Commit fd015330 authored by monty@mysql.com's avatar monty@mysql.com

Fixed some byte order bugs with prepared statements on machines with high-byte-first. (Bug #4173)

Fixed problem with NULL and derived tables (Bug #4097)
Cleanup of new pushed code
parent 2f76413e
...@@ -780,3 +780,4 @@ ndb/src/common/mgmcommon/printConfig/*.d ...@@ -780,3 +780,4 @@ ndb/src/common/mgmcommon/printConfig/*.d
ndb/src/mgmclient/test_cpcd/*.d ndb/src/mgmclient/test_cpcd/*.d
*.d *.d
libmysqld/examples/client_test.c libmysqld/examples/client_test.c
mysql-test/ndb/ndbcluster
...@@ -59,6 +59,7 @@ ...@@ -59,6 +59,7 @@
#include <sys/stat.h> #include <sys/stat.h>
#include <violite.h> #include <violite.h>
#define MAX_VAR_NAME 256
#define MAX_QUERY 65536 #define MAX_QUERY 65536
#define MAX_COLUMNS 256 #define MAX_COLUMNS 256
#define PAD_SIZE 128 #define PAD_SIZE 128
...@@ -628,6 +629,7 @@ static int check_result(DYNAMIC_STRING* ds, const char* fname, ...@@ -628,6 +629,7 @@ static int check_result(DYNAMIC_STRING* ds, const char* fname,
return error; return error;
} }
VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw, VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw,
my_bool ignore_not_existing) my_bool ignore_not_existing)
{ {
...@@ -642,25 +644,26 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw, ...@@ -642,25 +644,26 @@ VAR* var_get(const char* var_name, const char** var_name_end, my_bool raw,
if (!(digit < 10 && digit >= 0)) if (!(digit < 10 && digit >= 0))
{ {
const char* save_var_name = var_name, *end; const char* save_var_name = var_name, *end;
uint length;
end = (var_name_end) ? *var_name_end : 0; end = (var_name_end) ? *var_name_end : 0;
while (my_isvar(charset_info,*var_name) && var_name != end) while (my_isvar(charset_info,*var_name) && var_name != end)
++var_name; var_name++;
if (var_name == save_var_name) if (var_name == save_var_name)
{ {
if (ignore_not_existing) if (ignore_not_existing)
DBUG_RETURN(0); DBUG_RETURN(0);
die("Empty variable"); die("Empty variable");
} }
length= (uint) (var_name - save_var_name);
if (!(v = (VAR*) hash_search(&var_hash, save_var_name, if (!(v = (VAR*) hash_search(&var_hash, save_var_name, length)) &&
var_name - save_var_name))) length < MAX_VAR_NAME)
{ {
char c=*var_name, *s=(char*)var_name;; char buff[MAX_VAR_NAME+1];
*s=0; strmake(buff, save_var_name, length);
v=var_from_env(save_var_name, ""); v= var_from_env(buff, "");
*s=c;
} }
--var_name; /* Point at last character */ var_name--; /* Point at last character */
} }
else else
v = var_reg + digit; v = var_reg + digit;
......
...@@ -368,7 +368,7 @@ os_file_handle_error( ...@@ -368,7 +368,7 @@ os_file_handle_error(
#undef USE_FILE_LOCK #undef USE_FILE_LOCK
#define USE_FILE_LOCK #define USE_FILE_LOCK
#if defined(UNIV_HOTBACKUP) || defined(__WIN__) || defined(__FreeBSD__) #if defined(UNIV_HOTBACKUP) || defined(__WIN__) || defined(__FreeBSD__) || defined(__NETWARE__)
/* InnoDB Hot Backup does not lock the data files. /* InnoDB Hot Backup does not lock the data files.
* On Windows, mandatory locking is used. * On Windows, mandatory locking is used.
* On FreeBSD with LinuxThreads, advisory locking does not work properly. * On FreeBSD with LinuxThreads, advisory locking does not work properly.
......
...@@ -3001,6 +3001,7 @@ static uint read_binary_date(MYSQL_TIME *tm, uchar **pos) ...@@ -3001,6 +3001,7 @@ static uint read_binary_date(MYSQL_TIME *tm, uchar **pos)
return length; return length;
} }
/* Convert Numeric to buffer types */ /* Convert Numeric to buffer types */
static void send_data_long(MYSQL_BIND *param, MYSQL_FIELD *field, static void send_data_long(MYSQL_BIND *param, MYSQL_FIELD *field,
longlong value) longlong value)
...@@ -3015,26 +3016,26 @@ static void send_data_long(MYSQL_BIND *param, MYSQL_FIELD *field, ...@@ -3015,26 +3016,26 @@ static void send_data_long(MYSQL_BIND *param, MYSQL_FIELD *field,
*param->buffer= (uchar) value; *param->buffer= (uchar) value;
break; break;
case MYSQL_TYPE_SHORT: case MYSQL_TYPE_SHORT:
int2store(buffer, value); shortstore(buffer, value);
break; break;
case MYSQL_TYPE_LONG: case MYSQL_TYPE_LONG:
int4store(buffer, value); longstore(buffer, value);
break; break;
case MYSQL_TYPE_LONGLONG: case MYSQL_TYPE_LONGLONG:
int8store(buffer, value); longlongstore(buffer, value);
break; break;
case MYSQL_TYPE_FLOAT: case MYSQL_TYPE_FLOAT:
{ {
float data= (field_is_unsigned ? (float) ulonglong2double(value) : float data= (field_is_unsigned ? (float) ulonglong2double(value) :
(float) value); (float) value);
float4store(buffer, data); floatstore(buffer, data);
break; break;
} }
case MYSQL_TYPE_DOUBLE: case MYSQL_TYPE_DOUBLE:
{ {
double data= (field_is_unsigned ? ulonglong2double(value) : double data= (field_is_unsigned ? ulonglong2double(value) :
(double) value); (double) value);
float8store(buffer, data); doublestore(buffer, data);
break; break;
} }
default: default:
...@@ -3070,24 +3071,26 @@ static void send_data_double(MYSQL_BIND *param, double value) ...@@ -3070,24 +3071,26 @@ static void send_data_double(MYSQL_BIND *param, double value)
*buffer= (uchar)value; *buffer= (uchar)value;
break; break;
case MYSQL_TYPE_SHORT: case MYSQL_TYPE_SHORT:
int2store(buffer, (short)value); shortstore(buffer, (short)value);
break; break;
case MYSQL_TYPE_LONG: case MYSQL_TYPE_LONG:
int4store(buffer, (long)value); longstore(buffer, (long)value);
break; break;
case MYSQL_TYPE_LONGLONG: case MYSQL_TYPE_LONGLONG:
int8store(buffer, (longlong)value); {
longlong val= (longlong) value;
longlongstore(buffer, val);
break; break;
}
case MYSQL_TYPE_FLOAT: case MYSQL_TYPE_FLOAT:
{ {
float data= (float)value; float data= (float) value;
float4store(buffer, data); floatstore(buffer, data);
break; break;
} }
case MYSQL_TYPE_DOUBLE: case MYSQL_TYPE_DOUBLE:
{ {
double data= (double)value; doublestore(buffer, value);
float8store(buffer, data);
break; break;
} }
default: default:
......
...@@ -53,10 +53,11 @@ static double _nwghts[11]= ...@@ -53,10 +53,11 @@ static double _nwghts[11]=
-3.796875000000000}; -3.796875000000000};
static double *nwghts=_nwghts+5; /* nwghts[i] = -0.5*1.5**i */ static double *nwghts=_nwghts+5; /* nwghts[i] = -0.5*1.5**i */
#define FTB_FLAG_TRUNC 1 /* */ #define FTB_FLAG_TRUNC 1
#define FTB_FLAG_YES 2 /* no two from these three */ /* At most one of the following flags can be set */
#define FTB_FLAG_NO 4 /* YES, NO, WONLY */ #define FTB_FLAG_YES 2
#define FTB_FLAG_WONLY 8 /* should be _ever_ set both */ #define FTB_FLAG_NO 4
#define FTB_FLAG_WONLY 8
typedef struct st_ftb_expr FTB_EXPR; typedef struct st_ftb_expr FTB_EXPR;
struct st_ftb_expr struct st_ftb_expr
......
...@@ -376,8 +376,6 @@ int chk_key(MI_CHECK *param, register MI_INFO *info) ...@@ -376,8 +376,6 @@ int chk_key(MI_CHECK *param, register MI_INFO *info)
for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ; for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ;
rec_per_key_part+=keyinfo->keysegs, key++, keyinfo++) rec_per_key_part+=keyinfo->keysegs, key++, keyinfo++)
{ {
if (*killed_ptr(param))
DBUG_RETURN(-1);
param->key_crc[key]=0; param->key_crc[key]=0;
if (!(((ulonglong) 1 << key) & share->state.key_map)) if (!(((ulonglong) 1 << key) & share->state.key_map))
{ {
......
...@@ -69,7 +69,7 @@ my_bool mi_check_unique(MI_INFO *info, MI_UNIQUEDEF *def, byte *record, ...@@ -69,7 +69,7 @@ my_bool mi_check_unique(MI_INFO *info, MI_UNIQUEDEF *def, byte *record,
ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record) ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record)
{ {
const byte *pos, *end; const byte *pos, *end;
ulong crc= 0; ha_checksum crc= 0;
ulong seed= 4; ulong seed= 4;
HA_KEYSEG *keyseg; HA_KEYSEG *keyseg;
...@@ -109,8 +109,11 @@ ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record) ...@@ -109,8 +109,11 @@ ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record)
end= pos+length; end= pos+length;
if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT) if (type == HA_KEYTYPE_TEXT || type == HA_KEYTYPE_VARTEXT)
{ {
ulong tmp= 0;
keyseg->charset->coll->hash_sort(keyseg->charset, keyseg->charset->coll->hash_sort(keyseg->charset,
(const uchar*) pos, length, &crc, &seed); (const uchar*) pos, length, &tmp,
&seed);
crc^= tmp;
} }
else else
while (pos != end) while (pos != end)
...@@ -118,7 +121,7 @@ ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record) ...@@ -118,7 +121,7 @@ ha_checksum mi_unique_hash(MI_UNIQUEDEF *def, const byte *record)
(((uchar) *(uchar*) pos++))) + (((uchar) *(uchar*) pos++))) +
(crc >> (8*sizeof(ha_checksum)-8)); (crc >> (8*sizeof(ha_checksum)-8));
} }
return (ha_checksum)crc; return crc;
} }
/* /*
......
...@@ -362,13 +362,13 @@ static void usage(void) ...@@ -362,13 +362,13 @@ static void usage(void)
this option is deprecated; you can set variables\n\ this option is deprecated; you can set variables\n\
directly with '--variable-name=value'.\n\ directly with '--variable-name=value'.\n\
-t, --tmpdir=path Path for temporary files. Multiple paths can be\n\ -t, --tmpdir=path Path for temporary files. Multiple paths can be\n\
specified, separated by " specified, separated by ");
#if defined( __WIN__) || defined(OS2) || defined(__NETWARE__) #if defined( __WIN__) || defined(OS2) || defined(__NETWARE__)
"semicolon (;)" puts("semicolon (;)");
#else #else
"colon (:)" puts("colon (:)");
#endif #endif
", they will be used\n\ puts(", they will be used\n\
in a round-robin fashion.\n\ in a round-robin fashion.\n\
-s, --silent Only print errors. One can use two -s to make\n\ -s, --silent Only print errors. One can use two -s to make\n\
myisamchk very silent.\n\ myisamchk very silent.\n\
......
...@@ -16,7 +16,7 @@ USE_MANAGER=0 ...@@ -16,7 +16,7 @@ USE_MANAGER=0
MY_TZ=GMT-3 MY_TZ=GMT-3
TZ=$MY_TZ; export TZ # for UNIX_TIMESTAMP tests to work TZ=$MY_TZ; export TZ # for UNIX_TIMESTAMP tests to work
LOCAL_SOCKET=@MYSQL_UNIX_ADDR@ LOCAL_SOCKET=@MYSQL_UNIX_ADDR@
MYSQL_TCP_PORT=@MYSQL_TCP_PORT@; export MYSQL_TCP_PORT MYSQL_TCP_PORT=@MYSQL_TCP_PORT@
# For query_cache test # For query_cache test
case `uname` in case `uname` in
...@@ -434,7 +434,7 @@ SLAVE_MYERR="$MYSQL_TEST_DIR/var/log/slave.err" ...@@ -434,7 +434,7 @@ SLAVE_MYERR="$MYSQL_TEST_DIR/var/log/slave.err"
CURRENT_TEST="$MYSQL_TEST_DIR/var/log/current_test" CURRENT_TEST="$MYSQL_TEST_DIR/var/log/current_test"
SMALL_SERVER="--key_buffer_size=1M --sort_buffer=256K --max_heap_table_size=1M" SMALL_SERVER="--key_buffer_size=1M --sort_buffer=256K --max_heap_table_size=1M"
export MASTER_MYPORT SLAVE_MYPORT export MASTER_MYPORT SLAVE_MYPORT MYSQL_TCP_PORT
if [ x$SOURCE_DIST = x1 ] ; then if [ x$SOURCE_DIST = x1 ] ; then
MY_BASEDIR=$MYSQL_TEST_DIR MY_BASEDIR=$MYSQL_TEST_DIR
......
...@@ -1831,3 +1831,13 @@ Warnings: ...@@ -1831,3 +1831,13 @@ Warnings:
Note 1276 Field or reference 'up.a' of SELECT #2 was resolved in SELECT #1 Note 1276 Field or reference 'up.a' of SELECT #2 was resolved in SELECT #1
Note 1003 select test.up.a AS `a`,test.up.b AS `b` from test.t1 up where exists(select 1 AS `Not_used` from test.t1 where (test.t1.a = test.up.a)) Note 1003 select test.up.a AS `a`,test.up.b AS `b` from test.t1 up where exists(select 1 AS `Not_used` from test.t1 where (test.t1.a = test.up.a))
drop table t1; drop table t1;
CREATE TABLE t1 (id int(11) default NULL,name varchar(10) default NULL);
INSERT INTO t1 VALUES (1,'Tim'),(2,'Rebecca'),(3,NULL);
CREATE TABLE t2 (id int(11) default NULL, pet varchar(10) default NULL);
INSERT INTO t2 VALUES (1,'Fido'),(2,'Spot'),(3,'Felix');
SELECT a.*, b.* FROM (SELECT * FROM t1) AS a JOIN t2 as b on a.id=b.id;
id name id pet
1 Tim 1 Fido
2 Rebecca 2 Spot
3 NULL 3 Felix
drop table t1,t2;
...@@ -1167,3 +1167,14 @@ insert into t1 values (1,2),(3,4); ...@@ -1167,3 +1167,14 @@ insert into t1 values (1,2),(3,4);
select * from t1 up where exists (select * from t1 where t1.a=up.a); select * from t1 up where exists (select * from t1 where t1.a=up.a);
explain extended select * from t1 up where exists (select * from t1 where t1.a=up.a); explain extended select * from t1 up where exists (select * from t1 where t1.a=up.a);
drop table t1; drop table t1;
#
# Test problem with NULL and derived tables (Bug #4097)
#
CREATE TABLE t1 (id int(11) default NULL,name varchar(10) default NULL);
INSERT INTO t1 VALUES (1,'Tim'),(2,'Rebecca'),(3,NULL);
CREATE TABLE t2 (id int(11) default NULL, pet varchar(10) default NULL);
INSERT INTO t2 VALUES (1,'Fido'),(2,'Spot'),(3,'Felix');
SELECT a.*, b.* FROM (SELECT * FROM t1) AS a JOIN t2 as b on a.id=b.id;
drop table t1,t2;
...@@ -47,10 +47,6 @@ ...@@ -47,10 +47,6 @@
#define ONE_THREAD #define ONE_THREAD
#endif #endif
#define SHUTDOWN_THD
#define MAIN_THD
#define SIGNAL_THD
#ifdef HAVE_purify #ifdef HAVE_purify
#define IF_PURIFY(A,B) (A) #define IF_PURIFY(A,B) (A)
#else #else
...@@ -827,7 +823,6 @@ static void __cdecl kill_server(int sig_ptr) ...@@ -827,7 +823,6 @@ static void __cdecl kill_server(int sig_ptr)
#if defined(USE_ONE_SIGNAL_HAND) || (defined(__NETWARE__) && defined(SIGNALS_DONT_BREAK_READ)) #if defined(USE_ONE_SIGNAL_HAND) || (defined(__NETWARE__) && defined(SIGNALS_DONT_BREAK_READ))
extern "C" pthread_handler_decl(kill_server_thread,arg __attribute__((unused))) extern "C" pthread_handler_decl(kill_server_thread,arg __attribute__((unused)))
{ {
SHUTDOWN_THD;
my_thread_init(); // Initialize new thread my_thread_init(); // Initialize new thread
kill_server(0); kill_server(0);
my_thread_end(); // Normally never reached my_thread_end(); // Normally never reached
...@@ -1716,7 +1711,6 @@ static void init_signals(void) ...@@ -1716,7 +1711,6 @@ static void init_signals(void)
signal(SIGALRM, SIG_IGN); signal(SIGALRM, SIG_IGN);
signal(SIGBREAK,SIG_IGN); signal(SIGBREAK,SIG_IGN);
signal_thread = pthread_self(); signal_thread = pthread_self();
SIGNAL_THD;
} }
static void start_signal_handler(void) static void start_signal_handler(void)
...@@ -2116,7 +2110,6 @@ int uname(struct utsname *a) ...@@ -2116,7 +2110,6 @@ int uname(struct utsname *a)
extern "C" pthread_handler_decl(handle_shutdown,arg) extern "C" pthread_handler_decl(handle_shutdown,arg)
{ {
MSG msg; MSG msg;
SHUTDOWN_THD;
my_thread_init(); my_thread_init();
/* this call should create the message queue for this thread */ /* this call should create the message queue for this thread */
...@@ -2145,7 +2138,6 @@ int STDCALL handle_kill(ulong ctrl_type) ...@@ -2145,7 +2138,6 @@ int STDCALL handle_kill(ulong ctrl_type)
#ifdef OS2 #ifdef OS2
extern "C" pthread_handler_decl(handle_shutdown,arg) extern "C" pthread_handler_decl(handle_shutdown,arg)
{ {
SHUTDOWN_THD;
my_thread_init(); my_thread_init();
// wait semaphore // wait semaphore
......
...@@ -5137,6 +5137,10 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields, ...@@ -5137,6 +5137,10 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields,
recinfo->length=null_pack_length; recinfo->length=null_pack_length;
recinfo++; recinfo++;
bfill(null_flags,null_pack_length,255); // Set null fields bfill(null_flags,null_pack_length,255); // Set null fields
table->null_flags= (uchar*) table->record[0];
table->null_fields= null_count+ hidden_null_count;
table->null_bytes= null_pack_length;
} }
null_count= (blob_count == 0) ? 1 : 0; null_count= (blob_count == 0) ? 1 : 0;
hidden_field_count=param->hidden_field_count; hidden_field_count=param->hidden_field_count;
...@@ -5200,7 +5204,7 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields, ...@@ -5200,7 +5204,7 @@ create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields,
param->copy_field_end=copy; param->copy_field_end=copy;
param->recinfo=recinfo; param->recinfo=recinfo;
store_record(table,default_values); // Make empty default record store_record(table,default_values); // Make empty default record
if (thd->variables.tmp_table_size == ~(ulong) 0) // No limit if (thd->variables.tmp_table_size == ~(ulong) 0) // No limit
table->max_rows= ~(ha_rows) 0; table->max_rows= ~(ha_rows) 0;
...@@ -8326,10 +8330,11 @@ calc_group_buffer(JOIN *join,ORDER *group) ...@@ -8326,10 +8330,11 @@ calc_group_buffer(JOIN *join,ORDER *group)
join->tmp_table_param.group_null_parts=null_parts; join->tmp_table_param.group_null_parts=null_parts;
} }
/* /*
alloc group fields or take prepared (chached) allocate group fields or take prepared (cached)
SYNOPSYS SYNOPSIS
make_group_fields() make_group_fields()
main_join - join of current select main_join - join of current select
curr_join - current join (join of current select or temporary copy of it) curr_join - current join (join of current select or temporary copy of it)
...@@ -8342,22 +8347,21 @@ calc_group_buffer(JOIN *join,ORDER *group) ...@@ -8342,22 +8347,21 @@ calc_group_buffer(JOIN *join,ORDER *group)
static bool static bool
make_group_fields(JOIN *main_join, JOIN *curr_join) make_group_fields(JOIN *main_join, JOIN *curr_join)
{ {
if (main_join->group_fields_cache.elements) if (main_join->group_fields_cache.elements)
{ {
curr_join->group_fields= main_join->group_fields_cache; curr_join->group_fields= main_join->group_fields_cache;
curr_join->sort_and_group= 1; curr_join->sort_and_group= 1;
} }
else else
{ {
if (alloc_group_fields(curr_join, curr_join->group_list)) if (alloc_group_fields(curr_join, curr_join->group_list))
{ return (1);
return (1); main_join->group_fields_cache= curr_join->group_fields;
} }
main_join->group_fields_cache= curr_join->group_fields; return (0);
}
return (0);
} }
/* /*
Get a list of buffers for saveing last group Get a list of buffers for saveing last group
Groups are saved in reverse order for easyer check loop Groups are saved in reverse order for easyer check loop
...@@ -8398,7 +8402,6 @@ test_if_group_changed(List<Item_buff> &list) ...@@ -8398,7 +8402,6 @@ test_if_group_changed(List<Item_buff> &list)
} }
/* /*
Setup copy_fields to save fields at start of new group Setup copy_fields to save fields at start of new group
......
...@@ -370,7 +370,7 @@ bool String::copy(const char *str, uint32 arg_length, ...@@ -370,7 +370,7 @@ bool String::copy(const char *str, uint32 arg_length,
bool String::set_ascii(const char *str, uint32 arg_length) bool String::set_ascii(const char *str, uint32 arg_length)
{ {
if (!(str_charset->mbminlen > 1)) if (str_charset->mbminlen <= 1)
{ {
set(str, arg_length, str_charset); set(str, arg_length, str_charset);
return 0; return 0;
......
...@@ -803,7 +803,7 @@ verb_clause: ...@@ -803,7 +803,7 @@ verb_clause:
; ;
deallocate: deallocate:
DEALLOCATE_SYM PREPARE_SYM ident deallocate_or_drop PREPARE_SYM ident
{ {
THD *thd=YYTHD; THD *thd=YYTHD;
LEX *lex= thd->lex; LEX *lex= thd->lex;
...@@ -816,6 +816,12 @@ deallocate: ...@@ -816,6 +816,12 @@ deallocate:
lex->prepared_stmt_name= $3; lex->prepared_stmt_name= $3;
}; };
deallocate_or_drop:
DEALLOCATE_SYM |
DROP
;
prepare: prepare:
PREPARE_SYM ident FROM prepare_src PREPARE_SYM ident FROM prepare_src
{ {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment