2005-05-06 10:39:30 +02:00
|
|
|
/* Copyright (C) 2000-2003 MySQL AB
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
2006-12-23 20:17:15 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2005-02-17 22:52:40 +01:00
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
This file defines the NDB Cluster handler: the interface between MySQL and
|
|
|
|
NDB Cluster
|
|
|
|
*/
|
|
|
|
|
2005-05-26 12:09:14 +02:00
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
2005-06-03 22:46:03 +02:00
|
|
|
#pragma implementation // gcc: Class implementation
|
2004-04-15 09:14:14 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "mysql_priv.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
|
|
#include <my_dir.h>
|
|
|
|
#include "ha_ndbcluster.h"
|
|
|
|
#include <ndbapi/NdbApi.hpp>
|
2007-04-23 11:25:33 +02:00
|
|
|
#include "ha_ndbcluster_cond.h"
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-12-20 12:36:14 +01:00
|
|
|
// options from from mysqld.cc
|
|
|
|
extern my_bool opt_ndb_optimized_node_selection;
|
|
|
|
extern const char *opt_ndbcluster_connectstring;
|
2006-08-15 14:22:47 +02:00
|
|
|
extern ulong opt_ndb_cache_check_time;
|
2004-12-20 12:36:14 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Default value for parallelism
|
2004-12-06 09:05:30 +01:00
|
|
|
static const int parallelism= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
// Default value for max number of transactions
|
|
|
|
// createable against NDB from this handler
|
2005-06-09 15:39:14 +02:00
|
|
|
static const int max_transactions= 2;
|
2004-07-23 15:46:56 +02:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
static const char *ha_ndb_ext=".ndb";
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
static int ndbcluster_close_connection(THD *thd);
|
|
|
|
static int ndbcluster_commit(THD *thd, bool all);
|
|
|
|
static int ndbcluster_rollback(THD *thd, bool all);
|
|
|
|
|
2005-09-19 21:06:23 +02:00
|
|
|
handlerton ndbcluster_hton = {
|
2005-03-13 21:58:09 +01:00
|
|
|
"ndbcluster",
|
2005-10-03 04:44:28 +02:00
|
|
|
SHOW_OPTION_YES,
|
|
|
|
"Clustered, fault-tolerant, memory-based tables",
|
|
|
|
DB_TYPE_NDBCLUSTER,
|
|
|
|
ndbcluster_init,
|
2005-02-17 22:52:40 +01:00
|
|
|
0, /* slot */
|
|
|
|
0, /* savepoint size */
|
|
|
|
ndbcluster_close_connection,
|
|
|
|
NULL, /* savepoint_set */
|
|
|
|
NULL, /* savepoint_rollback */
|
|
|
|
NULL, /* savepoint_release */
|
|
|
|
ndbcluster_commit,
|
|
|
|
ndbcluster_rollback,
|
|
|
|
NULL, /* prepare */
|
2005-02-18 12:53:43 +01:00
|
|
|
NULL, /* recover */
|
|
|
|
NULL, /* commit_by_xid */
|
A fix and a test case for Bug#10760 and complementary cleanups.
The idea of the patch
is that every cursor gets its own lock id for table level locking.
Thus cursors are protected from updates performed within the same
connection. Additionally a list of transient (must be closed at
commit) cursors is maintained and all transient cursors are closed
when necessary. Lastly, this patch adds support for deadlock
timeouts to TLL locking when using cursors.
+ post-review fixes.
include/thr_lock.h:
- add a notion of lock owner to table level locking. When using
cursors, lock owner can not be identified by a thread id any more,
as we must protect cursors from updates issued within the same
connection (thread). So, each cursor has its own lock identifier to
use with table level locking.
- extend return values of thr_lock and thr_multi_lock with
THR_LOCK_TIMEOUT and THR_LOCK_DEADLOCK, since these conditions
are now possible (see comments to thr_lock.c)
mysys/thr_lock.c:
Better support for cursors:
- use THR_LOCK_OWNER * as lock identifier, not pthread_t.
- check and return an error for a trivial deadlock case, when an
update statement is issued to a table locked by a cursor which has
been previously opened in the same connection.
- add support for locking timeouts: with use of cursors, trivial
deadlocks can occur. For now the only remedy is the lock wait timeout,
which is initialized from a new global variable 'table_lock_wait_timeout'
Example of a deadlock (assuming the storage engine does not downgrade
locks):
con1: open cursor for select * from t1;
con2: open cursor for select * from t2;
con1: update t2 set id=id*2; -- blocked
con2: update t1 set id=id*2; -- deadlock
Lock timeouts are active only if a connection is using cursors.
- the check in the wait_for_lock loop has been changed from
data->cond != cond to data->cond != 0. data->cond is zeroed
in every place it's changed.
- added comments
sql/examples/ha_archive.cc:
- extend the handlerton with the info about cursor behaviour at commit.
sql/examples/ha_archive.h:
- ctor moved to .cc to make use of archive handlerton
sql/examples/ha_example.cc:
- add handlerton instance, init handler::ht with it
sql/examples/ha_example.h:
- ctor moved to .cc to make use of ha_example handlerton
sql/examples/ha_tina.cc:
- add handlerton instance, init handler::ht with it
sql/examples/ha_tina.h:
- ctor moved to .cc to make use of CSV handlerton
sql/ha_berkeley.cc:
- init handlerton::flags and handler::ht
sql/ha_berkeley.h:
- ctor moved to .cc to make use of BerkeleyDB handlerton
sql/ha_blackhole.cc:
- add handlerton instance, init handler::ht with it
sql/ha_blackhole.h:
- ctor moved to .cc to make use of blackhole handlerton
sql/ha_federated.cc:
- add handlerton instance, init handler::ht with it
sql/ha_federated.h:
- ctor moved to .cc to make use of federated handlerton
sql/ha_heap.cc:
- add handlerton instance, init handler::ht with it
sql/ha_heap.h:
- ctor moved to .cc to make use of ha_heap handlerton
sql/ha_innodb.cc:
- init handlerton::flags and handler::ht of innobase storage engine
sql/ha_innodb.h:
- ctor moved to .cc to make use of archive handlerton
sql/ha_myisam.cc:
- add handlerton instance, init handler::ht with it
sql/ha_myisam.h:
- ctor moved to .cc to make use of MyISAM handlerton
sql/ha_myisammrg.cc:
- init handler::ht in the ctor
sql/ha_myisammrg.h:
- ctor moved to .cc to make use of MyISAM MERGE handlerton
sql/ha_ndbcluster.cc:
- init handlerton::flags and handler::ht
sql/handler.cc:
- drop support for ISAM storage engine, which was removed from 5.0
- close all "transient" cursors at COMMIT/ROLLBACK. A "transient"
SQL level cursor is a cursor that uses tables that have a transaction-
specific state.
sql/handler.h:
- extend struct handlerton with flags, add handlerton *ht to every
handler instance.
sql/lock.cc:
- extend mysql_lock_tables to send error to the client if
thr_multi_lock returns a timeout or a deadlock error.
sql/mysqld.cc:
- add server option --table_lock_wait_timeout (in seconds)
sql/set_var.cc:
- add new global variable 'table_lock_wait_timeout' to specify
a wait timeout for table-level locks of MySQL (in seconds). The default
timeout is 50 seconds. The timeout is active only if the connection
has open cursors.
sql/sql_class.cc:
- implement Statement_map::close_transient_cursors
- safety suggests that we need an assert ensuring
llock_info->n_cursors is functioning properly, adjust destruction of
the Statement_map to allow such assert in THD::~THD
sql/sql_class.h:
- add support for Cursors registry to Statement map.
sql/sql_prepare.cc:
- maintain a list of cursors that must be closed at commit/rollback.
sql/sql_select.cc:
- extend class Cursor to support specific at-COMMIT/ROLLBACK behavior.
If a cursor uses tables of a storage engine that
invalidates all open tables at COMMIT/ROLLBACK, it must be closed
before COMMIT/ROLLBACK is executed.
sql/sql_select.h:
- add an own lock_id and commit/rollback status flag to class Cursor
tests/mysql_client_test.c:
A test case for Bug#10760 and complementary issues: test a simple
deadlock case too.
mysql-test/var:
New BitKeeper file ``mysql-test/var''
2005-07-19 20:21:12 +02:00
|
|
|
NULL, /* rollback_by_xid */
|
2005-07-20 18:02:36 +02:00
|
|
|
NULL, /* create_cursor_read_view */
|
|
|
|
NULL, /* set_cursor_read_view */
|
|
|
|
NULL, /* close_cursor_read_view */
|
2006-06-02 09:40:34 +02:00
|
|
|
HTON_CAN_RECREATE
|
2005-02-17 22:52:40 +01:00
|
|
|
};
|
|
|
|
|
2005-04-15 16:14:08 +02:00
|
|
|
#define NDB_AUTO_INCREMENT_RETRIES 10
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-11-17 12:45:23 +01:00
|
|
|
#define NDB_INVALID_SCHEMA_OBJECT 241
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
#define ERR_PRINT(err) \
|
2004-09-13 14:46:38 +02:00
|
|
|
DBUG_PRINT("error", ("%d message: %s", err.code, err.message))
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-02-16 14:18:32 +01:00
|
|
|
#define ERR_RETURN(err) \
|
|
|
|
{ \
|
2004-11-17 10:07:52 +01:00
|
|
|
const NdbError& tmp= err; \
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_PRINT(tmp); \
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(ndb_to_mysql_error(&tmp)); \
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Typedefs for long names
|
|
|
|
typedef NdbDictionary::Column NDBCOL;
|
2004-09-07 21:53:59 +02:00
|
|
|
typedef NdbDictionary::Table NDBTAB;
|
2004-04-15 09:14:14 +02:00
|
|
|
typedef NdbDictionary::Index NDBINDEX;
|
|
|
|
typedef NdbDictionary::Dictionary NDBDICT;
|
|
|
|
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
bool ndbcluster_inited= FALSE;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-05-11 13:59:22 +02:00
|
|
|
static Ndb* g_ndb= NULL;
|
2004-09-06 20:30:57 +02:00
|
|
|
static Ndb_cluster_connection* g_ndb_cluster_connection= NULL;
|
2004-05-11 13:59:22 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Handler synchronization
|
|
|
|
pthread_mutex_t ndbcluster_mutex;
|
|
|
|
|
|
|
|
// Table lock handling
|
|
|
|
static HASH ndbcluster_open_tables;
|
|
|
|
|
|
|
|
static byte *ndbcluster_get_key(NDB_SHARE *share,uint *length,
|
|
|
|
my_bool not_used __attribute__((unused)));
|
|
|
|
static NDB_SHARE *get_share(const char *table_name);
|
|
|
|
static void free_share(NDB_SHARE *share);
|
|
|
|
|
|
|
|
static int packfrm(const void *data, uint len, const void **pack_data, uint *pack_len);
|
|
|
|
static int unpackfrm(const void **data, uint *len,
|
2005-02-16 14:18:32 +01:00
|
|
|
const void* pack_data);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2006-10-23 15:46:35 +02:00
|
|
|
static int ndb_get_table_statistics(ha_ndbcluster*, bool, Ndb*, const char *,
|
2005-02-16 14:18:32 +01:00
|
|
|
struct Ndb_statistics *);
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
// Util thread variables
|
|
|
|
static pthread_t ndb_util_thread;
|
|
|
|
pthread_mutex_t LOCK_ndb_util_thread;
|
|
|
|
pthread_cond_t COND_ndb_util_thread;
|
2005-10-08 16:39:55 +02:00
|
|
|
pthread_handler_t ndb_util_thread_func(void *arg);
|
2005-02-11 22:33:52 +01:00
|
|
|
ulong ndb_cache_check_time;
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2004-10-20 19:22:58 +02:00
|
|
|
/*
|
|
|
|
Dummy buffer to read zero pack_length fields
|
|
|
|
which are mapped to 1 char
|
|
|
|
*/
|
2004-12-10 16:55:04 +01:00
|
|
|
static uint32 dummy_buf;
|
2004-10-20 19:22:58 +02:00
|
|
|
|
2005-01-13 18:24:19 +01:00
|
|
|
/*
|
|
|
|
Stats that can be retrieved from ndb
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct Ndb_statistics {
|
|
|
|
Uint64 row_count;
|
|
|
|
Uint64 commit_count;
|
|
|
|
Uint64 row_size;
|
|
|
|
Uint64 fragment_memory;
|
|
|
|
};
|
|
|
|
|
2005-01-14 12:32:33 +01:00
|
|
|
/* Status variables shown with 'show status like 'Ndb%' */
|
|
|
|
|
|
|
|
static long ndb_cluster_node_id= 0;
|
|
|
|
static const char * ndb_connected_host= 0;
|
|
|
|
static long ndb_connected_port= 0;
|
|
|
|
static long ndb_number_of_replicas= 0;
|
2006-08-11 09:10:08 +02:00
|
|
|
static long ndb_number_of_data_nodes= 0;
|
2005-01-14 12:32:33 +01:00
|
|
|
|
|
|
|
static int update_status_variables(Ndb_cluster_connection *c)
|
|
|
|
{
|
|
|
|
ndb_cluster_node_id= c->node_id();
|
|
|
|
ndb_connected_port= c->get_connected_port();
|
|
|
|
ndb_connected_host= c->get_connected_host();
|
|
|
|
ndb_number_of_replicas= 0;
|
2006-08-11 09:10:08 +02:00
|
|
|
ndb_number_of_data_nodes= c->no_db_nodes();
|
2005-01-14 12:32:33 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct show_var_st ndb_status_variables[]= {
|
|
|
|
{"cluster_node_id", (char*) &ndb_cluster_node_id, SHOW_LONG},
|
2006-06-26 15:31:10 +02:00
|
|
|
{"config_from_host", (char*) &ndb_connected_host, SHOW_CHAR_PTR},
|
|
|
|
{"config_from_port", (char*) &ndb_connected_port, SHOW_LONG},
|
2005-01-14 12:32:33 +01:00
|
|
|
// {"number_of_replicas", (char*) &ndb_number_of_replicas, SHOW_LONG},
|
2006-08-11 09:10:08 +02:00
|
|
|
{"number_of_data_nodes",(char*) &ndb_number_of_data_nodes, SHOW_LONG},
|
2005-01-14 12:32:33 +01:00
|
|
|
{NullS, NullS, SHOW_LONG}
|
|
|
|
};
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Error handling functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct err_code_mapping
|
|
|
|
{
|
|
|
|
int ndb_err;
|
|
|
|
int my_err;
|
2004-11-17 09:15:53 +01:00
|
|
|
int show_warning;
|
2004-04-15 09:14:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const err_code_mapping err_map[]=
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
{ 626, HA_ERR_KEY_NOT_FOUND, 0 },
|
2007-07-25 03:32:28 +02:00
|
|
|
{ 630, HA_ERR_FOUND_DUPP_KEY, 1 },
|
|
|
|
{ 893, HA_ERR_FOUND_DUPP_KEY, 1 },
|
2004-11-17 09:15:53 +01:00
|
|
|
{ 721, HA_ERR_TABLE_EXIST, 1 },
|
|
|
|
{ 4244, HA_ERR_TABLE_EXIST, 1 },
|
|
|
|
|
2005-02-17 14:22:44 +01:00
|
|
|
{ 709, HA_ERR_NO_SUCH_TABLE, 0 },
|
2004-11-17 09:15:53 +01:00
|
|
|
|
|
|
|
{ 266, HA_ERR_LOCK_WAIT_TIMEOUT, 1 },
|
|
|
|
{ 274, HA_ERR_LOCK_WAIT_TIMEOUT, 1 },
|
|
|
|
{ 296, HA_ERR_LOCK_WAIT_TIMEOUT, 1 },
|
|
|
|
{ 297, HA_ERR_LOCK_WAIT_TIMEOUT, 1 },
|
|
|
|
{ 237, HA_ERR_LOCK_WAIT_TIMEOUT, 1 },
|
|
|
|
|
|
|
|
{ 623, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
{ 624, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
{ 625, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
{ 826, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
{ 827, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
{ 832, HA_ERR_RECORD_FILE_FULL, 1 },
|
|
|
|
|
2005-04-07 20:17:37 +02:00
|
|
|
{ 284, HA_ERR_TABLE_DEF_CHANGED, 0 },
|
|
|
|
|
2007-04-11 15:21:11 +02:00
|
|
|
{4000, HA_ERR_OUT_OF_MEM, 1 },
|
2007-04-03 12:44:29 +02:00
|
|
|
{4009, HA_ERR_NO_CONNECTION, 1 },
|
|
|
|
|
2004-11-17 09:15:53 +01:00
|
|
|
{ 0, 1, 0 },
|
|
|
|
|
|
|
|
{ -1, -1, 1 }
|
2004-04-15 09:14:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int ndb_to_mysql_error(const NdbError *err)
|
|
|
|
{
|
|
|
|
uint i;
|
2004-11-17 09:15:53 +01:00
|
|
|
for (i=0; err_map[i].ndb_err != err->code && err_map[i].my_err != -1; i++);
|
|
|
|
if (err_map[i].show_warning)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
// Push the NDB error message as warning
|
|
|
|
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_ERROR,
|
2005-02-16 14:18:32 +01:00
|
|
|
ER_GET_ERRMSG, ER(ER_GET_ERRMSG),
|
|
|
|
err->code, err->message, "NDB");
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-11-17 09:15:53 +01:00
|
|
|
if (err_map[i].my_err == -1)
|
|
|
|
return err->code;
|
2004-04-15 09:14:14 +02:00
|
|
|
return err_map[i].my_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-17 16:58:08 +02:00
|
|
|
|
|
|
|
inline
|
2006-08-15 14:31:21 +02:00
|
|
|
int execute_no_commit(ha_ndbcluster *h, NdbTransaction *trans,
|
|
|
|
bool force_release)
|
2004-09-17 16:58:08 +02:00
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
h->release_completed_operations(trans, force_release);
|
2005-01-05 17:59:24 +01:00
|
|
|
return trans->execute(NdbTransaction::NoCommit,
|
2005-02-16 14:18:32 +01:00
|
|
|
NdbTransaction::AbortOnError,
|
|
|
|
h->m_force_send);
|
2004-09-28 21:11:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
inline
|
2005-01-04 13:47:16 +01:00
|
|
|
int execute_commit(ha_ndbcluster *h, NdbTransaction *trans)
|
2004-09-28 21:11:50 +02:00
|
|
|
{
|
2005-01-05 17:59:24 +01:00
|
|
|
return trans->execute(NdbTransaction::Commit,
|
2005-02-16 14:18:32 +01:00
|
|
|
NdbTransaction::AbortOnError,
|
|
|
|
h->m_force_send);
|
2004-11-17 09:15:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline
|
2005-01-04 13:47:16 +01:00
|
|
|
int execute_commit(THD *thd, NdbTransaction *trans)
|
2004-11-17 09:15:53 +01:00
|
|
|
{
|
2005-01-05 17:59:24 +01:00
|
|
|
return trans->execute(NdbTransaction::Commit,
|
2005-02-16 14:18:32 +01:00
|
|
|
NdbTransaction::AbortOnError,
|
|
|
|
thd->variables.ndb_force_send);
|
2004-09-28 21:11:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
inline
|
2006-08-15 14:31:21 +02:00
|
|
|
int execute_no_commit_ie(ha_ndbcluster *h, NdbTransaction *trans,
|
|
|
|
bool force_release)
|
2004-09-28 21:11:50 +02:00
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
h->release_completed_operations(trans, force_release);
|
2005-01-05 17:59:24 +01:00
|
|
|
return trans->execute(NdbTransaction::NoCommit,
|
2005-02-16 14:18:32 +01:00
|
|
|
NdbTransaction::AO_IgnoreError,
|
|
|
|
h->m_force_send);
|
2004-09-17 16:58:08 +02:00
|
|
|
}
|
|
|
|
|
2004-12-17 21:13:22 +01:00
|
|
|
/*
|
|
|
|
Place holder for ha_ndbcluster thread specific data
|
|
|
|
*/
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
Thd_ndb::Thd_ndb()
|
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
ndb= new Ndb(g_ndb_cluster_connection, "");
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
lock_count= 0;
|
|
|
|
count= 0;
|
2005-02-17 22:52:40 +01:00
|
|
|
all= NULL;
|
|
|
|
stmt= NULL;
|
2004-09-14 18:17:01 +02:00
|
|
|
error= 0;
|
2006-08-15 14:31:21 +02:00
|
|
|
query_state&= NDB_QUERY_NORMAL;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Thd_ndb::~Thd_ndb()
|
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
if (ndb)
|
2005-11-17 12:45:23 +01:00
|
|
|
{
|
|
|
|
#ifndef DBUG_OFF
|
2006-01-05 23:47:49 +01:00
|
|
|
Ndb::Free_list_usage tmp;
|
|
|
|
tmp.m_name= 0;
|
2005-11-17 12:45:23 +01:00
|
|
|
while (ndb->get_free_list_usage(&tmp))
|
|
|
|
{
|
|
|
|
uint leaked= (uint) tmp.m_created - tmp.m_free;
|
|
|
|
if (leaked)
|
|
|
|
fprintf(stderr, "NDB: Found %u %s%s that %s not been released\n",
|
|
|
|
leaked, tmp.m_name,
|
|
|
|
(leaked == 1)?"":"'s",
|
|
|
|
(leaked == 1)?"has":"have");
|
|
|
|
}
|
|
|
|
#endif
|
2004-09-14 14:47:34 +02:00
|
|
|
delete ndb;
|
2006-01-05 23:47:49 +01:00
|
|
|
ndb= NULL;
|
2005-11-17 12:45:23 +01:00
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
changed_tables.empty();
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
inline
|
|
|
|
Thd_ndb *
|
|
|
|
get_thd_ndb(THD *thd) { return (Thd_ndb *) thd->ha_data[ndbcluster_hton.slot]; }
|
|
|
|
|
|
|
|
inline
|
|
|
|
void
|
|
|
|
set_thd_ndb(THD *thd, Thd_ndb *thd_ndb) { thd->ha_data[ndbcluster_hton.slot]= thd_ndb; }
|
|
|
|
|
2004-12-30 19:56:09 +01:00
|
|
|
inline
|
|
|
|
Ndb *ha_ndbcluster::get_ndb()
|
|
|
|
{
|
2005-02-18 12:53:43 +01:00
|
|
|
return get_thd_ndb(current_thd)->ndb;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* manage uncommitted insert/deletes during transactio to get records correct
|
|
|
|
*/
|
|
|
|
|
2005-03-22 17:42:08 +01:00
|
|
|
struct Ndb_local_table_statistics {
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
int no_uncommitted_rows_count;
|
2004-09-14 14:47:34 +02:00
|
|
|
ulong last_count;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
ha_rows records;
|
|
|
|
};
|
|
|
|
|
2004-10-04 01:20:05 +02:00
|
|
|
void ha_ndbcluster::set_rec_per_key()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("ha_ndbcluster::get_status_const");
|
2005-01-06 12:00:13 +01:00
|
|
|
for (uint i=0 ; i < table->s->keys ; i++)
|
2004-10-04 01:20:05 +02:00
|
|
|
{
|
|
|
|
table->key_info[i].rec_per_key[table->key_info[i].key_parts-1]= 1;
|
|
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2006-08-10 16:55:20 +02:00
|
|
|
int ha_ndbcluster::records_update()
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
2006-08-10 16:55:20 +02:00
|
|
|
return 0;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::records_update");
|
2006-08-10 16:55:20 +02:00
|
|
|
int result= 0;
|
|
|
|
|
2006-12-14 23:51:37 +01:00
|
|
|
struct Ndb_local_table_statistics *local_info=
|
2005-03-22 17:42:08 +01:00
|
|
|
(struct Ndb_local_table_statistics *)m_table_info;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_PRINT("info", ("id=%d, no_uncommitted_rows_count=%d",
|
2005-02-16 14:18:32 +01:00
|
|
|
((const NDBTAB *)m_table)->getTableId(),
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->no_uncommitted_rows_count));
|
2004-09-20 17:52:19 +02:00
|
|
|
// if (info->records == ~(ha_rows)0)
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
2005-01-13 18:24:19 +01:00
|
|
|
struct Ndb_statistics stat;
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
return my_errno= HA_ERR_OUT_OF_MEM;
|
|
|
|
}
|
2006-10-23 16:18:57 +02:00
|
|
|
result= ndb_get_table_statistics(this, true, ndb, m_tabname, &stat);
|
|
|
|
if (result == 0)
|
|
|
|
{
|
2005-01-13 18:24:19 +01:00
|
|
|
mean_rec_length= stat.row_size;
|
|
|
|
data_file_length= stat.fragment_memory;
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->records= stat.row_count;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
}
|
2004-09-14 18:17:01 +02:00
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
2005-02-17 22:52:40 +01:00
|
|
|
if (get_thd_ndb(thd)->error)
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->no_uncommitted_rows_count= 0;
|
2004-09-14 18:17:01 +02:00
|
|
|
}
|
2006-10-23 15:46:35 +02:00
|
|
|
if(result==0)
|
2006-12-14 23:51:37 +01:00
|
|
|
records= local_info->records+ local_info->no_uncommitted_rows_count;
|
2006-08-10 16:55:20 +02:00
|
|
|
DBUG_RETURN(result);
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
|
2004-09-14 17:51:57 +02:00
|
|
|
void ha_ndbcluster::no_uncommitted_rows_execute_failure()
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
|
|
|
return;
|
2004-09-14 17:51:57 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::no_uncommitted_rows_execute_failure");
|
2005-02-18 12:53:43 +01:00
|
|
|
get_thd_ndb(current_thd)->error= 1;
|
2004-09-14 17:51:57 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
void ha_ndbcluster::no_uncommitted_rows_init(THD *thd)
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
|
|
|
return;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::no_uncommitted_rows_init");
|
2006-12-14 23:51:37 +01:00
|
|
|
struct Ndb_local_table_statistics *local_info=
|
2005-03-22 17:42:08 +01:00
|
|
|
(struct Ndb_local_table_statistics *)m_table_info;
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
2006-12-14 23:51:37 +01:00
|
|
|
if (local_info->last_count != thd_ndb->count)
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
{
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->last_count= thd_ndb->count;
|
|
|
|
local_info->no_uncommitted_rows_count= 0;
|
|
|
|
local_info->records= ~(ha_rows)0;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_PRINT("info", ("id=%d, no_uncommitted_rows_count=%d",
|
2005-02-16 14:18:32 +01:00
|
|
|
((const NDBTAB *)m_table)->getTableId(),
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->no_uncommitted_rows_count));
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ha_ndbcluster::no_uncommitted_rows_update(int c)
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
|
|
|
return;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::no_uncommitted_rows_update");
|
2006-12-14 23:51:37 +01:00
|
|
|
struct Ndb_local_table_statistics *local_info=
|
2005-03-22 17:42:08 +01:00
|
|
|
(struct Ndb_local_table_statistics *)m_table_info;
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->no_uncommitted_rows_count+= c;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_PRINT("info", ("id=%d, no_uncommitted_rows_count=%d",
|
2005-02-16 14:18:32 +01:00
|
|
|
((const NDBTAB *)m_table)->getTableId(),
|
2006-12-14 23:51:37 +01:00
|
|
|
local_info->no_uncommitted_rows_count));
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ha_ndbcluster::no_uncommitted_rows_reset(THD *thd)
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
|
|
|
return;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::no_uncommitted_rows_reset");
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
|
|
|
thd_ndb->count++;
|
|
|
|
thd_ndb->error= 0;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Take care of the error that occured in NDB
|
2005-02-17 22:52:40 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
RETURN
|
2005-02-16 14:18:32 +01:00
|
|
|
0 No error
|
2004-04-15 09:14:14 +02:00
|
|
|
# The mapped error code
|
|
|
|
*/
|
|
|
|
|
2005-04-27 18:17:41 +02:00
|
|
|
void ha_ndbcluster::invalidate_dictionary_cache(bool global)
|
2005-03-07 10:21:38 +01:00
|
|
|
{
|
|
|
|
NDBDICT *dict= get_ndb()->getDictionary();
|
2005-04-28 10:46:39 +02:00
|
|
|
DBUG_ENTER("invalidate_dictionary_cache");
|
2005-03-07 10:21:38 +01:00
|
|
|
DBUG_PRINT("info", ("invalidating %s", m_tabname));
|
2005-04-28 10:46:39 +02:00
|
|
|
|
2005-04-27 18:17:41 +02:00
|
|
|
if (global)
|
2005-04-28 09:39:29 +02:00
|
|
|
{
|
2005-04-28 10:46:39 +02:00
|
|
|
const NDBTAB *tab= dict->getTable(m_tabname);
|
|
|
|
if (!tab)
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
if (tab->getObjectStatus() == NdbDictionary::Object::Invalid)
|
2005-04-28 09:39:29 +02:00
|
|
|
{
|
|
|
|
// Global cache has already been invalidated
|
|
|
|
dict->removeCachedTable(m_tabname);
|
|
|
|
global= FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dict->invalidateTable(m_tabname);
|
|
|
|
}
|
2005-04-27 18:17:41 +02:00
|
|
|
else
|
|
|
|
dict->removeCachedTable(m_tabname);
|
2005-03-09 23:38:11 +01:00
|
|
|
table->s->version=0L; /* Free when thread is ready */
|
2005-03-07 10:21:38 +01:00
|
|
|
/* Invalidate indexes */
|
2005-03-09 23:38:11 +01:00
|
|
|
for (uint i= 0; i < table->s->keys; i++)
|
2005-03-07 10:21:38 +01:00
|
|
|
{
|
|
|
|
NDBINDEX *index = (NDBINDEX *) m_index[i].index;
|
|
|
|
NDBINDEX *unique_index = (NDBINDEX *) m_index[i].unique_index;
|
|
|
|
NDB_INDEX_TYPE idx_type= m_index[i].type;
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
switch (idx_type) {
|
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
|
|
|
case ORDERED_INDEX:
|
2005-04-27 18:17:41 +02:00
|
|
|
if (global)
|
|
|
|
dict->invalidateIndex(index->getName(), m_tabname);
|
|
|
|
else
|
|
|
|
dict->removeCachedIndex(index->getName(), m_tabname);
|
2005-03-10 00:10:10 +01:00
|
|
|
break;
|
2005-07-04 02:42:33 +02:00
|
|
|
case UNIQUE_ORDERED_INDEX:
|
2005-04-27 18:17:41 +02:00
|
|
|
if (global)
|
|
|
|
dict->invalidateIndex(index->getName(), m_tabname);
|
|
|
|
else
|
|
|
|
dict->removeCachedIndex(index->getName(), m_tabname);
|
2005-07-04 02:42:33 +02:00
|
|
|
case UNIQUE_INDEX:
|
2005-04-27 18:17:41 +02:00
|
|
|
if (global)
|
|
|
|
dict->invalidateIndex(unique_index->getName(), m_tabname);
|
|
|
|
else
|
|
|
|
dict->removeCachedIndex(unique_index->getName(), m_tabname);
|
2005-03-07 10:21:38 +01:00
|
|
|
break;
|
2005-07-04 02:42:33 +02:00
|
|
|
case PRIMARY_KEY_INDEX:
|
|
|
|
case UNDEFINED_INDEX:
|
2005-03-07 10:21:38 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-04-28 10:46:39 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2005-03-07 10:21:38 +01:00
|
|
|
}
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
|
2005-01-04 13:47:16 +01:00
|
|
|
int ha_ndbcluster::ndb_err(NdbTransaction *trans)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-08-31 12:07:48 +02:00
|
|
|
int res;
|
2005-04-04 12:26:05 +02:00
|
|
|
NdbError err= trans->getNdbError();
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("ndb_err");
|
|
|
|
|
|
|
|
ERR_PRINT(err);
|
|
|
|
switch (err.classification) {
|
|
|
|
case NdbError::SchemaError:
|
2006-04-11 09:46:03 +02:00
|
|
|
{
|
2006-01-10 11:35:21 +01:00
|
|
|
/* Close other open handlers not used by any thread */
|
|
|
|
TABLE_LIST table_list;
|
|
|
|
bzero((char*) &table_list,sizeof(table_list));
|
|
|
|
table_list.db= m_dbname;
|
|
|
|
table_list.alias= table_list.table_name= m_tabname;
|
|
|
|
close_cached_tables(current_thd, 0, &table_list);
|
|
|
|
|
2006-04-11 09:46:03 +02:00
|
|
|
invalidate_dictionary_cache(TRUE);
|
|
|
|
|
2005-04-04 12:26:05 +02:00
|
|
|
if (err.code==284)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Check if the table is _really_ gone or if the table has
|
|
|
|
been alterend and thus changed table id
|
|
|
|
*/
|
|
|
|
NDBDICT *dict= get_ndb()->getDictionary();
|
|
|
|
DBUG_PRINT("info", ("Check if table %s is really gone", m_tabname));
|
|
|
|
if (!(dict->getTable(m_tabname)))
|
|
|
|
{
|
|
|
|
err= dict->getNdbError();
|
|
|
|
DBUG_PRINT("info", ("Table not found, error: %d", err.code));
|
|
|
|
if (err.code != 709)
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2005-04-07 20:17:37 +02:00
|
|
|
DBUG_PRINT("info", ("Table exists but must have changed"));
|
2005-04-04 12:26:05 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
2006-04-11 09:46:03 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2004-08-31 12:07:48 +02:00
|
|
|
res= ndb_to_mysql_error(&err);
|
|
|
|
DBUG_PRINT("info", ("transformed ndbcluster error %d to mysql error %d",
|
2005-02-16 14:18:32 +01:00
|
|
|
err.code, res));
|
2004-08-31 12:07:48 +02:00
|
|
|
if (res == HA_ERR_FOUND_DUPP_KEY)
|
2005-06-17 14:45:54 +02:00
|
|
|
{
|
2007-12-07 13:42:44 +01:00
|
|
|
char *error_data= err.details;
|
2007-12-06 17:15:21 +01:00
|
|
|
uint dupkey= MAX_KEY;
|
|
|
|
|
|
|
|
for (uint i= 0; i < MAX_KEY; i++)
|
|
|
|
{
|
|
|
|
if (m_index[i].type == UNIQUE_INDEX ||
|
|
|
|
m_index[i].type == UNIQUE_ORDERED_INDEX)
|
|
|
|
{
|
|
|
|
const NDBINDEX *unique_index=
|
|
|
|
(const NDBINDEX *) m_index[i].unique_index;
|
|
|
|
if (unique_index &&
|
|
|
|
unique_index->getIndexTable() &&
|
2007-12-07 14:03:54 +01:00
|
|
|
(char *) unique_index->getIndexTable()->getTableId() == error_data)
|
2007-12-06 17:15:21 +01:00
|
|
|
{
|
|
|
|
dupkey= i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-06-17 14:45:54 +02:00
|
|
|
if (m_rows_to_insert == 1)
|
2006-09-21 16:49:07 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
We can only distinguish between primary and non-primary
|
|
|
|
violations here, so we need to return MAX_KEY for non-primary
|
|
|
|
to signal that key is unknown
|
|
|
|
*/
|
2007-12-06 17:15:21 +01:00
|
|
|
m_dupkey= err.code == 630 ? table->s->primary_key : dupkey;
|
2006-09-21 16:49:07 +02:00
|
|
|
}
|
2005-06-17 14:45:54 +02:00
|
|
|
else
|
2005-06-27 15:46:41 +02:00
|
|
|
{
|
|
|
|
/* We are batching inserts, offending key is not available */
|
2005-06-17 14:45:54 +02:00
|
|
|
m_dupkey= (uint) -1;
|
2005-06-27 15:46:41 +02:00
|
|
|
}
|
2005-06-17 14:45:54 +02:00
|
|
|
}
|
2004-08-31 12:07:48 +02:00
|
|
|
DBUG_RETURN(res);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-11 13:59:22 +02:00
|
|
|
/*
|
2004-05-13 11:56:45 +02:00
|
|
|
Override the default get_error_message in order to add the
|
2004-05-11 13:59:22 +02:00
|
|
|
error message of NDB
|
|
|
|
*/
|
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
bool ha_ndbcluster::get_error_message(int error,
|
2005-02-16 14:18:32 +01:00
|
|
|
String *buf)
|
2004-05-11 13:59:22 +02:00
|
|
|
{
|
2004-05-13 11:56:45 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::get_error_message");
|
2004-05-24 12:35:39 +02:00
|
|
|
DBUG_PRINT("enter", ("error: %d", error));
|
2004-05-11 13:59:22 +02:00
|
|
|
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
2004-09-15 14:44:21 +02:00
|
|
|
if (!ndb)
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
2004-05-13 11:56:45 +02:00
|
|
|
|
2004-09-15 14:44:21 +02:00
|
|
|
const NdbError err= ndb->getNdbError(error);
|
2004-05-24 12:35:39 +02:00
|
|
|
bool temporary= err.status==NdbError::TemporaryError;
|
|
|
|
buf->set(err.message, strlen(err.message), &my_charset_bin);
|
|
|
|
DBUG_PRINT("exit", ("message: %s, temporary: %d", buf->ptr(), temporary));
|
|
|
|
DBUG_RETURN(temporary);
|
2004-05-11 13:59:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-06-20 07:53:53 +02:00
|
|
|
#ifndef DBUG_OFF
|
2004-07-22 12:38:09 +02:00
|
|
|
/*
|
|
|
|
Check if type is supported by NDB.
|
|
|
|
*/
|
|
|
|
|
2005-06-20 07:53:53 +02:00
|
|
|
static bool ndb_supported_type(enum_field_types type)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
|
|
|
switch (type) {
|
2004-07-23 13:21:39 +02:00
|
|
|
case MYSQL_TYPE_TINY:
|
|
|
|
case MYSQL_TYPE_SHORT:
|
|
|
|
case MYSQL_TYPE_LONG:
|
|
|
|
case MYSQL_TYPE_INT24:
|
|
|
|
case MYSQL_TYPE_LONGLONG:
|
|
|
|
case MYSQL_TYPE_FLOAT:
|
|
|
|
case MYSQL_TYPE_DOUBLE:
|
2005-02-16 21:19:42 +01:00
|
|
|
case MYSQL_TYPE_DECIMAL:
|
|
|
|
case MYSQL_TYPE_NEWDECIMAL:
|
2004-07-23 13:21:39 +02:00
|
|
|
case MYSQL_TYPE_TIMESTAMP:
|
|
|
|
case MYSQL_TYPE_DATETIME:
|
|
|
|
case MYSQL_TYPE_DATE:
|
|
|
|
case MYSQL_TYPE_NEWDATE:
|
|
|
|
case MYSQL_TYPE_TIME:
|
|
|
|
case MYSQL_TYPE_YEAR:
|
|
|
|
case MYSQL_TYPE_STRING:
|
|
|
|
case MYSQL_TYPE_VAR_STRING:
|
2005-01-07 11:55:20 +01:00
|
|
|
case MYSQL_TYPE_VARCHAR:
|
2004-07-23 13:21:39 +02:00
|
|
|
case MYSQL_TYPE_TINY_BLOB:
|
|
|
|
case MYSQL_TYPE_BLOB:
|
|
|
|
case MYSQL_TYPE_MEDIUM_BLOB:
|
|
|
|
case MYSQL_TYPE_LONG_BLOB:
|
|
|
|
case MYSQL_TYPE_ENUM:
|
|
|
|
case MYSQL_TYPE_SET:
|
2004-12-23 15:28:41 +01:00
|
|
|
case MYSQL_TYPE_BIT:
|
2005-10-25 08:20:39 +02:00
|
|
|
case MYSQL_TYPE_GEOMETRY:
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return TRUE;
|
2004-07-22 12:38:09 +02:00
|
|
|
case MYSQL_TYPE_NULL:
|
2004-07-23 13:21:39 +02:00
|
|
|
break;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return FALSE;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
2005-06-20 07:53:53 +02:00
|
|
|
#endif /* !DBUG_OFF */
|
2004-07-22 12:38:09 +02:00
|
|
|
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Instruct NDB to set the value of the hidden primary key
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool ha_ndbcluster::set_hidden_key(NdbOperation *ndb_op,
|
2005-02-16 14:18:32 +01:00
|
|
|
uint fieldnr, const byte *field_ptr)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("set_hidden_key");
|
|
|
|
DBUG_RETURN(ndb_op->equal(fieldnr, (char*)field_ptr,
|
2005-02-16 14:18:32 +01:00
|
|
|
NDB_HIDDEN_PRIMARY_KEY_LENGTH) != 0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Instruct NDB to set the value of one primary key attribute
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::set_ndb_key(NdbOperation *ndb_op, Field *field,
|
|
|
|
uint fieldnr, const byte *field_ptr)
|
|
|
|
{
|
|
|
|
uint32 pack_len= field->pack_length();
|
|
|
|
DBUG_ENTER("set_ndb_key");
|
|
|
|
DBUG_PRINT("enter", ("%d: %s, ndb_type: %u, len=%d",
|
|
|
|
fieldnr, field->field_name, field->type(),
|
|
|
|
pack_len));
|
|
|
|
DBUG_DUMP("key", (char*)field_ptr, pack_len);
|
|
|
|
|
2005-06-20 07:53:53 +02:00
|
|
|
DBUG_ASSERT(ndb_supported_type(field->type()));
|
|
|
|
DBUG_ASSERT(! (field->flags & BLOB_FLAG));
|
|
|
|
// Common implementation for most field types
|
|
|
|
DBUG_RETURN(ndb_op->equal(fieldnr, (char*) field_ptr, pack_len) != 0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Instruct NDB to set the value of one attribute
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::set_ndb_value(NdbOperation *ndb_op, Field *field,
|
2004-10-01 13:16:49 +02:00
|
|
|
uint fieldnr, bool *set_blob_value)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
const byte* field_ptr= field->ptr;
|
|
|
|
uint32 pack_len= field->pack_length();
|
|
|
|
DBUG_ENTER("set_ndb_value");
|
|
|
|
DBUG_PRINT("enter", ("%d: %s, type: %u, len=%d, is_null=%s",
|
|
|
|
fieldnr, field->field_name, field->type(),
|
|
|
|
pack_len, field->is_null()?"Y":"N"));
|
|
|
|
DBUG_DUMP("value", (char*) field_ptr, pack_len);
|
2004-07-22 12:38:09 +02:00
|
|
|
|
2005-06-20 07:53:53 +02:00
|
|
|
DBUG_ASSERT(ndb_supported_type(field->type()));
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-10-20 19:22:58 +02:00
|
|
|
// ndb currently does not support size 0
|
2004-12-10 16:55:04 +01:00
|
|
|
uint32 empty_field;
|
2004-10-20 19:22:58 +02:00
|
|
|
if (pack_len == 0)
|
|
|
|
{
|
2004-12-10 16:55:04 +01:00
|
|
|
pack_len= sizeof(empty_field);
|
|
|
|
field_ptr= (byte *)&empty_field;
|
|
|
|
if (field->is_null())
|
2005-02-16 14:18:32 +01:00
|
|
|
empty_field= 0;
|
2004-12-10 16:55:04 +01:00
|
|
|
else
|
2005-02-16 14:18:32 +01:00
|
|
|
empty_field= 1;
|
2004-10-20 19:22:58 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
if (! (field->flags & BLOB_FLAG))
|
|
|
|
{
|
2004-12-23 15:28:41 +01:00
|
|
|
if (field->type() != MYSQL_TYPE_BIT)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
if (field->is_null())
|
|
|
|
// Set value to NULL
|
|
|
|
DBUG_RETURN((ndb_op->setValue(fieldnr,
|
|
|
|
(char*)NULL, pack_len) != 0));
|
|
|
|
// Common implementation for most field types
|
|
|
|
DBUG_RETURN(ndb_op->setValue(fieldnr,
|
|
|
|
(char*)field_ptr, pack_len) != 0);
|
2004-12-23 15:28:41 +01:00
|
|
|
}
|
|
|
|
else // if (field->type() == MYSQL_TYPE_BIT)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
longlong bits= field->val_int();
|
2004-12-23 15:28:41 +01:00
|
|
|
|
2005-02-16 14:18:32 +01:00
|
|
|
// Round up bit field length to nearest word boundry
|
|
|
|
pack_len= ((pack_len + 3) >> 2) << 2;
|
2004-12-23 15:28:41 +01:00
|
|
|
DBUG_ASSERT(pack_len <= 8);
|
|
|
|
if (field->is_null())
|
|
|
|
// Set value to NULL
|
|
|
|
DBUG_RETURN((ndb_op->setValue(fieldnr, (char*)NULL, pack_len) != 0));
|
|
|
|
DBUG_PRINT("info", ("bit field"));
|
2005-01-03 14:29:27 +01:00
|
|
|
DBUG_DUMP("value", (char*)&bits, pack_len);
|
2005-01-20 15:50:56 +01:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
2007-03-26 14:48:52 +02:00
|
|
|
/* store lsw first */
|
2007-04-18 16:41:38 +02:00
|
|
|
bits = ((bits >> 32) & 0x00000000FFFFFFFFLL)
|
|
|
|
| ((bits << 32) & 0xFFFFFFFF00000000LL);
|
2005-01-20 15:50:56 +01:00
|
|
|
#endif
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(ndb_op->setValue(fieldnr, (char*)&bits, pack_len) != 0);
|
2004-12-23 15:28:41 +01:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
// Blob type
|
2004-07-23 15:46:56 +02:00
|
|
|
NdbBlob *ndb_blob= ndb_op->getBlobHandle(fieldnr);
|
2004-07-22 12:38:09 +02:00
|
|
|
if (ndb_blob != NULL)
|
|
|
|
{
|
|
|
|
if (field->is_null())
|
|
|
|
DBUG_RETURN(ndb_blob->setNull() != 0);
|
|
|
|
|
|
|
|
Field_blob *field_blob= (Field_blob*)field;
|
|
|
|
|
|
|
|
// Get length and pointer to data
|
|
|
|
uint32 blob_len= field_blob->get_length(field_ptr);
|
|
|
|
char* blob_ptr= NULL;
|
|
|
|
field_blob->get_ptr(&blob_ptr);
|
|
|
|
|
2004-09-02 12:42:03 +02:00
|
|
|
// Looks like NULL ptr signals length 0 blob
|
|
|
|
if (blob_ptr == NULL) {
|
|
|
|
DBUG_ASSERT(blob_len == 0);
|
2004-09-15 17:31:15 +02:00
|
|
|
blob_ptr= (char*)"";
|
2004-09-02 12:42:03 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
|
2007-03-28 19:46:42 +02:00
|
|
|
DBUG_PRINT("value", ("set blob ptr: %p len: %u",
|
2005-07-19 22:04:43 +02:00
|
|
|
blob_ptr, blob_len));
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_DUMP("value", (char*)blob_ptr, min(blob_len, 26));
|
|
|
|
|
2004-10-01 13:16:49 +02:00
|
|
|
if (set_blob_value)
|
2005-02-16 14:18:32 +01:00
|
|
|
*set_blob_value= TRUE;
|
2004-07-22 12:38:09 +02:00
|
|
|
// No callback needed to write value
|
|
|
|
DBUG_RETURN(ndb_blob->setValue(blob_ptr, blob_len) != 0);
|
|
|
|
}
|
|
|
|
DBUG_RETURN(1);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Callback to read all blob values.
|
|
|
|
- not done in unpack_record because unpack_record is valid
|
|
|
|
after execute(Commit) but reading blobs is not
|
|
|
|
- may only generate read operations; they have to be executed
|
|
|
|
somewhere before the data is available
|
|
|
|
- due to single buffer for all blobs, we let the last blob
|
|
|
|
process all blobs (last so that all are active)
|
|
|
|
- null bit is still set in unpack_record
|
|
|
|
- TODO allocate blob part aligned buffers
|
|
|
|
*/
|
|
|
|
|
2004-07-29 10:44:53 +02:00
|
|
|
NdbBlob::ActiveHook g_get_ndb_blobs_value;
|
2004-07-22 12:38:09 +02:00
|
|
|
|
2004-07-29 10:44:53 +02:00
|
|
|
int g_get_ndb_blobs_value(NdbBlob *ndb_blob, void *arg)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
2004-07-29 10:44:53 +02:00
|
|
|
DBUG_ENTER("g_get_ndb_blobs_value");
|
2004-07-22 12:38:09 +02:00
|
|
|
if (ndb_blob->blobsNextBlob() != NULL)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
ha_ndbcluster *ha= (ha_ndbcluster *)arg;
|
2006-06-21 09:36:50 +02:00
|
|
|
DBUG_RETURN(ha->get_ndb_blobs_value(ndb_blob, ha->m_blobs_offset));
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
|
2006-06-21 09:36:50 +02:00
|
|
|
int ha_ndbcluster::get_ndb_blobs_value(NdbBlob *last_ndb_blob,
|
|
|
|
my_ptrdiff_t ptrdiff)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("get_ndb_blobs_value");
|
|
|
|
|
|
|
|
// Field has no field number so cannot use TABLE blob_field
|
|
|
|
// Loop twice, first only counting total buffer size
|
|
|
|
for (int loop= 0; loop <= 1; loop++)
|
|
|
|
{
|
|
|
|
uint32 offset= 0;
|
2005-01-06 12:00:13 +01:00
|
|
|
for (uint i= 0; i < table->s->fields; i++)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
|
|
|
NdbValue value= m_value[i];
|
|
|
|
if (value.ptr != NULL && (field->flags & BLOB_FLAG))
|
|
|
|
{
|
|
|
|
Field_blob *field_blob= (Field_blob *)field;
|
|
|
|
NdbBlob *ndb_blob= value.blob;
|
|
|
|
Uint64 blob_len= 0;
|
|
|
|
if (ndb_blob->getLength(blob_len) != 0)
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
// Align to Uint64
|
|
|
|
uint32 blob_size= blob_len;
|
|
|
|
if (blob_size % 8 != 0)
|
|
|
|
blob_size+= 8 - blob_size % 8;
|
|
|
|
if (loop == 1)
|
|
|
|
{
|
2004-11-03 15:53:26 +01:00
|
|
|
char *buf= m_blobs_buffer + offset;
|
2004-07-22 12:38:09 +02:00
|
|
|
uint32 len= 0xffffffff; // Max uint32
|
2006-11-30 20:56:03 +01:00
|
|
|
DBUG_PRINT("value", ("read blob ptr: 0x%lx len: %u",
|
|
|
|
(long)buf, (uint)blob_len));
|
2004-07-22 12:38:09 +02:00
|
|
|
if (ndb_blob->readData(buf, len) != 0)
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
DBUG_ASSERT(len == blob_len);
|
2006-06-21 09:36:50 +02:00
|
|
|
// Ugly hack assumes only ptr needs to be changed
|
|
|
|
field_blob->ptr+= ptrdiff;
|
2004-07-22 12:38:09 +02:00
|
|
|
field_blob->set_ptr(len, buf);
|
2006-06-21 09:36:50 +02:00
|
|
|
field_blob->ptr-= ptrdiff;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
offset+= blob_size;
|
|
|
|
}
|
|
|
|
}
|
2004-11-03 15:53:26 +01:00
|
|
|
if (loop == 0 && offset > m_blobs_buffer_size)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
2004-11-03 15:53:26 +01:00
|
|
|
my_free(m_blobs_buffer, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
m_blobs_buffer_size= 0;
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_PRINT("value", ("allocate blobs buffer size %u", offset));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_blobs_buffer= my_malloc(offset, MYF(MY_WME));
|
|
|
|
if (m_blobs_buffer == NULL)
|
2007-04-11 15:21:11 +02:00
|
|
|
{
|
|
|
|
sql_print_error("ha_ndbcluster::get_ndb_blobs_value: "
|
|
|
|
"my_malloc(%u) failed", offset);
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_RETURN(-1);
|
2007-04-11 15:21:11 +02:00
|
|
|
}
|
2004-11-03 15:53:26 +01:00
|
|
|
m_blobs_buffer_size= offset;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_RETURN(0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Instruct NDB to fetch one field
|
2004-07-22 12:38:09 +02:00
|
|
|
- data is read directly into buffer provided by field
|
|
|
|
if field is NULL, data is read into memory provided by NDBAPI
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2004-07-22 12:38:09 +02:00
|
|
|
int ha_ndbcluster::get_ndb_value(NdbOperation *ndb_op, Field *field,
|
2004-09-20 12:40:53 +02:00
|
|
|
uint fieldnr, byte* buf)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("get_ndb_value");
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_PRINT("enter", ("fieldnr: %d flags: %o", fieldnr,
|
|
|
|
(int)(field != NULL ? field->flags : 0)));
|
|
|
|
|
|
|
|
if (field != NULL)
|
|
|
|
{
|
2005-06-20 07:53:53 +02:00
|
|
|
DBUG_ASSERT(buf);
|
|
|
|
DBUG_ASSERT(ndb_supported_type(field->type()));
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_ASSERT(field->ptr != NULL);
|
|
|
|
if (! (field->flags & BLOB_FLAG))
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
2004-12-23 15:28:41 +01:00
|
|
|
if (field->type() != MYSQL_TYPE_BIT)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
byte *field_buf;
|
|
|
|
if (field->pack_length() != 0)
|
|
|
|
field_buf= buf + (field->ptr - table->record[0]);
|
|
|
|
else
|
|
|
|
field_buf= (byte *)&dummy_buf;
|
|
|
|
m_value[fieldnr].rec= ndb_op->getValue(fieldnr,
|
|
|
|
field_buf);
|
|
|
|
}
|
2004-12-23 15:28:41 +01:00
|
|
|
else // if (field->type() == MYSQL_TYPE_BIT)
|
|
|
|
{
|
|
|
|
m_value[fieldnr].rec= ndb_op->getValue(fieldnr);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_RETURN(m_value[fieldnr].rec == NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Blob type
|
|
|
|
NdbBlob *ndb_blob= ndb_op->getBlobHandle(fieldnr);
|
|
|
|
m_value[fieldnr].blob= ndb_blob;
|
|
|
|
if (ndb_blob != NULL)
|
|
|
|
{
|
|
|
|
// Set callback
|
2006-06-21 09:36:50 +02:00
|
|
|
m_blobs_offset= buf - (byte*) table->record[0];
|
2004-07-22 12:38:09 +02:00
|
|
|
void *arg= (void *)this;
|
2004-07-29 10:44:53 +02:00
|
|
|
DBUG_RETURN(ndb_blob->setActiveHook(g_get_ndb_blobs_value, arg) != 0);
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used for hidden key only
|
2006-02-10 17:40:22 +01:00
|
|
|
m_value[fieldnr].rec= ndb_op->getValue(fieldnr, m_ref);
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_RETURN(m_value[fieldnr].rec == NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Check if any set or get of blob value in current query.
|
|
|
|
*/
|
|
|
|
bool ha_ndbcluster::uses_blob_value(bool all_fields)
|
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->blob_fields == 0)
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return FALSE;
|
2004-07-22 12:38:09 +02:00
|
|
|
if (all_fields)
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return TRUE;
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
uint no_fields= table->s->fields;
|
2004-07-22 12:38:09 +02:00
|
|
|
int i;
|
2005-02-18 12:53:43 +01:00
|
|
|
THD *thd= current_thd;
|
2004-07-22 12:38:09 +02:00
|
|
|
// They always put blobs at the end..
|
|
|
|
for (i= no_fields - 1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
|
|
|
if (thd->query_id == field->query_id)
|
|
|
|
{
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return TRUE;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
return FALSE;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get metadata for this table from NDB
|
|
|
|
|
|
|
|
IMPLEMENTATION
|
|
|
|
- check that frm-file on disk is equal to frm-file
|
|
|
|
of table accessed in NDB
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::get_metadata(const char *path)
|
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-04-15 09:14:14 +02:00
|
|
|
const NDBTAB *tab;
|
|
|
|
int error;
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
bool invalidating_ndb_table= FALSE;
|
2004-09-24 18:58:25 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("get_metadata");
|
|
|
|
DBUG_PRINT("enter", ("m_tabname: %s, path: %s", m_tabname, path));
|
|
|
|
|
2004-09-24 18:58:25 +02:00
|
|
|
do {
|
2007-03-21 08:40:24 +01:00
|
|
|
const void *data= NULL, *pack_data= NULL;
|
2004-09-24 18:58:25 +02:00
|
|
|
uint length, pack_length;
|
|
|
|
|
|
|
|
if (!(tab= dict->getTable(m_tabname)))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
2005-04-28 09:39:29 +02:00
|
|
|
// Check if thread has stale local cache
|
2005-04-27 18:17:41 +02:00
|
|
|
if (tab->getObjectStatus() == NdbDictionary::Object::Invalid)
|
|
|
|
{
|
|
|
|
invalidate_dictionary_cache(FALSE);
|
|
|
|
if (!(tab= dict->getTable(m_tabname)))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
DBUG_PRINT("info", ("Table schema version: %d", tab->getObjectVersion()));
|
|
|
|
}
|
2004-09-24 18:58:25 +02:00
|
|
|
/*
|
|
|
|
Compare FrmData in NDB with frm file from disk.
|
|
|
|
*/
|
|
|
|
error= 0;
|
|
|
|
if (readfrm(path, &data, &length) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
packfrm(data, length, &pack_data, &pack_length))
|
2004-09-24 18:58:25 +02:00
|
|
|
{
|
|
|
|
my_free((char*)data, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
my_free((char*)pack_data, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-24 18:58:25 +02:00
|
|
|
if ((pack_length != tab->getFrmLength()) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
(memcmp(pack_data, tab->getFrmData(), pack_length)))
|
2004-09-24 18:58:25 +02:00
|
|
|
{
|
|
|
|
if (!invalidating_ndb_table)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_PRINT("info", ("Invalidating table"));
|
2005-04-27 18:17:41 +02:00
|
|
|
invalidate_dictionary_cache(TRUE);
|
2005-02-16 14:18:32 +01:00
|
|
|
invalidating_ndb_table= TRUE;
|
2004-09-24 18:58:25 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_PRINT("error",
|
|
|
|
("metadata, pack_length: %d getFrmLength: %d memcmp: %d",
|
|
|
|
pack_length, tab->getFrmLength(),
|
|
|
|
memcmp(pack_data, tab->getFrmData(), pack_length)));
|
|
|
|
DBUG_DUMP("pack_data", (char*)pack_data, pack_length);
|
|
|
|
DBUG_DUMP("frm", (char*)tab->getFrmData(), tab->getFrmLength());
|
|
|
|
error= 3;
|
|
|
|
invalidating_ndb_table= FALSE;
|
2004-09-24 18:58:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
invalidating_ndb_table= FALSE;
|
2004-09-24 18:58:25 +02:00
|
|
|
}
|
|
|
|
my_free((char*)data, MYF(0));
|
|
|
|
my_free((char*)pack_data, MYF(0));
|
|
|
|
} while (invalidating_ndb_table);
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
if (error)
|
|
|
|
DBUG_RETURN(error);
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2005-04-27 18:17:41 +02:00
|
|
|
m_table_version= tab->getObjectVersion();
|
2005-03-22 17:42:08 +01:00
|
|
|
m_table= (void *)tab;
|
|
|
|
m_table_info= NULL; // Set in external lock
|
|
|
|
|
|
|
|
DBUG_RETURN(build_index_list(ndb, table, ILBP_OPEN));
|
2004-05-24 12:35:39 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-01-26 11:31:46 +01:00
|
|
|
static int fix_unique_index_attr_order(NDB_INDEX_DATA &data,
|
2005-02-16 14:18:32 +01:00
|
|
|
const NDBINDEX *index,
|
|
|
|
KEY *key_info)
|
2005-01-26 11:31:46 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("fix_unique_index_attr_order");
|
|
|
|
unsigned sz= index->getNoOfIndexColumns();
|
|
|
|
|
|
|
|
if (data.unique_index_attrid_map)
|
|
|
|
my_free((char*)data.unique_index_attrid_map, MYF(0));
|
|
|
|
data.unique_index_attrid_map= (unsigned char*)my_malloc(sz,MYF(MY_WME));
|
2007-04-11 15:21:11 +02:00
|
|
|
if (data.unique_index_attrid_map == 0)
|
|
|
|
{
|
|
|
|
sql_print_error("fix_unique_index_attr_order: my_malloc(%u) failure",
|
|
|
|
(unsigned int)sz);
|
|
|
|
DBUG_RETURN(HA_ERR_OUT_OF_MEM);
|
|
|
|
}
|
2005-01-26 11:31:46 +01:00
|
|
|
|
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
|
|
|
DBUG_ASSERT(key_info->key_parts == sz);
|
|
|
|
for (unsigned i= 0; key_part != end; key_part++, i++)
|
|
|
|
{
|
|
|
|
const char *field_name= key_part->field->field_name;
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
data.unique_index_attrid_map[i]= 255;
|
|
|
|
#endif
|
|
|
|
for (unsigned j= 0; j < sz; j++)
|
|
|
|
{
|
2005-03-22 17:42:08 +01:00
|
|
|
const NDBCOL *c= index->getColumn(j);
|
BUG#9626
- Fix valgrind warnings
- Remove static db, schema and table name buffers from Ndb.cpp
mysys/thr_alarm.c:
Initialise sact to zero
ndb/include/kernel/ndb_limits.h:
Set MAX_ATTR_NAME_SIZE to NAME_LEN which is tha maximum in MySQL
ndb/include/kernel/signaldata/GetTabInfo.hpp:
Clean up
ndb/include/ndbapi/Ndb.hpp:
Remove NDB_MAX_INTERNAL_NAME_LENGTH and all char buffers for schema, db and tablename.
Made them dynamic and moved to NdbImpl.hpp
ndb/include/ndbapi/ndbapi_limits.h:
Remove the static length's of attr, db, schema and table name.
ndb/src/common/transporter/Packer.cpp:
Set theSignalId to ~0 when unpacking signal
ndb/src/ndbapi/Ndb.cpp:
Moved schema, database and tablename to NdbImpl.hpp
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
Add NdbIndexImpl::init and NdbEventImpl::init
Init all vars in NdbIndexImpl, NdbEventImpl, NdbTableImpl and NdbIndexImpl
Delete the pseudo column NDB$RANGE_NO
Copy tablename to internal buff in NdbDictInterface to get proper alignment.
Convert length of table name from bytes words, when setting sz of LinearSectionPtr
Set LinearSectionPtr array size to number of LinearSections used - save some stack.
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
Add NdbEventImpl::init and NdbIndexImpl::init
Remove clearNewProperties and copyNewProperties, it's easier to check if all vars are initied if it's done in the same func.
Add buffer for tabname ti NdbDictInterface, memset it to 0 in initializer.
ndb/src/ndbapi/NdbImpl.hpp:
Use BaseString for table, schema and db names.
ndb/src/ndbapi/Ndbinit.cpp:
Move schema and db name to NdbImpl and use BaseString
ndb/src/ndbapi/ndb_cluster_connection.cpp:
Destroy ndb_global_event_buffer_mutex and ndb_print_state_mutex
sql/ha_ndbcluster.cc:
Check if pTrans is not null before calling closeTransaction
Remove NDB_MAX_ATTR_NAME_SIZE
Remove truncation of attr names. When attr name length is same in NDB as in MySQL this will be checked in functin check_column_name
2005-05-18 20:50:29 +02:00
|
|
|
if (strcmp(field_name, c->getName()) == 0)
|
2005-01-26 11:31:46 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
data.unique_index_attrid_map[i]= j;
|
|
|
|
break;
|
2005-01-26 11:31:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_ASSERT(data.unique_index_attrid_map[i] != 255);
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-08-19 11:10:35 +02:00
|
|
|
|
2006-05-05 17:18:47 +02:00
|
|
|
|
|
|
|
|
2005-03-22 17:42:08 +01:00
|
|
|
int ha_ndbcluster::build_index_list(Ndb *ndb, TABLE *tab, enum ILBP phase)
|
2004-05-24 12:35:39 +02:00
|
|
|
{
|
2004-09-25 14:35:27 +02:00
|
|
|
uint i;
|
2004-08-19 11:10:35 +02:00
|
|
|
int error= 0;
|
2005-01-11 17:00:31 +01:00
|
|
|
const char *index_name;
|
2004-09-25 14:35:27 +02:00
|
|
|
char unique_index_name[FN_LEN];
|
2006-11-07 16:38:37 +01:00
|
|
|
bool null_in_unique_index= false;
|
2004-05-24 12:35:39 +02:00
|
|
|
static const char* unique_suffix= "$unique";
|
2004-08-19 11:10:35 +02:00
|
|
|
KEY* key_info= tab->key_info;
|
2005-01-06 12:00:13 +01:00
|
|
|
const char **key_name= tab->s->keynames.type_names;
|
2005-03-22 17:42:08 +01:00
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::build_index_list");
|
2004-05-24 12:35:39 +02:00
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
m_has_unique_index= FALSE;
|
2004-04-15 09:14:14 +02:00
|
|
|
// Save information about all known indexes
|
2005-01-06 12:00:13 +01:00
|
|
|
for (i= 0; i < tab->s->keys; i++, key_info++, key_name++)
|
2004-04-30 12:25:31 +02:00
|
|
|
{
|
2004-08-19 11:10:35 +02:00
|
|
|
index_name= *key_name;
|
2004-05-24 12:35:39 +02:00
|
|
|
NDB_INDEX_TYPE idx_type= get_index_type_from_table(i);
|
2004-08-18 19:13:39 +02:00
|
|
|
m_index[i].type= idx_type;
|
2004-05-24 12:35:39 +02:00
|
|
|
if (idx_type == UNIQUE_ORDERED_INDEX || idx_type == UNIQUE_INDEX)
|
2004-04-30 12:25:31 +02:00
|
|
|
{
|
2006-03-23 09:48:46 +01:00
|
|
|
m_has_unique_index= TRUE;
|
2004-09-25 14:35:27 +02:00
|
|
|
strxnmov(unique_index_name, FN_LEN, index_name, unique_suffix, NullS);
|
|
|
|
DBUG_PRINT("info", ("Created unique index name \'%s\' for index %d",
|
2005-02-16 14:18:32 +01:00
|
|
|
unique_index_name, i));
|
2004-04-30 12:25:31 +02:00
|
|
|
}
|
2004-08-19 11:10:35 +02:00
|
|
|
// Create secondary indexes if in create phase
|
|
|
|
if (phase == ILBP_CREATE)
|
|
|
|
{
|
2004-09-25 14:35:27 +02:00
|
|
|
DBUG_PRINT("info", ("Creating index %u: %s", i, index_name));
|
|
|
|
switch (idx_type){
|
2005-02-16 14:18:32 +01:00
|
|
|
|
2004-08-19 11:10:35 +02:00
|
|
|
case PRIMARY_KEY_INDEX:
|
2005-02-16 14:18:32 +01:00
|
|
|
// Do nothing, already created
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
2005-02-16 14:18:32 +01:00
|
|
|
error= create_ordered_index(index_name, key_info);
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
case UNIQUE_ORDERED_INDEX:
|
2005-02-16 14:18:32 +01:00
|
|
|
if (!(error= create_ordered_index(index_name, key_info)))
|
|
|
|
error= create_unique_index(unique_index_name, key_info);
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
case UNIQUE_INDEX:
|
2006-11-07 16:38:37 +01:00
|
|
|
if (check_index_fields_not_null(i))
|
|
|
|
{
|
|
|
|
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_WARN,
|
|
|
|
ER_NULL_COLUMN_IN_INDEX,
|
|
|
|
"Ndb does not support unique index on NULL valued attributes, index access with NULL value will become full table scan");
|
|
|
|
null_in_unique_index= true;
|
|
|
|
}
|
|
|
|
error= create_unique_index(unique_index_name, key_info);
|
2005-02-16 14:18:32 +01:00
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
case ORDERED_INDEX:
|
2007-01-08 10:38:53 +01:00
|
|
|
if (key_info->algorithm == HA_KEY_ALG_HASH)
|
|
|
|
{
|
|
|
|
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_ERROR,
|
|
|
|
ER_UNSUPPORTED_EXTENSION,
|
|
|
|
ER(ER_UNSUPPORTED_EXTENSION),
|
|
|
|
"Ndb does not support non-unique "
|
|
|
|
"hash based indexes");
|
|
|
|
error= HA_ERR_UNSUPPORTED;
|
|
|
|
break;
|
|
|
|
}
|
2005-02-16 14:18:32 +01:00
|
|
|
error= create_ordered_index(index_name, key_info);
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
default:
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_ASSERT(FALSE);
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
}
|
|
|
|
if (error)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_PRINT("error", ("Failed to create index %u", i));
|
|
|
|
drop_table();
|
|
|
|
break;
|
2004-08-19 11:10:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Add handles to index objects
|
2004-09-25 14:35:27 +02:00
|
|
|
if (idx_type != PRIMARY_KEY_INDEX && idx_type != UNIQUE_INDEX)
|
2004-08-18 19:13:39 +02:00
|
|
|
{
|
2004-09-25 14:35:27 +02:00
|
|
|
DBUG_PRINT("info", ("Get handle to index %s", index_name));
|
2004-08-19 11:10:35 +02:00
|
|
|
const NDBINDEX *index= dict->getIndex(index_name, m_tabname);
|
2007-04-12 11:07:30 +02:00
|
|
|
if (!index)
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
2004-08-19 07:33:35 +02:00
|
|
|
m_index[i].index= (void *) index;
|
2004-08-18 19:13:39 +02:00
|
|
|
}
|
2004-09-25 14:35:27 +02:00
|
|
|
if (idx_type == UNIQUE_ORDERED_INDEX || idx_type == UNIQUE_INDEX)
|
2004-08-18 19:13:39 +02:00
|
|
|
{
|
2004-09-25 14:35:27 +02:00
|
|
|
DBUG_PRINT("info", ("Get handle to unique_index %s", unique_index_name));
|
|
|
|
const NDBINDEX *index= dict->getIndex(unique_index_name, m_tabname);
|
2007-04-12 11:07:30 +02:00
|
|
|
if (!index)
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
2004-08-19 07:33:35 +02:00
|
|
|
m_index[i].unique_index= (void *) index;
|
2005-01-26 11:31:46 +01:00
|
|
|
error= fix_unique_index_attr_order(m_index[i], index, key_info);
|
|
|
|
}
|
2006-11-07 16:38:37 +01:00
|
|
|
if (idx_type == UNIQUE_INDEX &&
|
|
|
|
phase != ILBP_CREATE &&
|
|
|
|
check_index_fields_not_null(i))
|
|
|
|
null_in_unique_index= true;
|
|
|
|
m_index[i].null_in_unique_index= null_in_unique_index;
|
2004-08-18 19:13:39 +02:00
|
|
|
}
|
2004-08-19 11:10:35 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(error);
|
2004-08-18 19:13:39 +02:00
|
|
|
}
|
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Decode the type of an index from information
|
|
|
|
provided in table object
|
|
|
|
*/
|
2004-05-24 12:35:39 +02:00
|
|
|
NDB_INDEX_TYPE ha_ndbcluster::get_index_type_from_table(uint inx) const
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-05-24 12:35:39 +02:00
|
|
|
bool is_hash_index= (table->key_info[inx].algorithm == HA_KEY_ALG_HASH);
|
2005-01-06 12:00:13 +01:00
|
|
|
if (inx == table->s->primary_key)
|
2004-05-24 12:35:39 +02:00
|
|
|
return is_hash_index ? PRIMARY_KEY_INDEX : PRIMARY_KEY_ORDERED_INDEX;
|
2005-01-06 12:00:13 +01:00
|
|
|
|
|
|
|
return ((table->key_info[inx].flags & HA_NOSAME) ?
|
|
|
|
(is_hash_index ? UNIQUE_INDEX : UNIQUE_ORDERED_INDEX) :
|
|
|
|
ORDERED_INDEX);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-05-24 12:35:39 +02:00
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
bool ha_ndbcluster::check_index_fields_not_null(uint inx)
|
2004-11-22 10:35:03 +01:00
|
|
|
{
|
|
|
|
KEY* key_info= table->key_info + inx;
|
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::check_index_fields_not_null");
|
2004-11-22 10:35:03 +01:00
|
|
|
|
|
|
|
for (; key_part != end; key_part++)
|
|
|
|
{
|
|
|
|
Field* field= key_part->field;
|
|
|
|
if (field->maybe_null())
|
2006-11-07 16:38:37 +01:00
|
|
|
DBUG_RETURN(true);
|
2004-11-22 10:35:03 +01:00
|
|
|
}
|
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
DBUG_RETURN(false);
|
2004-11-22 10:35:03 +01:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
void ha_ndbcluster::release_metadata()
|
|
|
|
{
|
2004-05-24 12:35:39 +02:00
|
|
|
uint i;
|
2004-04-30 12:25:31 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("release_metadata");
|
|
|
|
DBUG_PRINT("enter", ("m_tabname: %s", m_tabname));
|
|
|
|
|
|
|
|
m_table= NULL;
|
2004-09-21 14:52:56 +02:00
|
|
|
m_table_info= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
// Release index list
|
2004-04-30 12:25:31 +02:00
|
|
|
for (i= 0; i < MAX_KEY; i++)
|
|
|
|
{
|
2004-08-18 19:13:39 +02:00
|
|
|
m_index[i].unique_index= NULL;
|
|
|
|
m_index[i].index= NULL;
|
2005-01-26 11:31:46 +01:00
|
|
|
if (m_index[i].unique_index_attrid_map)
|
|
|
|
{
|
|
|
|
my_free((char *)m_index[i].unique_index_attrid_map, MYF(0));
|
|
|
|
m_index[i].unique_index_attrid_map= NULL;
|
|
|
|
}
|
2004-04-30 12:25:31 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2004-07-22 12:38:09 +02:00
|
|
|
int ha_ndbcluster::get_ndb_lock_type(enum thr_lock_type type)
|
2004-05-10 14:12:28 +02:00
|
|
|
{
|
2006-06-12 09:37:19 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::get_ndb_lock_type");
|
2004-10-13 10:08:18 +02:00
|
|
|
if (type >= TL_WRITE_ALLOW_WRITE)
|
2006-06-12 09:37:19 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Using exclusive lock"));
|
|
|
|
DBUG_RETURN(NdbOperation::LM_Exclusive);
|
|
|
|
}
|
|
|
|
else if (type == TL_READ_WITH_SHARED_LOCKS ||
|
|
|
|
uses_blob_value(m_retrieve_all_fields))
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Using read lock"));
|
|
|
|
DBUG_RETURN(NdbOperation::LM_Read);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
else
|
2006-06-12 09:37:19 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Using committed read"));
|
|
|
|
DBUG_RETURN(NdbOperation::LM_CommittedRead);
|
|
|
|
}
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
static const ulong index_type_flags[]=
|
|
|
|
{
|
|
|
|
/* UNDEFINED_INDEX */
|
|
|
|
0,
|
|
|
|
|
|
|
|
/* PRIMARY_KEY_INDEX */
|
2004-06-24 12:04:22 +02:00
|
|
|
HA_ONLY_WHOLE_INDEX,
|
2004-05-24 12:35:39 +02:00
|
|
|
|
|
|
|
/* PRIMARY_KEY_ORDERED_INDEX */
|
2004-05-10 14:46:06 +02:00
|
|
|
/*
|
2004-06-24 14:49:34 +02:00
|
|
|
Enable HA_KEYREAD_ONLY when "sorted" indexes are supported,
|
2004-05-10 14:46:06 +02:00
|
|
|
thus ORDERD BY clauses can be optimized by reading directly
|
|
|
|
through the index.
|
|
|
|
*/
|
2004-06-24 14:49:34 +02:00
|
|
|
// HA_KEYREAD_ONLY |
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_NEXT |
|
2004-12-20 15:12:42 +01:00
|
|
|
HA_READ_PREV |
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_RANGE |
|
|
|
|
HA_READ_ORDER,
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
/* UNIQUE_INDEX */
|
2004-06-24 12:04:22 +02:00
|
|
|
HA_ONLY_WHOLE_INDEX,
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
/* UNIQUE_ORDERED_INDEX */
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_NEXT |
|
2004-12-20 15:12:42 +01:00
|
|
|
HA_READ_PREV |
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_RANGE |
|
|
|
|
HA_READ_ORDER,
|
2004-05-24 12:35:39 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/* ORDERED_INDEX */
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_NEXT |
|
2004-12-20 15:12:42 +01:00
|
|
|
HA_READ_PREV |
|
2004-09-03 13:55:40 +02:00
|
|
|
HA_READ_RANGE |
|
|
|
|
HA_READ_ORDER
|
2004-04-15 09:14:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const int index_flags_size= sizeof(index_type_flags)/sizeof(ulong);
|
|
|
|
|
|
|
|
inline NDB_INDEX_TYPE ha_ndbcluster::get_index_type(uint idx_no) const
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(idx_no < MAX_KEY);
|
2004-08-18 19:13:39 +02:00
|
|
|
return m_index[idx_no].type;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
inline bool ha_ndbcluster::has_null_in_unique_index(uint idx_no) const
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(idx_no < MAX_KEY);
|
|
|
|
return m_index[idx_no].null_in_unique_index;
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Get the flags for an index
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
flags depending on the type of the index.
|
|
|
|
*/
|
|
|
|
|
2004-07-08 14:45:25 +02:00
|
|
|
inline ulong ha_ndbcluster::index_flags(uint idx_no, uint part,
|
|
|
|
bool all_parts) const
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_flags");
|
2004-05-10 14:46:06 +02:00
|
|
|
DBUG_PRINT("info", ("idx_no: %d", idx_no));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ASSERT(get_index_type_from_table(idx_no) < index_flags_size);
|
2005-09-30 12:20:15 +02:00
|
|
|
DBUG_RETURN(index_type_flags[get_index_type_from_table(idx_no)] |
|
|
|
|
HA_KEY_SCAN_NOT_ROR);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2005-01-07 11:55:20 +01:00
|
|
|
static void shrink_varchar(Field* field, const byte* & ptr, char* buf)
|
|
|
|
{
|
2005-09-01 14:12:48 +02:00
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR && ptr != NULL) {
|
2005-01-07 11:55:20 +01:00
|
|
|
Field_varstring* f= (Field_varstring*)field;
|
2005-04-21 12:53:40 +02:00
|
|
|
if (f->length_bytes == 1) {
|
2005-01-07 11:55:20 +01:00
|
|
|
uint pack_len= field->pack_length();
|
|
|
|
DBUG_ASSERT(1 <= pack_len && pack_len <= 256);
|
|
|
|
if (ptr[1] == 0) {
|
|
|
|
buf[0]= ptr[0];
|
|
|
|
} else {
|
2005-05-06 10:39:30 +02:00
|
|
|
DBUG_ASSERT(FALSE);
|
2005-01-07 11:55:20 +01:00
|
|
|
buf[0]= 255;
|
|
|
|
}
|
|
|
|
memmove(buf + 1, ptr + 2, pack_len - 1);
|
|
|
|
ptr= buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
int ha_ndbcluster::set_primary_key(NdbOperation *op, const byte *key)
|
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
KEY* key_info= table->key_info + table->s->primary_key;
|
2004-04-15 09:14:14 +02:00
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
|
|
|
DBUG_ENTER("set_primary_key");
|
|
|
|
|
|
|
|
for (; key_part != end; key_part++)
|
|
|
|
{
|
|
|
|
Field* field= key_part->field;
|
2005-01-07 11:55:20 +01:00
|
|
|
const byte* ptr= key;
|
|
|
|
char buf[256];
|
|
|
|
shrink_varchar(field, ptr, buf);
|
2004-04-15 09:14:14 +02:00
|
|
|
if (set_ndb_key(op, field,
|
2005-02-16 14:18:32 +01:00
|
|
|
key_part->fieldnr-1, ptr))
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2005-01-07 11:55:20 +01:00
|
|
|
key += key_part->store_length;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-06 11:23:36 +02:00
|
|
|
int ha_ndbcluster::set_primary_key_from_record(NdbOperation *op, const byte *record)
|
2004-05-17 09:51:02 +02:00
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
KEY* key_info= table->key_info + table->s->primary_key;
|
2004-05-17 09:51:02 +02:00
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
2005-06-29 16:13:50 +02:00
|
|
|
DBUG_ENTER("set_primary_key_from_record");
|
2004-05-17 09:51:02 +02:00
|
|
|
|
|
|
|
for (; key_part != end; key_part++)
|
|
|
|
{
|
|
|
|
Field* field= key_part->field;
|
|
|
|
if (set_ndb_key(op, field,
|
2005-07-06 11:23:36 +02:00
|
|
|
key_part->fieldnr-1, record+key_part->offset))
|
2004-05-17 09:51:02 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2007-10-02 13:36:13 +02:00
|
|
|
bool ha_ndbcluster::check_index_fields_in_write_set(uint keyno)
|
|
|
|
{
|
|
|
|
KEY* key_info= table->key_info + keyno;
|
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
|
|
|
uint i;
|
|
|
|
DBUG_ENTER("check_index_fields_in_write_set");
|
|
|
|
|
|
|
|
if (m_retrieve_all_fields)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
|
|
|
for (i= 0; key_part != end; key_part++, i++)
|
|
|
|
{
|
|
|
|
Field* field= key_part->field;
|
|
|
|
if (field->query_id != current_thd->query_id)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
int ha_ndbcluster::set_index_key_from_record(NdbOperation *op, const byte *record, uint keyno)
|
|
|
|
{
|
|
|
|
KEY* key_info= table->key_info + keyno;
|
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
|
|
|
uint i;
|
|
|
|
DBUG_ENTER("set_index_key_from_record");
|
|
|
|
|
|
|
|
for (i= 0; key_part != end; key_part++, i++)
|
|
|
|
{
|
|
|
|
Field* field= key_part->field;
|
|
|
|
if (set_ndb_key(op, field, m_index[keyno].unique_index_attrid_map[i],
|
|
|
|
record+key_part->offset))
|
|
|
|
ERR_RETURN(m_active_trans->getNdbError());
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-11-30 13:37:03 +01:00
|
|
|
int
|
|
|
|
ha_ndbcluster::set_index_key(NdbOperation *op,
|
2005-02-16 14:18:32 +01:00
|
|
|
const KEY *key_info,
|
|
|
|
const byte * key_ptr)
|
2004-11-30 13:37:03 +01:00
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::set_index_key");
|
2004-11-30 13:37:03 +01:00
|
|
|
uint i;
|
|
|
|
KEY_PART_INFO* key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO* end= key_part+key_info->key_parts;
|
|
|
|
|
|
|
|
for (i= 0; key_part != end; key_part++, i++)
|
|
|
|
{
|
2005-01-07 11:55:20 +01:00
|
|
|
Field* field= key_part->field;
|
|
|
|
const byte* ptr= key_part->null_bit ? key_ptr + 1 : key_ptr;
|
|
|
|
char buf[256];
|
|
|
|
shrink_varchar(field, ptr, buf);
|
2005-01-26 13:22:07 +01:00
|
|
|
if (set_ndb_key(op, field, m_index[active_index].unique_index_attrid_map[i], ptr))
|
2004-11-30 13:37:03 +01:00
|
|
|
ERR_RETURN(m_active_trans->getNdbError());
|
|
|
|
key_ptr+= key_part->store_length;
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-01-18 16:31:05 +01:00
|
|
|
inline
|
|
|
|
int ha_ndbcluster::define_read_attrs(byte* buf, NdbOperation* op)
|
|
|
|
{
|
|
|
|
uint i;
|
|
|
|
THD *thd= current_thd;
|
|
|
|
|
|
|
|
DBUG_ENTER("define_read_attrs");
|
|
|
|
|
|
|
|
// Define attributes to read
|
|
|
|
for (i= 0; i < table->s->fields; i++)
|
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
|
|
|
if ((thd->query_id == field->query_id) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
((field->flags & PRI_KEY_FLAG)) ||
|
|
|
|
m_retrieve_all_fields)
|
2005-01-18 16:31:05 +01:00
|
|
|
{
|
|
|
|
if (get_ndb_value(op, field, i, buf))
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2005-01-18 16:31:05 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_value[i].ptr= NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (table->s->primary_key == MAX_KEY)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Getting hidden key"));
|
|
|
|
// Scanning table with no primary key
|
|
|
|
int hidden_no= table->s->fields;
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
const NDBTAB *tab= (const NDBTAB *) m_table;
|
|
|
|
if (!tab->getColumn(hidden_no))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
#endif
|
|
|
|
if (get_ndb_value(op, NULL, hidden_no, NULL))
|
|
|
|
ERR_RETURN(op->getNdbError());
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Read one record from NDB using primary key
|
|
|
|
*/
|
|
|
|
|
2004-07-22 18:35:51 +02:00
|
|
|
int ha_ndbcluster::pk_read(const byte *key, uint key_len, byte *buf)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-01-11 17:00:31 +01:00
|
|
|
uint no_fields= table->s->fields;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbConnection *trans= m_active_trans;
|
|
|
|
NdbOperation *op;
|
2005-01-11 17:00:31 +01:00
|
|
|
|
2004-11-17 10:07:52 +01:00
|
|
|
int res;
|
|
|
|
DBUG_ENTER("pk_read");
|
|
|
|
DBUG_PRINT("enter", ("key_len: %u", key_len));
|
|
|
|
DBUG_DUMP("key", (char*)key, key_len);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-28 19:35:07 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2004-09-07 21:53:59 +02:00
|
|
|
if (!(op= trans->getNdbOperation((const NDBTAB *) m_table)) ||
|
2004-09-28 19:35:07 +02:00
|
|
|
op->readTuple(lm) != 0)
|
2004-07-23 10:33:51 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-11-17 10:07:52 +01:00
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key == MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
// This table has no primary key, use "hidden" primary key
|
|
|
|
DBUG_PRINT("info", ("Using hidden key"));
|
|
|
|
DBUG_DUMP("key", (char*)key, 8);
|
|
|
|
if (set_hidden_key(op, no_fields, key))
|
2004-07-23 10:33:51 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-11-17 10:07:52 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Read key at the same time, for future reference
|
2004-09-20 12:40:53 +02:00
|
|
|
if (get_ndb_value(op, NULL, no_fields, NULL))
|
2004-07-23 10:33:51 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((res= set_primary_key(op, key)))
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= define_read_attrs(buf, op)))
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit_ie(this,trans,false) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
}
|
|
|
|
|
|
|
|
// The value have now been fetched from NDB
|
|
|
|
unpack_record(buf);
|
|
|
|
table->status= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-07-22 18:35:51 +02:00
|
|
|
/*
|
|
|
|
Read one complementing record from NDB using primary key from old_data
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::complemented_pk_read(const byte *old_data, byte *new_data)
|
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
uint no_fields= table->s->fields, i;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-07-22 18:35:51 +02:00
|
|
|
NdbOperation *op;
|
|
|
|
THD *thd= current_thd;
|
|
|
|
DBUG_ENTER("complemented_pk_read");
|
|
|
|
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_retrieve_all_fields)
|
2004-07-22 18:35:51 +02:00
|
|
|
// We have allready retrieved all fields, nothing to complement
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
2004-09-28 19:35:07 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2004-09-07 21:53:59 +02:00
|
|
|
if (!(op= trans->getNdbOperation((const NDBTAB *) m_table)) ||
|
2004-09-28 19:35:07 +02:00
|
|
|
op->readTuple(lm) != 0)
|
2004-07-23 10:33:51 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-12-01 12:43:30 +01:00
|
|
|
int res;
|
2005-07-04 20:36:38 +02:00
|
|
|
if ((res= set_primary_key_from_record(op, old_data)))
|
2004-12-01 12:43:30 +01:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-07-22 18:35:51 +02:00
|
|
|
// Read all unreferenced non-key field(s)
|
|
|
|
for (i= 0; i < no_fields; i++)
|
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
2004-12-01 12:43:30 +01:00
|
|
|
if (!((field->flags & PRI_KEY_FLAG) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
(thd->query_id == field->query_id)))
|
2004-07-22 18:35:51 +02:00
|
|
|
{
|
2004-09-20 12:40:53 +02:00
|
|
|
if (get_ndb_value(op, field, i, new_data))
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-07-22 18:35:51 +02:00
|
|
|
}
|
|
|
|
}
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2004-07-22 18:35:51 +02:00
|
|
|
{
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
}
|
|
|
|
|
|
|
|
// The value have now been fetched from NDB
|
|
|
|
unpack_record(new_data);
|
|
|
|
table->status= 0;
|
2004-12-01 12:43:30 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* restore m_value
|
|
|
|
*/
|
|
|
|
for (i= 0; i < no_fields; i++)
|
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
|
|
|
if (!((field->flags & PRI_KEY_FLAG) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
(thd->query_id == field->query_id)))
|
2004-12-01 12:43:30 +01:00
|
|
|
{
|
|
|
|
m_value[i].ptr= NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-22 18:35:51 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-11-18 12:11:56 +01:00
|
|
|
/*
|
2006-03-23 09:48:46 +01:00
|
|
|
* Check that all operations between first and last all
|
|
|
|
* have gotten the errcode
|
|
|
|
* If checking for HA_ERR_KEY_NOT_FOUND then update m_dupkey
|
|
|
|
* for all succeeding operations
|
|
|
|
*/
|
|
|
|
bool ha_ndbcluster::check_all_operations_for_error(NdbTransaction *trans,
|
|
|
|
const NdbOperation *first,
|
|
|
|
const NdbOperation *last,
|
|
|
|
uint errcode)
|
|
|
|
{
|
|
|
|
const NdbOperation *op= first;
|
|
|
|
DBUG_ENTER("ha_ndbcluster::check_all_operations_for_error");
|
|
|
|
|
|
|
|
while(op)
|
|
|
|
{
|
|
|
|
NdbError err= op->getNdbError();
|
|
|
|
if (err.status != NdbError::Success)
|
|
|
|
{
|
|
|
|
if (ndb_to_mysql_error(&err) != (int) errcode)
|
|
|
|
DBUG_RETURN(false);
|
|
|
|
if (op == last) break;
|
|
|
|
op= trans->getNextCompletedOperation(op);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We found a duplicate
|
|
|
|
if (op->getType() == NdbOperation::UniqueIndexAccess)
|
|
|
|
{
|
|
|
|
if (errcode == HA_ERR_KEY_NOT_FOUND)
|
|
|
|
{
|
|
|
|
NdbIndexOperation *iop= (NdbIndexOperation *) op;
|
|
|
|
const NDBINDEX *index= iop->getIndex();
|
|
|
|
// Find the key_no of the index
|
|
|
|
for(uint i= 0; i<table->s->keys; i++)
|
|
|
|
{
|
|
|
|
if (m_index[i].unique_index == index)
|
|
|
|
{
|
|
|
|
m_dupkey= i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Must have been primary key access
|
|
|
|
DBUG_ASSERT(op->getType() == NdbOperation::PrimaryKeyAccess);
|
|
|
|
if (errcode == HA_ERR_KEY_NOT_FOUND)
|
|
|
|
m_dupkey= table->s->primary_key;
|
|
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
|
|
|
|
2007-05-02 10:02:27 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if record contains any null valued columns that are part of a key
|
|
|
|
*/
|
|
|
|
static
|
|
|
|
int
|
|
|
|
check_null_in_record(const KEY* key_info, const byte *record)
|
|
|
|
{
|
|
|
|
KEY_PART_INFO *curr_part, *end_part;
|
|
|
|
curr_part= key_info->key_part;
|
|
|
|
end_part= curr_part + key_info->key_parts;
|
|
|
|
|
|
|
|
while (curr_part != end_part)
|
|
|
|
{
|
|
|
|
if (curr_part->null_bit &&
|
|
|
|
(record[curr_part->null_offset] & curr_part->null_bit))
|
|
|
|
return 1;
|
|
|
|
curr_part++;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
/*
|
|
|
|
We could instead pre-compute a bitmask in table_share with one bit for
|
|
|
|
every null-bit in the key, and so check this just by OR'ing the bitmask
|
|
|
|
with the null bitmap in the record.
|
|
|
|
But not sure it's worth it.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
/*
|
|
|
|
* Peek to check if any rows already exist with conflicting
|
|
|
|
* primary key or unique index values
|
2004-11-18 12:11:56 +01:00
|
|
|
*/
|
|
|
|
|
2007-10-02 13:36:13 +02:00
|
|
|
int ha_ndbcluster::peek_indexed_rows(const byte *record,
|
|
|
|
NDB_WRITE_OP write_op)
|
2004-11-18 12:11:56 +01:00
|
|
|
{
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-11-18 12:11:56 +01:00
|
|
|
NdbOperation *op;
|
2006-03-23 09:48:46 +01:00
|
|
|
const NdbOperation *first, *last;
|
|
|
|
uint i;
|
|
|
|
int res;
|
|
|
|
DBUG_ENTER("peek_indexed_rows");
|
2004-12-08 00:36:40 +01:00
|
|
|
|
2006-10-19 09:27:58 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
first= NULL;
|
2007-10-02 13:36:13 +02:00
|
|
|
if (write_op != NDB_UPDATE && table->s->primary_key != MAX_KEY)
|
2006-03-23 09:48:46 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Fetch any row with colliding primary key
|
|
|
|
*/
|
|
|
|
if (!(op= trans->getNdbOperation((const NDBTAB *) m_table)) ||
|
|
|
|
op->readTuple(lm) != 0)
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
|
|
|
first= op;
|
|
|
|
if ((res= set_primary_key_from_record(op, record)))
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Fetch any rows with colliding unique indexes
|
|
|
|
*/
|
|
|
|
KEY* key_info;
|
|
|
|
KEY_PART_INFO *key_part, *end;
|
|
|
|
for (i= 0, key_info= table->key_info; i < table->s->keys; i++, key_info++)
|
|
|
|
{
|
|
|
|
if (i != table->s->primary_key &&
|
|
|
|
key_info->flags & HA_NOSAME)
|
|
|
|
{
|
2007-05-02 10:02:27 +02:00
|
|
|
/*
|
|
|
|
A unique index is defined on table.
|
|
|
|
We cannot look up a NULL field value in a unique index. But since
|
|
|
|
keys with NULLs are not indexed, such rows cannot conflict anyway, so
|
|
|
|
we just skip the index in this case.
|
|
|
|
*/
|
|
|
|
if (check_null_in_record(key_info, record))
|
|
|
|
{
|
2007-10-03 09:29:10 +02:00
|
|
|
DBUG_PRINT("info", ("skipping check for key with NULL"));
|
2007-05-02 10:02:27 +02:00
|
|
|
continue;
|
|
|
|
}
|
2007-10-02 13:36:13 +02:00
|
|
|
if (write_op != NDB_INSERT && !check_index_fields_in_write_set(i))
|
|
|
|
{
|
2007-10-03 09:29:10 +02:00
|
|
|
DBUG_PRINT("info", ("skipping check for key %u not in write_set", i));
|
2007-10-02 13:36:13 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
NdbIndexOperation *iop;
|
|
|
|
NDBINDEX *unique_index = (NDBINDEX *) m_index[i].unique_index;
|
|
|
|
key_part= key_info->key_part;
|
|
|
|
end= key_part + key_info->key_parts;
|
|
|
|
if (!(iop= trans->getNdbIndexOperation(unique_index,
|
|
|
|
(const NDBTAB *) m_table)) ||
|
|
|
|
iop->readTuple(lm) != 0)
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-12-08 00:36:40 +01:00
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
if (!first)
|
|
|
|
first= iop;
|
|
|
|
if ((res= set_index_key_from_record(iop, record, i)))
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
last= trans->getLastDefinedOperation();
|
|
|
|
if (first)
|
2006-08-15 14:31:21 +02:00
|
|
|
res= execute_no_commit_ie(this,trans,false);
|
2006-03-23 09:48:46 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Table has no keys
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(HA_ERR_KEY_NOT_FOUND);
|
|
|
|
}
|
|
|
|
if (check_all_operations_for_error(trans, first, last,
|
|
|
|
HA_ERR_KEY_NOT_FOUND))
|
2004-12-08 00:36:40 +01:00
|
|
|
{
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
}
|
2006-03-23 09:48:46 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("m_dupkey %d", m_dupkey));
|
|
|
|
}
|
2004-11-18 12:11:56 +01:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-07-22 18:35:51 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Read one record from NDB using unique secondary index
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::unique_index_read(const byte *key,
|
2005-02-16 14:18:32 +01:00
|
|
|
uint key_len, byte *buf)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-11-17 10:07:52 +01:00
|
|
|
int res;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbIndexOperation *op;
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::unique_index_read");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("key_len: %u, index: %u", key_len, active_index));
|
|
|
|
DBUG_DUMP("key", (char*)key, key_len);
|
|
|
|
|
2004-09-28 19:35:07 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2004-08-18 19:13:39 +02:00
|
|
|
if (!(op= trans->getNdbIndexOperation((NDBINDEX *)
|
2005-02-16 14:18:32 +01:00
|
|
|
m_index[active_index].unique_index,
|
2004-09-07 21:53:59 +02:00
|
|
|
(const NDBTAB *) m_table)) ||
|
2004-09-28 19:35:07 +02:00
|
|
|
op->readTuple(lm) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
|
|
|
// Set secondary index key(s)
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= set_index_key(op, table->key_info + active_index, key)))
|
2004-11-30 13:37:03 +01:00
|
|
|
DBUG_RETURN(res);
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= define_read_attrs(buf, op)))
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit_ie(this,trans,false) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2007-08-13 09:22:42 +02:00
|
|
|
int err= ndb_err(trans);
|
|
|
|
if(err==HA_ERR_KEY_NOT_FOUND)
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
else
|
|
|
|
table->status= STATUS_GARBAGE;
|
|
|
|
|
|
|
|
DBUG_RETURN(err);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2007-08-13 09:22:42 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// The value have now been fetched from NDB
|
|
|
|
unpack_record(buf);
|
|
|
|
table->status= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
inline int ha_ndbcluster::fetch_next(NdbScanOperation* cursor)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("fetch_next");
|
2006-12-14 23:51:37 +01:00
|
|
|
int local_check;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2006-06-12 09:37:19 +02:00
|
|
|
if (m_lock_tuple)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Lock level m_lock.type either TL_WRITE_ALLOW_WRITE
|
|
|
|
(SELECT FOR UPDATE) or TL_READ_WITH_SHARED_LOCKS (SELECT
|
|
|
|
LOCK WITH SHARE MODE) and row was not explictly unlocked
|
|
|
|
with unlock_row() call
|
|
|
|
*/
|
2006-12-14 23:51:37 +01:00
|
|
|
NdbConnection *con_trans= m_active_trans;
|
2006-06-12 09:37:19 +02:00
|
|
|
NdbOperation *op;
|
|
|
|
// Lock row
|
|
|
|
DBUG_PRINT("info", ("Keeping lock on scanned row"));
|
|
|
|
|
|
|
|
if (!(op= m_active_cursor->lockCurrentTuple()))
|
|
|
|
{
|
2006-12-14 23:51:37 +01:00
|
|
|
/* purecov: begin inspected */
|
2006-06-12 09:37:19 +02:00
|
|
|
m_lock_tuple= false;
|
2006-12-14 23:51:37 +01:00
|
|
|
ERR_RETURN(con_trans->getNdbError());
|
|
|
|
/* purecov: end */
|
2006-06-12 09:37:19 +02:00
|
|
|
}
|
|
|
|
m_ops_pending++;
|
|
|
|
}
|
|
|
|
m_lock_tuple= false;
|
|
|
|
|
|
|
|
bool contact_ndb= m_lock.type < TL_WRITE_ALLOW_WRITE &&
|
|
|
|
m_lock.type != TL_READ_WITH_SHARED_LOCKS;
|
2004-05-10 14:12:28 +02:00
|
|
|
do {
|
|
|
|
DBUG_PRINT("info", ("Call nextResult, contact_ndb: %d", contact_ndb));
|
2004-07-22 12:38:09 +02:00
|
|
|
/*
|
|
|
|
We can only handle one tuple with blobs at a time.
|
|
|
|
*/
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_ops_pending && m_blobs_pending)
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending= 0;
|
|
|
|
m_blobs_pending= FALSE;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2006-12-14 23:51:37 +01:00
|
|
|
if ((local_check= cursor->nextResult(contact_ndb, m_force_send)) == 0)
|
2004-05-10 14:12:28 +02:00
|
|
|
{
|
2006-06-12 09:37:19 +02:00
|
|
|
/*
|
|
|
|
Explicitly lock tuple if "select for update" or
|
|
|
|
"select lock in share mode"
|
|
|
|
*/
|
|
|
|
m_lock_tuple= (m_lock.type == TL_WRITE_ALLOW_WRITE
|
|
|
|
||
|
|
|
|
m_lock.type == TL_READ_WITH_SHARED_LOCKS);
|
2004-05-10 14:12:28 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2006-12-14 23:51:37 +01:00
|
|
|
else if (local_check == 1 || local_check == 2)
|
2004-05-10 14:12:28 +02:00
|
|
|
{
|
|
|
|
// 1: No more records
|
|
|
|
// 2: No more cached records
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2004-05-10 14:12:28 +02:00
|
|
|
/*
|
2005-02-16 14:18:32 +01:00
|
|
|
Before fetching more rows and releasing lock(s),
|
|
|
|
all pending update or delete operations should
|
|
|
|
be sent to NDB
|
2004-05-10 14:12:28 +02:00
|
|
|
*/
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("info", ("ops_pending: %ld", (long) m_ops_pending));
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_ops_pending)
|
2004-09-03 17:11:09 +02:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
if (m_transaction_on)
|
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (execute_commit(this,trans) != 0)
|
|
|
|
DBUG_RETURN(-1);
|
2005-07-04 02:42:33 +02:00
|
|
|
if (trans->restart() != 0)
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_ops_pending= 0;
|
2004-09-03 17:11:09 +02:00
|
|
|
}
|
2006-12-14 23:51:37 +01:00
|
|
|
contact_ndb= (local_check == 2);
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
2004-12-08 00:36:40 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
2006-12-14 23:51:37 +01:00
|
|
|
} while (local_check == 2);
|
2004-12-08 00:36:40 +01:00
|
|
|
|
2004-12-07 15:15:49 +01:00
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get the next record of a started scan. Try to fetch
|
|
|
|
it locally from NdbApi cached records if possible,
|
|
|
|
otherwise ask NDB for more.
|
|
|
|
|
|
|
|
NOTE
|
|
|
|
If this is a update/delete make sure to not contact
|
|
|
|
NDB before any pending ops have been sent to NDB.
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-12-07 15:15:49 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
inline int ha_ndbcluster::next_result(byte *buf)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
DBUG_ENTER("next_result");
|
|
|
|
|
2004-12-15 15:13:11 +01:00
|
|
|
if (!m_active_cursor)
|
|
|
|
DBUG_RETURN(HA_ERR_END_OF_FILE);
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= fetch_next(m_active_cursor)) == 0)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("One more record found"));
|
|
|
|
|
|
|
|
unpack_record(buf);
|
|
|
|
table->status= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2005-07-04 02:42:33 +02:00
|
|
|
else if (res == 1)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
|
|
|
// No more records
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
|
|
|
|
DBUG_PRINT("info", ("No more records"));
|
|
|
|
DBUG_RETURN(HA_ERR_END_OF_FILE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_RETURN(ndb_err(m_active_trans));
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2004-04-30 13:38:41 +02:00
|
|
|
/*
|
2004-10-08 10:50:50 +02:00
|
|
|
Set bounds for ordered index scan.
|
2004-04-30 13:38:41 +02:00
|
|
|
*/
|
|
|
|
|
2004-06-11 13:49:22 +02:00
|
|
|
int ha_ndbcluster::set_bounds(NdbIndexScanOperation *op,
|
2005-02-16 14:18:32 +01:00
|
|
|
const key_range *keys[2],
|
|
|
|
uint range_no)
|
2004-04-30 13:38:41 +02:00
|
|
|
{
|
2004-10-08 10:50:50 +02:00
|
|
|
const KEY *const key_info= table->key_info + active_index;
|
|
|
|
const uint key_parts= key_info->key_parts;
|
|
|
|
uint key_tot_len[2];
|
|
|
|
uint tot_len;
|
2004-10-25 22:35:46 +02:00
|
|
|
uint i, j;
|
2004-04-30 13:38:41 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("set_bounds");
|
2004-10-08 10:50:50 +02:00
|
|
|
DBUG_PRINT("info", ("key_parts=%d", key_parts));
|
2004-04-30 13:38:41 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
for (j= 0; j <= 1; j++)
|
2004-04-30 13:38:41 +02:00
|
|
|
{
|
2004-10-08 10:50:50 +02:00
|
|
|
const key_range *key= keys[j];
|
|
|
|
if (key != NULL)
|
|
|
|
{
|
|
|
|
// for key->flag see ha_rkey_function
|
|
|
|
DBUG_PRINT("info", ("key %d length=%d flag=%d",
|
|
|
|
j, key->length, key->flag));
|
|
|
|
key_tot_len[j]= key->length;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("key %d not present", j));
|
|
|
|
key_tot_len[j]= 0;
|
|
|
|
}
|
2004-04-30 13:38:41 +02:00
|
|
|
}
|
|
|
|
tot_len= 0;
|
2004-08-31 10:19:10 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
for (i= 0; i < key_parts; i++)
|
|
|
|
{
|
|
|
|
KEY_PART_INFO *key_part= &key_info->key_part[i];
|
|
|
|
Field *field= key_part->field;
|
2005-01-11 17:00:31 +01:00
|
|
|
#ifndef DBUG_OFF
|
2004-10-08 10:50:50 +02:00
|
|
|
uint part_len= key_part->length;
|
2005-01-11 17:00:31 +01:00
|
|
|
#endif
|
2004-10-08 10:50:50 +02:00
|
|
|
uint part_store_len= key_part->store_length;
|
|
|
|
// Info about each key part
|
|
|
|
struct part_st {
|
|
|
|
bool part_last;
|
|
|
|
const key_range *key;
|
|
|
|
const byte *part_ptr;
|
|
|
|
bool part_null;
|
|
|
|
int bound_type;
|
|
|
|
const char* bound_ptr;
|
|
|
|
};
|
|
|
|
struct part_st part[2];
|
|
|
|
|
|
|
|
for (j= 0; j <= 1; j++)
|
|
|
|
{
|
2004-12-07 15:15:49 +01:00
|
|
|
struct part_st &p= part[j];
|
2004-10-08 10:50:50 +02:00
|
|
|
p.key= NULL;
|
|
|
|
p.bound_type= -1;
|
|
|
|
if (tot_len < key_tot_len[j])
|
|
|
|
{
|
|
|
|
p.part_last= (tot_len + part_store_len >= key_tot_len[j]);
|
|
|
|
p.key= keys[j];
|
|
|
|
p.part_ptr= &p.key->key[tot_len];
|
2004-11-08 11:11:49 +01:00
|
|
|
p.part_null= key_part->null_bit && *p.part_ptr;
|
2004-10-08 10:50:50 +02:00
|
|
|
p.bound_ptr= (const char *)
|
2004-11-08 11:11:49 +01:00
|
|
|
p.part_null ? 0 : key_part->null_bit ? p.part_ptr + 1 : p.part_ptr;
|
2004-10-08 10:50:50 +02:00
|
|
|
|
|
|
|
if (j == 0)
|
|
|
|
{
|
|
|
|
switch (p.key->flag)
|
|
|
|
{
|
|
|
|
case HA_READ_KEY_EXACT:
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundEQ;
|
|
|
|
break;
|
2004-12-20 15:12:42 +01:00
|
|
|
// ascending
|
2004-10-08 10:50:50 +02:00
|
|
|
case HA_READ_KEY_OR_NEXT:
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundLE;
|
|
|
|
break;
|
|
|
|
case HA_READ_AFTER_KEY:
|
|
|
|
if (! p.part_last)
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundLE;
|
|
|
|
else
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundLT;
|
|
|
|
break;
|
2004-12-20 15:12:42 +01:00
|
|
|
// descending
|
|
|
|
case HA_READ_PREFIX_LAST: // weird
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundEQ;
|
|
|
|
break;
|
|
|
|
case HA_READ_PREFIX_LAST_OR_PREV: // weird
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGE;
|
|
|
|
break;
|
|
|
|
case HA_READ_BEFORE_KEY:
|
|
|
|
if (! p.part_last)
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGE;
|
|
|
|
else
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGT;
|
|
|
|
break;
|
2004-10-08 10:50:50 +02:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (j == 1) {
|
|
|
|
switch (p.key->flag)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
// ascending
|
2004-10-08 10:50:50 +02:00
|
|
|
case HA_READ_BEFORE_KEY:
|
|
|
|
if (! p.part_last)
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGE;
|
|
|
|
else
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGT;
|
|
|
|
break;
|
|
|
|
case HA_READ_AFTER_KEY: // weird
|
|
|
|
p.bound_type= NdbIndexScanOperation::BoundGE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2004-12-20 15:12:42 +01:00
|
|
|
// descending strangely sets no end key
|
2004-10-08 10:50:50 +02:00
|
|
|
}
|
|
|
|
}
|
2004-04-30 13:38:41 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
if (p.bound_type == -1)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("key %d unknown flag %d", j, p.key->flag));
|
2005-05-06 10:39:30 +02:00
|
|
|
DBUG_ASSERT(FALSE);
|
2004-10-08 10:50:50 +02:00
|
|
|
// Stop setting bounds but continue with what we have
|
2007-02-07 11:36:34 +01:00
|
|
|
DBUG_RETURN(op->end_of_bound(range_no));
|
2004-10-08 10:50:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-04-30 13:38:41 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
// Seen with e.g. b = 1 and c > 1
|
|
|
|
if (part[0].bound_type == NdbIndexScanOperation::BoundLE &&
|
|
|
|
part[1].bound_type == NdbIndexScanOperation::BoundGE &&
|
|
|
|
memcmp(part[0].part_ptr, part[1].part_ptr, part_store_len) == 0)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("replace LE/GE pair by EQ"));
|
|
|
|
part[0].bound_type= NdbIndexScanOperation::BoundEQ;
|
|
|
|
part[1].bound_type= -1;
|
|
|
|
}
|
|
|
|
// Not seen but was in previous version
|
|
|
|
if (part[0].bound_type == NdbIndexScanOperation::BoundEQ &&
|
|
|
|
part[1].bound_type == NdbIndexScanOperation::BoundGE &&
|
|
|
|
memcmp(part[0].part_ptr, part[1].part_ptr, part_store_len) == 0)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("remove GE from EQ/GE pair"));
|
|
|
|
part[1].bound_type= -1;
|
|
|
|
}
|
2004-09-06 10:33:12 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
for (j= 0; j <= 1; j++)
|
|
|
|
{
|
2004-12-07 15:15:49 +01:00
|
|
|
struct part_st &p= part[j];
|
2004-10-08 10:50:50 +02:00
|
|
|
// Set bound if not done with this key
|
|
|
|
if (p.key != NULL)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("key %d:%d offset=%d length=%d last=%d bound=%d",
|
|
|
|
j, i, tot_len, part_len, p.part_last, p.bound_type));
|
|
|
|
DBUG_DUMP("info", (const char*)p.part_ptr, part_store_len);
|
|
|
|
|
|
|
|
// Set bound if not cancelled via type -1
|
|
|
|
if (p.bound_type != -1)
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
2005-01-07 11:55:20 +01:00
|
|
|
const char* ptr= p.bound_ptr;
|
|
|
|
char buf[256];
|
|
|
|
shrink_varchar(field, ptr, buf);
|
2005-01-26 13:22:07 +01:00
|
|
|
if (op->setBound(i, p.bound_type, ptr))
|
2004-10-08 10:50:50 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2005-02-16 14:18:32 +01:00
|
|
|
}
|
2004-10-08 10:50:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tot_len+= part_store_len;
|
2004-09-06 10:33:12 +02:00
|
|
|
}
|
2007-02-07 11:36:34 +01:00
|
|
|
DBUG_RETURN(op->end_of_bound(range_no));
|
2004-04-30 13:38:41 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
2004-05-10 14:12:28 +02:00
|
|
|
Start ordered index scan in NDB
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2004-04-30 13:38:41 +02:00
|
|
|
int ha_ndbcluster::ordered_index_scan(const key_range *start_key,
|
2005-02-16 14:18:32 +01:00
|
|
|
const key_range *end_key,
|
|
|
|
bool sorted, bool descending, byte* buf)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-11-17 10:07:52 +01:00
|
|
|
int res;
|
2004-10-05 12:17:42 +02:00
|
|
|
bool restart;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-06-11 13:49:22 +02:00
|
|
|
NdbIndexScanOperation *op;
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::ordered_index_scan");
|
|
|
|
DBUG_PRINT("enter", ("index: %u, sorted: %d, descending: %d",
|
|
|
|
active_index, sorted, descending));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("Starting new ordered scan on %s", m_tabname));
|
2004-07-22 12:38:09 +02:00
|
|
|
|
2004-10-07 11:57:48 +02:00
|
|
|
// Check that sorted seems to be initialised
|
|
|
|
DBUG_ASSERT(sorted == 0 || sorted == 1);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-10-08 10:50:50 +02:00
|
|
|
if (m_active_cursor == 0)
|
2004-10-05 12:17:42 +02:00
|
|
|
{
|
2005-05-06 10:39:30 +02:00
|
|
|
restart= FALSE;
|
2004-10-05 12:17:42 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2006-06-12 09:37:19 +02:00
|
|
|
bool need_pk = (lm == NdbOperation::LM_Read);
|
2004-10-05 12:17:42 +02:00
|
|
|
if (!(op= trans->getNdbIndexScanOperation((NDBINDEX *)
|
2005-02-16 14:18:32 +01:00
|
|
|
m_index[active_index].index,
|
|
|
|
(const NDBTAB *) m_table)) ||
|
2006-06-12 10:22:32 +02:00
|
|
|
op->readTuples(lm, 0, parallelism, sorted, descending, false, need_pk))
|
2004-10-05 12:17:42 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-12-07 15:15:49 +01:00
|
|
|
m_active_cursor= op;
|
2004-10-05 12:17:42 +02:00
|
|
|
} else {
|
2005-05-06 10:39:30 +02:00
|
|
|
restart= TRUE;
|
2004-12-07 15:15:49 +01:00
|
|
|
op= (NdbIndexScanOperation*)m_active_cursor;
|
2004-10-05 12:17:42 +02:00
|
|
|
|
|
|
|
DBUG_ASSERT(op->getSorted() == sorted);
|
|
|
|
DBUG_ASSERT(op->getLockMode() ==
|
2005-02-16 14:18:32 +01:00
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type));
|
2005-07-04 02:42:33 +02:00
|
|
|
if (op->reset_bounds(m_force_send))
|
2004-10-05 12:17:42 +02:00
|
|
|
DBUG_RETURN(ndb_err(m_active_trans));
|
|
|
|
}
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2004-05-10 14:46:06 +02:00
|
|
|
{
|
2004-10-08 10:50:50 +02:00
|
|
|
const key_range *keys[2]= { start_key, end_key };
|
2004-11-17 10:07:52 +01:00
|
|
|
res= set_bounds(op, keys);
|
|
|
|
if (res)
|
|
|
|
DBUG_RETURN(res);
|
2004-05-10 14:46:06 +02:00
|
|
|
}
|
2005-01-07 15:33:24 +01:00
|
|
|
|
2007-04-23 11:25:33 +02:00
|
|
|
if (!restart && m_cond && m_cond->generate_scan_filter(op))
|
2005-01-07 15:33:24 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2004-11-17 10:07:52 +01:00
|
|
|
if (!restart && (res= define_read_attrs(buf, op)))
|
2004-05-10 14:46:06 +02:00
|
|
|
{
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-10-05 12:17:42 +02:00
|
|
|
}
|
2004-11-17 10:07:52 +01:00
|
|
|
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
|
|
|
|
DBUG_RETURN(next_result(buf));
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
/*
|
|
|
|
Unique index scan in NDB (full table scan with scan filter)
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::unique_index_scan(const KEY* key_info,
|
|
|
|
const byte *key,
|
|
|
|
uint key_len,
|
|
|
|
byte *buf)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
NdbScanOperation *op;
|
|
|
|
NdbTransaction *trans= m_active_trans;
|
|
|
|
|
|
|
|
DBUG_ENTER("unique_index_scan");
|
|
|
|
DBUG_PRINT("enter", ("Starting new scan on %s", m_tabname));
|
|
|
|
|
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
|
|
|
bool need_pk = (lm == NdbOperation::LM_Read);
|
|
|
|
if (!(op=trans->getNdbScanOperation((const NDBTAB *) m_table)) ||
|
|
|
|
op->readTuples(lm,
|
|
|
|
(need_pk)?NdbScanOperation::SF_KeyInfo:0,
|
|
|
|
parallelism))
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
m_active_cursor= op;
|
2007-04-23 11:25:33 +02:00
|
|
|
if (!m_cond)
|
|
|
|
m_cond= new ha_ndbcluster_cond;
|
|
|
|
if (!m_cond)
|
|
|
|
{
|
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
}
|
|
|
|
if (m_cond->generate_scan_filter_from_key(op, key_info, key, key_len, buf))
|
2006-11-07 16:38:37 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
if ((res= define_read_attrs(buf, op)))
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
|
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
DBUG_PRINT("exit", ("Scan started successfully"));
|
|
|
|
DBUG_RETURN(next_result(buf));
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
2004-05-10 14:12:28 +02:00
|
|
|
Start full table scan in NDB
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::full_table_scan(byte *buf)
|
|
|
|
{
|
2004-11-17 10:07:52 +01:00
|
|
|
int res;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbScanOperation *op;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("full_table_scan");
|
|
|
|
DBUG_PRINT("enter", ("Starting new scan on %s", m_tabname));
|
|
|
|
|
2004-09-28 19:35:07 +02:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2006-06-12 09:37:19 +02:00
|
|
|
bool need_pk = (lm == NdbOperation::LM_Read);
|
2004-09-28 19:35:07 +02:00
|
|
|
if (!(op=trans->getNdbScanOperation((const NDBTAB *) m_table)) ||
|
2006-06-12 09:37:19 +02:00
|
|
|
op->readTuples(lm,
|
|
|
|
(need_pk)?NdbScanOperation::SF_KeyInfo:0,
|
|
|
|
parallelism))
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2004-12-07 15:15:49 +01:00
|
|
|
m_active_cursor= op;
|
2007-04-23 11:25:33 +02:00
|
|
|
if (m_cond && m_cond->generate_scan_filter(op))
|
2005-01-07 15:33:24 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= define_read_attrs(buf, op)))
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(res);
|
|
|
|
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
|
|
|
DBUG_PRINT("exit", ("Scan started successfully"));
|
|
|
|
DBUG_RETURN(next_result(buf));
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
|
2007-11-05 20:18:22 +01:00
|
|
|
int
|
|
|
|
ha_ndbcluster::set_auto_inc(Field *field)
|
|
|
|
{
|
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
Uint64 next_val= (Uint64) field->val_int() + 1;
|
|
|
|
DBUG_ENTER("ha_ndbcluster::set_auto_inc");
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
char buff[22];
|
|
|
|
DBUG_PRINT("info",
|
|
|
|
("Trying to set next auto increment value to %s",
|
|
|
|
llstr(next_val, buff)));
|
|
|
|
#endif
|
|
|
|
if (ndb->setAutoIncrementValue((const NDBTAB *) m_table, next_val, TRUE)
|
|
|
|
== -1)
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Insert one record into NDB
|
|
|
|
*/
|
|
|
|
int ha_ndbcluster::write_row(byte *record)
|
|
|
|
{
|
2004-08-11 15:06:17 +02:00
|
|
|
bool has_auto_increment;
|
2004-04-15 09:14:14 +02:00
|
|
|
uint i;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbOperation *op;
|
|
|
|
int res;
|
2006-12-14 23:51:37 +01:00
|
|
|
THD *thd= table->in_use;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("write_row");
|
2004-09-15 14:44:21 +02:00
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
has_auto_increment= (table->next_number_field && record == table->record[0]);
|
|
|
|
if (table->s->primary_key != MAX_KEY)
|
2004-09-15 14:44:21 +02:00
|
|
|
{
|
2006-03-23 09:48:46 +01:00
|
|
|
/*
|
|
|
|
* Increase any auto_incremented primary key
|
|
|
|
*/
|
|
|
|
if (has_auto_increment)
|
|
|
|
{
|
2006-08-30 22:20:39 +02:00
|
|
|
int error;
|
2007-05-09 14:14:27 +02:00
|
|
|
|
2006-03-23 09:48:46 +01:00
|
|
|
m_skip_auto_increment= FALSE;
|
2006-08-30 22:20:39 +02:00
|
|
|
if ((error= update_auto_increment()))
|
|
|
|
DBUG_RETURN(error);
|
2006-03-23 09:48:46 +01:00
|
|
|
m_skip_auto_increment= !auto_increment_column_changed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If IGNORE the ignore constraint violations on primary and unique keys
|
|
|
|
*/
|
2006-03-24 10:45:00 +01:00
|
|
|
if (!m_use_write && m_ignore_dup_key)
|
2006-03-23 09:48:46 +01:00
|
|
|
{
|
2006-06-14 20:16:32 +02:00
|
|
|
/*
|
|
|
|
compare if expression with that in start_bulk_insert()
|
|
|
|
start_bulk_insert will set parameters to ensure that each
|
|
|
|
write_row is committed individually
|
|
|
|
*/
|
2007-10-02 13:36:13 +02:00
|
|
|
int peek_res= peek_indexed_rows(record, NDB_INSERT);
|
2004-11-18 12:11:56 +01:00
|
|
|
|
|
|
|
if (!peek_res)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(HA_ERR_FOUND_DUPP_KEY);
|
|
|
|
}
|
|
|
|
if (peek_res != HA_ERR_KEY_NOT_FOUND)
|
|
|
|
DBUG_RETURN(peek_res);
|
2004-09-15 14:44:21 +02:00
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(thd->status_var.ha_write_count, &LOCK_status);
|
2004-10-04 00:27:04 +02:00
|
|
|
if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT)
|
|
|
|
table->timestamp_field->set_time();
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-07 21:53:59 +02:00
|
|
|
if (!(op= trans->getNdbOperation((const NDBTAB *) m_table)))
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
|
|
|
res= (m_use_write) ? op->writeTuple() :op->insertTuple();
|
|
|
|
if (res != 0)
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key == MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
// Table has hidden primary key
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
2006-05-17 12:40:10 +02:00
|
|
|
Uint64 auto_value;
|
2005-04-15 16:14:08 +02:00
|
|
|
uint retries= NDB_AUTO_INCREMENT_RETRIES;
|
2007-05-14 12:15:27 +02:00
|
|
|
int retry_sleep= 30; /* 30 milliseconds, transaction */
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
if (ndb->getAutoIncrementValue((const NDBTAB *) m_table,
|
|
|
|
auto_value, 1) == -1)
|
|
|
|
{
|
|
|
|
if (--retries &&
|
2007-10-25 08:40:42 +02:00
|
|
|
ndb->getNdbError().status == NdbError::TemporaryError)
|
2007-05-14 12:15:27 +02:00
|
|
|
{
|
|
|
|
my_sleep(retry_sleep);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-01-06 12:00:13 +01:00
|
|
|
if (set_hidden_key(op, table->s->fields, (const byte*)&auto_value))
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
|
|
|
}
|
2007-05-09 14:14:27 +02:00
|
|
|
else
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-06-29 16:13:50 +02:00
|
|
|
if ((res= set_primary_key_from_record(op, record)))
|
2005-04-19 11:21:26 +02:00
|
|
|
return res;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set non-key attribute(s)
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
bool set_blob_value= FALSE;
|
2005-01-06 12:00:13 +01:00
|
|
|
for (i= 0; i < table->s->fields; i++)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
|
|
|
if (!(field->flags & PRI_KEY_FLAG) &&
|
2005-02-16 14:18:32 +01:00
|
|
|
set_ndb_value(op, field, i, &set_blob_value))
|
2004-08-13 13:51:18 +02:00
|
|
|
{
|
2004-11-03 15:53:26 +01:00
|
|
|
m_skip_auto_increment= TRUE;
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2004-08-13 13:51:18 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2005-03-15 15:03:25 +01:00
|
|
|
m_rows_changed++;
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Execute write operation
|
|
|
|
NOTE When doing inserts with many values in
|
|
|
|
each INSERT statement it should not be necessary
|
|
|
|
to NoCommit the transaction between each row.
|
|
|
|
Find out how this is detected!
|
|
|
|
*/
|
2004-11-03 15:53:26 +01:00
|
|
|
m_rows_inserted++;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
no_uncommitted_rows_update(1);
|
2004-11-03 15:53:26 +01:00
|
|
|
m_bulk_insert_not_flushed= TRUE;
|
2005-04-13 12:42:23 +02:00
|
|
|
if ((m_rows_to_insert == (ha_rows) 1) ||
|
2004-11-03 15:53:26 +01:00
|
|
|
((m_rows_inserted % m_bulk_insert_rows) == 0) ||
|
2005-04-21 13:09:19 +02:00
|
|
|
m_primary_key_update ||
|
2004-10-01 13:16:49 +02:00
|
|
|
set_blob_value)
|
2004-04-29 14:38:35 +02:00
|
|
|
{
|
|
|
|
// Send rows to NDB
|
|
|
|
DBUG_PRINT("info", ("Sending inserts to NDB, "\
|
2005-02-16 14:18:32 +01:00
|
|
|
"rows_inserted:%d, bulk_insert_rows: %d",
|
|
|
|
(int)m_rows_inserted, (int)m_bulk_insert_rows));
|
2004-09-03 17:11:09 +02:00
|
|
|
|
2004-11-03 15:53:26 +01:00
|
|
|
m_bulk_insert_not_flushed= FALSE;
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_transaction_on)
|
2004-08-13 13:51:18 +02:00
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0)
|
2004-09-07 14:30:11 +02:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
m_skip_auto_increment= TRUE;
|
|
|
|
no_uncommitted_rows_execute_failure();
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-09-07 14:30:11 +02:00
|
|
|
}
|
2004-09-03 17:11:09 +02:00
|
|
|
}
|
|
|
|
else
|
2004-08-13 13:51:18 +02:00
|
|
|
{
|
2004-09-28 21:11:50 +02:00
|
|
|
if (execute_commit(this,trans) != 0)
|
2004-09-03 17:11:09 +02:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
m_skip_auto_increment= TRUE;
|
|
|
|
no_uncommitted_rows_execute_failure();
|
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-09-03 17:11:09 +02:00
|
|
|
}
|
2005-07-04 02:42:33 +02:00
|
|
|
if (trans->restart() != 0)
|
2005-01-11 17:00:31 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_ASSERT(0);
|
|
|
|
DBUG_RETURN(-1);
|
2005-01-11 17:00:31 +01:00
|
|
|
}
|
2004-08-13 13:51:18 +02:00
|
|
|
}
|
2004-04-29 14:38:35 +02:00
|
|
|
}
|
2004-11-03 15:53:26 +01:00
|
|
|
if ((has_auto_increment) && (m_skip_auto_increment))
|
2004-08-10 14:38:24 +02:00
|
|
|
{
|
2007-11-05 20:18:22 +01:00
|
|
|
int ret_val;
|
|
|
|
if ((ret_val= set_auto_inc(table->next_number_field)))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(ret_val);
|
|
|
|
}
|
2004-07-29 11:35:38 +02:00
|
|
|
}
|
2004-11-03 15:53:26 +01:00
|
|
|
m_skip_auto_increment= TRUE;
|
2004-07-29 11:35:38 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare if a key in a row has changed */
|
|
|
|
|
|
|
|
int ha_ndbcluster::key_cmp(uint keynr, const byte * old_row,
|
2005-02-16 14:18:32 +01:00
|
|
|
const byte * new_row)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
KEY_PART_INFO *key_part=table->key_info[keynr].key_part;
|
|
|
|
KEY_PART_INFO *end=key_part+table->key_info[keynr].key_parts;
|
|
|
|
|
|
|
|
for (; key_part != end ; key_part++)
|
|
|
|
{
|
|
|
|
if (key_part->null_bit)
|
|
|
|
{
|
|
|
|
if ((old_row[key_part->null_offset] & key_part->null_bit) !=
|
2005-02-16 14:18:32 +01:00
|
|
|
(new_row[key_part->null_offset] & key_part->null_bit))
|
|
|
|
return 1;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-12-06 01:00:37 +01:00
|
|
|
if (key_part->key_part_flag & (HA_BLOB_PART | HA_VAR_LENGTH_PART))
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
if (key_part->field->cmp_binary((char*) (old_row + key_part->offset),
|
2005-02-16 14:18:32 +01:00
|
|
|
(char*) (new_row + key_part->offset),
|
|
|
|
(ulong) key_part->length))
|
|
|
|
return 1;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (memcmp(old_row+key_part->offset, new_row+key_part->offset,
|
2005-02-16 14:18:32 +01:00
|
|
|
key_part->length))
|
|
|
|
return 1;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Update one record in NDB using primary key
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::update_row(const byte *old_data, byte *new_data)
|
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-12-07 15:15:49 +01:00
|
|
|
NdbScanOperation* cursor= m_active_cursor;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbOperation *op;
|
|
|
|
uint i;
|
2007-11-05 20:18:22 +01:00
|
|
|
int auto_res;
|
2006-11-30 22:52:23 +01:00
|
|
|
bool pk_update= (table->s->primary_key != MAX_KEY &&
|
|
|
|
key_cmp(table->s->primary_key, old_data, new_data));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("update_row");
|
|
|
|
|
2006-11-30 15:02:33 +01:00
|
|
|
/*
|
2006-11-30 17:22:04 +01:00
|
|
|
* If IGNORE the ignore constraint violations on primary and unique keys,
|
|
|
|
* but check that it is not part of INSERT ... ON DUPLICATE KEY UPDATE
|
2006-11-30 15:02:33 +01:00
|
|
|
*/
|
2007-05-28 13:23:11 +02:00
|
|
|
if (m_ignore_dup_key && (thd->lex->sql_command == SQLCOM_UPDATE ||
|
|
|
|
thd->lex->sql_command == SQLCOM_UPDATE_MULTI))
|
2006-11-30 15:02:33 +01:00
|
|
|
{
|
2007-10-02 13:36:13 +02:00
|
|
|
NDB_WRITE_OP write_op= (pk_update) ? NDB_PK_UPDATE : NDB_UPDATE;
|
|
|
|
int peek_res= peek_indexed_rows(new_data, write_op);
|
2006-11-30 15:02:33 +01:00
|
|
|
|
|
|
|
if (!peek_res)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(HA_ERR_FOUND_DUPP_KEY);
|
|
|
|
}
|
|
|
|
if (peek_res != HA_ERR_KEY_NOT_FOUND)
|
|
|
|
DBUG_RETURN(peek_res);
|
|
|
|
}
|
|
|
|
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(thd->status_var.ha_update_count, &LOCK_status);
|
2004-10-04 00:27:04 +02:00
|
|
|
if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE)
|
2005-04-01 17:59:46 +02:00
|
|
|
{
|
2004-10-04 00:27:04 +02:00
|
|
|
table->timestamp_field->set_time();
|
2005-04-01 17:59:46 +02:00
|
|
|
// Set query_id so that field is really updated
|
|
|
|
table->timestamp_field->query_id= thd->query_id;
|
|
|
|
}
|
2004-10-04 00:27:04 +02:00
|
|
|
|
2004-07-22 18:35:51 +02:00
|
|
|
/* Check for update of primary key for special handling */
|
2006-11-30 22:52:23 +01:00
|
|
|
if (pk_update)
|
2004-07-22 18:35:51 +02:00
|
|
|
{
|
2005-07-06 11:23:36 +02:00
|
|
|
int read_res, insert_res, delete_res, undo_res;
|
2004-07-22 18:35:51 +02:00
|
|
|
|
2005-04-19 11:21:26 +02:00
|
|
|
DBUG_PRINT("info", ("primary key update, doing pk read+delete+insert"));
|
2004-07-22 18:35:51 +02:00
|
|
|
// Get all old fields, since we optimize away fields not in query
|
2004-07-23 15:46:56 +02:00
|
|
|
read_res= complemented_pk_read(old_data, new_data);
|
2004-07-22 18:35:51 +02:00
|
|
|
if (read_res)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("pk read failed"));
|
|
|
|
DBUG_RETURN(read_res);
|
|
|
|
}
|
2004-07-23 15:46:56 +02:00
|
|
|
// Delete old row
|
2004-11-03 15:53:26 +01:00
|
|
|
m_primary_key_update= TRUE;
|
2004-07-23 15:46:56 +02:00
|
|
|
delete_res= delete_row(old_data);
|
2004-11-03 15:53:26 +01:00
|
|
|
m_primary_key_update= FALSE;
|
2004-07-23 15:46:56 +02:00
|
|
|
if (delete_res)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("delete failed"));
|
2005-04-19 11:21:26 +02:00
|
|
|
DBUG_RETURN(delete_res);
|
2004-07-23 15:46:56 +02:00
|
|
|
}
|
2005-04-19 11:21:26 +02:00
|
|
|
// Insert new row
|
|
|
|
DBUG_PRINT("info", ("delete succeded"));
|
2005-04-21 13:09:19 +02:00
|
|
|
m_primary_key_update= TRUE;
|
2007-11-05 20:18:22 +01:00
|
|
|
/*
|
|
|
|
If we are updating a primary key with auto_increment
|
|
|
|
then we need to update the auto_increment counter
|
|
|
|
*/
|
|
|
|
if (table->found_next_number_field &&
|
|
|
|
table->found_next_number_field->query_id == thd->query_id &&
|
|
|
|
(auto_res= set_auto_inc(table->found_next_number_field)))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(auto_res);
|
|
|
|
}
|
2005-04-19 11:21:26 +02:00
|
|
|
insert_res= write_row(new_data);
|
2005-04-21 13:09:19 +02:00
|
|
|
m_primary_key_update= FALSE;
|
2005-04-19 11:21:26 +02:00
|
|
|
if (insert_res)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("insert failed"));
|
|
|
|
if (trans->commitStatus() == NdbConnection::Started)
|
|
|
|
{
|
2005-07-06 11:23:36 +02:00
|
|
|
// Undo delete_row(old_data)
|
2005-04-19 11:21:26 +02:00
|
|
|
m_primary_key_update= TRUE;
|
2005-07-06 11:23:36 +02:00
|
|
|
undo_res= write_row((byte *)old_data);
|
|
|
|
if (undo_res)
|
|
|
|
push_warning(current_thd,
|
|
|
|
MYSQL_ERROR::WARN_LEVEL_WARN,
|
|
|
|
undo_res,
|
|
|
|
"NDB failed undoing delete at primary key update");
|
2005-04-19 11:21:26 +02:00
|
|
|
m_primary_key_update= FALSE;
|
|
|
|
}
|
|
|
|
DBUG_RETURN(insert_res);
|
|
|
|
}
|
|
|
|
DBUG_PRINT("info", ("delete+insert succeeded"));
|
2004-07-23 15:46:56 +02:00
|
|
|
DBUG_RETURN(0);
|
2004-07-22 18:35:51 +02:00
|
|
|
}
|
2007-11-05 20:18:22 +01:00
|
|
|
/*
|
|
|
|
If we are updating a unique key with auto_increment
|
|
|
|
then we need to update the auto_increment counter
|
|
|
|
*/
|
|
|
|
if (table->found_next_number_field &&
|
|
|
|
table->found_next_number_field->query_id == thd->query_id &&
|
|
|
|
(auto_res= set_auto_inc(table->found_next_number_field)))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(auto_res);
|
|
|
|
}
|
2004-05-10 14:12:28 +02:00
|
|
|
if (cursor)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-05-10 14:12:28 +02:00
|
|
|
/*
|
|
|
|
We are scanning records and want to update the record
|
|
|
|
that was just found, call updateTuple on the cursor
|
|
|
|
to take over the lock to a new update operation
|
|
|
|
And thus setting the primary key of the record from
|
|
|
|
the active record in cursor
|
|
|
|
*/
|
|
|
|
DBUG_PRINT("info", ("Calling updateTuple on cursor"));
|
2004-12-07 15:15:49 +01:00
|
|
|
if (!(op= cursor->updateCurrentTuple()))
|
2004-05-10 14:12:28 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2006-06-12 09:37:19 +02:00
|
|
|
m_lock_tuple= false;
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending++;
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
if (uses_blob_value(FALSE))
|
2004-11-03 15:53:26 +01:00
|
|
|
m_blobs_pending= TRUE;
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-09-07 21:53:59 +02:00
|
|
|
if (!(op= trans->getNdbOperation((const NDBTAB *) m_table)) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
op->updateTuple() != 0)
|
2004-05-10 14:12:28 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key == MAX_KEY)
|
2004-05-10 14:12:28 +02:00
|
|
|
{
|
|
|
|
// This table has no primary key, use "hidden" primary key
|
|
|
|
DBUG_PRINT("info", ("Using hidden key"));
|
|
|
|
|
|
|
|
// Require that the PK for this record has previously been
|
2006-02-10 17:40:22 +01:00
|
|
|
// read into m_ref
|
|
|
|
DBUG_DUMP("key", m_ref, NDB_HIDDEN_PRIMARY_KEY_LENGTH);
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2006-02-10 17:59:53 +01:00
|
|
|
if (set_hidden_key(op, table->s->fields, m_ref))
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int res;
|
2005-06-29 16:13:50 +02:00
|
|
|
if ((res= set_primary_key_from_record(op, old_data)))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2005-03-15 15:03:25 +01:00
|
|
|
m_rows_changed++;
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Set non-key attribute(s)
|
2005-01-06 12:00:13 +01:00
|
|
|
for (i= 0; i < table->s->fields; i++)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
Field *field= table->field[i];
|
2004-11-17 09:15:53 +01:00
|
|
|
if (((thd->query_id == field->query_id) || m_retrieve_all_fields) &&
|
2004-04-15 09:14:14 +02:00
|
|
|
(!(field->flags & PRI_KEY_FLAG)) &&
|
2005-02-16 14:18:32 +01:00
|
|
|
set_ndb_value(op, field, i))
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(op->getNdbError());
|
|
|
|
}
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2007-04-04 12:50:39 +02:00
|
|
|
/*
|
|
|
|
Execute update operation if we are not doing a scan for update
|
|
|
|
and there exist UPDATE AFTER triggers
|
|
|
|
*/
|
|
|
|
|
|
|
|
if ((!cursor || m_update_cannot_batch) &&
|
|
|
|
execute_no_commit(this,trans,false) != 0) {
|
2004-09-14 17:51:57 +02:00
|
|
|
no_uncommitted_rows_execute_failure();
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-09-14 17:51:57 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Delete one record from NDB, using primary key
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::delete_row(const byte *record)
|
|
|
|
{
|
2004-09-14 21:02:23 +02:00
|
|
|
THD *thd= current_thd;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-12-07 15:15:49 +01:00
|
|
|
NdbScanOperation* cursor= m_active_cursor;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbOperation *op;
|
|
|
|
DBUG_ENTER("delete_row");
|
|
|
|
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(thd->status_var.ha_delete_count,&LOCK_status);
|
2005-03-31 14:33:50 +02:00
|
|
|
m_rows_changed++;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-05-10 14:12:28 +02:00
|
|
|
if (cursor)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-05-10 14:12:28 +02:00
|
|
|
/*
|
2004-07-23 10:33:51 +02:00
|
|
|
We are scanning records and want to delete the record
|
2004-05-10 14:12:28 +02:00
|
|
|
that was just found, call deleteTuple on the cursor
|
2004-07-23 13:13:50 +02:00
|
|
|
to take over the lock to a new delete operation
|
2004-05-10 14:12:28 +02:00
|
|
|
And thus setting the primary key of the record from
|
|
|
|
the active record in cursor
|
|
|
|
*/
|
|
|
|
DBUG_PRINT("info", ("Calling deleteTuple on cursor"));
|
2004-12-07 15:15:49 +01:00
|
|
|
if (cursor->deleteCurrentTuple() != 0)
|
2004-05-10 14:12:28 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
2006-06-12 09:37:19 +02:00
|
|
|
m_lock_tuple= false;
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending++;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
no_uncommitted_rows_update(-1);
|
|
|
|
|
2007-04-04 12:50:39 +02:00
|
|
|
if (!(m_primary_key_update || m_delete_cannot_batch))
|
2005-04-19 11:21:26 +02:00
|
|
|
// If deleting from cursor, NoCommit will be handled in next_result
|
|
|
|
DBUG_RETURN(0);
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
else
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2004-09-07 21:53:59 +02:00
|
|
|
if (!(op=trans->getNdbOperation((const NDBTAB *) m_table)) ||
|
2005-02-16 14:18:32 +01:00
|
|
|
op->deleteTuple() != 0)
|
2004-05-10 14:12:28 +02:00
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
no_uncommitted_rows_update(-1);
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key == MAX_KEY)
|
2004-05-10 14:12:28 +02:00
|
|
|
{
|
|
|
|
// This table has no primary key, use "hidden" primary key
|
|
|
|
DBUG_PRINT("info", ("Using hidden key"));
|
|
|
|
|
2006-02-10 17:59:53 +01:00
|
|
|
if (set_hidden_key(op, table->s->fields, m_ref))
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(op->getNdbError());
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int res;
|
2005-06-29 16:13:50 +02:00
|
|
|
if ((res= set_primary_key_from_record(op, record)))
|
|
|
|
return res;
|
2004-05-10 14:12:28 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Execute delete operation
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0) {
|
2004-09-14 17:51:57 +02:00
|
|
|
no_uncommitted_rows_execute_failure();
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-09-14 17:51:57 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Unpack a record read from NDB
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
unpack_record()
|
2005-02-16 14:18:32 +01:00
|
|
|
buf Buffer to store read row
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
NOTE
|
|
|
|
The data for each row is read directly into the
|
|
|
|
destination buffer. This function is primarily
|
|
|
|
called in order to check if any fields should be
|
|
|
|
set to null.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::unpack_record(byte* buf)
|
|
|
|
{
|
|
|
|
uint row_offset= (uint) (buf - table->record[0]);
|
|
|
|
Field **field, **end;
|
2004-07-22 12:38:09 +02:00
|
|
|
NdbValue *value= m_value;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("unpack_record");
|
2004-12-01 12:43:30 +01:00
|
|
|
|
2005-01-07 07:15:08 +01:00
|
|
|
end= table->field + table->s->fields;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
// Set null flag(s)
|
2005-01-06 12:00:13 +01:00
|
|
|
bzero(buf, table->s->null_bytes);
|
2005-01-07 07:15:08 +01:00
|
|
|
for (field= table->field;
|
2004-04-15 09:14:14 +02:00
|
|
|
field < end;
|
|
|
|
field++, value++)
|
|
|
|
{
|
2004-07-22 12:38:09 +02:00
|
|
|
if ((*value).ptr)
|
|
|
|
{
|
|
|
|
if (! ((*field)->flags & BLOB_FLAG))
|
|
|
|
{
|
|
|
|
if ((*value).rec->isNULL())
|
|
|
|
(*field)->set_null(row_offset);
|
2004-12-23 15:28:41 +01:00
|
|
|
else if ((*field)->type() == MYSQL_TYPE_BIT)
|
|
|
|
{
|
|
|
|
uint pack_len= (*field)->pack_length();
|
|
|
|
if (pack_len < 5)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("bit field H'%.8X",
|
2005-02-16 14:18:32 +01:00
|
|
|
(*value).rec->u_32_value()));
|
2004-12-23 15:28:41 +01:00
|
|
|
((Field_bit *) *field)->store((longlong)
|
Added option --valgrind-mysqltest to mysql-test-run
Added flag to Field::store(longlong) to specify if value is unsigned.
This fixes bug #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
Fixed warning from valgrind in CREATE ... SELECT
Fixed double free of mysql.options if reconnect failed
mysql-test/mysql-test-run.sh:
Added option --valgrind-mysqltest to allow one to run mysqltest with valgrind
mysql-test/r/bigint.result:
Update results after fix for Field::store(longlong)
mysql-test/r/range.result:
Update results after fix for Field::store(longlong)
mysql-test/r/strict.result:
Update results after fix for Field::store(longlong)
(This fixes some wrong results when storing things into bigint columns)
mysql-test/r/type_ranges.result:
Update results after fix for Field::store(longlong)
mysql-test/t/bigint.test:
Added testing for #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
mysql-test/t/innodb.test:
Removed comments affected by this bug fix
mysql-test/t/mysqldump.test:
Fixed result to not depend on existing config files
mysql-test/t/range.test:
0xff numbers are now unsigned
mysql-test/t/strict.test:
Added errors for things that previously (wrongly) succeeded
sql-common/client.c:
Fixed double free of mysql.options if reconnect failed
sql/field.cc:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field.h:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field_conv.cc:
Fixed calls to Field::store(longlong,flag)
sql/ha_ndbcluster.cc:
Fixed calls to Field::store(longlong,flag)
sql/handler.cc:
Fixed calls to Field::store(longlong,flag)
sql/item.cc:
Fixed calls to Field::store(longlong,flag)
sql/item_sum.cc:
Fixed calls to Field::store(longlong,flag)
sql/sp.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_acl.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_help.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_show.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_table.cc:
Fixed varning from valgrind
sql/sql_udf.cc:
Fixed calls to Field::store(longlong,flag)
sql/tztime.cc:
Fixed calls to Field::store(longlong,flag)
sql/unireg.cc:
Fixed calls to Field::store(longlong,flag)
2005-09-14 00:41:44 +02:00
|
|
|
(*value).rec->u_32_value(),
|
|
|
|
FALSE);
|
2004-12-23 15:28:41 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("bit field H'%.8X%.8X",
|
|
|
|
*(Uint32 *)(*value).rec->aRef(),
|
|
|
|
*((Uint32 *)(*value).rec->aRef()+1)));
|
2007-03-26 14:48:52 +02:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
/* lsw is stored first */
|
|
|
|
Uint32 *buf= (Uint32 *)(*value).rec->aRef();
|
2007-03-26 17:57:00 +02:00
|
|
|
((Field_bit *) *field)->store((((longlong)*buf)
|
2007-04-18 16:41:38 +02:00
|
|
|
& 0x000000000FFFFFFFFLL)
|
2007-03-26 17:57:00 +02:00
|
|
|
|
|
|
|
|
((((longlong)*(buf+1)) << 32)
|
2007-04-18 16:41:38 +02:00
|
|
|
& 0xFFFFFFFF00000000LL),
|
2007-03-26 14:48:52 +02:00
|
|
|
TRUE);
|
|
|
|
#else
|
2004-12-23 15:28:41 +01:00
|
|
|
((Field_bit *) *field)->store((longlong)
|
Added option --valgrind-mysqltest to mysql-test-run
Added flag to Field::store(longlong) to specify if value is unsigned.
This fixes bug #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
Fixed warning from valgrind in CREATE ... SELECT
Fixed double free of mysql.options if reconnect failed
mysql-test/mysql-test-run.sh:
Added option --valgrind-mysqltest to allow one to run mysqltest with valgrind
mysql-test/r/bigint.result:
Update results after fix for Field::store(longlong)
mysql-test/r/range.result:
Update results after fix for Field::store(longlong)
mysql-test/r/strict.result:
Update results after fix for Field::store(longlong)
(This fixes some wrong results when storing things into bigint columns)
mysql-test/r/type_ranges.result:
Update results after fix for Field::store(longlong)
mysql-test/t/bigint.test:
Added testing for #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
mysql-test/t/innodb.test:
Removed comments affected by this bug fix
mysql-test/t/mysqldump.test:
Fixed result to not depend on existing config files
mysql-test/t/range.test:
0xff numbers are now unsigned
mysql-test/t/strict.test:
Added errors for things that previously (wrongly) succeeded
sql-common/client.c:
Fixed double free of mysql.options if reconnect failed
sql/field.cc:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field.h:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field_conv.cc:
Fixed calls to Field::store(longlong,flag)
sql/ha_ndbcluster.cc:
Fixed calls to Field::store(longlong,flag)
sql/handler.cc:
Fixed calls to Field::store(longlong,flag)
sql/item.cc:
Fixed calls to Field::store(longlong,flag)
sql/item_sum.cc:
Fixed calls to Field::store(longlong,flag)
sql/sp.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_acl.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_help.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_show.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_table.cc:
Fixed varning from valgrind
sql/sql_udf.cc:
Fixed calls to Field::store(longlong,flag)
sql/tztime.cc:
Fixed calls to Field::store(longlong,flag)
sql/unireg.cc:
Fixed calls to Field::store(longlong,flag)
2005-09-14 00:41:44 +02:00
|
|
|
(*value).rec->u_64_value(), TRUE);
|
2007-03-26 14:48:52 +02:00
|
|
|
#endif
|
Added option --valgrind-mysqltest to mysql-test-run
Added flag to Field::store(longlong) to specify if value is unsigned.
This fixes bug #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
Fixed warning from valgrind in CREATE ... SELECT
Fixed double free of mysql.options if reconnect failed
mysql-test/mysql-test-run.sh:
Added option --valgrind-mysqltest to allow one to run mysqltest with valgrind
mysql-test/r/bigint.result:
Update results after fix for Field::store(longlong)
mysql-test/r/range.result:
Update results after fix for Field::store(longlong)
mysql-test/r/strict.result:
Update results after fix for Field::store(longlong)
(This fixes some wrong results when storing things into bigint columns)
mysql-test/r/type_ranges.result:
Update results after fix for Field::store(longlong)
mysql-test/t/bigint.test:
Added testing for #12750: Incorrect storage of 9999999999999999999 in DECIMAL(19, 0)
mysql-test/t/innodb.test:
Removed comments affected by this bug fix
mysql-test/t/mysqldump.test:
Fixed result to not depend on existing config files
mysql-test/t/range.test:
0xff numbers are now unsigned
mysql-test/t/strict.test:
Added errors for things that previously (wrongly) succeeded
sql-common/client.c:
Fixed double free of mysql.options if reconnect failed
sql/field.cc:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field.h:
Added flag to Field::store(longlong) to specify if value is unsigned
sql/field_conv.cc:
Fixed calls to Field::store(longlong,flag)
sql/ha_ndbcluster.cc:
Fixed calls to Field::store(longlong,flag)
sql/handler.cc:
Fixed calls to Field::store(longlong,flag)
sql/item.cc:
Fixed calls to Field::store(longlong,flag)
sql/item_sum.cc:
Fixed calls to Field::store(longlong,flag)
sql/sp.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_acl.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_help.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_show.cc:
Fixed calls to Field::store(longlong,flag)
sql/sql_table.cc:
Fixed varning from valgrind
sql/sql_udf.cc:
Fixed calls to Field::store(longlong,flag)
sql/tztime.cc:
Fixed calls to Field::store(longlong,flag)
sql/unireg.cc:
Fixed calls to Field::store(longlong,flag)
2005-09-14 00:41:44 +02:00
|
|
|
}
|
2004-12-23 15:28:41 +01:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NdbBlob* ndb_blob= (*value).blob;
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
bool isNull= TRUE;
|
2005-01-11 17:00:31 +01:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
int ret=
|
|
|
|
#endif
|
2005-02-16 14:18:32 +01:00
|
|
|
ndb_blob->getNull(isNull);
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_ASSERT(ret == 0);
|
|
|
|
if (isNull)
|
2005-02-16 14:18:32 +01:00
|
|
|
(*field)->set_null(row_offset);
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-12-01 12:43:30 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
// Read and print all values that was fetched
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key == MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
// Table with hidden primary key
|
2005-01-06 12:00:13 +01:00
|
|
|
int hidden_no= table->s->fields;
|
2006-06-30 03:10:27 +02:00
|
|
|
char buff[22];
|
2004-09-07 21:53:59 +02:00
|
|
|
const NDBTAB *tab= (const NDBTAB *) m_table;
|
2004-04-15 09:14:14 +02:00
|
|
|
const NDBCOL *hidden_col= tab->getColumn(hidden_no);
|
2004-12-01 12:43:30 +01:00
|
|
|
const NdbRecAttr* rec= m_value[hidden_no].rec;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ASSERT(rec);
|
2006-06-30 03:10:27 +02:00
|
|
|
DBUG_PRINT("hidden", ("%d: %s \"%s\"", hidden_no,
|
2006-06-30 18:15:18 +02:00
|
|
|
hidden_col->getName(),
|
2006-06-30 03:10:27 +02:00
|
|
|
llstr(rec->u_64_value(), buff)));
|
2006-06-30 18:15:18 +02:00
|
|
|
}
|
2007-04-19 15:48:17 +02:00
|
|
|
//print_results();
|
2004-04-15 09:14:14 +02:00
|
|
|
#endif
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Utility function to print/dump the fetched field
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::print_results()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("print_results");
|
|
|
|
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
if (!_db_on_)
|
|
|
|
DBUG_VOID_RETURN;
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2005-01-11 17:00:31 +01:00
|
|
|
char buf_type[MAX_FIELD_WIDTH], buf_val[MAX_FIELD_WIDTH];
|
2005-02-11 22:33:52 +01:00
|
|
|
String type(buf_type, sizeof(buf_type), &my_charset_bin);
|
2005-01-11 17:00:31 +01:00
|
|
|
String val(buf_val, sizeof(buf_val), &my_charset_bin);
|
2005-02-17 14:22:44 +01:00
|
|
|
for (uint f= 0; f < table->s->fields; f++)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-02-11 22:33:52 +01:00
|
|
|
/* Use DBUG_PRINT since DBUG_FILE cannot be filtered out */
|
2004-12-17 09:01:54 +01:00
|
|
|
char buf[2000];
|
2004-04-15 09:14:14 +02:00
|
|
|
Field *field;
|
2004-12-20 15:12:42 +01:00
|
|
|
void* ptr;
|
2004-07-22 12:38:09 +02:00
|
|
|
NdbValue value;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-01-11 17:00:31 +01:00
|
|
|
buf[0]= 0;
|
2005-02-11 22:33:52 +01:00
|
|
|
field= table->field[f];
|
2004-07-22 12:38:09 +02:00
|
|
|
if (!(value= m_value[f]).ptr)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
strmov(buf, "not read");
|
2004-12-17 09:01:54 +01:00
|
|
|
goto print_value;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2005-01-11 17:00:31 +01:00
|
|
|
|
2004-12-20 15:12:42 +01:00
|
|
|
ptr= field->ptr;
|
2004-07-22 12:38:09 +02:00
|
|
|
|
|
|
|
if (! (field->flags & BLOB_FLAG))
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-07-22 12:38:09 +02:00
|
|
|
if (value.rec->isNULL())
|
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
strmov(buf, "NULL");
|
2004-12-17 09:01:54 +01:00
|
|
|
goto print_value;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
2005-01-11 17:00:31 +01:00
|
|
|
type.length(0);
|
|
|
|
val.length(0);
|
|
|
|
field->sql_type(type);
|
|
|
|
field->val_str(&val);
|
|
|
|
my_snprintf(buf, sizeof(buf), "%s %s", type.c_ptr(), val.c_ptr());
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-17 14:22:44 +01:00
|
|
|
NdbBlob *ndb_blob= value.blob;
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
bool isNull= TRUE;
|
2004-07-22 12:38:09 +02:00
|
|
|
ndb_blob->getNull(isNull);
|
2005-07-04 02:42:33 +02:00
|
|
|
if (isNull)
|
|
|
|
strmov(buf, "NULL");
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2004-12-17 09:01:54 +01:00
|
|
|
print_value:
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_PRINT("value", ("%u,%s: %s", f, field->field_name, buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_init(uint index)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_init");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("index: %u", index));
|
2006-06-12 09:37:19 +02:00
|
|
|
/*
|
|
|
|
Locks are are explicitly released in scan
|
|
|
|
unless m_lock.type == TL_READ_HIGH_PRIORITY
|
|
|
|
and no sub-sequent call to unlock_row()
|
|
|
|
*/
|
|
|
|
m_lock_tuple= false;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(handler::index_init(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_end()
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_end");
|
2004-05-11 13:59:22 +02:00
|
|
|
DBUG_RETURN(close_scan());
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2004-11-30 07:41:26 +01:00
|
|
|
/**
|
2007-05-02 10:02:27 +02:00
|
|
|
* Check if key contains nullable columns
|
2004-11-30 07:41:26 +01:00
|
|
|
*/
|
|
|
|
static
|
|
|
|
int
|
|
|
|
check_null_in_key(const KEY* key_info, const byte *key, uint key_len)
|
|
|
|
{
|
|
|
|
KEY_PART_INFO *curr_part, *end_part;
|
2004-12-07 15:15:49 +01:00
|
|
|
const byte* end_ptr= key + key_len;
|
2004-11-30 07:41:26 +01:00
|
|
|
curr_part= key_info->key_part;
|
|
|
|
end_part= curr_part + key_info->key_parts;
|
|
|
|
|
|
|
|
|
|
|
|
for (; curr_part != end_part && key < end_ptr; curr_part++)
|
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
if (curr_part->null_bit && *key)
|
2004-11-30 07:41:26 +01:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
key += curr_part->store_length;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
int ha_ndbcluster::index_read(byte *buf,
|
2005-02-16 14:18:32 +01:00
|
|
|
const byte *key, uint key_len,
|
|
|
|
enum ha_rkey_function find_flag)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_read");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("active_index: %u, key_len: %u, find_flag: %d",
|
|
|
|
active_index, key_len, find_flag));
|
|
|
|
|
2004-10-06 08:48:26 +02:00
|
|
|
int error;
|
2004-12-07 15:15:49 +01:00
|
|
|
ndb_index_type type= get_index_type(active_index);
|
|
|
|
const KEY* key_info= table->key_info+active_index;
|
2004-10-06 08:48:26 +02:00
|
|
|
switch (type){
|
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
|
|
|
case PRIMARY_KEY_INDEX:
|
|
|
|
if (find_flag == HA_READ_KEY_EXACT && key_info->key_length == key_len)
|
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
if (m_active_cursor && (error= close_scan()))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(error);
|
2004-10-06 08:48:26 +02:00
|
|
|
DBUG_RETURN(pk_read(key, key_len, buf));
|
|
|
|
}
|
|
|
|
else if (type == PRIMARY_KEY_INDEX)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case UNIQUE_ORDERED_INDEX:
|
|
|
|
case UNIQUE_INDEX:
|
2004-11-30 07:41:26 +01:00
|
|
|
if (find_flag == HA_READ_KEY_EXACT && key_info->key_length == key_len &&
|
2005-02-16 14:18:32 +01:00
|
|
|
!check_null_in_key(key_info, key, key_len))
|
2004-10-06 08:48:26 +02:00
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
if (m_active_cursor && (error= close_scan()))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(error);
|
2004-10-06 08:48:26 +02:00
|
|
|
DBUG_RETURN(unique_index_read(key, key_len, buf));
|
|
|
|
}
|
|
|
|
else if (type == UNIQUE_INDEX)
|
|
|
|
{
|
2006-11-07 16:38:37 +01:00
|
|
|
DBUG_RETURN(unique_index_scan(key_info, key, key_len, buf));
|
2004-10-06 08:48:26 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ORDERED_INDEX:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
case UNDEFINED_INDEX:
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
DBUG_ASSERT(FALSE);
|
2004-10-07 14:19:46 +02:00
|
|
|
DBUG_RETURN(1);
|
2004-10-06 08:48:26 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-05-17 09:51:02 +02:00
|
|
|
key_range start_key;
|
2004-12-07 15:15:49 +01:00
|
|
|
start_key.key= key;
|
|
|
|
start_key.length= key_len;
|
|
|
|
start_key.flag= find_flag;
|
2004-12-20 15:12:42 +01:00
|
|
|
bool descending= FALSE;
|
|
|
|
switch (find_flag) {
|
|
|
|
case HA_READ_KEY_OR_PREV:
|
|
|
|
case HA_READ_BEFORE_KEY:
|
|
|
|
case HA_READ_PREFIX_LAST:
|
|
|
|
case HA_READ_PREFIX_LAST_OR_PREV:
|
|
|
|
descending= TRUE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
error= ordered_index_scan(&start_key, 0, TRUE, descending, buf);
|
2004-10-06 08:48:26 +02:00
|
|
|
DBUG_RETURN(error == HA_ERR_END_OF_FILE ? HA_ERR_KEY_NOT_FOUND : error);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_read_idx(byte *buf, uint index_no,
|
2005-02-16 14:18:32 +01:00
|
|
|
const byte *key, uint key_len,
|
|
|
|
enum ha_rkey_function find_flag)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_key_count, &LOCK_status);
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_read_idx");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("index_no: %u, key_len: %u", index_no, key_len));
|
|
|
|
index_init(index_no);
|
|
|
|
DBUG_RETURN(index_read(buf, key, key_len, find_flag));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_next(byte *buf)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_next");
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_next_count,
|
2005-02-16 14:18:32 +01:00
|
|
|
&LOCK_status);
|
2004-05-10 14:12:28 +02:00
|
|
|
DBUG_RETURN(next_result(buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_prev(byte *buf)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_prev");
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_prev_count,
|
2005-02-16 14:18:32 +01:00
|
|
|
&LOCK_status);
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_RETURN(next_result(buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_first(byte *buf)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_first");
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_first_count,
|
2005-02-16 14:18:32 +01:00
|
|
|
&LOCK_status);
|
2004-09-03 13:55:40 +02:00
|
|
|
// Start the ordered index scan and fetch the first row
|
|
|
|
|
|
|
|
// Only HA_READ_ORDER indexes get called by index_first
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_RETURN(ordered_index_scan(0, 0, TRUE, FALSE, buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::index_last(byte *buf)
|
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::index_last");
|
2004-09-13 15:48:01 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_last_count,&LOCK_status);
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_RETURN(ordered_index_scan(0, 0, TRUE, TRUE, buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2004-12-20 15:12:42 +01:00
|
|
|
int ha_ndbcluster::index_read_last(byte * buf, const byte * key, uint key_len)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("ha_ndbcluster::index_read_last");
|
|
|
|
DBUG_RETURN(index_read(buf, key, key_len, HA_READ_PREFIX_LAST));
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-07 16:22:42 +02:00
|
|
|
inline
|
|
|
|
int ha_ndbcluster::read_range_first_to_buf(const key_range *start_key,
|
2005-02-16 14:18:32 +01:00
|
|
|
const key_range *end_key,
|
|
|
|
bool eq_r, bool sorted,
|
|
|
|
byte* buf)
|
2004-04-30 13:38:41 +02:00
|
|
|
{
|
2006-11-07 16:38:37 +01:00
|
|
|
ndb_index_type type= get_index_type(active_index);
|
|
|
|
KEY* key_info;
|
2004-09-07 16:22:42 +02:00
|
|
|
int error= 1;
|
|
|
|
DBUG_ENTER("ha_ndbcluster::read_range_first_to_buf");
|
2005-01-18 16:31:05 +01:00
|
|
|
DBUG_PRINT("info", ("eq_r: %d, sorted: %d", eq_r, sorted));
|
2004-04-30 13:38:41 +02:00
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
switch (type){
|
2004-05-24 12:35:39 +02:00
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
2004-04-30 13:38:41 +02:00
|
|
|
case PRIMARY_KEY_INDEX:
|
2004-05-10 14:46:06 +02:00
|
|
|
key_info= table->key_info + active_index;
|
|
|
|
if (start_key &&
|
2005-02-16 14:18:32 +01:00
|
|
|
start_key->length == key_info->key_length &&
|
|
|
|
start_key->flag == HA_READ_KEY_EXACT)
|
2004-05-17 09:51:02 +02:00
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
if (m_active_cursor && (error= close_scan()))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(error);
|
2004-05-17 09:51:02 +02:00
|
|
|
error= pk_read(start_key->key, start_key->length, buf);
|
|
|
|
DBUG_RETURN(error == HA_ERR_KEY_NOT_FOUND ? HA_ERR_END_OF_FILE : error);
|
|
|
|
}
|
2004-04-30 13:38:41 +02:00
|
|
|
break;
|
2004-05-24 12:35:39 +02:00
|
|
|
case UNIQUE_ORDERED_INDEX:
|
2004-04-30 13:38:41 +02:00
|
|
|
case UNIQUE_INDEX:
|
2004-05-10 14:46:06 +02:00
|
|
|
key_info= table->key_info + active_index;
|
2004-11-30 07:41:26 +01:00
|
|
|
if (start_key && start_key->length == key_info->key_length &&
|
2005-02-16 14:18:32 +01:00
|
|
|
start_key->flag == HA_READ_KEY_EXACT &&
|
|
|
|
!check_null_in_key(key_info, start_key->key, start_key->length))
|
2004-05-17 09:51:02 +02:00
|
|
|
{
|
2005-07-04 02:42:33 +02:00
|
|
|
if (m_active_cursor && (error= close_scan()))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(error);
|
2004-05-17 09:51:02 +02:00
|
|
|
error= unique_index_read(start_key->key, start_key->length, buf);
|
|
|
|
DBUG_RETURN(error == HA_ERR_KEY_NOT_FOUND ? HA_ERR_END_OF_FILE : error);
|
|
|
|
}
|
2006-11-07 16:38:37 +01:00
|
|
|
else if (type == UNIQUE_INDEX)
|
2006-12-14 16:27:14 +01:00
|
|
|
DBUG_RETURN(unique_index_scan(key_info,
|
|
|
|
start_key->key,
|
|
|
|
start_key->length,
|
|
|
|
buf));
|
2004-04-30 13:38:41 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2004-05-10 14:46:06 +02:00
|
|
|
|
|
|
|
// Start the ordered index scan and fetch the first row
|
2004-12-20 15:12:42 +01:00
|
|
|
error= ordered_index_scan(start_key, end_key, sorted, FALSE, buf);
|
2004-04-30 13:38:41 +02:00
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2004-10-21 12:39:08 +02:00
|
|
|
int ha_ndbcluster::read_range_first(const key_range *start_key,
|
2005-02-16 14:18:32 +01:00
|
|
|
const key_range *end_key,
|
|
|
|
bool eq_r, bool sorted)
|
2004-10-21 12:39:08 +02:00
|
|
|
{
|
|
|
|
byte* buf= table->record[0];
|
|
|
|
DBUG_ENTER("ha_ndbcluster::read_range_first");
|
|
|
|
|
|
|
|
DBUG_RETURN(read_range_first_to_buf(start_key,
|
2005-02-16 14:18:32 +01:00
|
|
|
end_key,
|
|
|
|
eq_r,
|
|
|
|
sorted,
|
|
|
|
buf));
|
2004-10-21 12:39:08 +02:00
|
|
|
}
|
|
|
|
|
2004-05-24 12:35:39 +02:00
|
|
|
int ha_ndbcluster::read_range_next()
|
2004-04-30 13:38:41 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("ha_ndbcluster::read_range_next");
|
|
|
|
DBUG_RETURN(next_result(table->record[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
int ha_ndbcluster::rnd_init(bool scan)
|
|
|
|
{
|
2004-12-07 15:15:49 +01:00
|
|
|
NdbScanOperation *cursor= m_active_cursor;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("rnd_init");
|
|
|
|
DBUG_PRINT("enter", ("scan: %d", scan));
|
2004-07-09 13:18:56 +02:00
|
|
|
// Check if scan is to be restarted
|
2004-08-24 23:13:32 +02:00
|
|
|
if (cursor)
|
|
|
|
{
|
|
|
|
if (!scan)
|
|
|
|
DBUG_RETURN(1);
|
2005-07-04 02:42:33 +02:00
|
|
|
if (cursor->restart(m_force_send) != 0)
|
2005-01-11 17:00:31 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
2004-08-24 23:13:32 +02:00
|
|
|
}
|
2005-01-06 12:00:13 +01:00
|
|
|
index_init(table->s->primary_key);
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-05-11 13:59:22 +02:00
|
|
|
int ha_ndbcluster::close_scan()
|
|
|
|
{
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-05-11 13:59:22 +02:00
|
|
|
DBUG_ENTER("close_scan");
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
m_multi_cursor= 0;
|
|
|
|
if (!m_active_cursor && !m_multi_cursor)
|
2004-05-11 13:59:22 +02:00
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
NdbScanOperation *cursor= m_active_cursor ? m_active_cursor : m_multi_cursor;
|
2004-08-04 10:54:42 +02:00
|
|
|
|
2006-08-15 13:12:27 +02:00
|
|
|
if (m_lock_tuple)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Lock level m_lock.type either TL_WRITE_ALLOW_WRITE
|
|
|
|
(SELECT FOR UPDATE) or TL_READ_WITH_SHARED_LOCKS (SELECT
|
|
|
|
LOCK WITH SHARE MODE) and row was not explictly unlocked
|
|
|
|
with unlock_row() call
|
|
|
|
*/
|
|
|
|
NdbOperation *op;
|
|
|
|
// Lock row
|
|
|
|
DBUG_PRINT("info", ("Keeping lock on scanned row"));
|
|
|
|
|
2006-08-15 14:31:21 +02:00
|
|
|
if (!(op= cursor->lockCurrentTuple()))
|
2006-08-15 13:12:27 +02:00
|
|
|
{
|
|
|
|
m_lock_tuple= false;
|
|
|
|
ERR_RETURN(trans->getNdbError());
|
|
|
|
}
|
|
|
|
m_ops_pending++;
|
|
|
|
}
|
2006-06-08 16:12:38 +02:00
|
|
|
m_lock_tuple= false;
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_ops_pending)
|
2004-08-04 10:54:42 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Take over any pending transactions to the
|
|
|
|
deleteing/updating transaction before closing the scan
|
|
|
|
*/
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("info", ("ops_pending: %ld", (long) m_ops_pending));
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this,trans,false) != 0) {
|
2004-09-14 17:51:57 +02:00
|
|
|
no_uncommitted_rows_execute_failure();
|
2004-08-04 10:54:42 +02:00
|
|
|
DBUG_RETURN(ndb_err(trans));
|
2004-09-14 17:51:57 +02:00
|
|
|
}
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending= 0;
|
2004-08-04 10:54:42 +02:00
|
|
|
}
|
|
|
|
|
2005-05-06 10:39:30 +02:00
|
|
|
cursor->close(m_force_send, TRUE);
|
2004-12-08 00:36:40 +01:00
|
|
|
m_active_cursor= m_multi_cursor= NULL;
|
2004-05-28 11:23:44 +02:00
|
|
|
DBUG_RETURN(0);
|
2004-05-11 13:59:22 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
int ha_ndbcluster::rnd_end()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("rnd_end");
|
2004-05-11 13:59:22 +02:00
|
|
|
DBUG_RETURN(close_scan());
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::rnd_next(byte *buf)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("rnd_next");
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_rnd_next_count,
|
2005-02-16 14:18:32 +01:00
|
|
|
&LOCK_status);
|
2004-05-10 14:12:28 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
if (!m_active_cursor)
|
2004-05-10 14:12:28 +02:00
|
|
|
DBUG_RETURN(full_table_scan(buf));
|
|
|
|
DBUG_RETURN(next_result(buf));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
An "interesting" record has been found and it's pk
|
|
|
|
retrieved by calling position
|
|
|
|
Now it's time to read the record from db once
|
|
|
|
again
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::rnd_pos(byte *buf, byte *pos)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("rnd_pos");
|
2004-09-14 21:02:23 +02:00
|
|
|
statistic_increment(current_thd->status_var.ha_read_rnd_count,
|
2005-02-16 14:18:32 +01:00
|
|
|
&LOCK_status);
|
2004-04-15 09:14:14 +02:00
|
|
|
// The primary key for the record is stored in pos
|
|
|
|
// Perform a pk_read using primary key "index"
|
|
|
|
DBUG_RETURN(pk_read(pos, ref_length, buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Store the primary key of this record in ref
|
|
|
|
variable, so that the row can be retrieved again later
|
|
|
|
using "reference" in rnd_pos
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::position(const byte *record)
|
|
|
|
{
|
|
|
|
KEY *key_info;
|
|
|
|
KEY_PART_INFO *key_part;
|
|
|
|
KEY_PART_INFO *end;
|
|
|
|
byte *buff;
|
|
|
|
DBUG_ENTER("position");
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key != MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
key_info= table->key_info + table->s->primary_key;
|
2004-04-15 09:14:14 +02:00
|
|
|
key_part= key_info->key_part;
|
|
|
|
end= key_part + key_info->key_parts;
|
|
|
|
buff= ref;
|
|
|
|
|
|
|
|
for (; key_part != end; key_part++)
|
|
|
|
{
|
|
|
|
if (key_part->null_bit) {
|
|
|
|
/* Store 0 if the key part is a NULL part */
|
|
|
|
if (record[key_part->null_offset]
|
|
|
|
& key_part->null_bit) {
|
|
|
|
*buff++= 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
*buff++= 0;
|
|
|
|
}
|
2005-12-14 10:57:09 +01:00
|
|
|
|
|
|
|
size_t len = key_part->length;
|
|
|
|
const byte * ptr = record + key_part->offset;
|
|
|
|
Field *field = key_part->field;
|
2007-01-23 01:12:03 +01:00
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR)
|
2005-12-14 10:57:09 +01:00
|
|
|
{
|
2007-01-22 18:34:00 +01:00
|
|
|
if (((Field_varstring*)field)->length_bytes == 1)
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Keys always use 2 bytes length
|
|
|
|
*/
|
|
|
|
buff[0] = ptr[0];
|
|
|
|
buff[1] = 0;
|
|
|
|
memcpy(buff+2, ptr + 1, len);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memcpy(buff, ptr, len + 2);
|
|
|
|
}
|
|
|
|
len += 2;
|
2005-12-14 10:57:09 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-01-22 18:34:00 +01:00
|
|
|
memcpy(buff, ptr, len);
|
2005-12-14 10:57:09 +01:00
|
|
|
}
|
|
|
|
buff += len;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No primary key, get hidden key
|
|
|
|
DBUG_PRINT("info", ("Getting hidden key"));
|
2005-01-11 17:00:31 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-02-10 17:50:17 +01:00
|
|
|
int hidden_no= table->s->fields;
|
2004-09-07 21:53:59 +02:00
|
|
|
const NDBTAB *tab= (const NDBTAB *) m_table;
|
2004-04-15 09:14:14 +02:00
|
|
|
const NDBCOL *hidden_col= tab->getColumn(hidden_no);
|
|
|
|
DBUG_ASSERT(hidden_col->getPrimaryKey() &&
|
|
|
|
hidden_col->getAutoIncrement() &&
|
|
|
|
ref_length == NDB_HIDDEN_PRIMARY_KEY_LENGTH);
|
2005-01-11 17:00:31 +01:00
|
|
|
#endif
|
2006-02-10 17:40:22 +01:00
|
|
|
memcpy(ref, m_ref, ref_length);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_DUMP("ref", (char*)ref, ref_length);
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-10 16:55:20 +02:00
|
|
|
int ha_ndbcluster::info(uint flag)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2006-08-10 16:55:20 +02:00
|
|
|
int result= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("info");
|
|
|
|
DBUG_PRINT("enter", ("flag: %d", flag));
|
|
|
|
|
|
|
|
if (flag & HA_STATUS_POS)
|
|
|
|
DBUG_PRINT("info", ("HA_STATUS_POS"));
|
|
|
|
if (flag & HA_STATUS_NO_LOCK)
|
|
|
|
DBUG_PRINT("info", ("HA_STATUS_NO_LOCK"));
|
|
|
|
if (flag & HA_STATUS_TIME)
|
|
|
|
DBUG_PRINT("info", ("HA_STATUS_TIME"));
|
|
|
|
if (flag & HA_STATUS_VARIABLE)
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
{
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("info", ("HA_STATUS_VARIABLE"));
|
2004-09-20 17:52:19 +02:00
|
|
|
if (m_table_info)
|
|
|
|
{
|
2004-11-17 09:15:53 +01:00
|
|
|
if (m_ha_not_exact_count)
|
2005-02-16 14:18:32 +01:00
|
|
|
records= 100;
|
2004-11-17 09:15:53 +01:00
|
|
|
else
|
2006-08-10 16:55:20 +02:00
|
|
|
result= records_update();
|
2004-09-20 17:52:19 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
if ((my_errno= check_ndb_connection()))
|
2006-08-10 16:55:20 +02:00
|
|
|
DBUG_RETURN(my_errno);
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
2005-01-13 18:24:19 +01:00
|
|
|
struct Ndb_statistics stat;
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= HA_ERR_OUT_OF_MEM);
|
|
|
|
}
|
2005-01-13 18:24:19 +01:00
|
|
|
if (current_thd->variables.ndb_use_exact_count &&
|
2006-10-23 16:18:57 +02:00
|
|
|
(result= ndb_get_table_statistics(this, true, ndb, m_tabname, &stat))
|
|
|
|
== 0)
|
2005-01-13 18:24:19 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
mean_rec_length= stat.row_size;
|
|
|
|
data_file_length= stat.fragment_memory;
|
|
|
|
records= stat.row_count;
|
2005-01-13 18:24:19 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
mean_rec_length= 0;
|
|
|
|
records= 100;
|
2005-01-13 18:24:19 +01:00
|
|
|
}
|
2004-09-20 17:52:19 +02:00
|
|
|
}
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
2004-10-04 01:20:05 +02:00
|
|
|
if (flag & HA_STATUS_CONST)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("HA_STATUS_CONST"));
|
|
|
|
set_rec_per_key();
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
if (flag & HA_STATUS_ERRKEY)
|
2004-08-31 12:07:48 +02:00
|
|
|
{
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("info", ("HA_STATUS_ERRKEY"));
|
2004-11-03 15:53:26 +01:00
|
|
|
errkey= m_dupkey;
|
2004-08-31 12:07:48 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
if (flag & HA_STATUS_AUTO)
|
2005-08-10 09:38:36 +02:00
|
|
|
{
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("info", ("HA_STATUS_AUTO"));
|
2007-02-14 09:16:26 +01:00
|
|
|
if (m_table && table->found_next_number_field)
|
2005-08-10 09:38:36 +02:00
|
|
|
{
|
2007-07-26 12:25:05 +02:00
|
|
|
if ((my_errno= check_ndb_connection()))
|
|
|
|
DBUG_RETURN(my_errno);
|
2005-08-10 09:38:36 +02:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
|
2006-05-25 18:00:55 +02:00
|
|
|
Uint64 auto_increment_value64;
|
2006-05-17 12:40:10 +02:00
|
|
|
if (ndb->readAutoIncrementValue((const NDBTAB *) m_table,
|
2006-05-25 18:00:55 +02:00
|
|
|
auto_increment_value64) == -1)
|
2006-05-17 12:40:10 +02:00
|
|
|
{
|
|
|
|
const NdbError err= ndb->getNdbError();
|
|
|
|
sql_print_error("Error %lu in readAutoIncrementValue(): %s",
|
|
|
|
(ulong) err.code, err.message);
|
|
|
|
auto_increment_value= ~(Uint64)0;
|
|
|
|
}
|
2006-05-25 21:27:56 +02:00
|
|
|
else
|
|
|
|
auto_increment_value= (ulonglong)auto_increment_value64;
|
2005-08-10 09:38:36 +02:00
|
|
|
}
|
|
|
|
}
|
2006-08-10 16:55:20 +02:00
|
|
|
|
|
|
|
if(result == -1)
|
|
|
|
result= HA_ERR_NO_CONNECTION;
|
|
|
|
|
|
|
|
DBUG_RETURN(result);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ha_ndbcluster::extra(enum ha_extra_function operation)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("extra");
|
|
|
|
switch (operation) {
|
|
|
|
case HA_EXTRA_NORMAL: /* Optimize for space (def) */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NORMAL"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_QUICK: /* Optimize for speed */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_QUICK"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_RESET: /* Reset database to after open */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_RESET"));
|
2007-03-22 12:42:13 +01:00
|
|
|
reset();
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case HA_EXTRA_CACHE: /* Cash record in HA_rrnd() */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_CACHE: /* End cacheing of records (def) */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_READCHECK: /* No readcheck on update */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_READCHECK"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_READCHECK: /* Use readcheck (def) */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_READCHECK"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_KEYREAD: /* Read only key to database */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_KEYREAD"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_KEYREAD: /* Normal read of records (def) */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_KEYREAD"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_USER_CHANGE: /* No user is allowed to write */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_USER_CHANGE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_KEY_CACHE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_KEY_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_KEY_CACHE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_KEY_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_WAIT_LOCK: /* Wait until file is avalably (def) */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_WAIT_LOCK"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_WAIT_LOCK: /* If file is locked, return quickly */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_WAIT_LOCK"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_WRITE_CACHE: /* Use write cache in ha_write() */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_WRITE_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_FLUSH_CACHE: /* flush write_record_cache */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_FLUSH_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_KEYS: /* Remove all update of keys */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_KEYS"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_KEYREAD_CHANGE_POS: /* Keyread, but change pos */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_KEYREAD_CHANGE_POS")); /* xxxxchk -r must be used */
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_REMEMBER_POS: /* Remember pos for next/prev */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_REMEMBER_POS"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_RESTORE_POS:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_RESTORE_POS"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_REINIT_CACHE: /* init cache from current record */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_REINIT_CACHE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_FORCE_REOPEN: /* Datafile have changed on disk */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_FORCE_REOPEN"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_FLUSH: /* Flush tables to disk */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_FLUSH"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_ROWS: /* Don't write rows */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_ROWS"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_RESET_STATE: /* Reset positions */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_RESET_STATE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_IGNORE_DUP_KEY: /* Dup keys don't rollback everything*/
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_IGNORE_DUP_KEY"));
|
2006-07-01 23:51:10 +02:00
|
|
|
DBUG_PRINT("info", ("Ignoring duplicate key"));
|
|
|
|
m_ignore_dup_key= TRUE;
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case HA_EXTRA_NO_IGNORE_DUP_KEY:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_NO_IGNORE_DUP_KEY"));
|
2004-11-18 12:11:56 +01:00
|
|
|
m_ignore_dup_key= FALSE;
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case HA_EXTRA_RETRIEVE_ALL_COLS: /* Retrieve all columns, not just those
|
2005-02-16 14:18:32 +01:00
|
|
|
where field->query_id is the same as
|
|
|
|
the current query id */
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_RETRIEVE_ALL_COLS"));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_retrieve_all_fields= TRUE;
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case HA_EXTRA_PREPARE_FOR_DELETE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_PREPARE_FOR_DELETE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_PREPARE_FOR_UPDATE: /* Remove read cache if problems */
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_PREPARE_FOR_UPDATE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_PRELOAD_BUFFER_SIZE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_PRELOAD_BUFFER_SIZE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_RETRIEVE_PRIMARY_KEY:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_RETRIEVE_PRIMARY_KEY"));
|
2004-11-18 12:11:56 +01:00
|
|
|
m_retrieve_primary_key= TRUE;
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case HA_EXTRA_CHANGE_KEY_TO_UNIQUE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_CHANGE_KEY_TO_UNIQUE"));
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_CHANGE_KEY_TO_DUP:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_CHANGE_KEY_TO_DUP"));
|
2004-09-09 05:59:26 +02:00
|
|
|
case HA_EXTRA_KEYREAD_PRESERVE_FIELDS:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_KEYREAD_PRESERVE_FIELDS"));
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
2006-07-01 23:51:10 +02:00
|
|
|
case HA_EXTRA_WRITE_CAN_REPLACE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_WRITE_CAN_REPLACE"));
|
|
|
|
if (!m_has_unique_index)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Turning ON use of write instead of insert"));
|
|
|
|
m_use_write= TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_WRITE_CANNOT_REPLACE:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_WRITE_CANNOT_REPLACE"));
|
|
|
|
DBUG_PRINT("info", ("Turning OFF use of write instead of insert"));
|
|
|
|
m_use_write= FALSE;
|
|
|
|
break;
|
2007-04-04 12:50:39 +02:00
|
|
|
case HA_EXTRA_DELETE_CANNOT_BATCH:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_DELETE_CANNOT_BATCH"));
|
|
|
|
m_delete_cannot_batch= TRUE;
|
|
|
|
break;
|
|
|
|
case HA_EXTRA_UPDATE_CANNOT_BATCH:
|
|
|
|
DBUG_PRINT("info", ("HA_EXTRA_UPDATE_CANNOT_BATCH"));
|
|
|
|
m_update_cannot_batch= TRUE;
|
|
|
|
break;
|
2007-04-05 07:28:09 +02:00
|
|
|
default:
|
|
|
|
break;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2007-03-22 12:42:13 +01:00
|
|
|
|
|
|
|
int ha_ndbcluster::reset()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("ha_ndbcluster::reset");
|
2007-04-23 11:25:33 +02:00
|
|
|
if (m_cond)
|
|
|
|
{
|
|
|
|
m_cond->cond_clear();
|
|
|
|
}
|
2007-03-22 12:42:13 +01:00
|
|
|
|
|
|
|
/* reset flags set by extra calls */
|
|
|
|
m_retrieve_all_fields= FALSE;
|
|
|
|
m_retrieve_primary_key= FALSE;
|
|
|
|
m_ignore_dup_key= FALSE;
|
|
|
|
m_use_write= FALSE;
|
2007-04-05 08:39:12 +02:00
|
|
|
m_delete_cannot_batch= FALSE;
|
|
|
|
m_update_cannot_batch= FALSE;
|
2007-03-22 12:42:13 +01:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-29 14:38:35 +02:00
|
|
|
/*
|
|
|
|
Start of an insert, remember number of rows to be inserted, it will
|
|
|
|
be used in write_row and get_autoincrement to send an optimal number
|
|
|
|
of rows in each roundtrip to the server
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
rows number of rows to insert, 0 if unknown
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::start_bulk_insert(ha_rows rows)
|
|
|
|
{
|
|
|
|
int bytes, batch;
|
2004-09-07 21:53:59 +02:00
|
|
|
const NDBTAB *tab= (const NDBTAB *) m_table;
|
2004-04-29 14:38:35 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("start_bulk_insert");
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_PRINT("enter", ("rows: %d", (int)rows));
|
2004-04-29 14:38:35 +02:00
|
|
|
|
2005-04-13 12:42:23 +02:00
|
|
|
m_rows_inserted= (ha_rows) 0;
|
2006-06-14 20:25:51 +02:00
|
|
|
if (!m_use_write && m_ignore_dup_key)
|
2006-06-14 20:16:32 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
compare if expression with that in write_row
|
2006-06-14 20:25:51 +02:00
|
|
|
we have a situation where peek_indexed_rows() will be called
|
2006-06-14 20:16:32 +02:00
|
|
|
so we cannot batch
|
|
|
|
*/
|
|
|
|
DBUG_PRINT("info", ("Batching turned off as duplicate key is "
|
|
|
|
"ignored by using peek_row"));
|
|
|
|
m_rows_to_insert= 1;
|
|
|
|
m_bulk_insert_rows= 1;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
2005-04-13 12:42:23 +02:00
|
|
|
if (rows == (ha_rows) 0)
|
2005-05-07 22:04:52 +02:00
|
|
|
{
|
2005-04-07 17:44:09 +02:00
|
|
|
/* We don't know how many will be inserted, guess */
|
|
|
|
m_rows_to_insert= m_autoincrement_prefetch;
|
2005-05-07 22:04:52 +02:00
|
|
|
}
|
2005-04-07 17:44:09 +02:00
|
|
|
else
|
|
|
|
m_rows_to_insert= rows;
|
2004-04-29 14:38:35 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Calculate how many rows that should be inserted
|
|
|
|
per roundtrip to NDB. This is done in order to minimize the
|
|
|
|
number of roundtrips as much as possible. However performance will
|
|
|
|
degrade if too many bytes are inserted, thus it's limited by this
|
|
|
|
calculation.
|
|
|
|
*/
|
2004-07-23 15:46:56 +02:00
|
|
|
const int bytesperbatch= 8192;
|
2004-04-29 14:38:35 +02:00
|
|
|
bytes= 12 + tab->getRowSizeInBytes() + 4 * tab->getNoOfColumns();
|
2004-05-10 14:46:06 +02:00
|
|
|
batch= bytesperbatch/bytes;
|
2004-04-29 14:38:35 +02:00
|
|
|
batch= batch == 0 ? 1 : batch;
|
|
|
|
DBUG_PRINT("info", ("batch: %d, bytes: %d", batch, bytes));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_bulk_insert_rows= batch;
|
2004-04-29 14:38:35 +02:00
|
|
|
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
End of an insert
|
|
|
|
*/
|
|
|
|
int ha_ndbcluster::end_bulk_insert()
|
|
|
|
{
|
2004-07-23 15:46:56 +02:00
|
|
|
int error= 0;
|
|
|
|
|
2004-04-29 14:38:35 +02:00
|
|
|
DBUG_ENTER("end_bulk_insert");
|
2004-07-23 15:46:56 +02:00
|
|
|
// Check if last inserts need to be flushed
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_bulk_insert_not_flushed)
|
2004-07-23 15:46:56 +02:00
|
|
|
{
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction *trans= m_active_trans;
|
2004-07-23 15:46:56 +02:00
|
|
|
// Send rows to NDB
|
|
|
|
DBUG_PRINT("info", ("Sending inserts to NDB, "\
|
|
|
|
"rows_inserted:%d, bulk_insert_rows: %d",
|
2005-04-13 12:42:23 +02:00
|
|
|
(int) m_rows_inserted, (int) m_bulk_insert_rows));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_bulk_insert_not_flushed= FALSE;
|
2006-02-07 00:03:39 +01:00
|
|
|
if (m_transaction_on)
|
|
|
|
{
|
2006-08-15 14:31:21 +02:00
|
|
|
if (execute_no_commit(this, trans,false) != 0)
|
2006-02-07 00:03:39 +01:00
|
|
|
{
|
|
|
|
no_uncommitted_rows_execute_failure();
|
|
|
|
my_errno= error= ndb_err(trans);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (execute_commit(this, trans) != 0)
|
|
|
|
{
|
|
|
|
no_uncommitted_rows_execute_failure();
|
|
|
|
my_errno= error= ndb_err(trans);
|
|
|
|
}
|
2006-02-07 11:30:42 +01:00
|
|
|
else
|
|
|
|
{
|
2007-02-28 21:23:35 +01:00
|
|
|
IF_DBUG(int res=) trans->restart();
|
2006-02-07 11:30:42 +01:00
|
|
|
DBUG_ASSERT(res == 0);
|
|
|
|
}
|
2004-09-14 17:51:57 +02:00
|
|
|
}
|
2004-07-23 15:46:56 +02:00
|
|
|
}
|
|
|
|
|
2005-04-13 12:42:23 +02:00
|
|
|
m_rows_inserted= (ha_rows) 0;
|
|
|
|
m_rows_to_insert= (ha_rows) 1;
|
2004-07-23 15:46:56 +02:00
|
|
|
DBUG_RETURN(error);
|
2004-04-29 14:38:35 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
int ha_ndbcluster::extra_opt(enum ha_extra_function operation, ulong cache_size)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("extra_opt");
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_PRINT("enter", ("cache_size: %lu", cache_size));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(extra(operation));
|
|
|
|
}
|
|
|
|
|
2005-04-27 11:25:08 +02:00
|
|
|
static const char *ha_ndbcluster_exts[] = {
|
|
|
|
ha_ndb_ext,
|
|
|
|
NullS
|
|
|
|
};
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-03-21 12:29:08 +01:00
|
|
|
const char** ha_ndbcluster::bas_ext() const
|
2005-04-27 11:25:08 +02:00
|
|
|
{
|
|
|
|
return ha_ndbcluster_exts;
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
How many seeks it will take to read through the table
|
|
|
|
This is to be comparable to the number returned by records_in_range so
|
|
|
|
that we can decide if we should scan the table or use keys.
|
|
|
|
*/
|
|
|
|
|
|
|
|
double ha_ndbcluster::scan_time()
|
|
|
|
{
|
2004-09-03 13:55:40 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::scan_time()");
|
|
|
|
double res= rows2double(records*1000);
|
|
|
|
DBUG_PRINT("exit", ("table: %s value: %f",
|
2005-02-16 14:18:32 +01:00
|
|
|
m_tabname, res));
|
2004-09-03 13:55:40 +02:00
|
|
|
DBUG_RETURN(res);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2005-06-20 17:16:22 +02:00
|
|
|
/*
|
|
|
|
Convert MySQL table locks into locks supported by Ndb Cluster.
|
|
|
|
Note that MySQL Cluster does currently not support distributed
|
|
|
|
table locks, so to be safe one should set cluster in Single
|
|
|
|
User Mode, before relying on table locks when updating tables
|
|
|
|
from several MySQL servers
|
|
|
|
*/
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
THR_LOCK_DATA **ha_ndbcluster::store_lock(THD *thd,
|
|
|
|
THR_LOCK_DATA **to,
|
|
|
|
enum thr_lock_type lock_type)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("store_lock");
|
|
|
|
if (lock_type != TL_IGNORE && m_lock.type == TL_UNLOCK)
|
|
|
|
{
|
2004-09-28 21:11:50 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/* If we are not doing a LOCK TABLE, then allow multiple
|
|
|
|
writers */
|
|
|
|
|
2004-10-13 10:08:18 +02:00
|
|
|
/* Since NDB does not currently have table locks
|
|
|
|
this is treated as a ordinary lock */
|
|
|
|
|
2005-06-09 13:28:29 +02:00
|
|
|
if ((lock_type >= TL_WRITE_CONCURRENT_INSERT &&
|
2004-04-15 09:14:14 +02:00
|
|
|
lock_type <= TL_WRITE) && !thd->in_lock_tables)
|
|
|
|
lock_type= TL_WRITE_ALLOW_WRITE;
|
|
|
|
|
|
|
|
/* In queries of type INSERT INTO t1 SELECT ... FROM t2 ...
|
|
|
|
MySQL would use the lock TL_READ_NO_INSERT on t2, and that
|
|
|
|
would conflict with TL_WRITE_ALLOW_WRITE, blocking all inserts
|
|
|
|
to t2. Convert the lock to a normal read lock to allow
|
|
|
|
concurrent inserts to t2. */
|
|
|
|
|
|
|
|
if (lock_type == TL_READ_NO_INSERT && !thd->in_lock_tables)
|
|
|
|
lock_type= TL_READ;
|
|
|
|
|
|
|
|
m_lock.type=lock_type;
|
|
|
|
}
|
|
|
|
*to++= &m_lock;
|
2004-05-10 14:12:28 +02:00
|
|
|
|
|
|
|
DBUG_PRINT("exit", ("lock_type: %d", lock_type));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(to);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
#define PRINT_OPTION_FLAGS(t) { \
|
|
|
|
if (t->options & OPTION_NOT_AUTOCOMMIT) \
|
|
|
|
DBUG_PRINT("thd->options", ("OPTION_NOT_AUTOCOMMIT")); \
|
|
|
|
if (t->options & OPTION_BEGIN) \
|
|
|
|
DBUG_PRINT("thd->options", ("OPTION_BEGIN")); \
|
|
|
|
if (t->options & OPTION_TABLE_LOCK) \
|
|
|
|
DBUG_PRINT("thd->options", ("OPTION_TABLE_LOCK")); \
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define PRINT_OPTION_FLAGS(t)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
As MySQL will execute an external lock for every new table it uses
|
|
|
|
we can use this to start the transactions.
|
|
|
|
If we are in auto_commit mode we just need to start a transaction
|
2005-02-17 22:52:40 +01:00
|
|
|
for the statement, this will be stored in thd_ndb.stmt.
|
2004-04-15 09:14:14 +02:00
|
|
|
If not, we have to start a master transaction if there doesn't exist
|
2005-02-17 22:52:40 +01:00
|
|
|
one from before, this will be stored in thd_ndb.all
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
When a table lock is held one transaction will be started which holds
|
|
|
|
the table lock and for each statement a hupp transaction will be started
|
2005-03-22 17:48:28 +01:00
|
|
|
If we are locking the table then:
|
2005-03-22 17:42:08 +01:00
|
|
|
- save the NdbDictionary::Table for easy access
|
2005-03-22 17:48:28 +01:00
|
|
|
- save reference to table statistics
|
|
|
|
- refresh list of the indexes for the table if needed (if altered)
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::external_lock(THD *thd, int lock_type)
|
|
|
|
{
|
|
|
|
int error=0;
|
2005-01-04 13:47:16 +01:00
|
|
|
NdbTransaction* trans= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("external_lock");
|
|
|
|
/*
|
|
|
|
Check that this handler instance has a connection
|
|
|
|
set up to the Ndb object of thd
|
|
|
|
*/
|
2005-03-15 15:03:25 +01:00
|
|
|
if (check_ndb_connection(thd))
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(1);
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= thd_ndb->ndb;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("enter", ("thd: 0x%lx thd_ndb: 0x%lx thd_ndb->lock_count: %d",
|
|
|
|
(long) thd, (long) thd_ndb, thd_ndb->lock_count));
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
if (lock_type != F_UNLCK)
|
|
|
|
{
|
2004-05-24 12:35:39 +02:00
|
|
|
DBUG_PRINT("info", ("lock_type != F_UNLCK"));
|
Fix for BUG#11151 "LOAD DATA INFILE commits transaction in 5.0".
In 5.0 we made LOAD DATA INFILE autocommit in all engines, while
only NDB wanted that. Users and trainers complained that it affected
InnoDB and was a change compared to 4.1 where only NDB autocommitted.
To revert to the behaviour of 4.1, we move the autocommit logic out of mysql_load() into
ha_ndbcluster::external_lock().
The result is that LOAD DATA INFILE commits all uncommitted changes
of NDB if this is an NDB table, its own changes if this is an NDB
table, but does not affect other engines.
Note: even though there is no "commit the full transaction at end"
anymore, LOAD DATA INFILE stays disabled in routines (re-entrency
problems per a comment of Pem).
Note: ha_ndbcluster::has_transactions() does not give reliable results
because it says "yes" even if transactions are disabled in this engine...
sql/ha_ndbcluster.cc:
NDB wants to do autocommit if this is LOAD DATA INFILE.
For this to not affect all other engines, we move the logic
inside ha_ndbcluster.
sql/sql_load.cc:
This ha_enable_transaction() in mysql_load() forced an autocommit
in all engines, while only NDB wants to do that.
So we move the logic inside ha_ndbcluster.cc.
mysql-test/include/loaddata_autocom.inc:
test for engines to see if they autocommit or not in LOAD DATA INFILE
mysql-test/r/loaddata_autocom_innodb.result:
result for InnoDB (no autocommit)
mysql-test/r/loaddata_autocom_ndb.result:
result for NDB (autocommit)
mysql-test/r/rpl_ndb_innodb_trans.result:
result for InnoDB+NDB transactions. Observe that when ROLLBACK
cannot rollback the LOAD DATA INFILE in NDB it issues warning 1196
as appropriate.
mysql-test/t/loaddata_autocom_innodb.test:
test that InnoDB does not autocommit in LOAD DATA INFILE.
mysql-test/t/loaddata_autocom_ndb.test:
test that NDB does autocommit in LOAD DATA INFIL
mysql-test/t/rpl_ndb_innodb_trans-slave.opt:
need to tell the slave to use innodb
mysql-test/t/rpl_ndb_innodb_trans.test:
test of transactions mixing NDB and InnoDB. To see if ROLLBACK
rolls back in both engines, with the exception of LOAD DATA INFILE
which does not roll back NDB: we see that a LOAD DATA INFILE in NDB
commits all what has been done in NDB so far, commits its changes,
but does not commit in other engines.
2006-09-05 16:52:05 +02:00
|
|
|
if (thd->lex->sql_command == SQLCOM_LOAD)
|
|
|
|
{
|
|
|
|
m_transaction_on= FALSE;
|
|
|
|
/* Would be simpler if has_transactions() didn't always say "yes" */
|
(pushing for Andrei)
Bug #27417 thd->no_trans_update.stmt lost value inside of SF-exec-stack
Once had been set the flag might later got reset inside of a stored routine
execution stack.
The reason was in that there was no check if a new statement started at time
of resetting.
The artifact affects most of binlogable DML queries. Notice, that multi-update
is wrapped up within
bug@27716 fix, multi-delete bug@29136.
Fixed with saving parent's statement flag of whether the statement modified
non-transactional table, and unioning (merging) the value with that was gained
in mysql_execute_command.
Resettling thd->no_trans_update members into thd->transaction.`member`;
Asserting code;
Effectively the following properties are held.
1. At the end of a substatement thd->transaction.stmt.modified_non_trans_table
reflects the fact if such a table got modified by the substatement.
That also respects THD::really_abort_on_warnin() requirements.
2. Eventually thd->transaction.stmt.modified_non_trans_table will be computed as
the union of the values of all invoked sub-statements.
That fixes this bug#27417;
Computing of thd->transaction.all.modified_non_trans_table is refined to base to
the stmt's value for all the case including insert .. select statement which
before the patch had an extra issue bug@28960.
Minor issues are covered with mysql_load, mysql_delete, and binloggin of insert in
to temp_table select.
The supplied test verifies limitely, mostly asserts. The ultimate testing is defered
for bug@13270, bug@23333.
mysql-test/r/mix_innodb_myisam_binlog.result:
results changed
mysql-test/t/mix_innodb_myisam_binlog.test:
regression test incl the related bug#28960.
sql/ha_ndbcluster.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/handler.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/handler.h:
new member added
sql/log.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/set_var.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/sp_head.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
and saving and merging stmt's flag at the end of a substatement.
sql/sql_class.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/sql_class.h:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/sql_delete.cc:
correcting basic delete incl truncate branch and multi-delete queries to set
stmt.modified_non_trans_table;
optimization to set the flag at the end of per-row loop;
multi-delete still has an extra issue similar to bug#27716 of multi-update
- to be address with bug_29136 fix.
sql/sql_insert.cc:
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/sql_load.cc:
eliminating a separate issue where the stmt flag was saved and re-stored after
write_record that actually could change it and the change would be lost but
should remain permanent;
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
sql/sql_parse.cc:
initialization to transaction.stmt.modified_non_trans_table at the common part
of all types of statements processing - mysql_execute_command().
sql/sql_table.cc:
moving the reset up to the mysql_execute_command() caller
sql/sql_update.cc:
correcting update query case (multi-update part of the issues covered by other
bug#27716 fix)
thd->transaction.{all,stmt}.modified_non_trans_table
instead of
thd->no_trans_update.{all,stmt}
2007-07-30 17:27:36 +02:00
|
|
|
thd->transaction.all.modified_non_trans_table= thd->transaction.stmt.modified_non_trans_table= TRUE;
|
Fix for BUG#11151 "LOAD DATA INFILE commits transaction in 5.0".
In 5.0 we made LOAD DATA INFILE autocommit in all engines, while
only NDB wanted that. Users and trainers complained that it affected
InnoDB and was a change compared to 4.1 where only NDB autocommitted.
To revert to the behaviour of 4.1, we move the autocommit logic out of mysql_load() into
ha_ndbcluster::external_lock().
The result is that LOAD DATA INFILE commits all uncommitted changes
of NDB if this is an NDB table, its own changes if this is an NDB
table, but does not affect other engines.
Note: even though there is no "commit the full transaction at end"
anymore, LOAD DATA INFILE stays disabled in routines (re-entrency
problems per a comment of Pem).
Note: ha_ndbcluster::has_transactions() does not give reliable results
because it says "yes" even if transactions are disabled in this engine...
sql/ha_ndbcluster.cc:
NDB wants to do autocommit if this is LOAD DATA INFILE.
For this to not affect all other engines, we move the logic
inside ha_ndbcluster.
sql/sql_load.cc:
This ha_enable_transaction() in mysql_load() forced an autocommit
in all engines, while only NDB wants to do that.
So we move the logic inside ha_ndbcluster.cc.
mysql-test/include/loaddata_autocom.inc:
test for engines to see if they autocommit or not in LOAD DATA INFILE
mysql-test/r/loaddata_autocom_innodb.result:
result for InnoDB (no autocommit)
mysql-test/r/loaddata_autocom_ndb.result:
result for NDB (autocommit)
mysql-test/r/rpl_ndb_innodb_trans.result:
result for InnoDB+NDB transactions. Observe that when ROLLBACK
cannot rollback the LOAD DATA INFILE in NDB it issues warning 1196
as appropriate.
mysql-test/t/loaddata_autocom_innodb.test:
test that InnoDB does not autocommit in LOAD DATA INFILE.
mysql-test/t/loaddata_autocom_ndb.test:
test that NDB does autocommit in LOAD DATA INFIL
mysql-test/t/rpl_ndb_innodb_trans-slave.opt:
need to tell the slave to use innodb
mysql-test/t/rpl_ndb_innodb_trans.test:
test of transactions mixing NDB and InnoDB. To see if ROLLBACK
rolls back in both engines, with the exception of LOAD DATA INFILE
which does not roll back NDB: we see that a LOAD DATA INFILE in NDB
commits all what has been done in NDB so far, commits its changes,
but does not commit in other engines.
2006-09-05 16:52:05 +02:00
|
|
|
}
|
|
|
|
else if (!thd->transaction.on)
|
2005-12-10 00:00:33 +01:00
|
|
|
m_transaction_on= FALSE;
|
|
|
|
else
|
|
|
|
m_transaction_on= thd->variables.ndb_use_transactions;
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
if (!thd_ndb->lock_count++)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
PRINT_OPTION_FLAGS(thd);
|
2005-10-14 11:23:02 +02:00
|
|
|
if (!(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)))
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
// Autocommit transaction
|
2005-02-17 22:52:40 +01:00
|
|
|
DBUG_ASSERT(!thd_ndb->stmt);
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("trans",("Starting transaction stmt"));
|
|
|
|
|
2004-12-30 19:56:09 +01:00
|
|
|
trans= ndb->startTransaction();
|
2004-04-15 09:14:14 +02:00
|
|
|
if (trans == NULL)
|
2004-12-30 19:56:09 +01:00
|
|
|
ERR_RETURN(ndb->getNdbError());
|
2005-02-16 14:18:32 +01:00
|
|
|
no_uncommitted_rows_reset(thd);
|
2005-02-17 22:52:40 +01:00
|
|
|
thd_ndb->stmt= trans;
|
2006-08-15 14:31:21 +02:00
|
|
|
thd_ndb->query_state&= NDB_QUERY_NORMAL;
|
2006-02-07 00:26:47 +01:00
|
|
|
trans_register_ha(thd, FALSE, &ndbcluster_hton);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-17 22:52:40 +01:00
|
|
|
if (!thd_ndb->all)
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
2004-04-15 09:14:14 +02:00
|
|
|
// Not autocommit transaction
|
|
|
|
// A "master" transaction ha not been started yet
|
|
|
|
DBUG_PRINT("trans",("starting transaction, all"));
|
|
|
|
|
2004-12-30 19:56:09 +01:00
|
|
|
trans= ndb->startTransaction();
|
2004-04-15 09:14:14 +02:00
|
|
|
if (trans == NULL)
|
2004-12-30 19:56:09 +01:00
|
|
|
ERR_RETURN(ndb->getNdbError());
|
2005-02-16 14:18:32 +01:00
|
|
|
no_uncommitted_rows_reset(thd);
|
2005-02-17 22:52:40 +01:00
|
|
|
thd_ndb->all= trans;
|
2006-08-15 14:31:21 +02:00
|
|
|
thd_ndb->query_state&= NDB_QUERY_NORMAL;
|
2006-02-07 00:26:47 +01:00
|
|
|
trans_register_ha(thd, TRUE, &ndbcluster_hton);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
If this is the start of a LOCK TABLE, a table look
|
|
|
|
should be taken on the table in NDB
|
|
|
|
|
|
|
|
Check if it should be read or write lock
|
|
|
|
*/
|
|
|
|
if (thd->options & (OPTION_TABLE_LOCK))
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
2004-04-15 09:14:14 +02:00
|
|
|
//lockThisTable();
|
|
|
|
DBUG_PRINT("info", ("Locking the table..." ));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
This is the place to make sure this handler instance
|
|
|
|
has a started transaction.
|
|
|
|
|
|
|
|
The transaction is started by the first handler on which
|
|
|
|
MySQL Server calls external lock
|
|
|
|
|
|
|
|
Other handlers in the same stmt or transaction should use
|
|
|
|
the same NDB transaction. This is done by setting up the m_active_trans
|
|
|
|
pointer to point to the NDB transaction.
|
|
|
|
*/
|
|
|
|
|
2004-11-17 09:15:53 +01:00
|
|
|
// store thread specific data first to set the right context
|
|
|
|
m_force_send= thd->variables.ndb_force_send;
|
|
|
|
m_ha_not_exact_count= !thd->variables.ndb_use_exact_count;
|
2007-11-05 20:18:22 +01:00
|
|
|
m_autoincrement_prefetch=
|
|
|
|
(thd->variables.ndb_autoincrement_prefetch_sz >
|
|
|
|
NDB_DEFAULT_AUTO_PREFETCH) ?
|
|
|
|
(ha_rows) thd->variables.ndb_autoincrement_prefetch_sz
|
|
|
|
: (ha_rows) NDB_DEFAULT_AUTO_PREFETCH;
|
2005-02-17 22:52:40 +01:00
|
|
|
m_active_trans= thd_ndb->all ? thd_ndb->all : thd_ndb->stmt;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ASSERT(m_active_trans);
|
2004-05-10 14:12:28 +02:00
|
|
|
// Start of transaction
|
2005-03-15 15:03:25 +01:00
|
|
|
m_rows_changed= 0;
|
2004-11-03 15:53:26 +01:00
|
|
|
m_retrieve_all_fields= FALSE;
|
2004-11-18 12:11:56 +01:00
|
|
|
m_retrieve_primary_key= FALSE;
|
2005-03-15 15:03:25 +01:00
|
|
|
m_ops_pending= 0;
|
2004-09-20 17:52:19 +02:00
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-09-20 17:52:19 +02:00
|
|
|
const NDBTAB *tab;
|
|
|
|
void *tab_info;
|
|
|
|
if (!(tab= dict->getTable(m_tabname, &tab_info)))
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
2005-04-27 18:17:41 +02:00
|
|
|
DBUG_PRINT("info", ("Table schema version: %d",
|
|
|
|
tab->getObjectVersion()));
|
|
|
|
// Check if thread has stale local cache
|
2005-08-18 14:02:25 +02:00
|
|
|
// New transaction must not use old tables... (trans != 0)
|
|
|
|
// Running might...
|
|
|
|
if ((trans && tab->getObjectStatus() != NdbDictionary::Object::Retrieved)
|
|
|
|
|| tab->getObjectStatus() == NdbDictionary::Object::Invalid)
|
2005-04-27 18:17:41 +02:00
|
|
|
{
|
|
|
|
invalidate_dictionary_cache(FALSE);
|
2005-04-29 11:37:47 +02:00
|
|
|
if (!(tab= dict->getTable(m_tabname, &tab_info)))
|
2005-04-27 18:17:41 +02:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
DBUG_PRINT("info", ("Table schema version: %d",
|
|
|
|
tab->getObjectVersion()));
|
|
|
|
}
|
2005-11-24 16:21:07 +01:00
|
|
|
if (m_table_version < tab->getObjectVersion())
|
2005-11-17 12:45:23 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
The table has been altered, caller has to retry
|
|
|
|
*/
|
|
|
|
NdbError err= ndb->getNdbError(NDB_INVALID_SCHEMA_OBJECT);
|
|
|
|
DBUG_RETURN(ndb_to_mysql_error(&err));
|
|
|
|
}
|
2005-11-24 16:21:07 +01:00
|
|
|
if (m_table != (void *)tab)
|
|
|
|
{
|
|
|
|
m_table= (void *)tab;
|
|
|
|
m_table_version = tab->getObjectVersion();
|
2006-05-15 16:23:59 +02:00
|
|
|
if ((my_errno= build_index_list(ndb, table, ILBP_OPEN)))
|
2005-11-24 16:21:07 +01:00
|
|
|
DBUG_RETURN(my_errno);
|
2006-05-15 16:23:59 +02:00
|
|
|
|
2007-03-21 08:40:24 +01:00
|
|
|
const void *data= NULL, *pack_data= NULL;
|
2006-05-15 16:23:59 +02:00
|
|
|
uint length, pack_length;
|
2006-05-15 20:29:50 +02:00
|
|
|
if (readfrm(table->s->path, &data, &length) ||
|
2006-05-15 16:23:59 +02:00
|
|
|
packfrm(data, length, &pack_data, &pack_length) ||
|
|
|
|
pack_length != tab->getFrmLength() ||
|
|
|
|
memcmp(pack_data, tab->getFrmData(), pack_length))
|
|
|
|
{
|
|
|
|
my_free((char*)data, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
my_free((char*)pack_data, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
NdbError err= ndb->getNdbError(NDB_INVALID_SCHEMA_OBJECT);
|
|
|
|
DBUG_RETURN(ndb_to_mysql_error(&err));
|
|
|
|
}
|
|
|
|
my_free((char*)data, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
my_free((char*)pack_data, MYF(MY_ALLOW_ZERO_PTR));
|
2005-11-17 12:45:23 +01:00
|
|
|
}
|
2004-09-20 17:52:19 +02:00
|
|
|
m_table_info= tab_info;
|
|
|
|
}
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
no_uncommitted_rows_init(thd);
|
2005-03-15 15:03:25 +01:00
|
|
|
}
|
|
|
|
else
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-05-24 12:35:39 +02:00
|
|
|
DBUG_PRINT("info", ("lock_type == F_UNLCK"));
|
2005-03-15 15:03:25 +01:00
|
|
|
|
|
|
|
if (ndb_cache_check_time && m_rows_changed)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Rows has changed and util thread is running"));
|
|
|
|
if (thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Add share to list of tables to be invalidated"));
|
|
|
|
/* NOTE push_back allocates memory using transactions mem_root! */
|
|
|
|
thd_ndb->changed_tables.push_back(m_share, &thd->transaction.mem_root);
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_mutex_lock(&m_share->mutex);
|
|
|
|
DBUG_PRINT("info", ("Invalidating commit_count"));
|
|
|
|
m_share->commit_count= 0;
|
|
|
|
m_share->commit_count_lock++;
|
|
|
|
pthread_mutex_unlock(&m_share->mutex);
|
|
|
|
}
|
|
|
|
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
if (!--thd_ndb->lock_count)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("trans", ("Last external_lock"));
|
|
|
|
PRINT_OPTION_FLAGS(thd);
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
if (thd_ndb->stmt)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Unlock is done without a transaction commit / rollback.
|
|
|
|
This happens if the thread didn't update any rows
|
|
|
|
We must in this case close the transaction to release resources
|
|
|
|
*/
|
|
|
|
DBUG_PRINT("trans",("ending non-updating transaction"));
|
2004-12-30 19:56:09 +01:00
|
|
|
ndb->closeTransaction(m_active_trans);
|
2005-02-17 22:52:40 +01:00
|
|
|
thd_ndb->stmt= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
}
|
2004-09-21 14:52:56 +02:00
|
|
|
m_table_info= NULL;
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
/*
|
|
|
|
This is the place to make sure this handler instance
|
|
|
|
no longer are connected to the active transaction.
|
|
|
|
|
|
|
|
And since the handler is no longer part of the transaction
|
|
|
|
it can't have open cursors, ops or blobs pending.
|
|
|
|
*/
|
|
|
|
m_active_trans= NULL;
|
|
|
|
|
2004-09-07 12:54:31 +02:00
|
|
|
if (m_active_cursor)
|
|
|
|
DBUG_PRINT("warning", ("m_active_cursor != NULL"));
|
2004-09-13 14:46:38 +02:00
|
|
|
m_active_cursor= NULL;
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
if (m_multi_cursor)
|
|
|
|
DBUG_PRINT("warning", ("m_multi_cursor != NULL"));
|
|
|
|
m_multi_cursor= NULL;
|
|
|
|
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_blobs_pending)
|
2004-09-07 12:54:31 +02:00
|
|
|
DBUG_PRINT("warning", ("blobs_pending != 0"));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_blobs_pending= 0;
|
2004-09-13 14:46:38 +02:00
|
|
|
|
2004-11-03 15:53:26 +01:00
|
|
|
if (m_ops_pending)
|
2004-09-07 12:54:31 +02:00
|
|
|
DBUG_PRINT("warning", ("ops_pending != 0L"));
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
2006-06-12 09:37:19 +02:00
|
|
|
/*
|
|
|
|
Unlock the last row read in an open scan.
|
|
|
|
Rows are unlocked by default in ndb, but
|
|
|
|
for SELECT FOR UPDATE and SELECT LOCK WIT SHARE MODE
|
|
|
|
locks are kept if unlock_row() is not called.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::unlock_row()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("unlock_row");
|
|
|
|
|
|
|
|
DBUG_PRINT("info", ("Unlocking row"));
|
|
|
|
m_lock_tuple= false;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
2005-10-14 11:23:02 +02:00
|
|
|
Start a transaction for running a statement if one is not
|
|
|
|
already running in a transaction. This will be the case in
|
|
|
|
a BEGIN; COMMIT; block
|
|
|
|
When using LOCK TABLE's external_lock will start a transaction
|
|
|
|
since ndb does not currently does not support table locking
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2005-09-30 20:20:10 +02:00
|
|
|
int ha_ndbcluster::start_stmt(THD *thd, thr_lock_type lock_type)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
int error=0;
|
|
|
|
DBUG_ENTER("start_stmt");
|
|
|
|
PRINT_OPTION_FLAGS(thd);
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
2005-10-14 11:28:01 +02:00
|
|
|
NdbTransaction *trans= (thd_ndb->stmt)?thd_ndb->stmt:thd_ndb->all;
|
2004-04-15 09:14:14 +02:00
|
|
|
if (!trans){
|
2005-02-17 22:52:40 +01:00
|
|
|
Ndb *ndb= thd_ndb->ndb;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("trans",("Starting transaction stmt"));
|
2004-12-30 19:56:09 +01:00
|
|
|
trans= ndb->startTransaction();
|
2004-04-15 09:14:14 +02:00
|
|
|
if (trans == NULL)
|
2004-12-30 19:56:09 +01:00
|
|
|
ERR_RETURN(ndb->getNdbError());
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
no_uncommitted_rows_reset(thd);
|
2005-02-17 22:52:40 +01:00
|
|
|
thd_ndb->stmt= trans;
|
2007-01-31 22:38:06 +01:00
|
|
|
thd_ndb->query_state&= NDB_QUERY_NORMAL;
|
2005-02-17 22:52:40 +01:00
|
|
|
trans_register_ha(thd, FALSE, &ndbcluster_hton);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
m_active_trans= trans;
|
2004-05-10 14:12:28 +02:00
|
|
|
// Start of statement
|
2004-11-03 15:53:26 +01:00
|
|
|
m_retrieve_all_fields= FALSE;
|
2004-11-18 12:11:56 +01:00
|
|
|
m_retrieve_primary_key= FALSE;
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2005-03-15 15:03:25 +01:00
|
|
|
Commit a transaction started in NDB
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
int ndbcluster_commit(THD *thd, bool all)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
int res= 0;
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
|
|
|
Ndb *ndb= thd_ndb->ndb;
|
|
|
|
NdbTransaction *trans= all ? thd_ndb->all : thd_ndb->stmt;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("ndbcluster_commit");
|
|
|
|
DBUG_PRINT("transaction",("%s",
|
2005-03-15 15:03:25 +01:00
|
|
|
trans == thd_ndb->stmt ?
|
2004-04-15 09:14:14 +02:00
|
|
|
"stmt" : "all"));
|
|
|
|
DBUG_ASSERT(ndb && trans);
|
|
|
|
|
2004-11-17 09:15:53 +01:00
|
|
|
if (execute_commit(thd,trans) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
const NdbError err= trans->getNdbError();
|
2004-07-02 16:14:08 +02:00
|
|
|
const NdbOperation *error_op= trans->getNdbErrorOperation();
|
2005-03-15 15:03:25 +01:00
|
|
|
ERR_PRINT(err);
|
2004-04-15 09:14:14 +02:00
|
|
|
res= ndb_to_mysql_error(&err);
|
2005-03-15 15:03:25 +01:00
|
|
|
if (res != -1)
|
2004-07-02 16:14:08 +02:00
|
|
|
ndbcluster_print_error(res, error_op);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
ndb->closeTransaction(trans);
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if (all)
|
2005-02-21 10:53:19 +01:00
|
|
|
thd_ndb->all= NULL;
|
|
|
|
else
|
|
|
|
thd_ndb->stmt= NULL;
|
2005-03-15 15:03:25 +01:00
|
|
|
|
|
|
|
/* Clear commit_count for tables changed by transaction */
|
|
|
|
NDB_SHARE* share;
|
|
|
|
List_iterator_fast<NDB_SHARE> it(thd_ndb->changed_tables);
|
|
|
|
while ((share= it++))
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&share->mutex);
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("info", ("Invalidate commit_count for %s, share->commit_count: %lu",
|
|
|
|
share->table_name, (ulong) share->commit_count));
|
2005-03-15 15:03:25 +01:00
|
|
|
share->commit_count= 0;
|
|
|
|
share->commit_count_lock++;
|
|
|
|
pthread_mutex_unlock(&share->mutex);
|
|
|
|
}
|
|
|
|
thd_ndb->changed_tables.empty();
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Rollback a transaction started in NDB
|
|
|
|
*/
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
int ndbcluster_rollback(THD *thd, bool all)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
int res= 0;
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
|
|
|
Ndb *ndb= thd_ndb->ndb;
|
|
|
|
NdbTransaction *trans= all ? thd_ndb->all : thd_ndb->stmt;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("ndbcluster_rollback");
|
|
|
|
DBUG_PRINT("transaction",("%s",
|
2005-02-17 22:52:40 +01:00
|
|
|
trans == thd_ndb->stmt ?
|
2004-04-15 09:14:14 +02:00
|
|
|
"stmt" : "all"));
|
|
|
|
DBUG_ASSERT(ndb && trans);
|
|
|
|
|
2005-01-05 17:59:24 +01:00
|
|
|
if (trans->execute(NdbTransaction::Rollback) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
const NdbError err= trans->getNdbError();
|
2004-07-02 16:14:08 +02:00
|
|
|
const NdbOperation *error_op= trans->getNdbErrorOperation();
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_PRINT(err);
|
|
|
|
res= ndb_to_mysql_error(&err);
|
2004-07-02 16:14:08 +02:00
|
|
|
if (res != -1)
|
|
|
|
ndbcluster_print_error(res, error_op);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
ndb->closeTransaction(trans);
|
2005-02-21 10:53:19 +01:00
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if (all)
|
2005-02-21 10:53:19 +01:00
|
|
|
thd_ndb->all= NULL;
|
|
|
|
else
|
|
|
|
thd_ndb->stmt= NULL;
|
|
|
|
|
2005-03-15 15:03:25 +01:00
|
|
|
/* Clear list of tables changed by transaction */
|
|
|
|
thd_ndb->changed_tables.empty();
|
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2004-07-22 12:38:09 +02:00
|
|
|
Define NDB column based on Field.
|
|
|
|
Returns 0 or mysql error code.
|
|
|
|
Not member of ha_ndbcluster because NDBCOL cannot be declared.
|
2005-01-08 00:03:19 +01:00
|
|
|
|
|
|
|
MySQL text types with character set "binary" are mapped to true
|
|
|
|
NDB binary types without a character set. This may change.
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2004-07-22 12:38:09 +02:00
|
|
|
static int create_ndb_column(NDBCOL &col,
|
|
|
|
Field *field,
|
|
|
|
HA_CREATE_INFO *info)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-07-22 12:38:09 +02:00
|
|
|
// Set name
|
2007-04-11 15:21:11 +02:00
|
|
|
if (col.setName(field->field_name))
|
|
|
|
{
|
|
|
|
return (my_errno= errno);
|
|
|
|
}
|
2004-09-15 17:44:13 +02:00
|
|
|
// Get char set
|
|
|
|
CHARSET_INFO *cs= field->charset();
|
2004-07-22 12:38:09 +02:00
|
|
|
// Set type and sizes
|
|
|
|
const enum enum_field_types mysql_type= field->real_type();
|
|
|
|
switch (mysql_type) {
|
|
|
|
// Numeric types
|
2004-04-15 09:14:14 +02:00
|
|
|
case MYSQL_TYPE_TINY:
|
2004-07-22 12:38:09 +02:00
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
col.setType(NDBCOL::Tinyunsigned);
|
|
|
|
else
|
|
|
|
col.setType(NDBCOL::Tinyint);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2004-04-15 09:14:14 +02:00
|
|
|
case MYSQL_TYPE_SHORT:
|
2004-07-22 12:38:09 +02:00
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
col.setType(NDBCOL::Smallunsigned);
|
|
|
|
else
|
|
|
|
col.setType(NDBCOL::Smallint);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2004-04-15 09:14:14 +02:00
|
|
|
case MYSQL_TYPE_LONG:
|
2004-07-22 12:38:09 +02:00
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
col.setType(NDBCOL::Unsigned);
|
|
|
|
else
|
|
|
|
col.setType(NDBCOL::Int);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2004-04-15 09:14:14 +02:00
|
|
|
case MYSQL_TYPE_INT24:
|
2004-07-22 12:38:09 +02:00
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
col.setType(NDBCOL::Mediumunsigned);
|
|
|
|
else
|
|
|
|
col.setType(NDBCOL::Mediumint);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
|
|
|
case MYSQL_TYPE_LONGLONG:
|
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
col.setType(NDBCOL::Bigunsigned);
|
|
|
|
else
|
|
|
|
col.setType(NDBCOL::Bigint);
|
|
|
|
col.setLength(1);
|
2004-04-15 09:14:14 +02:00
|
|
|
break;
|
|
|
|
case MYSQL_TYPE_FLOAT:
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Float);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2004-04-15 09:14:14 +02:00
|
|
|
case MYSQL_TYPE_DOUBLE:
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Double);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2005-02-16 12:45:59 +01:00
|
|
|
case MYSQL_TYPE_DECIMAL:
|
|
|
|
{
|
|
|
|
Field_decimal *f= (Field_decimal*)field;
|
|
|
|
uint precision= f->pack_length();
|
|
|
|
uint scale= f->decimals();
|
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Olddecimalunsigned);
|
|
|
|
precision-= (scale > 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Olddecimal);
|
|
|
|
precision-= 1 + (scale > 0);
|
|
|
|
}
|
|
|
|
col.setPrecision(precision);
|
|
|
|
col.setScale(scale);
|
|
|
|
col.setLength(1);
|
|
|
|
}
|
|
|
|
break;
|
2005-02-16 21:19:42 +01:00
|
|
|
case MYSQL_TYPE_NEWDECIMAL:
|
|
|
|
{
|
|
|
|
Field_new_decimal *f= (Field_new_decimal*)field;
|
2005-05-05 17:06:49 +02:00
|
|
|
uint precision= f->precision;
|
2005-02-16 21:19:42 +01:00
|
|
|
uint scale= f->decimals();
|
|
|
|
if (field->flags & UNSIGNED_FLAG)
|
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Decimalunsigned);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Decimal);
|
|
|
|
}
|
|
|
|
col.setPrecision(precision);
|
|
|
|
col.setScale(scale);
|
|
|
|
col.setLength(1);
|
|
|
|
}
|
|
|
|
break;
|
2004-07-22 12:38:09 +02:00
|
|
|
// Date types
|
|
|
|
case MYSQL_TYPE_DATETIME:
|
|
|
|
col.setType(NDBCOL::Datetime);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2005-01-27 17:33:35 +01:00
|
|
|
case MYSQL_TYPE_DATE: // ?
|
|
|
|
col.setType(NDBCOL::Char);
|
|
|
|
col.setLength(field->pack_length());
|
|
|
|
break;
|
2004-07-22 12:38:09 +02:00
|
|
|
case MYSQL_TYPE_NEWDATE:
|
2005-01-08 16:57:51 +01:00
|
|
|
col.setType(NDBCOL::Date);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2004-07-22 12:38:09 +02:00
|
|
|
case MYSQL_TYPE_TIME:
|
2005-01-08 19:28:44 +01:00
|
|
|
col.setType(NDBCOL::Time);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
2005-01-27 17:33:35 +01:00
|
|
|
case MYSQL_TYPE_YEAR:
|
|
|
|
col.setType(NDBCOL::Year);
|
|
|
|
col.setLength(1);
|
|
|
|
break;
|
|
|
|
case MYSQL_TYPE_TIMESTAMP:
|
|
|
|
col.setType(NDBCOL::Timestamp);
|
|
|
|
col.setLength(1);
|
2004-07-22 12:38:09 +02:00
|
|
|
break;
|
|
|
|
// Char types
|
|
|
|
case MYSQL_TYPE_STRING:
|
2004-10-20 19:22:58 +02:00
|
|
|
if (field->pack_length() == 0)
|
2004-12-09 14:17:58 +01:00
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Bit);
|
|
|
|
col.setLength(1);
|
|
|
|
}
|
2005-01-08 00:03:19 +01:00
|
|
|
else if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2004-12-10 16:55:04 +01:00
|
|
|
{
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Binary);
|
2004-12-10 16:55:04 +01:00
|
|
|
col.setLength(field->pack_length());
|
2004-09-15 17:44:13 +02:00
|
|
|
}
|
2004-10-20 19:22:58 +02:00
|
|
|
else
|
2004-12-10 16:55:04 +01:00
|
|
|
{
|
|
|
|
col.setType(NDBCOL::Char);
|
|
|
|
col.setCharset(cs);
|
2004-10-20 19:22:58 +02:00
|
|
|
col.setLength(field->pack_length());
|
2004-12-10 16:55:04 +01:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
break;
|
2005-01-07 11:55:20 +01:00
|
|
|
case MYSQL_TYPE_VAR_STRING: // ?
|
|
|
|
case MYSQL_TYPE_VARCHAR:
|
|
|
|
{
|
|
|
|
Field_varstring* f= (Field_varstring*)field;
|
|
|
|
if (f->length_bytes == 1)
|
|
|
|
{
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2005-01-07 11:55:20 +01:00
|
|
|
col.setType(NDBCOL::Varbinary);
|
|
|
|
else {
|
|
|
|
col.setType(NDBCOL::Varchar);
|
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (f->length_bytes == 2)
|
|
|
|
{
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2005-01-07 11:55:20 +01:00
|
|
|
col.setType(NDBCOL::Longvarbinary);
|
|
|
|
else {
|
|
|
|
col.setType(NDBCOL::Longvarchar);
|
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return HA_ERR_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
col.setLength(field->field_length);
|
2004-09-15 17:44:13 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
break;
|
|
|
|
// Blob types (all come in as MYSQL_TYPE_BLOB)
|
|
|
|
mysql_type_tiny_blob:
|
|
|
|
case MYSQL_TYPE_TINY_BLOB:
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Blob);
|
2004-09-15 17:44:13 +02:00
|
|
|
else {
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Text);
|
2004-09-15 17:44:13 +02:00
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setInlineSize(256);
|
|
|
|
// No parts
|
|
|
|
col.setPartSize(0);
|
|
|
|
col.setStripeSize(0);
|
|
|
|
break;
|
2004-11-29 13:50:13 +01:00
|
|
|
//mysql_type_blob:
|
2005-10-25 08:20:39 +02:00
|
|
|
case MYSQL_TYPE_GEOMETRY:
|
2004-07-22 12:38:09 +02:00
|
|
|
case MYSQL_TYPE_BLOB:
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Blob);
|
2004-09-15 17:44:13 +02:00
|
|
|
else {
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Text);
|
2004-09-15 17:44:13 +02:00
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
{
|
2007-01-23 12:58:10 +01:00
|
|
|
Field_blob *field_blob= (Field_blob *)field;
|
|
|
|
/*
|
|
|
|
* max_data_length is 2^8-1, 2^16-1, 2^24-1 for tiny, blob, medium.
|
|
|
|
* Tinyblob gets no blob parts. The other cases are just a crude
|
|
|
|
* way to control part size and striping.
|
|
|
|
*
|
|
|
|
* In mysql blob(256) is promoted to blob(65535) so it does not
|
|
|
|
* in fact fit "inline" in NDB.
|
|
|
|
*/
|
|
|
|
if (field_blob->max_data_length() < (1 << 8))
|
|
|
|
goto mysql_type_tiny_blob;
|
|
|
|
else if (field_blob->max_data_length() < (1 << 16))
|
|
|
|
{
|
|
|
|
col.setInlineSize(256);
|
|
|
|
col.setPartSize(2000);
|
|
|
|
col.setStripeSize(16);
|
|
|
|
}
|
|
|
|
else if (field_blob->max_data_length() < (1 << 24))
|
|
|
|
goto mysql_type_medium_blob;
|
|
|
|
else
|
|
|
|
goto mysql_type_long_blob;
|
2004-07-22 12:38:09 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
mysql_type_medium_blob:
|
|
|
|
case MYSQL_TYPE_MEDIUM_BLOB:
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Blob);
|
2004-09-15 17:44:13 +02:00
|
|
|
else {
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Text);
|
2004-09-15 17:44:13 +02:00
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setInlineSize(256);
|
|
|
|
col.setPartSize(4000);
|
|
|
|
col.setStripeSize(8);
|
|
|
|
break;
|
|
|
|
mysql_type_long_blob:
|
|
|
|
case MYSQL_TYPE_LONG_BLOB:
|
2005-01-08 00:03:19 +01:00
|
|
|
if ((field->flags & BINARY_FLAG) && cs == &my_charset_bin)
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Blob);
|
2004-09-15 17:44:13 +02:00
|
|
|
else {
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setType(NDBCOL::Text);
|
2004-09-15 17:44:13 +02:00
|
|
|
col.setCharset(cs);
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setInlineSize(256);
|
|
|
|
col.setPartSize(8000);
|
|
|
|
col.setStripeSize(4);
|
|
|
|
break;
|
|
|
|
// Other types
|
|
|
|
case MYSQL_TYPE_ENUM:
|
|
|
|
col.setType(NDBCOL::Char);
|
|
|
|
col.setLength(field->pack_length());
|
|
|
|
break;
|
|
|
|
case MYSQL_TYPE_SET:
|
|
|
|
col.setType(NDBCOL::Char);
|
|
|
|
col.setLength(field->pack_length());
|
|
|
|
break;
|
2005-07-04 02:42:33 +02:00
|
|
|
case MYSQL_TYPE_BIT:
|
|
|
|
{
|
2006-04-05 02:54:58 +02:00
|
|
|
int no_of_bits= field->field_length;
|
2004-12-23 15:28:41 +01:00
|
|
|
col.setType(NDBCOL::Bit);
|
|
|
|
if (!no_of_bits)
|
|
|
|
col.setLength(1);
|
|
|
|
else
|
|
|
|
col.setLength(no_of_bits);
|
|
|
|
break;
|
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
case MYSQL_TYPE_NULL:
|
|
|
|
goto mysql_type_unsupported;
|
|
|
|
mysql_type_unsupported:
|
|
|
|
default:
|
|
|
|
return HA_ERR_UNSUPPORTED;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
// Set nullable and pk
|
|
|
|
col.setNullable(field->maybe_null());
|
|
|
|
col.setPrimaryKey(field->flags & PRI_KEY_FLAG);
|
|
|
|
// Set autoincrement
|
|
|
|
if (field->flags & AUTO_INCREMENT_FLAG)
|
|
|
|
{
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setAutoIncrement(TRUE);
|
|
|
|
ulonglong value= info->auto_increment_value ?
|
2004-09-02 17:51:44 +02:00
|
|
|
info->auto_increment_value : (ulonglong) 1;
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("info", ("Autoincrement key, initial: %s", llstr(value, buff)));
|
2004-07-22 12:38:09 +02:00
|
|
|
col.setAutoIncrementInitialValue(value);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-07-22 12:38:09 +02:00
|
|
|
else
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
col.setAutoIncrement(FALSE);
|
2004-07-22 12:38:09 +02:00
|
|
|
return 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Create a table in NDB Cluster
|
|
|
|
*/
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
static void ndb_set_fragmentation(NDBTAB &tab, TABLE *form, uint pk_length)
|
|
|
|
{
|
2006-06-27 10:02:58 +02:00
|
|
|
ha_rows max_rows= form->s->max_rows;
|
|
|
|
ha_rows min_rows= form->s->min_rows;
|
|
|
|
if (max_rows < min_rows)
|
|
|
|
max_rows= min_rows;
|
|
|
|
if (max_rows == (ha_rows)0) /* default setting, don't set fragmentation */
|
2005-02-11 22:33:52 +01:00
|
|
|
return;
|
|
|
|
/**
|
|
|
|
* get the number of fragments right
|
|
|
|
*/
|
|
|
|
uint no_fragments;
|
|
|
|
{
|
|
|
|
#if MYSQL_VERSION_ID >= 50000
|
|
|
|
uint acc_row_size= 25 + /*safety margin*/ 2;
|
|
|
|
#else
|
|
|
|
uint acc_row_size= pk_length*4;
|
|
|
|
/* add acc overhead */
|
|
|
|
if (pk_length <= 8) /* main page will set the limit */
|
|
|
|
acc_row_size+= 25 + /*safety margin*/ 2;
|
|
|
|
else /* overflow page will set the limit */
|
|
|
|
acc_row_size+= 4 + /*safety margin*/ 4;
|
|
|
|
#endif
|
|
|
|
ulonglong acc_fragment_size= 512*1024*1024;
|
2006-08-22 14:26:56 +02:00
|
|
|
/*
|
|
|
|
* if not --with-big-tables then max_rows is ulong
|
|
|
|
* the warning in this case is misleading though
|
|
|
|
*/
|
|
|
|
ulonglong big_max_rows = (ulonglong)max_rows;
|
2005-02-11 22:33:52 +01:00
|
|
|
#if MYSQL_VERSION_ID >= 50100
|
2006-08-22 14:26:56 +02:00
|
|
|
no_fragments= (big_max_rows*acc_row_size)/acc_fragment_size+1;
|
2005-02-11 22:33:52 +01:00
|
|
|
#else
|
2006-08-22 14:26:56 +02:00
|
|
|
no_fragments= ((big_max_rows*acc_row_size)/acc_fragment_size+1
|
2005-02-16 14:18:32 +01:00
|
|
|
+1/*correct rounding*/)/2;
|
2005-02-11 22:33:52 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
{
|
|
|
|
uint no_nodes= g_ndb_cluster_connection->no_db_nodes();
|
|
|
|
NDBTAB::FragmentType ftype;
|
|
|
|
if (no_fragments > 2*no_nodes)
|
|
|
|
{
|
|
|
|
ftype= NDBTAB::FragAllLarge;
|
|
|
|
if (no_fragments > 4*no_nodes)
|
2005-02-16 14:18:32 +01:00
|
|
|
push_warning(current_thd, MYSQL_ERROR::WARN_LEVEL_WARN, ER_UNKNOWN_ERROR,
|
|
|
|
"Ndb might have problems storing the max amount of rows specified");
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
else if (no_fragments > no_nodes)
|
|
|
|
ftype= NDBTAB::FragAllMedium;
|
|
|
|
else
|
|
|
|
ftype= NDBTAB::FragAllSmall;
|
|
|
|
tab.setFragmentType(ftype);
|
|
|
|
}
|
2006-06-27 10:02:58 +02:00
|
|
|
tab.setMaxRows(max_rows);
|
|
|
|
tab.setMinRows(min_rows);
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
int ha_ndbcluster::create(const char *name,
|
2005-02-16 14:18:32 +01:00
|
|
|
TABLE *form,
|
2006-12-14 23:51:37 +01:00
|
|
|
HA_CREATE_INFO *create_info)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
NDBTAB tab;
|
|
|
|
NDBCOL col;
|
2004-11-08 11:06:36 +01:00
|
|
|
uint pack_length, length, i, pk_length= 0;
|
2007-03-21 08:40:24 +01:00
|
|
|
const void *data= NULL, *pack_data= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
char name2[FN_HEADLEN];
|
2006-12-14 23:51:37 +01:00
|
|
|
bool create_from_engine= (create_info->table_options & HA_OPTION_CREATE_FROM_ENGINE);
|
2005-11-25 11:57:13 +01:00
|
|
|
|
2005-01-07 11:55:20 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::create");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("name: %s", name));
|
|
|
|
fn_format(name2, name, "", "",2); // Remove the .frm extension
|
|
|
|
set_dbname(name2);
|
2004-09-13 14:46:38 +02:00
|
|
|
set_tabname(name2);
|
|
|
|
|
2006-06-02 07:26:45 +02:00
|
|
|
if (current_thd->lex->sql_command == SQLCOM_TRUNCATE)
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Dropping and re-creating table for TRUNCATE"));
|
|
|
|
if ((my_errno= delete_table(name)))
|
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
}
|
2004-09-13 14:46:38 +02:00
|
|
|
if (create_from_engine)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Table alreay exists in NDB and frm file has been created by
|
|
|
|
caller.
|
|
|
|
Do Ndb specific stuff, such as create a .ndb file
|
|
|
|
*/
|
|
|
|
my_errno= write_ndb_file();
|
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_PRINT("table", ("name: %s", m_tabname));
|
2007-04-11 15:21:11 +02:00
|
|
|
if (tab.setName(m_tabname))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2006-12-14 23:51:37 +01:00
|
|
|
tab.setLogging(!(create_info->options & HA_LEX_CREATE_TMP_TABLE));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
// Save frm data for this table
|
|
|
|
if (readfrm(name, &data, &length))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
if (packfrm(data, length, &pack_data, &pack_length))
|
2007-03-21 08:40:24 +01:00
|
|
|
{
|
|
|
|
my_free((char*)data, MYF(0));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(2);
|
2007-03-21 08:40:24 +01:00
|
|
|
}
|
|
|
|
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("info", ("setFrm data: 0x%lx len: %d", (long) pack_data, pack_length));
|
2004-04-15 09:14:14 +02:00
|
|
|
tab.setFrm(pack_data, pack_length);
|
|
|
|
my_free((char*)data, MYF(0));
|
|
|
|
my_free((char*)pack_data, MYF(0));
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
for (i= 0; i < form->s->fields; i++)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
Field *field= form->field[i];
|
2007-03-22 19:32:07 +01:00
|
|
|
DBUG_PRINT("info", ("name: %s type: %u pack_length: %d",
|
2004-04-15 09:14:14 +02:00
|
|
|
field->field_name, field->real_type(),
|
2005-02-16 14:18:32 +01:00
|
|
|
field->pack_length()));
|
2006-12-14 23:51:37 +01:00
|
|
|
if ((my_errno= create_ndb_column(col, field, create_info)))
|
2004-07-22 12:38:09 +02:00
|
|
|
DBUG_RETURN(my_errno);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (tab.addColumn(col))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2005-07-04 02:42:33 +02:00
|
|
|
if (col.getPrimaryKey())
|
2004-11-08 11:06:36 +01:00
|
|
|
pk_length += (field->pack_length() + 3) / 4;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// No primary key, create shadow key as 64 bit, auto increment
|
2005-01-06 12:00:13 +01:00
|
|
|
if (form->s->primary_key == MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Generating shadow key"));
|
2007-04-11 15:21:11 +02:00
|
|
|
if (col.setName("$PK"))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
col.setType(NdbDictionary::Column::Bigunsigned);
|
|
|
|
col.setLength(1);
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
col.setNullable(FALSE);
|
2004-04-15 09:14:14 +02:00
|
|
|
col.setPrimaryKey(TRUE);
|
|
|
|
col.setAutoIncrement(TRUE);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (tab.addColumn(col))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2004-11-08 11:06:36 +01:00
|
|
|
pk_length += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure that blob tables don't have to big part size
|
2005-01-06 12:00:13 +01:00
|
|
|
for (i= 0; i < form->s->fields; i++)
|
2004-11-08 11:06:36 +01:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The extra +7 concists
|
|
|
|
* 2 - words from pk in blob table
|
|
|
|
* 5 - from extra words added by tup/dict??
|
|
|
|
*/
|
|
|
|
switch (form->field[i]->real_type()) {
|
2005-10-25 08:20:39 +02:00
|
|
|
case MYSQL_TYPE_GEOMETRY:
|
2004-11-08 11:06:36 +01:00
|
|
|
case MYSQL_TYPE_BLOB:
|
|
|
|
case MYSQL_TYPE_MEDIUM_BLOB:
|
|
|
|
case MYSQL_TYPE_LONG_BLOB:
|
|
|
|
{
|
2006-12-14 23:51:37 +01:00
|
|
|
NdbDictionary::Column * column= tab.getColumn(i);
|
|
|
|
int size= pk_length + (column->getPartSize()+3)/4 + 7;
|
2005-07-04 02:42:33 +02:00
|
|
|
if (size > NDB_MAX_TUPLE_SIZE_IN_WORDS &&
|
2005-02-16 14:18:32 +01:00
|
|
|
(pk_length+7) < NDB_MAX_TUPLE_SIZE_IN_WORDS)
|
2004-11-08 11:06:36 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
size= NDB_MAX_TUPLE_SIZE_IN_WORDS - pk_length - 7;
|
2006-12-14 23:51:37 +01:00
|
|
|
column->setPartSize(4*size);
|
2004-11-08 11:06:36 +01:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* If size > NDB_MAX and pk_length+7 >= NDB_MAX
|
|
|
|
* then the table can't be created anyway, so skip
|
|
|
|
* changing part size, and have error later
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
ndb_set_fragmentation(tab, form, pk_length);
|
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
if ((my_errno= check_ndb_connection()))
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
|
|
|
|
// Create the table in NDB
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-09-13 14:46:38 +02:00
|
|
|
if (dict->createTable(tab) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
const NdbError err= dict->getNdbError();
|
|
|
|
ERR_PRINT(err);
|
|
|
|
my_errno= ndb_to_mysql_error(&err);
|
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
}
|
|
|
|
DBUG_PRINT("info", ("Table %s/%s created successfully",
|
|
|
|
m_dbname, m_tabname));
|
2004-04-30 12:25:31 +02:00
|
|
|
|
2004-08-19 11:10:35 +02:00
|
|
|
// Create secondary indexes
|
2005-03-22 17:42:08 +01:00
|
|
|
my_errno= build_index_list(ndb, form, ILBP_CREATE);
|
2004-08-18 19:13:39 +02:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
if (!my_errno)
|
|
|
|
my_errno= write_ndb_file();
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(my_errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-30 12:25:31 +02:00
|
|
|
int ha_ndbcluster::create_ordered_index(const char *name,
|
2005-02-16 14:18:32 +01:00
|
|
|
KEY *key_info)
|
2004-04-30 12:25:31 +02:00
|
|
|
{
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::create_ordered_index");
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
DBUG_RETURN(create_index(name, key_info, FALSE));
|
2004-04-30 12:25:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int ha_ndbcluster::create_unique_index(const char *name,
|
2005-02-16 14:18:32 +01:00
|
|
|
KEY *key_info)
|
2004-04-30 12:25:31 +02:00
|
|
|
{
|
|
|
|
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::create_unique_index");
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
DBUG_RETURN(create_index(name, key_info, TRUE));
|
2004-04-30 12:25:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Create an index in NDB Cluster
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::create_index(const char *name,
|
2005-02-16 14:18:32 +01:00
|
|
|
KEY *key_info,
|
|
|
|
bool unique)
|
2004-04-30 12:25:31 +02:00
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
NdbDictionary::Dictionary *dict= ndb->getDictionary();
|
2004-04-15 09:14:14 +02:00
|
|
|
KEY_PART_INFO *key_part= key_info->key_part;
|
|
|
|
KEY_PART_INFO *end= key_part + key_info->key_parts;
|
|
|
|
|
2004-12-20 15:12:42 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::create_index");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("name: %s ", name));
|
2004-04-30 12:25:31 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbDictionary::Index ndb_index(name);
|
2004-04-30 12:25:31 +02:00
|
|
|
if (unique)
|
2004-04-15 09:14:14 +02:00
|
|
|
ndb_index.setType(NdbDictionary::Index::UniqueHashIndex);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ndb_index.setType(NdbDictionary::Index::OrderedIndex);
|
|
|
|
// TODO Only temporary ordered indexes supported
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
ndb_index.setLogging(FALSE);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb_index.setTable(m_tabname))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
for (; key_part != end; key_part++)
|
|
|
|
{
|
|
|
|
Field *field= key_part->field;
|
|
|
|
DBUG_PRINT("info", ("attr: %s", field->field_name));
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb_index.addColumnName(field->field_name))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dict->createIndex(ndb_index))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
|
|
|
|
// Success
|
|
|
|
DBUG_PRINT("info", ("Created index %s", name));
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Rename a table in NDB Cluster
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::rename_table(const char *from, const char *to)
|
|
|
|
{
|
2004-12-06 14:51:10 +01:00
|
|
|
NDBDICT *dict;
|
2004-04-15 09:14:14 +02:00
|
|
|
char new_tabname[FN_HEADLEN];
|
2006-05-05 17:18:47 +02:00
|
|
|
char new_dbname[FN_HEADLEN];
|
2004-12-06 14:51:10 +01:00
|
|
|
const NDBTAB *orig_tab;
|
|
|
|
int result;
|
2006-05-05 17:18:47 +02:00
|
|
|
bool recreate_indexes= FALSE;
|
|
|
|
NDBDICT::List index_list;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("ha_ndbcluster::rename_table");
|
2004-12-06 14:51:10 +01:00
|
|
|
DBUG_PRINT("info", ("Renaming %s to %s", from, to));
|
2004-04-15 09:14:14 +02:00
|
|
|
set_dbname(from);
|
2006-05-05 17:18:47 +02:00
|
|
|
set_dbname(to, new_dbname);
|
2004-04-15 09:14:14 +02:00
|
|
|
set_tabname(from);
|
|
|
|
set_tabname(to, new_tabname);
|
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
if (check_ndb_connection())
|
|
|
|
DBUG_RETURN(my_errno= HA_ERR_NO_CONNECTION);
|
|
|
|
|
2005-01-03 11:56:51 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
dict= ndb->getDictionary();
|
2004-12-06 14:51:10 +01:00
|
|
|
if (!(orig_tab= dict->getTable(m_tabname)))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
2005-04-28 09:39:29 +02:00
|
|
|
// Check if thread has stale local cache
|
|
|
|
if (orig_tab->getObjectStatus() == NdbDictionary::Object::Invalid)
|
|
|
|
{
|
|
|
|
dict->removeCachedTable(m_tabname);
|
|
|
|
if (!(orig_tab= dict->getTable(m_tabname)))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
}
|
2006-05-05 17:18:47 +02:00
|
|
|
if (my_strcasecmp(system_charset_info, new_dbname, m_dbname))
|
|
|
|
{
|
|
|
|
dict->listIndexes(index_list, m_tabname);
|
|
|
|
recreate_indexes= TRUE;
|
|
|
|
}
|
|
|
|
|
2004-12-06 14:51:10 +01:00
|
|
|
m_table= (void *)orig_tab;
|
|
|
|
// Change current database to that of target table
|
|
|
|
set_dbname(to);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2004-12-06 14:51:10 +01:00
|
|
|
if (!(result= alter_table_name(new_tabname)))
|
2004-09-13 14:46:38 +02:00
|
|
|
{
|
2004-12-06 14:51:10 +01:00
|
|
|
// Rename .ndb file
|
|
|
|
result= handler::rename_table(from, to);
|
2004-09-13 14:46:38 +02:00
|
|
|
}
|
2004-12-06 14:51:10 +01:00
|
|
|
|
2006-05-05 17:18:47 +02:00
|
|
|
// If we are moving tables between databases, we need to recreate
|
|
|
|
// indexes
|
|
|
|
if (recreate_indexes)
|
|
|
|
{
|
|
|
|
const NDBTAB *new_tab;
|
|
|
|
set_tabname(to);
|
|
|
|
if (!(new_tab= dict->getTable(m_tabname)))
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < index_list.count; i++) {
|
|
|
|
NDBDICT::List::Element& index_el = index_list.elements[i];
|
|
|
|
set_dbname(from);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2006-05-05 17:18:47 +02:00
|
|
|
const NDBINDEX * index= dict->getIndex(index_el.name, *new_tab);
|
|
|
|
set_dbname(to);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2006-05-05 17:18:47 +02:00
|
|
|
DBUG_PRINT("info", ("Creating index %s/%s",
|
|
|
|
m_dbname, index->getName()));
|
|
|
|
dict->createIndex(*index);
|
|
|
|
DBUG_PRINT("info", ("Dropping index %s/%s",
|
|
|
|
m_dbname, index->getName()));
|
|
|
|
|
|
|
|
set_dbname(from);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2006-05-05 17:18:47 +02:00
|
|
|
dict->dropIndex(*index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Rename a table in NDB Cluster using alter table
|
|
|
|
*/
|
|
|
|
|
2004-12-06 14:51:10 +01:00
|
|
|
int ha_ndbcluster::alter_table_name(const char *to)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-12-06 14:51:10 +01:00
|
|
|
const NDBTAB *orig_tab= (const NDBTAB *) m_table;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("alter_table_name_table");
|
|
|
|
|
2004-12-07 10:27:22 +01:00
|
|
|
NdbDictionary::Table new_tab= *orig_tab;
|
2007-04-11 15:21:11 +02:00
|
|
|
if (new_tab.setName(to))
|
|
|
|
{
|
|
|
|
DBUG_RETURN(my_errno= errno);
|
|
|
|
}
|
2004-12-06 14:51:10 +01:00
|
|
|
if (dict->alterTable(new_tab) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
|
|
|
|
m_table= NULL;
|
2004-09-21 14:52:56 +02:00
|
|
|
m_table_info= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2005-02-17 14:22:44 +01:00
|
|
|
Delete table from NDB Cluster
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::delete_table(const char *name)
|
|
|
|
{
|
2005-02-17 14:22:44 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::delete_table");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_PRINT("enter", ("name: %s", name));
|
|
|
|
set_dbname(name);
|
|
|
|
set_tabname(name);
|
2005-02-17 14:22:44 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
if (check_ndb_connection())
|
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
2005-02-17 14:22:44 +01:00
|
|
|
|
|
|
|
/* Call ancestor function to delete .ndb file */
|
2004-09-13 14:46:38 +02:00
|
|
|
handler::delete_table(name);
|
2005-02-17 14:22:44 +01:00
|
|
|
|
|
|
|
/* Drop the table from NDB */
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(drop_table());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2005-02-17 14:22:44 +01:00
|
|
|
Drop table in NDB Cluster
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::drop_table()
|
|
|
|
{
|
2006-06-14 12:54:28 +02:00
|
|
|
THD *thd= current_thd;
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
|
|
|
NdbDictionary::Dictionary *dict= ndb->getDictionary();
|
2005-02-17 14:22:44 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("drop_table");
|
|
|
|
DBUG_PRINT("enter", ("Deleting %s", m_tabname));
|
|
|
|
|
|
|
|
release_metadata();
|
2006-06-14 12:54:28 +02:00
|
|
|
while (dict->dropTable(m_tabname))
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
const NdbError err= dict->getNdbError();
|
2006-06-14 12:54:28 +02:00
|
|
|
switch (err.status)
|
|
|
|
{
|
|
|
|
case NdbError::TemporaryError:
|
|
|
|
if (!thd->killed)
|
|
|
|
continue; // retry indefinitly
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2005-02-17 14:22:44 +01:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
2006-06-14 12:54:28 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-15 21:10:31 +02:00
|
|
|
ulonglong ha_ndbcluster::get_auto_increment()
|
2004-04-29 14:38:35 +02:00
|
|
|
{
|
2007-11-06 10:28:11 +01:00
|
|
|
uint cache_size;
|
2004-09-15 21:10:31 +02:00
|
|
|
Uint64 auto_value;
|
2007-11-05 20:18:22 +01:00
|
|
|
THD *thd= current_thd;
|
|
|
|
Uint64 step= thd->variables.auto_increment_increment;
|
|
|
|
Uint64 start= thd->variables.auto_increment_offset;
|
2004-07-02 11:50:28 +02:00
|
|
|
DBUG_ENTER("get_auto_increment");
|
|
|
|
DBUG_PRINT("enter", ("m_tabname: %s", m_tabname));
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb= get_ndb();
|
2005-04-08 16:49:14 +02:00
|
|
|
|
2005-04-07 20:08:01 +02:00
|
|
|
if (m_rows_inserted > m_rows_to_insert)
|
2005-05-07 22:04:52 +02:00
|
|
|
{
|
2005-04-07 20:08:01 +02:00
|
|
|
/* We guessed too low */
|
|
|
|
m_rows_to_insert+= m_autoincrement_prefetch;
|
2005-05-07 22:04:52 +02:00
|
|
|
}
|
2007-11-06 10:28:11 +01:00
|
|
|
uint remaining= m_rows_to_insert - m_rows_inserted;
|
|
|
|
uint min_prefetch=
|
2007-11-05 20:18:22 +01:00
|
|
|
(remaining < thd->variables.ndb_autoincrement_prefetch_sz) ?
|
|
|
|
thd->variables.ndb_autoincrement_prefetch_sz
|
|
|
|
: remaining;
|
2007-11-06 10:28:11 +01:00
|
|
|
cache_size= ((remaining < m_autoincrement_prefetch) ?
|
|
|
|
min_prefetch
|
|
|
|
: remaining);
|
2005-04-15 16:14:08 +02:00
|
|
|
uint retries= NDB_AUTO_INCREMENT_RETRIES;
|
2007-05-14 12:15:27 +02:00
|
|
|
int retry_sleep= 30; /* 30 milliseconds, transaction */
|
|
|
|
for (;;)
|
2005-10-14 11:23:02 +02:00
|
|
|
{
|
2007-05-14 12:15:27 +02:00
|
|
|
if (m_skip_auto_increment &&
|
|
|
|
ndb->readAutoIncrementValue((const NDBTAB *) m_table, auto_value) ||
|
|
|
|
ndb->getAutoIncrementValue((const NDBTAB *) m_table,
|
2007-05-28 15:23:28 +02:00
|
|
|
auto_value, cache_size, step, start))
|
2007-05-14 12:15:27 +02:00
|
|
|
{
|
|
|
|
if (--retries &&
|
2007-10-25 08:40:42 +02:00
|
|
|
ndb->getNdbError().status == NdbError::TemporaryError)
|
2007-05-14 12:15:27 +02:00
|
|
|
{
|
|
|
|
my_sleep(retry_sleep);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const NdbError err= ndb->getNdbError();
|
|
|
|
sql_print_error("Error %lu in ::get_auto_increment(): %s",
|
|
|
|
(ulong) err.code, err.message);
|
|
|
|
DBUG_RETURN(~(ulonglong) 0);
|
|
|
|
}
|
|
|
|
break;
|
2005-10-14 11:23:02 +02:00
|
|
|
}
|
2004-07-02 11:50:28 +02:00
|
|
|
DBUG_RETURN((longlong)auto_value);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Constructor for the NDB Cluster table handler
|
|
|
|
*/
|
|
|
|
|
|
|
|
ha_ndbcluster::ha_ndbcluster(TABLE *table_arg):
|
A fix and a test case for Bug#10760 and complementary cleanups.
The idea of the patch
is that every cursor gets its own lock id for table level locking.
Thus cursors are protected from updates performed within the same
connection. Additionally a list of transient (must be closed at
commit) cursors is maintained and all transient cursors are closed
when necessary. Lastly, this patch adds support for deadlock
timeouts to TLL locking when using cursors.
+ post-review fixes.
include/thr_lock.h:
- add a notion of lock owner to table level locking. When using
cursors, lock owner can not be identified by a thread id any more,
as we must protect cursors from updates issued within the same
connection (thread). So, each cursor has its own lock identifier to
use with table level locking.
- extend return values of thr_lock and thr_multi_lock with
THR_LOCK_TIMEOUT and THR_LOCK_DEADLOCK, since these conditions
are now possible (see comments to thr_lock.c)
mysys/thr_lock.c:
Better support for cursors:
- use THR_LOCK_OWNER * as lock identifier, not pthread_t.
- check and return an error for a trivial deadlock case, when an
update statement is issued to a table locked by a cursor which has
been previously opened in the same connection.
- add support for locking timeouts: with use of cursors, trivial
deadlocks can occur. For now the only remedy is the lock wait timeout,
which is initialized from a new global variable 'table_lock_wait_timeout'
Example of a deadlock (assuming the storage engine does not downgrade
locks):
con1: open cursor for select * from t1;
con2: open cursor for select * from t2;
con1: update t2 set id=id*2; -- blocked
con2: update t1 set id=id*2; -- deadlock
Lock timeouts are active only if a connection is using cursors.
- the check in the wait_for_lock loop has been changed from
data->cond != cond to data->cond != 0. data->cond is zeroed
in every place it's changed.
- added comments
sql/examples/ha_archive.cc:
- extend the handlerton with the info about cursor behaviour at commit.
sql/examples/ha_archive.h:
- ctor moved to .cc to make use of archive handlerton
sql/examples/ha_example.cc:
- add handlerton instance, init handler::ht with it
sql/examples/ha_example.h:
- ctor moved to .cc to make use of ha_example handlerton
sql/examples/ha_tina.cc:
- add handlerton instance, init handler::ht with it
sql/examples/ha_tina.h:
- ctor moved to .cc to make use of CSV handlerton
sql/ha_berkeley.cc:
- init handlerton::flags and handler::ht
sql/ha_berkeley.h:
- ctor moved to .cc to make use of BerkeleyDB handlerton
sql/ha_blackhole.cc:
- add handlerton instance, init handler::ht with it
sql/ha_blackhole.h:
- ctor moved to .cc to make use of blackhole handlerton
sql/ha_federated.cc:
- add handlerton instance, init handler::ht with it
sql/ha_federated.h:
- ctor moved to .cc to make use of federated handlerton
sql/ha_heap.cc:
- add handlerton instance, init handler::ht with it
sql/ha_heap.h:
- ctor moved to .cc to make use of ha_heap handlerton
sql/ha_innodb.cc:
- init handlerton::flags and handler::ht of innobase storage engine
sql/ha_innodb.h:
- ctor moved to .cc to make use of archive handlerton
sql/ha_myisam.cc:
- add handlerton instance, init handler::ht with it
sql/ha_myisam.h:
- ctor moved to .cc to make use of MyISAM handlerton
sql/ha_myisammrg.cc:
- init handler::ht in the ctor
sql/ha_myisammrg.h:
- ctor moved to .cc to make use of MyISAM MERGE handlerton
sql/ha_ndbcluster.cc:
- init handlerton::flags and handler::ht
sql/handler.cc:
- drop support for ISAM storage engine, which was removed from 5.0
- close all "transient" cursors at COMMIT/ROLLBACK. A "transient"
SQL level cursor is a cursor that uses tables that have a transaction-
specific state.
sql/handler.h:
- extend struct handlerton with flags, add handlerton *ht to every
handler instance.
sql/lock.cc:
- extend mysql_lock_tables to send error to the client if
thr_multi_lock returns a timeout or a deadlock error.
sql/mysqld.cc:
- add server option --table_lock_wait_timeout (in seconds)
sql/set_var.cc:
- add new global variable 'table_lock_wait_timeout' to specify
a wait timeout for table-level locks of MySQL (in seconds). The default
timeout is 50 seconds. The timeout is active only if the connection
has open cursors.
sql/sql_class.cc:
- implement Statement_map::close_transient_cursors
- safety suggests that we need an assert ensuring
llock_info->n_cursors is functioning properly, adjust destruction of
the Statement_map to allow such assert in THD::~THD
sql/sql_class.h:
- add support for Cursors registry to Statement map.
sql/sql_prepare.cc:
- maintain a list of cursors that must be closed at commit/rollback.
sql/sql_select.cc:
- extend class Cursor to support specific at-COMMIT/ROLLBACK behavior.
If a cursor uses tables of a storage engine that
invalidates all open tables at COMMIT/ROLLBACK, it must be closed
before COMMIT/ROLLBACK is executed.
sql/sql_select.h:
- add an own lock_id and commit/rollback status flag to class Cursor
tests/mysql_client_test.c:
A test case for Bug#10760 and complementary issues: test a simple
deadlock case too.
mysql-test/var:
New BitKeeper file ``mysql-test/var''
2005-07-19 20:21:12 +02:00
|
|
|
handler(&ndbcluster_hton, table_arg),
|
2004-04-15 09:14:14 +02:00
|
|
|
m_active_trans(NULL),
|
|
|
|
m_active_cursor(NULL),
|
|
|
|
m_table(NULL),
|
2005-04-27 18:17:41 +02:00
|
|
|
m_table_version(-1),
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
m_table_info(NULL),
|
2004-04-15 09:14:14 +02:00
|
|
|
m_table_flags(HA_REC_NOT_IN_SEQ |
|
2005-02-16 14:18:32 +01:00
|
|
|
HA_NULL_IN_KEY |
|
|
|
|
HA_AUTO_PART_KEY |
|
|
|
|
HA_NO_PREFIX_CHAR_KEYS |
|
|
|
|
HA_NEED_READ_RANGE_BUFFER |
|
2005-10-25 08:20:39 +02:00
|
|
|
HA_CAN_GEOMETRY |
|
2006-07-04 11:43:06 +02:00
|
|
|
HA_CAN_BIT_FIELD |
|
2007-05-09 14:14:27 +02:00
|
|
|
HA_PARTIAL_COLUMN_READ |
|
|
|
|
HA_EXTERNAL_AUTO_INCREMENT),
|
2004-09-06 20:30:57 +02:00
|
|
|
m_share(0),
|
true,false -> TRUE, FALSE
Simple fixes/optimization of things discovered during review of new pushed code
include/my_sys.h:
Ensure that clear_alloc_root() interacts correctly with alloc_root_inited()
mysys/hash.c:
More comments
Simple optimization (merge identical code)
mysys/my_bitmap.c:
Change inline -> static inline
sql/examples/ha_archive.cc:
Fixed compiler warning
sql/ha_ndbcluster.cc:
true,false -> TRUE, FALSE
Change if (false) -> #ifdef NOT_USED
sql/ha_ndbcluster.h:
true,false -> TRUE, FALSE
sql/handler.cc:
More comments
Remove not needed initializations.
#ifdef not used code
sql/item_cmpfunc.h:
true,false -> TRUE, FALSE
sql/item_strfunc.cc:
Move local variables to function beginning
Remove wrong comments
sql/log_event.h:
true,false -> TRUE, FALSE
sql/sql_base.cc:
true,false -> TRUE, FALSE
More comments
sql/sql_help.cc:
true,false -> TRUE, FALSE
sql/sql_lex.cc:
Simple optimization of new code
sql/sql_parse.cc:
true,false -> TRUE, FALSE
sql/sql_prepare.cc:
true,false -> TRUE, FALSE
sql/sql_table.cc:
true,false -> TRUE, FALSE
sql/sql_yacc.yy:
true,false -> TRUE, FALSE
2004-10-14 17:03:46 +02:00
|
|
|
m_use_write(FALSE),
|
2004-11-18 12:11:56 +01:00
|
|
|
m_ignore_dup_key(FALSE),
|
2006-03-23 09:48:46 +01:00
|
|
|
m_has_unique_index(FALSE),
|
2004-11-03 15:53:26 +01:00
|
|
|
m_primary_key_update(FALSE),
|
|
|
|
m_retrieve_all_fields(FALSE),
|
2004-11-18 12:11:56 +01:00
|
|
|
m_retrieve_primary_key(FALSE),
|
2005-04-13 12:42:23 +02:00
|
|
|
m_rows_to_insert((ha_rows) 1),
|
|
|
|
m_rows_inserted((ha_rows) 0),
|
|
|
|
m_bulk_insert_rows((ha_rows) 1024),
|
2005-04-13 16:24:17 +02:00
|
|
|
m_rows_changed((ha_rows) 0),
|
2004-11-03 15:53:26 +01:00
|
|
|
m_bulk_insert_not_flushed(FALSE),
|
2007-04-04 12:50:39 +02:00
|
|
|
m_delete_cannot_batch(FALSE),
|
|
|
|
m_update_cannot_batch(FALSE),
|
2004-11-03 15:53:26 +01:00
|
|
|
m_ops_pending(0),
|
|
|
|
m_skip_auto_increment(TRUE),
|
|
|
|
m_blobs_pending(0),
|
2006-06-21 09:36:50 +02:00
|
|
|
m_blobs_offset(0),
|
2004-11-03 15:53:26 +01:00
|
|
|
m_blobs_buffer(0),
|
|
|
|
m_blobs_buffer_size(0),
|
2004-11-17 09:15:53 +01:00
|
|
|
m_dupkey((uint) -1),
|
|
|
|
m_ha_not_exact_count(FALSE),
|
|
|
|
m_force_send(TRUE),
|
2007-11-05 20:18:22 +01:00
|
|
|
m_autoincrement_prefetch((ha_rows) NDB_DEFAULT_AUTO_PREFETCH),
|
2004-11-22 14:41:46 +01:00
|
|
|
m_transaction_on(TRUE),
|
2007-04-23 11:25:33 +02:00
|
|
|
m_cond(NULL),
|
2005-02-11 22:05:24 +01:00
|
|
|
m_multi_cursor(NULL)
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
2004-04-30 12:25:31 +02:00
|
|
|
int i;
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster");
|
|
|
|
|
|
|
|
m_tabname[0]= '\0';
|
|
|
|
m_dbname[0]= '\0';
|
|
|
|
|
2004-09-14 14:47:34 +02:00
|
|
|
records= ~(ha_rows)0; // uninitialized
|
2004-04-15 09:14:14 +02:00
|
|
|
block_size= 1024;
|
|
|
|
|
2004-04-30 12:25:31 +02:00
|
|
|
for (i= 0; i < MAX_KEY; i++)
|
|
|
|
{
|
2005-01-26 11:31:46 +01:00
|
|
|
m_index[i].type= UNDEFINED_INDEX;
|
|
|
|
m_index[i].unique_index= NULL;
|
|
|
|
m_index[i].index= NULL;
|
|
|
|
m_index[i].unique_index_attrid_map= NULL;
|
2004-04-30 12:25:31 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Destructor for NDB Cluster table handler
|
|
|
|
*/
|
|
|
|
|
|
|
|
ha_ndbcluster::~ha_ndbcluster()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("~ha_ndbcluster");
|
|
|
|
|
2004-09-03 17:11:09 +02:00
|
|
|
if (m_share)
|
|
|
|
free_share(m_share);
|
2004-04-15 09:14:14 +02:00
|
|
|
release_metadata();
|
2004-11-03 15:53:26 +01:00
|
|
|
my_free(m_blobs_buffer, MYF(MY_ALLOW_ZERO_PTR));
|
|
|
|
m_blobs_buffer= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
// Check for open cursor/transaction
|
2004-09-07 12:54:31 +02:00
|
|
|
if (m_active_cursor) {
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ASSERT(m_active_cursor == NULL);
|
2004-09-07 12:54:31 +02:00
|
|
|
if (m_active_trans) {
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ASSERT(m_active_trans == NULL);
|
|
|
|
|
2007-04-23 11:25:33 +02:00
|
|
|
// Discard any generated condition
|
|
|
|
DBUG_PRINT("info", ("Deleting generated condition"));
|
|
|
|
if (m_cond)
|
|
|
|
{
|
|
|
|
delete m_cond;
|
|
|
|
m_cond= NULL;
|
|
|
|
}
|
2004-12-17 21:13:22 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
Open a table for further use
|
|
|
|
- fetch metadata for this table from NDB
|
|
|
|
- check that table exists
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::open(const char *name, int mode, uint test_if_locked)
|
|
|
|
{
|
2004-10-04 01:20:05 +02:00
|
|
|
int res;
|
2004-04-15 09:14:14 +02:00
|
|
|
KEY *key;
|
|
|
|
DBUG_ENTER("open");
|
|
|
|
DBUG_PRINT("enter", ("name: %s mode: %d test_if_locked: %d",
|
|
|
|
name, mode, test_if_locked));
|
|
|
|
|
|
|
|
// Setup ref_length to make room for the whole
|
|
|
|
// primary key to be written in the ref variable
|
|
|
|
|
2005-01-06 12:00:13 +01:00
|
|
|
if (table->s->primary_key != MAX_KEY)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-01-06 12:00:13 +01:00
|
|
|
key= table->key_info+table->s->primary_key;
|
2004-04-15 09:14:14 +02:00
|
|
|
ref_length= key->key_length;
|
|
|
|
DBUG_PRINT("info", (" ref_length: %d", ref_length));
|
|
|
|
}
|
|
|
|
// Init table lock structure
|
|
|
|
if (!(m_share=get_share(name)))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
thr_lock_data_init(&m_share->lock,&m_lock,(void*) 0);
|
|
|
|
|
|
|
|
set_dbname(name);
|
|
|
|
set_tabname(name);
|
|
|
|
|
2007-05-11 08:07:42 +02:00
|
|
|
if ((res= check_ndb_connection()) ||
|
|
|
|
(res= get_metadata(name)))
|
|
|
|
{
|
|
|
|
free_share(m_share);
|
|
|
|
m_share= 0;
|
|
|
|
DBUG_RETURN(res);
|
2004-09-03 17:11:09 +02:00
|
|
|
}
|
2007-05-11 08:07:42 +02:00
|
|
|
while (1)
|
2006-10-23 15:46:35 +02:00
|
|
|
{
|
|
|
|
Ndb *ndb= get_ndb();
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
2007-05-11 08:07:42 +02:00
|
|
|
res= ndb_to_mysql_error(&ndb->getNdbError());
|
|
|
|
break;
|
2007-04-11 15:21:11 +02:00
|
|
|
}
|
2006-10-23 16:18:57 +02:00
|
|
|
struct Ndb_statistics stat;
|
|
|
|
res= ndb_get_table_statistics(NULL, false, ndb, m_tabname, &stat);
|
|
|
|
records= stat.row_count;
|
2006-10-23 15:46:35 +02:00
|
|
|
if(!res)
|
|
|
|
res= info(HA_STATUS_CONST);
|
2007-05-11 08:07:42 +02:00
|
|
|
break;
|
2006-10-23 15:46:35 +02:00
|
|
|
}
|
2007-05-11 08:07:42 +02:00
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
free_share(m_share);
|
|
|
|
m_share= 0;
|
|
|
|
release_metadata();
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Close the table
|
|
|
|
- release resources setup by open()
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::close(void)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("close");
|
2004-09-03 17:11:09 +02:00
|
|
|
free_share(m_share); m_share= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
release_metadata();
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-14 14:47:34 +02:00
|
|
|
Thd_ndb* ha_ndbcluster::seize_thd_ndb()
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
Thd_ndb *thd_ndb;
|
|
|
|
DBUG_ENTER("seize_thd_ndb");
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-14 14:47:34 +02:00
|
|
|
thd_ndb= new Thd_ndb();
|
2007-04-11 15:21:11 +02:00
|
|
|
if (thd_ndb == NULL)
|
|
|
|
{
|
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
|
|
|
return NULL;
|
|
|
|
}
|
2005-03-22 17:42:08 +01:00
|
|
|
thd_ndb->ndb->getDictionary()->set_local_table_data_size(
|
|
|
|
sizeof(Ndb_local_table_statistics)
|
|
|
|
);
|
2004-09-14 14:47:34 +02:00
|
|
|
if (thd_ndb->ndb->init(max_transactions) != 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
ERR_PRINT(thd_ndb->ndb->getNdbError());
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
|
|
|
TODO
|
|
|
|
Alt.1 If init fails because to many allocated Ndb
|
|
|
|
wait on condition for a Ndb object to be released.
|
|
|
|
Alt.2 Seize/release from pool, wait until next release
|
|
|
|
*/
|
2004-09-14 14:47:34 +02:00
|
|
|
delete thd_ndb;
|
|
|
|
thd_ndb= NULL;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-09-14 14:47:34 +02:00
|
|
|
DBUG_RETURN(thd_ndb);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-14 14:47:34 +02:00
|
|
|
void ha_ndbcluster::release_thd_ndb(Thd_ndb* thd_ndb)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
DBUG_ENTER("release_thd_ndb");
|
|
|
|
delete thd_ndb;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2004-09-20 17:34:32 +02:00
|
|
|
If this thread already has a Thd_ndb object allocated
|
2004-04-15 09:14:14 +02:00
|
|
|
in current THD, reuse it. Otherwise
|
2004-09-20 17:34:32 +02:00
|
|
|
seize a Thd_ndb object, assign it to current THD and use it.
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
*/
|
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
Ndb* check_ndb_in_thd(THD* thd)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
2004-09-14 14:47:34 +02:00
|
|
|
if (!thd_ndb)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-20 17:34:32 +02:00
|
|
|
if (!(thd_ndb= ha_ndbcluster::seize_thd_ndb()))
|
2005-02-11 22:33:52 +01:00
|
|
|
return NULL;
|
2005-02-17 22:52:40 +01:00
|
|
|
set_thd_ndb(thd, thd_ndb);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
return thd_ndb->ndb;
|
2004-09-13 14:46:38 +02:00
|
|
|
}
|
|
|
|
|
2004-09-20 17:34:32 +02:00
|
|
|
|
2004-12-06 14:51:10 +01:00
|
|
|
|
2005-03-15 15:03:25 +01:00
|
|
|
int ha_ndbcluster::check_ndb_connection(THD* thd)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-12-30 19:56:09 +01:00
|
|
|
Ndb *ndb;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("check_ndb_connection");
|
|
|
|
|
2004-12-30 19:56:09 +01:00
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
2004-09-13 14:46:38 +02:00
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-09-20 17:34:32 +02:00
|
|
|
|
2005-02-17 22:52:40 +01:00
|
|
|
int ndbcluster_close_connection(THD *thd)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-02-17 22:52:40 +01:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(thd);
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("ndbcluster_close_connection");
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
if (thd_ndb)
|
|
|
|
{
|
2004-09-14 14:47:34 +02:00
|
|
|
ha_ndbcluster::release_thd_ndb(thd_ndb);
|
2005-02-17 22:52:40 +01:00
|
|
|
set_thd_ndb(thd, NULL); // not strictly required but does not hurt either
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
}
|
2005-02-17 22:52:40 +01:00
|
|
|
DBUG_RETURN(0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Try to discover one table from NDB
|
|
|
|
*/
|
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
int ndbcluster_discover(THD* thd, const char *db, const char *name,
|
2005-02-16 14:18:32 +01:00
|
|
|
const void** frmblob, uint* frmlen)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
uint len;
|
|
|
|
const void* data;
|
|
|
|
const NDBTAB* tab;
|
2004-09-13 14:46:38 +02:00
|
|
|
Ndb* ndb;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("ndbcluster_discover");
|
2004-09-13 14:46:38 +02:00
|
|
|
DBUG_PRINT("enter", ("db: %s, name: %s", db, name));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(db))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2004-09-13 14:46:38 +02:00
|
|
|
NDBDICT* dict= ndb->getDictionary();
|
2005-03-22 17:42:08 +01:00
|
|
|
dict->set_local_table_data_size(sizeof(Ndb_local_table_statistics));
|
2004-09-13 14:46:38 +02:00
|
|
|
dict->invalidateTable(name);
|
|
|
|
if (!(tab= dict->getTable(name)))
|
|
|
|
{
|
|
|
|
const NdbError err= dict->getNdbError();
|
|
|
|
if (err.code == 709)
|
2005-06-16 15:17:47 +02:00
|
|
|
DBUG_RETURN(-1);
|
2004-09-13 14:46:38 +02:00
|
|
|
ERR_RETURN(err);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
DBUG_PRINT("info", ("Found table %s", tab->getName()));
|
|
|
|
|
|
|
|
len= tab->getFrmLength();
|
|
|
|
if (len == 0 || tab->getFrmData() == NULL)
|
|
|
|
{
|
BUG#10365 Cluster handler uses non-standard error codes
- Added better error messages when trying to open a table that can't be discovered or unpacked. The most likely cause of this is that it does not have any frm data, probably since it has been created from NdbApi or is a NDB system table.
- Separated functionality that was in ha_create_table_from_engine into two functions. One that checks if the table exists and another one that tries to create the table from the engine.
mysql-test/r/ndb_autodiscover.result:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
mysql-test/t/ndb_autodiscover.test:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
ndb/test/ndbapi/create_tab.cpp:
Set connectstring before creating Ndb object.
sql/ha_ndbcluster.cc:
Rename and use the function ndbcluster_table_exists_in_engine.
Correct return valu from ndbcluster_discover
Remove old code "ndb_discover_tables"
sql/ha_ndbcluster.h:
Rename function ndbcluster_table_exists to ndb ndbcluster_table_exists_in_engine
sql/handler.cc:
Update comment of ha_create_table_from_engine
Remove parameter create_if_found from ha_create_table_from_engine, the function ha_table_exists_in_engine is now used toi check if table is found in engine.
Cleanup return codes from ha_create_table_from_engine.
Change name of ha_table_exists to ha_table_exists_in_engine, update comment and returne codes.
sql/handler.h:
Remove paramter create_if_cound from ha_create_table_from_engine
Rename ha_table_exists to ha_table_exists_in_engine
sql/sql_base.cc:
Use the function ha_table_exists_in_engine to detect if table exists in enegine.
If it exists, call function ha_create_table_from_engine to try and create it.
If create of table fails, set correct error message.
sql/sql_table.cc:
Add comments, remove parameter create_if_found to ha_create_table_from_engine.
When dropping a table, try to discover it from engine. If discover fails, use same error message as if the table didn't exists.
Maybe another message should be displayed here, ex: "Table could not be dropped, unpack failed"
When creating a new table, use ha_table_exists_in_engine to check if a table with the given name already exists.
2005-06-08 13:31:59 +02:00
|
|
|
DBUG_PRINT("error", ("No frm data found."));
|
|
|
|
DBUG_RETURN(1);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (unpackfrm(&data, &len, tab->getFrmData()))
|
BUG#10365 Cluster handler uses non-standard error codes
- Added better error messages when trying to open a table that can't be discovered or unpacked. The most likely cause of this is that it does not have any frm data, probably since it has been created from NdbApi or is a NDB system table.
- Separated functionality that was in ha_create_table_from_engine into two functions. One that checks if the table exists and another one that tries to create the table from the engine.
mysql-test/r/ndb_autodiscover.result:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
mysql-test/t/ndb_autodiscover.test:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
ndb/test/ndbapi/create_tab.cpp:
Set connectstring before creating Ndb object.
sql/ha_ndbcluster.cc:
Rename and use the function ndbcluster_table_exists_in_engine.
Correct return valu from ndbcluster_discover
Remove old code "ndb_discover_tables"
sql/ha_ndbcluster.h:
Rename function ndbcluster_table_exists to ndb ndbcluster_table_exists_in_engine
sql/handler.cc:
Update comment of ha_create_table_from_engine
Remove parameter create_if_found from ha_create_table_from_engine, the function ha_table_exists_in_engine is now used toi check if table is found in engine.
Cleanup return codes from ha_create_table_from_engine.
Change name of ha_table_exists to ha_table_exists_in_engine, update comment and returne codes.
sql/handler.h:
Remove paramter create_if_cound from ha_create_table_from_engine
Rename ha_table_exists to ha_table_exists_in_engine
sql/sql_base.cc:
Use the function ha_table_exists_in_engine to detect if table exists in enegine.
If it exists, call function ha_create_table_from_engine to try and create it.
If create of table fails, set correct error message.
sql/sql_table.cc:
Add comments, remove parameter create_if_found to ha_create_table_from_engine.
When dropping a table, try to discover it from engine. If discover fails, use same error message as if the table didn't exists.
Maybe another message should be displayed here, ex: "Table could not be dropped, unpack failed"
When creating a new table, use ha_table_exists_in_engine to check if a table with the given name already exists.
2005-06-08 13:31:59 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("Could not unpack table"));
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
*frmlen= len;
|
|
|
|
*frmblob= data;
|
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2004-09-13 14:46:38 +02:00
|
|
|
Check if a table exists in NDB
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
*/
|
2004-04-15 09:14:14 +02:00
|
|
|
|
BUG#10365 Cluster handler uses non-standard error codes
- Added better error messages when trying to open a table that can't be discovered or unpacked. The most likely cause of this is that it does not have any frm data, probably since it has been created from NdbApi or is a NDB system table.
- Separated functionality that was in ha_create_table_from_engine into two functions. One that checks if the table exists and another one that tries to create the table from the engine.
mysql-test/r/ndb_autodiscover.result:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
mysql-test/t/ndb_autodiscover.test:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
ndb/test/ndbapi/create_tab.cpp:
Set connectstring before creating Ndb object.
sql/ha_ndbcluster.cc:
Rename and use the function ndbcluster_table_exists_in_engine.
Correct return valu from ndbcluster_discover
Remove old code "ndb_discover_tables"
sql/ha_ndbcluster.h:
Rename function ndbcluster_table_exists to ndb ndbcluster_table_exists_in_engine
sql/handler.cc:
Update comment of ha_create_table_from_engine
Remove parameter create_if_found from ha_create_table_from_engine, the function ha_table_exists_in_engine is now used toi check if table is found in engine.
Cleanup return codes from ha_create_table_from_engine.
Change name of ha_table_exists to ha_table_exists_in_engine, update comment and returne codes.
sql/handler.h:
Remove paramter create_if_cound from ha_create_table_from_engine
Rename ha_table_exists to ha_table_exists_in_engine
sql/sql_base.cc:
Use the function ha_table_exists_in_engine to detect if table exists in enegine.
If it exists, call function ha_create_table_from_engine to try and create it.
If create of table fails, set correct error message.
sql/sql_table.cc:
Add comments, remove parameter create_if_found to ha_create_table_from_engine.
When dropping a table, try to discover it from engine. If discover fails, use same error message as if the table didn't exists.
Maybe another message should be displayed here, ex: "Table could not be dropped, unpack failed"
When creating a new table, use ha_table_exists_in_engine to check if a table with the given name already exists.
2005-06-08 13:31:59 +02:00
|
|
|
int ndbcluster_table_exists_in_engine(THD* thd, const char *db, const char *name)
|
2004-09-13 14:46:38 +02:00
|
|
|
{
|
|
|
|
const NDBTAB* tab;
|
|
|
|
Ndb* ndb;
|
BUG#10365 Cluster handler uses non-standard error codes
- Added better error messages when trying to open a table that can't be discovered or unpacked. The most likely cause of this is that it does not have any frm data, probably since it has been created from NdbApi or is a NDB system table.
- Separated functionality that was in ha_create_table_from_engine into two functions. One that checks if the table exists and another one that tries to create the table from the engine.
mysql-test/r/ndb_autodiscover.result:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
mysql-test/t/ndb_autodiscover.test:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
ndb/test/ndbapi/create_tab.cpp:
Set connectstring before creating Ndb object.
sql/ha_ndbcluster.cc:
Rename and use the function ndbcluster_table_exists_in_engine.
Correct return valu from ndbcluster_discover
Remove old code "ndb_discover_tables"
sql/ha_ndbcluster.h:
Rename function ndbcluster_table_exists to ndb ndbcluster_table_exists_in_engine
sql/handler.cc:
Update comment of ha_create_table_from_engine
Remove parameter create_if_found from ha_create_table_from_engine, the function ha_table_exists_in_engine is now used toi check if table is found in engine.
Cleanup return codes from ha_create_table_from_engine.
Change name of ha_table_exists to ha_table_exists_in_engine, update comment and returne codes.
sql/handler.h:
Remove paramter create_if_cound from ha_create_table_from_engine
Rename ha_table_exists to ha_table_exists_in_engine
sql/sql_base.cc:
Use the function ha_table_exists_in_engine to detect if table exists in enegine.
If it exists, call function ha_create_table_from_engine to try and create it.
If create of table fails, set correct error message.
sql/sql_table.cc:
Add comments, remove parameter create_if_found to ha_create_table_from_engine.
When dropping a table, try to discover it from engine. If discover fails, use same error message as if the table didn't exists.
Maybe another message should be displayed here, ex: "Table could not be dropped, unpack failed"
When creating a new table, use ha_table_exists_in_engine to check if a table with the given name already exists.
2005-06-08 13:31:59 +02:00
|
|
|
DBUG_ENTER("ndbcluster_table_exists_in_engine");
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_PRINT("enter", ("db: %s, name: %s", db, name));
|
2004-09-13 14:46:38 +02:00
|
|
|
|
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(db))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2004-09-13 14:46:38 +02:00
|
|
|
NDBDICT* dict= ndb->getDictionary();
|
2005-03-22 17:42:08 +01:00
|
|
|
dict->set_local_table_data_size(sizeof(Ndb_local_table_statistics));
|
2004-09-13 14:46:38 +02:00
|
|
|
dict->invalidateTable(name);
|
|
|
|
if (!(tab= dict->getTable(name)))
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
2007-04-03 12:44:29 +02:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
2004-09-13 14:46:38 +02:00
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
DBUG_PRINT("info", ("Found table %s", tab->getName()));
|
2007-04-03 12:44:29 +02:00
|
|
|
DBUG_RETURN(HA_ERR_TABLE_EXIST);
|
2004-09-13 14:46:38 +02:00
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
extern "C" byte* tables_get_key(const char *entry, uint *length,
|
2005-02-16 14:18:32 +01:00
|
|
|
my_bool not_used __attribute__((unused)))
|
2004-09-21 12:13:58 +02:00
|
|
|
{
|
|
|
|
*length= strlen(entry);
|
|
|
|
return (byte*) entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-04-22 15:28:23 +02:00
|
|
|
/*
|
|
|
|
Drop a database in NDB Cluster
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ndbcluster_drop_database(const char *path)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("ndbcluster_drop_database");
|
|
|
|
THD *thd= current_thd;
|
|
|
|
char dbname[FN_HEADLEN];
|
|
|
|
Ndb* ndb;
|
|
|
|
NdbDictionary::Dictionary::List list;
|
|
|
|
uint i;
|
|
|
|
char *tabname;
|
|
|
|
List<char> drop_list;
|
2005-04-25 11:31:16 +02:00
|
|
|
int ret= 0;
|
2005-04-22 15:28:23 +02:00
|
|
|
ha_ndbcluster::set_dbname(path, (char *)&dbname);
|
|
|
|
DBUG_PRINT("enter", ("db: %s", dbname));
|
|
|
|
|
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
|
|
|
|
|
|
|
// List tables in NDB
|
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
|
|
|
if (dict->listObjects(list,
|
|
|
|
NdbDictionary::Object::UserTable) != 0)
|
|
|
|
ERR_RETURN(dict->getNdbError());
|
|
|
|
for (i= 0 ; i < list.count ; i++)
|
|
|
|
{
|
|
|
|
NdbDictionary::Dictionary::List::Element& t= list.elements[i];
|
|
|
|
DBUG_PRINT("info", ("Found %s/%s in NDB", t.database, t.name));
|
|
|
|
|
|
|
|
// Add only tables that belongs to db
|
|
|
|
if (my_strcasecmp(system_charset_info, t.database, dbname))
|
|
|
|
continue;
|
|
|
|
DBUG_PRINT("info", ("%s must be dropped", t.name));
|
|
|
|
drop_list.push_back(thd->strdup(t.name));
|
|
|
|
}
|
|
|
|
// Drop any tables belonging to database
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2005-04-22 15:28:23 +02:00
|
|
|
List_iterator_fast<char> it(drop_list);
|
|
|
|
while ((tabname=it++))
|
2005-04-25 11:31:16 +02:00
|
|
|
{
|
2006-06-14 12:54:28 +02:00
|
|
|
while (dict->dropTable(tabname))
|
2005-04-25 11:31:16 +02:00
|
|
|
{
|
|
|
|
const NdbError err= dict->getNdbError();
|
2006-06-14 12:54:28 +02:00
|
|
|
switch (err.status)
|
|
|
|
{
|
|
|
|
case NdbError::TemporaryError:
|
|
|
|
if (!thd->killed)
|
|
|
|
continue; // retry indefinitly
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (err.code != 709) // 709: No such table existed
|
2005-04-25 11:34:47 +02:00
|
|
|
{
|
|
|
|
ERR_PRINT(err);
|
2005-04-25 11:31:16 +02:00
|
|
|
ret= ndb_to_mysql_error(&err);
|
2005-04-25 11:34:47 +02:00
|
|
|
}
|
2006-06-14 12:54:28 +02:00
|
|
|
break;
|
2005-04-25 11:31:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_RETURN(ret);
|
2005-04-22 15:28:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-21 12:13:58 +02:00
|
|
|
int ndbcluster_find_files(THD *thd,const char *db,const char *path,
|
2005-02-16 14:18:32 +01:00
|
|
|
const char *wild, bool dir, List<char> *files)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-10-25 22:35:46 +02:00
|
|
|
DBUG_ENTER("ndbcluster_find_files");
|
|
|
|
DBUG_PRINT("enter", ("db: %s", db));
|
|
|
|
{ // extra bracket to avoid gcc 2.95.3 warning
|
2004-04-15 09:14:14 +02:00
|
|
|
uint i;
|
2004-09-13 14:46:38 +02:00
|
|
|
Ndb* ndb;
|
2004-09-21 12:13:58 +02:00
|
|
|
char name[FN_REFLEN];
|
2004-09-26 16:11:24 +02:00
|
|
|
HASH ndb_tables, ok_tables;
|
2004-04-15 09:14:14 +02:00
|
|
|
NdbDictionary::Dictionary::List list;
|
2004-09-13 14:46:38 +02:00
|
|
|
|
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
|
|
|
DBUG_RETURN(HA_ERR_NO_CONNECTION);
|
|
|
|
|
2004-09-21 12:13:58 +02:00
|
|
|
if (dir)
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_RETURN(0); // Discover of databases not yet supported
|
2004-09-21 12:13:58 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
// List tables in NDB
|
2004-09-13 14:46:38 +02:00
|
|
|
NDBDICT *dict= ndb->getDictionary();
|
2004-04-15 09:14:14 +02:00
|
|
|
if (dict->listObjects(list,
|
2005-02-16 14:18:32 +01:00
|
|
|
NdbDictionary::Object::UserTable) != 0)
|
2004-09-13 06:40:25 +02:00
|
|
|
ERR_RETURN(dict->getNdbError());
|
2004-09-13 14:46:38 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
if (hash_init(&ndb_tables, system_charset_info,list.count,0,0,
|
2005-02-16 14:18:32 +01:00
|
|
|
(hash_get_key)tables_get_key,0,0))
|
2004-09-26 16:11:24 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("Failed to init HASH ndb_tables"));
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_init(&ok_tables, system_charset_info,32,0,0,
|
2005-02-16 14:18:32 +01:00
|
|
|
(hash_get_key)tables_get_key,0,0))
|
2004-09-26 16:11:24 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("Failed to init HASH ok_tables"));
|
|
|
|
hash_free(&ndb_tables);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
for (i= 0 ; i < list.count ; i++)
|
|
|
|
{
|
|
|
|
NdbDictionary::Dictionary::List::Element& t= list.elements[i];
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("Found %s/%s in NDB", t.database, t.name));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-10-08 13:37:13 +02:00
|
|
|
// Add only tables that belongs to db
|
|
|
|
if (my_strcasecmp(system_charset_info, t.database, db))
|
|
|
|
continue;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
// Apply wildcard to list of tables in NDB
|
2004-09-21 12:13:58 +02:00
|
|
|
if (wild)
|
2004-09-13 14:46:38 +02:00
|
|
|
{
|
2004-09-21 12:13:58 +02:00
|
|
|
if (lower_case_table_names)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
if (wild_case_compare(files_charset_info, t.name, wild))
|
|
|
|
continue;
|
2004-09-21 12:13:58 +02:00
|
|
|
}
|
|
|
|
else if (wild_compare(t.name,wild,0))
|
2005-02-16 14:18:32 +01:00
|
|
|
continue;
|
2004-09-21 12:13:58 +02:00
|
|
|
}
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("Inserting %s into ndb_tables hash", t.name));
|
|
|
|
my_hash_insert(&ndb_tables, (byte*)thd->strdup(t.name));
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
char *file_name;
|
|
|
|
List_iterator<char> it(*files);
|
|
|
|
List<char> delete_list;
|
|
|
|
while ((file_name=it++))
|
|
|
|
{
|
2006-09-12 16:04:52 +02:00
|
|
|
bool file_on_disk= false;
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("%s", file_name));
|
|
|
|
if (hash_search(&ndb_tables, file_name, strlen(file_name)))
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("%s existed in NDB _and_ on disk ", file_name));
|
2006-09-12 16:04:52 +02:00
|
|
|
file_on_disk= true;
|
2004-09-26 16:11:24 +02:00
|
|
|
}
|
|
|
|
|
2006-09-12 16:04:52 +02:00
|
|
|
// Check for .ndb file with this name
|
2004-09-21 12:13:58 +02:00
|
|
|
(void)strxnmov(name, FN_REFLEN,
|
2005-02-16 14:18:32 +01:00
|
|
|
mysql_data_home,"/",db,"/",file_name,ha_ndb_ext,NullS);
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("Check access for %s", name));
|
2004-09-21 12:13:58 +02:00
|
|
|
if (access(name, F_OK))
|
2004-09-26 16:11:24 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("%s did not exist on disk", name));
|
|
|
|
// .ndb file did not exist on disk, another table type
|
2006-09-12 16:04:52 +02:00
|
|
|
if (file_on_disk)
|
2006-09-13 18:49:05 +02:00
|
|
|
{
|
|
|
|
// Ignore this ndb table
|
|
|
|
gptr record= hash_search(&ndb_tables, file_name, strlen(file_name));
|
|
|
|
DBUG_ASSERT(record);
|
|
|
|
hash_delete(&ndb_tables, record);
|
2006-09-12 16:04:52 +02:00
|
|
|
push_warning_printf(current_thd, MYSQL_ERROR::WARN_LEVEL_WARN,
|
|
|
|
ER_TABLE_EXISTS_ERROR,
|
|
|
|
"Local table %s.%s shadows ndb table",
|
|
|
|
db, file_name);
|
2006-09-13 18:49:05 +02:00
|
|
|
}
|
2006-09-12 16:04:52 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (file_on_disk)
|
|
|
|
{
|
2006-09-13 18:49:05 +02:00
|
|
|
// File existed in NDB and as frm file, put in ok_tables list
|
2006-09-12 16:04:52 +02:00
|
|
|
my_hash_insert(&ok_tables, (byte*)file_name);
|
2004-09-26 16:11:24 +02:00
|
|
|
continue;
|
2004-09-13 14:46:38 +02:00
|
|
|
}
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("%s existed on disk", name));
|
|
|
|
// The .ndb file exists on disk, but it's not in list of tables in ndb
|
|
|
|
// Verify that handler agrees table is gone.
|
2007-04-03 12:44:29 +02:00
|
|
|
if (ndbcluster_table_exists_in_engine(thd, db, file_name) == HA_ERR_NO_SUCH_TABLE)
|
2004-09-26 16:11:24 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("NDB says %s does not exists", file_name));
|
|
|
|
it.remove();
|
|
|
|
// Put in list of tables to remove from disk
|
|
|
|
delete_list.push_back(thd->strdup(file_name));
|
|
|
|
}
|
|
|
|
}
|
2004-09-21 12:13:58 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
// Check for new files to discover
|
|
|
|
DBUG_PRINT("info", ("Checking for new files to discover"));
|
|
|
|
List<char> create_list;
|
|
|
|
for (i= 0 ; i < ndb_tables.records ; i++)
|
2004-09-21 12:13:58 +02:00
|
|
|
{
|
2004-09-26 16:11:24 +02:00
|
|
|
file_name= hash_element(&ndb_tables, i);
|
|
|
|
if (!hash_search(&ok_tables, file_name, strlen(file_name)))
|
2004-09-21 12:13:58 +02:00
|
|
|
{
|
2004-09-26 16:11:24 +02:00
|
|
|
DBUG_PRINT("info", ("%s must be discovered", file_name));
|
|
|
|
// File is in list of ndb tables and not in ok_tables
|
|
|
|
// This table need to be created
|
|
|
|
create_list.push_back(thd->strdup(file_name));
|
|
|
|
}
|
|
|
|
}
|
2004-09-21 12:13:58 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
// Lock mutex before deleting and creating frm files
|
|
|
|
pthread_mutex_lock(&LOCK_open);
|
2004-09-21 12:13:58 +02:00
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
if (!global_read_lock)
|
|
|
|
{
|
|
|
|
// Delete old files
|
|
|
|
List_iterator_fast<char> it3(delete_list);
|
|
|
|
while ((file_name=it3++))
|
2005-02-17 14:22:44 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Remove table %s/%s", db, file_name));
|
2004-09-26 16:11:24 +02:00
|
|
|
// Delete the table and all related files
|
|
|
|
TABLE_LIST table_list;
|
|
|
|
bzero((char*) &table_list,sizeof(table_list));
|
|
|
|
table_list.db= (char*) db;
|
2005-01-06 12:00:13 +01:00
|
|
|
table_list.alias= table_list.table_name= (char*)file_name;
|
2005-02-17 14:22:44 +01:00
|
|
|
(void)mysql_rm_table_part2(thd, &table_list,
|
2005-02-21 16:13:29 +01:00
|
|
|
/* if_exists */ FALSE,
|
|
|
|
/* drop_temporary */ FALSE,
|
|
|
|
/* drop_view */ FALSE,
|
|
|
|
/* dont_log_query*/ TRUE);
|
2005-02-17 14:22:44 +01:00
|
|
|
/* Clear error message that is returned when table is deleted */
|
|
|
|
thd->clear_error();
|
2004-09-21 12:13:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-09-26 16:11:24 +02:00
|
|
|
// Create new files
|
|
|
|
List_iterator_fast<char> it2(create_list);
|
|
|
|
while ((file_name=it2++))
|
|
|
|
{
|
BUG#9626 Several serious errors reported by Valgrind in latest 5.0 bk tree
- Fix several valgrind warnings.
ndb/include/kernel/AttributeList.hpp:
Include ndb_limits.h
ndb/include/ndbapi/Ndb.hpp:
Add new function internalize_table_name
Proper formatting
ndb/include/util/SimpleProperties.hpp:
Add function "add" for adding strings/data to word buffers without reading after the string/data
Fix comment for UtilBufferWriter
ndb/src/common/util/SimpleProperties.cpp:
Add function "add" for adding strings/data to word buffers without reading after the string/data
ndb/src/ndbapi/DictCache.cpp:
Add DBUG printouts to all functions in GlobalDictCache
Change ndbout_c to DBUG_PRINT
Add function GlobalDictCache::printCache
ndb/src/ndbapi/DictCache.hpp:
Add function GlobalDictCache::printCache
ndb/src/ndbapi/Ndb.cpp:
Change DEBUG_TRACE to DBUG_PRINT, DBUG_ENTER
Use new function internalize_table_name, and create internal tabname on stack
ndb/src/ndbapi/NdbDictionary.cpp:
Add DBUG_ENTER calls
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
Pass BaseString as reference when passing internal table name to internal functions.
Remove check for exceeding MAX_SECTION_SIZE.
Remove pekkas fix that saves internalName in save_me variable, not needed when tablename is stack variable.
NdbDictInterface::gettable(int tableid, ...) - Not used
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
Remove include of ndb_limits.h
Use BaseString& for internal functions
Remove m_namebuf, not needed, uses m_buffer
Formatting
NdbDictInterface::getTable(int tableId, ...) - Not used
ndb/src/ndbapi/NdbImpl.hpp:
Remove buffer for m_internalname, placed on stack of caller.
Moved impl of internalize_table_name and internalize_index_anme to Ndb::internalize_table_name and Ndb::internalize_index_name
ndb/src/ndbapi/NdbLinHash.hpp:
Save also the terminating null character in chain->str to get nicer printouts.
ndb/src/ndbapi/NdbTransaction.cpp:
Remove hardcoded buffer sizes.
Remove include of ndb_limits.h
sql/ha_ndbcluster.cc:
Fix "typo", use file_name var since that is the current files name,
2005-06-02 14:32:06 +02:00
|
|
|
DBUG_PRINT("info", ("Table %s need discovery", file_name));
|
BUG#10365 Cluster handler uses non-standard error codes
- Added better error messages when trying to open a table that can't be discovered or unpacked. The most likely cause of this is that it does not have any frm data, probably since it has been created from NdbApi or is a NDB system table.
- Separated functionality that was in ha_create_table_from_engine into two functions. One that checks if the table exists and another one that tries to create the table from the engine.
mysql-test/r/ndb_autodiscover.result:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
mysql-test/t/ndb_autodiscover.test:
Add tests for reading from a table that can't be discovered(SYSTAB_0)
Discovery is not performed during create table anymore.
ndb/test/ndbapi/create_tab.cpp:
Set connectstring before creating Ndb object.
sql/ha_ndbcluster.cc:
Rename and use the function ndbcluster_table_exists_in_engine.
Correct return valu from ndbcluster_discover
Remove old code "ndb_discover_tables"
sql/ha_ndbcluster.h:
Rename function ndbcluster_table_exists to ndb ndbcluster_table_exists_in_engine
sql/handler.cc:
Update comment of ha_create_table_from_engine
Remove parameter create_if_found from ha_create_table_from_engine, the function ha_table_exists_in_engine is now used toi check if table is found in engine.
Cleanup return codes from ha_create_table_from_engine.
Change name of ha_table_exists to ha_table_exists_in_engine, update comment and returne codes.
sql/handler.h:
Remove paramter create_if_cound from ha_create_table_from_engine
Rename ha_table_exists to ha_table_exists_in_engine
sql/sql_base.cc:
Use the function ha_table_exists_in_engine to detect if table exists in enegine.
If it exists, call function ha_create_table_from_engine to try and create it.
If create of table fails, set correct error message.
sql/sql_table.cc:
Add comments, remove parameter create_if_found to ha_create_table_from_engine.
When dropping a table, try to discover it from engine. If discover fails, use same error message as if the table didn't exists.
Maybe another message should be displayed here, ex: "Table could not be dropped, unpack failed"
When creating a new table, use ha_table_exists_in_engine to check if a table with the given name already exists.
2005-06-08 13:31:59 +02:00
|
|
|
if (ha_create_table_from_engine(thd, db, file_name) == 0)
|
2004-10-07 09:16:46 +02:00
|
|
|
files->push_back(thd->strdup(file_name));
|
2004-09-26 16:11:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&LOCK_open);
|
|
|
|
|
|
|
|
hash_free(&ok_tables);
|
2004-09-21 12:13:58 +02:00
|
|
|
hash_free(&ndb_tables);
|
2004-10-25 22:35:46 +02:00
|
|
|
} // extra bracket to avoid gcc 2.95.3 warning
|
2004-09-21 12:13:58 +02:00
|
|
|
DBUG_RETURN(0);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Initialise all gloal variables before creating
|
|
|
|
a NDB Cluster table handler
|
|
|
|
*/
|
|
|
|
|
2005-01-14 12:32:33 +01:00
|
|
|
/* Call back after cluster connect */
|
|
|
|
static int connect_callback()
|
|
|
|
{
|
|
|
|
update_status_variables(g_ndb_cluster_connection);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-10-03 04:44:28 +02:00
|
|
|
bool ndbcluster_init()
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-09-13 06:40:25 +02:00
|
|
|
int res;
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_ENTER("ndbcluster_init");
|
2005-10-03 04:44:28 +02:00
|
|
|
|
|
|
|
if (have_ndbcluster != SHOW_OPTION_YES)
|
|
|
|
goto ndbcluster_init_error;
|
|
|
|
|
2004-08-20 18:10:47 +02:00
|
|
|
// Set connectstring if specified
|
2004-12-20 12:36:14 +01:00
|
|
|
if (opt_ndbcluster_connectstring != 0)
|
|
|
|
DBUG_PRINT("connectstring", ("%s", opt_ndbcluster_connectstring));
|
2004-09-06 20:30:57 +02:00
|
|
|
if ((g_ndb_cluster_connection=
|
2004-12-20 12:36:14 +01:00
|
|
|
new Ndb_cluster_connection(opt_ndbcluster_connectstring)) == 0)
|
2004-09-06 20:30:57 +02:00
|
|
|
{
|
2004-12-20 12:36:14 +01:00
|
|
|
DBUG_PRINT("error",("Ndb_cluster_connection(%s)",
|
2005-02-16 14:18:32 +01:00
|
|
|
opt_ndbcluster_connectstring));
|
2007-04-11 15:21:11 +02:00
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
2004-11-30 18:47:47 +01:00
|
|
|
goto ndbcluster_init_error;
|
2004-08-20 18:10:47 +02:00
|
|
|
}
|
2006-02-13 12:58:12 +01:00
|
|
|
{
|
|
|
|
char buf[128];
|
2006-11-20 21:42:06 +01:00
|
|
|
my_snprintf(buf, sizeof(buf), "mysqld --server-id=%lu", server_id);
|
2006-02-13 12:58:12 +01:00
|
|
|
g_ndb_cluster_connection->set_name(buf);
|
|
|
|
}
|
2004-12-20 12:36:14 +01:00
|
|
|
g_ndb_cluster_connection->set_optimized_node_selection
|
|
|
|
(opt_ndb_optimized_node_selection);
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
// Create a Ndb object to open the connection to NDB
|
2005-02-21 23:29:07 +01:00
|
|
|
if ( (g_ndb= new Ndb(g_ndb_cluster_connection, "sys")) == 0 )
|
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("failed to create global ndb object"));
|
2007-04-11 15:21:11 +02:00
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
2005-02-21 23:29:07 +01:00
|
|
|
goto ndbcluster_init_error;
|
|
|
|
}
|
2005-03-22 17:42:08 +01:00
|
|
|
g_ndb->getDictionary()->set_local_table_data_size(sizeof(Ndb_local_table_statistics));
|
2004-04-15 09:14:14 +02:00
|
|
|
if (g_ndb->init() != 0)
|
|
|
|
{
|
|
|
|
ERR_PRINT (g_ndb->getNdbError());
|
2004-11-30 18:47:47 +01:00
|
|
|
goto ndbcluster_init_error;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-09-13 06:40:25 +02:00
|
|
|
|
2004-11-30 18:47:47 +01:00
|
|
|
if ((res= g_ndb_cluster_connection->connect(0,0,0)) == 0)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-01-14 12:32:33 +01:00
|
|
|
connect_callback();
|
2004-11-30 18:47:47 +01:00
|
|
|
DBUG_PRINT("info",("NDBCLUSTER storage engine at %s on port %d",
|
2005-02-16 14:18:32 +01:00
|
|
|
g_ndb_cluster_connection->get_connected_host(),
|
|
|
|
g_ndb_cluster_connection->get_connected_port()));
|
2005-02-24 18:58:56 +01:00
|
|
|
g_ndb_cluster_connection->wait_until_ready(10,3);
|
2004-09-13 06:40:25 +02:00
|
|
|
}
|
2005-07-04 02:42:33 +02:00
|
|
|
else if (res == 1)
|
2004-09-13 06:40:25 +02:00
|
|
|
{
|
2005-01-14 12:32:33 +01:00
|
|
|
if (g_ndb_cluster_connection->start_connect_thread(connect_callback))
|
2005-01-11 17:00:31 +01:00
|
|
|
{
|
2004-09-27 02:24:23 +02:00
|
|
|
DBUG_PRINT("error", ("g_ndb_cluster_connection->start_connect_thread()"));
|
2004-11-30 18:47:47 +01:00
|
|
|
goto ndbcluster_init_error;
|
|
|
|
}
|
2005-01-11 17:00:31 +01:00
|
|
|
#ifndef DBUG_OFF
|
2004-11-30 18:47:47 +01:00
|
|
|
{
|
|
|
|
char buf[1024];
|
2005-01-11 17:00:31 +01:00
|
|
|
DBUG_PRINT("info",
|
2005-02-16 14:18:32 +01:00
|
|
|
("NDBCLUSTER storage engine not started, "
|
|
|
|
"will connect using %s",
|
|
|
|
g_ndb_cluster_connection->
|
|
|
|
get_connectstring(buf,sizeof(buf))));
|
2004-09-27 02:24:23 +02:00
|
|
|
}
|
2005-01-11 17:00:31 +01:00
|
|
|
#endif
|
2004-09-13 06:40:25 +02:00
|
|
|
}
|
2004-09-27 02:24:23 +02:00
|
|
|
else
|
2004-09-13 06:40:25 +02:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(res == -1);
|
|
|
|
DBUG_PRINT("error", ("permanent error"));
|
2004-11-30 18:47:47 +01:00
|
|
|
goto ndbcluster_init_error;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
2004-09-13 06:40:25 +02:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
(void) hash_init(&ndbcluster_open_tables,system_charset_info,32,0,0,
|
|
|
|
(hash_get_key) ndbcluster_get_key,0,0);
|
|
|
|
pthread_mutex_init(&ndbcluster_mutex,MY_MUTEX_INIT_FAST);
|
2005-02-11 22:33:52 +01:00
|
|
|
pthread_mutex_init(&LOCK_ndb_util_thread, MY_MUTEX_INIT_FAST);
|
|
|
|
pthread_cond_init(&COND_ndb_util_thread, NULL);
|
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
new methods to keep "records" up to datecorrect record field in ndbcluster handler
new method for ndbcluster handler to store/retrieve table and thread specific data
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
hanged deleteKey to return ponter to deleted object
moved heavy global cache fetch from inline to separate method
mysql-test/r/ndb_alter_table.result:
correct record field in ndbcluster handler
mysql-test/r/ndb_blob.result:
correct record field in ndbcluster handler
ndb/include/ndbapi/NdbDictionary.hpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/DictCache.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/DictCache.hpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
ndb/src/ndbapi/Ndb.cpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbDictionary.cpp:
new method for ndbcluster handler to store/retrieve table and thread specific data
ndb/src/ndbapi/NdbDictionaryImpl.cpp:
changed local hash to store new table_info object, with placeholders for local data, instead of TableImpl
moved heavy global cache fetch from inline to separate method
ndb/src/ndbapi/NdbDictionaryImpl.hpp:
replaced method DictionaryImpl::getTable with DictionaryImpl::get_local_table_info
ndb/src/ndbapi/NdbLinHash.hpp:
changed deleteKey to return ponter to deleted object
sql/ha_ndbcluster.cc:
moved all ndb thread specific data into new placeholder
new methods to keep "records" up to date
unset flag HA_NOT_EXACT_COUNT to make handler read "records" field, for count() optim and join optimization
sql/ha_ndbcluster.h:
new methods to keep "records" up to date
sql/sql_class.h:
moved all ndb thread specific data into new placeholder
2004-09-14 10:52:21 +02:00
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2006-08-15 14:22:47 +02:00
|
|
|
ndb_cache_check_time = opt_ndb_cache_check_time;
|
2005-02-11 22:33:52 +01:00
|
|
|
// Create utility thread
|
|
|
|
pthread_t tmp;
|
|
|
|
if (pthread_create(&tmp, &connection_attrib, ndb_util_thread_func, 0))
|
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("Could not create ndb utility thread"));
|
2005-02-21 23:29:07 +01:00
|
|
|
hash_free(&ndbcluster_open_tables);
|
|
|
|
pthread_mutex_destroy(&ndbcluster_mutex);
|
|
|
|
pthread_mutex_destroy(&LOCK_ndb_util_thread);
|
|
|
|
pthread_cond_destroy(&COND_ndb_util_thread);
|
2005-02-11 22:33:52 +01:00
|
|
|
goto ndbcluster_init_error;
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
ndbcluster_inited= 1;
|
2005-10-03 04:44:28 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2005-10-03 04:44:28 +02:00
|
|
|
ndbcluster_init_error:
|
2005-07-04 02:42:33 +02:00
|
|
|
if (g_ndb)
|
2005-02-21 23:29:07 +01:00
|
|
|
delete g_ndb;
|
|
|
|
g_ndb= NULL;
|
|
|
|
if (g_ndb_cluster_connection)
|
|
|
|
delete g_ndb_cluster_connection;
|
|
|
|
g_ndb_cluster_connection= NULL;
|
2005-10-03 04:44:28 +02:00
|
|
|
have_ndbcluster= SHOW_OPTION_DISABLED; // If we couldn't use handler
|
|
|
|
DBUG_RETURN(TRUE);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
End use of the NDB Cluster table handler
|
|
|
|
- free all global variables allocated by
|
2005-02-11 22:33:52 +01:00
|
|
|
ndbcluster_init()
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
bool ndbcluster_end()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("ndbcluster_end");
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2005-02-21 23:29:07 +01:00
|
|
|
if (!ndbcluster_inited)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
// Kill ndb utility thread
|
|
|
|
(void) pthread_mutex_lock(&LOCK_ndb_util_thread);
|
|
|
|
DBUG_PRINT("exit",("killing ndb util thread: %lx", ndb_util_thread));
|
|
|
|
(void) pthread_cond_signal(&COND_ndb_util_thread);
|
|
|
|
(void) pthread_mutex_unlock(&LOCK_ndb_util_thread);
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if (g_ndb)
|
2005-11-17 12:45:23 +01:00
|
|
|
{
|
|
|
|
#ifndef DBUG_OFF
|
2006-01-05 23:47:49 +01:00
|
|
|
Ndb::Free_list_usage tmp;
|
|
|
|
tmp.m_name= 0;
|
2005-11-17 12:45:23 +01:00
|
|
|
while (g_ndb->get_free_list_usage(&tmp))
|
|
|
|
{
|
|
|
|
uint leaked= (uint) tmp.m_created - tmp.m_free;
|
|
|
|
if (leaked)
|
|
|
|
fprintf(stderr, "NDB: Found %u %s%s that %s not been released\n",
|
|
|
|
leaked, tmp.m_name,
|
|
|
|
(leaked == 1)?"":"'s",
|
|
|
|
(leaked == 1)?"has":"have");
|
|
|
|
}
|
|
|
|
#endif
|
2004-09-13 06:40:25 +02:00
|
|
|
delete g_ndb;
|
2006-01-05 23:47:49 +01:00
|
|
|
g_ndb= NULL;
|
2005-11-17 12:45:23 +01:00
|
|
|
}
|
2006-01-05 23:47:49 +01:00
|
|
|
delete g_ndb_cluster_connection;
|
2004-09-06 20:30:57 +02:00
|
|
|
g_ndb_cluster_connection= NULL;
|
2005-02-21 23:29:07 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
hash_free(&ndbcluster_open_tables);
|
|
|
|
pthread_mutex_destroy(&ndbcluster_mutex);
|
2005-02-11 22:33:52 +01:00
|
|
|
pthread_mutex_destroy(&LOCK_ndb_util_thread);
|
|
|
|
pthread_cond_destroy(&COND_ndb_util_thread);
|
2004-04-15 09:14:14 +02:00
|
|
|
ndbcluster_inited= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2004-06-23 15:34:45 +02:00
|
|
|
/*
|
|
|
|
Static error print function called from
|
|
|
|
static handler method ndbcluster_commit
|
|
|
|
and ndbcluster_rollback
|
|
|
|
*/
|
2004-07-02 16:14:08 +02:00
|
|
|
|
|
|
|
void ndbcluster_print_error(int error, const NdbOperation *error_op)
|
2004-06-22 09:54:00 +02:00
|
|
|
{
|
2004-06-23 15:34:45 +02:00
|
|
|
DBUG_ENTER("ndbcluster_print_error");
|
|
|
|
TABLE tab;
|
2004-07-02 16:14:08 +02:00
|
|
|
const char *tab_name= (error_op) ? error_op->getTableName() : "";
|
2005-01-06 12:00:13 +01:00
|
|
|
tab.alias= (char *) tab_name;
|
2004-06-23 15:34:45 +02:00
|
|
|
ha_ndbcluster error_handler(&tab);
|
2004-07-02 16:14:08 +02:00
|
|
|
tab.file= &error_handler;
|
2004-06-23 15:34:45 +02:00
|
|
|
error_handler.print_error(error, MYF(0));
|
2004-07-06 08:43:57 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2004-06-22 09:54:00 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-04-22 15:28:23 +02:00
|
|
|
/**
|
|
|
|
* Set a given location from full pathname to database name
|
|
|
|
*
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
2005-04-22 15:28:23 +02:00
|
|
|
void ha_ndbcluster::set_dbname(const char *path_name, char *dbname)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
char *end, *ptr;
|
|
|
|
|
|
|
|
/* Scan name from the end */
|
2005-04-22 15:28:23 +02:00
|
|
|
ptr= strend(path_name)-1;
|
|
|
|
while (ptr >= path_name && *ptr != '\\' && *ptr != '/') {
|
|
|
|
ptr--;
|
|
|
|
}
|
|
|
|
ptr--;
|
|
|
|
end= ptr;
|
2004-04-15 09:14:14 +02:00
|
|
|
while (ptr >= path_name && *ptr != '\\' && *ptr != '/') {
|
|
|
|
ptr--;
|
|
|
|
}
|
|
|
|
uint name_len= end - ptr;
|
2005-04-22 15:28:23 +02:00
|
|
|
memcpy(dbname, ptr + 1, name_len);
|
|
|
|
dbname[name_len]= '\0';
|
2004-04-15 09:14:14 +02:00
|
|
|
#ifdef __WIN__
|
|
|
|
/* Put to lower case */
|
2005-04-22 15:28:23 +02:00
|
|
|
|
|
|
|
ptr= dbname;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
while (*ptr != '\0') {
|
2004-07-23 15:46:56 +02:00
|
|
|
*ptr= tolower(*ptr);
|
2004-04-15 09:14:14 +02:00
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2005-04-22 15:28:23 +02:00
|
|
|
/*
|
|
|
|
Set m_dbname from full pathname to table file
|
|
|
|
*/
|
|
|
|
|
|
|
|
void ha_ndbcluster::set_dbname(const char *path_name)
|
|
|
|
{
|
|
|
|
set_dbname(path_name, m_dbname);
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/**
|
|
|
|
* Set a given location from full pathname to table file
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ha_ndbcluster::set_tabname(const char *path_name, char * tabname)
|
|
|
|
{
|
|
|
|
char *end, *ptr;
|
|
|
|
|
|
|
|
/* Scan name from the end */
|
2004-07-23 15:46:56 +02:00
|
|
|
end= strend(path_name)-1;
|
|
|
|
ptr= end;
|
2004-04-15 09:14:14 +02:00
|
|
|
while (ptr >= path_name && *ptr != '\\' && *ptr != '/') {
|
|
|
|
ptr--;
|
|
|
|
}
|
2004-07-23 15:46:56 +02:00
|
|
|
uint name_len= end - ptr;
|
2004-04-15 09:14:14 +02:00
|
|
|
memcpy(tabname, ptr + 1, end - ptr);
|
2004-07-23 15:46:56 +02:00
|
|
|
tabname[name_len]= '\0';
|
2004-04-15 09:14:14 +02:00
|
|
|
#ifdef __WIN__
|
|
|
|
/* Put to lower case */
|
2004-07-23 15:46:56 +02:00
|
|
|
ptr= tabname;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
while (*ptr != '\0') {
|
|
|
|
*ptr= tolower(*ptr);
|
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2005-04-22 15:28:23 +02:00
|
|
|
Set m_tabname from full pathname to table file
|
2004-04-15 09:14:14 +02:00
|
|
|
*/
|
|
|
|
|
2005-04-22 15:28:23 +02:00
|
|
|
void ha_ndbcluster::set_tabname(const char *path_name)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2005-04-22 15:28:23 +02:00
|
|
|
set_tabname(path_name, m_tabname);
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ha_rows
|
2004-05-16 13:48:32 +02:00
|
|
|
ha_ndbcluster::records_in_range(uint inx, key_range *min_key,
|
|
|
|
key_range *max_key)
|
|
|
|
{
|
|
|
|
KEY *key_info= table->key_info + inx;
|
2004-04-15 09:14:14 +02:00
|
|
|
uint key_length= key_info->key_length;
|
2004-05-24 12:35:39 +02:00
|
|
|
NDB_INDEX_TYPE idx_type= get_index_type(inx);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("records_in_range");
|
2004-05-24 12:35:39 +02:00
|
|
|
// Prevent partial read of hash indexes by returning HA_POS_ERROR
|
|
|
|
if ((idx_type == UNIQUE_INDEX || idx_type == PRIMARY_KEY_INDEX) &&
|
|
|
|
((min_key && min_key->length < key_length) ||
|
|
|
|
(max_key && max_key->length < key_length)))
|
|
|
|
DBUG_RETURN(HA_POS_ERROR);
|
|
|
|
|
|
|
|
// Read from hash index with full key
|
|
|
|
// This is a "const" table which returns only one record!
|
|
|
|
if ((idx_type != ORDERED_INDEX) &&
|
|
|
|
((min_key && min_key->length == key_length) ||
|
|
|
|
(max_key && max_key->length == key_length)))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
|
|
|
DBUG_RETURN(10); /* Good guess when you don't know anything */
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
|
2004-11-17 09:15:53 +01:00
|
|
|
ulong ha_ndbcluster::table_flags(void) const
|
|
|
|
{
|
|
|
|
if (m_ha_not_exact_count)
|
|
|
|
return m_table_flags | HA_NOT_EXACT_COUNT;
|
|
|
|
else
|
|
|
|
return m_table_flags;
|
|
|
|
}
|
|
|
|
const char * ha_ndbcluster::table_type() const
|
|
|
|
{
|
|
|
|
return("ndbcluster");
|
|
|
|
}
|
|
|
|
uint ha_ndbcluster::max_supported_record_length() const
|
|
|
|
{
|
|
|
|
return NDB_MAX_TUPLE_SIZE;
|
|
|
|
}
|
|
|
|
uint ha_ndbcluster::max_supported_keys() const
|
|
|
|
{
|
|
|
|
return MAX_KEY;
|
|
|
|
}
|
|
|
|
uint ha_ndbcluster::max_supported_key_parts() const
|
|
|
|
{
|
|
|
|
return NDB_MAX_NO_OF_ATTRIBUTES_IN_KEY;
|
|
|
|
}
|
|
|
|
uint ha_ndbcluster::max_supported_key_length() const
|
|
|
|
{
|
|
|
|
return NDB_MAX_KEY_SIZE;
|
|
|
|
}
|
2006-02-07 19:57:31 +01:00
|
|
|
uint ha_ndbcluster::max_supported_key_part_length() const
|
|
|
|
{
|
|
|
|
return NDB_MAX_KEY_SIZE;
|
|
|
|
}
|
2004-11-17 09:15:53 +01:00
|
|
|
bool ha_ndbcluster::low_byte_first() const
|
|
|
|
{
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
return FALSE;
|
|
|
|
#else
|
|
|
|
return TRUE;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
bool ha_ndbcluster::has_transactions()
|
|
|
|
{
|
2006-02-07 00:03:39 +01:00
|
|
|
return TRUE;
|
2004-11-17 09:15:53 +01:00
|
|
|
}
|
|
|
|
const char* ha_ndbcluster::index_type(uint key_number)
|
|
|
|
{
|
|
|
|
switch (get_index_type(key_number)) {
|
|
|
|
case ORDERED_INDEX:
|
|
|
|
case UNIQUE_ORDERED_INDEX:
|
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
|
|
|
return "BTREE";
|
|
|
|
case UNIQUE_INDEX:
|
|
|
|
case PRIMARY_KEY_INDEX:
|
|
|
|
default:
|
|
|
|
return "HASH";
|
|
|
|
}
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2004-11-17 09:15:53 +01:00
|
|
|
uint8 ha_ndbcluster::table_cache_type()
|
|
|
|
{
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::table_cache_type=HA_CACHE_TBL_ASKTRANSACT");
|
|
|
|
DBUG_RETURN(HA_CACHE_TBL_ASKTRANSACT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint ndb_get_commitcount(THD *thd, char *dbname, char *tabname,
|
2005-02-16 14:18:32 +01:00
|
|
|
Uint64 *commit_count)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("ndb_get_commitcount");
|
|
|
|
|
2005-03-15 15:03:25 +01:00
|
|
|
char name[FN_REFLEN];
|
|
|
|
NDB_SHARE *share;
|
|
|
|
(void)strxnmov(name, FN_REFLEN, "./",dbname,"/",tabname,NullS);
|
|
|
|
DBUG_PRINT("enter", ("name: %s", name));
|
|
|
|
pthread_mutex_lock(&ndbcluster_mutex);
|
|
|
|
if (!(share=(NDB_SHARE*) hash_search(&ndbcluster_open_tables,
|
|
|
|
(byte*) name,
|
|
|
|
strlen(name))))
|
|
|
|
{
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
DBUG_PRINT("info", ("Table %s not found in ndbcluster_open_tables",
|
|
|
|
name));
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
share->use_count++;
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
|
|
|
|
pthread_mutex_lock(&share->mutex);
|
2005-02-11 22:33:52 +01:00
|
|
|
if (ndb_cache_check_time > 0)
|
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
if (share->commit_count != 0)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
*commit_count= share->commit_count;
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("info", ("Getting commit_count: %s from share",
|
|
|
|
llstr(share->commit_count, buff)));
|
2005-03-15 15:03:25 +01:00
|
|
|
pthread_mutex_unlock(&share->mutex);
|
|
|
|
free_share(share);
|
|
|
|
DBUG_RETURN(0);
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_PRINT("info", ("Get commit_count from NDB"));
|
2005-02-11 22:33:52 +01:00
|
|
|
Ndb *ndb;
|
|
|
|
if (!(ndb= check_ndb_in_thd(thd)))
|
|
|
|
DBUG_RETURN(1);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(dbname))
|
|
|
|
{
|
|
|
|
ERR_RETURN(ndb->getNdbError());
|
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
uint lock= share->commit_count_lock;
|
|
|
|
pthread_mutex_unlock(&share->mutex);
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
struct Ndb_statistics stat;
|
2006-10-24 15:26:58 +02:00
|
|
|
if (ndb_get_table_statistics(NULL, false, ndb, tabname, &stat))
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
|
|
|
free_share(share);
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(1);
|
2005-03-15 15:03:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pthread_mutex_lock(&share->mutex);
|
2005-07-04 02:42:33 +02:00
|
|
|
if (share->commit_count_lock == lock)
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("info", ("Setting commit_count to %s",
|
|
|
|
llstr(stat.commit_count, buff)));
|
2005-03-15 15:03:25 +01:00
|
|
|
share->commit_count= stat.commit_count;
|
|
|
|
*commit_count= stat.commit_count;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("Discarding commit_count, comit_count_lock changed"));
|
|
|
|
*commit_count= 0;
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&share->mutex);
|
|
|
|
free_share(share);
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Check if a cached query can be used.
|
|
|
|
This is done by comparing the supplied engine_data to commit_count of
|
|
|
|
the table.
|
|
|
|
The commit_count is either retrieved from the share for the table, where
|
|
|
|
it has been cached by the util thread. If the util thread is not started,
|
|
|
|
NDB has to be contacetd to retrieve the commit_count, this will introduce
|
|
|
|
a small delay while waiting for NDB to answer.
|
|
|
|
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
ndbcluster_cache_retrieval_allowed
|
|
|
|
thd thread handle
|
|
|
|
full_name concatenation of database name,
|
|
|
|
the null character '\0', and the table
|
|
|
|
name
|
|
|
|
full_name_len length of the full name,
|
|
|
|
i.e. len(dbname) + len(tablename) + 1
|
|
|
|
|
|
|
|
engine_data parameter retrieved when query was first inserted into
|
|
|
|
the cache. If the value of engine_data is changed,
|
|
|
|
all queries for this table should be invalidated.
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
TRUE Yes, use the query from cache
|
|
|
|
FALSE No, don't use the cached query, and if engine_data
|
|
|
|
has changed, all queries for this table should be invalidated
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
static my_bool
|
|
|
|
ndbcluster_cache_retrieval_allowed(THD *thd,
|
2005-02-16 14:18:32 +01:00
|
|
|
char *full_name, uint full_name_len,
|
|
|
|
ulonglong *engine_data)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
Uint64 commit_count;
|
|
|
|
bool is_autocommit= !(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN));
|
|
|
|
char *dbname= full_name;
|
|
|
|
char *tabname= dbname+strlen(dbname)+1;
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22], buff2[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_ENTER("ndbcluster_cache_retrieval_allowed");
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_PRINT("enter", ("dbname: %s, tabname: %s, is_autocommit: %d",
|
|
|
|
dbname, tabname, is_autocommit));
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
if (!is_autocommit)
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("exit", ("No, don't use cache in transaction"));
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
2005-03-15 15:03:25 +01:00
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
if (ndb_get_commitcount(thd, dbname, tabname, &commit_count))
|
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
*engine_data= 0; /* invalidate */
|
|
|
|
DBUG_PRINT("exit", ("No, could not retrieve commit_count"));
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("info", ("*engine_data: %s, commit_count: %s",
|
|
|
|
llstr(*engine_data, buff), llstr(commit_count, buff2)));
|
2005-03-15 15:03:25 +01:00
|
|
|
if (commit_count == 0)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
*engine_data= 0; /* invalidate */
|
|
|
|
DBUG_PRINT("exit", ("No, local commit has been performed"));
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
2005-03-15 15:03:25 +01:00
|
|
|
else if (*engine_data != commit_count)
|
|
|
|
{
|
|
|
|
*engine_data= commit_count; /* invalidate */
|
|
|
|
DBUG_PRINT("exit", ("No, commit_count has changed"));
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("exit", ("OK to use cache, engine_data: %s",
|
|
|
|
llstr(*engine_data, buff)));
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Register a table for use in the query cache. Fetch the commit_count
|
|
|
|
for the table and return it in engine_data, this will later be used
|
|
|
|
to check if the table has changed, before the cached query is reused.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
ha_ndbcluster::can_query_cache_table
|
|
|
|
thd thread handle
|
|
|
|
full_name concatenation of database name,
|
|
|
|
the null character '\0', and the table
|
|
|
|
name
|
|
|
|
full_name_len length of the full name,
|
|
|
|
i.e. len(dbname) + len(tablename) + 1
|
|
|
|
qc_engine_callback function to be called before using cache on this table
|
|
|
|
engine_data out, commit_count for this table
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
TRUE Yes, it's ok to cahce this query
|
|
|
|
FALSE No, don't cach the query
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
my_bool
|
|
|
|
ha_ndbcluster::register_query_cache_table(THD *thd,
|
2005-02-16 14:18:32 +01:00
|
|
|
char *full_name, uint full_name_len,
|
|
|
|
qc_engine_callback *engine_callback,
|
|
|
|
ulonglong *engine_data)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
2006-06-30 18:15:18 +02:00
|
|
|
Uint64 commit_count;
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2005-02-11 22:33:52 +01:00
|
|
|
bool is_autocommit= !(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN));
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_ENTER("ha_ndbcluster::register_query_cache_table");
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_PRINT("enter",("dbname: %s, tabname: %s, is_autocommit: %d",
|
|
|
|
m_dbname, m_tabname, is_autocommit));
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
if (!is_autocommit)
|
2005-03-15 15:03:25 +01:00
|
|
|
{
|
|
|
|
DBUG_PRINT("exit", ("Can't register table during transaction"))
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
2005-03-15 15:03:25 +01:00
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
if (ndb_get_commitcount(thd, m_dbname, m_tabname, &commit_count))
|
|
|
|
{
|
|
|
|
*engine_data= 0;
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_PRINT("exit", ("Error, could not get commitcount"))
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
*engine_data= commit_count;
|
|
|
|
*engine_callback= ndbcluster_cache_retrieval_allowed;
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("exit", ("commit_count: %s", llstr(commit_count, buff)));
|
2005-03-15 15:03:25 +01:00
|
|
|
DBUG_RETURN(commit_count > 0);
|
2004-11-17 09:15:53 +01:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
/*
|
2005-02-11 22:33:52 +01:00
|
|
|
Handling the shared NDB_SHARE structure that is needed to
|
2004-04-15 09:14:14 +02:00
|
|
|
provide table locking.
|
|
|
|
It's also used for sharing data with other NDB handlers
|
|
|
|
in the same MySQL Server. There is currently not much
|
|
|
|
data we want to or can share.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static byte* ndbcluster_get_key(NDB_SHARE *share,uint *length,
|
2005-02-16 14:18:32 +01:00
|
|
|
my_bool not_used __attribute__((unused)))
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
*length=share->table_name_length;
|
|
|
|
return (byte*) share->table_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
static NDB_SHARE* get_share(const char *table_name)
|
|
|
|
{
|
|
|
|
NDB_SHARE *share;
|
|
|
|
pthread_mutex_lock(&ndbcluster_mutex);
|
|
|
|
uint length=(uint) strlen(table_name);
|
|
|
|
if (!(share=(NDB_SHARE*) hash_search(&ndbcluster_open_tables,
|
|
|
|
(byte*) table_name,
|
|
|
|
length)))
|
|
|
|
{
|
|
|
|
if ((share=(NDB_SHARE *) my_malloc(sizeof(*share)+length+1,
|
|
|
|
MYF(MY_WME | MY_ZEROFILL))))
|
|
|
|
{
|
|
|
|
share->table_name_length=length;
|
|
|
|
share->table_name=(char*) (share+1);
|
|
|
|
strmov(share->table_name,table_name);
|
|
|
|
if (my_hash_insert(&ndbcluster_open_tables, (byte*) share))
|
|
|
|
{
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
my_free((gptr) share,0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
thr_lock_init(&share->lock);
|
|
|
|
pthread_mutex_init(&share->mutex,MY_MUTEX_INIT_FAST);
|
2005-02-11 22:33:52 +01:00
|
|
|
share->commit_count= 0;
|
2005-03-15 15:03:25 +01:00
|
|
|
share->commit_count_lock= 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("error", ("Failed to alloc share"));
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
2007-04-11 15:21:11 +02:00
|
|
|
sql_print_error("get_share: my_malloc(%u) failed",
|
|
|
|
(unsigned int)(sizeof(*share)+length+1));
|
2005-03-15 15:03:25 +01:00
|
|
|
return 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
share->use_count++;
|
2005-03-15 15:03:25 +01:00
|
|
|
|
|
|
|
DBUG_PRINT("share",
|
2006-11-20 21:42:06 +01:00
|
|
|
("table_name: %s length: %d use_count: %d commit_count: %lu",
|
2005-03-15 15:03:25 +01:00
|
|
|
share->table_name, share->table_name_length, share->use_count,
|
2006-11-20 21:42:06 +01:00
|
|
|
(ulong) share->commit_count));
|
2004-04-15 09:14:14 +02:00
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
return share;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void free_share(NDB_SHARE *share)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&ndbcluster_mutex);
|
|
|
|
if (!--share->use_count)
|
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
hash_delete(&ndbcluster_open_tables, (byte*) share);
|
2004-04-15 09:14:14 +02:00
|
|
|
thr_lock_delete(&share->lock);
|
|
|
|
pthread_mutex_destroy(&share->mutex);
|
|
|
|
my_free((gptr) share, MYF(0));
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Internal representation of the frm blob
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct frm_blob_struct
|
|
|
|
{
|
|
|
|
struct frm_blob_header
|
|
|
|
{
|
|
|
|
uint ver; // Version of header
|
|
|
|
uint orglen; // Original length of compressed data
|
|
|
|
uint complen; // Compressed length of data, 0=uncompressed
|
|
|
|
} head;
|
|
|
|
char data[1];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int packfrm(const void *data, uint len,
|
2005-02-16 14:18:32 +01:00
|
|
|
const void **pack_data, uint *pack_len)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
|
|
|
int error;
|
|
|
|
ulong org_len, comp_len;
|
|
|
|
uint blob_len;
|
|
|
|
frm_blob_struct* blob;
|
|
|
|
DBUG_ENTER("packfrm");
|
2007-03-28 19:46:42 +02:00
|
|
|
DBUG_PRINT("enter", ("data: 0x%lx len: %d", (long) data, len));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
error= 1;
|
2004-07-23 15:46:56 +02:00
|
|
|
org_len= len;
|
2004-04-15 09:14:14 +02:00
|
|
|
if (my_compress((byte*)data, &org_len, &comp_len))
|
2007-04-11 15:21:11 +02:00
|
|
|
{
|
|
|
|
sql_print_error("packfrm: my_compress(org_len: %u)",
|
|
|
|
(unsigned int)org_len);
|
2004-04-15 09:14:14 +02:00
|
|
|
goto err;
|
2007-04-11 15:21:11 +02:00
|
|
|
}
|
|
|
|
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("info", ("org_len: %lu comp_len: %lu", org_len, comp_len));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_DUMP("compressed", (char*)data, org_len);
|
|
|
|
|
|
|
|
error= 2;
|
|
|
|
blob_len= sizeof(frm_blob_struct::frm_blob_header)+org_len;
|
|
|
|
if (!(blob= (frm_blob_struct*) my_malloc(blob_len,MYF(MY_WME))))
|
2007-04-11 15:21:11 +02:00
|
|
|
{
|
|
|
|
sql_print_error("packfrm: my_malloc(%u)", blob_len);
|
2004-04-15 09:14:14 +02:00
|
|
|
goto err;
|
2007-04-11 15:21:11 +02:00
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
// Store compressed blob in machine independent format
|
|
|
|
int4store((char*)(&blob->head.ver), 1);
|
|
|
|
int4store((char*)(&blob->head.orglen), comp_len);
|
|
|
|
int4store((char*)(&blob->head.complen), org_len);
|
|
|
|
|
|
|
|
// Copy frm data into blob, already in machine independent format
|
|
|
|
memcpy(blob->data, data, org_len);
|
|
|
|
|
2004-07-23 15:46:56 +02:00
|
|
|
*pack_data= blob;
|
|
|
|
*pack_len= blob_len;
|
|
|
|
error= 0;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2007-03-28 19:46:42 +02:00
|
|
|
DBUG_PRINT("exit", ("pack_data: 0x%lx pack_len: %d", (long) *pack_data,
|
2007-03-22 19:32:07 +01:00
|
|
|
*pack_len));
|
2004-04-15 09:14:14 +02:00
|
|
|
err:
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int unpackfrm(const void **unpack_data, uint *unpack_len,
|
2005-02-16 14:18:32 +01:00
|
|
|
const void *pack_data)
|
2004-04-15 09:14:14 +02:00
|
|
|
{
|
2004-07-23 15:46:56 +02:00
|
|
|
const frm_blob_struct *blob= (frm_blob_struct*)pack_data;
|
2004-04-15 09:14:14 +02:00
|
|
|
byte *data;
|
|
|
|
ulong complen, orglen, ver;
|
|
|
|
DBUG_ENTER("unpackfrm");
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("enter", ("pack_data: 0x%lx", (long) pack_data));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2005-02-16 14:18:32 +01:00
|
|
|
complen= uint4korr((char*)&blob->head.complen);
|
|
|
|
orglen= uint4korr((char*)&blob->head.orglen);
|
|
|
|
ver= uint4korr((char*)&blob->head.ver);
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("blob",("ver: %lu complen: %lu orglen: %lu",
|
2005-02-16 14:18:32 +01:00
|
|
|
ver,complen,orglen));
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_DUMP("blob->data", (char*) blob->data, complen);
|
|
|
|
|
|
|
|
if (ver != 1)
|
2007-04-11 15:21:11 +02:00
|
|
|
{
|
|
|
|
sql_print_error("unpackfrm: ver != 1");
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(1);
|
2007-04-11 15:21:11 +02:00
|
|
|
}
|
2004-07-23 15:46:56 +02:00
|
|
|
if (!(data= my_malloc(max(orglen, complen), MYF(MY_WME))))
|
2007-04-11 15:21:11 +02:00
|
|
|
{
|
|
|
|
sql_print_error("unpackfrm: my_malloc(%u)",
|
|
|
|
(unsigned int)max(orglen, complen));
|
|
|
|
DBUG_RETURN(HA_ERR_OUT_OF_MEM);
|
|
|
|
}
|
2004-04-15 09:14:14 +02:00
|
|
|
memcpy(data, blob->data, complen);
|
|
|
|
|
|
|
|
if (my_uncompress(data, &complen, &orglen))
|
|
|
|
{
|
|
|
|
my_free((char*)data, MYF(0));
|
2007-04-11 15:21:11 +02:00
|
|
|
sql_print_error("unpackfrm: my_uncompress(complen: %u, orglen: %u)",
|
|
|
|
(unsigned int)complen, (unsigned int)orglen);
|
2004-04-15 09:14:14 +02:00
|
|
|
DBUG_RETURN(3);
|
|
|
|
}
|
|
|
|
|
2004-07-23 15:46:56 +02:00
|
|
|
*unpack_data= data;
|
|
|
|
*unpack_len= complen;
|
2004-04-15 09:14:14 +02:00
|
|
|
|
2007-03-28 19:46:42 +02:00
|
|
|
DBUG_PRINT("exit", ("frmdata: 0x%lx len: %d", (long) *unpack_data,
|
2007-03-22 19:32:07 +01:00
|
|
|
*unpack_len));
|
2004-04-15 09:14:14 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
|
|
|
static
|
|
|
|
int
|
2006-10-23 15:46:35 +02:00
|
|
|
ndb_get_table_statistics(ha_ndbcluster* file, bool report_error, Ndb* ndb,
|
2006-10-23 16:18:57 +02:00
|
|
|
const char* table,
|
2005-02-16 14:18:32 +01:00
|
|
|
struct Ndb_statistics * ndbstat)
|
2004-09-03 13:55:40 +02:00
|
|
|
{
|
2006-04-26 16:57:44 +02:00
|
|
|
NdbTransaction* pTrans;
|
2006-06-29 20:55:21 +02:00
|
|
|
NdbError error;
|
2006-04-26 16:57:44 +02:00
|
|
|
int retries= 10;
|
2006-10-23 15:57:28 +02:00
|
|
|
int reterr= 0;
|
2007-04-25 15:21:38 +02:00
|
|
|
int retry_sleep= 30; /* 30 milliseconds, transaction */
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22], buff2[22], buff3[22], buff4[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_ENTER("ndb_get_table_statistics");
|
|
|
|
DBUG_PRINT("enter", ("table: %s", table));
|
2006-04-26 16:57:44 +02:00
|
|
|
|
|
|
|
do
|
2004-09-03 13:55:40 +02:00
|
|
|
{
|
2006-06-29 21:05:43 +02:00
|
|
|
Uint64 rows, commits, mem;
|
|
|
|
Uint32 size;
|
2006-06-29 22:11:29 +02:00
|
|
|
Uint32 count= 0;
|
2006-06-29 20:55:21 +02:00
|
|
|
Uint64 sum_rows= 0;
|
|
|
|
Uint64 sum_commits= 0;
|
2006-06-29 22:11:29 +02:00
|
|
|
Uint64 sum_row_size= 0;
|
|
|
|
Uint64 sum_mem= 0;
|
2006-06-29 20:55:21 +02:00
|
|
|
NdbScanOperation*pOp;
|
|
|
|
int check;
|
|
|
|
|
|
|
|
if ((pTrans= ndb->startTransaction()) == NULL)
|
2006-04-26 16:57:44 +02:00
|
|
|
{
|
2006-06-29 20:55:21 +02:00
|
|
|
error= ndb->getNdbError();
|
|
|
|
goto retry;
|
|
|
|
}
|
2004-11-22 14:41:46 +01:00
|
|
|
|
2006-06-29 20:55:21 +02:00
|
|
|
if ((pOp= pTrans->getNdbScanOperation(table)) == NULL)
|
|
|
|
{
|
|
|
|
error= pTrans->getNdbError();
|
|
|
|
goto retry;
|
2006-04-26 16:57:44 +02:00
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2004-12-07 15:15:49 +01:00
|
|
|
if (pOp->readTuples(NdbOperation::LM_CommittedRead))
|
2006-06-29 20:55:21 +02:00
|
|
|
{
|
|
|
|
error= pOp->getNdbError();
|
|
|
|
goto retry;
|
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2006-06-29 20:55:21 +02:00
|
|
|
if (pOp->interpret_exit_last_row() == -1)
|
|
|
|
{
|
|
|
|
error= pOp->getNdbError();
|
|
|
|
goto retry;
|
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
|
|
|
pOp->getValue(NdbDictionary::Column::ROW_COUNT, (char*)&rows);
|
|
|
|
pOp->getValue(NdbDictionary::Column::COMMIT_COUNT, (char*)&commits);
|
2005-01-13 18:24:19 +01:00
|
|
|
pOp->getValue(NdbDictionary::Column::ROW_SIZE, (char*)&size);
|
|
|
|
pOp->getValue(NdbDictionary::Column::FRAGMENT_MEMORY, (char*)&mem);
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2006-06-29 21:05:43 +02:00
|
|
|
if (pTrans->execute(NdbTransaction::NoCommit,
|
|
|
|
NdbTransaction::AbortOnError,
|
|
|
|
TRUE) == -1)
|
2006-04-26 16:57:44 +02:00
|
|
|
{
|
2006-06-29 20:55:21 +02:00
|
|
|
error= pTrans->getNdbError();
|
|
|
|
goto retry;
|
2006-04-26 16:57:44 +02:00
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2005-07-18 14:33:18 +02:00
|
|
|
while ((check= pOp->nextResult(TRUE, TRUE)) == 0)
|
2004-09-03 13:55:40 +02:00
|
|
|
{
|
|
|
|
sum_rows+= rows;
|
|
|
|
sum_commits+= commits;
|
2005-01-13 18:24:19 +01:00
|
|
|
if (sum_row_size < size)
|
2005-02-16 14:18:32 +01:00
|
|
|
sum_row_size= size;
|
2005-01-13 18:24:19 +01:00
|
|
|
sum_mem+= mem;
|
2005-03-15 15:03:25 +01:00
|
|
|
count++;
|
2004-09-03 13:55:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (check == -1)
|
2006-06-29 20:55:21 +02:00
|
|
|
{
|
|
|
|
error= pOp->getNdbError();
|
|
|
|
goto retry;
|
|
|
|
}
|
2004-09-03 13:55:40 +02:00
|
|
|
|
2004-12-07 15:15:49 +01:00
|
|
|
pOp->close(TRUE);
|
2004-11-22 14:41:46 +01:00
|
|
|
|
2004-09-03 13:55:40 +02:00
|
|
|
ndb->closeTransaction(pTrans);
|
2005-01-13 18:24:19 +01:00
|
|
|
|
|
|
|
ndbstat->row_count= sum_rows;
|
|
|
|
ndbstat->commit_count= sum_commits;
|
|
|
|
ndbstat->row_size= sum_row_size;
|
|
|
|
ndbstat->fragment_memory= sum_mem;
|
|
|
|
|
2006-06-30 18:15:18 +02:00
|
|
|
DBUG_PRINT("exit", ("records: %s commits: %s "
|
|
|
|
"row_size: %s mem: %s count: %u",
|
|
|
|
llstr(sum_rows, buff),
|
|
|
|
llstr(sum_commits, buff2),
|
|
|
|
llstr(sum_row_size, buff3),
|
|
|
|
llstr(sum_mem, buff4),
|
|
|
|
count));
|
2005-03-15 15:03:25 +01:00
|
|
|
|
2004-09-03 13:55:40 +02:00
|
|
|
DBUG_RETURN(0);
|
2006-06-29 20:55:21 +02:00
|
|
|
retry:
|
2007-01-17 10:41:52 +01:00
|
|
|
if(report_error)
|
|
|
|
{
|
|
|
|
if (file && pTrans)
|
|
|
|
{
|
|
|
|
reterr= file->ndb_err(pTrans);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const NdbError& tmp= error;
|
|
|
|
ERR_PRINT(tmp);
|
|
|
|
reterr= ndb_to_mysql_error(&tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
reterr= error.code;
|
2006-10-24 15:26:58 +02:00
|
|
|
|
2007-01-17 10:41:52 +01:00
|
|
|
if (pTrans)
|
2006-06-29 20:55:21 +02:00
|
|
|
{
|
|
|
|
ndb->closeTransaction(pTrans);
|
|
|
|
pTrans= NULL;
|
|
|
|
}
|
|
|
|
if (error.status == NdbError::TemporaryError && retries--)
|
|
|
|
{
|
|
|
|
my_sleep(retry_sleep);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
2006-04-26 16:57:44 +02:00
|
|
|
} while(1);
|
2006-10-23 15:46:35 +02:00
|
|
|
DBUG_PRINT("exit", ("failed, reterr: %u, NdbError %u(%s)", reterr,
|
|
|
|
error.code, error.message));
|
|
|
|
DBUG_RETURN(reterr);
|
2004-09-03 13:55:40 +02:00
|
|
|
}
|
|
|
|
|
2004-09-13 14:46:38 +02:00
|
|
|
/*
|
|
|
|
Create a .ndb file to serve as a placeholder indicating
|
|
|
|
that the table with this name is a ndb table
|
|
|
|
*/
|
|
|
|
|
|
|
|
int ha_ndbcluster::write_ndb_file()
|
|
|
|
{
|
|
|
|
File file;
|
|
|
|
bool error=1;
|
|
|
|
char path[FN_REFLEN];
|
|
|
|
|
|
|
|
DBUG_ENTER("write_ndb_file");
|
|
|
|
DBUG_PRINT("enter", ("db: %s, name: %s", m_dbname, m_tabname));
|
|
|
|
|
|
|
|
(void)strxnmov(path, FN_REFLEN,
|
2005-02-16 14:18:32 +01:00
|
|
|
mysql_data_home,"/",m_dbname,"/",m_tabname,ha_ndb_ext,NullS);
|
2004-09-13 14:46:38 +02:00
|
|
|
|
|
|
|
if ((file=my_create(path, CREATE_MODE,O_RDWR | O_TRUNC,MYF(MY_WME))) >= 0)
|
|
|
|
{
|
|
|
|
// It's an empty file
|
|
|
|
error=0;
|
|
|
|
my_close(file,MYF(0));
|
|
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
2006-08-15 13:12:27 +02:00
|
|
|
void
|
2006-08-15 14:31:21 +02:00
|
|
|
ha_ndbcluster::release_completed_operations(NdbTransaction *trans,
|
|
|
|
bool force_release)
|
2006-08-15 13:12:27 +02:00
|
|
|
{
|
|
|
|
if (trans->hasBlobOperation())
|
|
|
|
{
|
|
|
|
/* We are reading/writing BLOB fields,
|
|
|
|
releasing operation records is unsafe
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
}
|
2006-08-15 14:31:21 +02:00
|
|
|
if (!force_release)
|
|
|
|
{
|
|
|
|
if (get_thd_ndb(current_thd)->query_state & NDB_QUERY_MULTI_READ_RANGE)
|
|
|
|
{
|
|
|
|
/* We are batching reads and have not consumed all fetched
|
|
|
|
rows yet, releasing operation records is unsafe
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2006-08-15 13:12:27 +02:00
|
|
|
trans->releaseCompletedOperations();
|
2006-08-15 14:31:21 +02:00
|
|
|
}
|
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
bool
|
|
|
|
ha_ndbcluster::null_value_index_search(KEY_MULTI_RANGE *ranges,
|
|
|
|
KEY_MULTI_RANGE *end_range,
|
|
|
|
HANDLER_BUFFER *buffer)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("null_value_index_search");
|
|
|
|
KEY* key_info= table->key_info + active_index;
|
|
|
|
KEY_MULTI_RANGE *range= ranges;
|
|
|
|
ulong reclength= table->s->reclength;
|
|
|
|
byte *curr= (byte*)buffer->buffer;
|
|
|
|
byte *end_of_buffer= (byte*)buffer->buffer_end;
|
|
|
|
|
|
|
|
for (; range<end_range && curr+reclength <= end_of_buffer;
|
|
|
|
range++)
|
|
|
|
{
|
|
|
|
const byte *key= range->start_key.key;
|
|
|
|
uint key_len= range->start_key.length;
|
|
|
|
if (check_null_in_key(key_info, key, key_len))
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
curr += reclength;
|
|
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
|
|
}
|
|
|
|
|
2004-11-17 10:07:52 +01:00
|
|
|
int
|
2004-12-28 17:01:07 +01:00
|
|
|
ha_ndbcluster::read_multi_range_first(KEY_MULTI_RANGE **found_range_p,
|
2005-02-16 14:18:32 +01:00
|
|
|
KEY_MULTI_RANGE *ranges,
|
|
|
|
uint range_count,
|
|
|
|
bool sorted,
|
|
|
|
HANDLER_BUFFER *buffer)
|
2004-11-17 10:07:52 +01:00
|
|
|
{
|
|
|
|
int res;
|
|
|
|
KEY* key_info= table->key_info + active_index;
|
2006-12-14 23:51:37 +01:00
|
|
|
NDB_INDEX_TYPE cur_index_type= get_index_type(active_index);
|
2005-01-07 07:15:08 +01:00
|
|
|
ulong reclength= table->s->reclength;
|
2004-11-17 10:07:52 +01:00
|
|
|
NdbOperation* op;
|
2006-08-15 14:31:21 +02:00
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(current_thd);
|
2006-12-15 05:21:15 +01:00
|
|
|
DBUG_ENTER("ha_ndbcluster::read_multi_range_first");
|
2004-11-17 10:07:52 +01:00
|
|
|
|
2006-11-07 16:38:37 +01:00
|
|
|
/**
|
|
|
|
* blobs and unique hash index with NULL can't be batched currently
|
|
|
|
*/
|
|
|
|
if (uses_blob_value(m_retrieve_all_fields) ||
|
2006-12-15 05:21:15 +01:00
|
|
|
(cur_index_type == UNIQUE_INDEX &&
|
2006-11-07 16:38:37 +01:00
|
|
|
has_null_in_unique_index(active_index) &&
|
2007-08-13 11:36:30 +02:00
|
|
|
null_value_index_search(ranges, ranges+range_count, buffer))
|
|
|
|
|| m_delete_cannot_batch || m_update_cannot_batch)
|
2004-11-17 14:21:56 +01:00
|
|
|
{
|
2005-05-06 10:39:30 +02:00
|
|
|
m_disable_multi_read= TRUE;
|
2004-11-25 19:12:05 +01:00
|
|
|
DBUG_RETURN(handler::read_multi_range_first(found_range_p,
|
2005-02-16 14:18:32 +01:00
|
|
|
ranges,
|
|
|
|
range_count,
|
|
|
|
sorted,
|
|
|
|
buffer));
|
2004-11-17 14:21:56 +01:00
|
|
|
}
|
2006-08-15 14:31:21 +02:00
|
|
|
thd_ndb->query_state|= NDB_QUERY_MULTI_READ_RANGE;
|
2005-05-06 10:39:30 +02:00
|
|
|
m_disable_multi_read= FALSE;
|
2004-11-30 07:58:55 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy arguments into member variables
|
|
|
|
*/
|
2004-12-28 17:01:07 +01:00
|
|
|
m_multi_ranges= ranges;
|
|
|
|
multi_range_curr= ranges;
|
|
|
|
multi_range_end= ranges+range_count;
|
2004-11-17 10:07:52 +01:00
|
|
|
multi_range_sorted= sorted;
|
|
|
|
multi_range_buffer= buffer;
|
|
|
|
|
2004-11-30 07:58:55 +01:00
|
|
|
/**
|
|
|
|
* read multi range will read ranges as follows (if not ordered)
|
|
|
|
*
|
|
|
|
* input read order
|
|
|
|
* ====== ==========
|
|
|
|
* pk-op 1 pk-op 1
|
|
|
|
* pk-op 2 pk-op 2
|
|
|
|
* range 3 range (3,5) NOTE result rows will be intermixed
|
|
|
|
* pk-op 4 pk-op 4
|
|
|
|
* range 5
|
|
|
|
* pk-op 6 pk-ok 6
|
2005-01-11 17:00:31 +01:00
|
|
|
*/
|
|
|
|
|
2005-01-11 14:06:44 +01:00
|
|
|
/**
|
2004-11-30 07:58:55 +01:00
|
|
|
* Variables for loop
|
|
|
|
*/
|
2004-12-08 00:36:40 +01:00
|
|
|
byte *curr= (byte*)buffer->buffer;
|
|
|
|
byte *end_of_buffer= (byte*)buffer->buffer_end;
|
2004-11-17 10:07:52 +01:00
|
|
|
NdbOperation::LockMode lm=
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type);
|
2006-06-12 09:37:19 +02:00
|
|
|
bool need_pk = (lm == NdbOperation::LM_Read);
|
2004-11-17 10:07:52 +01:00
|
|
|
const NDBTAB *tab= (const NDBTAB *) m_table;
|
|
|
|
const NDBINDEX *unique_idx= (NDBINDEX *) m_index[active_index].unique_index;
|
2004-11-30 07:58:55 +01:00
|
|
|
const NDBINDEX *idx= (NDBINDEX *) m_index[active_index].index;
|
2004-11-26 15:15:22 +01:00
|
|
|
const NdbOperation* lastOp= m_active_trans->getLastDefinedOperation();
|
|
|
|
NdbIndexScanOperation* scanOp= 0;
|
2005-06-01 15:35:09 +02:00
|
|
|
for (; multi_range_curr<multi_range_end && curr+reclength <= end_of_buffer;
|
|
|
|
multi_range_curr++)
|
2004-11-26 15:15:22 +01:00
|
|
|
{
|
2006-12-14 23:51:37 +01:00
|
|
|
switch (cur_index_type) {
|
2005-07-04 02:42:33 +02:00
|
|
|
case PRIMARY_KEY_ORDERED_INDEX:
|
|
|
|
if (!(multi_range_curr->start_key.length == key_info->key_length &&
|
|
|
|
multi_range_curr->start_key.flag == HA_READ_KEY_EXACT))
|
|
|
|
goto range;
|
|
|
|
/* fall through */
|
2004-11-26 15:15:22 +01:00
|
|
|
case PRIMARY_KEY_INDEX:
|
2004-12-28 17:01:07 +01:00
|
|
|
multi_range_curr->range_flag |= UNIQUE_RANGE;
|
2004-11-17 10:07:52 +01:00
|
|
|
if ((op= m_active_trans->getNdbOperation(tab)) &&
|
2005-02-16 14:18:32 +01:00
|
|
|
!op->readTuple(lm) &&
|
|
|
|
!set_primary_key(op, multi_range_curr->start_key.key) &&
|
|
|
|
!define_read_attrs(curr, op) &&
|
2005-05-06 10:39:30 +02:00
|
|
|
(op->setAbortOption(AO_IgnoreError), TRUE))
|
2005-02-16 14:18:32 +01:00
|
|
|
curr += reclength;
|
2004-11-17 10:07:52 +01:00
|
|
|
else
|
2005-02-16 14:18:32 +01:00
|
|
|
ERR_RETURN(op ? op->getNdbError() : m_active_trans->getNdbError());
|
2004-11-26 15:15:22 +01:00
|
|
|
break;
|
2005-07-04 02:42:33 +02:00
|
|
|
case UNIQUE_ORDERED_INDEX:
|
|
|
|
if (!(multi_range_curr->start_key.length == key_info->key_length &&
|
|
|
|
multi_range_curr->start_key.flag == HA_READ_KEY_EXACT &&
|
|
|
|
!check_null_in_key(key_info, multi_range_curr->start_key.key,
|
|
|
|
multi_range_curr->start_key.length)))
|
|
|
|
goto range;
|
|
|
|
/* fall through */
|
2004-11-26 15:15:22 +01:00
|
|
|
case UNIQUE_INDEX:
|
2004-12-28 17:01:07 +01:00
|
|
|
multi_range_curr->range_flag |= UNIQUE_RANGE;
|
2004-11-17 10:07:52 +01:00
|
|
|
if ((op= m_active_trans->getNdbIndexOperation(unique_idx, tab)) &&
|
2006-11-07 16:38:37 +01:00
|
|
|
!op->readTuple(lm) &&
|
|
|
|
!set_index_key(op, key_info, multi_range_curr->start_key.key) &&
|
|
|
|
!define_read_attrs(curr, op) &&
|
|
|
|
(op->setAbortOption(AO_IgnoreError), TRUE))
|
|
|
|
curr += reclength;
|
2004-11-17 10:07:52 +01:00
|
|
|
else
|
2006-11-07 16:38:37 +01:00
|
|
|
ERR_RETURN(op ? op->getNdbError() : m_active_trans->getNdbError());
|
2004-11-26 15:15:22 +01:00
|
|
|
break;
|
2005-07-04 02:42:33 +02:00
|
|
|
case ORDERED_INDEX:
|
|
|
|
{
|
2004-11-26 15:15:22 +01:00
|
|
|
range:
|
2004-12-28 17:01:07 +01:00
|
|
|
multi_range_curr->range_flag &= ~(uint)UNIQUE_RANGE;
|
2004-11-26 15:15:22 +01:00
|
|
|
if (scanOp == 0)
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
if (m_multi_cursor)
|
|
|
|
{
|
|
|
|
scanOp= m_multi_cursor;
|
|
|
|
DBUG_ASSERT(scanOp->getSorted() == sorted);
|
|
|
|
DBUG_ASSERT(scanOp->getLockMode() ==
|
|
|
|
(NdbOperation::LockMode)get_ndb_lock_type(m_lock.type));
|
2005-07-04 02:42:33 +02:00
|
|
|
if (scanOp->reset_bounds(m_force_send))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(ndb_err(m_active_trans));
|
|
|
|
|
|
|
|
end_of_buffer -= reclength;
|
|
|
|
}
|
|
|
|
else if ((scanOp= m_active_trans->getNdbIndexScanOperation(idx, tab))
|
2006-06-12 09:37:19 +02:00
|
|
|
&&!scanOp->readTuples(lm, 0, parallelism, sorted,
|
2007-02-06 23:06:58 +01:00
|
|
|
FALSE, TRUE, need_pk, TRUE)
|
2007-04-23 11:25:33 +02:00
|
|
|
&&!(m_cond && m_cond->generate_scan_filter(scanOp))
|
2005-02-16 14:18:32 +01:00
|
|
|
&&!define_read_attrs(end_of_buffer-reclength, scanOp))
|
|
|
|
{
|
|
|
|
m_multi_cursor= scanOp;
|
|
|
|
m_multi_range_cursor_result_ptr= end_of_buffer-reclength;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERR_RETURN(scanOp ? scanOp->getNdbError() :
|
|
|
|
m_active_trans->getNdbError());
|
|
|
|
}
|
2004-11-26 15:15:22 +01:00
|
|
|
}
|
2005-01-07 15:33:24 +01:00
|
|
|
|
2004-12-28 17:01:07 +01:00
|
|
|
const key_range *keys[2]= { &multi_range_curr->start_key,
|
2005-02-16 14:18:32 +01:00
|
|
|
&multi_range_curr->end_key };
|
2004-12-28 17:01:07 +01:00
|
|
|
if ((res= set_bounds(scanOp, keys, multi_range_curr-ranges)))
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_RETURN(res);
|
2004-11-26 15:15:22 +01:00
|
|
|
break;
|
2004-11-17 10:07:52 +01:00
|
|
|
}
|
2005-07-04 02:42:33 +02:00
|
|
|
case UNDEFINED_INDEX:
|
2005-01-11 14:06:44 +01:00
|
|
|
DBUG_ASSERT(FALSE);
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
break;
|
|
|
|
}
|
2004-11-17 10:07:52 +01:00
|
|
|
}
|
|
|
|
|
2004-12-28 17:01:07 +01:00
|
|
|
if (multi_range_curr != multi_range_end)
|
2004-11-17 10:07:52 +01:00
|
|
|
{
|
2004-11-30 07:58:55 +01:00
|
|
|
/**
|
|
|
|
* Mark that we're using entire buffer (even if might not) as
|
|
|
|
* we haven't read all ranges for some reason
|
|
|
|
* This as we don't want mysqld to reuse the buffer when we read
|
|
|
|
* the remaining ranges
|
|
|
|
*/
|
2004-11-26 15:15:22 +01:00
|
|
|
buffer->end_of_used_area= (byte*)buffer->buffer_end;
|
2004-11-17 10:07:52 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
buffer->end_of_used_area= curr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set first operation in multi range
|
|
|
|
*/
|
|
|
|
m_current_multi_operation=
|
|
|
|
lastOp ? lastOp->next() : m_active_trans->getFirstDefinedOperation();
|
2006-08-15 14:31:21 +02:00
|
|
|
if (!(res= execute_no_commit_ie(this, m_active_trans, true)))
|
2004-11-17 10:07:52 +01:00
|
|
|
{
|
2004-12-28 17:01:07 +01:00
|
|
|
m_multi_range_defined= multi_range_curr;
|
|
|
|
multi_range_curr= ranges;
|
2004-11-26 15:15:22 +01:00
|
|
|
m_multi_range_result_ptr= (byte*)buffer->buffer;
|
|
|
|
DBUG_RETURN(read_multi_range_next(found_range_p));
|
2004-11-17 10:07:52 +01:00
|
|
|
}
|
|
|
|
ERR_RETURN(m_active_trans->getNdbError());
|
|
|
|
}
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
#if 0
|
|
|
|
#define DBUG_MULTI_RANGE(x) printf("read_multi_range_next: case %d\n", x);
|
|
|
|
#else
|
|
|
|
#define DBUG_MULTI_RANGE(x)
|
|
|
|
#endif
|
|
|
|
|
2004-11-17 10:07:52 +01:00
|
|
|
int
|
2004-12-28 17:01:07 +01:00
|
|
|
ha_ndbcluster::read_multi_range_next(KEY_MULTI_RANGE ** multi_range_found_p)
|
2004-11-17 10:07:52 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("ha_ndbcluster::read_multi_range_next");
|
2004-11-26 15:15:22 +01:00
|
|
|
if (m_disable_multi_read)
|
2004-11-30 07:58:55 +01:00
|
|
|
{
|
2004-11-26 15:15:22 +01:00
|
|
|
DBUG_RETURN(handler::read_multi_range_next(multi_range_found_p));
|
2004-11-30 07:58:55 +01:00
|
|
|
}
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2004-11-26 15:15:22 +01:00
|
|
|
int res;
|
2004-12-07 15:15:49 +01:00
|
|
|
int range_no;
|
2005-01-07 07:15:08 +01:00
|
|
|
ulong reclength= table->s->reclength;
|
2004-11-26 15:15:22 +01:00
|
|
|
const NdbOperation* op= m_current_multi_operation;
|
2005-06-01 15:35:09 +02:00
|
|
|
for (;multi_range_curr < m_multi_range_defined; multi_range_curr++)
|
2004-11-17 10:07:52 +01:00
|
|
|
{
|
2004-12-28 17:01:07 +01:00
|
|
|
if (multi_range_curr->range_flag & UNIQUE_RANGE)
|
2004-11-26 15:15:22 +01:00
|
|
|
{
|
2004-12-07 15:15:49 +01:00
|
|
|
if (op->getNdbError().code == 0)
|
2005-02-16 14:18:32 +01:00
|
|
|
goto found_next;
|
2004-11-26 15:15:22 +01:00
|
|
|
|
|
|
|
op= m_active_trans->getNextCompletedOperation(op);
|
|
|
|
m_multi_range_result_ptr += reclength;
|
2004-12-07 15:15:49 +01:00
|
|
|
continue;
|
2004-11-26 15:15:22 +01:00
|
|
|
}
|
2004-12-07 15:15:49 +01:00
|
|
|
else if (m_multi_cursor && !multi_range_sorted)
|
2004-11-26 15:15:22 +01:00
|
|
|
{
|
2004-12-08 00:36:40 +01:00
|
|
|
DBUG_MULTI_RANGE(1);
|
|
|
|
if ((res= fetch_next(m_multi_cursor)) == 0)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_MULTI_RANGE(2);
|
|
|
|
range_no= m_multi_cursor->get_range_no();
|
|
|
|
goto found;
|
2004-12-07 15:15:49 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
goto close_scan;
|
2004-12-07 15:15:49 +01:00
|
|
|
}
|
|
|
|
}
|
2004-12-08 00:36:40 +01:00
|
|
|
else if (m_multi_cursor && multi_range_sorted)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
2004-12-08 00:36:40 +01:00
|
|
|
if (m_active_cursor && (res= fetch_next(m_multi_cursor)))
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_MULTI_RANGE(3);
|
|
|
|
goto close_scan;
|
2004-12-08 00:36:40 +01:00
|
|
|
}
|
2004-11-29 13:50:13 +01:00
|
|
|
|
2004-12-07 15:15:49 +01:00
|
|
|
range_no= m_multi_cursor->get_range_no();
|
2004-12-28 17:01:07 +01:00
|
|
|
uint current_range_no= multi_range_curr - m_multi_ranges;
|
2005-01-11 14:06:44 +01:00
|
|
|
if ((uint) range_no == current_range_no)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_MULTI_RANGE(4);
|
2004-12-07 15:15:49 +01:00
|
|
|
// return current row
|
2005-02-16 14:18:32 +01:00
|
|
|
goto found;
|
2004-12-07 15:15:49 +01:00
|
|
|
}
|
2004-12-28 17:01:07 +01:00
|
|
|
else if (range_no > (int)current_range_no)
|
2004-12-07 15:15:49 +01:00
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_MULTI_RANGE(5);
|
|
|
|
// wait with current row
|
|
|
|
m_active_cursor= 0;
|
|
|
|
continue;
|
2004-12-07 15:15:49 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-02-16 14:18:32 +01:00
|
|
|
DBUG_MULTI_RANGE(6);
|
|
|
|
// First fetch from cursor
|
|
|
|
DBUG_ASSERT(range_no == -1);
|
2005-07-04 02:42:33 +02:00
|
|
|
if ((res= m_multi_cursor->nextResult(true)))
|
2005-02-16 14:18:32 +01:00
|
|
|
{
|
|
|
|
goto close_scan;
|
|
|
|
}
|
|
|
|
multi_range_curr--; // Will be increased in for-loop
|
|
|
|
continue;
|
2004-12-07 15:15:49 +01:00
|
|
|
}
|
2004-11-26 15:15:22 +01:00
|
|
|
}
|
2004-12-08 00:36:40 +01:00
|
|
|
else /** m_multi_cursor == 0 */
|
2004-11-26 15:15:22 +01:00
|
|
|
{
|
2004-12-08 00:36:40 +01:00
|
|
|
DBUG_MULTI_RANGE(7);
|
2004-11-30 07:58:55 +01:00
|
|
|
/**
|
|
|
|
* Corresponds to range 5 in example in read_multi_range_first
|
|
|
|
*/
|
|
|
|
(void)1;
|
2004-12-07 15:15:49 +01:00
|
|
|
continue;
|
2004-11-26 15:15:22 +01:00
|
|
|
}
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2005-05-06 10:39:30 +02:00
|
|
|
DBUG_ASSERT(FALSE); // Should only get here via goto's
|
2004-12-07 15:15:49 +01:00
|
|
|
close_scan:
|
|
|
|
if (res == 1)
|
|
|
|
{
|
2005-05-06 10:39:30 +02:00
|
|
|
m_multi_cursor->close(FALSE, TRUE);
|
2004-12-07 15:15:49 +01:00
|
|
|
m_active_cursor= m_multi_cursor= 0;
|
2004-12-08 00:36:40 +01:00
|
|
|
DBUG_MULTI_RANGE(8);
|
2004-12-07 15:15:49 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_RETURN(ndb_err(m_active_trans));
|
|
|
|
}
|
|
|
|
}
|
2004-11-26 15:15:22 +01:00
|
|
|
|
2004-12-28 17:01:07 +01:00
|
|
|
if (multi_range_curr == multi_range_end)
|
2007-01-31 22:38:06 +01:00
|
|
|
{
|
|
|
|
Thd_ndb *thd_ndb= get_thd_ndb(current_thd);
|
|
|
|
thd_ndb->query_state&= NDB_QUERY_NORMAL;
|
2004-11-17 10:07:52 +01:00
|
|
|
DBUG_RETURN(HA_ERR_END_OF_FILE);
|
2007-01-31 22:38:06 +01:00
|
|
|
}
|
2004-11-26 15:15:22 +01:00
|
|
|
|
2004-11-17 10:07:52 +01:00
|
|
|
/**
|
|
|
|
* Read remaining ranges
|
|
|
|
*/
|
|
|
|
DBUG_RETURN(read_multi_range_first(multi_range_found_p,
|
2005-02-16 14:18:32 +01:00
|
|
|
multi_range_curr,
|
|
|
|
multi_range_end - multi_range_curr,
|
|
|
|
multi_range_sorted,
|
|
|
|
multi_range_buffer));
|
2004-11-26 15:15:22 +01:00
|
|
|
|
|
|
|
found:
|
2004-11-30 07:58:55 +01:00
|
|
|
/**
|
|
|
|
* Found a record belonging to a scan
|
|
|
|
*/
|
2004-12-07 15:15:49 +01:00
|
|
|
m_active_cursor= m_multi_cursor;
|
2004-12-28 17:01:07 +01:00
|
|
|
* multi_range_found_p= m_multi_ranges + range_no;
|
2004-12-07 15:15:49 +01:00
|
|
|
memcpy(table->record[0], m_multi_range_cursor_result_ptr, reclength);
|
|
|
|
setup_recattr(m_active_cursor->getFirstRecAttr());
|
2004-11-26 15:15:22 +01:00
|
|
|
unpack_record(table->record[0]);
|
|
|
|
table->status= 0;
|
|
|
|
DBUG_RETURN(0);
|
2004-12-07 15:15:49 +01:00
|
|
|
|
2004-11-29 13:50:13 +01:00
|
|
|
found_next:
|
2004-11-30 07:58:55 +01:00
|
|
|
/**
|
|
|
|
* Found a record belonging to a pk/index op,
|
|
|
|
* copy result and move to next to prepare for next call
|
|
|
|
*/
|
2004-12-28 17:01:07 +01:00
|
|
|
* multi_range_found_p= multi_range_curr;
|
2004-11-29 13:50:13 +01:00
|
|
|
memcpy(table->record[0], m_multi_range_result_ptr, reclength);
|
2004-12-01 12:43:30 +01:00
|
|
|
setup_recattr(op->getFirstRecAttr());
|
2004-11-29 13:50:13 +01:00
|
|
|
unpack_record(table->record[0]);
|
2004-12-01 12:43:30 +01:00
|
|
|
table->status= 0;
|
|
|
|
|
2004-11-29 13:50:13 +01:00
|
|
|
multi_range_curr++;
|
2004-12-01 12:43:30 +01:00
|
|
|
m_current_multi_operation= m_active_trans->getNextCompletedOperation(op);
|
2004-11-29 13:50:13 +01:00
|
|
|
m_multi_range_result_ptr += reclength;
|
|
|
|
DBUG_RETURN(0);
|
2004-11-17 10:07:52 +01:00
|
|
|
}
|
|
|
|
|
2004-12-01 12:43:30 +01:00
|
|
|
int
|
|
|
|
ha_ndbcluster::setup_recattr(const NdbRecAttr* curr)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("setup_recattr");
|
|
|
|
|
|
|
|
Field **field, **end;
|
|
|
|
NdbValue *value= m_value;
|
|
|
|
|
2005-01-07 07:15:08 +01:00
|
|
|
end= table->field + table->s->fields;
|
2004-12-01 12:43:30 +01:00
|
|
|
|
|
|
|
for (field= table->field; field < end; field++, value++)
|
|
|
|
{
|
|
|
|
if ((* value).ptr)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(curr != 0);
|
2006-03-28 14:38:16 +02:00
|
|
|
NdbValue* val= m_value + curr->getColumn()->getColumnNo();
|
|
|
|
DBUG_ASSERT(val->ptr);
|
|
|
|
val->rec= curr;
|
2004-12-07 15:15:49 +01:00
|
|
|
curr= curr->next();
|
2004-12-01 12:43:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-08 00:36:40 +01:00
|
|
|
DBUG_RETURN(0);
|
2004-12-01 12:43:30 +01:00
|
|
|
}
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
char*
|
|
|
|
ha_ndbcluster::update_table_comment(
|
2005-02-16 14:18:32 +01:00
|
|
|
/* out: table comment + additional */
|
|
|
|
const char* comment)/* in: table comment defined by user */
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
uint length= strlen(comment);
|
2005-07-04 02:42:33 +02:00
|
|
|
if (length > 64000 - 3)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
return((char*)comment); /* string too long */
|
|
|
|
}
|
|
|
|
|
|
|
|
Ndb* ndb;
|
|
|
|
if (!(ndb= get_ndb()))
|
|
|
|
{
|
|
|
|
return((char*)comment);
|
|
|
|
}
|
|
|
|
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(m_dbname))
|
|
|
|
{
|
|
|
|
return((char*)comment);
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
NDBDICT* dict= ndb->getDictionary();
|
|
|
|
const NDBTAB* tab;
|
|
|
|
if (!(tab= dict->getTable(m_tabname)))
|
|
|
|
{
|
|
|
|
return((char*)comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *str;
|
|
|
|
const char *fmt="%s%snumber_of_replicas: %d";
|
|
|
|
const unsigned fmt_len_plus_extra= length + strlen(fmt);
|
|
|
|
if ((str= my_malloc(fmt_len_plus_extra, MYF(0))) == NULL)
|
|
|
|
{
|
2007-04-11 15:21:11 +02:00
|
|
|
sql_print_error("ha_ndbcluster::update_table_comment: "
|
|
|
|
"my_malloc(%u) failed", (unsigned int)fmt_len_plus_extra);
|
2005-02-11 22:33:52 +01:00
|
|
|
return (char*)comment;
|
|
|
|
}
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
my_snprintf(str,fmt_len_plus_extra,fmt,comment,
|
|
|
|
length > 0 ? " ":"",
|
|
|
|
tab->getReplicaCount());
|
2005-02-11 22:33:52 +01:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Utility thread main loop
|
2005-10-08 16:39:55 +02:00
|
|
|
pthread_handler_t ndb_util_thread_func(void *arg __attribute__((unused)))
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
THD *thd; /* needs to be first for thread_stack */
|
2005-03-15 15:03:25 +01:00
|
|
|
Ndb* ndb;
|
2005-02-11 22:33:52 +01:00
|
|
|
struct timespec abstime;
|
|
|
|
|
|
|
|
my_thread_init();
|
|
|
|
DBUG_ENTER("ndb_util_thread");
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("enter", ("ndb_cache_check_time: %lu", ndb_cache_check_time));
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
thd= new THD; /* note that contructor of THD uses DBUG_ */
|
2007-04-11 15:21:11 +02:00
|
|
|
if (thd == NULL)
|
|
|
|
{
|
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
|
|
|
DBUG_RETURN(NULL);
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
THD_CHECK_SENTRY(thd);
|
2005-03-15 15:03:25 +01:00
|
|
|
ndb= new Ndb(g_ndb_cluster_connection, "");
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb == NULL)
|
|
|
|
{
|
|
|
|
thd->cleanup();
|
|
|
|
delete thd;
|
|
|
|
DBUG_RETURN(NULL);
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
pthread_detach_this_thread();
|
|
|
|
ndb_util_thread= pthread_self();
|
|
|
|
|
|
|
|
thd->thread_stack= (char*)&thd; /* remember where our stack is */
|
2006-02-27 10:29:55 +01:00
|
|
|
if (thd->store_globals() || (ndb->init() != 0))
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
thd->cleanup();
|
|
|
|
delete thd;
|
2005-04-06 12:27:43 +02:00
|
|
|
delete ndb;
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_RETURN(NULL);
|
|
|
|
}
|
|
|
|
|
2007-04-02 18:21:05 +02:00
|
|
|
uint share_list_size= 0;
|
|
|
|
NDB_SHARE **share_list= NULL;
|
2005-02-21 14:04:54 +01:00
|
|
|
set_timespec(abstime, 0);
|
2005-02-11 22:33:52 +01:00
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
|
2007-02-21 18:38:40 +01:00
|
|
|
if (abort_loop)
|
|
|
|
break; /* Shutting down server */
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
pthread_mutex_lock(&LOCK_ndb_util_thread);
|
2005-10-11 23:58:22 +02:00
|
|
|
pthread_cond_timedwait(&COND_ndb_util_thread,
|
|
|
|
&LOCK_ndb_util_thread,
|
|
|
|
&abstime);
|
2005-02-11 22:33:52 +01:00
|
|
|
pthread_mutex_unlock(&LOCK_ndb_util_thread);
|
|
|
|
|
2006-11-20 21:42:06 +01:00
|
|
|
DBUG_PRINT("ndb_util_thread", ("Started, ndb_cache_check_time: %lu",
|
2005-02-11 22:33:52 +01:00
|
|
|
ndb_cache_check_time));
|
|
|
|
|
|
|
|
if (abort_loop)
|
|
|
|
break; /* Shutting down server */
|
|
|
|
|
|
|
|
if (ndb_cache_check_time == 0)
|
|
|
|
{
|
2005-03-15 15:03:25 +01:00
|
|
|
/* Wake up in 1 second to check if value has changed */
|
|
|
|
set_timespec(abstime, 1);
|
2005-02-11 22:33:52 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Lock mutex and fill list with pointers to all open tables */
|
|
|
|
NDB_SHARE *share;
|
|
|
|
pthread_mutex_lock(&ndbcluster_mutex);
|
2007-04-02 18:21:05 +02:00
|
|
|
uint i, record_count= ndbcluster_open_tables.records;
|
|
|
|
if (share_list_size < record_count)
|
|
|
|
{
|
|
|
|
NDB_SHARE ** new_share_list= new NDB_SHARE * [record_count];
|
|
|
|
if (!new_share_list)
|
|
|
|
{
|
|
|
|
sql_print_warning("ndb util thread: malloc failure, "
|
|
|
|
"query cache not maintained properly");
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
goto next; // At least do not crash
|
|
|
|
}
|
|
|
|
delete [] share_list;
|
|
|
|
share_list_size= record_count;
|
|
|
|
share_list= new_share_list;
|
|
|
|
}
|
|
|
|
for (i= 0; i < record_count; i++)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
|
|
|
share= (NDB_SHARE *)hash_element(&ndbcluster_open_tables, i);
|
|
|
|
share->use_count++; /* Make sure the table can't be closed */
|
|
|
|
DBUG_PRINT("ndb_util_thread",
|
|
|
|
("Found open table[%d]: %s, use_count: %d",
|
|
|
|
i, share->table_name, share->use_count));
|
|
|
|
|
|
|
|
/* Store pointer to table */
|
2007-04-02 18:21:05 +02:00
|
|
|
share_list[i]= share;
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&ndbcluster_mutex);
|
|
|
|
|
2007-04-02 18:21:05 +02:00
|
|
|
/* Iterate through the open files list */
|
|
|
|
for (i= 0; i < record_count; i++)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
2007-04-02 18:21:05 +02:00
|
|
|
share= share_list[i];
|
2005-02-11 22:33:52 +01:00
|
|
|
/* Split tab- and dbname */
|
|
|
|
char buf[FN_REFLEN];
|
|
|
|
char *tabname, *db;
|
|
|
|
uint length= dirname_length(share->table_name);
|
|
|
|
tabname= share->table_name+length;
|
|
|
|
memcpy(buf, share->table_name, length-1);
|
|
|
|
buf[length-1]= 0;
|
|
|
|
db= buf+dirname_length(buf);
|
|
|
|
DBUG_PRINT("ndb_util_thread",
|
2005-03-15 15:03:25 +01:00
|
|
|
("Fetching commit count for: %s",
|
|
|
|
share->table_name));
|
2005-02-11 22:33:52 +01:00
|
|
|
|
|
|
|
/* Contact NDB to get commit count for table */
|
2005-03-15 15:03:25 +01:00
|
|
|
struct Ndb_statistics stat;
|
|
|
|
uint lock;
|
|
|
|
pthread_mutex_lock(&share->mutex);
|
|
|
|
lock= share->commit_count_lock;
|
|
|
|
pthread_mutex_unlock(&share->mutex);
|
2007-04-11 15:21:11 +02:00
|
|
|
if (ndb->setDatabaseName(db))
|
|
|
|
{
|
|
|
|
goto loop_next;
|
|
|
|
}
|
2006-10-23 16:18:57 +02:00
|
|
|
if (ndb_get_table_statistics(NULL, false, ndb, tabname, &stat) == 0)
|
2005-02-11 22:33:52 +01:00
|
|
|
{
|
2007-02-28 21:23:35 +01:00
|
|
|
#ifndef DBUG_OFF
|
2006-06-30 18:15:18 +02:00
|
|
|
char buff[22], buff2[22];
|
2007-02-28 21:23:35 +01:00
|
|
|
#endif
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_PRINT("ndb_util_thread",
|
2006-06-30 18:15:18 +02:00
|
|
|
("Table: %s commit_count: %s rows: %s",
|
|
|
|
share->table_name,
|
|
|
|
llstr(stat.commit_count, buff),
|
2006-06-30 19:07:33 +02:00
|
|
|
llstr(stat.row_count, buff2)));
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DBUG_PRINT("ndb_util_thread",
|
|
|
|
("Error: Could not get commit count for table %s",
|
|
|
|
share->table_name));
|
2005-03-15 15:03:25 +01:00
|
|
|
stat.commit_count= 0;
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
2007-04-11 15:21:11 +02:00
|
|
|
loop_next:
|
2005-03-15 15:03:25 +01:00
|
|
|
pthread_mutex_lock(&share->mutex);
|
|
|
|
if (share->commit_count_lock == lock)
|
|
|
|
share->commit_count= stat.commit_count;
|
|
|
|
pthread_mutex_unlock(&share->mutex);
|
|
|
|
|
2005-02-11 22:33:52 +01:00
|
|
|
/* Decrease the use count and possibly free share */
|
|
|
|
free_share(share);
|
|
|
|
}
|
2007-04-02 18:21:05 +02:00
|
|
|
next:
|
2005-03-15 15:03:25 +01:00
|
|
|
/* Calculate new time to wake up */
|
|
|
|
int secs= 0;
|
|
|
|
int msecs= ndb_cache_check_time;
|
|
|
|
|
|
|
|
struct timeval tick_time;
|
|
|
|
gettimeofday(&tick_time, 0);
|
|
|
|
abstime.tv_sec= tick_time.tv_sec;
|
|
|
|
abstime.tv_nsec= tick_time.tv_usec * 1000;
|
|
|
|
|
2005-07-04 02:42:33 +02:00
|
|
|
if (msecs >= 1000){
|
2005-03-15 15:03:25 +01:00
|
|
|
secs= msecs / 1000;
|
|
|
|
msecs= msecs % 1000;
|
|
|
|
}
|
|
|
|
|
|
|
|
abstime.tv_sec+= secs;
|
|
|
|
abstime.tv_nsec+= msecs * 1000000;
|
|
|
|
if (abstime.tv_nsec >= 1000000000) {
|
|
|
|
abstime.tv_sec+= 1;
|
|
|
|
abstime.tv_nsec-= 1000000000;
|
|
|
|
}
|
2005-02-11 22:33:52 +01:00
|
|
|
}
|
|
|
|
|
2007-04-02 18:21:05 +02:00
|
|
|
if (share_list)
|
|
|
|
delete [] share_list;
|
2005-02-11 22:33:52 +01:00
|
|
|
thd->cleanup();
|
|
|
|
delete thd;
|
2005-04-06 12:27:43 +02:00
|
|
|
delete ndb;
|
2005-02-11 22:33:52 +01:00
|
|
|
DBUG_PRINT("exit", ("ndb_util_thread"));
|
|
|
|
my_thread_end();
|
|
|
|
pthread_exit(0);
|
|
|
|
DBUG_RETURN(NULL);
|
|
|
|
}
|
|
|
|
|
2005-09-19 15:35:07 +02:00
|
|
|
int
|
|
|
|
ndbcluster_show_status(THD* thd)
|
|
|
|
{
|
|
|
|
Protocol *protocol= thd->protocol;
|
|
|
|
DBUG_ENTER("ndbcluster_show_status");
|
|
|
|
|
|
|
|
if (have_ndbcluster != SHOW_OPTION_YES)
|
|
|
|
{
|
|
|
|
my_message(ER_NOT_SUPPORTED_YET,
|
2006-01-05 23:47:49 +01:00
|
|
|
"Cannot call SHOW NDBCLUSTER STATUS because skip-ndbcluster is "
|
|
|
|
"defined",
|
2005-09-19 15:35:07 +02:00
|
|
|
MYF(0));
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
List<Item> field_list;
|
|
|
|
field_list.push_back(new Item_empty_string("free_list", 255));
|
|
|
|
field_list.push_back(new Item_return_int("created", 10,MYSQL_TYPE_LONG));
|
|
|
|
field_list.push_back(new Item_return_int("free", 10,MYSQL_TYPE_LONG));
|
|
|
|
field_list.push_back(new Item_return_int("sizeof", 10,MYSQL_TYPE_LONG));
|
|
|
|
|
2006-01-05 23:47:49 +01:00
|
|
|
if (protocol->send_fields(&field_list,
|
|
|
|
Protocol::SEND_NUM_ROWS | Protocol::SEND_EOF))
|
2005-09-19 15:35:07 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
2005-09-20 12:07:18 +02:00
|
|
|
if (get_thd_ndb(thd) && get_thd_ndb(thd)->ndb)
|
2005-09-19 15:35:07 +02:00
|
|
|
{
|
2005-09-20 12:07:18 +02:00
|
|
|
Ndb* ndb= (get_thd_ndb(thd))->ndb;
|
2006-01-05 23:47:49 +01:00
|
|
|
Ndb::Free_list_usage tmp;
|
|
|
|
tmp.m_name= 0;
|
2005-09-19 15:35:07 +02:00
|
|
|
while (ndb->get_free_list_usage(&tmp))
|
|
|
|
{
|
|
|
|
protocol->prepare_for_resend();
|
|
|
|
|
|
|
|
protocol->store(tmp.m_name, &my_charset_bin);
|
|
|
|
protocol->store((uint)tmp.m_created);
|
|
|
|
protocol->store((uint)tmp.m_free);
|
|
|
|
protocol->store((uint)tmp.m_sizeof);
|
|
|
|
if (protocol->write())
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
send_eof(thd);
|
|
|
|
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
2007-04-23 11:25:33 +02:00
|
|
|
/*
|
|
|
|
Condition pushdown
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
Push a condition to ndbcluster storage engine for evaluation
|
|
|
|
during table and index scans. The conditions will be stored on a stack
|
|
|
|
for possibly storing several conditions. The stack can be popped
|
|
|
|
by calling cond_pop, handler::extra(HA_EXTRA_RESET) (handler::reset())
|
|
|
|
will clear the stack.
|
|
|
|
The current implementation supports arbitrary AND/OR nested conditions
|
|
|
|
with comparisons between columns and constants (including constant
|
|
|
|
expressions and function calls) and the following comparison operators:
|
|
|
|
=, !=, >, >=, <, <=, "is null", and "is not null".
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
NULL The condition was supported and will be evaluated for each
|
|
|
|
row found during the scan
|
|
|
|
cond The condition was not supported and all rows will be returned from
|
|
|
|
the scan for evaluation (and thus not saved on stack)
|
|
|
|
*/
|
|
|
|
const
|
|
|
|
COND*
|
|
|
|
ha_ndbcluster::cond_push(const COND *cond)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("cond_push");
|
|
|
|
if (!m_cond)
|
|
|
|
m_cond= new ha_ndbcluster_cond;
|
|
|
|
if (!m_cond)
|
|
|
|
{
|
|
|
|
my_errno= HA_ERR_OUT_OF_MEM;
|
|
|
|
DBUG_RETURN(NULL);
|
|
|
|
}
|
|
|
|
DBUG_EXECUTE("where",print_where((COND *)cond, m_tabname););
|
|
|
|
DBUG_RETURN(m_cond->cond_push(cond, table, (NDBTAB *)m_table));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Pop the top condition from the condition stack of the handler instance.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ha_ndbcluster::cond_pop()
|
|
|
|
{
|
|
|
|
if (m_cond)
|
|
|
|
m_cond->cond_pop();
|
|
|
|
}
|
|
|
|
|
2004-04-15 09:14:14 +02:00
|
|
|
#endif /* HAVE_NDBCLUSTER_DB */
|