39
34
#define MTEST_VERSION "3.3"
41
36
#include "client_priv.h"
37
#include <mysql_version.h>
38
#include <mysqld_error.h>
44
#include "my_regex.h" /* Our own version of regex */
50
45
#ifdef HAVE_SYS_WAIT_H
51
46
#include <sys/wait.h>
55
#include <sys/types.h>
64
/* Added this for string translation. */
65
#include "drizzled/gettext.h"
66
#include "drizzled/hash.h"
67
#include "drizzled/my_time.h"
68
#include "drizzled/charset.h"
70
#ifndef DRIZZLE_RETURN_SERVER_GONE
71
#define DRIZZLE_RETURN_HANDSHAKE_FAILED DRIZZLE_RETURN_ERROR_CODE
78
unsigned char *get_var_key(const unsigned char* var, size_t *len, bool);
79
bool get_one_option(int optid, const struct my_option *, char *argument);
82
51
#define MAX_VAR_NAME_LENGTH 256
83
52
#define MAX_COLUMNS 256
53
#define MAX_EMBEDDED_SERVER_ARGS 64
84
54
#define MAX_DELIMITER_LENGTH 16
85
56
/* Flags controlling send and reap */
86
57
#define QUERY_SEND_FLAG 1
87
58
#define QUERY_REAP_FLAG 2
89
ErrorCodes global_error_names;
61
OPT_SKIP_SAFEMALLOC=OPT_MAX_CLIENT_OPTION,
92
62
OPT_PS_PROTOCOL, OPT_SP_PROTOCOL, OPT_CURSOR_PROTOCOL, OPT_VIEW_PROTOCOL,
93
OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR, OPT_TAIL_LINES,
63
OPT_MAX_CONNECT_RETRIES, OPT_MARK_PROGRESS, OPT_LOG_DIR, OPT_TAIL_LINES
97
66
static int record= 0, opt_sleep= -1;
98
static char *opt_db= NULL, *opt_pass= NULL;
99
const char *opt_user= NULL, *opt_host= NULL, *unix_sock= NULL,
67
static char *opt_db= 0, *opt_pass= 0;
68
const char *opt_user= 0, *opt_host= 0, *unix_sock= 0, *opt_basedir= "./";
101
69
const char *opt_logdir= "";
102
const char *opt_include= NULL, *opt_charsets_dir;
103
const char *opt_testdir= NULL;
104
static uint32_t opt_port= 0;
70
const char *opt_include= 0, *opt_charsets_dir;
71
static int opt_port= 0;
105
72
static int opt_max_connect_retries;
106
static bool silent= false, verbose= false;
107
static bool tty_password= false;
108
static bool opt_mark_progress= false;
109
static bool parsing_disabled= false;
110
static bool display_result_vertically= false,
111
display_metadata= false, display_result_sorted= false;
112
static bool disable_query_log= false, disable_result_log= false;
113
static bool disable_warnings= false;
114
static bool disable_info= true;
115
static bool abort_on_error= true;
116
static bool server_initialized= false;
117
static bool is_windows= false;
73
static my_bool opt_compress= 0, silent= 0, verbose= 0;
74
static my_bool debug_info_flag= 0, debug_check_flag= 0;
75
static my_bool tty_password= 0;
76
static my_bool opt_mark_progress= 0;
77
static my_bool ps_protocol= 0, ps_protocol_enabled= 0;
78
static my_bool sp_protocol= 0, sp_protocol_enabled= 0;
79
static my_bool view_protocol= 0, view_protocol_enabled= 0;
80
static my_bool cursor_protocol= 0, cursor_protocol_enabled= 0;
81
static my_bool parsing_disabled= 0;
82
static my_bool display_result_vertically= FALSE,
83
display_metadata= FALSE, display_result_sorted= FALSE;
84
static my_bool disable_query_log= 0, disable_result_log= 0;
85
static my_bool disable_warnings= 0;
86
static my_bool disable_info= 1;
87
static my_bool abort_on_error= 1;
88
static my_bool server_initialized= 0;
89
static my_bool is_windows= 0;
118
90
static char **default_argv;
119
static const char *load_default_groups[]= { "drizzletest", "client", 0 };
91
static const char *load_default_groups[]= { "mysqltest", "client", 0 };
120
92
static char line_buffer[MAX_DELIMITER_LENGTH], *line_buffer_pos= line_buffer;
122
static uint32_t start_lineno= 0; /* Start line of current command */
94
static uint start_lineno= 0; /* Start line of current command */
95
static uint my_end_arg= 0;
124
97
/* Number of lines of the result to include in failure report */
125
static uint32_t opt_tail_lines= 0;
98
static uint opt_tail_lines= 0;
127
100
static char delimiter[MAX_DELIMITER_LENGTH]= ";";
128
static uint32_t delimiter_length= 1;
101
static uint delimiter_length= 1;
130
103
static char TMPDIR[FN_REFLEN];
387
387
char *query, *query_buf,*first_argument,*last_argument,*end;
388
388
int first_word_len, query_len;
390
st_expected_errors expected_errors;
389
my_bool abort_on_error;
390
struct st_expected_errors expected_errors;
391
char require_file[FN_REFLEN];
392
392
enum enum_commands type;
395
: query(NULL), query_buf(NULL), first_argument(NULL), last_argument(NULL),
396
end(NULL), first_word_len(0), query_len(0), abort_on_error(false),
397
require_file(""), type(Q_CONNECTION)
399
memset(&expected_errors, 0, sizeof(st_expected_errors));
404
if (query_buf != NULL)
411
395
TYPELIB command_typelib= {array_elements(command_names),"",
414
string ds_res, ds_progress, ds_warning_messages;
398
DYNAMIC_STRING ds_res, ds_progress, ds_warning_messages;
416
400
char builtin_echo[FN_REFLEN];
418
402
void die(const char *fmt, ...)
419
__attribute__((format(printf, 1, 2)));
403
ATTRIBUTE_FORMAT(printf, 1, 2);
420
404
void abort_not_supported_test(const char *fmt, ...)
421
__attribute__((format(printf, 1, 2)));
405
ATTRIBUTE_FORMAT(printf, 1, 2);
422
406
void verbose_msg(const char *fmt, ...)
423
__attribute__((format(printf, 1, 2)));
407
ATTRIBUTE_FORMAT(printf, 1, 2);
424
408
void warning_msg(const char *fmt, ...)
425
__attribute__((format(printf, 1, 2)));
409
ATTRIBUTE_FORMAT(printf, 1, 2);
426
410
void log_msg(const char *fmt, ...)
427
__attribute__((format(printf, 1, 2)));
411
ATTRIBUTE_FORMAT(printf, 1, 2);
429
413
VAR* var_from_env(const char *, const char *);
430
414
VAR* var_init(VAR* v, const char *name, int name_len, const char *val,
432
void var_free(pair<string, VAR*> v);
416
void var_free(void* v);
433
417
VAR* var_get(const char *var_name, const char** var_name_end,
434
bool raw, bool ignore_not_existing);
418
my_bool raw, my_bool ignore_not_existing);
435
419
void eval_expr(VAR* v, const char *p, const char** p_end);
436
bool match_delimiter(int c, const char *delim, uint32_t length);
420
my_bool match_delimiter(int c, const char *delim, uint length);
437
421
void dump_result_to_reject_file(char *buf, int size);
438
void dump_result_to_log_file(const char *buf, int size);
439
void dump_warning_messages(void);
440
void dump_progress(void);
422
void dump_result_to_log_file(char *buf, int size);
423
void dump_warning_messages();
424
void dump_progress();
442
void do_eval(string *query_eval, const char *query,
443
const char *query_end, bool pass_through_escape_chars);
444
void str_to_file(const char *fname, const char *str, int size);
445
void str_to_file2(const char *fname, const char *str, int size, bool append);
426
void do_eval(DYNAMIC_STRING *query_eval, const char *query,
427
const char *query_end, my_bool pass_through_escape_chars);
428
void str_to_file(const char *fname, char *str, int size);
429
void str_to_file2(const char *fname, char *str, int size, my_bool append);
447
431
/* For replace_column */
448
432
static char *replace_column[MAX_COLUMNS];
449
static uint32_t max_replace_column= 0;
433
static uint max_replace_column= 0;
450
434
void do_get_replace_column(struct st_command*);
451
void free_replace_column(void);
435
void free_replace_column();
453
437
/* For replace */
454
438
void do_get_replace(struct st_command *command);
455
void free_replace(void);
457
441
/* For replace_regex */
458
442
void do_get_replace_regex(struct st_command *command);
459
void free_replace_regex(void);
462
void free_all_replace(void);
465
void free_all_replace(void){
443
void free_replace_regex();
446
void free_all_replace(){
467
448
free_replace_regex();
468
449
free_replace_column();
471
void replace_append_mem(string *ds, const char *val,
473
void replace_append(string *ds, const char *val);
474
void replace_append_uint(string *ds, uint32_t val);
475
void append_sorted(string* ds, string* ds_input);
452
void replace_dynstr_append_mem(DYNAMIC_STRING *ds, const char *val,
454
void replace_dynstr_append(DYNAMIC_STRING *ds, const char *val);
455
void replace_dynstr_append_uint(DYNAMIC_STRING *ds, uint val);
456
void dynstr_append_sorted(DYNAMIC_STRING* ds, DYNAMIC_STRING* ds_input);
477
458
void handle_error(struct st_command*,
478
459
unsigned int err_errno, const char *err_error,
479
const char *err_sqlstate, string *ds);
460
const char *err_sqlstate, DYNAMIC_STRING *ds);
480
461
void handle_no_error(struct st_command*);
483
void do_eval(string *query_eval, const char *query,
484
const char *query_end, bool pass_through_escape_chars)
463
#ifdef EMBEDDED_LIBRARY
465
/* attributes of the query thread */
466
pthread_attr_t cn_thd_attrib;
469
send_one_query executes query in separate thread, which is
470
necessary in embedded library to run 'send' in proper way.
471
This implementation doesn't handle errors returned
472
by mysql_send_query. It's technically possible, though
473
I don't see where it is needed.
475
pthread_handler_t send_one_query(void *arg)
477
struct st_connection *cn= (struct st_connection*)arg;
480
VOID(mysql_send_query(&cn->mysql, cn->cur_query, cn->cur_query_len));
483
pthread_mutex_lock(&cn->mutex);
485
VOID(pthread_cond_signal(&cn->cond));
486
pthread_mutex_unlock(&cn->mutex);
491
static int do_send_query(struct st_connection *cn, const char *q, int q_len,
496
if (flags & QUERY_REAP_FLAG)
497
return mysql_send_query(&cn->mysql, q, q_len);
499
if (pthread_mutex_init(&cn->mutex, NULL) ||
500
pthread_cond_init(&cn->cond, NULL))
501
die("Error in the thread library");
504
cn->cur_query_len= q_len;
506
if (pthread_create(&tid, &cn_thd_attrib, send_one_query, (void*)cn))
507
die("Cannot start new thread for query");
512
static void wait_query_thread_end(struct st_connection *con)
514
if (!con->query_done)
516
pthread_mutex_lock(&con->mutex);
517
while (!con->query_done)
518
pthread_cond_wait(&con->cond, &con->mutex);
519
pthread_mutex_unlock(&con->mutex);
523
#else /*EMBEDDED_LIBRARY*/
525
#define do_send_query(cn,q,q_len,flags) mysql_send_query(&cn->mysql, q, q_len)
527
#endif /*EMBEDDED_LIBRARY*/
529
void do_eval(DYNAMIC_STRING *query_eval, const char *query,
530
const char *query_end, my_bool pass_through_escape_chars)
487
533
register char c, next_c;
488
534
register int escaped = 0;
536
DBUG_ENTER("do_eval");
492
538
for (p= query; (c= *p) && p < query_end; ++p)
1945
1980
<query to run> - The query that should be sent to the server
1946
1981
<column name> - Name of the column that holds the field be compared
1947
against the expected value
1982
against the expected value
1948
1983
<row no> - Number of the row that holds the field to be
1949
compared against the expected value
1984
compared against the expected value
1953
static void var_set_query_get_value(struct st_command *command, VAR *var)
1988
void var_set_query_get_value(struct st_command *command, VAR *var)
1956
1991
int col_no= -1;
1957
drizzle_result_st res;
1958
drizzle_return_t ret;
1959
drizzle_con_st *con= &cur_con->con;
1993
MYSQL* mysql= &cur_con->mysql;
1995
static DYNAMIC_STRING ds_query;
1996
static DYNAMIC_STRING ds_col;
1997
static DYNAMIC_STRING ds_row;
1964
1998
const struct command_arg query_get_value_args[] = {
1965
{"query", ARG_STRING, true, &ds_query, "Query to run"},
1966
{"column name", ARG_STRING, true, &ds_col, "Name of column"},
1967
{"row number", ARG_STRING, true, &ds_row, "Number for row"}
1999
"query", ARG_STRING, TRUE, &ds_query, "Query to run",
2000
"column name", ARG_STRING, TRUE, &ds_col, "Name of column",
2001
"row number", ARG_STRING, TRUE, &ds_row, "Number for row"
2004
DBUG_ENTER("var_set_query_get_value");
1972
2006
strip_parentheses(command);
2007
DBUG_PRINT("info", ("query: %s", command->query));
1973
2008
check_command_args(command, command->first_argument, query_get_value_args,
1974
2009
sizeof(query_get_value_args)/sizeof(struct command_arg),
2012
DBUG_PRINT("info", ("query: %s", ds_query.str));
2013
DBUG_PRINT("info", ("col: %s", ds_col.str));
1977
2015
/* Convert row number to int */
1978
row_no= atoi(ds_row.c_str());
1980
istringstream buff(ds_row);
1981
if ((buff >> row_no).fail())
1982
die("Invalid row number: '%s'", ds_row.c_str());
2016
if (!str2int(ds_row.str, 10, (long) 0, (long) INT_MAX, &row_no))
2017
die("Invalid row number: '%s'", ds_row.str);
2018
DBUG_PRINT("info", ("row: %s, row_no: %ld", ds_row.str, row_no));
2019
dynstr_free(&ds_row);
1984
2021
/* Remove any surrounding "'s from the query - if there is any */
1985
// (Don't get me started on this)
1986
char * unstripped_query= strdup(ds_query.c_str());
1987
if (strip_surrounding(unstripped_query, '"', '"'))
1988
die("Mismatched \"'s around query '%s'", ds_query.c_str());
1990
ds_query.append(unstripped_query);
2022
if (strip_surrounding(ds_query.str, '"', '"'))
2023
die("Mismatched \"'s around query '%s'", ds_query.str);
1992
2025
/* Run the query */
1993
if (drizzle_query(con, &res, ds_query.c_str(), ds_query.length(),
1995
ret != DRIZZLE_RETURN_OK)
1997
if (ret == DRIZZLE_RETURN_ERROR_CODE)
1999
die("Error running query '%s': %d %s", ds_query.c_str(),
2000
drizzle_result_error_code(&res), drizzle_result_error(&res));
2001
drizzle_result_free(&res);
2005
die("Error running query '%s': %d %s", ds_query.c_str(), ret,
2006
drizzle_con_error(con));
2009
if (drizzle_result_column_count(&res) == 0 ||
2010
drizzle_result_buffer(&res) != DRIZZLE_RETURN_OK)
2011
die("Query '%s' didn't return a result set", ds_query.c_str());
2026
if (mysql_real_query(mysql, ds_query.str, ds_query.length))
2027
die("Error running query '%s': %d %s", ds_query.str,
2028
mysql_errno(mysql), mysql_error(mysql));
2029
if (!(res= mysql_store_result(mysql)))
2030
die("Query '%s' didn't return a result set", ds_query.str);
2014
2033
/* Find column number from the given column name */
2016
uint32_t num_fields= drizzle_result_column_count(&res);
2017
drizzle_column_st *column;
2035
uint num_fields= mysql_num_fields(res);
2036
MYSQL_FIELD *fields= mysql_fetch_fields(res);
2019
2038
for (i= 0; i < num_fields; i++)
2021
column= drizzle_column_next(&res);
2022
if (strcmp(drizzle_column_name(column), ds_col.c_str()) == 0 &&
2023
strlen(drizzle_column_name(column)) == ds_col.length())
2040
if (strcmp(fields[i].name, ds_col.str) == 0 &&
2041
strlen(fields[i].name) == ds_col.length)
2539
2616
/* Parse what mode to set */
2540
istringstream buff(ds_mode);
2541
if (ds_mode.length() != 4 ||
2542
(buff >> oct >> mode).fail())
2617
if (ds_mode.length != 4 ||
2618
str2int(ds_mode.str, 8, 0, INT_MAX, &mode) == NullS)
2543
2619
die("You must write a 4 digit octal number for mode");
2545
handle_command_error(command, chmod(ds_file.c_str(), mode));
2621
DBUG_PRINT("info", ("chmod %o %s", (uint)mode, ds_file.str));
2622
handle_command_error(command, chmod(ds_file.str, mode));
2623
dynstr_free(&ds_mode);
2624
dynstr_free(&ds_file);
2553
command called command
2632
command called command
2556
2635
fiile_exist <file_name>
2557
2636
Check if file <file_name> exists
2560
static void do_file_exist(struct st_command *command)
2639
void do_file_exist(struct st_command *command)
2642
static DYNAMIC_STRING ds_filename;
2564
2643
const struct command_arg file_exist_args[] = {
2565
{ "filename", ARG_STRING, true, &ds_filename, "File to check if it exist" }
2644
{ "filename", ARG_STRING, TRUE, &ds_filename, "File to check if it exist" }
2646
DBUG_ENTER("do_file_exist");
2569
2648
check_command_args(command, command->first_argument,
2570
2649
file_exist_args,
2571
2650
sizeof(file_exist_args)/sizeof(struct command_arg),
2574
error= (access(ds_filename.c_str(), F_OK) != 0);
2653
DBUG_PRINT("info", ("Checking for existence of file: %s", ds_filename.str));
2654
error= (access(ds_filename.str, F_OK) != 0);
2575
2655
handle_command_error(command, error);
2656
dynstr_free(&ds_filename);
2583
command called command
2664
command called command
2586
2667
mkdir <dir_name>
2587
2668
Create the directory <dir_name>
2590
static void do_mkdir(struct st_command *command)
2671
void do_mkdir(struct st_command *command)
2674
static DYNAMIC_STRING ds_dirname;
2594
2675
const struct command_arg mkdir_args[] = {
2595
{"dirname", ARG_STRING, true, &ds_dirname, "Directory to create"}
2676
"dirname", ARG_STRING, TRUE, &ds_dirname, "Directory to create"
2678
DBUG_ENTER("do_mkdir");
2599
2680
check_command_args(command, command->first_argument,
2600
2681
mkdir_args, sizeof(mkdir_args)/sizeof(struct command_arg),
2603
error= mkdir(ds_dirname.c_str(), (0777 & my_umask_dir)) != 0;
2684
DBUG_PRINT("info", ("creating directory: %s", ds_dirname.str));
2685
error= my_mkdir(ds_dirname.str, 0777, MYF(0)) != 0;
2604
2686
handle_command_error(command, error);
2687
dynstr_free(&ds_dirname);
2611
command called command
2694
command called command
2614
2697
rmdir <dir_name>
2615
2698
Remove the empty directory <dir_name>
2618
static void do_rmdir(struct st_command *command)
2701
void do_rmdir(struct st_command *command)
2704
static DYNAMIC_STRING ds_dirname;
2622
2705
const struct command_arg rmdir_args[] = {
2623
{"dirname", ARG_STRING, true, &ds_dirname, "Directory to remove"}
2706
"dirname", ARG_STRING, TRUE, &ds_dirname, "Directory to remove"
2708
DBUG_ENTER("do_rmdir");
2627
2710
check_command_args(command, command->first_argument,
2628
2711
rmdir_args, sizeof(rmdir_args)/sizeof(struct command_arg),
2631
error= rmdir(ds_dirname.c_str()) != 0;
2714
DBUG_PRINT("info", ("removing directory: %s", ds_dirname.str));
2715
error= rmdir(ds_dirname.str) != 0;
2632
2716
handle_command_error(command, error);
2717
dynstr_free(&ds_dirname);
3191
3313
when ndb binlog is on, this call will wait until last updated epoch
3192
(locally in the drizzled) has been received into the binlog
3314
(locally in the mysqld) has been received into the binlog
3194
static int do_save_master_pos(void)
3316
int do_save_master_pos()
3196
drizzle_result_st res;
3197
drizzle_return_t ret;
3199
drizzle_con_st *con= &cur_con->con;
3320
MYSQL *mysql = &cur_con->mysql;
3200
3321
const char *query;
3203
if (drizzle_query_str(con, &res, query= "show master status", &ret) == NULL ||
3204
ret != DRIZZLE_RETURN_OK)
3322
DBUG_ENTER("do_save_master_pos");
3324
#ifdef HAVE_NDB_BINLOG
3326
Wait for ndb binlog to be up-to-date with all changes
3327
done on the local mysql server
3206
if (ret == DRIZZLE_RETURN_ERROR_CODE)
3330
ulong have_ndbcluster;
3331
if (mysql_query(mysql, query= "show variables like 'have_ndbcluster'"))
3332
die("'%s' failed: %d %s", query,
3333
mysql_errno(mysql), mysql_error(mysql));
3334
if (!(res= mysql_store_result(mysql)))
3335
die("mysql_store_result() returned NULL for '%s'", query);
3336
if (!(row= mysql_fetch_row(res)))
3337
die("Query '%s' returned empty result", query);
3339
have_ndbcluster= strcmp("YES", row[1]) == 0;
3340
mysql_free_result(res);
3342
if (have_ndbcluster)
3208
die("failed in '%s': %d: %s", query, drizzle_result_error_code(&res),
3209
drizzle_result_error(&res));
3210
drizzle_result_free(&res);
3344
ulonglong start_epoch= 0, handled_epoch= 0,
3345
latest_epoch=0, latest_trans_epoch=0,
3346
latest_handled_binlog_epoch= 0, latest_received_binlog_epoch= 0,
3347
latest_applied_binlog_epoch= 0;
3352
const char binlog[]= "binlog";
3353
const char latest_epoch_str[]=
3355
const char latest_trans_epoch_str[]=
3356
"latest_trans_epoch=";
3357
const char latest_received_binlog_epoch_str[]=
3358
"latest_received_binlog_epoch";
3359
const char latest_handled_binlog_epoch_str[]=
3360
"latest_handled_binlog_epoch=";
3361
const char latest_applied_binlog_epoch_str[]=
3362
"latest_applied_binlog_epoch=";
3365
if (mysql_query(mysql, query= "show engine ndb status"))
3366
die("failed in '%s': %d %s", query,
3367
mysql_errno(mysql), mysql_error(mysql));
3368
if (!(res= mysql_store_result(mysql)))
3369
die("mysql_store_result() returned NULL for '%s'", query);
3370
while ((row= mysql_fetch_row(res)))
3372
if (strcmp(row[1], binlog) == 0)
3374
const char *status= row[2];
3377
while (*status && strncmp(status, latest_epoch_str,
3378
sizeof(latest_epoch_str)-1))
3382
status+= sizeof(latest_epoch_str)-1;
3383
latest_epoch= strtoull(status, (char**) 0, 10);
3386
die("result does not contain '%s' in '%s'",
3387
latest_epoch_str, query);
3388
/* latest_trans_epoch */
3389
while (*status && strncmp(status, latest_trans_epoch_str,
3390
sizeof(latest_trans_epoch_str)-1))
3394
status+= sizeof(latest_trans_epoch_str)-1;
3395
latest_trans_epoch= strtoull(status, (char**) 0, 10);
3398
die("result does not contain '%s' in '%s'",
3399
latest_trans_epoch_str, query);
3400
/* latest_received_binlog_epoch */
3402
strncmp(status, latest_received_binlog_epoch_str,
3403
sizeof(latest_received_binlog_epoch_str)-1))
3407
status+= sizeof(latest_received_binlog_epoch_str)-1;
3408
latest_received_binlog_epoch= strtoull(status, (char**) 0, 10);
3411
die("result does not contain '%s' in '%s'",
3412
latest_received_binlog_epoch_str, query);
3413
/* latest_handled_binlog */
3415
strncmp(status, latest_handled_binlog_epoch_str,
3416
sizeof(latest_handled_binlog_epoch_str)-1))
3420
status+= sizeof(latest_handled_binlog_epoch_str)-1;
3421
latest_handled_binlog_epoch= strtoull(status, (char**) 0, 10);
3424
die("result does not contain '%s' in '%s'",
3425
latest_handled_binlog_epoch_str, query);
3426
/* latest_applied_binlog_epoch */
3428
strncmp(status, latest_applied_binlog_epoch_str,
3429
sizeof(latest_applied_binlog_epoch_str)-1))
3433
status+= sizeof(latest_applied_binlog_epoch_str)-1;
3434
latest_applied_binlog_epoch= strtoull(status, (char**) 0, 10);
3437
die("result does not contain '%s' in '%s'",
3438
latest_applied_binlog_epoch_str, query);
3440
start_epoch= latest_trans_epoch;
3445
die("result does not contain '%s' in '%s'",
3447
if (latest_handled_binlog_epoch > handled_epoch)
3449
handled_epoch= latest_handled_binlog_epoch;
3451
if (latest_handled_binlog_epoch >= start_epoch)
3453
else if (count > 30)
3457
mysql_free_result(res);
3213
die("failed in '%s': %d: %s", query, ret, drizzle_con_error(con));
3462
if (mysql_query(mysql, query= "show master status"))
3463
die("failed in 'show master status': %d %s",
3464
mysql_errno(mysql), mysql_error(mysql));
3216
if (drizzle_result_column_count(&res) == 0 ||
3217
drizzle_result_buffer(&res) != DRIZZLE_RETURN_OK)
3218
die("drizzleclient_store_result() retuned NULL for '%s'", query);
3219
if (!(row = drizzle_row_next(&res)))
3466
if (!(res = mysql_store_result(mysql)))
3467
die("mysql_store_result() retuned NULL for '%s'", query);
3468
if (!(row = mysql_fetch_row(res)))
3220
3469
die("empty result in show master status");
3221
strncpy(master_pos.file, row[0], sizeof(master_pos.file)-1);
3470
strnmov(master_pos.file, row[0], sizeof(master_pos.file)-1);
3222
3471
master_pos.pos = strtoul(row[1], (char**) 0, 10);
3223
drizzle_result_free(&res);
3472
mysql_free_result(res);
3828
4141
<port> - server port
3829
4142
<sock> - server socket
3830
4143
<opts> - options to use for the connection
3831
* SSL - use SSL if available
3832
* COMPRESS - use compression if available
3836
static void do_connect(struct st_command *command)
4144
* SSL - use SSL if available
4145
* COMPRESS - use compression if available
4149
void do_connect(struct st_command *command)
3838
4151
int con_port= opt_port;
3839
const char *con_options;
3840
bool con_ssl= 0, con_compress= 0;
4153
my_bool con_ssl= 0, con_compress= 0;
3841
4154
struct st_connection* con_slot;
3843
string ds_connection_name;
4156
static DYNAMIC_STRING ds_connection_name;
4157
static DYNAMIC_STRING ds_host;
4158
static DYNAMIC_STRING ds_user;
4159
static DYNAMIC_STRING ds_password;
4160
static DYNAMIC_STRING ds_database;
4161
static DYNAMIC_STRING ds_port;
4162
static DYNAMIC_STRING ds_sock;
4163
static DYNAMIC_STRING ds_options;
3851
4164
const struct command_arg connect_args[] = {
3852
{ "connection name", ARG_STRING, true, &ds_connection_name, "Name of the connection" },
3853
{ "host", ARG_STRING, true, &ds_host, "Host to connect to" },
3854
{ "user", ARG_STRING, false, &ds_user, "User to connect as" },
3855
{ "passsword", ARG_STRING, false, &ds_password, "Password used when connecting" },
3856
{ "database", ARG_STRING, false, &ds_database, "Database to select after connect" },
3857
{ "port", ARG_STRING, false, &ds_port, "Port to connect to" },
3858
{ "socket", ARG_STRING, false, &ds_sock, "Socket to connect with" },
3859
{ "options", ARG_STRING, false, &ds_options, "Options to use while connecting" }
4165
{ "connection name", ARG_STRING, TRUE, &ds_connection_name, "Name of the connection" },
4166
{ "host", ARG_STRING, TRUE, &ds_host, "Host to connect to" },
4167
{ "user", ARG_STRING, FALSE, &ds_user, "User to connect as" },
4168
{ "passsword", ARG_STRING, FALSE, &ds_password, "Password used when connecting" },
4169
{ "database", ARG_STRING, FALSE, &ds_database, "Database to select after connect" },
4170
{ "port", ARG_STRING, FALSE, &ds_port, "Port to connect to" },
4171
{ "socket", ARG_STRING, FALSE, &ds_sock, "Socket to connect with" },
4172
{ "options", ARG_STRING, FALSE, &ds_options, "Options to use while connecting" }
4175
DBUG_ENTER("do_connect");
4176
DBUG_PRINT("enter",("connect: %s", command->first_argument));
3863
4178
strip_parentheses(command);
3864
4179
check_command_args(command, command->first_argument, connect_args,
4580
4932
{"help", '?', "Display this help and exit.", 0, 0, 0, GET_NO_ARG, NO_ARG,
4581
4933
0, 0, 0, 0, 0, 0},
4582
{"basedir", 'b', "Basedir for tests.", (char**) &opt_basedir,
4583
(char**) &opt_basedir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4934
{"basedir", 'b', "Basedir for tests.", (uchar**) &opt_basedir,
4935
(uchar**) &opt_basedir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4584
4936
{"character-sets-dir", OPT_CHARSETS_DIR,
4585
"Directory where character sets are.", (char**) &opt_charsets_dir,
4586
(char**) &opt_charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4587
{"database", 'D', "Database to use.", (char**) &opt_db, (char**) &opt_db, 0,
4588
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4589
{"host", 'h', "Connect to host.", (char**) &opt_host, (char**) &opt_host, 0,
4590
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4591
{"include", 'i', "Include SQL before each test case.", (char**) &opt_include,
4592
(char**) &opt_include, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4593
{"testdir", OPT_TESTDIR, "Path to use to search for test files",
4594
(char**) &opt_testdir,
4595
(char**) &opt_testdir, 0,GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4596
{"logdir", OPT_LOG_DIR, "Directory for log files", (char**) &opt_logdir,
4597
(char**) &opt_logdir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4937
"Directory where character sets are.", (uchar**) &opt_charsets_dir,
4938
(uchar**) &opt_charsets_dir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4939
{"compress", 'C', "Use the compressed server/client protocol.",
4940
(uchar**) &opt_compress, (uchar**) &opt_compress, 0, GET_BOOL, NO_ARG, 0, 0, 0,
4942
{"cursor-protocol", OPT_CURSOR_PROTOCOL, "Use cursors for prepared statements.",
4943
(uchar**) &cursor_protocol, (uchar**) &cursor_protocol, 0,
4944
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4945
{"database", 'D', "Database to use.", (uchar**) &opt_db, (uchar**) &opt_db, 0,
4946
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4948
{"debug", '#', "This is a non-debug version. Catch this and exit",
4949
0,0, 0, GET_DISABLED, OPT_ARG, 0, 0, 0, 0, 0, 0},
4951
{"debug", '#', "Output debug log. Often this is 'd:t:o,filename'.",
4952
0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0},
4954
{"debug-check", OPT_DEBUG_CHECK, "Check memory and open file usage at exit.",
4955
(uchar**) &debug_check_flag, (uchar**) &debug_check_flag, 0,
4956
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4957
{"debug-info", OPT_DEBUG_INFO, "Print some debug info at exit.",
4958
(uchar**) &debug_info_flag, (uchar**) &debug_info_flag,
4959
0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4960
{"host", 'h', "Connect to host.", (uchar**) &opt_host, (uchar**) &opt_host, 0,
4961
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4962
{"include", 'i', "Include SQL before each test case.", (uchar**) &opt_include,
4963
(uchar**) &opt_include, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4964
{"logdir", OPT_LOG_DIR, "Directory for log files", (uchar**) &opt_logdir,
4965
(uchar**) &opt_logdir, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4598
4966
{"mark-progress", OPT_MARK_PROGRESS,
4599
4967
"Write linenumber and elapsed time to <testname>.progress ",
4600
(char**) &opt_mark_progress, (char**) &opt_mark_progress, 0,
4968
(uchar**) &opt_mark_progress, (uchar**) &opt_mark_progress, 0,
4601
4969
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4602
4970
{"max-connect-retries", OPT_MAX_CONNECT_RETRIES,
4603
4971
"Max number of connection attempts when connecting to server",
4604
(char**) &opt_max_connect_retries, (char**) &opt_max_connect_retries, 0,
4972
(uchar**) &opt_max_connect_retries, (uchar**) &opt_max_connect_retries, 0,
4605
4973
GET_INT, REQUIRED_ARG, 500, 1, 10000, 0, 0, 0},
4606
{"password", 'P', "Password to use when connecting to server.",
4974
{"password", 'p', "Password to use when connecting to server.",
4607
4975
0, 0, 0, GET_STR, OPT_ARG, 0, 0, 0, 0, 0, 0},
4608
{"port", 'p', "Port number to use for connection or 0 for default to, in "
4609
"order of preference, drizzle.cnf, $DRIZZLE_TCP_PORT, "
4610
"built-in default (" STRINGIFY_ARG(DRIZZLE_PORT) ").",
4611
0, 0, 0, GET_UINT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4612
{"quiet", 's', "Suppress all normal output.", (char**) &silent,
4613
(char**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4976
{"port", 'P', "Port number to use for connection or 0 for default to, in "
4977
"order of preference, my.cnf, $MYSQL_TCP_PORT, "
4978
#if MYSQL_PORT_DEFAULT == 0
4981
"built-in default (" STRINGIFY_ARG(MYSQL_PORT) ").",
4982
(uchar**) &opt_port,
4983
(uchar**) &opt_port, 0, GET_INT, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4984
{"protocol", OPT_MYSQL_PROTOCOL,
4985
"The protocol of connection (tcp,socket,pipe,memory).",
4986
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4987
{"ps-protocol", OPT_PS_PROTOCOL, "Use prepared statements protocol for communication",
4988
(uchar**) &ps_protocol, (uchar**) &ps_protocol, 0,
4989
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4990
{"quiet", 's', "Suppress all normal output.", (uchar**) &silent,
4991
(uchar**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4614
4992
{"record", 'r', "Record output of test_file into result file.",
4615
4993
0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
4616
4994
{"result-file", 'R', "Read/Store result from/in this file.",
4617
(char**) &result_file_name, (char**) &result_file_name, 0,
4995
(uchar**) &result_file_name, (uchar**) &result_file_name, 0,
4618
4996
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4997
{"server-arg", 'A', "Send option value to embedded server as a parameter.",
4998
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4999
{"server-file", 'F', "Read embedded server arguments from file.",
5000
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4619
5001
{"silent", 's', "Suppress all normal output. Synonym for --quiet.",
4620
(char**) &silent, (char**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
5002
(uchar**) &silent, (uchar**) &silent, 0, GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4621
5003
{"sleep", 'T', "Sleep always this many seconds on sleep commands.",
4622
(char**) &opt_sleep, (char**) &opt_sleep, 0, GET_INT, REQUIRED_ARG, -1, -1, 0,
5004
(uchar**) &opt_sleep, (uchar**) &opt_sleep, 0, GET_INT, REQUIRED_ARG, -1, -1, 0,
5006
{"socket", 'S', "Socket file to use for connection.",
5007
(uchar**) &unix_sock, (uchar**) &unix_sock, 0, GET_STR, REQUIRED_ARG, 0, 0, 0,
5009
{"sp-protocol", OPT_SP_PROTOCOL, "Use stored procedures for select",
5010
(uchar**) &sp_protocol, (uchar**) &sp_protocol, 0,
5011
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4624
5012
{"tail-lines", OPT_TAIL_LINES,
4625
5013
"Number of lines of the resul to include in a failure report",
4626
(char**) &opt_tail_lines, (char**) &opt_tail_lines, 0,
5014
(uchar**) &opt_tail_lines, (uchar**) &opt_tail_lines, 0,
4627
5015
GET_INT, REQUIRED_ARG, 0, 0, 10000, 0, 0, 0},
4628
5016
{"test-file", 'x', "Read test from/in this file (default stdin).",
4629
5017
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4631
5019
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4632
5020
{"tmpdir", 't', "Temporary directory where sockets are put.",
4633
5021
0, 0, 0, GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4634
{"user", 'u', "User for login.", (char**) &opt_user, (char**) &opt_user, 0,
5022
{"user", 'u', "User for login.", (uchar**) &opt_user, (uchar**) &opt_user, 0,
4635
5023
GET_STR, REQUIRED_ARG, 0, 0, 0, 0, 0, 0},
4636
{"verbose", 'v', "Write more.", (char**) &verbose, (char**) &verbose, 0,
5024
{"verbose", 'v', "Write more.", (uchar**) &verbose, (uchar**) &verbose, 0,
4637
5025
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4638
5026
{"version", 'V', "Output version information and exit.",
4639
5027
0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0},
5028
{"view-protocol", OPT_VIEW_PROTOCOL, "Use views for select",
5029
(uchar**) &view_protocol, (uchar**) &view_protocol, 0,
5030
GET_BOOL, NO_ARG, 0, 0, 0, 0, 0, 0},
4640
5031
{ 0, 0, 0, 0, 0, 0, GET_NO_ARG, NO_ARG, 0, 0, 0, 0, 0, 0}
4644
static void print_version(void)
5035
#include <help_start.h>
5037
void print_version(void)
4646
printf("%s Ver %s Distrib %s, for %s-%s (%s)\n",my_progname,MTEST_VERSION,
4647
drizzle_version(),HOST_VENDOR,HOST_OS,HOST_CPU);
5039
printf("%s Ver %s Distrib %s, for %s (%s)\n",my_progname,MTEST_VERSION,
5040
MYSQL_SERVER_VERSION,SYSTEM_TYPE,MACHINE_TYPE);
4650
static void usage(void)
4652
5045
print_version();
4653
5046
printf("MySQL AB, by Sasha, Matt, Monty & Jani\n");
4654
printf("Drizzle version modified by Brian, Jay, Monty Taylor, PatG and Stewart\n");
4655
5047
printf("This software comes with ABSOLUTELY NO WARRANTY\n\n");
4656
printf("Runs a test against the DRIZZLE server and compares output with a results file.\n\n");
5048
printf("Runs a test against the mysql server and compares output with a results file.\n\n");
4657
5049
printf("Usage: %s [OPTIONS] [database] < test_file\n", my_progname);
4658
5050
my_print_help(my_long_options);
4659
5051
printf(" --no-defaults Don't read default options from any options file.\n");
4660
5052
my_print_variables(my_long_options);
4663
bool get_one_option(int optid, const struct my_option *, char *argument)
4665
char *endchar= NULL;
4666
uint64_t temp_drizzle_port= 0;
5055
#include <help_end.h>
5059
Read arguments for embedded server and put them into
5060
embedded_server_args[]
5063
void read_embedded_server_arguments(const char *name)
5065
char argument[1024],buff[FN_REFLEN], *str=0;
5068
if (!test_if_hard_path(name))
5070
strxmov(buff, opt_basedir, name, NullS);
5073
fn_format(buff, name, "", "", MY_UNPACK_FILENAME);
5075
if (!embedded_server_arg_count)
5077
embedded_server_arg_count=1;
5078
embedded_server_args[0]= (char*) ""; /* Progname */
5080
if (!(file=my_fopen(buff, O_RDONLY | FILE_BINARY, MYF(MY_WME))))
5081
die("Failed to open file '%s'", buff);
5083
while (embedded_server_arg_count < MAX_EMBEDDED_SERVER_ARGS &&
5084
(str=fgets(argument,sizeof(argument), file)))
5086
*(strend(str)-1)=0; /* Remove end newline */
5087
if (!(embedded_server_args[embedded_server_arg_count]=
5088
(char*) my_strdup(str,MYF(MY_WME))))
5090
my_fclose(file,MYF(0));
5091
die("Out of memory");
5094
embedded_server_arg_count++;
5096
my_fclose(file,MYF(0));
5098
die("Too many arguments in option file: %s",name);
5105
get_one_option(int optid, const struct my_option *opt __attribute__((unused)),
4668
5108
switch(optid) {
5111
DBUG_PUSH(argument ? argument : "d:t:S:i:O,/tmp/mysqltest.trace");
5112
debug_check_flag= 1;
4895
5343
Values may be converted with 'replace_column'
4898
static void append_result(string *ds, drizzle_result_st *res)
5346
void append_result(DYNAMIC_STRING *ds, MYSQL_RES *res)
4901
uint32_t num_fields= drizzle_result_column_count(res);
4902
drizzle_column_st *column;
5349
uint num_fields= mysql_num_fields(res);
5350
MYSQL_FIELD *fields= mysql_fetch_fields(res);
4905
while ((row = drizzle_row_next(res)))
5353
while ((row = mysql_fetch_row(res)))
4908
lengths = drizzle_row_field_sizes(res);
4909
drizzle_column_seek(res, 0);
5356
lengths = mysql_fetch_lengths(res);
4910
5357
for (i = 0; i < num_fields; i++)
4912
column= drizzle_column_next(res);
4913
append_field(ds, i, column,
5358
append_field(ds, i, &fields[i],
4914
5359
(const char*)row[i], lengths[i], !row[i]);
4916
if (!display_result_vertically)
5360
if (!display_result_vertically)
5361
dynstr_append_mem(ds, "\n", 1);
5367
Append all results from ps execution to the dynamic string separated
5368
with '\t'. Values may be converted with 'replace_column'
5371
void append_stmt_result(DYNAMIC_STRING *ds, MYSQL_STMT *stmt,
5372
MYSQL_FIELD *fields, uint num_fields)
5374
MYSQL_BIND *my_bind;
5379
/* Allocate array with bind structs, lengths and NULL flags */
5380
my_bind= (MYSQL_BIND*) my_malloc(num_fields * sizeof(MYSQL_BIND),
5381
MYF(MY_WME | MY_FAE | MY_ZEROFILL));
5382
length= (ulong*) my_malloc(num_fields * sizeof(ulong),
5383
MYF(MY_WME | MY_FAE));
5384
is_null= (my_bool*) my_malloc(num_fields * sizeof(my_bool),
5385
MYF(MY_WME | MY_FAE));
5387
/* Allocate data for the result of each field */
5388
for (i= 0; i < num_fields; i++)
5390
uint max_length= fields[i].max_length + 1;
5391
my_bind[i].buffer_type= MYSQL_TYPE_STRING;
5392
my_bind[i].buffer= (char *)my_malloc(max_length, MYF(MY_WME | MY_FAE));
5393
my_bind[i].buffer_length= max_length;
5394
my_bind[i].is_null= &is_null[i];
5395
my_bind[i].length= &length[i];
5397
DBUG_PRINT("bind", ("col[%d]: buffer_type: %d, buffer_length: %lu",
5398
i, my_bind[i].buffer_type, my_bind[i].buffer_length));
5401
if (mysql_stmt_bind_result(stmt, my_bind))
5402
die("mysql_stmt_bind_result failed: %d: %s",
5403
mysql_stmt_errno(stmt), mysql_stmt_error(stmt));
5405
while (mysql_stmt_fetch(stmt) == 0)
5407
for (i= 0; i < num_fields; i++)
5408
append_field(ds, i, &fields[i], (const char *) my_bind[i].buffer,
5409
*my_bind[i].length, *my_bind[i].is_null);
5410
if (!display_result_vertically)
5411
dynstr_append_mem(ds, "\n", 1);
5414
if (mysql_stmt_fetch(stmt) != MYSQL_NO_DATA)
5415
die("fetch didn't end with MYSQL_NO_DATA from statement: %d %s",
5416
mysql_stmt_errno(stmt), mysql_stmt_error(stmt));
5418
for (i= 0; i < num_fields; i++)
5420
/* Free data for output */
5421
my_free(my_bind[i].buffer, MYF(MY_WME | MY_FAE));
5423
/* Free array with bind structs, lengths and NULL flags */
5424
my_free(my_bind , MYF(MY_WME | MY_FAE));
5425
my_free(length , MYF(MY_WME | MY_FAE));
5426
my_free(is_null , MYF(MY_WME | MY_FAE));
4924
5431
Append metadata for fields to output
4927
static void append_metadata(string *ds, drizzle_result_st *res)
5434
void append_metadata(DYNAMIC_STRING *ds,
4929
drizzle_column_st *column;
4930
ds->append("Catalog\tDatabase\tTable\tTable_alias\tColumn\t"
4931
"Column_alias\tType\tLength\tMax length\tIs_null\t"
4932
"Flags\tDecimals\tCharsetnr\n");
5438
MYSQL_FIELD *field_end;
5439
dynstr_append(ds,"Catalog\tDatabase\tTable\tTable_alias\tColumn\t"
5440
"Column_alias\tType\tLength\tMax length\tIs_null\t"
5441
"Flags\tDecimals\tCharsetnr\n");
4934
drizzle_column_seek(res, 0);
4935
while ((column= drizzle_column_next(res)))
5443
for (field_end= field+num_fields ;
4937
ds->append(drizzle_column_catalog(column),
4938
strlen(drizzle_column_catalog(column)));
4939
ds->append("\t", 1);
4940
ds->append(drizzle_column_db(column), strlen(drizzle_column_db(column)));
4941
ds->append("\t", 1);
4942
ds->append(drizzle_column_orig_table(column),
4943
strlen(drizzle_column_orig_table(column)));
4944
ds->append("\t", 1);
4945
ds->append(drizzle_column_table(column),
4946
strlen(drizzle_column_table(column)));
4947
ds->append("\t", 1);
4948
ds->append(drizzle_column_orig_name(column),
4949
strlen(drizzle_column_orig_name(column)));
4950
ds->append("\t", 1);
4951
ds->append(drizzle_column_name(column),
4952
strlen(drizzle_column_name(column)));
4953
ds->append("\t", 1);
4954
replace_append_uint(ds, drizzle_column_type_drizzle(column));
4955
ds->append("\t", 1);
4956
replace_append_uint(ds, drizzle_column_size(column));
4957
ds->append("\t", 1);
4958
replace_append_uint(ds, drizzle_column_max_size(column));
4959
ds->append("\t", 1);
4960
ds->append((char*) ((drizzle_column_flags(column) & DRIZZLE_COLUMN_FLAGS_NOT_NULL) ? "N" : "Y"), 1);
4961
ds->append("\t", 1);
4962
replace_append_uint(ds, drizzle_column_flags(column));
4963
ds->append("\t", 1);
4964
replace_append_uint(ds, drizzle_column_decimals(column));
4965
ds->append("\t", 1);
4966
replace_append_uint(ds, drizzle_column_charset(column));
4967
ds->append("\n", 1);
5447
dynstr_append_mem(ds, field->catalog,
5448
field->catalog_length);
5449
dynstr_append_mem(ds, "\t", 1);
5450
dynstr_append_mem(ds, field->db, field->db_length);
5451
dynstr_append_mem(ds, "\t", 1);
5452
dynstr_append_mem(ds, field->org_table,
5453
field->org_table_length);
5454
dynstr_append_mem(ds, "\t", 1);
5455
dynstr_append_mem(ds, field->table,
5456
field->table_length);
5457
dynstr_append_mem(ds, "\t", 1);
5458
dynstr_append_mem(ds, field->org_name,
5459
field->org_name_length);
5460
dynstr_append_mem(ds, "\t", 1);
5461
dynstr_append_mem(ds, field->name, field->name_length);
5462
dynstr_append_mem(ds, "\t", 1);
5463
replace_dynstr_append_uint(ds, field->type);
5464
dynstr_append_mem(ds, "\t", 1);
5465
replace_dynstr_append_uint(ds, field->length);
5466
dynstr_append_mem(ds, "\t", 1);
5467
replace_dynstr_append_uint(ds, field->max_length);
5468
dynstr_append_mem(ds, "\t", 1);
5469
dynstr_append_mem(ds, (char*) (IS_NOT_NULL(field->flags) ?
5471
dynstr_append_mem(ds, "\t", 1);
5472
replace_dynstr_append_uint(ds, field->flags);
5473
dynstr_append_mem(ds, "\t", 1);
5474
replace_dynstr_append_uint(ds, field->decimals);
5475
dynstr_append_mem(ds, "\t", 1);
5476
replace_dynstr_append_uint(ds, field->charsetnr);
5477
dynstr_append_mem(ds, "\n", 1);
5014
5523
Number of warnings appended to ds
5017
static int append_warnings(string *ds, drizzle_con_st *con,
5018
drizzle_result_st *res)
5526
int append_warnings(DYNAMIC_STRING *ds, MYSQL* mysql)
5021
drizzle_result_st warn_res;
5022
drizzle_return_t ret;
5025
if (!(count= drizzle_result_warning_count(res)))
5028
if (drizzle_query_str(con, &warn_res, "SHOW WARNINGS", &ret) == NULL ||
5029
ret != DRIZZLE_RETURN_OK)
5031
if (ret == DRIZZLE_RETURN_ERROR_CODE)
5032
die("Error running query \"SHOW WARNINGS\": %s", drizzle_result_error(&warn_res));
5034
die("Error running query \"SHOW WARNINGS\": %s", drizzle_con_error(con));
5037
if (drizzle_result_column_count(&warn_res) == 0 ||
5038
drizzle_result_buffer(&warn_res) != DRIZZLE_RETURN_OK)
5039
die("Warning count is %u but didn't get any warnings", count);
5041
append_result(ds, &warn_res);
5042
drizzle_result_free(&warn_res);
5529
MYSQL_RES *warn_res;
5530
DBUG_ENTER("append_warnings");
5532
if (!(count= mysql_warning_count(mysql)))
5536
If one day we will support execution of multi-statements
5537
through PS API we should not issue SHOW WARNINGS until
5538
we have not read all results...
5540
DBUG_ASSERT(!mysql_more_results(mysql));
5542
if (mysql_real_query(mysql, "SHOW WARNINGS", 13))
5543
die("Error running query \"SHOW WARNINGS\": %s", mysql_error(mysql));
5545
if (!(warn_res= mysql_store_result(mysql)))
5546
die("Warning count is %u but didn't get any warnings",
5549
append_result(ds, warn_res);
5550
mysql_free_result(warn_res);
5552
DBUG_PRINT("warnings", ("%s", ds->str));
5049
Run query using DRIZZLE C API
5559
Run query using MySQL C API
5053
drizzle DRIZZLE handle
5054
command current command pointer
5055
flags flags indicating if we should SEND and/or REAP
5056
query query string to execute
5057
query_len length query string to execute
5058
ds output buffer where to store result form query
5564
command current command pointer
5565
flags flags indicating if we should SEND and/or REAP
5566
query query string to execute
5567
query_len length query string to execute
5568
ds output buffer where to store result form query
5061
static void run_query_normal(struct st_connection *cn,
5062
struct st_command *command,
5063
int flags, char *query, int query_len,
5064
string *ds, string *ds_warnings)
5571
void run_query_normal(struct st_connection *cn, struct st_command *command,
5572
int flags, char *query, int query_len,
5573
DYNAMIC_STRING *ds, DYNAMIC_STRING *ds_warnings)
5066
drizzle_result_st res;
5067
drizzle_return_t ret;
5068
drizzle_con_st *con= &cn->con;
5071
drizzle_con_add_options(con, DRIZZLE_CON_NO_RESULT_READ);
5576
MYSQL *mysql= &cn->mysql;
5577
int err= 0, counter= 0;
5578
DBUG_ENTER("run_query_normal");
5579
DBUG_PRINT("enter",("flags: %d", flags));
5580
DBUG_PRINT("enter", ("query: '%-.60s'", query));
5073
5582
if (flags & QUERY_SEND_FLAG)
5079
(void) drizzle_query(con, &res, query, query_len, &ret);
5080
if (ret != DRIZZLE_RETURN_OK)
5587
if (do_send_query(cn, query, query_len, flags))
5082
if (ret == DRIZZLE_RETURN_ERROR_CODE ||
5083
ret == DRIZZLE_RETURN_HANDSHAKE_FAILED)
5085
err= drizzle_result_error_code(&res);
5086
handle_error(command, err, drizzle_result_error(&res),
5087
drizzle_result_sqlstate(&res), ds);
5088
if (ret == DRIZZLE_RETURN_ERROR_CODE)
5089
drizzle_result_free(&res);
5093
handle_error(command, ret, drizzle_con_error(con), "", ds);
5589
handle_error(command, mysql_errno(mysql), mysql_error(mysql),
5590
mysql_sqlstate(mysql), ds);
5594
#ifdef EMBEDDED_LIBRARY
5596
Here we handle 'reap' command, so we need to check if the
5597
query's thread was finished and probably wait
5599
else if (flags & QUERY_REAP_FLAG)
5600
wait_query_thread_end(cn);
5601
#endif /*EMBEDDED_LIBRARY*/
5099
5602
if (!(flags & QUERY_REAP_FLAG))
5104
* Read the result packet
5106
if (drizzle_result_read(con, &res, &ret) == NULL ||
5107
ret != DRIZZLE_RETURN_OK)
5608
When on first result set, call mysql_read_query_result to retrieve
5609
answer to the query sent earlier
5611
if ((counter==0) && mysql_read_query_result(mysql))
5109
if (ret == DRIZZLE_RETURN_ERROR_CODE)
5111
handle_error(command, drizzle_result_error_code(&res),
5112
drizzle_result_error(&res), drizzle_result_sqlstate(&res),
5116
handle_error(command, ret, drizzle_con_error(con), "", ds);
5117
drizzle_result_free(&res);
5613
handle_error(command, mysql_errno(mysql), mysql_error(mysql),
5614
mysql_sqlstate(mysql), ds);
5123
5620
Store the result of the query if it will return any fields
5125
if (drizzle_result_column_count(&res) &&
5126
(ret= drizzle_result_buffer(&res)) != DRIZZLE_RETURN_OK)
5622
if (mysql_field_count(mysql) && ((res= mysql_store_result(mysql)) == 0))
5128
if (ret == DRIZZLE_RETURN_ERROR_CODE)
5130
handle_error(command, drizzle_result_error_code(&res),
5131
drizzle_result_error(&res), drizzle_result_sqlstate(&res),
5135
handle_error(command, ret, drizzle_con_error(con), "", ds);
5136
drizzle_result_free(&res);
5624
handle_error(command, mysql_errno(mysql), mysql_error(mysql),
5625
mysql_sqlstate(mysql), ds);
5141
5629
if (!disable_result_log)
5143
uint64_t affected_rows= 0; /* Ok to be undef if 'disable_info' is set */
5631
ulonglong affected_rows= 0; /* Ok to be undef if 'disable_info' is set */
5145
if (drizzle_result_column_count(&res))
5147
if (display_metadata)
5148
append_metadata(ds, &res);
5150
if (!display_result_vertically)
5151
append_table_headings(ds, &res);
5153
append_result(ds, &res);
5635
MYSQL_FIELD *fields= mysql_fetch_fields(res);
5636
uint num_fields= mysql_num_fields(res);
5638
if (display_metadata)
5639
append_metadata(ds, fields, num_fields);
5641
if (!display_result_vertically)
5642
append_table_headings(ds, fields, num_fields);
5644
append_result(ds, res);
5157
Need to call drizzle_result_affected_rows() before the "new"
5648
Need to call mysql_affected_rows() before the "new"
5158
5649
query to find the warnings
5160
5651
if (!disable_info)
5161
affected_rows= drizzle_result_affected_rows(&res);
5652
affected_rows= mysql_affected_rows(mysql);
5164
5655
Add all warnings to the result. We can't do this if we are in
5165
5656
the middle of processing results from multi-statement, because
5166
5657
this will break protocol.
5168
if (!disable_warnings)
5659
if (!disable_warnings && !mysql_more_results(mysql))
5170
drizzle_con_remove_options(con, DRIZZLE_CON_NO_RESULT_READ);
5171
if (append_warnings(ds_warnings, con, &res) || ds_warnings->length())
5173
ds->append("Warnings:\n", 10);
5174
ds->append(ds_warnings->c_str(), ds_warnings->length());
5661
if (append_warnings(ds_warnings, mysql) || ds_warnings->length)
5663
dynstr_append_mem(ds, "Warnings:\n", 10);
5664
dynstr_append_mem(ds, ds_warnings->str, ds_warnings->length);
5178
5668
if (!disable_info)
5179
append_info(ds, affected_rows, drizzle_result_info(&res));
5669
append_info(ds, affected_rows, mysql_info(mysql));
5182
drizzle_result_free(&res);
5674
mysql_free_result(res);
5678
} while (!(err= mysql_next_result(mysql)));
5681
/* We got an error from mysql_next_result, maybe expected */
5682
handle_error(command, mysql_errno(mysql), mysql_error(mysql),
5683
mysql_sqlstate(mysql), ds);
5686
DBUG_ASSERT(err == -1); /* Successful and there are no more results */
5185
5688
/* If we come here the query is both executed and read successfully */
5186
5689
handle_no_error(command);
5325
5834
command->query, command->expected_errors.err[0].code.sqlstate);
5842
Run query using prepared statement C API
5846
mysql - mysql handle
5847
command - currrent command pointer
5848
query - query string to execute
5849
query_len - length query string to execute
5850
ds - output buffer where to store result form query
5853
error - function will not return
5856
void run_query_stmt(MYSQL *mysql, struct st_command *command,
5857
char *query, int query_len, DYNAMIC_STRING *ds,
5858
DYNAMIC_STRING *ds_warnings)
5860
MYSQL_RES *res= NULL; /* Note that here 'res' is meta data result set */
5862
DYNAMIC_STRING ds_prepare_warnings;
5863
DYNAMIC_STRING ds_execute_warnings;
5864
DBUG_ENTER("run_query_stmt");
5865
DBUG_PRINT("query", ("'%-.60s'", query));
5868
Init a new stmt if it's not already one created for this connection
5870
if(!(stmt= cur_con->stmt))
5872
if (!(stmt= mysql_stmt_init(mysql)))
5873
die("unable to init stmt structure");
5874
cur_con->stmt= stmt;
5877
/* Init dynamic strings for warnings */
5878
if (!disable_warnings)
5880
init_dynamic_string(&ds_prepare_warnings, NULL, 0, 256);
5881
init_dynamic_string(&ds_execute_warnings, NULL, 0, 256);
5887
if (mysql_stmt_prepare(stmt, query, query_len))
5889
handle_error(command, mysql_stmt_errno(stmt),
5890
mysql_stmt_error(stmt), mysql_stmt_sqlstate(stmt), ds);
5895
Get the warnings from mysql_stmt_prepare and keep them in a
5898
if (!disable_warnings)
5899
append_warnings(&ds_prepare_warnings, mysql);
5902
No need to call mysql_stmt_bind_param() because we have no
5906
#if MYSQL_VERSION_ID >= 50000
5907
if (cursor_protocol_enabled)
5910
Use cursor when retrieving result
5912
ulong type= CURSOR_TYPE_READ_ONLY;
5913
if (mysql_stmt_attr_set(stmt, STMT_ATTR_CURSOR_TYPE, (void*) &type))
5914
die("mysql_stmt_attr_set(STMT_ATTR_CURSOR_TYPE) failed': %d %s",
5915
mysql_stmt_errno(stmt), mysql_stmt_error(stmt));
5922
if (mysql_stmt_execute(stmt))
5924
handle_error(command, mysql_stmt_errno(stmt),
5925
mysql_stmt_error(stmt), mysql_stmt_sqlstate(stmt), ds);
5930
When running in cursor_protocol get the warnings from execute here
5931
and keep them in a separate string for later.
5933
if (cursor_protocol_enabled && !disable_warnings)
5934
append_warnings(&ds_execute_warnings, mysql);
5937
We instruct that we want to update the "max_length" field in
5938
mysql_stmt_store_result(), this is our only way to know how much
5939
buffer to allocate for result data
5943
if (mysql_stmt_attr_set(stmt, STMT_ATTR_UPDATE_MAX_LENGTH, (void*) &one))
5944
die("mysql_stmt_attr_set(STMT_ATTR_UPDATE_MAX_LENGTH) failed': %d %s",
5945
mysql_stmt_errno(stmt), mysql_stmt_error(stmt));
5949
If we got here the statement succeeded and was expected to do so,
5950
get data. Note that this can still give errors found during execution!
5951
Store the result of the query if if will return any fields
5953
if (mysql_stmt_field_count(stmt) && mysql_stmt_store_result(stmt))
5955
handle_error(command, mysql_stmt_errno(stmt),
5956
mysql_stmt_error(stmt), mysql_stmt_sqlstate(stmt), ds);
5960
/* If we got here the statement was both executed and read successfully */
5961
handle_no_error(command);
5962
if (!disable_result_log)
5965
Not all statements creates a result set. If there is one we can
5966
now create another normal result set that contains the meta
5967
data. This set can be handled almost like any other non prepared
5968
statement result set.
5970
if ((res= mysql_stmt_result_metadata(stmt)) != NULL)
5972
/* Take the column count from meta info */
5973
MYSQL_FIELD *fields= mysql_fetch_fields(res);
5974
uint num_fields= mysql_num_fields(res);
5976
if (display_metadata)
5977
append_metadata(ds, fields, num_fields);
5979
if (!display_result_vertically)
5980
append_table_headings(ds, fields, num_fields);
5982
append_stmt_result(ds, stmt, fields, num_fields);
5984
mysql_free_result(res); /* Free normal result set with meta data */
5986
/* Clear prepare warnings */
5987
dynstr_set(&ds_prepare_warnings, NULL);
5992
This is a query without resultset
5996
if (!disable_warnings)
5998
/* Get the warnings from execute */
6000
/* Append warnings to ds - if there are any */
6001
if (append_warnings(&ds_execute_warnings, mysql) ||
6002
ds_execute_warnings.length ||
6003
ds_prepare_warnings.length ||
6004
ds_warnings->length)
6006
dynstr_append_mem(ds, "Warnings:\n", 10);
6007
if (ds_warnings->length)
6008
dynstr_append_mem(ds, ds_warnings->str,
6009
ds_warnings->length);
6010
if (ds_prepare_warnings.length)
6011
dynstr_append_mem(ds, ds_prepare_warnings.str,
6012
ds_prepare_warnings.length);
6013
if (ds_execute_warnings.length)
6014
dynstr_append_mem(ds, ds_execute_warnings.str,
6015
ds_execute_warnings.length);
6020
append_info(ds, mysql_affected_rows(mysql), mysql_info(mysql));
6025
if (!disable_warnings)
6027
dynstr_free(&ds_prepare_warnings);
6028
dynstr_free(&ds_execute_warnings);
6032
/* Close the statement if - no reconnect, need new prepare */
6033
if (mysql->reconnect)
6035
mysql_stmt_close(stmt);
6036
cur_con->stmt= NULL;
6040
We save the return code (mysql_stmt_errno(stmt)) from the last call sent
6041
to the server into the mysqltest builtin variable $mysql_errno. This
6042
variable then can be used from the test case itself.
6045
var_set_errno(mysql_stmt_errno(stmt));
6053
Create a util connection if one does not already exists
6054
and use that to run the query
6055
This is done to avoid implict commit when creating/dropping objects such
6059
int util_query(MYSQL* org_mysql, const char* query){
6062
DBUG_ENTER("util_query");
6064
if(!(mysql= cur_con->util_mysql))
6066
int opt_protocol= MYSQL_PROTOCOL_TCP;
6067
DBUG_PRINT("info", ("Creating util_mysql"));
6068
if (!(mysql= mysql_init(mysql)))
6069
die("Failed in mysql_init()");
6070
mysql_options(mysql, MYSQL_OPT_PROTOCOL,(char*)&opt_protocol);
6072
/* enable local infile, in non-binary builds often disabled by default */
6073
mysql_options(mysql, MYSQL_OPT_LOCAL_INFILE, 0);
6074
safe_connect(mysql, "util", org_mysql->host, org_mysql->user,
6075
org_mysql->passwd, org_mysql->db, org_mysql->port);
6077
cur_con->util_mysql= mysql;
6080
return mysql_query(mysql, query);
5337
drizzle DRIZZLE handle
5338
command currrent command pointer
6091
command currrent command pointer
5340
6093
flags control the phased/stages of query execution to be performed
5341
6094
if QUERY_SEND_FLAG bit is on, the query will be sent. If QUERY_REAP_FLAG
5342
6095
is on the result will be read - for regular query, both bits must be on
5345
static void run_query(struct st_connection *cn,
5346
struct st_command *command,
6098
void run_query(struct st_connection *cn, struct st_command *command, int flags)
5350
string *save_ds= NULL;
6100
MYSQL *mysql= &cn->mysql;
6102
DYNAMIC_STRING *save_ds= NULL;
6103
DYNAMIC_STRING ds_result;
6104
DYNAMIC_STRING ds_sorted;
6105
DYNAMIC_STRING ds_warnings;
6106
DYNAMIC_STRING eval_query;
6109
my_bool view_created= 0, sp_created= 0;
6110
my_bool complete_query= ((flags & QUERY_SEND_FLAG) &&
6111
(flags & QUERY_REAP_FLAG));
6112
DBUG_ENTER("run_query");
6114
init_dynamic_string(&ds_warnings, NULL, 0, 256);
5359
6116
/* Scan for warning before sending to server */
5360
6117
scan_command_for_warnings(command);
5380
6138
Create a temporary dynamic string to contain the output from
5383
if (! command->require_file.empty())
6141
if (command->require_file[0])
6143
init_dynamic_string(&ds_result, "", 1024, 1024);
5385
6144
ds= &ds_result;
5392
6150
Log the query into the output buffer
5394
6152
if (!disable_query_log && (flags & QUERY_SEND_FLAG))
5396
replace_append_mem(ds, query, query_len);
5397
ds->append(delimiter, delimiter_length);
6154
replace_dynstr_append_mem(ds, query, query_len);
6155
dynstr_append_mem(ds, delimiter, delimiter_length);
6156
dynstr_append_mem(ds, "\n", 1);
6159
if (view_protocol_enabled &&
6161
match_re(&view_re, query))
6164
Create the query as a view.
6165
Use replace since view can exist from a failed mysqltest run
6167
DYNAMIC_STRING query_str;
6168
init_dynamic_string(&query_str,
6169
"CREATE OR REPLACE VIEW mysqltest_tmp_v AS ",
6171
dynstr_append_mem(&query_str, query, query_len);
6172
if (util_query(mysql, query_str.str))
6175
Failed to create the view, this is not fatal
6176
just run the query the normal way
6178
DBUG_PRINT("view_create_error",
6179
("Failed to create view '%s': %d: %s", query_str.str,
6180
mysql_errno(mysql), mysql_error(mysql)));
6182
/* Log error to create view */
6183
verbose_msg("Failed to create view '%s' %d: %s", query_str.str,
6184
mysql_errno(mysql), mysql_error(mysql));
6189
Yes, it was possible to create this query as a view
6192
query= (char*)"SELECT * FROM mysqltest_tmp_v";
6193
query_len = strlen(query);
6196
Collect warnings from create of the view that should otherwise
6197
have been produced when the SELECT was executed
6199
append_warnings(&ds_warnings, cur_con->util_mysql);
6202
dynstr_free(&query_str);
6206
if (sp_protocol_enabled &&
6208
match_re(&sp_re, query))
6211
Create the query as a stored procedure
6212
Drop first since sp can exist from a failed mysqltest run
6214
DYNAMIC_STRING query_str;
6215
init_dynamic_string(&query_str,
6216
"DROP PROCEDURE IF EXISTS mysqltest_tmp_sp;",
6218
util_query(mysql, query_str.str);
6219
dynstr_set(&query_str, "CREATE PROCEDURE mysqltest_tmp_sp()\n");
6220
dynstr_append_mem(&query_str, query, query_len);
6221
if (util_query(mysql, query_str.str))
6224
Failed to create the stored procedure for this query,
6225
this is not fatal just run the query the normal way
6227
DBUG_PRINT("sp_create_error",
6228
("Failed to create sp '%s': %d: %s", query_str.str,
6229
mysql_errno(mysql), mysql_error(mysql)));
6231
/* Log error to create sp */
6232
verbose_msg("Failed to create sp '%s' %d: %s", query_str.str,
6233
mysql_errno(mysql), mysql_error(mysql));
6240
query= (char*)"CALL mysqltest_tmp_sp()";
6241
query_len = strlen(query);
6243
dynstr_free(&query_str);
5401
6246
if (display_result_sorted)
5404
Collect the query output in a separate string
5405
that can be sorted before it's added to the
5406
global result string
6249
Collect the query output in a separate string
6250
that can be sorted before it's added to the
6251
global result string
6253
init_dynamic_string(&ds_sorted, "", 1024, 1024);
5408
6254
save_ds= ds; /* Remember original ds */
5409
6255
ds= &ds_sorted;
6259
Find out how to run this query
5413
6261
Always run with normal C API if it's not a complete
6264
If it is a '?' in the query it may be a SQL level prepared
6265
statement already and we can't do it twice
5416
run_query_normal(cn, command, flags, query, query_len,
6267
if (ps_protocol_enabled &&
6269
match_re(&ps_re, query))
6270
run_query_stmt(mysql, command, query, query_len, ds, &ds_warnings);
6272
run_query_normal(cn, command, flags, query, query_len,
5419
6275
if (display_result_sorted)
5421
6277
/* Sort the result set and append it to result */
5422
append_sorted(save_ds, &ds_sorted);
6278
dynstr_append_sorted(save_ds, &ds_sorted);
5426
if (! command->require_file.empty())
6280
dynstr_free(&ds_sorted);
6285
if (util_query(mysql, "DROP PROCEDURE mysqltest_tmp_sp "))
6286
die("Failed to drop sp: %d: %s", mysql_errno(mysql), mysql_error(mysql));
6291
if (util_query(mysql, "DROP VIEW mysqltest_tmp_v "))
6292
die("Failed to drop view: %d: %s",
6293
mysql_errno(mysql), mysql_error(mysql));
6296
if (command->require_file[0])
5428
6298
/* A result file was specified for _this_ query
5429
6299
and the output should be checked against an already
5593
6582
init_builtin_echo();
5595
ds_res.reserve(65536);
5596
ds_progress.reserve(2048);
5597
ds_warning_messages.reserve(2048);
6584
init_dynamic_string(&ds_res, "", 65536, 65536);
6585
init_dynamic_string(&ds_progress, "", 0, 2048);
6586
init_dynamic_string(&ds_warning_messages, "", 0, 2048);
5599
6587
parse_args(argc, argv);
6589
var_set_int("$PS_PROTOCOL", ps_protocol);
6590
var_set_int("$SP_PROTOCOL", sp_protocol);
6591
var_set_int("$VIEW_PROTOCOL", view_protocol);
6592
var_set_int("$CURSOR_PROTOCOL", cursor_protocol);
6594
DBUG_PRINT("info",("result_file: '%s'",
6595
result_file_name ? result_file_name : ""));
6596
if (mysql_server_init(embedded_server_arg_count,
6597
embedded_server_args,
6598
(char**) embedded_server_groups))
6599
die("Can't initialize MySQL server");
5601
6600
server_initialized= 1;
5602
6601
if (cur_file == file_stack && cur_file->file == 0)
5604
6603
cur_file->file= stdin;
5605
cur_file->file_name= strdup("<stdin>");
5606
if (cur_file->file_name == NULL)
5607
die("Out of memory");
6604
cur_file->file_name= my_strdup("<stdin>", MYF(MY_WME));
5608
6605
cur_file->lineno= 1;
6608
ps_protocol_enabled= ps_protocol;
6609
sp_protocol_enabled= sp_protocol;
6610
view_protocol_enabled= view_protocol;
6611
cursor_protocol_enabled= cursor_protocol;
6612
/* Cursor protcol implies ps protocol */
6613
if (cursor_protocol_enabled)
6614
ps_protocol_enabled= 1;
5610
6616
cur_con= connections;
5611
if ((cur_con->drizzle= drizzle_create(NULL)) == NULL)
5612
die("Failed in drizzle_create()");
5613
if (!( drizzle_con_create(cur_con->drizzle, &cur_con->con)))
5614
die("Failed in drizzle_con_create()");
5616
if (!(cur_con->name = strdup("default")))
6617
if (!( mysql_init(&cur_con->mysql)))
6618
die("Failed in mysql_init()");
6620
mysql_options(&cur_con->mysql,MYSQL_OPT_COMPRESS,NullS);
6621
mysql_options(&cur_con->mysql, MYSQL_OPT_LOCAL_INFILE, 0);
6622
mysql_options(&cur_con->mysql, MYSQL_SET_CHARSET_NAME,
6623
charset_info->csname);
6624
int opt_protocol= MYSQL_PROTOCOL_TCP;
6625
mysql_options(&cur_con->mysql,MYSQL_OPT_PROTOCOL,(char*)&opt_protocol);
6626
if (opt_charsets_dir)
6627
mysql_options(&cur_con->mysql, MYSQL_SET_CHARSET_DIR,
6630
#if defined(HAVE_OPENSSL) && !defined(EMBEDDED_LIBRARY)
6634
mysql_ssl_set(&cur_con->mysql, opt_ssl_key, opt_ssl_cert, opt_ssl_ca,
6635
opt_ssl_capath, opt_ssl_cipher);
6636
#if MYSQL_VERSION_ID >= 50000
6637
/* Turn on ssl_verify_server_cert only if host is "localhost" */
6638
opt_ssl_verify_server_cert= opt_host && !strcmp(opt_host, "localhost");
6639
mysql_options(&cur_con->mysql, MYSQL_OPT_SSL_VERIFY_SERVER_CERT,
6640
&opt_ssl_verify_server_cert);
6645
if (!(cur_con->name = my_strdup("default", MYF(MY_WME))))
5617
6646
die("Out of memory");
5619
safe_connect(&cur_con->con, cur_con->name, opt_host, opt_user, opt_pass,
6648
safe_connect(&cur_con->mysql, cur_con->name, opt_host, opt_user, opt_pass,
5620
6649
opt_db, opt_port);
5622
6651
/* Use all time until exit if no explicit 'start_timer' */
5623
6652
timer_start= timer_now();
5626
Initialize $drizzleclient_errno with -1, so we can
6655
Initialize $mysql_errno with -1, so we can
5627
6656
- distinguish it from valid values ( >= 0 ) and
5628
6657
- detect if there was never a command sent to the server
5630
6659
var_set_errno(-1);
5632
/* Update $drizzleclient_get_server_version to that of current connection */
5633
var_set_drizzleclient_get_server_version(&cur_con->con);
6661
/* Update $mysql_get_server_version to that of current connection */
6662
var_set_mysql_get_server_version(&cur_con->mysql);
5635
6664
if (opt_include)
6510
7550
icase - flag, if set to 1 the match is case insensitive
6512
7552
int reg_replace(char** buf_p, int* buf_len_p, char *pattern,
6513
char *replace, char *in_string, int icase, int global)
7553
char *replace, char *string, int icase)
6515
const char *error= NULL;
6518
pcre *re= pcre_compile(pattern,
6519
icase ? PCRE_CASELESS | PCRE_MULTILINE : PCRE_MULTILINE,
6520
&error, &erroffset, NULL);
7556
my_regmatch_t *subs;
7560
int buf_len, need_buf_len;
7561
int cflags= REG_EXTENDED;
7563
char *res_p,*str_p,*str_end;
7565
buf_len= *buf_len_p;
7566
len= strlen(string);
7567
str_end= string + len;
7569
/* start with a buffer of a reasonable size that hopefully will not
7570
need to be reallocated
7572
need_buf_len= len * 2 + 1;
7580
if ((err_code= my_regcomp(&r,pattern,cflags,&my_charset_latin1)))
7582
check_regerr(&r,err_code);
6527
int rc= pcre_exec(re, NULL, in_string, (int)strlen(in_string),
6535
char *substring_to_replace= in_string + ovector[0];
6536
int substring_length= ovector[1] - ovector[0];
6537
*buf_len_p= strlen(in_string) - substring_length + strlen(replace);
6538
char * new_buf = (char *)malloc(*buf_len_p+1);
6539
if (new_buf == NULL)
6545
memset(new_buf, 0, *buf_len_p+1);
6546
strncpy(new_buf, in_string, substring_to_replace-in_string);
6547
strncpy(new_buf+(substring_to_replace-in_string), replace, strlen(replace));
6548
strncpy(new_buf+(substring_to_replace-in_string)+strlen(replace),
6549
substring_to_replace + substring_length,
6552
- (substring_to_replace-in_string));
6560
/* Repeatedly replace the string with the matched regex */
6561
string subject(in_string);
6562
size_t replace_length= strlen(replace);
6563
size_t current_position= 0;
6565
while(0 >= (rc= pcre_exec(re, NULL, subject.c_str() + current_position, subject.length() - current_position,
6568
current_position= static_cast<size_t>(ovector[0]);
6569
replace_length= static_cast<size_t>(ovector[1] - ovector[0]);
6570
subject.replace(current_position, replace_length, replace, replace_length);
6573
char *new_buf = (char *) malloc(subject.length() + 1);
6574
if (new_buf == NULL)
6579
memset(new_buf, 0, subject.length() + 1);
6580
strncpy(new_buf, subject.c_str(), subject.length());
6581
*buf_len_p= subject.length() + 1;
7586
subs= (my_regmatch_t*)my_malloc(sizeof(my_regmatch_t) * (r.re_nsub+1),
7587
MYF(MY_WME+MY_FAE));
7591
replace_end= replace + strlen(replace);
7593
/* for each pattern match instance perform a replacement */
7596
/* find the match */
7597
err_code= my_regexec(&r,str_p, r.re_nsub+1, subs,
7598
(str_p == string) ? REG_NOTBOL : 0);
7600
/* if regular expression error (eg. bad syntax, or out of memory) */
7601
if (err_code && err_code != REG_NOMATCH)
7603
check_regerr(&r,err_code);
7608
/* if match found */
7611
char* expr_p= replace;
7615
we need at least what we have so far in the buffer + the part
7618
need_buf_len= (res_p - buf) + (int) subs[0].rm_so;
7620
/* on this pass, calculate the memory for the result buffer */
7621
while (expr_p < replace_end)
7623
int back_ref_num= -1;
7626
if (c == '\\' && expr_p + 1 < replace_end)
7628
back_ref_num= (int) (expr_p[1] - '0');
7631
/* found a valid back_ref (eg. \1)*/
7632
if (back_ref_num >= 0 && back_ref_num <= (int)r.re_nsub)
7634
regoff_t start_off, end_off;
7635
if ((start_off=subs[back_ref_num].rm_so) > -1 &&
7636
(end_off=subs[back_ref_num].rm_eo) > -1)
7638
need_buf_len += (int) (end_off - start_off);
7650
now that we know the size of the buffer,
7651
make sure it is big enough
7655
/* copy the pre-match part */
7658
memcpy(res_p, str_p, (size_t) subs[0].rm_so);
7659
res_p+= subs[0].rm_so;
7664
/* copy the match and expand back_refs */
7665
while (expr_p < replace_end)
7667
int back_ref_num= -1;
7670
if (c == '\\' && expr_p + 1 < replace_end)
7672
back_ref_num= expr_p[1] - '0';
7675
if (back_ref_num >= 0 && back_ref_num <= (int)r.re_nsub)
7677
regoff_t start_off, end_off;
7678
if ((start_off=subs[back_ref_num].rm_so) > -1 &&
7679
(end_off=subs[back_ref_num].rm_eo) > -1)
7681
int block_len= (int) (end_off - start_off);
7682
memcpy(res_p,str_p + start_off, block_len);
7689
*res_p++ = *expr_p++;
7693
/* handle the post-match part */
7694
if (subs[0].rm_so == subs[0].rm_eo)
7696
if (str_p + subs[0].rm_so >= str_end)
7698
str_p += subs[0].rm_eo ;
7699
*res_p++ = *str_p++;
7703
str_p += subs[0].rm_eo;
7706
else /* no match this time, just copy the string as is */
7708
int left_in_str= str_end-str_p;
7709
need_buf_len= (res_p-buf) + left_in_str;
7711
memcpy(res_p,str_p,left_in_str);
7712
res_p += left_in_str;
7716
my_free(subs, MYF(0));
7720
*buf_len_p= buf_len;
6590
7725
#ifndef WORD_BIT
6591
#define WORD_BIT (8*sizeof(uint32_t))
7726
#define WORD_BIT (8*sizeof(uint))
6594
7729
#define SET_MALLOC_HUNC 64
6595
7730
#define LAST_CHAR_CODE 259
6597
7732
typedef struct st_rep_set {
6598
uint32_t *bits; /* Pointer to used sets */
6599
short next[LAST_CHAR_CODE]; /* Pointer to next sets */
6600
uint32_t found_len; /* Best match to date */
6602
uint32_t table_offset;
6603
uint32_t size_of_bits; /* For convinience */
7733
uint *bits; /* Pointer to used sets */
7734
short next[LAST_CHAR_CODE]; /* Pointer to next sets */
7735
uint found_len; /* Best match to date */
7738
uint size_of_bits; /* For convinience */
6606
7741
typedef struct st_rep_sets {
6607
uint32_t count; /* Number of sets */
6608
uint32_t extra; /* Extra sets in buffer */
6609
uint32_t invisible; /* Sets not chown */
6610
uint32_t size_of_bits;
6611
REP_SET *set,*set_buffer;
6612
uint32_t *bit_buffer;
7742
uint count; /* Number of sets */
7743
uint extra; /* Extra sets in buffer */
7744
uint invisible; /* Sets not chown */
7746
REP_SET *set,*set_buffer;
6615
7750
typedef struct st_found_set {
6616
uint32_t table_offset;
6617
7752
int found_offset;
6620
7755
typedef struct st_follow {
6622
uint32_t table_offset;
6627
int init_sets(REP_SETS *sets,uint32_t states);
7762
int init_sets(REP_SETS *sets,uint states);
6628
7763
REP_SET *make_new_set(REP_SETS *sets);
6629
7764
void make_sets_invisible(REP_SETS *sets);
6630
7765
void free_last_set(REP_SETS *sets);
6631
7766
void free_sets(REP_SETS *sets);
6632
void internal_set_bit(REP_SET *set, uint32_t bit);
6633
void internal_clear_bit(REP_SET *set, uint32_t bit);
7767
void internal_set_bit(REP_SET *set, uint bit);
7768
void internal_clear_bit(REP_SET *set, uint bit);
6634
7769
void or_bits(REP_SET *to,REP_SET *from);
6635
7770
void copy_bits(REP_SET *to,REP_SET *from);
6636
7771
int cmp_bits(REP_SET *set1,REP_SET *set2);
6637
int get_next_bit(REP_SET *set,uint32_t lastpos);
7772
int get_next_bit(REP_SET *set,uint lastpos);
6638
7773
int find_set(REP_SETS *sets,REP_SET *find);
6639
int find_found(FOUND_SET *found_set,uint32_t table_offset,
7774
int find_found(FOUND_SET *found_set,uint table_offset,
6640
7775
int found_offset);
6641
uint32_t start_at_word(char * pos);
6642
uint32_t end_of_word(char * pos);
6644
static uint32_t found_sets=0;
6647
static uint32_t replace_len(char * str)
7776
uint start_at_word(char * pos);
7777
uint end_of_word(char * pos);
7779
static uint found_sets=0;
7782
uint replace_len(char * str)
6652
7787
if (str[0] == '\\' && str[1])
6786
7921
follow_ptr->table_offset=i;
6787
7922
follow_ptr->len=len;
6789
states+=(uint32_t) len+1;
7924
states+=(uint) len+1;
6793
7928
for (set_nr=0,pos=0 ; set_nr < sets.count ; set_nr++)
6795
7930
set=sets.set+set_nr;
6796
default_state= 0; /* Start from beginning */
7931
default_state= 0; /* Start from beginning */
6798
7933
/* If end of found-string not found or start-set with current set */
6800
for (i= UINT32_MAX; (i=get_next_bit(set,i)) ;)
7935
for (i= (uint) ~0; (i=get_next_bit(set,i)) ;)
6802
7937
if (!follow[i].chr)
6804
if (! default_state)
6805
default_state= find_found(found_set,set->table_offset,
6806
set->found_offset+1);
7939
if (! default_state)
7940
default_state= find_found(found_set,set->table_offset,
7941
set->found_offset+1);
6809
copy_bits(sets.set+used_sets,set); /* Save set for changes */
7944
copy_bits(sets.set+used_sets,set); /* Save set for changes */
6810
7945
if (!default_state)
6811
or_bits(sets.set+used_sets,sets.set); /* Can restart from start */
7946
or_bits(sets.set+used_sets,sets.set); /* Can restart from start */
6813
7948
/* Find all chars that follows current sets */
6814
memset(used_chars, 0, sizeof(used_chars));
6815
for (i= UINT32_MAX; (i=get_next_bit(sets.set+used_sets,i)) ;)
7949
bzero((char*) used_chars,sizeof(used_chars));
7950
for (i= (uint) ~0; (i=get_next_bit(sets.set+used_sets,i)) ;)
6817
7952
used_chars[follow[i].chr]=1;
6818
7953
if ((follow[i].chr == SPACE_CHAR && !follow[i+1].chr &&
6819
follow[i].len > 1) || follow[i].chr == END_OF_LINE)
7954
follow[i].len > 1) || follow[i].chr == END_OF_LINE)
6823
7958
/* Mark word_chars used if \b is in state */
6824
7959
if (used_chars[SPACE_CHAR])
6825
7960
for (pos= word_end_chars ; *pos ; pos++)
6826
used_chars[(int) (unsigned char) *pos] = 1;
7961
used_chars[(int) (uchar) *pos] = 1;
6828
7963
/* Handle other used characters */
6829
7964
for (chr= 0 ; chr < 256 ; chr++)
6831
7966
if (! used_chars[chr])
6832
set->next[chr]= chr ? default_state : -1;
7967
set->next[chr]= chr ? default_state : -1;
6835
new_set=make_new_set(&sets);
6836
set=sets.set+set_nr; /* if realloc */
6837
new_set->table_offset=set->table_offset;
6838
new_set->found_len=set->found_len;
6839
new_set->found_offset=set->found_offset+1;
7970
new_set=make_new_set(&sets);
7971
set=sets.set+set_nr; /* if realloc */
7972
new_set->table_offset=set->table_offset;
7973
new_set->found_len=set->found_len;
7974
new_set->found_offset=set->found_offset+1;
6842
for (i= UINT32_MAX ; (i=get_next_bit(sets.set+used_sets,i)) ; )
6844
if (!follow[i].chr || follow[i].chr == chr ||
6845
(follow[i].chr == SPACE_CHAR &&
6846
(is_word_end[chr] ||
6847
(!chr && follow[i].len > 1 && ! follow[i+1].chr))) ||
6848
(follow[i].chr == END_OF_LINE && ! chr))
6850
if ((! chr || (follow[i].chr && !follow[i+1].chr)) &&
6851
follow[i].len > found_end)
6852
found_end=follow[i].len;
6853
if (chr && follow[i].chr)
6854
internal_set_bit(new_set,i+1); /* To next set */
6856
internal_set_bit(new_set,i);
6861
new_set->found_len=0; /* Set for testing if first */
6863
for (i= UINT32_MAX; (i=get_next_bit(new_set,i)) ;)
6865
if ((follow[i].chr == SPACE_CHAR ||
6866
follow[i].chr == END_OF_LINE) && ! chr)
6870
if (follow[bit_nr-1].len < found_end ||
6871
(new_set->found_len &&
6872
(chr == 0 || !follow[bit_nr].chr)))
6873
internal_clear_bit(new_set,i);
6876
if (chr == 0 || !follow[bit_nr].chr)
6878
new_set->table_offset=follow[bit_nr].table_offset;
6879
if (chr || (follow[i].chr == SPACE_CHAR ||
6880
follow[i].chr == END_OF_LINE))
6881
new_set->found_offset=found_end; /* New match */
6882
new_set->found_len=found_end;
6889
set->next[chr] = find_found(found_set,
6890
new_set->table_offset,
6891
new_set->found_offset);
6892
free_last_set(&sets);
6895
set->next[chr] = find_set(&sets,new_set);
6898
set->next[chr] = find_set(&sets,new_set);
7977
for (i= (uint) ~0 ; (i=get_next_bit(sets.set+used_sets,i)) ; )
7979
if (!follow[i].chr || follow[i].chr == chr ||
7980
(follow[i].chr == SPACE_CHAR &&
7981
(is_word_end[chr] ||
7982
(!chr && follow[i].len > 1 && ! follow[i+1].chr))) ||
7983
(follow[i].chr == END_OF_LINE && ! chr))
7985
if ((! chr || (follow[i].chr && !follow[i+1].chr)) &&
7986
follow[i].len > found_end)
7987
found_end=follow[i].len;
7988
if (chr && follow[i].chr)
7989
internal_set_bit(new_set,i+1); /* To next set */
7991
internal_set_bit(new_set,i);
7996
new_set->found_len=0; /* Set for testing if first */
7998
for (i= (uint) ~0; (i=get_next_bit(new_set,i)) ;)
8000
if ((follow[i].chr == SPACE_CHAR ||
8001
follow[i].chr == END_OF_LINE) && ! chr)
8005
if (follow[bit_nr-1].len < found_end ||
8006
(new_set->found_len &&
8007
(chr == 0 || !follow[bit_nr].chr)))
8008
internal_clear_bit(new_set,i);
8011
if (chr == 0 || !follow[bit_nr].chr)
8013
new_set->table_offset=follow[bit_nr].table_offset;
8014
if (chr || (follow[i].chr == SPACE_CHAR ||
8015
follow[i].chr == END_OF_LINE))
8016
new_set->found_offset=found_end; /* New match */
8017
new_set->found_len=found_end;
8024
set->next[chr] = find_found(found_set,
8025
new_set->table_offset,
8026
new_set->found_offset);
8027
free_last_set(&sets);
8030
set->next[chr] = find_set(&sets,new_set);
8033
set->next[chr] = find_set(&sets,new_set);
6903
8038
/* Alloc replace structure for the replace-state-machine */
6905
if ((replace=(REPLACE*) malloc(sizeof(REPLACE)*(sets.count)+
6906
sizeof(REPLACE_STRING)*(found_sets+1)+
6907
sizeof(char *)*count+result_len)))
8040
if ((replace=(REPLACE*) my_malloc(sizeof(REPLACE)*(sets.count)+
8041
sizeof(REPLACE_STRING)*(found_sets+1)+
8042
sizeof(char *)*count+result_len,
8043
MYF(MY_WME | MY_ZEROFILL))))
6909
memset(replace, 0, sizeof(REPLACE)*(sets.count)+
6910
sizeof(REPLACE_STRING)*(found_sets+1)+
6911
sizeof(char *)*count+result_len);
6912
8045
rep_str=(REPLACE_STRING*) (replace+sets.count);
6913
8046
to_array= (char **) (rep_str+found_sets+1);
6914
8047
to_pos=(char *) (to_array+count);
6915
8048
for (i=0 ; i < count ; i++)
6917
8050
to_array[i]=to_pos;
6918
to_pos=strcpy(to_pos,to[i])+strlen(to[i])+1;
8051
to_pos=strmov(to_pos,to[i])+1;
6920
8053
rep_str[0].found=1;
6921
8054
rep_str[0].replace_string=0;
6922
8055
for (i=1 ; i <= found_sets ; i++)
6924
8057
pos=from[found_set[i-1].table_offset];
6925
rep_str[i].found= !memcmp(pos, "\\^", 3) ? 2 : 1;
8058
rep_str[i].found= !bcmp((const uchar*) pos,
8059
(const uchar*) "\\^", 3) ? 2 : 1;
6926
8060
rep_str[i].replace_string=to_array[found_set[i-1].table_offset];
6927
8061
rep_str[i].to_offset=found_set[i-1].found_offset-start_at_word(pos);
6928
8062
rep_str[i].from_offset=found_set[i-1].found_offset-replace_len(pos)+
6931
8065
for (i=0 ; i < sets.count ; i++)
6933
8067
for (j=0 ; j < 256 ; j++)
6934
if (sets.set[i].next[j] >= 0)
6935
replace[i].next[j]=replace+sets.set[i].next[j];
6937
replace[i].next[j]=(REPLACE*) (rep_str+(-sets.set[i].next[j]-1));
8068
if (sets.set[i].next[j] >= 0)
8069
replace[i].next[j]=replace+sets.set[i].next[j];
8071
replace[i].next[j]=(REPLACE*) (rep_str+(-sets.set[i].next[j]-1));
8074
my_free(follow,MYF(0));
6941
8075
free_sets(&sets);
8076
my_free(found_set,MYF(0));
8077
DBUG_PRINT("exit",("Replace table has %d states",sets.count));
8078
DBUG_RETURN(replace);
6947
int init_sets(REP_SETS *sets,uint32_t states)
8082
int init_sets(REP_SETS *sets,uint states)
6949
memset(sets, 0, sizeof(*sets));
8084
bzero((char*) sets,sizeof(*sets));
6950
8085
sets->size_of_bits=((states+7)/8);
6951
if (!(sets->set_buffer=(REP_SET*) malloc(sizeof(REP_SET)*SET_MALLOC_HUNC)))
8086
if (!(sets->set_buffer=(REP_SET*) my_malloc(sizeof(REP_SET)*SET_MALLOC_HUNC,
6953
if (!(sets->bit_buffer=(uint*) malloc(sizeof(uint32_t)*sets->size_of_bits*
8089
if (!(sets->bit_buffer=(uint*) my_malloc(sizeof(uint)*sets->size_of_bits*
8090
SET_MALLOC_HUNC,MYF(MY_WME))))
8092
my_free(sets->set,MYF(0));
7098
8237
set->next[] == -1 is reserved for end without replaces.
7101
int find_found(FOUND_SET *found_set,uint32_t table_offset, int found_offset)
8240
int find_found(FOUND_SET *found_set,uint table_offset, int found_offset)
7104
for (i=0 ; (uint32_t) i < found_sets ; i++)
8243
for (i=0 ; (uint) i < found_sets ; i++)
7105
8244
if (found_set[i].table_offset == table_offset &&
7106
found_set[i].found_offset == found_offset)
8245
found_set[i].found_offset == found_offset)
7108
8247
found_set[i].table_offset=table_offset;
7109
8248
found_set[i].found_offset=found_offset;
7111
return -i-2; /* return new postion */
8250
return -i-2; /* return new postion */
7114
8253
/* Return 1 if regexp starts with \b or ends with \b*/
7116
uint32_t start_at_word(char * pos)
8255
uint start_at_word(char * pos)
7118
return (((!memcmp(pos, "\\b",2) && pos[2]) ||
7119
!memcmp(pos, "\\^", 2)) ? 1 : 0);
8257
return (((!bcmp((const uchar*) pos, (const uchar*) "\\b",2) && pos[2]) ||
8258
!bcmp((const uchar*) pos, (const uchar*) "\\^", 2)) ? 1 : 0);
7122
uint32_t end_of_word(char * pos)
8261
uint end_of_word(char * pos)
7124
char * end= strchr(pos, '\0');
7125
return ((end > pos+2 && !memcmp(end-2, "\\b", 2)) ||
7126
(end >= pos+2 && !memcmp(end-2, "\\$",2))) ? 1 : 0;
8263
char * end=strend(pos);
8264
return ((end > pos+2 && !bcmp((const uchar*) end-2,
8265
(const uchar*) "\\b", 2)) ||
8266
(end >= pos+2 && !bcmp((const uchar*) end-2,
8267
(const uchar*) "\\$",2))) ? 1 : 0;
7129
8270
/****************************************************************************
7130
8271
* Handle replacement of strings
7131
8272
****************************************************************************/
7133
#define PC_MALLOC 256 /* Bytes for pointers */
7134
#define PS_MALLOC 512 /* Bytes for data */
8274
#define PC_MALLOC 256 /* Bytes for pointers */
8275
#define PS_MALLOC 512 /* Bytes for data */
7136
8277
int insert_pointer_name(POINTER_ARRAY *pa,char * name)
7138
uint32_t i,length,old_count;
7139
unsigned char *new_pos;
8279
uint i,length,old_count;
7140
8281
const char **new_array;
8282
DBUG_ENTER("insert_pointer_name");
7143
8284
if (! pa->typelib.count)
7145
8286
if (!(pa->typelib.type_names=(const char **)
7146
malloc(((PC_MALLOC-MALLOC_OVERHEAD)/
7147
(sizeof(char *)+sizeof(*pa->flag))*
7148
(sizeof(char *)+sizeof(*pa->flag))))))
7150
if (!(pa->str= (unsigned char*) malloc(PS_MALLOC-MALLOC_OVERHEAD)))
8287
my_malloc(((PC_MALLOC-MALLOC_OVERHEAD)/
8288
(sizeof(char *)+sizeof(*pa->flag))*
8289
(sizeof(char *)+sizeof(*pa->flag))),MYF(MY_WME))))
8291
if (!(pa->str= (uchar*) my_malloc((uint) (PS_MALLOC-MALLOC_OVERHEAD),
7152
free((char*) pa->typelib.type_names);
8294
my_free((char*) pa->typelib.type_names,MYF(0));
7155
pa->max_count=(PC_MALLOC-MALLOC_OVERHEAD)/(sizeof(unsigned char*)+
7157
pa->flag= (uint8_t*) (pa->typelib.type_names+pa->max_count);
8297
pa->max_count=(PC_MALLOC-MALLOC_OVERHEAD)/(sizeof(uchar*)+
8299
pa->flag= (int7*) (pa->typelib.type_names+pa->max_count);
7159
8301
pa->max_length=PS_MALLOC-MALLOC_OVERHEAD;
7160
8302
pa->array_allocs=1;
7162
length=(uint32_t) strlen(name)+1;
8304
length=(uint) strlen(name)+1;
7163
8305
if (pa->length+length >= pa->max_length)
7165
if (!(new_pos= (unsigned char*)realloc((unsigned char*)pa->str,
7166
(size_t)(pa->max_length+PS_MALLOC))))
8307
if (!(new_pos= (uchar*) my_realloc((uchar*) pa->str,
8308
(uint) (pa->max_length+PS_MALLOC),
7168
8311
if (new_pos != pa->str)
7170
ptrdiff_t diff= PTR_BYTE_DIFF(new_pos,pa->str);
8313
my_ptrdiff_t diff=PTR_BYTE_DIFF(new_pos,pa->str);
7171
8314
for (i=0 ; i < pa->typelib.count ; i++)
7172
pa->typelib.type_names[i]= ADD_TO_PTR(pa->typelib.type_names[i],diff,
8315
pa->typelib.type_names[i]= ADD_TO_PTR(pa->typelib.type_names[i],diff,
7174
8317
pa->str=new_pos;
7176
8319
pa->max_length+=PS_MALLOC;
7178
8321
if (pa->typelib.count >= pa->max_count-1)
7181
8324
pa->array_allocs++;
7182
8325
len=(PC_MALLOC*pa->array_allocs - MALLOC_OVERHEAD);
7184
(const char **)realloc((unsigned char*) pa->typelib.type_names,
7186
(sizeof(unsigned char*)+sizeof(*pa->flag))*
7187
(sizeof(unsigned char*)+sizeof(*pa->flag)))))
8326
if (!(new_array=(const char **) my_realloc((uchar*) pa->typelib.type_names,
8328
(sizeof(uchar*)+sizeof(*pa->flag))*
8329
(sizeof(uchar*)+sizeof(*pa->flag)),
7189
8332
pa->typelib.type_names=new_array;
7190
8333
old_count=pa->max_count;
7191
pa->max_count=len/(sizeof(unsigned char*) + sizeof(*pa->flag));
7192
pa->flag= (uint8_t*) (pa->typelib.type_names+pa->max_count);
7193
memcpy(pa->flag, pa->typelib.type_names+old_count,
7194
old_count*sizeof(*pa->flag));
8334
pa->max_count=len/(sizeof(uchar*) + sizeof(*pa->flag));
8335
pa->flag= (int7*) (pa->typelib.type_names+pa->max_count);
8336
memcpy((uchar*) pa->flag,(char *) (pa->typelib.type_names+old_count),
8337
old_count*sizeof(*pa->flag));
7196
pa->flag[pa->typelib.count]=0; /* Reset flag */
8339
pa->flag[pa->typelib.count]=0; /* Reset flag */
7197
8340
pa->typelib.type_names[pa->typelib.count++]= (char*) pa->str+pa->length;
7198
pa->typelib.type_names[pa->typelib.count]= NULL; /* Put end-mark */
7199
strcpy((char*) pa->str+pa->length,name);
8341
pa->typelib.type_names[pa->typelib.count]= NullS; /* Put end-mark */
8342
VOID(strmov((char*) pa->str+pa->length,name));
7200
8343
pa->length+=length;
7202
8345
} /* insert_pointer_name */