Commit 87e88fe7 authored by patg@govinda.patg.net's avatar patg@govinda.patg.net

Merge pgalbraith@bk-internal.mysql.com:/home/bk/mysql-5.0

into  govinda.patg.net:/home/patg/mysql-build/mysql-5.0-engines-bug19773
parents f242546e d2df283b
...@@ -1558,6 +1558,8 @@ id ...@@ -1558,6 +1558,8 @@ id
3 3
4 4
5 5
DROP TABLE federated.t1;
DROP TABLE federated.t1;
DROP TABLE IF EXISTS federated.bug_17377_table; DROP TABLE IF EXISTS federated.bug_17377_table;
CREATE TABLE federated.bug_17377_table ( CREATE TABLE federated.bug_17377_table (
`fld_cid` bigint(20) NOT NULL auto_increment, `fld_cid` bigint(20) NOT NULL auto_increment,
...@@ -1601,6 +1603,92 @@ fld_cid fld_name fld_parentid fld_delt ...@@ -1601,6 +1603,92 @@ fld_cid fld_name fld_parentid fld_delt
5 Torkel 0 0 5 Torkel 0 0
DROP TABLE federated.t1; DROP TABLE federated.t1;
DROP TABLE federated.bug_17377_table; DROP TABLE federated.bug_17377_table;
create table federated.t1 (i1 int, i2 int, i3 int);
create table federated.t2 (id int, c1 varchar(20), c2 varchar(20));
create table federated.t1 (i1 int, i2 int, i3 int) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:9308/federated/t1';
create table federated.t2 (id int, c1 varchar(20), c2 varchar(20)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:9308/federated/t2';
insert into federated.t1 values (1,5,10),(3,7,12),(4,5,2),(9,10,15),(2,2,2);
insert into federated.t2 values (9,"abc","def"),(5,"opq","lmn"),(2,"test t","t test");
select * from federated.t1 order by i1;
i1 i2 i3
1 5 10
2 2 2
3 7 12
4 5 2
9 10 15
select * from federated.t2;
id c1 c2
9 abc def
5 opq lmn
2 test t t test
update federated.t1,federated.t2 set t1.i2=15, t2.c2="ppc" where t1.i1=t2.id;
select * from federated.t1 order by i1;
i1 i2 i3
1 5 10
2 15 2
3 7 12
4 5 2
9 15 15
select * from federated.t2 order by id;
id c1 c2
2 test t ppc
5 opq lmn
9 abc ppc
delete federated.t1.*,federated.t2.* from federated.t1,federated.t2 where t1.i2=t2.id;
select * from federated.t1 order by i1;
i1 i2 i3
2 15 2
3 7 12
9 15 15
select * from federated.t2 order by id;
id c1 c2
2 test t ppc
9 abc ppc
drop table federated.t1, federated.t2;
drop table federated.t1, federated.t2;
create table federated.t1 (i1 int, i2 int, i3 int, primary key (i1));
create table federated.t2 (id int, c1 varchar(20), c2 varchar(20), primary key (id));
create table federated.t1 (i1 int auto_increment not null, i2 int, i3 int, primary key (i1)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:9308/federated/t1';
create table federated.t2 (id int auto_increment not null, c1 varchar(20), c2 varchar(20), primary key(id)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:9308/federated/t2';
insert into federated.t1 values (1,5,10),(3,7,12),(4,5,2),(9,10,15),(2,2,2);
insert into federated.t2 values (9,"abc","def"),(5,"opq","lmn"),(2,"test t","t test");
select * from federated.t1 order by i1;
i1 i2 i3
1 5 10
2 2 2
3 7 12
4 5 2
9 10 15
select * from federated.t2 order by id;
id c1 c2
2 test t t test
5 opq lmn
9 abc def
update federated.t1,federated.t2 set t1.i2=15, t2.c2="ppc" where t1.i1=t2.id;
select * from federated.t1 order by i1;
i1 i2 i3
1 5 10
2 15 2
3 7 12
4 5 2
9 15 15
select * from federated.t2 order by id;
id c1 c2
2 test t ppc
5 opq lmn
9 abc ppc
delete federated.t1.*,federated.t2.* from federated.t1,federated.t2 where t1.i2=t2.id;
select * from federated.t1 order by i1;
i1 i2 i3
2 15 2
3 7 12
9 15 15
select * from federated.t2 order by id;
id c1 c2
2 test t ppc
9 abc ppc
drop table federated.t1, federated.t2;
drop table federated.t1, federated.t2;
DROP TABLE IF EXISTS federated.t1; DROP TABLE IF EXISTS federated.t1;
DROP DATABASE IF EXISTS federated; DROP DATABASE IF EXISTS federated;
DROP TABLE IF EXISTS federated.t1; DROP TABLE IF EXISTS federated.t1;
......
...@@ -1254,6 +1254,10 @@ SELECT LAST_INSERT_ID(); ...@@ -1254,6 +1254,10 @@ SELECT LAST_INSERT_ID();
INSERT INTO federated.t1 VALUES (); INSERT INTO federated.t1 VALUES ();
SELECT LAST_INSERT_ID(); SELECT LAST_INSERT_ID();
SELECT * FROM federated.t1; SELECT * FROM federated.t1;
DROP TABLE federated.t1;
connection slave;
DROP TABLE federated.t1;
# #
# Bug#17377 Federated Engine returns wrong Data, always the rows # Bug#17377 Federated Engine returns wrong Data, always the rows
...@@ -1309,5 +1313,52 @@ DROP TABLE federated.t1; ...@@ -1309,5 +1313,52 @@ DROP TABLE federated.t1;
connection slave; connection slave;
DROP TABLE federated.bug_17377_table; DROP TABLE federated.bug_17377_table;
#
# BUG 19773 Crash when using multi-table updates, deletes
# with federated tables
#
connection slave;
create table federated.t1 (i1 int, i2 int, i3 int);
create table federated.t2 (id int, c1 varchar(20), c2 varchar(20));
connection master;
eval create table federated.t1 (i1 int, i2 int, i3 int) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:$SLAVE_MYPORT/federated/t1';
eval create table federated.t2 (id int, c1 varchar(20), c2 varchar(20)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:$SLAVE_MYPORT/federated/t2';
insert into federated.t1 values (1,5,10),(3,7,12),(4,5,2),(9,10,15),(2,2,2);
insert into federated.t2 values (9,"abc","def"),(5,"opq","lmn"),(2,"test t","t test");
select * from federated.t1 order by i1;
select * from federated.t2;
update federated.t1,federated.t2 set t1.i2=15, t2.c2="ppc" where t1.i1=t2.id;
select * from federated.t1 order by i1;
select * from federated.t2 order by id;
delete federated.t1.*,federated.t2.* from federated.t1,federated.t2 where t1.i2=t2.id;
select * from federated.t1 order by i1;
select * from federated.t2 order by id;
drop table federated.t1, federated.t2;
connection slave;
drop table federated.t1, federated.t2;
# Test multi updates and deletes with keys
connection slave;
create table federated.t1 (i1 int, i2 int, i3 int, primary key (i1));
create table federated.t2 (id int, c1 varchar(20), c2 varchar(20), primary key (id));
connection master;
eval create table federated.t1 (i1 int auto_increment not null, i2 int, i3 int, primary key (i1)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:$SLAVE_MYPORT/federated/t1';
eval create table federated.t2 (id int auto_increment not null, c1 varchar(20), c2 varchar(20), primary key(id)) ENGINE=FEDERATED CONNECTION='mysql://root@127.0.0.1:$SLAVE_MYPORT/federated/t2';
insert into federated.t1 values (1,5,10),(3,7,12),(4,5,2),(9,10,15),(2,2,2);
insert into federated.t2 values (9,"abc","def"),(5,"opq","lmn"),(2,"test t","t test");
select * from federated.t1 order by i1;
select * from federated.t2 order by id;
update federated.t1,federated.t2 set t1.i2=15, t2.c2="ppc" where t1.i1=t2.id;
select * from federated.t1 order by i1;
select * from federated.t2 order by id;
delete federated.t1.*,federated.t2.* from federated.t1,federated.t2 where t1.i2=t2.id;
select * from federated.t1 order by i1;
select * from federated.t2 order by id;
drop table federated.t1, federated.t2;
connection slave;
drop table federated.t1, federated.t2;
source include/federated_cleanup.inc; source include/federated_cleanup.inc;
...@@ -32,13 +32,14 @@ ...@@ -32,13 +32,14 @@
so to read, that data has to be parsed into fields, to write, fields have to so to read, that data has to be parsed into fields, to write, fields have to
be stored in this format to write to this data file. be stored in this format to write to this data file.
With MySQL Federated storage engine, there will be no local files for each With MySQL Federated storage engine, there will be no local files
table's data (such as .MYD). A foreign database will store the data that would for each table's data (such as .MYD). A foreign database will store
normally be in this file. This will necessitate the use of MySQL client API the data that would normally be in this file. This will necessitate
to read, delete, update, insert this data. The data will have to be retrieve the use of MySQL client API to read, delete, update, insert this
via an SQL call "SELECT * FROM users". Then, to read this data, it will have data. The data will have to be retrieve via an SQL call "SELECT *
to be retrieved via mysql_fetch_row one row at a time, then converted from FROM users". Then, to read this data, it will have to be retrieved
the column in this select into the format that the handler expects. via mysql_fetch_row one row at a time, then converted from the
column in this select into the format that the handler expects.
The create table will simply create the .frm file, and within the The create table will simply create the .frm file, and within the
"CREATE TABLE" SQL, there SHALL be any of the following : "CREATE TABLE" SQL, there SHALL be any of the following :
...@@ -395,8 +396,8 @@ handlerton federated_hton= { ...@@ -395,8 +396,8 @@ handlerton federated_hton= {
static byte *federated_get_key(FEDERATED_SHARE *share, uint *length, static byte *federated_get_key(FEDERATED_SHARE *share, uint *length,
my_bool not_used __attribute__ ((unused))) my_bool not_used __attribute__ ((unused)))
{ {
*length= share->table_name_length; *length= share->connect_string_length;
return (byte*) share->table_name; return (byte*) share->scheme;
} }
/* /*
...@@ -416,7 +417,7 @@ bool federated_db_init() ...@@ -416,7 +417,7 @@ bool federated_db_init()
DBUG_ENTER("federated_db_init"); DBUG_ENTER("federated_db_init");
if (pthread_mutex_init(&federated_mutex, MY_MUTEX_INIT_FAST)) if (pthread_mutex_init(&federated_mutex, MY_MUTEX_INIT_FAST))
goto error; goto error;
if (hash_init(&federated_open_tables, system_charset_info, 32, 0, 0, if (hash_init(&federated_open_tables, &my_charset_bin, 32, 0, 0,
(hash_get_key) federated_get_key, 0, 0)) (hash_get_key) federated_get_key, 0, 0))
{ {
VOID(pthread_mutex_destroy(&federated_mutex)); VOID(pthread_mutex_destroy(&federated_mutex));
...@@ -513,6 +514,7 @@ static int check_foreign_data_source(FEDERATED_SHARE *share, ...@@ -513,6 +514,7 @@ static int check_foreign_data_source(FEDERATED_SHARE *share,
} }
else else
{ {
int escaped_table_name_length= 0;
/* /*
Since we do not support transactions at this version, we can let the Since we do not support transactions at this version, we can let the
client API silently reconnect. For future versions, we will need more client API silently reconnect. For future versions, we will need more
...@@ -531,17 +533,16 @@ static int check_foreign_data_source(FEDERATED_SHARE *share, ...@@ -531,17 +533,16 @@ static int check_foreign_data_source(FEDERATED_SHARE *share,
query.append(FEDERATED_STAR); query.append(FEDERATED_STAR);
query.append(FEDERATED_FROM); query.append(FEDERATED_FROM);
query.append(FEDERATED_BTICK); query.append(FEDERATED_BTICK);
escape_string_for_mysql(&my_charset_bin, (char *)escaped_table_name, escaped_table_name_length=
escape_string_for_mysql(&my_charset_bin, (char*)escaped_table_name,
sizeof(escaped_table_name), sizeof(escaped_table_name),
share->table_name, share->table_name,
share->table_name_length); share->table_name_length);
query.append(escaped_table_name); query.append(escaped_table_name, escaped_table_name_length);
query.append(FEDERATED_BTICK); query.append(FEDERATED_BTICK);
query.append(FEDERATED_WHERE); query.append(FEDERATED_WHERE);
query.append(FEDERATED_FALSE); query.append(FEDERATED_FALSE);
DBUG_PRINT("info", ("check_foreign_data_source query %s",
query.c_ptr_quick()));
if (mysql_real_query(mysql, query.ptr(), query.length())) if (mysql_real_query(mysql, query.ptr(), query.length()))
{ {
error_code= table_create_flag ? error_code= table_create_flag ?
...@@ -636,8 +637,7 @@ static int parse_url(FEDERATED_SHARE *share, TABLE *table, ...@@ -636,8 +637,7 @@ static int parse_url(FEDERATED_SHARE *share, TABLE *table,
table->s->connect_string.length, table->s->connect_string.length,
MYF(0)); MYF(0));
// Add a null for later termination of table name share->connect_string_length= table->s->connect_string.length;
share->scheme[table->s->connect_string.length]= 0;
DBUG_PRINT("info",("parse_url alloced share->scheme %lx", share->scheme)); DBUG_PRINT("info",("parse_url alloced share->scheme %lx", share->scheme));
/* /*
...@@ -739,8 +739,7 @@ static int parse_url(FEDERATED_SHARE *share, TABLE *table, ...@@ -739,8 +739,7 @@ static int parse_url(FEDERATED_SHARE *share, TABLE *table,
ha_federated::ha_federated(TABLE *table_arg) ha_federated::ha_federated(TABLE *table_arg)
:handler(&federated_hton, table_arg), :handler(&federated_hton, table_arg),
mysql(0), stored_result(0), mysql(0), stored_result(0)
ref_length(sizeof(MYSQL_ROW_OFFSET)), current_position(0)
{} {}
...@@ -751,6 +750,7 @@ ha_federated::ha_federated(TABLE *table_arg) ...@@ -751,6 +750,7 @@ ha_federated::ha_federated(TABLE *table_arg)
convert_row_to_internal_format() convert_row_to_internal_format()
record Byte pointer to record record Byte pointer to record
row MySQL result set row from fetchrow() row MySQL result set row from fetchrow()
result Result set to use
DESCRIPTION DESCRIPTION
This method simply iterates through a row returned via fetchrow with This method simply iterates through a row returned via fetchrow with
...@@ -763,14 +763,15 @@ ha_federated::ha_federated(TABLE *table_arg) ...@@ -763,14 +763,15 @@ ha_federated::ha_federated(TABLE *table_arg)
0 After fields have had field values stored from record 0 After fields have had field values stored from record
*/ */
uint ha_federated::convert_row_to_internal_format(byte *record, MYSQL_ROW row) uint ha_federated::convert_row_to_internal_format(byte *record,
MYSQL_ROW row,
MYSQL_RES *result)
{ {
ulong *lengths; ulong *lengths;
Field **field; Field **field;
DBUG_ENTER("ha_federated::convert_row_to_internal_format"); DBUG_ENTER("ha_federated::convert_row_to_internal_format");
lengths= mysql_fetch_lengths(stored_result); lengths= mysql_fetch_lengths(result);
memset(record, 0, table->s->null_bytes);
for (field= table->field; *field; field++) for (field= table->field; *field; field++)
{ {
...@@ -1298,12 +1299,11 @@ bool ha_federated::create_where_from_key(String *to, ...@@ -1298,12 +1299,11 @@ bool ha_federated::create_where_from_key(String *to,
static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table) static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table)
{ {
char *select_query, *tmp_table_name; char *select_query;
char query_buffer[FEDERATED_QUERY_BUFFER_SIZE]; char query_buffer[FEDERATED_QUERY_BUFFER_SIZE];
uint tmp_table_name_length;
Field **field; Field **field;
String query(query_buffer, sizeof(query_buffer), &my_charset_bin); String query(query_buffer, sizeof(query_buffer), &my_charset_bin);
FEDERATED_SHARE *share; FEDERATED_SHARE *share= NULL, tmp_share;
/* /*
In order to use this string, we must first zero it's length, In order to use this string, we must first zero it's length,
or it will contain garbage or it will contain garbage
...@@ -1311,12 +1311,15 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table) ...@@ -1311,12 +1311,15 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table)
query.length(0); query.length(0);
pthread_mutex_lock(&federated_mutex); pthread_mutex_lock(&federated_mutex);
tmp_table_name= (char *)table->s->table_name;
tmp_table_name_length= (uint) strlen(tmp_table_name);
if (parse_url(&tmp_share, table, 0))
goto error;
/* TODO: change tmp_share.scheme to LEX_STRING object */
if (!(share= (FEDERATED_SHARE *) hash_search(&federated_open_tables, if (!(share= (FEDERATED_SHARE *) hash_search(&federated_open_tables,
(byte*) table_name, (byte*) tmp_share.scheme,
strlen(table_name)))) tmp_share.
connect_string_length)))
{ {
query.set_charset(system_charset_info); query.set_charset(system_charset_info);
query.append(FEDERATED_SELECT); query.append(FEDERATED_SELECT);
...@@ -1334,24 +1337,20 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table) ...@@ -1334,24 +1337,20 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table)
if (!(share= (FEDERATED_SHARE *) if (!(share= (FEDERATED_SHARE *)
my_multi_malloc(MYF(MY_WME), my_multi_malloc(MYF(MY_WME),
&share, sizeof(*share), &share, sizeof(*share),
&tmp_table_name, tmp_table_name_length+ 1,
&select_query, &select_query,
query.length()+table->s->connect_string.length+1, query.length()+table->s->connect_string.length+1,
NullS))) NullS)))
{
pthread_mutex_unlock(&federated_mutex);
return NULL;
}
if (parse_url(share, table, 0))
goto error; goto error;
memcpy(share, &tmp_share, sizeof(tmp_share));
share->table_name_length= strlen(share->table_name);
/* TODO: share->table_name to LEX_STRING object */
query.append(share->table_name, share->table_name_length); query.append(share->table_name, share->table_name_length);
query.append(FEDERATED_BTICK); query.append(FEDERATED_BTICK);
share->select_query= select_query; share->select_query= select_query;
strmov(share->select_query, query.ptr()); strmov(share->select_query, query.ptr());
share->use_count= 0; share->use_count= 0;
share->table_name_length= strlen(share->table_name);
DBUG_PRINT("info", DBUG_PRINT("info",
("share->select_query %s", share->select_query)); ("share->select_query %s", share->select_query));
...@@ -1367,11 +1366,8 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table) ...@@ -1367,11 +1366,8 @@ static FEDERATED_SHARE *get_share(const char *table_name, TABLE *table)
error: error:
pthread_mutex_unlock(&federated_mutex); pthread_mutex_unlock(&federated_mutex);
if (share->scheme) my_free((gptr) tmp_share.scheme, MYF(MY_ALLOW_ZERO_PTR));
{ my_free((gptr) share, MYF(MY_ALLOW_ZERO_PTR));
my_free((gptr) share->scheme, MYF(0));
share->scheme= 0;
}
return NULL; return NULL;
} }
...@@ -1391,13 +1387,7 @@ static int free_share(FEDERATED_SHARE *share) ...@@ -1391,13 +1387,7 @@ static int free_share(FEDERATED_SHARE *share)
{ {
hash_delete(&federated_open_tables, (byte*) share); hash_delete(&federated_open_tables, (byte*) share);
my_free((gptr) share->scheme, MYF(MY_ALLOW_ZERO_PTR)); my_free((gptr) share->scheme, MYF(MY_ALLOW_ZERO_PTR));
share->scheme= 0;
if (share->socket)
{
my_free((gptr) share->socket, MYF(MY_ALLOW_ZERO_PTR)); my_free((gptr) share->socket, MYF(MY_ALLOW_ZERO_PTR));
share->socket= 0;
}
thr_lock_delete(&share->lock); thr_lock_delete(&share->lock);
VOID(pthread_mutex_destroy(&share->mutex)); VOID(pthread_mutex_destroy(&share->mutex));
my_free((gptr) share, MYF(0)); my_free((gptr) share, MYF(0));
...@@ -1459,7 +1449,7 @@ int ha_federated::open(const char *name, int mode, uint test_if_locked) ...@@ -1459,7 +1449,7 @@ int ha_federated::open(const char *name, int mode, uint test_if_locked)
/* Connect to foreign database mysql_real_connect() */ /* Connect to foreign database mysql_real_connect() */
mysql= mysql_init(0); mysql= mysql_init(0);
if (!mysql_real_connect(mysql, if (!mysql || !mysql_real_connect(mysql,
share->hostname, share->hostname,
share->username, share->username,
share->password, share->password,
...@@ -1467,14 +1457,21 @@ int ha_federated::open(const char *name, int mode, uint test_if_locked) ...@@ -1467,14 +1457,21 @@ int ha_federated::open(const char *name, int mode, uint test_if_locked)
share->port, share->port,
share->socket, 0)) share->socket, 0))
{ {
free_share(share);
DBUG_RETURN(stash_remote_error()); DBUG_RETURN(stash_remote_error());
} }
/* /*
Since we do not support transactions at this version, we can let the client Since we do not support transactions at this version, we can let the client
API silently reconnect. For future versions, we will need more logic to deal API silently reconnect. For future versions, we will need more logic to
with transactions deal with transactions
*/ */
mysql->reconnect= 1; mysql->reconnect= 1;
ref_length= (table->s->primary_key != MAX_KEY ?
table->key_info[table->s->primary_key].key_length :
table->s->reclength);
DBUG_PRINT("info", ("ref_length: %u", ref_length));
DBUG_RETURN(0); DBUG_RETURN(0);
} }
...@@ -1498,12 +1495,11 @@ int ha_federated::close(void) ...@@ -1498,12 +1495,11 @@ int ha_federated::close(void)
/* free the result set */ /* free the result set */
if (stored_result) if (stored_result)
{ {
DBUG_PRINT("info",
("mysql_free_result result at address %lx", stored_result));
mysql_free_result(stored_result); mysql_free_result(stored_result);
stored_result= 0; stored_result= 0;
} }
/* Disconnect from mysql */ /* Disconnect from mysql */
if (mysql) // QQ is this really needed
mysql_close(mysql); mysql_close(mysql);
retval= free_share(share); retval= free_share(share);
DBUG_RETURN(retval); DBUG_RETURN(retval);
...@@ -1694,15 +1690,13 @@ int ha_federated::write_row(byte *buf) ...@@ -1694,15 +1690,13 @@ int ha_federated::write_row(byte *buf)
/* add the values */ /* add the values */
insert_string.append(values_string); insert_string.append(values_string);
DBUG_PRINT("info", ("insert query %s", insert_string.c_ptr_quick()));
if (mysql_real_query(mysql, insert_string.ptr(), insert_string.length())) if (mysql_real_query(mysql, insert_string.ptr(), insert_string.length()))
{ {
DBUG_RETURN(stash_remote_error()); DBUG_RETURN(stash_remote_error());
} }
/* /*
If the table we've just written a record to contains an auto_increment field, If the table we've just written a record to contains an auto_increment
then store the last_insert_id() value from the foreign server field, then store the last_insert_id() value from the foreign server
*/ */
if (table->next_number_field) if (table->next_number_field)
update_auto_increment(); update_auto_increment();
...@@ -1923,7 +1917,7 @@ int ha_federated::update_row(const byte *old_data, byte *new_data) ...@@ -1923,7 +1917,7 @@ int ha_federated::update_row(const byte *old_data, byte *new_data)
/* /*
This will delete a row. 'buf' will contain a copy of the row to be =deleted. This will delete a row. 'buf' will contain a copy of the row to be =deleted.
The server will call this right after the current row has been called (from The server will call this right after the current row has been called (from
either a previous rnd_nexT() or index call). either a previous rnd_next() or index call).
If you keep a pointer to the last row or can access a primary key it will If you keep a pointer to the last row or can access a primary key it will
make doing the deletion quite a bit easier. make doing the deletion quite a bit easier.
Keep in mind that the server does no guarentee consecutive deletions. Keep in mind that the server does no guarentee consecutive deletions.
...@@ -1983,6 +1977,7 @@ int ha_federated::delete_row(const byte *buf) ...@@ -1983,6 +1977,7 @@ int ha_federated::delete_row(const byte *buf)
DBUG_RETURN(stash_remote_error()); DBUG_RETURN(stash_remote_error());
} }
deleted+= mysql->affected_rows; deleted+= mysql->affected_rows;
records-= mysql->affected_rows;
DBUG_PRINT("info", DBUG_PRINT("info",
("rows deleted %d rows deleted for all time %d", ("rows deleted %d rows deleted for all time %d",
int(mysql->affected_rows), deleted)); int(mysql->affected_rows), deleted));
...@@ -1999,12 +1994,15 @@ int ha_federated::delete_row(const byte *buf) ...@@ -1999,12 +1994,15 @@ int ha_federated::delete_row(const byte *buf)
*/ */
int ha_federated::index_read(byte *buf, const byte *key, int ha_federated::index_read(byte *buf, const byte *key,
uint key_len, enum ha_rkey_function find_flag) uint key_len, ha_rkey_function find_flag)
{ {
int retval;
DBUG_ENTER("ha_federated::index_read"); DBUG_ENTER("ha_federated::index_read");
retval= index_read_idx(buf, active_index, key, key_len, find_flag);
DBUG_RETURN(retval); if (stored_result)
mysql_free_result(stored_result);
DBUG_RETURN(index_read_idx_with_result_set(buf, active_index, key,
key_len, find_flag,
&stored_result));
} }
...@@ -2013,12 +2011,45 @@ int ha_federated::index_read(byte *buf, const byte *key, ...@@ -2013,12 +2011,45 @@ int ha_federated::index_read(byte *buf, const byte *key,
row if any. This is only used to read whole keys. row if any. This is only used to read whole keys.
This method is called via index_read in the case of a WHERE clause using This method is called via index_read in the case of a WHERE clause using
a regular non-primary key index, OR is called DIRECTLY when the WHERE clause a primary key index OR is called DIRECTLY when the WHERE clause
uses a PRIMARY KEY index. uses a PRIMARY KEY index.
NOTES
This uses an internal result set that is deleted before function
returns. We need to be able to be calable from ha_rnd_pos()
*/ */
int ha_federated::index_read_idx(byte *buf, uint index, const byte *key, int ha_federated::index_read_idx(byte *buf, uint index, const byte *key,
uint key_len, enum ha_rkey_function find_flag) uint key_len, enum ha_rkey_function find_flag)
{
int retval;
MYSQL_RES *mysql_result;
DBUG_ENTER("ha_federated::index_read_idx");
if ((retval= index_read_idx_with_result_set(buf, index, key,
key_len, find_flag,
&mysql_result)))
DBUG_RETURN(retval);
mysql_free_result(mysql_result);
DBUG_RETURN(retval);
}
/*
Create result set for rows matching query and return first row
RESULT
0 ok In this case *result will contain the result set
table->status == 0
# error In this case *result will contain 0
table->status == STATUS_NOT_FOUND
*/
int ha_federated::index_read_idx_with_result_set(byte *buf, uint index,
const byte *key,
uint key_len,
ha_rkey_function find_flag,
MYSQL_RES **result)
{ {
int retval; int retval;
char error_buffer[FEDERATED_QUERY_BUFFER_SIZE]; char error_buffer[FEDERATED_QUERY_BUFFER_SIZE];
...@@ -2031,8 +2062,9 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key, ...@@ -2031,8 +2062,9 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key,
sizeof(sql_query_buffer), sizeof(sql_query_buffer),
&my_charset_bin); &my_charset_bin);
key_range range; key_range range;
DBUG_ENTER("ha_federated::index_read_idx"); DBUG_ENTER("ha_federated::index_read_idx_with_result_set");
*result= 0; // In case of errors
index_string.length(0); index_string.length(0);
sql_query.length(0); sql_query.length(0);
statistic_increment(table->in_use->status_var.ha_read_key_count, statistic_increment(table->in_use->status_var.ha_read_key_count,
...@@ -2049,20 +2081,6 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key, ...@@ -2049,20 +2081,6 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key,
NULL, 0); NULL, 0);
sql_query.append(index_string); sql_query.append(index_string);
DBUG_PRINT("info",
("current key %d key value %s index_string value %s length %d",
index, (char*) key, index_string.c_ptr_quick(),
index_string.length()));
DBUG_PRINT("info",
("current position %d sql_query %s", current_position,
sql_query.c_ptr_quick()));
if (stored_result)
{
mysql_free_result(stored_result);
stored_result= 0;
}
if (mysql_real_query(mysql, sql_query.ptr(), sql_query.length())) if (mysql_real_query(mysql, sql_query.ptr(), sql_query.length()))
{ {
my_sprintf(error_buffer, (error_buffer, "error: %d '%s'", my_sprintf(error_buffer, (error_buffer, "error: %d '%s'",
...@@ -2070,50 +2088,41 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key, ...@@ -2070,50 +2088,41 @@ int ha_federated::index_read_idx(byte *buf, uint index, const byte *key,
retval= ER_QUERY_ON_FOREIGN_DATA_SOURCE; retval= ER_QUERY_ON_FOREIGN_DATA_SOURCE;
goto error; goto error;
} }
stored_result= mysql_store_result(mysql); if (!(*result= mysql_store_result(mysql)))
if (!stored_result)
{ {
retval= HA_ERR_END_OF_FILE; retval= HA_ERR_END_OF_FILE;
goto error; goto error;
} }
/* if (!(retval= read_next(buf, *result)))
This basically says that the record in table->record[0] is legal, DBUG_RETURN(retval);
and that it is ok to use this record, for whatever reason, such
as with a join (without it, joins will not work)
*/
table->status= 0;
retval= rnd_next(buf); mysql_free_result(*result);
*result= 0;
table->status= STATUS_NOT_FOUND;
DBUG_RETURN(retval); DBUG_RETURN(retval);
error: error:
if (stored_result)
{
mysql_free_result(stored_result);
stored_result= 0;
}
table->status= STATUS_NOT_FOUND; table->status= STATUS_NOT_FOUND;
my_error(retval, MYF(0), error_buffer); my_error(retval, MYF(0), error_buffer);
DBUG_RETURN(retval); DBUG_RETURN(retval);
} }
/* Initialized at each key walk (called multiple times unlike rnd_init()) */ /* Initialized at each key walk (called multiple times unlike rnd_init()) */
int ha_federated::index_init(uint keynr) int ha_federated::index_init(uint keynr)
{ {
DBUG_ENTER("ha_federated::index_init"); DBUG_ENTER("ha_federated::index_init");
DBUG_PRINT("info", DBUG_PRINT("info", ("table: '%s' key: %u", table->s->table_name, keynr));
("table: '%s' key: %d", table->s->table_name, keynr));
active_index= keynr; active_index= keynr;
DBUG_RETURN(0); DBUG_RETURN(0);
} }
/*
int read_range_first(const key_range *start_key, /*
const key_range *end_key, Read first range
bool eq_range, bool sorted);
*/ */
int ha_federated::read_range_first(const key_range *start_key, int ha_federated::read_range_first(const key_range *start_key,
const key_range *end_key, const key_range *end_key,
bool eq_range, bool sorted) bool eq_range, bool sorted)
...@@ -2125,8 +2134,7 @@ int ha_federated::read_range_first(const key_range *start_key, ...@@ -2125,8 +2134,7 @@ int ha_federated::read_range_first(const key_range *start_key,
&my_charset_bin); &my_charset_bin);
DBUG_ENTER("ha_federated::read_range_first"); DBUG_ENTER("ha_federated::read_range_first");
if (start_key == NULL && end_key == NULL) DBUG_ASSERT(!(start_key == NULL && end_key == NULL));
DBUG_RETURN(0);
sql_query.length(0); sql_query.length(0);
sql_query.append(share->select_query); sql_query.append(share->select_query);
...@@ -2134,6 +2142,11 @@ int ha_federated::read_range_first(const key_range *start_key, ...@@ -2134,6 +2142,11 @@ int ha_federated::read_range_first(const key_range *start_key,
&table->key_info[active_index], &table->key_info[active_index],
start_key, end_key, 0); start_key, end_key, 0);
if (stored_result)
{
mysql_free_result(stored_result);
stored_result= 0;
}
if (mysql_real_query(mysql, sql_query.ptr(), sql_query.length())) if (mysql_real_query(mysql, sql_query.ptr(), sql_query.length()))
{ {
retval= ER_QUERY_ON_FOREIGN_DATA_SOURCE; retval= ER_QUERY_ON_FOREIGN_DATA_SOURCE;
...@@ -2141,38 +2154,21 @@ int ha_federated::read_range_first(const key_range *start_key, ...@@ -2141,38 +2154,21 @@ int ha_federated::read_range_first(const key_range *start_key,
} }
sql_query.length(0); sql_query.length(0);
if (stored_result) if (!(stored_result= mysql_store_result(mysql)))
{
DBUG_PRINT("info",
("mysql_free_result address %lx", stored_result));
mysql_free_result(stored_result);
stored_result= 0;
}
stored_result= mysql_store_result(mysql);
if (!stored_result)
{ {
retval= HA_ERR_END_OF_FILE; retval= HA_ERR_END_OF_FILE;
goto error; goto error;
} }
/* This was successful, please let it be known! */ retval= read_next(table->record[0], stored_result);
table->status= 0;
retval= rnd_next(table->record[0]);
DBUG_RETURN(retval); DBUG_RETURN(retval);
error: error:
table->status= STATUS_NOT_FOUND; table->status= STATUS_NOT_FOUND;
if (stored_result)
{
DBUG_PRINT("info", ("mysql_free_result address %lx", stored_result));
mysql_free_result(stored_result);
stored_result= 0;
}
DBUG_RETURN(retval); DBUG_RETURN(retval);
} }
int ha_federated::read_range_next() int ha_federated::read_range_next()
{ {
int retval; int retval;
...@@ -2185,13 +2181,13 @@ int ha_federated::read_range_next() ...@@ -2185,13 +2181,13 @@ int ha_federated::read_range_next()
/* Used to read forward through the index. */ /* Used to read forward through the index. */
int ha_federated::index_next(byte *buf) int ha_federated::index_next(byte *buf)
{ {
int retval;
DBUG_ENTER("ha_federated::index_next"); DBUG_ENTER("ha_federated::index_next");
statistic_increment(table->in_use->status_var.ha_read_next_count, statistic_increment(table->in_use->status_var.ha_read_next_count,
&LOCK_status); &LOCK_status);
retval= rnd_next(buf); DBUG_RETURN(read_next(buf, stored_result));
DBUG_RETURN(retval);
} }
/* /*
rnd_init() is called when the system wants the storage engine to do a table rnd_init() is called when the system wants the storage engine to do a table
scan. scan.
...@@ -2245,11 +2241,8 @@ int ha_federated::rnd_init(bool scan) ...@@ -2245,11 +2241,8 @@ int ha_federated::rnd_init(bool scan)
if (scan) if (scan)
{ {
DBUG_PRINT("info", ("share->select_query %s", share->select_query));
if (stored_result) if (stored_result)
{ {
DBUG_PRINT("info",
("mysql_free_result address %lx", stored_result));
mysql_free_result(stored_result); mysql_free_result(stored_result);
stored_result= 0; stored_result= 0;
} }
...@@ -2269,24 +2262,22 @@ int ha_federated::rnd_init(bool scan) ...@@ -2269,24 +2262,22 @@ int ha_federated::rnd_init(bool scan)
DBUG_RETURN(stash_remote_error()); DBUG_RETURN(stash_remote_error());
} }
int ha_federated::rnd_end() int ha_federated::rnd_end()
{ {
int retval;
DBUG_ENTER("ha_federated::rnd_end"); DBUG_ENTER("ha_federated::rnd_end");
DBUG_RETURN(index_end());
}
int ha_federated::index_end(void)
{
DBUG_ENTER("ha_federated::index_end");
if (stored_result) if (stored_result)
{ {
DBUG_PRINT("info", ("mysql_free_result address %lx", stored_result));
mysql_free_result(stored_result); mysql_free_result(stored_result);
stored_result= 0; stored_result= 0;
} }
retval= index_end();
DBUG_RETURN(retval);
}
int ha_federated::index_end(void)
{
DBUG_ENTER("ha_federated::index_end");
active_index= MAX_KEY; active_index= MAX_KEY;
DBUG_RETURN(0); DBUG_RETURN(0);
} }
...@@ -2303,8 +2294,6 @@ int ha_federated::index_end(void) ...@@ -2303,8 +2294,6 @@ int ha_federated::index_end(void)
int ha_federated::rnd_next(byte *buf) int ha_federated::rnd_next(byte *buf)
{ {
int retval;
MYSQL_ROW row;
DBUG_ENTER("ha_federated::rnd_next"); DBUG_ENTER("ha_federated::rnd_next");
if (stored_result == 0) if (stored_result == 0)
...@@ -2312,32 +2301,60 @@ int ha_federated::rnd_next(byte *buf) ...@@ -2312,32 +2301,60 @@ int ha_federated::rnd_next(byte *buf)
/* /*
Return value of rnd_init is not always checked (see records.cc), Return value of rnd_init is not always checked (see records.cc),
so we can get here _even_ if there is _no_ pre-fetched result-set! so we can get here _even_ if there is _no_ pre-fetched result-set!
TODO: fix it. TODO: fix it. We can delete this in 5.1 when rnd_init() is checked.
*/ */
DBUG_RETURN(1); DBUG_RETURN(1);
} }
DBUG_RETURN(read_next(buf, stored_result));
}
/*
ha_federated::read_next
reads from a result set and converts to mysql internal
format
SYNOPSIS
field_in_record_is_null()
buf byte pointer to record
result mysql result set
DESCRIPTION
This method is a wrapper method that reads one record from a result
set and converts it to the internal table format
RETURN VALUE
1 error
0 no error
*/
int ha_federated::read_next(byte *buf, MYSQL_RES *result)
{
int retval;
my_ulonglong num_rows;
MYSQL_ROW row;
DBUG_ENTER("ha_federated::read_next");
table->status= STATUS_NOT_FOUND; // For easier return
/* Fetch a row, insert it back in a row format. */ /* Fetch a row, insert it back in a row format. */
current_position= stored_result->data_cursor; if (!(row= mysql_fetch_row(result)))
DBUG_PRINT("info", ("current position %d", current_position));
if (!(row= mysql_fetch_row(stored_result)))
DBUG_RETURN(HA_ERR_END_OF_FILE); DBUG_RETURN(HA_ERR_END_OF_FILE);
retval= convert_row_to_internal_format(buf, row); if (!(retval= convert_row_to_internal_format(buf, row, result)))
table->status= 0;
DBUG_RETURN(retval); DBUG_RETURN(retval);
} }
/* /*
'position()' is called after each call to rnd_next() if the data needs to be store reference to current row so that we can later find it for
ordered. You can do something like the following to store the position: a re-read, update or delete.
my_store_ptr(ref, ref_length, current_position);
The server uses ref to store data. ref_length in the above case is the size In case of federated, a reference is either a primary key or
needed to store current_position. ref is just a byte array that the server the whole record.
will maintain. If you are using offsets to mark rows, then current_position
should be the offset. If it is a primary key like in BDB, then it needs to
be a primary key.
Called from filesort.cc, sql_select.cc, sql_delete.cc and sql_update.cc. Called from filesort.cc, sql_select.cc, sql_delete.cc and sql_update.cc.
*/ */
...@@ -2345,32 +2362,44 @@ int ha_federated::rnd_next(byte *buf) ...@@ -2345,32 +2362,44 @@ int ha_federated::rnd_next(byte *buf)
void ha_federated::position(const byte *record) void ha_federated::position(const byte *record)
{ {
DBUG_ENTER("ha_federated::position"); DBUG_ENTER("ha_federated::position");
/* my_store_ptr Add seek storage */ if (table->s->primary_key != MAX_KEY)
*(MYSQL_ROW_OFFSET *) ref= current_position; // ref is always aligned key_copy(ref, (byte *)record, table->key_info + table->s->primary_key,
ref_length);
else
memcpy(ref, record, ref_length);
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
/* /*
This is like rnd_next, but you are given a position to use to determine the This is like rnd_next, but you are given a position to use to determine the
row. The position will be of the type that you stored in ref. You can use row. The position will be of the type that you stored in ref.
ha_get_ptr(pos,ref_length) to retrieve whatever key or position you saved
when position() was called.
This method is required for an ORDER BY. This method is required for an ORDER BY
Called from filesort.cc records.cc sql_insert.cc sql_select.cc sql_update.cc. Called from filesort.cc records.cc sql_insert.cc sql_select.cc sql_update.cc.
*/ */
int ha_federated::rnd_pos(byte *buf, byte *pos) int ha_federated::rnd_pos(byte *buf, byte *pos)
{ {
int result;
DBUG_ENTER("ha_federated::rnd_pos"); DBUG_ENTER("ha_federated::rnd_pos");
statistic_increment(table->in_use->status_var.ha_read_rnd_count, statistic_increment(table->in_use->status_var.ha_read_rnd_count,
&LOCK_status); &LOCK_status);
memcpy_fixed(&current_position, pos, sizeof(MYSQL_ROW_OFFSET)); if (table->s->primary_key != MAX_KEY)
stored_result->current_row= 0; {
stored_result->data_cursor= current_position; /* We have a primary key, so use index_read_idx to find row */
DBUG_RETURN(rnd_next(buf)); result= index_read_idx(buf, table->s->primary_key, pos,
ref_length, HA_READ_KEY_EXACT);
}
else
{
/* otherwise, get the old record ref as obtained in ::position */
memcpy(buf, pos, ref_length);
result= 0;
}
table->status= result ? STATUS_NOT_FOUND : 0;
DBUG_RETURN(result);
} }
...@@ -2476,18 +2505,22 @@ void ha_federated::info(uint flag) ...@@ -2476,18 +2505,22 @@ void ha_federated::info(uint flag)
*/ */
if (row[4] != NULL) if (row[4] != NULL)
records= (ha_rows) my_strtoll10(row[4], (char**) 0, &error); records= (ha_rows) my_strtoll10(row[4], (char**) 0, &error);
if (row[5] != NULL)
mean_rec_length= (ha_rows) my_strtoll10(row[5], (char**) 0, &error); mean_rec_length= table->s->reclength;
data_file_length= records * mean_rec_length;
if (row[12] != NULL) if (row[12] != NULL)
update_time= (ha_rows) my_strtoll10(row[12], (char**) 0, &error); update_time= (ha_rows) my_strtoll10(row[12], (char**) 0, &error);
if (row[13] != NULL) if (row[13] != NULL)
check_time= (ha_rows) my_strtoll10(row[13], (char**) 0, &error); check_time= (ha_rows) my_strtoll10(row[13], (char**) 0, &error);
} }
if (flag & HA_STATUS_CONST)
{ /*
size of IO operations (This is based on a good guess, no high science
involved)
*/
block_size= 4096; block_size= 4096;
} }
}
if (result) if (result)
mysql_free_result(result); mysql_free_result(result);
...@@ -2497,6 +2530,7 @@ void ha_federated::info(uint flag) ...@@ -2497,6 +2530,7 @@ void ha_federated::info(uint flag)
error: error:
if (result) if (result)
mysql_free_result(result); mysql_free_result(result);
my_sprintf(error_buffer, (error_buffer, ": %d : %s", my_sprintf(error_buffer, (error_buffer, ": %d : %s",
mysql_errno(mysql), mysql_error(mysql))); mysql_errno(mysql), mysql_error(mysql)));
my_error(error_code, MYF(0), error_buffer); my_error(error_code, MYF(0), error_buffer);
...@@ -2577,6 +2611,7 @@ THR_LOCK_DATA **ha_federated::store_lock(THD *thd, ...@@ -2577,6 +2611,7 @@ THR_LOCK_DATA **ha_federated::store_lock(THD *thd,
THR_LOCK_DATA **to, THR_LOCK_DATA **to,
enum thr_lock_type lock_type) enum thr_lock_type lock_type)
{ {
DBUG_ENTER("ha_federated::store_lock");
if (lock_type != TL_IGNORE && lock.type == TL_UNLOCK) if (lock_type != TL_IGNORE && lock.type == TL_UNLOCK)
{ {
/* /*
...@@ -2606,7 +2641,7 @@ THR_LOCK_DATA **ha_federated::store_lock(THD *thd, ...@@ -2606,7 +2641,7 @@ THR_LOCK_DATA **ha_federated::store_lock(THD *thd,
*to++= &lock; *to++= &lock;
return to; DBUG_RETURN(to);
} }
/* /*
......
...@@ -78,7 +78,7 @@ ...@@ -78,7 +78,7 @@
#define FEDERATED_VALUES_LEN sizeof(FEDERATED_VALUES) #define FEDERATED_VALUES_LEN sizeof(FEDERATED_VALUES)
#define FEDERATED_UPDATE "UPDATE " #define FEDERATED_UPDATE "UPDATE "
#define FEDERATED_UPDATE_LEN sizeof(FEDERATED_UPDATE) #define FEDERATED_UPDATE_LEN sizeof(FEDERATED_UPDATE)
#define FEDERATED_SET "SET " #define FEDERATED_SET " SET "
#define FEDERATED_SET_LEN sizeof(FEDERATED_SET) #define FEDERATED_SET_LEN sizeof(FEDERATED_SET)
#define FEDERATED_AND " AND " #define FEDERATED_AND " AND "
#define FEDERATED_AND_LEN sizeof(FEDERATED_AND) #define FEDERATED_AND_LEN sizeof(FEDERATED_AND)
...@@ -130,6 +130,7 @@ typedef struct st_federated_share { ...@@ -130,6 +130,7 @@ typedef struct st_federated_share {
remote host info, parse_url supplies remote host info, parse_url supplies
*/ */
char *scheme; char *scheme;
char *connect_string;
char *hostname; char *hostname;
char *username; char *username;
char *password; char *password;
...@@ -139,7 +140,7 @@ typedef struct st_federated_share { ...@@ -139,7 +140,7 @@ typedef struct st_federated_share {
char *socket; char *socket;
char *sport; char *sport;
ushort port; ushort port;
uint table_name_length, use_count; uint table_name_length, connect_string_length, use_count;
pthread_mutex_t mutex; pthread_mutex_t mutex;
THR_LOCK lock; THR_LOCK lock;
} FEDERATED_SHARE; } FEDERATED_SHARE;
...@@ -153,7 +154,6 @@ class ha_federated: public handler ...@@ -153,7 +154,6 @@ class ha_federated: public handler
FEDERATED_SHARE *share; /* Shared lock info */ FEDERATED_SHARE *share; /* Shared lock info */
MYSQL *mysql; /* MySQL connection */ MYSQL *mysql; /* MySQL connection */
MYSQL_RES *stored_result; MYSQL_RES *stored_result;
uint ref_length;
uint fetch_num; // stores the fetch num uint fetch_num; // stores the fetch num
MYSQL_ROW_OFFSET current_position; // Current position used by ::position() MYSQL_ROW_OFFSET current_position; // Current position used by ::position()
int remote_error_number; int remote_error_number;
...@@ -164,7 +164,8 @@ class ha_federated: public handler ...@@ -164,7 +164,8 @@ class ha_federated: public handler
return 0 on success return 0 on success
return errorcode otherwise return errorcode otherwise
*/ */
uint convert_row_to_internal_format(byte *buf, MYSQL_ROW row); uint convert_row_to_internal_format(byte *buf, MYSQL_ROW row,
MYSQL_RES *row);
bool create_where_from_key(String *to, KEY *key_info, bool create_where_from_key(String *to, KEY *key_info,
const key_range *start_key, const key_range *start_key,
const key_range *end_key, const key_range *end_key,
...@@ -298,6 +299,13 @@ class ha_federated: public handler ...@@ -298,6 +299,13 @@ class ha_federated: public handler
THR_LOCK_DATA **store_lock(THD *thd, THR_LOCK_DATA **to, THR_LOCK_DATA **store_lock(THD *thd, THR_LOCK_DATA **to,
enum thr_lock_type lock_type); //required enum thr_lock_type lock_type); //required
virtual bool get_error_message(int error, String *buf); virtual bool get_error_message(int error, String *buf);
int read_next(byte *buf, MYSQL_RES *result);
int index_read_idx_with_result_set(byte *buf, uint index,
const byte *key,
uint key_len,
ha_rkey_function find_flag,
MYSQL_RES **result);
}; };
bool federated_db_init(void); bool federated_db_init(void);
......
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