mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 14:54:20 +01:00
d1bf0b7dab
row_remove_indexes_for_mysql(): Rename to row_merge_drop_indexes(), move from row0mysql.c to row0merge.c and make the return type void. row_merge_remove_index(): Rename to row_merge_drop_index() and make the return type void.
9170 lines
257 KiB
C++
9170 lines
257 KiB
C++
/* Copyright (C) 2000-2005 MySQL AB & Innobase Oy
|
|
|
|
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
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
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
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
/* This file defines the InnoDB handler: the interface between MySQL and InnoDB
|
|
NOTE: You can only use noninlined InnoDB functions in this file, because we
|
|
have disabled the InnoDB inlining in this file. */
|
|
|
|
/* TODO list for the InnoDB handler in 5.0:
|
|
- Remove the flag trx->active_trans and look at trx->conc_state
|
|
- fix savepoint functions to use savepoint storage area
|
|
- Find out what kind of problems the OS X case-insensitivity causes to
|
|
table and database names; should we 'normalize' the names like we do
|
|
in Windows?
|
|
*/
|
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
|
#pragma implementation // gcc: Class implementation
|
|
#endif
|
|
|
|
#include <mysql_priv.h>
|
|
#include <mysqld_error.h>
|
|
|
|
#include <m_ctype.h>
|
|
#include <hash.h>
|
|
#include <myisampack.h>
|
|
#include <mysys_err.h>
|
|
#include <my_sys.h>
|
|
#include "ha_innodb.h"
|
|
#include <mysql/plugin.h>
|
|
|
|
#ifdef MYSQL_SERVER
|
|
/* Define some macros until these functions are declared in <mysql/plugin.h>.
|
|
Once these functions are defined by MySQL, we may consider
|
|
removing -DMYSQL_SERVER from ../Makefile.am as well. */
|
|
#define thd_charset(thd) (thd)->charset()
|
|
#define thd_get_xid(thd,xid_) ((*xid_) = (thd)->transaction.xid_state.xid)
|
|
#define thd_memdup(thd,str,len) (thd)->memdup(str, len)
|
|
#define thd_killed(thd) (thd)->killed
|
|
#define thd_slave_thread(thd) (thd)->slave_thread
|
|
#define thd_query(thd) (&(thd)->query)
|
|
#define thd_non_transactional_update(thd) ((thd)->no_trans_update.all)
|
|
#define mysql_bin_log_file_name() mysql_bin_log.get_log_fname()
|
|
#define mysql_bin_log_file_pos() mysql_bin_log.get_log_file()->pos_in_file
|
|
#define mysql_tmpfile() fileno(tmpfile())/* BUGGY: leaks memory, Bug #3998 */
|
|
#define mysql_query_cache_invalidate4(a,b,c,d) query_cache.invalidate(a,b,c,d)
|
|
#else /* MYSQL_SERVER */
|
|
/* This is needed because of Bug #3596. Let us hope that pthread_mutex_t
|
|
is defined the same in both builds: the MySQL server and the InnoDB plugin. */
|
|
extern pthread_mutex_t LOCK_thread_count;
|
|
#endif /* MYSQL_SERVER */
|
|
|
|
/** to protect innobase_open_files */
|
|
static pthread_mutex_t innobase_share_mutex;
|
|
/** to force correct commit order in binlog */
|
|
static pthread_mutex_t prepare_commit_mutex;
|
|
static ulong commit_threads = 0;
|
|
static pthread_mutex_t commit_threads_m;
|
|
static pthread_cond_t commit_cond;
|
|
static pthread_mutex_t commit_cond_m;
|
|
static bool innodb_inited = 0;
|
|
|
|
/*
|
|
This needs to exist until the query cache callback is removed
|
|
or learns to pass hton.
|
|
*/
|
|
static handlerton *innodb_hton_ptr;
|
|
|
|
/* Store MySQL definition of 'byte': in Linux it is char while InnoDB
|
|
uses unsigned char; the header univ.i which we include next defines
|
|
'byte' as a macro which expands to 'unsigned char' */
|
|
|
|
typedef byte mysql_byte;
|
|
|
|
#define INSIDE_HA_INNOBASE_CC
|
|
|
|
/* Include necessary InnoDB headers */
|
|
extern "C" {
|
|
#include "univ.i"
|
|
#include "buf0buddy.h"
|
|
#include "os0file.h"
|
|
#include "os0thread.h"
|
|
#include "srv0start.h"
|
|
#include "srv0srv.h"
|
|
#include "trx0roll.h"
|
|
#include "trx0trx.h"
|
|
#include "trx0sys.h"
|
|
#include "mtr0mtr.h"
|
|
#include "row0ins.h"
|
|
#include "row0mysql.h"
|
|
#include "row0sel.h"
|
|
#include "row0upd.h"
|
|
#include "log0log.h"
|
|
#include "lock0lock.h"
|
|
#include "dict0crea.h"
|
|
#include "btr0cur.h"
|
|
#include "btr0btr.h"
|
|
#include "fsp0fsp.h"
|
|
#include "sync0sync.h"
|
|
#include "fil0fil.h"
|
|
#include "trx0xa.h"
|
|
#include "row0merge.h"
|
|
#include "thr0loc.h"
|
|
#include "dict0boot.h"
|
|
#include "ha_prototypes.h"
|
|
}
|
|
|
|
static long innobase_mirrored_log_groups, innobase_log_files_in_group,
|
|
innobase_log_buffer_size,
|
|
innobase_additional_mem_pool_size, innobase_file_io_threads,
|
|
innobase_lock_wait_timeout, innobase_force_recovery,
|
|
innobase_open_files;
|
|
|
|
static long long innobase_buffer_pool_size, innobase_log_file_size;
|
|
|
|
/* The default values for the following char* start-up parameters
|
|
are determined in innobase_init below: */
|
|
|
|
static char* innobase_data_home_dir = NULL;
|
|
static char* innobase_data_file_path = NULL;
|
|
static char* innobase_log_group_home_dir = NULL;
|
|
/* The following has a misleading name: starting from 4.0.5, this also
|
|
affects Windows: */
|
|
static char* innobase_unix_file_flush_method = NULL;
|
|
|
|
/* Below we have boolean-valued start-up parameters, and their default
|
|
values */
|
|
|
|
static ulong innobase_fast_shutdown = 1;
|
|
#ifdef UNIV_LOG_ARCHIVE
|
|
static my_bool innobase_log_archive = FALSE;
|
|
static char* innobase_log_arch_dir = NULL;
|
|
#endif /* UNIV_LOG_ARCHIVE */
|
|
static my_bool innobase_use_doublewrite = TRUE;
|
|
static my_bool innobase_use_checksums = TRUE;
|
|
static my_bool innobase_file_per_table = FALSE;
|
|
static my_bool innobase_locks_unsafe_for_binlog = FALSE;
|
|
static my_bool innobase_rollback_on_timeout = FALSE;
|
|
static my_bool innobase_create_status_file = FALSE;
|
|
static my_bool innobase_stats_on_metadata = TRUE;
|
|
|
|
static char* internal_innobase_data_file_path = NULL;
|
|
|
|
/* The following counter is used to convey information to InnoDB
|
|
about server activity: in selects it is not sensible to call
|
|
srv_active_wake_master_thread after each fetch or search, we only do
|
|
it every INNOBASE_WAKE_INTERVAL'th step. */
|
|
|
|
#define INNOBASE_WAKE_INTERVAL 32
|
|
static ulong innobase_active_counter = 0;
|
|
|
|
static HASH innobase_open_tables;
|
|
|
|
#ifdef __NETWARE__ /* some special cleanup for NetWare */
|
|
bool nw_panic = FALSE;
|
|
#endif
|
|
|
|
static mysql_byte* innobase_get_key(INNOBASE_SHARE *share,uint *length,
|
|
my_bool not_used __attribute__((unused)));
|
|
static INNOBASE_SHARE *get_share(const char *table_name);
|
|
static void free_share(INNOBASE_SHARE *share);
|
|
static int innobase_close_connection(handlerton *hton, THD* thd);
|
|
static int innobase_commit(handlerton *hton, THD* thd, bool all);
|
|
static int innobase_rollback(handlerton *hton, THD* thd, bool all);
|
|
static int innobase_rollback_to_savepoint(handlerton *hton, THD* thd,
|
|
void *savepoint);
|
|
static int innobase_savepoint(handlerton *hton, THD* thd, void *savepoint);
|
|
static int innobase_release_savepoint(handlerton *hton, THD* thd,
|
|
void *savepoint);
|
|
static handler *innobase_create_handler(handlerton *hton,
|
|
TABLE_SHARE *table,
|
|
MEM_ROOT *mem_root);
|
|
|
|
/********************************************************************
|
|
Return alter table flags supported in an InnoDB database. */
|
|
static
|
|
uint
|
|
innobase_alter_table_flags(
|
|
/*=======================*/
|
|
uint flags);
|
|
|
|
static const char innobase_hton_name[]= "InnoDB";
|
|
|
|
|
|
static MYSQL_THDVAR_BOOL(support_xa, PLUGIN_VAR_OPCMDARG,
|
|
"Enable InnoDB support for the XA two-phase commit",
|
|
/* check_func */ NULL, /* update_func */ NULL,
|
|
/* default */ TRUE);
|
|
|
|
static MYSQL_THDVAR_BOOL(table_locks, PLUGIN_VAR_OPCMDARG,
|
|
"Enable InnoDB locking in LOCK TABLES",
|
|
/* check_func */ NULL, /* update_func */ NULL,
|
|
/* default */ TRUE);
|
|
|
|
static handler *innobase_create_handler(handlerton *hton,
|
|
TABLE_SHARE *table,
|
|
MEM_ROOT *mem_root)
|
|
{
|
|
return new (mem_root) ha_innobase(hton, table);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function is used to prepare X/Open XA distributed transaction */
|
|
static
|
|
int
|
|
innobase_xa_prepare(
|
|
/*================*/
|
|
/* out: 0 or error number */
|
|
handlerton* hton,
|
|
THD* thd, /* in: handle to the MySQL thread of the user
|
|
whose XA transaction should be prepared */
|
|
bool all); /* in: TRUE - commit transaction
|
|
FALSE - the current SQL statement ended */
|
|
/***********************************************************************
|
|
This function is used to recover X/Open XA distributed transactions */
|
|
static
|
|
int
|
|
innobase_xa_recover(
|
|
/*================*/
|
|
/* out: number of prepared transactions
|
|
stored in xid_list */
|
|
handlerton* hton,
|
|
XID* xid_list, /* in/out: prepared transactions */
|
|
uint len); /* in: number of slots in xid_list */
|
|
/***********************************************************************
|
|
This function is used to commit one X/Open XA distributed transaction
|
|
which is in the prepared state */
|
|
static
|
|
int
|
|
innobase_commit_by_xid(
|
|
/*===================*/
|
|
/* out: 0 or error number */
|
|
handlerton* hton,
|
|
XID* xid); /* in: X/Open XA transaction identification */
|
|
/***********************************************************************
|
|
This function is used to rollback one X/Open XA distributed transaction
|
|
which is in the prepared state */
|
|
static
|
|
int
|
|
innobase_rollback_by_xid(
|
|
/*=====================*/
|
|
/* out: 0 or error number */
|
|
handlerton* hton,
|
|
XID *xid); /* in: X/Open XA transaction identification */
|
|
/***********************************************************************
|
|
Create a consistent view for a cursor based on current transaction
|
|
which is created if the corresponding MySQL thread still lacks one.
|
|
This consistent view is then used inside of MySQL when accessing records
|
|
using a cursor. */
|
|
static
|
|
void*
|
|
innobase_create_cursor_view(
|
|
/*========================*/
|
|
/* out: pointer to cursor view or NULL */
|
|
handlerton* hton, /* in: innobase hton */
|
|
THD* thd); /* in: user thread handle */
|
|
/***********************************************************************
|
|
Set the given consistent cursor view to a transaction which is created
|
|
if the corresponding MySQL thread still lacks one. If the given
|
|
consistent cursor view is NULL global read view of a transaction is
|
|
restored to a transaction read view. */
|
|
static
|
|
void
|
|
innobase_set_cursor_view(
|
|
/*=====================*/
|
|
handlerton* hton,
|
|
THD* thd, /* in: user thread handle */
|
|
void* curview);/* in: Consistent cursor view to be set */
|
|
/***********************************************************************
|
|
Close the given consistent cursor view of a transaction and restore
|
|
global read view to a transaction read view. Transaction is created if the
|
|
corresponding MySQL thread still lacks one. */
|
|
static
|
|
void
|
|
innobase_close_cursor_view(
|
|
/*=======================*/
|
|
handlerton* hton,
|
|
THD* thd, /* in: user thread handle */
|
|
void* curview);/* in: Consistent read view to be closed */
|
|
/*********************************************************************
|
|
Removes all tables in the named database inside InnoDB. */
|
|
static
|
|
void
|
|
innobase_drop_database(
|
|
/*===================*/
|
|
/* out: error number */
|
|
handlerton* hton, /* in: handlerton of Innodb */
|
|
char* path); /* in: database path; inside InnoDB the name
|
|
of the last directory in the path is used as
|
|
the database name: for example, in 'mysql/data/test'
|
|
the database name is 'test' */
|
|
/***********************************************************************
|
|
Closes an InnoDB database. */
|
|
static
|
|
int
|
|
innobase_end(handlerton *hton, ha_panic_function type);
|
|
|
|
/*********************************************************************
|
|
Creates an InnoDB transaction struct for the thd if it does not yet have one.
|
|
Starts a new InnoDB transaction if a transaction is not yet started. And
|
|
assigns a new snapshot for a consistent read if the transaction does not yet
|
|
have one. */
|
|
static
|
|
int
|
|
innobase_start_trx_and_assign_read_view(
|
|
/*====================================*/
|
|
/* out: 0 */
|
|
handlerton* hton, /* in: Innodb handlerton */
|
|
THD* thd); /* in: MySQL thread handle of the user for whom
|
|
the transaction should be committed */
|
|
/********************************************************************
|
|
Flushes InnoDB logs to disk and makes a checkpoint. Really, a commit flushes
|
|
the logs, and the name of this function should be innobase_checkpoint. */
|
|
static
|
|
bool
|
|
innobase_flush_logs(
|
|
/*================*/
|
|
/* out: TRUE if error */
|
|
handlerton* hton); /* in: InnoDB handlerton */
|
|
|
|
/****************************************************************************
|
|
Implements the SHOW INNODB STATUS command. Sends the output of the InnoDB
|
|
Monitor to the client. */
|
|
static
|
|
bool
|
|
innodb_show_status(
|
|
/*===============*/
|
|
handlerton* hton, /* in: the innodb handlerton */
|
|
THD* thd, /* in: the MySQL query thread of the caller */
|
|
stat_print_fn *stat_print);
|
|
static
|
|
bool innobase_show_status(handlerton *hton, THD* thd,
|
|
stat_print_fn* stat_print,
|
|
enum ha_stat_type stat_type);
|
|
|
|
/*********************************************************************
|
|
Commits a transaction in an InnoDB database. */
|
|
static
|
|
void
|
|
innobase_commit_low(
|
|
/*================*/
|
|
trx_t* trx); /* in: transaction handle */
|
|
|
|
static SHOW_VAR innodb_status_variables[]= {
|
|
{"buffer_pool_pages_data",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_data, SHOW_LONG},
|
|
{"buffer_pool_pages_dirty",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_dirty, SHOW_LONG},
|
|
{"buffer_pool_pages_flushed",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_flushed, SHOW_LONG},
|
|
{"buffer_pool_pages_free",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_free, SHOW_LONG},
|
|
{"buffer_pool_pages_latched",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_latched, SHOW_LONG},
|
|
{"buffer_pool_pages_misc",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_misc, SHOW_LONG},
|
|
{"buffer_pool_pages_total",
|
|
(char*) &export_vars.innodb_buffer_pool_pages_total, SHOW_LONG},
|
|
{"buffer_pool_read_ahead_rnd",
|
|
(char*) &export_vars.innodb_buffer_pool_read_ahead_rnd, SHOW_LONG},
|
|
{"buffer_pool_read_ahead_seq",
|
|
(char*) &export_vars.innodb_buffer_pool_read_ahead_seq, SHOW_LONG},
|
|
{"buffer_pool_read_requests",
|
|
(char*) &export_vars.innodb_buffer_pool_read_requests, SHOW_LONG},
|
|
{"buffer_pool_reads",
|
|
(char*) &export_vars.innodb_buffer_pool_reads, SHOW_LONG},
|
|
{"buffer_pool_wait_free",
|
|
(char*) &export_vars.innodb_buffer_pool_wait_free, SHOW_LONG},
|
|
{"buffer_pool_write_requests",
|
|
(char*) &export_vars.innodb_buffer_pool_write_requests, SHOW_LONG},
|
|
{"data_fsyncs",
|
|
(char*) &export_vars.innodb_data_fsyncs, SHOW_LONG},
|
|
{"data_pending_fsyncs",
|
|
(char*) &export_vars.innodb_data_pending_fsyncs, SHOW_LONG},
|
|
{"data_pending_reads",
|
|
(char*) &export_vars.innodb_data_pending_reads, SHOW_LONG},
|
|
{"data_pending_writes",
|
|
(char*) &export_vars.innodb_data_pending_writes, SHOW_LONG},
|
|
{"data_read",
|
|
(char*) &export_vars.innodb_data_read, SHOW_LONG},
|
|
{"data_reads",
|
|
(char*) &export_vars.innodb_data_reads, SHOW_LONG},
|
|
{"data_writes",
|
|
(char*) &export_vars.innodb_data_writes, SHOW_LONG},
|
|
{"data_written",
|
|
(char*) &export_vars.innodb_data_written, SHOW_LONG},
|
|
{"dblwr_pages_written",
|
|
(char*) &export_vars.innodb_dblwr_pages_written, SHOW_LONG},
|
|
{"dblwr_writes",
|
|
(char*) &export_vars.innodb_dblwr_writes, SHOW_LONG},
|
|
{"log_waits",
|
|
(char*) &export_vars.innodb_log_waits, SHOW_LONG},
|
|
{"log_write_requests",
|
|
(char*) &export_vars.innodb_log_write_requests, SHOW_LONG},
|
|
{"log_writes",
|
|
(char*) &export_vars.innodb_log_writes, SHOW_LONG},
|
|
{"os_log_fsyncs",
|
|
(char*) &export_vars.innodb_os_log_fsyncs, SHOW_LONG},
|
|
{"os_log_pending_fsyncs",
|
|
(char*) &export_vars.innodb_os_log_pending_fsyncs, SHOW_LONG},
|
|
{"os_log_pending_writes",
|
|
(char*) &export_vars.innodb_os_log_pending_writes, SHOW_LONG},
|
|
{"os_log_written",
|
|
(char*) &export_vars.innodb_os_log_written, SHOW_LONG},
|
|
{"page_size",
|
|
(char*) &export_vars.innodb_page_size, SHOW_LONG},
|
|
{"pages_created",
|
|
(char*) &export_vars.innodb_pages_created, SHOW_LONG},
|
|
{"pages_read",
|
|
(char*) &export_vars.innodb_pages_read, SHOW_LONG},
|
|
{"pages_written",
|
|
(char*) &export_vars.innodb_pages_written, SHOW_LONG},
|
|
{"row_lock_current_waits",
|
|
(char*) &export_vars.innodb_row_lock_current_waits, SHOW_LONG},
|
|
{"row_lock_time",
|
|
(char*) &export_vars.innodb_row_lock_time, SHOW_LONGLONG},
|
|
{"row_lock_time_avg",
|
|
(char*) &export_vars.innodb_row_lock_time_avg, SHOW_LONG},
|
|
{"row_lock_time_max",
|
|
(char*) &export_vars.innodb_row_lock_time_max, SHOW_LONG},
|
|
{"row_lock_waits",
|
|
(char*) &export_vars.innodb_row_lock_waits, SHOW_LONG},
|
|
{"rows_deleted",
|
|
(char*) &export_vars.innodb_rows_deleted, SHOW_LONG},
|
|
{"rows_inserted",
|
|
(char*) &export_vars.innodb_rows_inserted, SHOW_LONG},
|
|
{"rows_read",
|
|
(char*) &export_vars.innodb_rows_read, SHOW_LONG},
|
|
{"rows_updated",
|
|
(char*) &export_vars.innodb_rows_updated, SHOW_LONG},
|
|
{NullS, NullS, SHOW_LONG}
|
|
};
|
|
|
|
/* General functions */
|
|
|
|
/**********************************************************************
|
|
Returns true if the thread is the replication thread on the slave
|
|
server. Used in srv_conc_enter_innodb() to determine if the thread
|
|
should be allowed to enter InnoDB - the replication thread is treated
|
|
differently than other threads. Also used in
|
|
srv_conc_force_exit_innodb(). */
|
|
extern "C"
|
|
ibool
|
|
thd_is_replication_slave_thread(
|
|
/*============================*/
|
|
/* out: true if thd is the replication thread */
|
|
void* thd) /* in: thread handle (THD*) */
|
|
{
|
|
return((ibool) thd_slave_thread((THD*) thd));
|
|
}
|
|
|
|
/**********************************************************************
|
|
Save some CPU by testing the value of srv_thread_concurrency in inline
|
|
functions. */
|
|
inline
|
|
void
|
|
innodb_srv_conc_enter_innodb(
|
|
/*=========================*/
|
|
trx_t* trx) /* in: transaction handle */
|
|
{
|
|
if (UNIV_LIKELY(!srv_thread_concurrency)) {
|
|
|
|
return;
|
|
}
|
|
|
|
srv_conc_enter_innodb(trx);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Save some CPU by testing the value of srv_thread_concurrency in inline
|
|
functions. */
|
|
inline
|
|
void
|
|
innodb_srv_conc_exit_innodb(
|
|
/*========================*/
|
|
trx_t* trx) /* in: transaction handle */
|
|
{
|
|
if (UNIV_LIKELY(!srv_thread_concurrency)) {
|
|
|
|
return;
|
|
}
|
|
|
|
srv_conc_exit_innodb(trx);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Releases possible search latch and InnoDB thread FIFO ticket. These should
|
|
be released at each SQL statement end, and also when mysqld passes the
|
|
control to the client. It does no harm to release these also in the middle
|
|
of an SQL statement. */
|
|
inline
|
|
void
|
|
innobase_release_stat_resources(
|
|
/*============================*/
|
|
trx_t* trx) /* in: transaction object */
|
|
{
|
|
if (trx->has_search_latch) {
|
|
trx_search_latch_release_if_reserved(trx);
|
|
}
|
|
|
|
if (trx->declared_to_be_inside_innodb) {
|
|
/* Release our possible ticket in the FIFO */
|
|
|
|
srv_conc_force_exit_innodb(trx);
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
Returns true if the transaction this thread is processing has edited
|
|
non-transactional tables. Used by the deadlock detector when deciding
|
|
which transaction to rollback in case of a deadlock - we try to avoid
|
|
rolling back transactions that have edited non-transactional tables. */
|
|
extern "C"
|
|
ibool
|
|
thd_has_edited_nontrans_tables(
|
|
/*===========================*/
|
|
/* out: true if non-transactional tables have
|
|
been edited */
|
|
void* thd) /* in: thread handle (THD*) */
|
|
{
|
|
return((ibool) thd_non_transactional_update((THD*) thd));
|
|
}
|
|
|
|
/************************************************************************
|
|
Obtain the InnoDB transaction of a MySQL thread. */
|
|
inline
|
|
trx_t*&
|
|
thd_to_trx(
|
|
/*=======*/
|
|
/* out: reference to transaction pointer */
|
|
THD* thd) /* in: MySQL thread */
|
|
{
|
|
return(*(trx_t**) thd_ha_data(thd, innodb_hton_ptr));
|
|
}
|
|
|
|
/************************************************************************
|
|
Call this function when mysqld passes control to the client. That is to
|
|
avoid deadlocks on the adaptive hash S-latch possibly held by thd. For more
|
|
documentation, see handler.cc. */
|
|
static
|
|
int
|
|
innobase_release_temporary_latches(
|
|
/*===============================*/
|
|
/* out: 0 */
|
|
handlerton* hton, /* in: handlerton */
|
|
THD* thd) /* in: MySQL thread */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ASSERT(hton == innodb_hton_ptr);
|
|
|
|
if (!innodb_inited) {
|
|
|
|
return(0);
|
|
}
|
|
|
|
trx = thd_to_trx(thd);
|
|
|
|
if (trx) {
|
|
innobase_release_stat_resources(trx);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/************************************************************************
|
|
Increments innobase_active_counter and every INNOBASE_WAKE_INTERVALth
|
|
time calls srv_active_wake_master_thread. This function should be used
|
|
when a single database operation may introduce a small need for
|
|
server utility activity, like checkpointing. */
|
|
inline
|
|
void
|
|
innobase_active_small(void)
|
|
/*=======================*/
|
|
{
|
|
innobase_active_counter++;
|
|
|
|
if ((innobase_active_counter % INNOBASE_WAKE_INTERVAL) == 0) {
|
|
srv_active_wake_master_thread();
|
|
}
|
|
}
|
|
|
|
/************************************************************************
|
|
Converts an InnoDB error code to a MySQL error code and also tells to MySQL
|
|
about a possible transaction rollback inside InnoDB caused by a lock wait
|
|
timeout or a deadlock. */
|
|
static
|
|
int
|
|
convert_error_code_to_mysql(
|
|
/*========================*/
|
|
/* out: MySQL error code */
|
|
int error, /* in: InnoDB error code */
|
|
THD* thd) /* in: user thread handle or NULL */
|
|
{
|
|
if (error == DB_SUCCESS) {
|
|
|
|
return(0);
|
|
|
|
} else if (error == (int) DB_DUPLICATE_KEY) {
|
|
|
|
return(HA_ERR_FOUND_DUPP_KEY);
|
|
|
|
} else if (error == (int) DB_FOREIGN_DUPLICATE_KEY) {
|
|
|
|
return(HA_ERR_FOREIGN_DUPLICATE_KEY);
|
|
|
|
} else if (error == (int) DB_RECORD_NOT_FOUND) {
|
|
|
|
return(HA_ERR_NO_ACTIVE_RECORD);
|
|
|
|
} else if (error == (int) DB_ERROR) {
|
|
|
|
return(-1); /* unspecified error */
|
|
|
|
} else if (error == (int) DB_DEADLOCK) {
|
|
/* Since we rolled back the whole transaction, we must
|
|
tell it also to MySQL so that MySQL knows to empty the
|
|
cached binlog for this transaction */
|
|
|
|
if (thd) {
|
|
ha_rollback(thd);
|
|
}
|
|
|
|
return(HA_ERR_LOCK_DEADLOCK);
|
|
|
|
} else if (error == (int) DB_LOCK_WAIT_TIMEOUT) {
|
|
|
|
/* Starting from 5.0.13, we let MySQL just roll back the
|
|
latest SQL statement in a lock wait timeout. Previously, we
|
|
rolled back the whole transaction. */
|
|
|
|
if (thd && row_rollback_on_timeout) {
|
|
ha_rollback(thd);
|
|
}
|
|
|
|
return(HA_ERR_LOCK_WAIT_TIMEOUT);
|
|
|
|
} else if (error == (int) DB_NO_REFERENCED_ROW) {
|
|
|
|
return(HA_ERR_NO_REFERENCED_ROW);
|
|
|
|
} else if (error == (int) DB_ROW_IS_REFERENCED) {
|
|
|
|
return(HA_ERR_ROW_IS_REFERENCED);
|
|
|
|
} else if (error == (int) DB_CANNOT_ADD_CONSTRAINT) {
|
|
|
|
return(HA_ERR_CANNOT_ADD_FOREIGN);
|
|
|
|
} else if (error == (int) DB_CANNOT_DROP_CONSTRAINT) {
|
|
|
|
return(HA_ERR_ROW_IS_REFERENCED); /* TODO: This is a bit
|
|
misleading, a new MySQL error
|
|
code should be introduced */
|
|
} else if (error == (int) DB_COL_APPEARS_TWICE_IN_INDEX) {
|
|
|
|
return(HA_ERR_CRASHED);
|
|
|
|
} else if (error == (int) DB_OUT_OF_FILE_SPACE) {
|
|
|
|
return(HA_ERR_RECORD_FILE_FULL);
|
|
|
|
} else if (error == (int) DB_TABLE_IS_BEING_USED) {
|
|
|
|
return(HA_ERR_WRONG_COMMAND);
|
|
|
|
} else if (error == (int) DB_TABLE_NOT_FOUND) {
|
|
|
|
return(HA_ERR_KEY_NOT_FOUND);
|
|
|
|
} else if (error == (int) DB_TOO_BIG_RECORD) {
|
|
|
|
return(HA_ERR_TO_BIG_ROW);
|
|
|
|
} else if (error == (int) DB_CORRUPTION) {
|
|
|
|
return(HA_ERR_CRASHED);
|
|
} else if (error == (int) DB_NO_SAVEPOINT) {
|
|
|
|
return(HA_ERR_NO_SAVEPOINT);
|
|
} else if (error == (int) DB_LOCK_TABLE_FULL) {
|
|
/* Since we rolled back the whole transaction, we must
|
|
tell it also to MySQL so that MySQL knows to empty the
|
|
cached binlog for this transaction */
|
|
|
|
if (thd) {
|
|
ha_rollback(thd);
|
|
}
|
|
|
|
return(HA_ERR_LOCK_TABLE_FULL);
|
|
} else if (error == (int) DB_CANNOT_DROP_FOREIGN_INDEX) {
|
|
|
|
return(HA_ERR_DROP_INDEX_FK);
|
|
} else if (error == DB_TOO_MANY_CONCURRENT_TRXS) {
|
|
|
|
/* Once MySQL add the appropriate code to errmsg.txt then
|
|
we can get rid of this #ifdef. NOTE: The code checked by
|
|
the #ifdef is the suggested name for the error condition
|
|
and the actual error code name could very well be different.
|
|
This will require some monitoring, ie. the status
|
|
of this request on our part.*/
|
|
#ifdef ER_TOO_MANY_CONCURRENT_TRXS
|
|
return(ER_TOO_MANY_CONCURRENT_TRXS);
|
|
#else
|
|
return(HA_ERR_RECORD_FILE_FULL);
|
|
#endif
|
|
|
|
} else {
|
|
return(-1); // Unknown error
|
|
}
|
|
}
|
|
|
|
/*****************************************************************
|
|
If you want to print a thd that is not associated with the current thread,
|
|
you must call this function before reserving the InnoDB kernel_mutex, to
|
|
protect MySQL from setting thd->query NULL. If you print a thd of the current
|
|
thread, we know that MySQL cannot modify thd->query, and it is not necessary
|
|
to call this. Call innobase_mysql_end_print_arbitrary_thd() after you release
|
|
the kernel_mutex.
|
|
NOTE that /mysql/innobase/lock/lock0lock.c must contain the prototype for this
|
|
function! */
|
|
extern "C"
|
|
void
|
|
innobase_mysql_prepare_print_arbitrary_thd(void)
|
|
/*============================================*/
|
|
{
|
|
VOID(pthread_mutex_lock(&LOCK_thread_count));
|
|
}
|
|
|
|
/*****************************************************************
|
|
Releases the mutex reserved by innobase_mysql_prepare_print_arbitrary_thd().
|
|
NOTE that /mysql/innobase/lock/lock0lock.c must contain the prototype for this
|
|
function! */
|
|
extern "C"
|
|
void
|
|
innobase_mysql_end_print_arbitrary_thd(void)
|
|
/*========================================*/
|
|
{
|
|
VOID(pthread_mutex_unlock(&LOCK_thread_count));
|
|
}
|
|
|
|
/*****************************************************************
|
|
Prints info of a THD object (== user session thread) to the given file.
|
|
NOTE that /mysql/innobase/trx/trx0trx.c must contain the prototype for
|
|
this function! */
|
|
extern "C"
|
|
void
|
|
innobase_mysql_print_thd(
|
|
/*=====================*/
|
|
FILE* f, /* in: output stream */
|
|
void* input_thd, /* in: pointer to a MySQL THD object */
|
|
uint max_query_len) /* in: max query length to print, or 0 to
|
|
use the default max length */
|
|
{
|
|
THD* thd;
|
|
char buffer[1024];
|
|
|
|
thd = (THD*) input_thd;
|
|
fputs(thd_security_context(thd, buffer, sizeof(buffer),
|
|
max_query_len), f);
|
|
putc('\n', f);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Get the variable length bounds of the given character set.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/include/data0type.ic! */
|
|
extern "C"
|
|
void
|
|
innobase_get_cset_width(
|
|
/*====================*/
|
|
ulint cset, /* in: MySQL charset-collation code */
|
|
ulint* mbminlen, /* out: minimum length of a char (in bytes) */
|
|
ulint* mbmaxlen) /* out: maximum length of a char (in bytes) */
|
|
{
|
|
CHARSET_INFO* cs;
|
|
ut_ad(cset < 256);
|
|
ut_ad(mbminlen);
|
|
ut_ad(mbmaxlen);
|
|
|
|
cs = all_charsets[cset];
|
|
if (cs) {
|
|
*mbminlen = cs->mbminlen;
|
|
*mbmaxlen = cs->mbmaxlen;
|
|
} else {
|
|
ut_a(cset == 0);
|
|
*mbminlen = *mbmaxlen = 0;
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
Converts an identifier to a table name.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
void
|
|
innobase_convert_from_table_id(
|
|
/*===========================*/
|
|
char* to, /* out: converted identifier */
|
|
const char* from, /* in: identifier to convert */
|
|
ulint len) /* in: length of 'to', in bytes */
|
|
{
|
|
uint errors;
|
|
|
|
strconvert(thd_charset(current_thd), from,
|
|
&my_charset_filename, to, (uint) len, &errors);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Converts an identifier to UTF-8.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
void
|
|
innobase_convert_from_id(
|
|
/*=====================*/
|
|
char* to, /* out: converted identifier */
|
|
const char* from, /* in: identifier to convert */
|
|
ulint len) /* in: length of 'to', in bytes */
|
|
{
|
|
uint errors;
|
|
|
|
strconvert(thd_charset(current_thd), from,
|
|
system_charset_info, to, (uint) len, &errors);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Removes the filename encoding of a table or database name.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
void
|
|
innobase_convert_from_filename(
|
|
/*===========================*/
|
|
char* s) /* in: identifier; out: decoded identifier */
|
|
{
|
|
uint errors;
|
|
|
|
strconvert(&my_charset_filename, s,
|
|
system_charset_info, s, strlen(s), &errors);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Compares NUL-terminated UTF-8 strings case insensitively.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
int
|
|
innobase_strcasecmp(
|
|
/*================*/
|
|
/* out: 0 if a=b, <0 if a<b, >1 if a>b */
|
|
const char* a, /* in: first string to compare */
|
|
const char* b) /* in: second string to compare */
|
|
{
|
|
return(my_strcasecmp(system_charset_info, a, b));
|
|
}
|
|
|
|
/**********************************************************************
|
|
Makes all characters in a NUL-terminated UTF-8 string lower case.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
void
|
|
innobase_casedn_str(
|
|
/*================*/
|
|
char* a) /* in/out: string to put in lower case */
|
|
{
|
|
my_casedn_str(system_charset_info, a);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Determines the connection character set.
|
|
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/dict/dict0dict.c! */
|
|
extern "C"
|
|
struct charset_info_st*
|
|
innobase_get_charset(
|
|
/*=================*/
|
|
/* out: connection character set */
|
|
void* mysql_thd) /* in: MySQL thread handle */
|
|
{
|
|
return(thd_charset((THD*) mysql_thd));
|
|
}
|
|
|
|
/*************************************************************************
|
|
Creates a temporary file. */
|
|
extern "C"
|
|
int
|
|
innobase_mysql_tmpfile(void)
|
|
/*========================*/
|
|
/* out: temporary file descriptor, or < 0 on error */
|
|
{
|
|
return(mysql_tmpfile());
|
|
}
|
|
|
|
/*************************************************************************
|
|
Wrapper around MySQL's copy_and_convert function, see it for
|
|
documentation. */
|
|
extern "C"
|
|
ulint
|
|
innobase_convert_string(
|
|
/*====================*/
|
|
void* to,
|
|
ulint to_length,
|
|
CHARSET_INFO* to_cs,
|
|
const void* from,
|
|
ulint from_length,
|
|
CHARSET_INFO* from_cs,
|
|
uint* errors)
|
|
{
|
|
return(copy_and_convert((char*)to, (uint32) to_length, to_cs,
|
|
(const char*)from, (uint32) from_length, from_cs,
|
|
errors));
|
|
}
|
|
|
|
/*************************************************************************
|
|
Gets the InnoDB transaction handle for a MySQL handler object, creates
|
|
an InnoDB transaction struct if the corresponding MySQL thread struct still
|
|
lacks one. */
|
|
static
|
|
trx_t*
|
|
check_trx_exists(
|
|
/*=============*/
|
|
/* out: InnoDB transaction handle */
|
|
THD* thd) /* in: user thread handle */
|
|
{
|
|
trx_t*& trx = thd_to_trx(thd);
|
|
|
|
ut_ad(thd == current_thd);
|
|
|
|
if (trx == NULL) {
|
|
DBUG_ASSERT(thd != NULL);
|
|
trx = trx_allocate_for_mysql();
|
|
|
|
trx->mysql_thd = thd;
|
|
trx->mysql_query_str = thd_query(thd);
|
|
|
|
/* Update the info whether we should skip XA steps that eat
|
|
CPU time */
|
|
trx->support_xa = THDVAR(thd, support_xa);
|
|
} else {
|
|
if (trx->magic_n != TRX_MAGIC_N) {
|
|
mem_analyze_corruption(trx);
|
|
|
|
ut_error;
|
|
}
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
} else {
|
|
trx->check_foreigns = TRUE;
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_RELAXED_UNIQUE_CHECKS)) {
|
|
trx->check_unique_secondary = FALSE;
|
|
} else {
|
|
trx->check_unique_secondary = TRUE;
|
|
}
|
|
|
|
return(trx);
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
Construct ha_innobase handler. */
|
|
|
|
ha_innobase::ha_innobase(handlerton *hton, TABLE_SHARE *table_arg)
|
|
:handler(hton, table_arg),
|
|
int_table_flags(HA_REC_NOT_IN_SEQ |
|
|
HA_NULL_IN_KEY |
|
|
HA_CAN_INDEX_BLOBS |
|
|
HA_CAN_SQL_HANDLER |
|
|
HA_PRIMARY_KEY_REQUIRED_FOR_POSITION |
|
|
HA_PRIMARY_KEY_IN_READ_INDEX |
|
|
HA_CAN_GEOMETRY | HA_PARTIAL_COLUMN_READ |
|
|
HA_TABLE_SCAN_ON_INDEX),
|
|
start_of_scan(0),
|
|
num_write_row(0)
|
|
{}
|
|
|
|
/*************************************************************************
|
|
Updates the user_thd field in a handle and also allocates a new InnoDB
|
|
transaction handle if needed, and updates the transaction fields in the
|
|
prebuilt struct. */
|
|
inline
|
|
int
|
|
ha_innobase::update_thd(
|
|
/*====================*/
|
|
/* out: 0 or error code */
|
|
THD* thd) /* in: thd to use the handle */
|
|
{
|
|
trx_t* trx;
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
if (prebuilt->trx != trx) {
|
|
|
|
row_update_prebuilt_trx(prebuilt, trx);
|
|
}
|
|
|
|
user_thd = thd;
|
|
|
|
return(0);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Registers that InnoDB takes part in an SQL statement, so that MySQL knows to
|
|
roll back the statement if the statement results in an error. This MUST be
|
|
called for every SQL statement that may be rolled back by MySQL. Calling this
|
|
several times to register the same statement is allowed, too. */
|
|
inline
|
|
void
|
|
innobase_register_stmt(
|
|
/*===================*/
|
|
handlerton* hton, /* in: Innobase hton */
|
|
THD* thd) /* in: MySQL thd (connection) object */
|
|
{
|
|
/* Register the statement */
|
|
trans_register_ha(thd, FALSE, hton);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Registers an InnoDB transaction in MySQL, so that the MySQL XA code knows
|
|
to call the InnoDB prepare and commit, or rollback for the transaction. This
|
|
MUST be called for every transaction for which the user may call commit or
|
|
rollback. Calling this several times to register the same transaction is
|
|
allowed, too.
|
|
This function also registers the current SQL statement. */
|
|
inline
|
|
void
|
|
innobase_register_trx_and_stmt(
|
|
/*===========================*/
|
|
handlerton *hton, /* in: Innobase handlerton */
|
|
THD* thd) /* in: MySQL thd (connection) object */
|
|
{
|
|
/* NOTE that actually innobase_register_stmt() registers also
|
|
the transaction in the AUTOCOMMIT=1 mode. */
|
|
|
|
innobase_register_stmt(hton, thd);
|
|
|
|
if (thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
|
|
/* No autocommit mode, register for a transaction */
|
|
trans_register_ha(thd, TRUE, hton);
|
|
}
|
|
}
|
|
|
|
/* BACKGROUND INFO: HOW THE MYSQL QUERY CACHE WORKS WITH INNODB
|
|
------------------------------------------------------------
|
|
|
|
1) The use of the query cache for TBL is disabled when there is an
|
|
uncommitted change to TBL.
|
|
|
|
2) When a change to TBL commits, InnoDB stores the current value of
|
|
its global trx id counter, let us denote it by INV_TRX_ID, to the table object
|
|
in the InnoDB data dictionary, and does only allow such transactions whose
|
|
id <= INV_TRX_ID to use the query cache.
|
|
|
|
3) When InnoDB does an INSERT/DELETE/UPDATE to a table TBL, or an implicit
|
|
modification because an ON DELETE CASCADE, we invalidate the MySQL query cache
|
|
of TBL immediately.
|
|
|
|
How this is implemented inside InnoDB:
|
|
|
|
1) Since every modification always sets an IX type table lock on the InnoDB
|
|
table, it is easy to check if there can be uncommitted modifications for a
|
|
table: just check if there are locks in the lock list of the table.
|
|
|
|
2) When a transaction inside InnoDB commits, it reads the global trx id
|
|
counter and stores the value INV_TRX_ID to the tables on which it had a lock.
|
|
|
|
3) If there is an implicit table change from ON DELETE CASCADE or SET NULL,
|
|
InnoDB calls an invalidate method for the MySQL query cache for that table.
|
|
|
|
How this is implemented inside sql_cache.cc:
|
|
|
|
1) The query cache for an InnoDB table TBL is invalidated immediately at an
|
|
INSERT/UPDATE/DELETE, just like in the case of MyISAM. No need to delay
|
|
invalidation to the transaction commit.
|
|
|
|
2) To store or retrieve a value from the query cache of an InnoDB table TBL,
|
|
any query must first ask InnoDB's permission. We must pass the thd as a
|
|
parameter because InnoDB will look at the trx id, if any, associated with
|
|
that thd.
|
|
|
|
3) Use of the query cache for InnoDB tables is now allowed also when
|
|
AUTOCOMMIT==0 or we are inside BEGIN ... COMMIT. Thus transactions no longer
|
|
put restrictions on the use of the query cache.
|
|
*/
|
|
|
|
/**********************************************************************
|
|
The MySQL query cache uses this to check from InnoDB if the query cache at
|
|
the moment is allowed to operate on an InnoDB table. The SQL query must
|
|
be a non-locking SELECT.
|
|
|
|
The query cache is allowed to operate on certain query only if this function
|
|
returns TRUE for all tables in the query.
|
|
|
|
If thd is not in the autocommit state, this function also starts a new
|
|
transaction for thd if there is no active trx yet, and assigns a consistent
|
|
read view to it if there is no read view yet.
|
|
|
|
Why a deadlock of threads is not possible: the query cache calls this function
|
|
at the start of a SELECT processing. Then the calling thread cannot be
|
|
holding any InnoDB semaphores. The calling thread is holding the
|
|
query cache mutex, and this function will reserver the InnoDB kernel mutex.
|
|
Thus, the 'rank' in sync0sync.h of the MySQL query cache mutex is above
|
|
the InnoDB kernel mutex. */
|
|
static
|
|
my_bool
|
|
innobase_query_caching_of_table_permitted(
|
|
/*======================================*/
|
|
/* out: TRUE if permitted, FALSE if not;
|
|
note that the value FALSE does not mean
|
|
we should invalidate the query cache:
|
|
invalidation is called explicitly */
|
|
THD* thd, /* in: thd of the user who is trying to
|
|
store a result to the query cache or
|
|
retrieve it */
|
|
char* full_name, /* in: concatenation of database name,
|
|
the null character '\0', and the table
|
|
name */
|
|
uint full_name_len, /* in: length of the full name, i.e.
|
|
len(dbname) + len(tablename) + 1 */
|
|
ulonglong *unused) /* unused for this engine */
|
|
{
|
|
ibool is_autocommit;
|
|
trx_t* trx;
|
|
char norm_name[1000];
|
|
|
|
ut_a(full_name_len < 999);
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
if (trx->isolation_level == TRX_ISO_SERIALIZABLE) {
|
|
/* In the SERIALIZABLE mode we add LOCK IN SHARE MODE to every
|
|
plain SELECT if AUTOCOMMIT is not on. */
|
|
|
|
return((my_bool)FALSE);
|
|
}
|
|
|
|
if (trx->has_search_latch) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("The calling thread is holding the adaptive "
|
|
"search, latch though calling "
|
|
"innobase_query_caching_of_table_permitted.");
|
|
|
|
mutex_enter_noninline(&kernel_mutex);
|
|
trx_print(stderr, trx, 1024);
|
|
mutex_exit_noninline(&kernel_mutex);
|
|
}
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
if (!thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
|
|
is_autocommit = TRUE;
|
|
} else {
|
|
is_autocommit = FALSE;
|
|
|
|
}
|
|
|
|
if (is_autocommit && trx->n_mysql_tables_in_use == 0) {
|
|
/* We are going to retrieve the query result from the query
|
|
cache. This cannot be a store operation to the query cache
|
|
because then MySQL would have locks on tables already.
|
|
|
|
TODO: if the user has used LOCK TABLES to lock the table,
|
|
then we open a transaction in the call of row_.. below.
|
|
That trx can stay open until UNLOCK TABLES. The same problem
|
|
exists even if we do not use the query cache. MySQL should be
|
|
modified so that it ALWAYS calls some cleanup function when
|
|
the processing of a query ends!
|
|
|
|
We can imagine we instantaneously serialize this consistent
|
|
read trx to the current trx id counter. If trx2 would have
|
|
changed the tables of a query result stored in the cache, and
|
|
trx2 would have already committed, making the result obsolete,
|
|
then trx2 would have already invalidated the cache. Thus we
|
|
can trust the result in the cache is ok for this query. */
|
|
|
|
return((my_bool)TRUE);
|
|
}
|
|
|
|
/* Normalize the table name to InnoDB format */
|
|
|
|
memcpy(norm_name, full_name, full_name_len);
|
|
|
|
norm_name[strlen(norm_name)] = '/'; /* InnoDB uses '/' as the
|
|
separator between db and table */
|
|
norm_name[full_name_len] = '\0';
|
|
#ifdef __WIN__
|
|
innobase_casedn_str(norm_name);
|
|
#endif
|
|
/* The call of row_search_.. will start a new transaction if it is
|
|
not yet started */
|
|
|
|
if (trx->active_trans == 0) {
|
|
|
|
innobase_register_trx_and_stmt(innodb_hton_ptr, thd);
|
|
trx->active_trans = 1;
|
|
}
|
|
|
|
if (row_search_check_if_query_cache_permitted(trx, norm_name)) {
|
|
|
|
/* printf("Query cache for %s permitted\n", norm_name); */
|
|
|
|
return((my_bool)TRUE);
|
|
}
|
|
|
|
/* printf("Query cache for %s NOT permitted\n", norm_name); */
|
|
|
|
return((my_bool)FALSE);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Invalidates the MySQL query cache for the table.
|
|
NOTE that the exact prototype of this function has to be in
|
|
/innobase/row/row0ins.c! */
|
|
extern "C"
|
|
void
|
|
innobase_invalidate_query_cache(
|
|
/*============================*/
|
|
trx_t* trx, /* in: transaction which modifies the table */
|
|
char* full_name, /* in: concatenation of database name, null
|
|
char '\0', table name, null char'\0';
|
|
NOTE that in Windows this is always
|
|
in LOWER CASE! */
|
|
ulint full_name_len) /* in: full name length where also the null
|
|
chars count */
|
|
{
|
|
/* Note that the sync0sync.h rank of the query cache mutex is just
|
|
above the InnoDB kernel mutex. The caller of this function must not
|
|
have latches of a lower rank. */
|
|
|
|
/* Argument TRUE below means we are using transactions */
|
|
#ifdef HAVE_QUERY_CACHE
|
|
mysql_query_cache_invalidate4((THD*) trx->mysql_thd,
|
|
(const char*) full_name,
|
|
(uint32) full_name_len,
|
|
TRUE);
|
|
#endif
|
|
}
|
|
|
|
/*********************************************************************
|
|
Display an SQL identifier.
|
|
This definition must match the one in innobase/ut/ut0ut.c! */
|
|
extern "C"
|
|
void
|
|
innobase_print_identifier(
|
|
/*======================*/
|
|
FILE* f, /* in: output stream */
|
|
trx_t* trx, /* in: transaction */
|
|
ibool table_id,/* in: TRUE=print a table name,
|
|
FALSE=print other identifier */
|
|
const char* name, /* in: name to print */
|
|
ulint namelen)/* in: length of name */
|
|
{
|
|
const char* s = name;
|
|
char* qname = NULL;
|
|
int q;
|
|
|
|
if (table_id) {
|
|
/* Decode the table name. The filename_to_tablename()
|
|
function expects a NUL-terminated string. The input and
|
|
output strings buffers must not be shared. The function
|
|
only produces more output when the name contains other
|
|
characters than [0-9A-Z_a-z]. */
|
|
char* temp_name = my_malloc((uint) namelen + 1, MYF(MY_WME));
|
|
uint qnamelen = (uint) (namelen
|
|
+ (1 + sizeof srv_mysql50_table_name_prefix));
|
|
|
|
if (temp_name) {
|
|
qname = my_malloc(qnamelen, MYF(MY_WME));
|
|
if (qname) {
|
|
memcpy(temp_name, name, namelen);
|
|
temp_name[namelen] = 0;
|
|
s = qname;
|
|
namelen = filename_to_tablename(temp_name,
|
|
qname, qnamelen);
|
|
}
|
|
my_free(temp_name, MYF(0));
|
|
}
|
|
}
|
|
|
|
if (!trx || !trx->mysql_thd) {
|
|
|
|
q = '"';
|
|
} else {
|
|
q = get_quote_char_for_identifier((THD*) trx->mysql_thd,
|
|
s, (int) namelen);
|
|
}
|
|
|
|
if (q == EOF) {
|
|
fwrite(s, 1, namelen, f);
|
|
} else {
|
|
const char* e = s + namelen;
|
|
putc(q, f);
|
|
while (s < e) {
|
|
int c = *s++;
|
|
if (c == q) {
|
|
putc(c, f);
|
|
}
|
|
putc(c, f);
|
|
}
|
|
putc(q, f);
|
|
}
|
|
|
|
my_free(qname, MYF(MY_ALLOW_ZERO_PTR));
|
|
}
|
|
|
|
/**************************************************************************
|
|
Determines if the currently running transaction has been interrupted. */
|
|
extern "C"
|
|
ibool
|
|
trx_is_interrupted(
|
|
/*===============*/
|
|
/* out: TRUE if interrupted */
|
|
trx_t* trx) /* in: transaction */
|
|
{
|
|
return(trx && trx->mysql_thd && thd_killed((THD*) trx->mysql_thd));
|
|
}
|
|
|
|
/******************************************************************
|
|
Resets some fields of a prebuilt struct. The template is used in fast
|
|
retrieval of just those column values MySQL needs in its processing. */
|
|
static
|
|
void
|
|
reset_template(
|
|
/*===========*/
|
|
row_prebuilt_t* prebuilt) /* in/out: prebuilt struct */
|
|
{
|
|
prebuilt->keep_other_fields_on_keyread = 0;
|
|
prebuilt->read_just_key = 0;
|
|
}
|
|
|
|
/*********************************************************************
|
|
Call this when you have opened a new table handle in HANDLER, before you
|
|
call index_read_idx() etc. Actually, we can let the cursor stay open even
|
|
over a transaction commit! Then you should call this before every operation,
|
|
fetch next etc. This function inits the necessary things even after a
|
|
transaction commit. */
|
|
|
|
void
|
|
ha_innobase::init_table_handle_for_HANDLER(void)
|
|
/*============================================*/
|
|
{
|
|
/* If current thd does not yet have a trx struct, create one.
|
|
If the current handle does not yet have a prebuilt struct, create
|
|
one. Update the trx pointers in the prebuilt struct. Normally
|
|
this operation is done in external_lock. */
|
|
|
|
update_thd(ha_thd());
|
|
|
|
/* Initialize the prebuilt struct much like it would be inited in
|
|
external_lock */
|
|
|
|
innobase_release_stat_resources(prebuilt->trx);
|
|
|
|
/* If the transaction is not started yet, start it */
|
|
|
|
trx_start_if_not_started_noninline(prebuilt->trx);
|
|
|
|
/* Assign a read view if the transaction does not have it yet */
|
|
|
|
trx_assign_read_view(prebuilt->trx);
|
|
|
|
/* Set the MySQL flag to mark that there is an active transaction */
|
|
|
|
if (prebuilt->trx->active_trans == 0) {
|
|
|
|
innobase_register_trx_and_stmt(ht, user_thd);
|
|
|
|
prebuilt->trx->active_trans = 1;
|
|
}
|
|
|
|
/* We did the necessary inits in this function, no need to repeat them
|
|
in row_search_for_mysql */
|
|
|
|
prebuilt->sql_stat_start = FALSE;
|
|
|
|
/* We let HANDLER always to do the reads as consistent reads, even
|
|
if the trx isolation level would have been specified as SERIALIZABLE */
|
|
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
prebuilt->stored_select_lock_type = LOCK_NONE;
|
|
|
|
/* Always fetch all columns in the index record */
|
|
|
|
prebuilt->hint_need_to_fetch_extra_cols = ROW_RETRIEVE_ALL_COLS;
|
|
|
|
/* We want always to fetch all columns in the whole row? Or do
|
|
we???? */
|
|
|
|
prebuilt->used_in_HANDLER = TRUE;
|
|
reset_template(prebuilt);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Opens an InnoDB database. */
|
|
static
|
|
int
|
|
innobase_init(
|
|
/*==========*/
|
|
/* out: 0 on success, error code on failure */
|
|
void *p) /* in: InnoDB handlerton */
|
|
{
|
|
static char current_dir[3]; /* Set if using current lib */
|
|
int err;
|
|
bool ret;
|
|
char *default_path;
|
|
|
|
DBUG_ENTER("innobase_init");
|
|
handlerton *innobase_hton= (handlerton *)p;
|
|
innodb_hton_ptr = innobase_hton;
|
|
|
|
innobase_hton->state = SHOW_OPTION_YES;
|
|
innobase_hton->db_type= DB_TYPE_INNODB;
|
|
innobase_hton->savepoint_offset=sizeof(trx_named_savept_t);
|
|
innobase_hton->close_connection=innobase_close_connection;
|
|
innobase_hton->savepoint_set=innobase_savepoint;
|
|
innobase_hton->savepoint_rollback=innobase_rollback_to_savepoint;
|
|
innobase_hton->savepoint_release=innobase_release_savepoint;
|
|
innobase_hton->commit=innobase_commit;
|
|
innobase_hton->rollback=innobase_rollback;
|
|
innobase_hton->prepare=innobase_xa_prepare;
|
|
innobase_hton->recover=innobase_xa_recover;
|
|
innobase_hton->commit_by_xid=innobase_commit_by_xid;
|
|
innobase_hton->rollback_by_xid=innobase_rollback_by_xid;
|
|
innobase_hton->create_cursor_read_view=innobase_create_cursor_view;
|
|
innobase_hton->set_cursor_read_view=innobase_set_cursor_view;
|
|
innobase_hton->close_cursor_read_view=innobase_close_cursor_view;
|
|
innobase_hton->create=innobase_create_handler;
|
|
innobase_hton->drop_database=innobase_drop_database;
|
|
innobase_hton->panic=innobase_end;
|
|
innobase_hton->start_consistent_snapshot=innobase_start_trx_and_assign_read_view;
|
|
innobase_hton->flush_logs=innobase_flush_logs;
|
|
innobase_hton->show_status=innobase_show_status;
|
|
innobase_hton->flags=HTON_NO_FLAGS;
|
|
innobase_hton->release_temporary_latches=innobase_release_temporary_latches;
|
|
innobase_hton->alter_table_flags = innobase_alter_table_flags;
|
|
|
|
ut_a(DATA_MYSQL_TRUE_VARCHAR == (ulint)MYSQL_TYPE_VARCHAR);
|
|
|
|
#ifdef UNIV_DEBUG
|
|
static const char test_filename[] = "-@";
|
|
char test_tablename[sizeof test_filename
|
|
+ sizeof srv_mysql50_table_name_prefix];
|
|
if ((sizeof test_tablename) - 1
|
|
!= filename_to_tablename(test_filename, test_tablename,
|
|
sizeof test_tablename)
|
|
|| strncmp(test_tablename,
|
|
srv_mysql50_table_name_prefix,
|
|
sizeof srv_mysql50_table_name_prefix)
|
|
|| strcmp(test_tablename
|
|
+ sizeof srv_mysql50_table_name_prefix,
|
|
test_filename)) {
|
|
sql_print_error("tablename encoding has been changed");
|
|
goto error;
|
|
}
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
/* Check that values don't overflow on 32-bit systems. */
|
|
if (sizeof(ulint) == 4) {
|
|
if (innobase_buffer_pool_size > UINT_MAX32) {
|
|
sql_print_error(
|
|
"innobase_buffer_pool_size can't be over 4GB"
|
|
" on 32-bit systems");
|
|
|
|
goto error;
|
|
}
|
|
|
|
if (innobase_log_file_size > UINT_MAX32) {
|
|
sql_print_error(
|
|
"innobase_log_file_size can't be over 4GB"
|
|
" on 32-bit systems");
|
|
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
os_innodb_umask = (ulint)my_umask;
|
|
|
|
/* First calculate the default path for innodb_data_home_dir etc.,
|
|
in case the user has not given any value.
|
|
|
|
Note that when using the embedded server, the datadirectory is not
|
|
necessarily the current directory of this program. */
|
|
|
|
if (mysqld_embedded) {
|
|
default_path = mysql_real_data_home;
|
|
fil_path_to_mysql_datadir = mysql_real_data_home;
|
|
} else {
|
|
/* It's better to use current lib, to keep paths short */
|
|
current_dir[0] = FN_CURLIB;
|
|
current_dir[1] = FN_LIBCHAR;
|
|
current_dir[2] = 0;
|
|
default_path = current_dir;
|
|
}
|
|
|
|
ut_a(default_path);
|
|
|
|
if (specialflag & SPECIAL_NO_PRIOR) {
|
|
srv_set_thread_priorities = FALSE;
|
|
} else {
|
|
srv_set_thread_priorities = TRUE;
|
|
srv_query_thread_priority = QUERY_PRIOR;
|
|
}
|
|
|
|
/* Set InnoDB initialization parameters according to the values
|
|
read from MySQL .cnf file */
|
|
|
|
/*--------------- Data files -------------------------*/
|
|
|
|
/* The default dir for data files is the datadir of MySQL */
|
|
|
|
srv_data_home = (innobase_data_home_dir ? innobase_data_home_dir :
|
|
default_path);
|
|
|
|
/* Set default InnoDB data file size to 10 MB and let it be
|
|
auto-extending. Thus users can use InnoDB in >= 4.0 without having
|
|
to specify any startup options. */
|
|
|
|
if (!innobase_data_file_path) {
|
|
innobase_data_file_path = (char*) "ibdata1:10M:autoextend";
|
|
}
|
|
|
|
/* Since InnoDB edits the argument in the next call, we make another
|
|
copy of it: */
|
|
|
|
internal_innobase_data_file_path = my_strdup(innobase_data_file_path,
|
|
MYF(MY_FAE));
|
|
|
|
ret = (bool) srv_parse_data_file_paths_and_sizes(
|
|
internal_innobase_data_file_path,
|
|
&srv_data_file_names,
|
|
&srv_data_file_sizes,
|
|
&srv_data_file_is_raw_partition,
|
|
&srv_n_data_files,
|
|
&srv_auto_extend_last_data_file,
|
|
&srv_last_file_size_max);
|
|
if (ret == FALSE) {
|
|
sql_print_error(
|
|
"InnoDB: syntax error in innodb_data_file_path");
|
|
my_free(internal_innobase_data_file_path,
|
|
MYF(MY_ALLOW_ZERO_PTR));
|
|
goto error;
|
|
}
|
|
|
|
/* -------------- Log files ---------------------------*/
|
|
|
|
/* The default dir for log files is the datadir of MySQL */
|
|
|
|
if (!innobase_log_group_home_dir) {
|
|
innobase_log_group_home_dir = default_path;
|
|
}
|
|
|
|
#ifdef UNIV_LOG_ARCHIVE
|
|
/* Since innodb_log_arch_dir has no relevance under MySQL,
|
|
starting from 4.0.6 we always set it the same as
|
|
innodb_log_group_home_dir: */
|
|
|
|
innobase_log_arch_dir = innobase_log_group_home_dir;
|
|
|
|
srv_arch_dir = innobase_log_arch_dir;
|
|
#endif /* UNIG_LOG_ARCHIVE */
|
|
|
|
ret = (bool)
|
|
srv_parse_log_group_home_dirs(innobase_log_group_home_dir,
|
|
&srv_log_group_home_dirs);
|
|
|
|
if (ret == FALSE || innobase_mirrored_log_groups != 1) {
|
|
sql_print_error("syntax error in innodb_log_group_home_dir, or a "
|
|
"wrong number of mirrored log groups");
|
|
|
|
my_free(internal_innobase_data_file_path,
|
|
MYF(MY_ALLOW_ZERO_PTR));
|
|
goto error;
|
|
}
|
|
|
|
/* --------------------------------------------------*/
|
|
|
|
srv_file_flush_method_str = innobase_unix_file_flush_method;
|
|
|
|
srv_n_log_groups = (ulint) innobase_mirrored_log_groups;
|
|
srv_n_log_files = (ulint) innobase_log_files_in_group;
|
|
srv_log_file_size = (ulint) innobase_log_file_size;
|
|
|
|
#ifdef UNIV_LOG_ARCHIVE
|
|
srv_log_archive_on = (ulint) innobase_log_archive;
|
|
#endif /* UNIV_LOG_ARCHIVE */
|
|
srv_log_buffer_size = (ulint) innobase_log_buffer_size;
|
|
|
|
srv_buf_pool_size = (ulint) innobase_buffer_pool_size;
|
|
|
|
srv_mem_pool_size = (ulint) innobase_additional_mem_pool_size;
|
|
|
|
srv_n_file_io_threads = (ulint) innobase_file_io_threads;
|
|
|
|
srv_lock_wait_timeout = (ulint) innobase_lock_wait_timeout;
|
|
srv_force_recovery = (ulint) innobase_force_recovery;
|
|
|
|
srv_use_doublewrite_buf = (ibool) innobase_use_doublewrite;
|
|
srv_use_checksums = (ibool) innobase_use_checksums;
|
|
|
|
#ifdef HAVE_LARGE_PAGES
|
|
if ((os_use_large_pages = (ibool) my_use_large_pages))
|
|
os_large_page_size = (ulint) opt_large_page_size;
|
|
#endif
|
|
|
|
row_rollback_on_timeout = (ibool) innobase_rollback_on_timeout;
|
|
|
|
srv_file_per_table = (ibool) innobase_file_per_table;
|
|
srv_locks_unsafe_for_binlog = (ibool) innobase_locks_unsafe_for_binlog;
|
|
|
|
srv_max_n_open_files = (ulint) innobase_open_files;
|
|
srv_innodb_status = (ibool) innobase_create_status_file;
|
|
|
|
srv_stats_on_metadata = (ibool) innobase_stats_on_metadata;
|
|
|
|
srv_print_verbose_log = mysqld_embedded ? 0 : 1;
|
|
|
|
/* Store the default charset-collation number of this MySQL
|
|
installation */
|
|
|
|
data_mysql_default_charset_coll = (ulint)default_charset_info->number;
|
|
|
|
ut_a(DATA_MYSQL_LATIN1_SWEDISH_CHARSET_COLL ==
|
|
my_charset_latin1.number);
|
|
ut_a(DATA_MYSQL_BINARY_CHARSET_COLL == my_charset_bin.number);
|
|
|
|
/* Store the latin1_swedish_ci character ordering table to InnoDB. For
|
|
non-latin1_swedish_ci charsets we use the MySQL comparison functions,
|
|
and consequently we do not need to know the ordering internally in
|
|
InnoDB. */
|
|
|
|
ut_a(0 == strcmp((char*)my_charset_latin1.name,
|
|
(char*)"latin1_swedish_ci"));
|
|
memcpy(srv_latin1_ordering, my_charset_latin1.sort_order, 256);
|
|
|
|
/* Since we in this module access directly the fields of a trx
|
|
struct, and due to different headers and flags it might happen that
|
|
mutex_t has a different size in this module and in InnoDB
|
|
modules, we check at run time that the size is the same in
|
|
these compilation modules. */
|
|
|
|
srv_sizeof_trx_t_in_ha_innodb_cc = sizeof(trx_t);
|
|
|
|
err = innobase_start_or_create_for_mysql();
|
|
|
|
if (err != DB_SUCCESS) {
|
|
my_free(internal_innobase_data_file_path,
|
|
MYF(MY_ALLOW_ZERO_PTR));
|
|
goto error;
|
|
}
|
|
|
|
(void) hash_init(&innobase_open_tables,system_charset_info, 32, 0, 0,
|
|
(hash_get_key) innobase_get_key, 0, 0);
|
|
pthread_mutex_init(&innobase_share_mutex, MY_MUTEX_INIT_FAST);
|
|
pthread_mutex_init(&prepare_commit_mutex, MY_MUTEX_INIT_FAST);
|
|
pthread_mutex_init(&commit_threads_m, MY_MUTEX_INIT_FAST);
|
|
pthread_mutex_init(&commit_cond_m, MY_MUTEX_INIT_FAST);
|
|
pthread_cond_init(&commit_cond, NULL);
|
|
innodb_inited= 1;
|
|
|
|
DBUG_RETURN(FALSE);
|
|
error:
|
|
innobase_hton->state = SHOW_OPTION_DISABLED;
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Closes an InnoDB database. */
|
|
static
|
|
int
|
|
innobase_end(handlerton *hton, ha_panic_function type)
|
|
/*==============*/
|
|
/* out: TRUE if error */
|
|
{
|
|
int err= 0;
|
|
|
|
DBUG_ENTER("innobase_end");
|
|
|
|
#ifdef __NETWARE__ /* some special cleanup for NetWare */
|
|
if (nw_panic) {
|
|
set_panic_flag_for_netware();
|
|
}
|
|
#endif
|
|
if (innodb_inited) {
|
|
|
|
srv_fast_shutdown = (ulint) innobase_fast_shutdown;
|
|
innodb_inited = 0;
|
|
if (innobase_shutdown_for_mysql() != DB_SUCCESS) {
|
|
err = 1;
|
|
}
|
|
hash_free(&innobase_open_tables);
|
|
my_free(internal_innobase_data_file_path,
|
|
MYF(MY_ALLOW_ZERO_PTR));
|
|
pthread_mutex_destroy(&innobase_share_mutex);
|
|
pthread_mutex_destroy(&prepare_commit_mutex);
|
|
pthread_mutex_destroy(&commit_threads_m);
|
|
pthread_mutex_destroy(&commit_cond_m);
|
|
pthread_cond_destroy(&commit_cond);
|
|
}
|
|
|
|
DBUG_RETURN(err);
|
|
}
|
|
|
|
/********************************************************************
|
|
Flushes InnoDB logs to disk and makes a checkpoint. Really, a commit flushes
|
|
the logs, and the name of this function should be innobase_checkpoint. */
|
|
static
|
|
bool
|
|
innobase_flush_logs(handlerton *hton)
|
|
/*=====================*/
|
|
/* out: TRUE if error */
|
|
{
|
|
bool result = 0;
|
|
|
|
DBUG_ENTER("innobase_flush_logs");
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
/********************************************************************
|
|
Return alter table flags supported in an InnoDB database. */
|
|
static
|
|
uint
|
|
innobase_alter_table_flags(
|
|
/*=======================*/
|
|
uint flags)
|
|
{
|
|
return(HA_ONLINE_ADD_INDEX_NO_WRITES
|
|
| HA_ONLINE_DROP_INDEX_NO_WRITES
|
|
| HA_ONLINE_ADD_UNIQUE_INDEX_NO_WRITES
|
|
| HA_ONLINE_DROP_UNIQUE_INDEX_NO_WRITES
|
|
| HA_ONLINE_ADD_PK_INDEX_NO_WRITES);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Commits a transaction in an InnoDB database. */
|
|
static
|
|
void
|
|
innobase_commit_low(
|
|
/*================*/
|
|
trx_t* trx) /* in: transaction handle */
|
|
{
|
|
if (trx->conc_state == TRX_NOT_STARTED) {
|
|
|
|
return;
|
|
}
|
|
|
|
trx_commit_for_mysql(trx);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates an InnoDB transaction struct for the thd if it does not yet have one.
|
|
Starts a new InnoDB transaction if a transaction is not yet started. And
|
|
assigns a new snapshot for a consistent read if the transaction does not yet
|
|
have one. */
|
|
static
|
|
int
|
|
innobase_start_trx_and_assign_read_view(
|
|
/*====================================*/
|
|
/* out: 0 */
|
|
handlerton *hton, /* in: Innodb handlerton */
|
|
THD* thd) /* in: MySQL thread handle of the user for whom
|
|
the transaction should be committed */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("innobase_start_trx_and_assign_read_view");
|
|
|
|
/* Create a new trx struct for thd, if it does not yet have one */
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* This is just to play safe: release a possible FIFO ticket and
|
|
search latch. Since we will reserve the kernel mutex, we have to
|
|
release the search system latch first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
/* If the transaction is not started yet, start it */
|
|
|
|
trx_start_if_not_started_noninline(trx);
|
|
|
|
/* Assign a read view if the transaction does not have it yet */
|
|
|
|
trx_assign_read_view(trx);
|
|
|
|
/* Set the MySQL flag to mark that there is an active transaction */
|
|
|
|
if (trx->active_trans == 0) {
|
|
innobase_register_trx_and_stmt(hton, current_thd);
|
|
trx->active_trans = 1;
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Commits a transaction in an InnoDB database or marks an SQL statement
|
|
ended. */
|
|
static
|
|
int
|
|
innobase_commit(
|
|
/*============*/
|
|
/* out: 0 */
|
|
handlerton *hton, /* in: Innodb handlerton */
|
|
THD* thd, /* in: MySQL thread handle of the user for whom
|
|
the transaction should be committed */
|
|
bool all) /* in: TRUE - commit transaction
|
|
FALSE - the current SQL statement ended */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("innobase_commit");
|
|
DBUG_PRINT("trans", ("ending transaction"));
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* Update the info whether we should skip XA steps that eat CPU time */
|
|
trx->support_xa = THDVAR(thd, support_xa);
|
|
|
|
/* Since we will reserve the kernel mutex, we have to release
|
|
the search system latch first to obey the latching order. */
|
|
|
|
if (trx->has_search_latch) {
|
|
trx_search_latch_release_if_reserved(trx);
|
|
}
|
|
|
|
/* The flag trx->active_trans is set to 1 in
|
|
|
|
1. ::external_lock(),
|
|
2. ::start_stmt(),
|
|
3. innobase_query_caching_of_table_permitted(),
|
|
4. innobase_savepoint(),
|
|
5. ::init_table_handle_for_HANDLER(),
|
|
6. innobase_start_trx_and_assign_read_view(),
|
|
7. ::transactional_table_lock()
|
|
|
|
and it is only set to 0 in a commit or a rollback. If it is 0 we know
|
|
there cannot be resources to be freed and we could return immediately.
|
|
For the time being, we play safe and do the cleanup though there should
|
|
be nothing to clean up. */
|
|
|
|
if (trx->active_trans == 0
|
|
&& trx->conc_state != TRX_NOT_STARTED) {
|
|
|
|
sql_print_error("trx->active_trans == 0, but"
|
|
" trx->conc_state != TRX_NOT_STARTED");
|
|
}
|
|
if (all
|
|
|| (!thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))) {
|
|
|
|
/* We were instructed to commit the whole transaction, or
|
|
this is an SQL statement end and autocommit is on */
|
|
|
|
/* We need current binlog position for ibbackup to work.
|
|
Note, the position is current because of
|
|
prepare_commit_mutex */
|
|
retry:
|
|
if (srv_commit_concurrency > 0) {
|
|
pthread_mutex_lock(&commit_cond_m);
|
|
commit_threads++;
|
|
|
|
if (commit_threads > srv_commit_concurrency) {
|
|
commit_threads--;
|
|
pthread_cond_wait(&commit_cond,
|
|
&commit_cond_m);
|
|
pthread_mutex_unlock(&commit_cond_m);
|
|
goto retry;
|
|
}
|
|
else {
|
|
pthread_mutex_unlock(&commit_cond_m);
|
|
}
|
|
}
|
|
|
|
trx->mysql_log_file_name = mysql_bin_log_file_name();
|
|
trx->mysql_log_offset = (ib_longlong) mysql_bin_log_file_pos();
|
|
|
|
innobase_commit_low(trx);
|
|
|
|
if (srv_commit_concurrency > 0) {
|
|
pthread_mutex_lock(&commit_cond_m);
|
|
commit_threads--;
|
|
pthread_cond_signal(&commit_cond);
|
|
pthread_mutex_unlock(&commit_cond_m);
|
|
}
|
|
|
|
if (trx->active_trans == 2) {
|
|
|
|
pthread_mutex_unlock(&prepare_commit_mutex);
|
|
}
|
|
|
|
trx->active_trans = 0;
|
|
|
|
} else {
|
|
/* We just mark the SQL statement ended and do not do a
|
|
transaction commit */
|
|
|
|
if (trx->auto_inc_lock) {
|
|
/* If we had reserved the auto-inc lock for some
|
|
table in this SQL statement we release it now */
|
|
|
|
row_unlock_table_autoinc_for_mysql(trx);
|
|
}
|
|
/* Store the current undo_no of the transaction so that we
|
|
know where to roll back if we have to roll back the next
|
|
SQL statement */
|
|
|
|
trx_mark_sql_stat_end(trx);
|
|
}
|
|
|
|
if (trx->declared_to_be_inside_innodb) {
|
|
/* Release our possible ticket in the FIFO */
|
|
|
|
srv_conc_force_exit_innodb(trx);
|
|
}
|
|
|
|
/* Tell the InnoDB server that there might be work for utility
|
|
threads: */
|
|
srv_active_wake_master_thread();
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
#if 0
|
|
/* TODO: put the
|
|
MySQL-4.1 functionality back to 5.0. This is needed to get InnoDB Hot Backup
|
|
to work. */
|
|
|
|
/*********************************************************************
|
|
This is called when MySQL writes the binlog entry for the current
|
|
transaction. Writes to the InnoDB tablespace info which tells where the
|
|
MySQL binlog entry for the current transaction ended. Also commits the
|
|
transaction inside InnoDB but does NOT flush InnoDB log files to disk.
|
|
To flush you have to call innobase_commit_complete(). We have separated
|
|
flushing to eliminate the bottleneck of LOCK_log in log.cc which disabled
|
|
InnoDB's group commit capability. */
|
|
static
|
|
int
|
|
innobase_report_binlog_offset_and_commit(
|
|
/*=====================================*/
|
|
/* out: 0 */
|
|
handlerton *hton, /* in: Innodb handlerton */
|
|
THD* thd, /* in: user thread */
|
|
void* trx_handle, /* in: InnoDB trx handle */
|
|
char* log_file_name, /* in: latest binlog file name */
|
|
my_off_t end_offset) /* in: the offset in the binlog file
|
|
up to which we wrote */
|
|
{
|
|
trx_t* trx;
|
|
|
|
trx = (trx_t*)trx_handle;
|
|
|
|
ut_a(trx != NULL);
|
|
|
|
trx->mysql_log_file_name = log_file_name;
|
|
trx->mysql_log_offset = (ib_longlong)end_offset;
|
|
|
|
trx->flush_log_later = TRUE;
|
|
|
|
innobase_commit(hton, thd, TRUE);
|
|
|
|
trx->flush_log_later = FALSE;
|
|
|
|
return(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function stores the binlog offset and flushes logs. */
|
|
static
|
|
void
|
|
innobase_store_binlog_offset_and_flush_log(
|
|
/*=======================================*/
|
|
char* binlog_name, /* in: binlog name */
|
|
longlong offset) /* in: binlog offset */
|
|
{
|
|
mtr_t mtr;
|
|
|
|
assert(binlog_name != NULL);
|
|
|
|
/* Start a mini-transaction */
|
|
mtr_start_noninline(&mtr);
|
|
|
|
/* Update the latest MySQL binlog name and offset info
|
|
in trx sys header */
|
|
|
|
trx_sys_update_mysql_binlog_offset(
|
|
binlog_name,
|
|
offset,
|
|
TRX_SYS_MYSQL_LOG_INFO, &mtr);
|
|
|
|
/* Commits the mini-transaction */
|
|
mtr_commit(&mtr);
|
|
|
|
/* Synchronous flush of the log buffer to disk */
|
|
log_buffer_flush_to_disk();
|
|
}
|
|
|
|
/*********************************************************************
|
|
This is called after MySQL has written the binlog entry for the current
|
|
transaction. Flushes the InnoDB log files to disk if required. */
|
|
static
|
|
int
|
|
innobase_commit_complete(
|
|
/*=====================*/
|
|
/* out: 0 */
|
|
THD* thd) /* in: user thread */
|
|
{
|
|
trx_t* trx;
|
|
|
|
trx = thd_to_trx(thd);
|
|
|
|
if (trx && trx->active_trans) {
|
|
|
|
trx->active_trans = 0;
|
|
|
|
if (UNIV_UNLIKELY(srv_flush_log_at_trx_commit == 0)) {
|
|
|
|
return(0);
|
|
}
|
|
|
|
trx_commit_complete_for_mysql(trx);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
#endif
|
|
|
|
/*********************************************************************
|
|
Rolls back a transaction or the latest SQL statement. */
|
|
static
|
|
int
|
|
innobase_rollback(
|
|
/*==============*/
|
|
/* out: 0 or error number */
|
|
handlerton *hton, /* in: Innodb handlerton */
|
|
THD* thd, /* in: handle to the MySQL thread of the user
|
|
whose transaction should be rolled back */
|
|
bool all) /* in: TRUE - commit transaction
|
|
FALSE - the current SQL statement ended */
|
|
{
|
|
int error = 0;
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("innobase_rollback");
|
|
DBUG_PRINT("trans", ("aborting transaction"));
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* Update the info whether we should skip XA steps that eat CPU time */
|
|
trx->support_xa = THDVAR(thd, support_xa);
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we will
|
|
reserve the kernel mutex, we have to release the search system latch
|
|
first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
if (trx->auto_inc_lock) {
|
|
/* If we had reserved the auto-inc lock for some table (if
|
|
we come here to roll back the latest SQL statement) we
|
|
release it now before a possibly lengthy rollback */
|
|
|
|
row_unlock_table_autoinc_for_mysql(trx);
|
|
}
|
|
|
|
if (all
|
|
|| !thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
|
|
error = trx_rollback_for_mysql(trx);
|
|
trx->active_trans = 0;
|
|
} else {
|
|
error = trx_rollback_last_sql_stat_for_mysql(trx);
|
|
}
|
|
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, NULL));
|
|
}
|
|
|
|
/*********************************************************************
|
|
Rolls back a transaction */
|
|
static
|
|
int
|
|
innobase_rollback_trx(
|
|
/*==================*/
|
|
/* out: 0 or error number */
|
|
trx_t* trx) /* in: transaction */
|
|
{
|
|
int error = 0;
|
|
|
|
DBUG_ENTER("innobase_rollback_trx");
|
|
DBUG_PRINT("trans", ("aborting transaction"));
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we will
|
|
reserve the kernel mutex, we have to release the search system latch
|
|
first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
if (trx->auto_inc_lock) {
|
|
/* If we had reserved the auto-inc lock for some table (if
|
|
we come here to roll back the latest SQL statement) we
|
|
release it now before a possibly lengthy rollback */
|
|
|
|
row_unlock_table_autoinc_for_mysql(trx);
|
|
}
|
|
|
|
error = trx_rollback_for_mysql(trx);
|
|
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, NULL));
|
|
}
|
|
|
|
/*********************************************************************
|
|
Rolls back a transaction to a savepoint. */
|
|
static
|
|
int
|
|
innobase_rollback_to_savepoint(
|
|
/*===========================*/
|
|
/* out: 0 if success, HA_ERR_NO_SAVEPOINT if
|
|
no savepoint with the given name */
|
|
handlerton *hton, /* in: Innodb handlerton */
|
|
THD* thd, /* in: handle to the MySQL thread of the user
|
|
whose transaction should be rolled back */
|
|
void* savepoint) /* in: savepoint data */
|
|
{
|
|
ib_longlong mysql_binlog_cache_pos;
|
|
int error = 0;
|
|
trx_t* trx;
|
|
char name[64];
|
|
|
|
DBUG_ENTER("innobase_rollback_to_savepoint");
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we will
|
|
reserve the kernel mutex, we have to release the search system latch
|
|
first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
/* TODO: use provided savepoint data area to store savepoint data */
|
|
|
|
longlong2str((ulint)savepoint, name, 36);
|
|
|
|
error = (int) trx_rollback_to_savepoint_for_mysql(trx, name,
|
|
&mysql_binlog_cache_pos);
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, NULL));
|
|
}
|
|
|
|
/*********************************************************************
|
|
Release transaction savepoint name. */
|
|
static
|
|
int
|
|
innobase_release_savepoint(
|
|
/*=======================*/
|
|
/* out: 0 if success, HA_ERR_NO_SAVEPOINT if
|
|
no savepoint with the given name */
|
|
handlerton* hton, /* in: handlerton for Innodb */
|
|
THD* thd, /* in: handle to the MySQL thread of the user
|
|
whose transaction should be rolled back */
|
|
void* savepoint) /* in: savepoint data */
|
|
{
|
|
int error = 0;
|
|
trx_t* trx;
|
|
char name[64];
|
|
|
|
DBUG_ENTER("innobase_release_savepoint");
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* TODO: use provided savepoint data area to store savepoint data */
|
|
|
|
longlong2str((ulint)savepoint, name, 36);
|
|
|
|
error = (int) trx_release_savepoint_for_mysql(trx, name);
|
|
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, NULL));
|
|
}
|
|
|
|
/*********************************************************************
|
|
Sets a transaction savepoint. */
|
|
static
|
|
int
|
|
innobase_savepoint(
|
|
/*===============*/
|
|
/* out: always 0, that is, always succeeds */
|
|
handlerton* hton, /* in: handle to the Innodb handlerton */
|
|
THD* thd, /* in: handle to the MySQL thread */
|
|
void* savepoint) /* in: savepoint data */
|
|
{
|
|
int error = 0;
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("innobase_savepoint");
|
|
|
|
/*
|
|
In the autocommit mode there is no sense to set a savepoint
|
|
(unless we are in sub-statement), so SQL layer ensures that
|
|
this method is never called in such situation.
|
|
*/
|
|
#ifdef MYSQL_SERVER /* plugins cannot access thd->in_sub_stmt */
|
|
DBUG_ASSERT(thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN) ||
|
|
thd->in_sub_stmt);
|
|
#endif /* MYSQL_SERVER */
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we will
|
|
reserve the kernel mutex, we have to release the search system latch
|
|
first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
/* cannot happen outside of transaction */
|
|
DBUG_ASSERT(trx->active_trans);
|
|
|
|
/* TODO: use provided savepoint data area to store savepoint data */
|
|
char name[64];
|
|
longlong2str((ulint)savepoint,name,36);
|
|
|
|
error = (int) trx_savepoint_for_mysql(trx, name, (ib_longlong)0);
|
|
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, NULL));
|
|
}
|
|
|
|
/*********************************************************************
|
|
Frees a possible InnoDB trx object associated with the current THD. */
|
|
static
|
|
int
|
|
innobase_close_connection(
|
|
/*======================*/
|
|
/* out: 0 or error number */
|
|
handlerton* hton, /* in: innobase handlerton */
|
|
THD* thd) /* in: handle to the MySQL thread of the user
|
|
whose resources should be free'd */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("innobase_close_connection");
|
|
DBUG_ASSERT(hton == innodb_hton_ptr);
|
|
trx = thd_to_trx(thd);
|
|
|
|
ut_a(trx);
|
|
|
|
if (trx->active_trans == 0
|
|
&& trx->conc_state != TRX_NOT_STARTED) {
|
|
|
|
sql_print_error("trx->active_trans == 0, but"
|
|
" trx->conc_state != TRX_NOT_STARTED");
|
|
}
|
|
|
|
|
|
if (trx->conc_state != TRX_NOT_STARTED &&
|
|
global_system_variables.log_warnings) {
|
|
sql_print_warning(
|
|
"MySQL is closing a connection that has an active "
|
|
"InnoDB transaction. %lu row modifications will "
|
|
"roll back.",
|
|
(ulong) trx->undo_no.low);
|
|
}
|
|
|
|
innobase_rollback_trx(trx);
|
|
|
|
thr_local_free(trx->mysql_thread_id);
|
|
trx_free_for_mysql(trx);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
** InnoDB database tables
|
|
*****************************************************************************/
|
|
|
|
/********************************************************************
|
|
Get the record format from the data dictionary. */
|
|
enum row_type
|
|
ha_innobase::get_row_type() const
|
|
/*=============================*/
|
|
/* out: ROW_TYPE_REDUNDANT or ROW_TYPE_COMPACT */
|
|
{
|
|
if (prebuilt && prebuilt->table) {
|
|
if (dict_table_is_comp_noninline(prebuilt->table)) {
|
|
return(ROW_TYPE_COMPACT);
|
|
} else {
|
|
return(ROW_TYPE_REDUNDANT);
|
|
}
|
|
}
|
|
ut_ad(0);
|
|
return(ROW_TYPE_NOT_USED);
|
|
}
|
|
|
|
/********************************************************************
|
|
Gives the file extension of an InnoDB single-table tablespace. */
|
|
static const char* ha_innobase_exts[] = {
|
|
".ibd",
|
|
NullS
|
|
};
|
|
|
|
const char**
|
|
ha_innobase::bas_ext() const
|
|
/*========================*/
|
|
/* out: file extension string */
|
|
{
|
|
return(ha_innobase_exts);
|
|
}
|
|
|
|
|
|
/*********************************************************************
|
|
Normalizes a table name string. A normalized name consists of the
|
|
database name catenated to '/' and table name. An example:
|
|
test/mytable. On Windows normalization puts both the database name and the
|
|
table name always to lower case. */
|
|
static
|
|
void
|
|
normalize_table_name(
|
|
/*=================*/
|
|
char* norm_name, /* out: normalized name as a
|
|
null-terminated string */
|
|
const char* name) /* in: table name string */
|
|
{
|
|
char* name_ptr;
|
|
char* db_ptr;
|
|
char* ptr;
|
|
|
|
/* Scan name from the end */
|
|
|
|
ptr = strend(name)-1;
|
|
|
|
while (ptr >= name && *ptr != '\\' && *ptr != '/') {
|
|
ptr--;
|
|
}
|
|
|
|
name_ptr = ptr + 1;
|
|
|
|
DBUG_ASSERT(ptr > name);
|
|
|
|
ptr--;
|
|
|
|
while (ptr >= name && *ptr != '\\' && *ptr != '/') {
|
|
ptr--;
|
|
}
|
|
|
|
db_ptr = ptr + 1;
|
|
|
|
memcpy(norm_name, db_ptr, strlen(name) + 1 - (db_ptr - name));
|
|
|
|
norm_name[name_ptr - db_ptr - 1] = '/';
|
|
|
|
#ifdef __WIN__
|
|
innobase_casedn_str(norm_name);
|
|
#endif
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates and opens a handle to a table which already exists in an InnoDB
|
|
database. */
|
|
|
|
int
|
|
ha_innobase::open(
|
|
/*==============*/
|
|
/* out: 1 if error, 0 if success */
|
|
const char* name, /* in: table name */
|
|
int mode, /* in: not used */
|
|
uint test_if_locked) /* in: not used */
|
|
{
|
|
dict_table_t* ib_table;
|
|
char norm_name[1000];
|
|
THD* thd;
|
|
|
|
DBUG_ENTER("ha_innobase::open");
|
|
|
|
UT_NOT_USED(mode);
|
|
UT_NOT_USED(test_if_locked);
|
|
|
|
thd = ha_thd();
|
|
normalize_table_name(norm_name, name);
|
|
|
|
user_thd = NULL;
|
|
|
|
if (!(share=get_share(name))) {
|
|
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
/* Create buffers for packing the fields of a record. Why
|
|
table->reclength did not work here? Obviously, because char
|
|
fields when packed actually became 1 byte longer, when we also
|
|
stored the string length as the first byte. */
|
|
|
|
upd_and_key_val_buff_len =
|
|
table->s->reclength + table->s->max_key_length
|
|
+ MAX_REF_PARTS * 3;
|
|
if (!(mysql_byte*) my_multi_malloc(MYF(MY_WME),
|
|
&upd_buff, upd_and_key_val_buff_len,
|
|
&key_val_buff, upd_and_key_val_buff_len,
|
|
NullS)) {
|
|
free_share(share);
|
|
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
/* Get pointer to a table object in InnoDB dictionary cache */
|
|
|
|
ib_table = dict_table_get(norm_name, TRUE);
|
|
|
|
if (NULL == ib_table) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("Cannot find table %s from the internal data "
|
|
"dictionary\nof InnoDB though the .frm file "
|
|
"for the table exists. Maybe you\nhave "
|
|
"deleted and recreated InnoDB data files but "
|
|
"have forgotten\nto delete the corresponding "
|
|
".frm files of InnoDB tables, or you\n"
|
|
"have moved .frm files to another database?\n"
|
|
"See http://dev.mysql.com/doc/refman/5.1/en/innodb-troubleshooting.html\n"
|
|
"how you can resolve the problem.\n",
|
|
norm_name);
|
|
free_share(share);
|
|
my_free((gptr) upd_buff, MYF(0));
|
|
my_errno = ENOENT;
|
|
|
|
DBUG_RETURN(HA_ERR_NO_SUCH_TABLE);
|
|
}
|
|
|
|
if (ib_table->ibd_file_missing && !thd_tablespace_op(thd)) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("MySQL is trying to open a table handle but "
|
|
"the .ibd file for\ntable %s does not exist.\n"
|
|
"Have you deleted the .ibd file from the "
|
|
"database directory under\nthe MySQL datadir, "
|
|
"or have you used DISCARD TABLESPACE?\n"
|
|
"See http://dev.mysql.com/doc/refman/5.1/en/innodb-troubleshooting.html\n"
|
|
"how you can resolve the problem.\n",
|
|
norm_name);
|
|
free_share(share);
|
|
my_free((gptr) upd_buff, MYF(0));
|
|
my_errno = ENOENT;
|
|
|
|
dict_table_decrement_handle_count(ib_table);
|
|
DBUG_RETURN(HA_ERR_NO_SUCH_TABLE);
|
|
}
|
|
|
|
prebuilt = row_create_prebuilt(ib_table);
|
|
|
|
prebuilt->mysql_row_len = table->s->reclength;
|
|
|
|
/* Looks like MySQL-3.23 sometimes has primary key number != 0 */
|
|
|
|
primary_key = table->s->primary_key;
|
|
key_used_on_scan = primary_key;
|
|
|
|
/* Allocate a buffer for a 'row reference'. A row reference is
|
|
a string of bytes of length ref_length which uniquely specifies
|
|
a row in our table. Note that MySQL may also compare two row
|
|
references for equality by doing a simple memcmp on the strings
|
|
of length ref_length! */
|
|
|
|
if (!row_table_got_default_clust_index(ib_table)) {
|
|
if (primary_key >= MAX_KEY) {
|
|
sql_print_error("Table %s has a primary key in InnoDB data "
|
|
"dictionary, but not in MySQL!", name);
|
|
}
|
|
|
|
prebuilt->clust_index_was_generated = FALSE;
|
|
|
|
/* MySQL allocates the buffer for ref. key_info->key_length
|
|
includes space for all key columns + one byte for each column
|
|
that may be NULL. ref_length must be as exact as possible to
|
|
save space, because all row reference buffers are allocated
|
|
based on ref_length. */
|
|
|
|
ref_length = table->key_info[primary_key].key_length;
|
|
} else {
|
|
if (primary_key != MAX_KEY) {
|
|
sql_print_error("Table %s has no primary key in InnoDB data "
|
|
"dictionary, but has one in MySQL! If you "
|
|
"created the table with a MySQL version < "
|
|
"3.23.54 and did not define a primary key, "
|
|
"but defined a unique key with all non-NULL "
|
|
"columns, then MySQL internally treats that "
|
|
"key as the primary key. You can fix this "
|
|
"error by dump + DROP + CREATE + reimport "
|
|
"of the table.", name);
|
|
}
|
|
|
|
prebuilt->clust_index_was_generated = TRUE;
|
|
|
|
ref_length = DATA_ROW_ID_LEN;
|
|
|
|
/* If we automatically created the clustered index, then
|
|
MySQL does not know about it, and MySQL must NOT be aware
|
|
of the index used on scan, to make it avoid checking if we
|
|
update the column of the index. That is why we assert below
|
|
that key_used_on_scan is the undefined value MAX_KEY.
|
|
The column is the row id in the automatical generation case,
|
|
and it will never be updated anyway. */
|
|
|
|
if (key_used_on_scan != MAX_KEY) {
|
|
sql_print_warning(
|
|
"Table %s key_used_on_scan is %lu even "
|
|
"though there is no primary key inside "
|
|
"InnoDB.", name, (ulong) key_used_on_scan);
|
|
}
|
|
}
|
|
|
|
stats.block_size = 16 * 1024; /* Index block size in InnoDB: used by MySQL
|
|
in query optimization */
|
|
|
|
/* Init table lock structure */
|
|
thr_lock_data_init(&share->lock,&lock,(void*) 0);
|
|
|
|
info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
uint
|
|
ha_innobase::max_supported_key_part_length() const
|
|
{
|
|
return(DICT_MAX_INDEX_COL_LEN - 1);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Closes a handle to an InnoDB table. */
|
|
|
|
int
|
|
ha_innobase::close(void)
|
|
/*====================*/
|
|
/* out: 0 */
|
|
{
|
|
THD* thd;
|
|
|
|
DBUG_ENTER("ha_innobase::close");
|
|
|
|
thd = current_thd; // avoid calling current_thd twice, it may be slow
|
|
if (thd != NULL) {
|
|
innobase_release_temporary_latches(ht, thd);
|
|
}
|
|
|
|
row_prebuilt_free(prebuilt);
|
|
|
|
my_free((gptr) upd_buff, MYF(0));
|
|
free_share(share);
|
|
|
|
/* Tell InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* The following accessor functions should really be inside MySQL code! */
|
|
|
|
/******************************************************************
|
|
Gets field offset for a field in a table. */
|
|
inline
|
|
uint
|
|
get_field_offset(
|
|
/*=============*/
|
|
/* out: offset */
|
|
TABLE* table, /* in: MySQL table object */
|
|
Field* field) /* in: MySQL field object */
|
|
{
|
|
return((uint) (field->ptr - (char*) table->record[0]));
|
|
}
|
|
|
|
/******************************************************************
|
|
Checks if a field in a record is SQL NULL. Uses the record format
|
|
information in table to track the null bit in record. */
|
|
static inline
|
|
uint
|
|
field_in_record_is_null(
|
|
/*====================*/
|
|
/* out: 1 if NULL, 0 otherwise */
|
|
TABLE* table, /* in: MySQL table object */
|
|
Field* field, /* in: MySQL field object */
|
|
char* record) /* in: a row in MySQL format */
|
|
{
|
|
int null_offset;
|
|
|
|
if (!field->null_ptr) {
|
|
|
|
return(0);
|
|
}
|
|
|
|
null_offset = (uint) ((char*) field->null_ptr
|
|
- (char*) table->record[0]);
|
|
|
|
if (record[null_offset] & field->null_bit) {
|
|
|
|
return(1);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/******************************************************************
|
|
Sets a field in a record to SQL NULL. Uses the record format
|
|
information in table to track the null bit in record. */
|
|
inline
|
|
void
|
|
set_field_in_record_to_null(
|
|
/*========================*/
|
|
TABLE* table, /* in: MySQL table object */
|
|
Field* field, /* in: MySQL field object */
|
|
char* record) /* in: a row in MySQL format */
|
|
{
|
|
int null_offset;
|
|
|
|
null_offset = (uint) ((char*) field->null_ptr
|
|
- (char*) table->record[0]);
|
|
|
|
record[null_offset] = record[null_offset] | field->null_bit;
|
|
}
|
|
|
|
extern "C" {
|
|
/*****************************************************************
|
|
InnoDB uses this function to compare two data fields for which the data type
|
|
is such that we must use MySQL code to compare them. NOTE that the prototype
|
|
of this function is in rem0cmp.c in InnoDB source code! If you change this
|
|
function, remember to update the prototype there! */
|
|
|
|
int
|
|
innobase_mysql_cmp(
|
|
/*===============*/
|
|
/* out: 1, 0, -1, if a is greater,
|
|
equal, less than b, respectively */
|
|
int mysql_type, /* in: MySQL type */
|
|
uint charset_number, /* in: number of the charset */
|
|
const unsigned char* a, /* in: data field */
|
|
unsigned int a_length, /* in: data field length,
|
|
not UNIV_SQL_NULL */
|
|
const unsigned char* b, /* in: data field */
|
|
unsigned int b_length) /* in: data field length,
|
|
not UNIV_SQL_NULL */
|
|
{
|
|
CHARSET_INFO* charset;
|
|
enum_field_types mysql_tp;
|
|
int ret;
|
|
|
|
DBUG_ASSERT(a_length != UNIV_SQL_NULL);
|
|
DBUG_ASSERT(b_length != UNIV_SQL_NULL);
|
|
|
|
mysql_tp = (enum_field_types) mysql_type;
|
|
|
|
switch (mysql_tp) {
|
|
|
|
case MYSQL_TYPE_BIT:
|
|
case MYSQL_TYPE_STRING:
|
|
case MYSQL_TYPE_VAR_STRING:
|
|
case MYSQL_TYPE_TINY_BLOB:
|
|
case MYSQL_TYPE_MEDIUM_BLOB:
|
|
case MYSQL_TYPE_BLOB:
|
|
case MYSQL_TYPE_LONG_BLOB:
|
|
case MYSQL_TYPE_VARCHAR:
|
|
/* Use the charset number to pick the right charset struct for
|
|
the comparison. Since the MySQL function get_charset may be
|
|
slow before Bar removes the mutex operation there, we first
|
|
look at 2 common charsets directly. */
|
|
|
|
if (charset_number == default_charset_info->number) {
|
|
charset = default_charset_info;
|
|
} else if (charset_number == my_charset_latin1.number) {
|
|
charset = &my_charset_latin1;
|
|
} else {
|
|
charset = get_charset(charset_number, MYF(MY_WME));
|
|
|
|
if (charset == NULL) {
|
|
sql_print_error("InnoDB needs charset %lu for doing "
|
|
"a comparison, but MySQL cannot "
|
|
"find that charset.",
|
|
(ulong) charset_number);
|
|
ut_a(0);
|
|
}
|
|
}
|
|
|
|
/* Starting from 4.1.3, we use strnncollsp() in comparisons of
|
|
non-latin1_swedish_ci strings. NOTE that the collation order
|
|
changes then: 'b\0\0...' is ordered BEFORE 'b ...'. Users
|
|
having indexes on such data need to rebuild their tables! */
|
|
|
|
ret = charset->coll->strnncollsp(charset,
|
|
a, a_length,
|
|
b, b_length, 0);
|
|
if (ret < 0) {
|
|
return(-1);
|
|
} else if (ret > 0) {
|
|
return(1);
|
|
} else {
|
|
return(0);
|
|
}
|
|
default:
|
|
ut_error;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
}
|
|
|
|
/******************************************************************
|
|
Converts a MySQL type to an InnoDB type. Note that this function returns
|
|
the 'mtype' of InnoDB. InnoDB differentiates between MySQL's old <= 4.1
|
|
VARCHAR and the new true VARCHAR in >= 5.0.3 by the 'prtype'. */
|
|
inline
|
|
ulint
|
|
get_innobase_type_from_mysql_type(
|
|
/*==============================*/
|
|
/* out: DATA_BINARY, DATA_VARCHAR, ... */
|
|
ulint* unsigned_flag, /* out: DATA_UNSIGNED if an 'unsigned type';
|
|
at least ENUM and SET, and unsigned integer
|
|
types are 'unsigned types' */
|
|
Field* field) /* in: MySQL field */
|
|
{
|
|
/* The following asserts try to check that the MySQL type code fits in
|
|
8 bits: this is used in ibuf and also when DATA_NOT_NULL is ORed to
|
|
the type */
|
|
|
|
DBUG_ASSERT((ulint)MYSQL_TYPE_STRING < 256);
|
|
DBUG_ASSERT((ulint)MYSQL_TYPE_VAR_STRING < 256);
|
|
DBUG_ASSERT((ulint)MYSQL_TYPE_DOUBLE < 256);
|
|
DBUG_ASSERT((ulint)MYSQL_TYPE_FLOAT < 256);
|
|
DBUG_ASSERT((ulint)MYSQL_TYPE_DECIMAL < 256);
|
|
|
|
if (field->flags & UNSIGNED_FLAG) {
|
|
|
|
*unsigned_flag = DATA_UNSIGNED;
|
|
} else {
|
|
*unsigned_flag = 0;
|
|
}
|
|
|
|
if (field->real_type() == MYSQL_TYPE_ENUM
|
|
|| field->real_type() == MYSQL_TYPE_SET) {
|
|
|
|
/* MySQL has field->type() a string type for these, but the
|
|
data is actually internally stored as an unsigned integer
|
|
code! */
|
|
|
|
*unsigned_flag = DATA_UNSIGNED; /* MySQL has its own unsigned
|
|
flag set to zero, even though
|
|
internally this is an unsigned
|
|
integer type */
|
|
return(DATA_INT);
|
|
}
|
|
|
|
switch (field->type()) {
|
|
/* NOTE that we only allow string types in DATA_MYSQL and
|
|
DATA_VARMYSQL */
|
|
case MYSQL_TYPE_VAR_STRING: /* old <= 4.1 VARCHAR */
|
|
case MYSQL_TYPE_VARCHAR: /* new >= 5.0.3 true VARCHAR */
|
|
if (field->binary()) {
|
|
return(DATA_BINARY);
|
|
} else if (strcmp(
|
|
field->charset()->name,
|
|
"latin1_swedish_ci") == 0) {
|
|
return(DATA_VARCHAR);
|
|
} else {
|
|
return(DATA_VARMYSQL);
|
|
}
|
|
case MYSQL_TYPE_BIT:
|
|
case MYSQL_TYPE_STRING: if (field->binary()) {
|
|
|
|
return(DATA_FIXBINARY);
|
|
} else if (strcmp(
|
|
field->charset()->name,
|
|
"latin1_swedish_ci") == 0) {
|
|
return(DATA_CHAR);
|
|
} else {
|
|
return(DATA_MYSQL);
|
|
}
|
|
case MYSQL_TYPE_NEWDECIMAL:
|
|
return(DATA_FIXBINARY);
|
|
case MYSQL_TYPE_LONG:
|
|
case MYSQL_TYPE_LONGLONG:
|
|
case MYSQL_TYPE_TINY:
|
|
case MYSQL_TYPE_SHORT:
|
|
case MYSQL_TYPE_INT24:
|
|
case MYSQL_TYPE_DATE:
|
|
case MYSQL_TYPE_DATETIME:
|
|
case MYSQL_TYPE_YEAR:
|
|
case MYSQL_TYPE_NEWDATE:
|
|
case MYSQL_TYPE_TIME:
|
|
case MYSQL_TYPE_TIMESTAMP:
|
|
return(DATA_INT);
|
|
case MYSQL_TYPE_FLOAT:
|
|
return(DATA_FLOAT);
|
|
case MYSQL_TYPE_DOUBLE:
|
|
return(DATA_DOUBLE);
|
|
case MYSQL_TYPE_DECIMAL:
|
|
return(DATA_DECIMAL);
|
|
case MYSQL_TYPE_GEOMETRY:
|
|
case MYSQL_TYPE_TINY_BLOB:
|
|
case MYSQL_TYPE_MEDIUM_BLOB:
|
|
case MYSQL_TYPE_BLOB:
|
|
case MYSQL_TYPE_LONG_BLOB:
|
|
return(DATA_BLOB);
|
|
default:
|
|
ut_error;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Writes an unsigned integer value < 64k to 2 bytes, in the little-endian
|
|
storage format. */
|
|
inline
|
|
void
|
|
innobase_write_to_2_little_endian(
|
|
/*==============================*/
|
|
byte* buf, /* in: where to store */
|
|
ulint val) /* in: value to write, must be < 64k */
|
|
{
|
|
ut_a(val < 256 * 256);
|
|
|
|
buf[0] = (byte)(val & 0xFF);
|
|
buf[1] = (byte)(val / 256);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Reads an unsigned integer value < 64k from 2 bytes, in the little-endian
|
|
storage format. */
|
|
inline
|
|
uint
|
|
innobase_read_from_2_little_endian(
|
|
/*===============================*/
|
|
/* out: value */
|
|
const mysql_byte* buf) /* in: from where to read */
|
|
{
|
|
return (uint) ((ulint)(buf[0]) + 256 * ((ulint)(buf[1])));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Stores a key value for a row to a buffer. */
|
|
|
|
uint
|
|
ha_innobase::store_key_val_for_row(
|
|
/*===============================*/
|
|
/* out: key value length as stored in buff */
|
|
uint keynr, /* in: key number */
|
|
char* buff, /* in/out: buffer for the key value (in MySQL
|
|
format) */
|
|
uint buff_len,/* in: buffer length */
|
|
const mysql_byte* record)/* in: row in MySQL format */
|
|
{
|
|
KEY* key_info = table->key_info + keynr;
|
|
KEY_PART_INFO* key_part = key_info->key_part;
|
|
KEY_PART_INFO* end = key_part + key_info->key_parts;
|
|
char* buff_start = buff;
|
|
enum_field_types mysql_type;
|
|
Field* field;
|
|
ibool is_null;
|
|
|
|
DBUG_ENTER("store_key_val_for_row");
|
|
|
|
/* The format for storing a key field in MySQL is the following:
|
|
|
|
1. If the column can be NULL, then in the first byte we put 1 if the
|
|
field value is NULL, 0 otherwise.
|
|
|
|
2. If the column is of a BLOB type (it must be a column prefix field
|
|
in this case), then we put the length of the data in the field to the
|
|
next 2 bytes, in the little-endian format. If the field is SQL NULL,
|
|
then these 2 bytes are set to 0. Note that the length of data in the
|
|
field is <= column prefix length.
|
|
|
|
3. In a column prefix field, prefix_len next bytes are reserved for
|
|
data. In a normal field the max field length next bytes are reserved
|
|
for data. For a VARCHAR(n) the max field length is n. If the stored
|
|
value is the SQL NULL then these data bytes are set to 0.
|
|
|
|
4. We always use a 2 byte length for a true >= 5.0.3 VARCHAR. Note that
|
|
in the MySQL row format, the length is stored in 1 or 2 bytes,
|
|
depending on the maximum allowed length. But in the MySQL key value
|
|
format, the length always takes 2 bytes.
|
|
|
|
We have to zero-fill the buffer so that MySQL is able to use a
|
|
simple memcmp to compare two key values to determine if they are
|
|
equal. MySQL does this to compare contents of two 'ref' values. */
|
|
|
|
bzero(buff, buff_len);
|
|
|
|
for (; key_part != end; key_part++) {
|
|
is_null = FALSE;
|
|
|
|
if (key_part->null_bit) {
|
|
if (record[key_part->null_offset]
|
|
& key_part->null_bit) {
|
|
*buff = 1;
|
|
is_null = TRUE;
|
|
} else {
|
|
*buff = 0;
|
|
}
|
|
buff++;
|
|
}
|
|
|
|
field = key_part->field;
|
|
mysql_type = field->type();
|
|
|
|
if (mysql_type == MYSQL_TYPE_VARCHAR) {
|
|
/* >= 5.0.3 true VARCHAR */
|
|
ulint lenlen;
|
|
ulint len;
|
|
byte* data;
|
|
ulint key_len;
|
|
ulint true_len;
|
|
CHARSET_INFO* cs;
|
|
int error=0;
|
|
|
|
key_len = key_part->length;
|
|
|
|
if (is_null) {
|
|
buff += key_len + 2;
|
|
|
|
continue;
|
|
}
|
|
cs = field->charset();
|
|
|
|
lenlen = (ulint)
|
|
(((Field_varstring*)field)->length_bytes);
|
|
|
|
data = row_mysql_read_true_varchar(&len,
|
|
(byte*) (record
|
|
+ (ulint)get_field_offset(table, field)),
|
|
lenlen);
|
|
|
|
true_len = len;
|
|
|
|
/* For multi byte character sets we need to calculate
|
|
the true length of the key */
|
|
|
|
if (len > 0 && cs->mbmaxlen > 1) {
|
|
true_len = (ulint) cs->cset->well_formed_len(cs,
|
|
(const char *) data,
|
|
(const char *) data + len,
|
|
(uint) (key_len /
|
|
cs->mbmaxlen),
|
|
&error);
|
|
}
|
|
|
|
/* In a column prefix index, we may need to truncate
|
|
the stored value: */
|
|
|
|
if (true_len > key_len) {
|
|
true_len = key_len;
|
|
}
|
|
|
|
/* The length in a key value is always stored in 2
|
|
bytes */
|
|
|
|
row_mysql_store_true_var_len((byte*)buff, true_len, 2);
|
|
buff += 2;
|
|
|
|
memcpy(buff, data, true_len);
|
|
|
|
/* Note that we always reserve the maximum possible
|
|
length of the true VARCHAR in the key value, though
|
|
only len first bytes after the 2 length bytes contain
|
|
actual data. The rest of the space was reset to zero
|
|
in the bzero() call above. */
|
|
|
|
buff += key_len;
|
|
|
|
} else if (mysql_type == MYSQL_TYPE_TINY_BLOB
|
|
|| mysql_type == MYSQL_TYPE_MEDIUM_BLOB
|
|
|| mysql_type == MYSQL_TYPE_BLOB
|
|
|| mysql_type == MYSQL_TYPE_LONG_BLOB) {
|
|
|
|
CHARSET_INFO* cs;
|
|
ulint key_len;
|
|
ulint true_len;
|
|
int error=0;
|
|
ulint blob_len;
|
|
byte* blob_data;
|
|
|
|
ut_a(key_part->key_part_flag & HA_PART_KEY_SEG);
|
|
|
|
key_len = key_part->length;
|
|
|
|
if (is_null) {
|
|
buff += key_len + 2;
|
|
|
|
continue;
|
|
}
|
|
|
|
cs = field->charset();
|
|
|
|
blob_data = row_mysql_read_blob_ref(&blob_len,
|
|
(byte*) (record
|
|
+ (ulint)get_field_offset(table, field)),
|
|
(ulint) field->pack_length());
|
|
|
|
true_len = blob_len;
|
|
|
|
ut_a(get_field_offset(table, field)
|
|
== key_part->offset);
|
|
|
|
/* For multi byte character sets we need to calculate
|
|
the true length of the key */
|
|
|
|
if (blob_len > 0 && cs->mbmaxlen > 1) {
|
|
true_len = (ulint) cs->cset->well_formed_len(cs,
|
|
(const char *) blob_data,
|
|
(const char *) blob_data
|
|
+ blob_len,
|
|
(uint) (key_len /
|
|
cs->mbmaxlen),
|
|
&error);
|
|
}
|
|
|
|
/* All indexes on BLOB and TEXT are column prefix
|
|
indexes, and we may need to truncate the data to be
|
|
stored in the key value: */
|
|
|
|
if (true_len > key_len) {
|
|
true_len = key_len;
|
|
}
|
|
|
|
/* MySQL reserves 2 bytes for the length and the
|
|
storage of the number is little-endian */
|
|
|
|
innobase_write_to_2_little_endian(
|
|
(byte*)buff, true_len);
|
|
buff += 2;
|
|
|
|
memcpy(buff, blob_data, true_len);
|
|
|
|
/* Note that we always reserve the maximum possible
|
|
length of the BLOB prefix in the key value. */
|
|
|
|
buff += key_len;
|
|
} else {
|
|
/* Here we handle all other data types except the
|
|
true VARCHAR, BLOB and TEXT. Note that the column
|
|
value we store may be also in a column prefix
|
|
index. */
|
|
|
|
CHARSET_INFO* cs;
|
|
ulint true_len;
|
|
ulint key_len;
|
|
const mysql_byte* src_start;
|
|
int error=0;
|
|
enum_field_types real_type;
|
|
|
|
key_len = key_part->length;
|
|
|
|
if (is_null) {
|
|
buff += key_len;
|
|
|
|
continue;
|
|
}
|
|
|
|
src_start = record + key_part->offset;
|
|
real_type = field->real_type();
|
|
true_len = key_len;
|
|
|
|
/* Character set for the field is defined only
|
|
to fields whose type is string and real field
|
|
type is not enum or set. For these fields check
|
|
if character set is multi byte. */
|
|
|
|
if (real_type != MYSQL_TYPE_ENUM
|
|
&& real_type != MYSQL_TYPE_SET
|
|
&& ( mysql_type == MYSQL_TYPE_VAR_STRING
|
|
|| mysql_type == MYSQL_TYPE_STRING)) {
|
|
|
|
cs = field->charset();
|
|
|
|
/* For multi byte character sets we need to
|
|
calculate the true length of the key */
|
|
|
|
if (key_len > 0 && cs->mbmaxlen > 1) {
|
|
|
|
true_len = (ulint)
|
|
cs->cset->well_formed_len(cs,
|
|
(const char *)src_start,
|
|
(const char *)src_start
|
|
+ key_len,
|
|
(uint) (key_len /
|
|
cs->mbmaxlen),
|
|
&error);
|
|
}
|
|
}
|
|
|
|
memcpy(buff, src_start, true_len);
|
|
buff += true_len;
|
|
|
|
/* Pad the unused space with spaces. Note that no
|
|
padding is ever needed for UCS-2 because in MySQL,
|
|
all UCS2 characters are 2 bytes, as MySQL does not
|
|
support surrogate pairs, which are needed to represent
|
|
characters in the range U+10000 to U+10FFFF. */
|
|
|
|
if (true_len < key_len) {
|
|
ulint pad_len = key_len - true_len;
|
|
memset(buff, ' ', pad_len);
|
|
buff += pad_len;
|
|
}
|
|
}
|
|
}
|
|
|
|
ut_a(buff <= buff_start + buff_len);
|
|
|
|
DBUG_RETURN((uint)(buff - buff_start));
|
|
}
|
|
|
|
/******************************************************************
|
|
Builds a 'template' to the prebuilt struct. The template is used in fast
|
|
retrieval of just those column values MySQL needs in its processing. */
|
|
static
|
|
void
|
|
build_template(
|
|
/*===========*/
|
|
row_prebuilt_t* prebuilt, /* in/out: prebuilt struct */
|
|
THD* thd, /* in: current user thread, used
|
|
only if templ_type is
|
|
ROW_MYSQL_REC_FIELDS */
|
|
TABLE* table, /* in: MySQL table */
|
|
uint templ_type) /* in: ROW_MYSQL_WHOLE_ROW or
|
|
ROW_MYSQL_REC_FIELDS */
|
|
{
|
|
dict_index_t* index;
|
|
dict_index_t* clust_index;
|
|
mysql_row_templ_t* templ;
|
|
Field* field;
|
|
ulint n_fields;
|
|
ulint n_requested_fields = 0;
|
|
ibool fetch_all_in_key = FALSE;
|
|
ibool fetch_primary_key_cols = FALSE;
|
|
ulint i;
|
|
/* byte offset of the end of last requested column */
|
|
ulint mysql_prefix_len = 0;
|
|
|
|
if (prebuilt->select_lock_type == LOCK_X) {
|
|
/* We always retrieve the whole clustered index record if we
|
|
use exclusive row level locks, for example, if the read is
|
|
done in an UPDATE statement. */
|
|
|
|
templ_type = ROW_MYSQL_WHOLE_ROW;
|
|
}
|
|
|
|
if (templ_type == ROW_MYSQL_REC_FIELDS) {
|
|
if (prebuilt->hint_need_to_fetch_extra_cols
|
|
== ROW_RETRIEVE_ALL_COLS) {
|
|
|
|
/* We know we must at least fetch all columns in the
|
|
key, or all columns in the table */
|
|
|
|
if (prebuilt->read_just_key) {
|
|
/* MySQL has instructed us that it is enough
|
|
to fetch the columns in the key; looks like
|
|
MySQL can set this flag also when there is
|
|
only a prefix of the column in the key: in
|
|
that case we retrieve the whole column from
|
|
the clustered index */
|
|
|
|
fetch_all_in_key = TRUE;
|
|
} else {
|
|
templ_type = ROW_MYSQL_WHOLE_ROW;
|
|
}
|
|
} else if (prebuilt->hint_need_to_fetch_extra_cols
|
|
== ROW_RETRIEVE_PRIMARY_KEY) {
|
|
/* We must at least fetch all primary key cols. Note
|
|
that if the clustered index was internally generated
|
|
by InnoDB on the row id (no primary key was
|
|
defined), then row_search_for_mysql() will always
|
|
retrieve the row id to a special buffer in the
|
|
prebuilt struct. */
|
|
|
|
fetch_primary_key_cols = TRUE;
|
|
}
|
|
}
|
|
|
|
clust_index = dict_table_get_first_index_noninline(prebuilt->table);
|
|
|
|
if (templ_type == ROW_MYSQL_REC_FIELDS) {
|
|
index = prebuilt->index;
|
|
} else {
|
|
index = clust_index;
|
|
}
|
|
|
|
if (index == clust_index) {
|
|
prebuilt->need_to_access_clustered = TRUE;
|
|
} else {
|
|
prebuilt->need_to_access_clustered = FALSE;
|
|
/* Below we check column by column if we need to access
|
|
the clustered index */
|
|
}
|
|
|
|
n_fields = (ulint)table->s->fields; /* number of columns */
|
|
|
|
if (!prebuilt->mysql_template) {
|
|
prebuilt->mysql_template = (mysql_row_templ_t*)
|
|
mem_alloc_noninline(
|
|
n_fields * sizeof(mysql_row_templ_t));
|
|
}
|
|
|
|
prebuilt->template_type = templ_type;
|
|
prebuilt->null_bitmap_len = table->s->null_bytes;
|
|
|
|
prebuilt->templ_contains_blob = FALSE;
|
|
|
|
/* Note that in InnoDB, i is the column number. MySQL calls columns
|
|
'fields'. */
|
|
for (i = 0; i < n_fields; i++) {
|
|
templ = prebuilt->mysql_template + n_requested_fields;
|
|
field = table->field[i];
|
|
|
|
if (UNIV_LIKELY(templ_type == ROW_MYSQL_REC_FIELDS)) {
|
|
/* Decide which columns we should fetch
|
|
and which we can skip. */
|
|
register const ibool index_contains_field =
|
|
dict_index_contains_col_or_prefix(index, i);
|
|
|
|
if (!index_contains_field && prebuilt->read_just_key) {
|
|
/* If this is a 'key read', we do not need
|
|
columns that are not in the key */
|
|
|
|
goto skip_field;
|
|
}
|
|
|
|
if (index_contains_field && fetch_all_in_key) {
|
|
/* This field is needed in the query */
|
|
|
|
goto include_field;
|
|
}
|
|
|
|
if (bitmap_is_set(table->read_set, i) ||
|
|
bitmap_is_set(table->write_set, i)) {
|
|
/* This field is needed in the query */
|
|
|
|
goto include_field;
|
|
}
|
|
|
|
if (fetch_primary_key_cols
|
|
&& dict_table_col_in_clustered_key(
|
|
index->table, i)) {
|
|
/* This field is needed in the query */
|
|
|
|
goto include_field;
|
|
}
|
|
|
|
/* This field is not needed in the query, skip it */
|
|
|
|
goto skip_field;
|
|
}
|
|
include_field:
|
|
n_requested_fields++;
|
|
|
|
templ->col_no = i;
|
|
|
|
if (index == clust_index) {
|
|
templ->rec_field_no = dict_col_get_clust_pos_noninline(
|
|
&index->table->cols[i], index);
|
|
} else {
|
|
templ->rec_field_no = dict_index_get_nth_col_pos(
|
|
index, i);
|
|
}
|
|
|
|
if (templ->rec_field_no == ULINT_UNDEFINED) {
|
|
prebuilt->need_to_access_clustered = TRUE;
|
|
}
|
|
|
|
if (field->null_ptr) {
|
|
templ->mysql_null_byte_offset =
|
|
(ulint) ((char*) field->null_ptr
|
|
- (char*) table->record[0]);
|
|
|
|
templ->mysql_null_bit_mask = (ulint) field->null_bit;
|
|
} else {
|
|
templ->mysql_null_bit_mask = 0;
|
|
}
|
|
|
|
templ->mysql_col_offset = (ulint)
|
|
get_field_offset(table, field);
|
|
|
|
templ->mysql_col_len = (ulint) field->pack_length();
|
|
if (mysql_prefix_len < templ->mysql_col_offset
|
|
+ templ->mysql_col_len) {
|
|
mysql_prefix_len = templ->mysql_col_offset
|
|
+ templ->mysql_col_len;
|
|
}
|
|
templ->type = index->table->cols[i].mtype;
|
|
templ->mysql_type = (ulint)field->type();
|
|
|
|
if (templ->mysql_type == DATA_MYSQL_TRUE_VARCHAR) {
|
|
templ->mysql_length_bytes = (ulint)
|
|
(((Field_varstring*)field)->length_bytes);
|
|
}
|
|
|
|
templ->charset = dtype_get_charset_coll_noninline(
|
|
index->table->cols[i].prtype);
|
|
templ->mbminlen = index->table->cols[i].mbminlen;
|
|
templ->mbmaxlen = index->table->cols[i].mbmaxlen;
|
|
templ->is_unsigned = index->table->cols[i].prtype
|
|
& DATA_UNSIGNED;
|
|
if (templ->type == DATA_BLOB) {
|
|
prebuilt->templ_contains_blob = TRUE;
|
|
}
|
|
skip_field:
|
|
;
|
|
}
|
|
|
|
prebuilt->n_template = n_requested_fields;
|
|
prebuilt->mysql_prefix_len = mysql_prefix_len;
|
|
|
|
if (index != clust_index && prebuilt->need_to_access_clustered) {
|
|
/* Change rec_field_no's to correspond to the clustered index
|
|
record */
|
|
for (i = 0; i < n_requested_fields; i++) {
|
|
templ = prebuilt->mysql_template + i;
|
|
|
|
templ->rec_field_no = dict_col_get_clust_pos_noninline(
|
|
&index->table->cols[templ->col_no],
|
|
clust_index);
|
|
}
|
|
}
|
|
}
|
|
|
|
/************************************************************************
|
|
Stores a row in an InnoDB database, to the table specified in this
|
|
handle. */
|
|
|
|
int
|
|
ha_innobase::write_row(
|
|
/*===================*/
|
|
/* out: error code */
|
|
mysql_byte* record) /* in: a row in MySQL format */
|
|
{
|
|
int error;
|
|
longlong auto_inc;
|
|
longlong dummy;
|
|
ibool auto_inc_used= FALSE;
|
|
ulint sql_command;
|
|
trx_t* trx = thd_to_trx(user_thd);
|
|
|
|
DBUG_ENTER("ha_innobase::write_row");
|
|
|
|
if (prebuilt->trx != trx) {
|
|
sql_print_error("The transaction object for the table handle is at "
|
|
"%p, but for the current thread it is at %p",
|
|
prebuilt->trx, trx);
|
|
|
|
fputs("InnoDB: Dump of 200 bytes around prebuilt: ", stderr);
|
|
ut_print_buf(stderr, ((const byte*)prebuilt) - 100, 200);
|
|
fputs("\n"
|
|
"InnoDB: Dump of 200 bytes around ha_data: ",
|
|
stderr);
|
|
ut_print_buf(stderr, ((const byte*) trx) - 100, 200);
|
|
putc('\n', stderr);
|
|
ut_error;
|
|
}
|
|
|
|
ha_statistic_increment(&SSV::ha_write_count);
|
|
|
|
if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_INSERT)
|
|
table->timestamp_field->set_time();
|
|
|
|
sql_command = thd_sql_command(user_thd);
|
|
|
|
if ((sql_command == SQLCOM_ALTER_TABLE
|
|
|| sql_command == SQLCOM_OPTIMIZE
|
|
|| sql_command == SQLCOM_CREATE_INDEX
|
|
|| sql_command == SQLCOM_DROP_INDEX)
|
|
&& num_write_row >= 10000) {
|
|
/* ALTER TABLE is COMMITted at every 10000 copied rows.
|
|
The IX table lock for the original table has to be re-issued.
|
|
As this method will be called on a temporary table where the
|
|
contents of the original table is being copied to, it is
|
|
a bit tricky to determine the source table. The cursor
|
|
position in the source table need not be adjusted after the
|
|
intermediate COMMIT, since writes by other transactions are
|
|
being blocked by a MySQL table lock TL_WRITE_ALLOW_READ. */
|
|
|
|
dict_table_t* src_table;
|
|
enum lock_mode mode;
|
|
|
|
num_write_row = 0;
|
|
|
|
/* Commit the transaction. This will release the table
|
|
locks, so they have to be acquired again. */
|
|
|
|
/* Altering an InnoDB table */
|
|
/* Get the source table. */
|
|
src_table = lock_get_src_table(
|
|
prebuilt->trx, prebuilt->table, &mode);
|
|
if (!src_table) {
|
|
no_commit:
|
|
/* Unknown situation: do not commit */
|
|
/*
|
|
ut_print_timestamp(stderr);
|
|
fprintf(stderr,
|
|
" InnoDB error: ALTER TABLE is holding lock"
|
|
" on %lu tables!\n",
|
|
prebuilt->trx->mysql_n_tables_locked);
|
|
*/
|
|
;
|
|
} else if (src_table == prebuilt->table) {
|
|
/* Source table is not in InnoDB format:
|
|
no need to re-acquire locks on it. */
|
|
|
|
/* Altering to InnoDB format */
|
|
innobase_commit(ht, user_thd, 1);
|
|
/* Note that this transaction is still active. */
|
|
prebuilt->trx->active_trans = 1;
|
|
/* We will need an IX lock on the destination table. */
|
|
prebuilt->sql_stat_start = TRUE;
|
|
} else {
|
|
/* Ensure that there are no other table locks than
|
|
LOCK_IX and LOCK_AUTO_INC on the destination table. */
|
|
|
|
if (!lock_is_table_exclusive(prebuilt->table,
|
|
prebuilt->trx)) {
|
|
goto no_commit;
|
|
}
|
|
|
|
/* Commit the transaction. This will release the table
|
|
locks, so they have to be acquired again. */
|
|
innobase_commit(ht, user_thd, 1);
|
|
/* Note that this transaction is still active. */
|
|
prebuilt->trx->active_trans = 1;
|
|
/* Re-acquire the table lock on the source table. */
|
|
row_lock_table_for_mysql(prebuilt, src_table, mode);
|
|
/* We will need an IX lock on the destination table. */
|
|
prebuilt->sql_stat_start = TRUE;
|
|
}
|
|
}
|
|
|
|
num_write_row++;
|
|
|
|
if (table->next_number_field && record == table->record[0]) {
|
|
/* This is the case where the table has an
|
|
auto-increment column */
|
|
|
|
/* Initialize the auto-inc counter if it has not been
|
|
initialized yet */
|
|
|
|
if (0 == dict_table_autoinc_peek(prebuilt->table)) {
|
|
|
|
/* This call initializes the counter */
|
|
error = innobase_read_and_init_auto_inc(&dummy);
|
|
|
|
if (error) {
|
|
/* Deadlock or lock wait timeout */
|
|
|
|
goto func_exit;
|
|
}
|
|
|
|
/* We have to set sql_stat_start to TRUE because
|
|
the above call probably has called a select, and
|
|
has reset that flag; row_insert_for_mysql has to
|
|
know to set the IX intention lock on the table,
|
|
something it only does at the start of each
|
|
statement */
|
|
|
|
prebuilt->sql_stat_start = TRUE;
|
|
}
|
|
|
|
/* We have to use the transactional lock mechanism on the
|
|
auto-inc counter of the table to ensure that replication and
|
|
roll-forward of the binlog exactly imitates also the given
|
|
auto-inc values. The lock is released at each SQL statement's
|
|
end. This lock also prevents a race where two threads would
|
|
call ::get_auto_increment() simultaneously. */
|
|
|
|
error = row_lock_table_autoinc_for_mysql(prebuilt);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
/* Deadlock or lock wait timeout */
|
|
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
goto func_exit;
|
|
}
|
|
|
|
/* We must use the handler code to update the auto-increment
|
|
value to be sure that we increment it correctly. */
|
|
|
|
if ((error= update_auto_increment()))
|
|
goto func_exit;
|
|
auto_inc_used = 1;
|
|
|
|
}
|
|
|
|
if (prebuilt->mysql_template == NULL
|
|
|| prebuilt->template_type != ROW_MYSQL_WHOLE_ROW) {
|
|
/* Build the template used in converting quickly between
|
|
the two database formats */
|
|
|
|
build_template(prebuilt, NULL, table, ROW_MYSQL_WHOLE_ROW);
|
|
}
|
|
|
|
innodb_srv_conc_enter_innodb(prebuilt->trx);
|
|
|
|
error = row_insert_for_mysql((byte*) record, prebuilt);
|
|
|
|
if (error == DB_SUCCESS && auto_inc_used) {
|
|
|
|
/* Fetch the value that was set in the autoincrement field */
|
|
|
|
auto_inc = table->next_number_field->val_int();
|
|
|
|
if (auto_inc != 0) {
|
|
/* This call will update the counter according to the
|
|
value that was inserted in the table */
|
|
|
|
dict_table_autoinc_update(prebuilt->table, auto_inc);
|
|
}
|
|
}
|
|
|
|
/* A REPLACE command and LOAD DATA INFILE REPLACE handle a duplicate
|
|
key error themselves, and we must update the autoinc counter if we are
|
|
performing those statements. */
|
|
|
|
if (error == DB_DUPLICATE_KEY && auto_inc_used
|
|
&& (sql_command == SQLCOM_REPLACE
|
|
|| sql_command == SQLCOM_REPLACE_SELECT
|
|
|| (sql_command == SQLCOM_INSERT
|
|
&& ((trx->duplicates
|
|
& (TRX_DUP_IGNORE | TRX_DUP_REPLACE))
|
|
== TRX_DUP_IGNORE))
|
|
|| (sql_command == SQLCOM_LOAD
|
|
&& ((trx->duplicates
|
|
& (TRX_DUP_IGNORE | TRX_DUP_REPLACE))
|
|
== (TRX_DUP_IGNORE | TRX_DUP_REPLACE))))) {
|
|
|
|
auto_inc = table->next_number_field->val_int();
|
|
|
|
if (auto_inc != 0) {
|
|
dict_table_autoinc_update(prebuilt->table, auto_inc);
|
|
}
|
|
}
|
|
|
|
innodb_srv_conc_exit_innodb(prebuilt->trx);
|
|
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
/* Tell InnoDB server that there might be work for
|
|
utility threads: */
|
|
func_exit:
|
|
innobase_active_small();
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Checks which fields have changed in a row and stores information
|
|
of them to an update vector. */
|
|
static
|
|
int
|
|
calc_row_difference(
|
|
/*================*/
|
|
/* out: error number or 0 */
|
|
upd_t* uvect, /* in/out: update vector */
|
|
mysql_byte* old_row, /* in: old row in MySQL format */
|
|
mysql_byte* new_row, /* in: new row in MySQL format */
|
|
struct st_table* table, /* in: table in MySQL data
|
|
dictionary */
|
|
mysql_byte* upd_buff, /* in: buffer to use */
|
|
ulint buff_len, /* in: buffer length */
|
|
row_prebuilt_t* prebuilt, /* in: InnoDB prebuilt struct */
|
|
THD* thd) /* in: user thread */
|
|
{
|
|
mysql_byte* original_upd_buff = upd_buff;
|
|
Field* field;
|
|
enum_field_types field_mysql_type;
|
|
uint n_fields;
|
|
ulint o_len;
|
|
ulint n_len;
|
|
ulint col_pack_len;
|
|
byte* new_mysql_row_col;
|
|
byte* o_ptr;
|
|
byte* n_ptr;
|
|
byte* buf;
|
|
upd_field_t* ufield;
|
|
ulint col_type;
|
|
ulint n_changed = 0;
|
|
dfield_t dfield;
|
|
dict_index_t* clust_index;
|
|
uint i;
|
|
|
|
n_fields = table->s->fields;
|
|
clust_index = dict_table_get_first_index_noninline(prebuilt->table);
|
|
|
|
/* We use upd_buff to convert changed fields */
|
|
buf = (byte*) upd_buff;
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
field = table->field[i];
|
|
|
|
o_ptr = (byte*) old_row + get_field_offset(table, field);
|
|
n_ptr = (byte*) new_row + get_field_offset(table, field);
|
|
|
|
/* Use new_mysql_row_col and col_pack_len save the values */
|
|
|
|
new_mysql_row_col = n_ptr;
|
|
col_pack_len = field->pack_length();
|
|
|
|
o_len = col_pack_len;
|
|
n_len = col_pack_len;
|
|
|
|
/* We use o_ptr and n_ptr to dig up the actual data for
|
|
comparison. */
|
|
|
|
field_mysql_type = field->type();
|
|
|
|
col_type = prebuilt->table->cols[i].mtype;
|
|
|
|
switch (col_type) {
|
|
|
|
case DATA_BLOB:
|
|
o_ptr = row_mysql_read_blob_ref(&o_len, o_ptr, o_len);
|
|
n_ptr = row_mysql_read_blob_ref(&n_len, n_ptr, n_len);
|
|
|
|
break;
|
|
|
|
case DATA_VARCHAR:
|
|
case DATA_BINARY:
|
|
case DATA_VARMYSQL:
|
|
if (field_mysql_type == MYSQL_TYPE_VARCHAR) {
|
|
/* This is a >= 5.0.3 type true VARCHAR where
|
|
the real payload data length is stored in
|
|
1 or 2 bytes */
|
|
|
|
o_ptr = row_mysql_read_true_varchar(
|
|
&o_len, o_ptr,
|
|
(ulint)
|
|
(((Field_varstring*)field)->length_bytes));
|
|
|
|
n_ptr = row_mysql_read_true_varchar(
|
|
&n_len, n_ptr,
|
|
(ulint)
|
|
(((Field_varstring*)field)->length_bytes));
|
|
}
|
|
|
|
break;
|
|
default:
|
|
;
|
|
}
|
|
|
|
if (field->null_ptr) {
|
|
if (field_in_record_is_null(table, field,
|
|
(char*) old_row)) {
|
|
o_len = UNIV_SQL_NULL;
|
|
}
|
|
|
|
if (field_in_record_is_null(table, field,
|
|
(char*) new_row)) {
|
|
n_len = UNIV_SQL_NULL;
|
|
}
|
|
}
|
|
|
|
if (o_len != n_len || (o_len != UNIV_SQL_NULL &&
|
|
0 != memcmp(o_ptr, n_ptr, o_len))) {
|
|
/* The field has changed */
|
|
|
|
ufield = uvect->fields + n_changed;
|
|
|
|
/* Let us use a dummy dfield to make the conversion
|
|
from the MySQL column format to the InnoDB format */
|
|
|
|
dict_col_copy_type_noninline(prebuilt->table->cols + i,
|
|
&dfield.type);
|
|
|
|
if (n_len != UNIV_SQL_NULL) {
|
|
buf = row_mysql_store_col_in_innobase_format(
|
|
&dfield,
|
|
(byte*)buf,
|
|
TRUE,
|
|
new_mysql_row_col,
|
|
col_pack_len,
|
|
dict_table_is_comp_noninline(
|
|
prebuilt->table));
|
|
ufield->new_val.data = dfield.data;
|
|
ufield->new_val.len = dfield.len;
|
|
} else {
|
|
ufield->new_val.data = NULL;
|
|
ufield->new_val.len = UNIV_SQL_NULL;
|
|
}
|
|
|
|
ufield->exp = NULL;
|
|
ufield->field_no = dict_col_get_clust_pos_noninline(
|
|
&prebuilt->table->cols[i], clust_index);
|
|
n_changed++;
|
|
}
|
|
}
|
|
|
|
uvect->n_fields = n_changed;
|
|
uvect->info_bits = 0;
|
|
|
|
ut_a(buf <= (byte*)original_upd_buff + buff_len);
|
|
|
|
return(0);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Updates a row given as a parameter to a new value. Note that we are given
|
|
whole rows, not just the fields which are updated: this incurs some
|
|
overhead for CPU when we check which fields are actually updated.
|
|
TODO: currently InnoDB does not prevent the 'Halloween problem':
|
|
in a searched update a single row can get updated several times
|
|
if its index columns are updated! */
|
|
|
|
int
|
|
ha_innobase::update_row(
|
|
/*====================*/
|
|
/* out: error number or 0 */
|
|
const mysql_byte* old_row,/* in: old row in MySQL format */
|
|
mysql_byte* new_row)/* in: new row in MySQL format */
|
|
{
|
|
upd_t* uvect;
|
|
int error = 0;
|
|
trx_t* trx = thd_to_trx(user_thd);
|
|
|
|
DBUG_ENTER("ha_innobase::update_row");
|
|
|
|
ut_a(prebuilt->trx == trx);
|
|
|
|
if (table->timestamp_field_type & TIMESTAMP_AUTO_SET_ON_UPDATE)
|
|
table->timestamp_field->set_time();
|
|
|
|
if (prebuilt->upd_node) {
|
|
uvect = prebuilt->upd_node->update;
|
|
} else {
|
|
uvect = row_get_prebuilt_update_vector(prebuilt);
|
|
}
|
|
|
|
/* Build an update vector from the modified fields in the rows
|
|
(uses upd_buff of the handle) */
|
|
|
|
calc_row_difference(uvect, (mysql_byte*) old_row, new_row, table,
|
|
upd_buff, (ulint)upd_and_key_val_buff_len,
|
|
prebuilt, user_thd);
|
|
|
|
/* This is not a delete */
|
|
prebuilt->upd_node->is_delete = FALSE;
|
|
|
|
ut_a(prebuilt->template_type == ROW_MYSQL_WHOLE_ROW);
|
|
|
|
innodb_srv_conc_enter_innodb(trx);
|
|
|
|
error = row_update_for_mysql((byte*) old_row, prebuilt);
|
|
|
|
innodb_srv_conc_exit_innodb(trx);
|
|
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
/* Tell InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
innobase_active_small();
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Deletes a row given as the parameter. */
|
|
|
|
int
|
|
ha_innobase::delete_row(
|
|
/*====================*/
|
|
/* out: error number or 0 */
|
|
const mysql_byte* record) /* in: a row in MySQL format */
|
|
{
|
|
int error = 0;
|
|
trx_t* trx = thd_to_trx(user_thd);
|
|
|
|
DBUG_ENTER("ha_innobase::delete_row");
|
|
|
|
ut_a(prebuilt->trx == trx);
|
|
|
|
if (!prebuilt->upd_node) {
|
|
row_get_prebuilt_update_vector(prebuilt);
|
|
}
|
|
|
|
/* This is a delete */
|
|
|
|
prebuilt->upd_node->is_delete = TRUE;
|
|
|
|
innodb_srv_conc_enter_innodb(trx);
|
|
|
|
error = row_update_for_mysql((byte*) record, prebuilt);
|
|
|
|
innodb_srv_conc_exit_innodb(trx);
|
|
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
innobase_active_small();
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Removes a new lock set on a row, if it was not read optimistically. This can
|
|
be called after a row has been read in the processing of an UPDATE or a DELETE
|
|
query, if the option innodb_locks_unsafe_for_binlog is set. */
|
|
|
|
void
|
|
ha_innobase::unlock_row(void)
|
|
/*=========================*/
|
|
{
|
|
DBUG_ENTER("ha_innobase::unlock_row");
|
|
|
|
/* Consistent read does not take any locks, thus there is
|
|
nothing to unlock. */
|
|
|
|
if (prebuilt->select_lock_type == LOCK_NONE) {
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
switch (prebuilt->row_read_type) {
|
|
case ROW_READ_WITH_LOCKS:
|
|
if (!srv_locks_unsafe_for_binlog
|
|
|| prebuilt->trx->isolation_level == TRX_ISO_READ_COMMITTED) {
|
|
break;
|
|
}
|
|
/* fall through */
|
|
case ROW_READ_TRY_SEMI_CONSISTENT:
|
|
row_unlock_for_mysql(prebuilt, FALSE);
|
|
break;
|
|
case ROW_READ_DID_SEMI_CONSISTENT:
|
|
prebuilt->row_read_type = ROW_READ_TRY_SEMI_CONSISTENT;
|
|
break;
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/* See handler.h and row0mysql.h for docs on this function. */
|
|
bool
|
|
ha_innobase::was_semi_consistent_read(void)
|
|
/*=======================================*/
|
|
{
|
|
return(prebuilt->row_read_type == ROW_READ_DID_SEMI_CONSISTENT);
|
|
}
|
|
|
|
/* See handler.h and row0mysql.h for docs on this function. */
|
|
void
|
|
ha_innobase::try_semi_consistent_read(bool yes)
|
|
/*===========================================*/
|
|
{
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
/* Row read type is set to semi consistent read if this was
|
|
requested by the MySQL and either innodb_locks_unsafe_for_binlog
|
|
option is used or this session is using READ COMMITTED isolation
|
|
level. */
|
|
|
|
if (yes
|
|
&& (srv_locks_unsafe_for_binlog
|
|
|| prebuilt->trx->isolation_level == TRX_ISO_READ_COMMITTED)) {
|
|
prebuilt->row_read_type = ROW_READ_TRY_SEMI_CONSISTENT;
|
|
} else {
|
|
prebuilt->row_read_type = ROW_READ_WITH_LOCKS;
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
Check if an index can be used by the optimizer. */
|
|
|
|
bool
|
|
ha_innobase::is_index_available(
|
|
/*============================*/
|
|
/* out: true if available else false*/
|
|
uint keynr) /* in: index number to check */
|
|
{
|
|
DBUG_ENTER("is_index_available");
|
|
|
|
if (table && keynr != MAX_KEY && table->s->keys > 0) {
|
|
dict_index_t* index;
|
|
KEY* key = table->key_info + keynr;
|
|
|
|
ut_ad(user_thd == ha_thd());
|
|
ut_a(prebuilt->trx == thd_to_trx(user_thd));
|
|
|
|
index = dict_table_get_index_on_name(
|
|
prebuilt->table, key->name);
|
|
|
|
if (!row_merge_is_index_usable(prebuilt->trx, index)) {
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Initializes a handle to use an index. */
|
|
|
|
int
|
|
ha_innobase::index_init(
|
|
/*====================*/
|
|
/* out: 0 or error number */
|
|
uint keynr, /* in: key (index) number */
|
|
bool sorted) /* in: 1 if result MUST be sorted according to index */
|
|
{
|
|
DBUG_ENTER("index_init");
|
|
|
|
DBUG_RETURN(change_active_index(keynr));
|
|
}
|
|
|
|
/**********************************************************************
|
|
Currently does nothing. */
|
|
|
|
int
|
|
ha_innobase::index_end(void)
|
|
/*========================*/
|
|
{
|
|
int error = 0;
|
|
DBUG_ENTER("index_end");
|
|
active_index=MAX_KEY;
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Converts a search mode flag understood by MySQL to a flag understood
|
|
by InnoDB. */
|
|
inline
|
|
ulint
|
|
convert_search_mode_to_innobase(
|
|
/*============================*/
|
|
enum ha_rkey_function find_flag)
|
|
{
|
|
switch (find_flag) {
|
|
case HA_READ_KEY_EXACT: return(PAGE_CUR_GE);
|
|
/* the above does not require the index to be UNIQUE */
|
|
case HA_READ_KEY_OR_NEXT: return(PAGE_CUR_GE);
|
|
case HA_READ_KEY_OR_PREV: return(PAGE_CUR_LE);
|
|
case HA_READ_AFTER_KEY: return(PAGE_CUR_G);
|
|
case HA_READ_BEFORE_KEY: return(PAGE_CUR_L);
|
|
case HA_READ_PREFIX: return(PAGE_CUR_GE);
|
|
case HA_READ_PREFIX_LAST: return(PAGE_CUR_LE);
|
|
case HA_READ_PREFIX_LAST_OR_PREV:return(PAGE_CUR_LE);
|
|
/* In MySQL-4.0 HA_READ_PREFIX and HA_READ_PREFIX_LAST always
|
|
pass a complete-field prefix of a key value as the search
|
|
tuple. I.e., it is not allowed that the last field would
|
|
just contain n first bytes of the full field value.
|
|
MySQL uses a 'padding' trick to convert LIKE 'abc%'
|
|
type queries so that it can use as a search tuple
|
|
a complete-field-prefix of a key value. Thus, the InnoDB
|
|
search mode PAGE_CUR_LE_OR_EXTENDS is never used.
|
|
TODO: when/if MySQL starts to use also partial-field
|
|
prefixes, we have to deal with stripping of spaces
|
|
and comparison of non-latin1 char type fields in
|
|
innobase_mysql_cmp() to get PAGE_CUR_LE_OR_EXTENDS to
|
|
work correctly. */
|
|
|
|
default: ut_error;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/*
|
|
BACKGROUND INFO: HOW A SELECT SQL QUERY IS EXECUTED
|
|
---------------------------------------------------
|
|
The following does not cover all the details, but explains how we determine
|
|
the start of a new SQL statement, and what is associated with it.
|
|
|
|
For each table in the database the MySQL interpreter may have several
|
|
table handle instances in use, also in a single SQL query. For each table
|
|
handle instance there is an InnoDB 'prebuilt' struct which contains most
|
|
of the InnoDB data associated with this table handle instance.
|
|
|
|
A) if the user has not explicitly set any MySQL table level locks:
|
|
|
|
1) MySQL calls ::external_lock to set an 'intention' table level lock on
|
|
the table of the handle instance. There we set
|
|
prebuilt->sql_stat_start = TRUE. The flag sql_stat_start should be set
|
|
true if we are taking this table handle instance to use in a new SQL
|
|
statement issued by the user. We also increment trx->n_mysql_tables_in_use.
|
|
|
|
2) If prebuilt->sql_stat_start == TRUE we 'pre-compile' the MySQL search
|
|
instructions to prebuilt->template of the table handle instance in
|
|
::index_read. The template is used to save CPU time in large joins.
|
|
|
|
3) In row_search_for_mysql, if prebuilt->sql_stat_start is true, we
|
|
allocate a new consistent read view for the trx if it does not yet have one,
|
|
or in the case of a locking read, set an InnoDB 'intention' table level
|
|
lock on the table.
|
|
|
|
4) We do the SELECT. MySQL may repeatedly call ::index_read for the
|
|
same table handle instance, if it is a join.
|
|
|
|
5) When the SELECT ends, MySQL removes its intention table level locks
|
|
in ::external_lock. When trx->n_mysql_tables_in_use drops to zero,
|
|
(a) we execute a COMMIT there if the autocommit is on,
|
|
(b) we also release possible 'SQL statement level resources' InnoDB may
|
|
have for this SQL statement. The MySQL interpreter does NOT execute
|
|
autocommit for pure read transactions, though it should. That is why the
|
|
table handler in that case has to execute the COMMIT in ::external_lock.
|
|
|
|
B) If the user has explicitly set MySQL table level locks, then MySQL
|
|
does NOT call ::external_lock at the start of the statement. To determine
|
|
when we are at the start of a new SQL statement we at the start of
|
|
::index_read also compare the query id to the latest query id where the
|
|
table handle instance was used. If it has changed, we know we are at the
|
|
start of a new SQL statement. Since the query id can theoretically
|
|
overwrap, we use this test only as a secondary way of determining the
|
|
start of a new SQL statement. */
|
|
|
|
|
|
/**************************************************************************
|
|
Positions an index cursor to the index specified in the handle. Fetches the
|
|
row if any. */
|
|
|
|
int
|
|
ha_innobase::index_read(
|
|
/*====================*/
|
|
/* out: 0, HA_ERR_KEY_NOT_FOUND,
|
|
or error number */
|
|
mysql_byte* buf, /* in/out: buffer for the returned
|
|
row */
|
|
const mysql_byte* key_ptr,/* in: key value; if this is NULL
|
|
we position the cursor at the
|
|
start or end of index; this can
|
|
also contain an InnoDB row id, in
|
|
which case key_len is the InnoDB
|
|
row id length; the key value can
|
|
also be a prefix of a full key value,
|
|
and the last column can be a prefix
|
|
of a full column */
|
|
uint key_len,/* in: key value length */
|
|
enum ha_rkey_function find_flag)/* in: search flags from my_base.h */
|
|
{
|
|
ulint mode;
|
|
dict_index_t* index;
|
|
ulint match_mode = 0;
|
|
int error;
|
|
ulint ret;
|
|
|
|
DBUG_ENTER("index_read");
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(user_thd));
|
|
|
|
ha_statistic_increment(&SSV::ha_read_key_count);
|
|
|
|
index = prebuilt->index;
|
|
|
|
/* Note that if the index for which the search template is built is not
|
|
necessarily prebuilt->index, but can also be the clustered index */
|
|
|
|
if (prebuilt->sql_stat_start) {
|
|
build_template(prebuilt, user_thd, table, ROW_MYSQL_REC_FIELDS);
|
|
}
|
|
|
|
if (key_ptr) {
|
|
/* Convert the search key value to InnoDB format into
|
|
prebuilt->search_tuple */
|
|
|
|
row_sel_convert_mysql_key_to_innobase(
|
|
prebuilt->search_tuple,
|
|
(byte*) key_val_buff,
|
|
(ulint)upd_and_key_val_buff_len,
|
|
index,
|
|
(byte*) key_ptr,
|
|
(ulint) key_len,
|
|
prebuilt->trx);
|
|
} else {
|
|
/* We position the cursor to the last or the first entry
|
|
in the index */
|
|
|
|
dtuple_set_n_fields(prebuilt->search_tuple, 0);
|
|
}
|
|
|
|
mode = convert_search_mode_to_innobase(find_flag);
|
|
|
|
match_mode = 0;
|
|
|
|
if (find_flag == HA_READ_KEY_EXACT) {
|
|
|
|
match_mode = ROW_SEL_EXACT;
|
|
|
|
} else if (find_flag == HA_READ_PREFIX
|
|
|| find_flag == HA_READ_PREFIX_LAST) {
|
|
|
|
match_mode = ROW_SEL_EXACT_PREFIX;
|
|
}
|
|
|
|
last_match_mode = (uint) match_mode;
|
|
|
|
innodb_srv_conc_enter_innodb(prebuilt->trx);
|
|
|
|
ret = row_search_for_mysql((byte*) buf, mode, prebuilt, match_mode, 0);
|
|
|
|
innodb_srv_conc_exit_innodb(prebuilt->trx);
|
|
|
|
if (ret == DB_SUCCESS) {
|
|
error = 0;
|
|
table->status = 0;
|
|
|
|
} else if (ret == DB_RECORD_NOT_FOUND) {
|
|
error = HA_ERR_KEY_NOT_FOUND;
|
|
table->status = STATUS_NOT_FOUND;
|
|
|
|
} else if (ret == DB_END_OF_INDEX) {
|
|
error = HA_ERR_KEY_NOT_FOUND;
|
|
table->status = STATUS_NOT_FOUND;
|
|
} else {
|
|
error = convert_error_code_to_mysql((int) ret, user_thd);
|
|
table->status = STATUS_NOT_FOUND;
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/***********************************************************************
|
|
The following functions works like index_read, but it find the last
|
|
row with the current key value or prefix. */
|
|
|
|
int
|
|
ha_innobase::index_read_last(
|
|
/*=========================*/
|
|
/* out: 0, HA_ERR_KEY_NOT_FOUND, or an
|
|
error code */
|
|
mysql_byte* buf, /* out: fetched row */
|
|
const mysql_byte* key_ptr, /* in: key value, or a prefix of a full
|
|
key value */
|
|
uint key_len) /* in: length of the key val or prefix
|
|
in bytes */
|
|
{
|
|
return(index_read(buf, key_ptr, key_len, HA_READ_PREFIX_LAST));
|
|
}
|
|
|
|
/************************************************************************
|
|
Changes the active index of a handle. */
|
|
|
|
int
|
|
ha_innobase::change_active_index(
|
|
/*=============================*/
|
|
/* out: 0 or error code */
|
|
uint keynr) /* in: use this index; MAX_KEY means always clustered
|
|
index, even if it was internally generated by
|
|
InnoDB */
|
|
{
|
|
KEY* key = 0;
|
|
DBUG_ENTER("change_active_index");
|
|
ha_statistic_increment(&SSV::ha_read_key_count);
|
|
|
|
ut_ad(user_thd == ha_thd());
|
|
ut_a(prebuilt->trx == thd_to_trx(user_thd));
|
|
|
|
active_index = keynr;
|
|
|
|
if (keynr != MAX_KEY && table->s->keys > 0) {
|
|
key = table->key_info + active_index;
|
|
|
|
prebuilt->index = dict_table_get_index_on_name(
|
|
prebuilt->table, key->name);
|
|
} else {
|
|
prebuilt->index = dict_table_get_first_index_noninline(
|
|
prebuilt->table);
|
|
}
|
|
|
|
if (!prebuilt->index) {
|
|
sql_print_error(
|
|
"Innodb could not find key n:o %u with name %s "
|
|
"from dict cache for table %s",
|
|
keynr, key ? key->name : "NULL",
|
|
prebuilt->table->name);
|
|
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
ut_a(prebuilt->search_tuple != 0);
|
|
|
|
dtuple_set_n_fields(prebuilt->search_tuple, prebuilt->index->n_fields);
|
|
|
|
dict_index_copy_types(prebuilt->search_tuple, prebuilt->index,
|
|
prebuilt->index->n_fields);
|
|
|
|
/* MySQL changes the active index for a handle also during some
|
|
queries, for example SELECT MAX(a), SUM(a) first retrieves the MAX()
|
|
and then calculates the sum. Previously we played safe and used
|
|
the flag ROW_MYSQL_WHOLE_ROW below, but that caused unnecessary
|
|
copying. Starting from MySQL-4.1 we use a more efficient flag here. */
|
|
|
|
build_template(prebuilt, user_thd, table, ROW_MYSQL_REC_FIELDS);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Positions an index cursor to the index specified in keynr. Fetches the
|
|
row if any. */
|
|
/* ??? This is only used to read whole keys ??? */
|
|
|
|
int
|
|
ha_innobase::index_read_idx(
|
|
/*========================*/
|
|
/* out: error number or 0 */
|
|
mysql_byte* buf, /* in/out: buffer for the returned
|
|
row */
|
|
uint keynr, /* in: use this index */
|
|
const mysql_byte* key, /* in: key value; if this is NULL
|
|
we position the cursor at the
|
|
start or end of index */
|
|
uint key_len, /* in: key value length */
|
|
enum ha_rkey_function find_flag)/* in: search flags from my_base.h */
|
|
{
|
|
if (change_active_index(keynr)) {
|
|
|
|
return(1);
|
|
}
|
|
|
|
return(index_read(buf, key, key_len, find_flag));
|
|
}
|
|
|
|
/***************************************************************************
|
|
Reads the next or previous row from a cursor, which must have previously been
|
|
positioned using index_read. */
|
|
|
|
int
|
|
ha_innobase::general_fetch(
|
|
/*=======================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error
|
|
number */
|
|
mysql_byte* buf, /* in/out: buffer for next row in MySQL
|
|
format */
|
|
uint direction, /* in: ROW_SEL_NEXT or ROW_SEL_PREV */
|
|
uint match_mode) /* in: 0, ROW_SEL_EXACT, or
|
|
ROW_SEL_EXACT_PREFIX */
|
|
{
|
|
ulint ret;
|
|
int error = 0;
|
|
|
|
DBUG_ENTER("general_fetch");
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(user_thd));
|
|
|
|
innodb_srv_conc_enter_innodb(prebuilt->trx);
|
|
|
|
ret = row_search_for_mysql(
|
|
(byte*)buf, 0, prebuilt, match_mode, direction);
|
|
|
|
innodb_srv_conc_exit_innodb(prebuilt->trx);
|
|
|
|
if (ret == DB_SUCCESS) {
|
|
error = 0;
|
|
table->status = 0;
|
|
|
|
} else if (ret == DB_RECORD_NOT_FOUND) {
|
|
error = HA_ERR_END_OF_FILE;
|
|
table->status = STATUS_NOT_FOUND;
|
|
|
|
} else if (ret == DB_END_OF_INDEX) {
|
|
error = HA_ERR_END_OF_FILE;
|
|
table->status = STATUS_NOT_FOUND;
|
|
} else {
|
|
error = convert_error_code_to_mysql((int) ret, user_thd);
|
|
table->status = STATUS_NOT_FOUND;
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/***************************************************************************
|
|
Reads the next row from a cursor, which must have previously been
|
|
positioned using index_read. */
|
|
|
|
int
|
|
ha_innobase::index_next(
|
|
/*====================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error
|
|
number */
|
|
mysql_byte* buf) /* in/out: buffer for next row in MySQL
|
|
format */
|
|
{
|
|
ha_statistic_increment(&SSV::ha_read_next_count);
|
|
|
|
return(general_fetch(buf, ROW_SEL_NEXT, 0));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Reads the next row matching to the key value given as the parameter. */
|
|
|
|
int
|
|
ha_innobase::index_next_same(
|
|
/*=========================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error
|
|
number */
|
|
mysql_byte* buf, /* in/out: buffer for the row */
|
|
const mysql_byte* key, /* in: key value */
|
|
uint keylen) /* in: key value length */
|
|
{
|
|
ha_statistic_increment(&SSV::ha_read_next_count);
|
|
|
|
return(general_fetch(buf, ROW_SEL_NEXT, last_match_mode));
|
|
}
|
|
|
|
/***************************************************************************
|
|
Reads the previous row from a cursor, which must have previously been
|
|
positioned using index_read. */
|
|
|
|
int
|
|
ha_innobase::index_prev(
|
|
/*====================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error
|
|
number */
|
|
mysql_byte* buf) /* in/out: buffer for previous row in MySQL
|
|
format */
|
|
{
|
|
ha_statistic_increment(&SSV::ha_read_prev_count);
|
|
|
|
return(general_fetch(buf, ROW_SEL_PREV, 0));
|
|
}
|
|
|
|
/************************************************************************
|
|
Positions a cursor on the first record in an index and reads the
|
|
corresponding row to buf. */
|
|
|
|
int
|
|
ha_innobase::index_first(
|
|
/*=====================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE,
|
|
or error code */
|
|
mysql_byte* buf) /* in/out: buffer for the row */
|
|
{
|
|
int error;
|
|
|
|
DBUG_ENTER("index_first");
|
|
ha_statistic_increment(&SSV::ha_read_first_count);
|
|
|
|
error = index_read(buf, NULL, 0, HA_READ_AFTER_KEY);
|
|
|
|
/* MySQL does not seem to allow this to return HA_ERR_KEY_NOT_FOUND */
|
|
|
|
if (error == HA_ERR_KEY_NOT_FOUND) {
|
|
error = HA_ERR_END_OF_FILE;
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/************************************************************************
|
|
Positions a cursor on the last record in an index and reads the
|
|
corresponding row to buf. */
|
|
|
|
int
|
|
ha_innobase::index_last(
|
|
/*====================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error code */
|
|
mysql_byte* buf) /* in/out: buffer for the row */
|
|
{
|
|
int error;
|
|
|
|
DBUG_ENTER("index_last");
|
|
ha_statistic_increment(&SSV::ha_read_last_count);
|
|
|
|
error = index_read(buf, NULL, 0, HA_READ_BEFORE_KEY);
|
|
|
|
/* MySQL does not seem to allow this to return HA_ERR_KEY_NOT_FOUND */
|
|
|
|
if (error == HA_ERR_KEY_NOT_FOUND) {
|
|
error = HA_ERR_END_OF_FILE;
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/********************************************************************
|
|
Initialize a table scan. */
|
|
|
|
int
|
|
ha_innobase::rnd_init(
|
|
/*==================*/
|
|
/* out: 0 or error number */
|
|
bool scan) /* in: TRUE if table/index scan FALSE otherwise */
|
|
{
|
|
int err;
|
|
|
|
/* Store the active index value so that we can restore the original
|
|
value after a scan */
|
|
|
|
if (prebuilt->clust_index_was_generated) {
|
|
err = change_active_index(MAX_KEY);
|
|
} else {
|
|
err = change_active_index(primary_key);
|
|
}
|
|
|
|
/* Don't use semi-consistent read in random row reads (by position).
|
|
This means we must disable semi_consistent_read if scan is false */
|
|
|
|
if (!scan) {
|
|
try_semi_consistent_read(0);
|
|
}
|
|
|
|
start_of_scan = 1;
|
|
|
|
return(err);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Ends a table scan. */
|
|
|
|
int
|
|
ha_innobase::rnd_end(void)
|
|
/*======================*/
|
|
/* out: 0 or error number */
|
|
{
|
|
return(index_end());
|
|
}
|
|
|
|
/*********************************************************************
|
|
Reads the next row in a table scan (also used to read the FIRST row
|
|
in a table scan). */
|
|
|
|
int
|
|
ha_innobase::rnd_next(
|
|
/*==================*/
|
|
/* out: 0, HA_ERR_END_OF_FILE, or error number */
|
|
mysql_byte* buf)/* in/out: returns the row in this buffer,
|
|
in MySQL format */
|
|
{
|
|
int error;
|
|
|
|
DBUG_ENTER("rnd_next");
|
|
ha_statistic_increment(&SSV::ha_read_rnd_next_count);
|
|
|
|
if (start_of_scan) {
|
|
error = index_first(buf);
|
|
|
|
if (error == HA_ERR_KEY_NOT_FOUND) {
|
|
error = HA_ERR_END_OF_FILE;
|
|
}
|
|
|
|
start_of_scan = 0;
|
|
} else {
|
|
error = general_fetch(buf, ROW_SEL_NEXT, 0);
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Fetches a row from the table based on a row reference. */
|
|
|
|
int
|
|
ha_innobase::rnd_pos(
|
|
/*=================*/
|
|
/* out: 0, HA_ERR_KEY_NOT_FOUND,
|
|
or error code */
|
|
mysql_byte* buf, /* in/out: buffer for the row */
|
|
mysql_byte* pos) /* in: primary key value of the row in the
|
|
MySQL format, or the row id if the clustered
|
|
index was internally generated by InnoDB;
|
|
the length of data in pos has to be
|
|
ref_length */
|
|
{
|
|
int error;
|
|
uint keynr = active_index;
|
|
DBUG_ENTER("rnd_pos");
|
|
DBUG_DUMP("key", (char*) pos, ref_length);
|
|
|
|
ha_statistic_increment(&SSV::ha_read_rnd_count);
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
if (prebuilt->clust_index_was_generated) {
|
|
/* No primary key was defined for the table and we
|
|
generated the clustered index from the row id: the
|
|
row reference is the row id, not any key value
|
|
that MySQL knows of */
|
|
|
|
error = change_active_index(MAX_KEY);
|
|
} else {
|
|
error = change_active_index(primary_key);
|
|
}
|
|
|
|
if (error) {
|
|
DBUG_PRINT("error", ("Got error: %d", error));
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/* Note that we assume the length of the row reference is fixed
|
|
for the table, and it is == ref_length */
|
|
|
|
error = index_read(buf, pos, ref_length, HA_READ_KEY_EXACT);
|
|
|
|
if (error) {
|
|
DBUG_PRINT("error", ("Got error: %d", error));
|
|
}
|
|
|
|
change_active_index(keynr);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Stores a reference to the current row to 'ref' field of the handle. Note
|
|
that in the case where we have generated the clustered index for the
|
|
table, the function parameter is illogical: we MUST ASSUME that 'record'
|
|
is the current 'position' of the handle, because if row ref is actually
|
|
the row id internally generated in InnoDB, then 'record' does not contain
|
|
it. We just guess that the row id must be for the record where the handle
|
|
was positioned the last time. */
|
|
|
|
void
|
|
ha_innobase::position(
|
|
/*==================*/
|
|
const mysql_byte* record) /* in: row in MySQL format */
|
|
{
|
|
uint len;
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
if (prebuilt->clust_index_was_generated) {
|
|
/* No primary key was defined for the table and we
|
|
generated the clustered index from row id: the
|
|
row reference will be the row id, not any key value
|
|
that MySQL knows of */
|
|
|
|
len = DATA_ROW_ID_LEN;
|
|
|
|
memcpy(ref, prebuilt->row_id, len);
|
|
} else {
|
|
len = store_key_val_for_row(primary_key, (char*)ref,
|
|
ref_length, record);
|
|
}
|
|
|
|
/* We assume that the 'ref' value len is always fixed for the same
|
|
table. */
|
|
|
|
if (len != ref_length) {
|
|
sql_print_error("Stored ref len is %lu, but table ref len is %lu",
|
|
(ulong) len, (ulong) ref_length);
|
|
}
|
|
}
|
|
|
|
/*********************************************************************
|
|
If it's a DB_TOO_BIG_RECORD error then set a suitable message to
|
|
return to the client.*/
|
|
static
|
|
void
|
|
innodb_check_for_record_too_big_error(
|
|
/*==================================*/
|
|
dict_table_t* table, /* in: table to check */
|
|
int error) /* in: error code to check */
|
|
{
|
|
if (error == (int)DB_TOO_BIG_RECORD) {
|
|
ulint max_row_size;
|
|
|
|
max_row_size = page_get_free_space_of_empty_noninline(table);
|
|
|
|
my_error(ER_TOO_BIG_ROWSIZE, MYF(0), max_row_size);
|
|
}
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates a table definition to an InnoDB database. */
|
|
static
|
|
int
|
|
create_table_def(
|
|
/*=============*/
|
|
trx_t* trx, /* in: InnoDB transaction handle */
|
|
TABLE* form, /* in: information on table
|
|
columns and indexes */
|
|
const char* table_name, /* in: table name */
|
|
const char* path_of_temp_table,/* in: if this is a table explicitly
|
|
created by the user with the
|
|
TEMPORARY keyword, then this
|
|
parameter is the dir path where the
|
|
table should be placed if we create
|
|
an .ibd file for it (no .ibd extension
|
|
in the path, though); otherwise this
|
|
is NULL */
|
|
ulint flags) /* in: table flags */
|
|
{
|
|
Field* field;
|
|
dict_table_t* table;
|
|
ulint n_cols;
|
|
int error;
|
|
ulint col_type;
|
|
ulint col_len;
|
|
ulint nulls_allowed;
|
|
ulint unsigned_type;
|
|
ulint binary_type;
|
|
ulint long_true_varchar;
|
|
ulint charset_no;
|
|
ulint i;
|
|
|
|
DBUG_ENTER("create_table_def");
|
|
DBUG_PRINT("enter", ("table_name: %s", table_name));
|
|
|
|
n_cols = form->s->fields;
|
|
|
|
/* We pass 0 as the space id, and determine at a lower level the space
|
|
id where to store the table */
|
|
|
|
table = dict_mem_table_create(table_name, 0, n_cols, flags);
|
|
|
|
if (path_of_temp_table) {
|
|
table->dir_path_of_temp_table =
|
|
mem_heap_strdup(table->heap, path_of_temp_table);
|
|
}
|
|
|
|
for (i = 0; i < n_cols; i++) {
|
|
field = form->field[i];
|
|
|
|
col_type = get_innobase_type_from_mysql_type(&unsigned_type,
|
|
field);
|
|
if (field->null_ptr) {
|
|
nulls_allowed = 0;
|
|
} else {
|
|
nulls_allowed = DATA_NOT_NULL;
|
|
}
|
|
|
|
if (field->binary()) {
|
|
binary_type = DATA_BINARY_TYPE;
|
|
} else {
|
|
binary_type = 0;
|
|
}
|
|
|
|
charset_no = 0;
|
|
|
|
if (dtype_is_string_type(col_type)) {
|
|
|
|
charset_no = (ulint)field->charset()->number;
|
|
|
|
ut_a(charset_no < 256); /* in data0type.h we assume
|
|
that the number fits in one
|
|
byte */
|
|
}
|
|
|
|
ut_a(field->type() < 256); /* we assume in dtype_form_prtype()
|
|
that this fits in one byte */
|
|
col_len = field->pack_length();
|
|
|
|
/* The MySQL pack length contains 1 or 2 bytes length field
|
|
for a true VARCHAR. Let us subtract that, so that the InnoDB
|
|
column length in the InnoDB data dictionary is the real
|
|
maximum byte length of the actual data. */
|
|
|
|
long_true_varchar = 0;
|
|
|
|
if (field->type() == MYSQL_TYPE_VARCHAR) {
|
|
col_len -= ((Field_varstring*)field)->length_bytes;
|
|
|
|
if (((Field_varstring*)field)->length_bytes == 2) {
|
|
long_true_varchar = DATA_LONG_TRUE_VARCHAR;
|
|
}
|
|
}
|
|
|
|
dict_mem_table_add_col(table, table->heap,
|
|
(char*) field->field_name,
|
|
col_type,
|
|
dtype_form_prtype(
|
|
(ulint)field->type()
|
|
| nulls_allowed | unsigned_type
|
|
| binary_type | long_true_varchar,
|
|
charset_no),
|
|
col_len);
|
|
}
|
|
|
|
error = row_create_table_for_mysql(table, trx);
|
|
|
|
/* We need access to the table and so we do the error checking
|
|
and set the error message here, before the error translation.*/
|
|
innodb_check_for_record_too_big_error(table, error);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates an index in an InnoDB database. */
|
|
static
|
|
int
|
|
create_index(
|
|
/*=========*/
|
|
trx_t* trx, /* in: InnoDB transaction handle */
|
|
TABLE* form, /* in: information on table
|
|
columns and indexes */
|
|
const char* table_name, /* in: table name */
|
|
uint key_num) /* in: index number */
|
|
{
|
|
Field* field;
|
|
dict_index_t* index;
|
|
int error;
|
|
ulint n_fields;
|
|
KEY* key;
|
|
KEY_PART_INFO* key_part;
|
|
ulint ind_type;
|
|
ulint col_type;
|
|
ulint prefix_len;
|
|
ulint is_unsigned;
|
|
ulint i;
|
|
ulint j;
|
|
ulint* field_lengths;
|
|
|
|
DBUG_ENTER("create_index");
|
|
|
|
key = form->key_info + key_num;
|
|
|
|
n_fields = key->key_parts;
|
|
|
|
ind_type = 0;
|
|
|
|
if (key_num == form->s->primary_key) {
|
|
ind_type = ind_type | DICT_CLUSTERED;
|
|
}
|
|
|
|
if (key->flags & HA_NOSAME ) {
|
|
ind_type = ind_type | DICT_UNIQUE;
|
|
}
|
|
|
|
/* We pass 0 as the space id, and determine at a lower level the space
|
|
id where to store the table */
|
|
|
|
index = dict_mem_index_create((char*) table_name, key->name, 0,
|
|
ind_type, n_fields);
|
|
|
|
field_lengths = (ulint*) my_malloc(sizeof(ulint) * n_fields,
|
|
MYF(MY_FAE));
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
key_part = key->key_part + i;
|
|
|
|
/* (The flag HA_PART_KEY_SEG denotes in MySQL a column prefix
|
|
field in an index: we only store a specified number of first
|
|
bytes of the column to the index field.) The flag does not
|
|
seem to be properly set by MySQL. Let us fall back on testing
|
|
the length of the key part versus the column. */
|
|
|
|
field = NULL;
|
|
for (j = 0; j < form->s->fields; j++) {
|
|
|
|
field = form->field[j];
|
|
|
|
if (0 == innobase_strcasecmp(
|
|
field->field_name,
|
|
key_part->field->field_name)) {
|
|
/* Found the corresponding column */
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
ut_a(j < form->s->fields);
|
|
|
|
col_type = get_innobase_type_from_mysql_type(
|
|
&is_unsigned, key_part->field);
|
|
|
|
if (DATA_BLOB == col_type
|
|
|| (key_part->length < field->pack_length()
|
|
&& field->type() != MYSQL_TYPE_VARCHAR)
|
|
|| (field->type() == MYSQL_TYPE_VARCHAR
|
|
&& key_part->length < field->pack_length()
|
|
- ((Field_varstring*)field)->length_bytes)) {
|
|
|
|
prefix_len = key_part->length;
|
|
|
|
if (col_type == DATA_INT
|
|
|| col_type == DATA_FLOAT
|
|
|| col_type == DATA_DOUBLE
|
|
|| col_type == DATA_DECIMAL) {
|
|
sql_print_error(
|
|
"MySQL is trying to create a column "
|
|
"prefix index field, on an "
|
|
"inappropriate data type. Table "
|
|
"name %s, column name %s.",
|
|
table_name,
|
|
key_part->field->field_name);
|
|
|
|
prefix_len = 0;
|
|
}
|
|
} else {
|
|
prefix_len = 0;
|
|
}
|
|
|
|
field_lengths[i] = key_part->length;
|
|
|
|
dict_mem_index_add_field(index,
|
|
(char*) key_part->field->field_name, prefix_len);
|
|
}
|
|
|
|
/* Even though we've defined max_supported_key_part_length, we
|
|
still do our own checking using field_lengths to be absolutely
|
|
sure we don't create too long indexes. */
|
|
error = row_create_index_for_mysql(index, trx, field_lengths);
|
|
|
|
/* We need access to the table and so we do the error checking
|
|
and set the error message here, before the error translation.*/
|
|
innodb_check_for_record_too_big_error(index->table, error);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
my_free((gptr) field_lengths, MYF(0));
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates an index to an InnoDB table when the user has defined no
|
|
primary index. */
|
|
static
|
|
int
|
|
create_clustered_index_when_no_primary(
|
|
/*===================================*/
|
|
trx_t* trx, /* in: InnoDB transaction handle */
|
|
const char* table_name) /* in: table name */
|
|
{
|
|
dict_index_t* index;
|
|
int error;
|
|
|
|
/* We pass 0 as the space id, and determine at a lower level the space
|
|
id where to store the table */
|
|
|
|
index = dict_mem_index_create((char*) table_name,
|
|
(char*) "GEN_CLUST_INDEX", 0, DICT_CLUSTERED, 0);
|
|
|
|
error = row_create_index_for_mysql(index, trx, NULL);
|
|
|
|
/* We need access to the table and so we do the error checking
|
|
and set the error message here, before the error translation.*/
|
|
innodb_check_for_record_too_big_error(index->table, error);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
return(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Update create_info. Used in SHOW CREATE TABLE et al. */
|
|
|
|
void
|
|
ha_innobase::update_create_info(
|
|
/*============================*/
|
|
HA_CREATE_INFO* create_info) /* in/out: create info */
|
|
{
|
|
if (!(create_info->used_fields & HA_CREATE_USED_AUTO)) {
|
|
ha_innobase::info(HA_STATUS_AUTO);
|
|
create_info->auto_increment_value = stats.auto_increment_value;
|
|
}
|
|
}
|
|
|
|
/*********************************************************************
|
|
Creates a new table to an InnoDB database. */
|
|
|
|
int
|
|
ha_innobase::create(
|
|
/*================*/
|
|
/* out: error number */
|
|
const char* name, /* in: table name */
|
|
TABLE* form, /* in: information on table
|
|
columns and indexes */
|
|
HA_CREATE_INFO* create_info) /* in: more information of the
|
|
created table, contains also the
|
|
create statement string */
|
|
{
|
|
int error;
|
|
dict_table_t* innobase_table;
|
|
trx_t* parent_trx;
|
|
trx_t* trx;
|
|
int primary_key_no;
|
|
uint i;
|
|
char name2[FN_REFLEN];
|
|
char norm_name[FN_REFLEN];
|
|
THD* thd = ha_thd();
|
|
ib_longlong auto_inc_value;
|
|
ulint flags;
|
|
|
|
DBUG_ENTER("ha_innobase::create");
|
|
|
|
DBUG_ASSERT(thd != NULL);
|
|
|
|
if (form->s->fields > 1000) {
|
|
/* The limit probably should be REC_MAX_N_FIELDS - 3 = 1020,
|
|
but we play safe here */
|
|
|
|
DBUG_RETURN(HA_ERR_TO_BIG_ROW);
|
|
}
|
|
|
|
/* Get the transaction associated with the current thd, or create one
|
|
if not yet created */
|
|
|
|
parent_trx = check_trx_exists(thd);
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(parent_trx);
|
|
|
|
trx = trx_allocate_for_mysql();
|
|
|
|
trx->mysql_thd = thd;
|
|
trx->mysql_query_str = thd_query(thd);
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_RELAXED_UNIQUE_CHECKS)) {
|
|
trx->check_unique_secondary = FALSE;
|
|
}
|
|
|
|
if (lower_case_table_names) {
|
|
srv_lower_case_table_names = TRUE;
|
|
} else {
|
|
srv_lower_case_table_names = FALSE;
|
|
}
|
|
|
|
strcpy(name2, name);
|
|
|
|
normalize_table_name(norm_name, name2);
|
|
|
|
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
|
|
or lock waits can happen in it during a table create operation.
|
|
Drop table etc. do this latching in row0mysql.c. */
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
/* Create the table definition in InnoDB */
|
|
|
|
flags = 0;
|
|
|
|
if (form->s->row_type != ROW_TYPE_REDUNDANT) {
|
|
flags |= DICT_TF_COMPACT;
|
|
|
|
switch (create_info->key_block_size) {
|
|
case 1: case 2: case 4: case 8: case 16:
|
|
flags |= create_info->key_block_size
|
|
<< DICT_TF_COMPRESSED_SHIFT;
|
|
}
|
|
}
|
|
|
|
error = create_table_def(trx, form, norm_name,
|
|
create_info->options & HA_LEX_CREATE_TMP_TABLE ? name2 : NULL,
|
|
flags);
|
|
|
|
if (error) {
|
|
goto cleanup;
|
|
}
|
|
|
|
/* Look for a primary key */
|
|
|
|
primary_key_no= (form->s->primary_key != MAX_KEY ?
|
|
(int) form->s->primary_key :
|
|
-1);
|
|
|
|
/* Our function row_get_mysql_key_number_for_index assumes
|
|
the primary key is always number 0, if it exists */
|
|
|
|
DBUG_ASSERT(primary_key_no == -1 || primary_key_no == 0);
|
|
|
|
/* Create the keys */
|
|
|
|
if (form->s->keys == 0 || primary_key_no == -1) {
|
|
/* Create an index which is used as the clustered index;
|
|
order the rows by their row id which is internally generated
|
|
by InnoDB */
|
|
|
|
error = create_clustered_index_when_no_primary(trx,
|
|
norm_name);
|
|
if (error) {
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
if (primary_key_no != -1) {
|
|
/* In InnoDB the clustered index must always be created
|
|
first */
|
|
if ((error = create_index(trx, form, norm_name,
|
|
(uint) primary_key_no))) {
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < form->s->keys; i++) {
|
|
|
|
if (i != (uint) primary_key_no) {
|
|
|
|
if ((error = create_index(trx, form, norm_name, i))) {
|
|
goto cleanup;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (*trx->mysql_query_str) {
|
|
error = row_table_add_foreign_constraints(trx,
|
|
*trx->mysql_query_str, norm_name,
|
|
create_info->options & HA_LEX_CREATE_TMP_TABLE);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
if (error) {
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
innobase_commit_low(trx);
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
innobase_table = dict_table_get(norm_name, FALSE);
|
|
|
|
DBUG_ASSERT(innobase_table != 0);
|
|
|
|
if ((create_info->used_fields & HA_CREATE_USED_AUTO) &&
|
|
(create_info->auto_increment_value != 0)) {
|
|
|
|
/* Query was ALTER TABLE...AUTO_INCREMENT = x; or
|
|
CREATE TABLE ...AUTO_INCREMENT = x; Find out a table
|
|
definition from the dictionary and get the current value
|
|
of the auto increment field. Set a new value to the
|
|
auto increment field if the value is greater than the
|
|
maximum value in the column. */
|
|
|
|
auto_inc_value = create_info->auto_increment_value;
|
|
dict_table_autoinc_initialize(innobase_table, auto_inc_value);
|
|
}
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
trx_free_for_mysql(trx);
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
cleanup:
|
|
innobase_commit_low(trx);
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
trx_free_for_mysql(trx);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Discards or imports an InnoDB tablespace. */
|
|
|
|
int
|
|
ha_innobase::discard_or_import_tablespace(
|
|
/*======================================*/
|
|
/* out: 0 == success, -1 == error */
|
|
my_bool discard) /* in: TRUE if discard, else import */
|
|
{
|
|
dict_table_t* dict_table;
|
|
trx_t* trx;
|
|
int err;
|
|
|
|
DBUG_ENTER("ha_innobase::discard_or_import_tablespace");
|
|
|
|
ut_a(prebuilt->trx);
|
|
ut_a(prebuilt->trx->magic_n == TRX_MAGIC_N);
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
dict_table = prebuilt->table;
|
|
trx = prebuilt->trx;
|
|
|
|
if (discard) {
|
|
err = row_discard_tablespace_for_mysql(dict_table->name, trx);
|
|
} else {
|
|
err = row_import_tablespace_for_mysql(dict_table->name, trx);
|
|
}
|
|
|
|
err = convert_error_code_to_mysql(err, NULL);
|
|
|
|
DBUG_RETURN(err);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Deletes all rows of an InnoDB table. */
|
|
|
|
int
|
|
ha_innobase::delete_all_rows(void)
|
|
/*==============================*/
|
|
/* out: error number */
|
|
{
|
|
int error;
|
|
|
|
DBUG_ENTER("ha_innobase::delete_all_rows");
|
|
|
|
/* Get the transaction associated with the current thd, or create one
|
|
if not yet created, and update prebuilt->trx */
|
|
|
|
update_thd(ha_thd());
|
|
|
|
if (thd_sql_command(user_thd) != SQLCOM_TRUNCATE) {
|
|
fallback:
|
|
/* We only handle TRUNCATE TABLE t as a special case.
|
|
DELETE FROM t will have to use ha_innobase::delete_row(). */
|
|
DBUG_RETURN(my_errno=HA_ERR_WRONG_COMMAND);
|
|
}
|
|
|
|
/* Truncate the table in InnoDB */
|
|
|
|
error = row_truncate_table_for_mysql(prebuilt->table, prebuilt->trx);
|
|
if (error == DB_ERROR) {
|
|
/* Cannot truncate; resort to ha_innobase::delete_row() */
|
|
goto fallback;
|
|
}
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Drops a table from an InnoDB database. Before calling this function,
|
|
MySQL calls innobase_commit to commit the transaction of the current user.
|
|
Then the current user cannot have locks set on the table. Drop table
|
|
operation inside InnoDB will remove all locks any user has on the table
|
|
inside InnoDB. */
|
|
|
|
int
|
|
ha_innobase::delete_table(
|
|
/*======================*/
|
|
/* out: error number */
|
|
const char* name) /* in: table name */
|
|
{
|
|
ulint name_len;
|
|
int error;
|
|
trx_t* parent_trx;
|
|
trx_t* trx;
|
|
THD *thd = ha_thd();
|
|
char norm_name[1000];
|
|
|
|
DBUG_ENTER("ha_innobase::delete_table");
|
|
|
|
/* Get the transaction associated with the current thd, or create one
|
|
if not yet created */
|
|
|
|
parent_trx = check_trx_exists(thd);
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(parent_trx);
|
|
|
|
if (lower_case_table_names) {
|
|
srv_lower_case_table_names = TRUE;
|
|
} else {
|
|
srv_lower_case_table_names = FALSE;
|
|
}
|
|
|
|
trx = trx_allocate_for_mysql();
|
|
|
|
trx->mysql_thd = thd;
|
|
trx->mysql_query_str = thd_query(thd);
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_RELAXED_UNIQUE_CHECKS)) {
|
|
trx->check_unique_secondary = FALSE;
|
|
}
|
|
|
|
name_len = strlen(name);
|
|
|
|
ut_a(name_len < 1000);
|
|
|
|
/* Strangely, MySQL passes the table name without the '.frm'
|
|
extension, in contrast to ::create */
|
|
|
|
normalize_table_name(norm_name, name);
|
|
|
|
/* Drop the table in InnoDB */
|
|
|
|
error = row_drop_table_for_mysql(norm_name, trx,
|
|
thd_sql_command(thd)
|
|
== SQLCOM_DROP_DB);
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
innobase_commit_low(trx);
|
|
|
|
trx_free_for_mysql(trx);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Removes all tables in the named database inside InnoDB. */
|
|
static
|
|
void
|
|
innobase_drop_database(
|
|
/*===================*/
|
|
/* out: error number */
|
|
handlerton *hton, /* in: handlerton of Innodb */
|
|
char* path) /* in: database path; inside InnoDB the name
|
|
of the last directory in the path is used as
|
|
the database name: for example, in 'mysql/data/test'
|
|
the database name is 'test' */
|
|
{
|
|
ulint len = 0;
|
|
trx_t* parent_trx;
|
|
trx_t* trx;
|
|
char* ptr;
|
|
int error;
|
|
char* namebuf;
|
|
THD* thd = current_thd;
|
|
|
|
/* Get the transaction associated with the current thd, or create one
|
|
if not yet created */
|
|
|
|
parent_trx = check_trx_exists(thd);
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(parent_trx);
|
|
|
|
ptr = strend(path) - 2;
|
|
|
|
while (ptr >= path && *ptr != '\\' && *ptr != '/') {
|
|
ptr--;
|
|
len++;
|
|
}
|
|
|
|
ptr++;
|
|
namebuf = my_malloc((uint) len + 2, MYF(0));
|
|
|
|
memcpy(namebuf, ptr, len);
|
|
namebuf[len] = '/';
|
|
namebuf[len + 1] = '\0';
|
|
#ifdef __WIN__
|
|
innobase_casedn_str(namebuf);
|
|
#endif
|
|
trx = trx_allocate_for_mysql();
|
|
trx->mysql_thd = thd;
|
|
trx->mysql_query_str = thd_query(thd);
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
error = row_drop_database_for_mysql(namebuf, trx);
|
|
my_free(namebuf, MYF(0));
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
innobase_commit_low(trx);
|
|
trx_free_for_mysql(trx);
|
|
#ifdef NO_LONGER_INTERESTED_IN_DROP_DB_ERROR
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
return(error);
|
|
#else
|
|
return;
|
|
#endif
|
|
}
|
|
/*************************************************************************
|
|
Renames an InnoDB table. */
|
|
static
|
|
int
|
|
innobase_rename_table(
|
|
/*==================*/
|
|
/* out: 0 or error code */
|
|
trx_t* trx, /* in: transaction */
|
|
const char* from, /* in: old name of the table */
|
|
const char* to, /* in: new name of the table */
|
|
ibool commit_flag) /* in: if TRUE then commit */
|
|
{
|
|
int error;
|
|
char* norm_to;
|
|
char* norm_from;
|
|
|
|
if (lower_case_table_names) {
|
|
srv_lower_case_table_names = TRUE;
|
|
} else {
|
|
srv_lower_case_table_names = FALSE;
|
|
}
|
|
|
|
// Magic number 64 arbitrary
|
|
norm_to = (char*) my_malloc(strlen(to) + 64, MYF(0));
|
|
norm_from = (char*) my_malloc(strlen(from) + 64, MYF(0));
|
|
|
|
if (*to != TEMP_TABLE_PREFIX) {
|
|
normalize_table_name(norm_to, to);
|
|
} else {
|
|
strcpy(norm_to, to);
|
|
}
|
|
|
|
if (*from != TEMP_TABLE_PREFIX) {
|
|
normalize_table_name(norm_from, from);
|
|
} else {
|
|
strcpy(norm_from, from);
|
|
}
|
|
|
|
/* Serialize data dictionary operations with dictionary mutex:
|
|
no deadlocks can occur then in these operations */
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
error = row_rename_table_for_mysql(
|
|
norm_from, norm_to, trx, commit_flag);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
FILE* ef = dict_foreign_err_file;
|
|
|
|
fputs("InnoDB: Rename from old name ", ef);
|
|
|
|
if (*norm_from != TEMP_TABLE_PREFIX) {
|
|
ut_print_name(ef, trx, TRUE, norm_from);
|
|
} else {
|
|
fputs(norm_from, ef);
|
|
}
|
|
|
|
fputs(" to ", ef);
|
|
|
|
if (*norm_to != TEMP_TABLE_PREFIX) {
|
|
ut_print_name(ef, trx, TRUE, norm_to);
|
|
} else {
|
|
fputs(norm_to, ef);
|
|
}
|
|
|
|
fputs(" failed!\n", ef);
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
my_free(norm_to, MYF(0));
|
|
my_free(norm_from, MYF(0));
|
|
|
|
return error;
|
|
}
|
|
/*************************************************************************
|
|
Renames an InnoDB table. */
|
|
|
|
int
|
|
ha_innobase::rename_table(
|
|
/*======================*/
|
|
/* out: 0 or error code */
|
|
const char* from, /* in: old name of the table */
|
|
const char* to) /* in: new name of the table */
|
|
{
|
|
trx_t* trx;
|
|
int error;
|
|
trx_t* parent_trx;
|
|
THD* thd = ha_thd();
|
|
|
|
DBUG_ENTER("ha_innobase::rename_table");
|
|
|
|
/* Get the transaction associated with the current thd, or create one
|
|
if not yet created */
|
|
|
|
parent_trx = check_trx_exists(thd);
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(parent_trx);
|
|
|
|
trx = trx_allocate_for_mysql();
|
|
trx->mysql_thd = thd;
|
|
trx->mysql_query_str = thd_query(thd);
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
error = innobase_rename_table(trx, from, to, TRUE);
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
innobase_commit_low(trx);
|
|
trx_free_for_mysql(trx);
|
|
|
|
error = convert_error_code_to_mysql(error, NULL);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Estimates the number of index records in a range. */
|
|
|
|
ha_rows
|
|
ha_innobase::records_in_range(
|
|
/*==========================*/
|
|
/* out: estimated number of
|
|
rows */
|
|
uint keynr, /* in: index number */
|
|
key_range *min_key, /* in: start key value of the
|
|
range, may also be 0 */
|
|
key_range *max_key) /* in: range end key val, may
|
|
also be 0 */
|
|
{
|
|
KEY* key;
|
|
dict_index_t* index;
|
|
mysql_byte* key_val_buff2 = (mysql_byte*) my_malloc(
|
|
table->s->reclength
|
|
+ table->s->max_key_length + 100,
|
|
MYF(MY_FAE));
|
|
ulint buff2_len = table->s->reclength
|
|
+ table->s->max_key_length + 100;
|
|
dtuple_t* range_start;
|
|
dtuple_t* range_end;
|
|
ib_longlong n_rows;
|
|
ulint mode1;
|
|
ulint mode2;
|
|
void* heap1;
|
|
void* heap2;
|
|
|
|
DBUG_ENTER("records_in_range");
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
prebuilt->trx->op_info = (char*)"estimating records in index range";
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
active_index = keynr;
|
|
|
|
key = table->key_info + active_index;
|
|
|
|
index = dict_table_get_index_on_name(prebuilt->table, key->name);
|
|
|
|
/* MySQL knows about this index and so we must be able to find it.*/
|
|
ut_a(index);
|
|
|
|
range_start = dtuple_create_for_mysql(&heap1, key->key_parts);
|
|
dict_index_copy_types(range_start, index, key->key_parts);
|
|
|
|
range_end = dtuple_create_for_mysql(&heap2, key->key_parts);
|
|
dict_index_copy_types(range_end, index, key->key_parts);
|
|
|
|
row_sel_convert_mysql_key_to_innobase(
|
|
range_start, (byte*) key_val_buff,
|
|
(ulint)upd_and_key_val_buff_len,
|
|
index,
|
|
(byte*) (min_key ? min_key->key :
|
|
(const mysql_byte*) 0),
|
|
(ulint) (min_key ? min_key->length : 0),
|
|
prebuilt->trx);
|
|
|
|
row_sel_convert_mysql_key_to_innobase(
|
|
range_end, (byte*) key_val_buff2,
|
|
buff2_len, index,
|
|
(byte*) (max_key ? max_key->key :
|
|
(const mysql_byte*) 0),
|
|
(ulint) (max_key ? max_key->length : 0),
|
|
prebuilt->trx);
|
|
|
|
mode1 = convert_search_mode_to_innobase(min_key ? min_key->flag :
|
|
HA_READ_KEY_EXACT);
|
|
mode2 = convert_search_mode_to_innobase(max_key ? max_key->flag :
|
|
HA_READ_KEY_EXACT);
|
|
|
|
n_rows = btr_estimate_n_rows_in_range(index, range_start,
|
|
mode1, range_end, mode2);
|
|
dtuple_free_for_mysql(heap1);
|
|
dtuple_free_for_mysql(heap2);
|
|
|
|
my_free((gptr) key_val_buff2, MYF(0));
|
|
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
/* The MySQL optimizer seems to believe an estimate of 0 rows is
|
|
always accurate and may return the result 'Empty set' based on that.
|
|
The accuracy is not guaranteed, and even if it were, for a locking
|
|
read we should anyway perform the search to set the next-key lock.
|
|
Add 1 to the value to make sure MySQL does not make the assumption! */
|
|
|
|
if (n_rows == 0) {
|
|
n_rows = 1;
|
|
}
|
|
|
|
DBUG_RETURN((ha_rows) n_rows);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Gives an UPPER BOUND to the number of rows in a table. This is used in
|
|
filesort.cc. */
|
|
|
|
ha_rows
|
|
ha_innobase::estimate_rows_upper_bound(void)
|
|
/*======================================*/
|
|
/* out: upper bound of rows */
|
|
{
|
|
dict_index_t* index;
|
|
ulonglong estimate;
|
|
ulonglong local_data_file_length;
|
|
|
|
DBUG_ENTER("estimate_rows_upper_bound");
|
|
|
|
/* We do not know if MySQL can call this function before calling
|
|
external_lock(). To be safe, update the thd of the current table
|
|
handle. */
|
|
|
|
update_thd(ha_thd());
|
|
|
|
prebuilt->trx->op_info = (char*)
|
|
"calculating upper bound for table rows";
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
index = dict_table_get_first_index_noninline(prebuilt->table);
|
|
|
|
ut_a(index->stat_n_leaf_pages > 0);
|
|
|
|
local_data_file_length =
|
|
((ulonglong) index->stat_n_leaf_pages) * UNIV_PAGE_SIZE;
|
|
|
|
|
|
/* Calculate a minimum length for a clustered index record and from
|
|
that an upper bound for the number of rows. Since we only calculate
|
|
new statistics in row0mysql.c when a table has grown by a threshold
|
|
factor, we must add a safety factor 2 in front of the formula below. */
|
|
|
|
estimate = 2 * local_data_file_length /
|
|
dict_index_calc_min_rec_len(index);
|
|
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
DBUG_RETURN((ha_rows) estimate);
|
|
}
|
|
|
|
/*************************************************************************
|
|
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_innobase::scan_time()
|
|
/*====================*/
|
|
/* out: estimated time measured in disk seeks */
|
|
{
|
|
/* Since MySQL seems to favor table scans too much over index
|
|
searches, we pretend that a sequential read takes the same time
|
|
as a random disk read, that is, we do not divide the following
|
|
by 10, which would be physically realistic. */
|
|
|
|
return((double) (prebuilt->table->stat_clustered_index_size));
|
|
}
|
|
|
|
/**********************************************************************
|
|
Calculate the time it takes to read a set of ranges through an index
|
|
This enables us to optimise reads for clustered indexes. */
|
|
|
|
double
|
|
ha_innobase::read_time(
|
|
/*===================*/
|
|
/* out: estimated time measured in disk seeks */
|
|
uint index, /* in: key number */
|
|
uint ranges, /* in: how many ranges */
|
|
ha_rows rows) /* in: estimated number of rows in the ranges */
|
|
{
|
|
ha_rows total_rows;
|
|
double time_for_scan;
|
|
|
|
if (index != table->s->primary_key) {
|
|
/* Not clustered */
|
|
return(handler::read_time(index, ranges, rows));
|
|
}
|
|
|
|
if (rows <= 2) {
|
|
|
|
return((double) rows);
|
|
}
|
|
|
|
/* Assume that the read time is proportional to the scan time for all
|
|
rows + at most one seek per range. */
|
|
|
|
time_for_scan = scan_time();
|
|
|
|
if ((total_rows = estimate_rows_upper_bound()) < rows) {
|
|
|
|
return(time_for_scan);
|
|
}
|
|
|
|
return(ranges + (double) rows / (double) total_rows * time_for_scan);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Returns statistics information of the table to the MySQL interpreter,
|
|
in various fields of the handle object. */
|
|
|
|
int
|
|
ha_innobase::info(
|
|
/*==============*/
|
|
uint flag) /* in: what information MySQL requests */
|
|
{
|
|
dict_table_t* ib_table;
|
|
dict_index_t* index;
|
|
ha_rows rec_per_key;
|
|
ib_longlong n_rows;
|
|
ulong j;
|
|
ulong i;
|
|
char path[FN_REFLEN];
|
|
os_file_stat_t stat_info;
|
|
|
|
DBUG_ENTER("info");
|
|
|
|
/* If we are forcing recovery at a high level, we will suppress
|
|
statistics calculation on tables, because that may crash the
|
|
server if an index is badly corrupted. */
|
|
|
|
if (srv_force_recovery >= SRV_FORCE_NO_IBUF_MERGE) {
|
|
|
|
DBUG_RETURN(HA_ERR_CRASHED);
|
|
}
|
|
|
|
/* We do not know if MySQL can call this function before calling
|
|
external_lock(). To be safe, update the thd of the current table
|
|
handle. */
|
|
|
|
update_thd(ha_thd());
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
prebuilt->trx->op_info = (char*)"returning various info to MySQL";
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
ib_table = prebuilt->table;
|
|
|
|
if (flag & HA_STATUS_TIME) {
|
|
if (srv_stats_on_metadata) {
|
|
/* In sql_show we call with this flag: update
|
|
then statistics so that they are up-to-date */
|
|
|
|
prebuilt->trx->op_info = "updating table statistics";
|
|
|
|
dict_update_statistics(ib_table);
|
|
|
|
prebuilt->trx->op_info = "returning various info to MySQL";
|
|
}
|
|
|
|
my_snprintf(path, sizeof(path), "%s/%s%s",
|
|
mysql_data_home, ib_table->name, reg_ext);
|
|
|
|
unpack_filename(path,path);
|
|
|
|
/* Note that we do not know the access time of the table,
|
|
nor the CHECK TABLE time, nor the UPDATE or INSERT time. */
|
|
|
|
if (os_file_get_status(path,&stat_info)) {
|
|
stats.create_time = stat_info.ctime;
|
|
}
|
|
}
|
|
|
|
if (flag & HA_STATUS_VARIABLE) {
|
|
n_rows = ib_table->stat_n_rows;
|
|
|
|
/* Because we do not protect stat_n_rows by any mutex in a
|
|
delete, it is theoretically possible that the value can be
|
|
smaller than zero! TODO: fix this race.
|
|
|
|
The MySQL optimizer seems to assume in a left join that n_rows
|
|
is an accurate estimate if it is zero. Of course, it is not,
|
|
since we do not have any locks on the rows yet at this phase.
|
|
Since SHOW TABLE STATUS seems to call this function with the
|
|
HA_STATUS_TIME flag set, while the left join optimizer does not
|
|
set that flag, we add one to a zero value if the flag is not
|
|
set. That way SHOW TABLE STATUS will show the best estimate,
|
|
while the optimizer never sees the table empty. */
|
|
|
|
if (n_rows < 0) {
|
|
n_rows = 0;
|
|
}
|
|
|
|
if (n_rows == 0 && !(flag & HA_STATUS_TIME)) {
|
|
n_rows++;
|
|
}
|
|
|
|
stats.records = (ha_rows)n_rows;
|
|
stats.deleted = 0;
|
|
stats.data_file_length = ((ulonglong)
|
|
ib_table->stat_clustered_index_size)
|
|
* UNIV_PAGE_SIZE;
|
|
stats.index_file_length = ((ulonglong)
|
|
ib_table->stat_sum_of_other_index_sizes)
|
|
* UNIV_PAGE_SIZE;
|
|
stats.delete_length = 0;
|
|
stats.check_time = 0;
|
|
|
|
if (stats.records == 0) {
|
|
stats.mean_rec_length = 0;
|
|
} else {
|
|
stats.mean_rec_length = (ulong) (stats.data_file_length / stats.records);
|
|
}
|
|
}
|
|
|
|
if (flag & HA_STATUS_CONST) {
|
|
index = dict_table_get_first_index_noninline(ib_table);
|
|
|
|
if (prebuilt->clust_index_was_generated) {
|
|
index = dict_table_get_next_index_noninline(index);
|
|
}
|
|
|
|
for (i = 0; i < table->s->keys; i++) {
|
|
if (index == NULL) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("Table %s contains fewer "
|
|
"indexes inside InnoDB than "
|
|
"are defined in the MySQL "
|
|
".frm file. Have you mixed up "
|
|
".frm files from different "
|
|
"installations? See "
|
|
"http://dev.mysql.com/doc/refman/5.1/en/innodb-troubleshooting.html\n",
|
|
|
|
ib_table->name);
|
|
break;
|
|
}
|
|
|
|
for (j = 0; j < table->key_info[i].key_parts; j++) {
|
|
|
|
if (j + 1 > index->n_uniq) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error(
|
|
"Index %s of %s has %lu columns unique inside InnoDB, but MySQL is asking "
|
|
"statistics for %lu columns. Have you mixed up .frm files from different "
|
|
"installations? "
|
|
"See http://dev.mysql.com/doc/refman/5.1/en/innodb-troubleshooting.html\n",
|
|
index->name,
|
|
ib_table->name,
|
|
(unsigned long)
|
|
index->n_uniq, j + 1);
|
|
break;
|
|
}
|
|
|
|
if (index->stat_n_diff_key_vals[j + 1] == 0) {
|
|
|
|
rec_per_key = stats.records;
|
|
} else {
|
|
rec_per_key = (ha_rows)(stats.records /
|
|
index->stat_n_diff_key_vals[j + 1]);
|
|
}
|
|
|
|
/* Since MySQL seems to favor table scans
|
|
too much over index searches, we pretend
|
|
index selectivity is 2 times better than
|
|
our estimate: */
|
|
|
|
rec_per_key = rec_per_key / 2;
|
|
|
|
if (rec_per_key == 0) {
|
|
rec_per_key = 1;
|
|
}
|
|
|
|
table->key_info[i].rec_per_key[j]=
|
|
rec_per_key >= ~(ulong) 0 ? ~(ulong) 0 :
|
|
rec_per_key;
|
|
}
|
|
|
|
index = dict_table_get_next_index_noninline(index);
|
|
}
|
|
}
|
|
|
|
if (flag & HA_STATUS_ERRKEY) {
|
|
dict_index_t* err_index;
|
|
|
|
ut_a(prebuilt->trx);
|
|
ut_a(prebuilt->trx->magic_n == TRX_MAGIC_N);
|
|
|
|
err_index = (dict_index_t*)trx_get_error_info(prebuilt->trx);
|
|
|
|
if (err_index) {
|
|
errkey = (unsigned int)
|
|
row_get_mysql_key_number_for_index(err_index);
|
|
} else {
|
|
errkey = (unsigned int) prebuilt->trx->error_key_num;
|
|
}
|
|
}
|
|
|
|
if (flag & HA_STATUS_AUTO && table->found_next_number_field) {
|
|
longlong auto_inc;
|
|
int ret;
|
|
|
|
/* The following function call can the first time fail in
|
|
a lock wait timeout error because it reserves the auto-inc
|
|
lock on the table. If it fails, then someone is already initing
|
|
the auto-inc counter, and the second call is guaranteed to
|
|
succeed. */
|
|
|
|
ret = innobase_read_and_init_auto_inc(&auto_inc);
|
|
|
|
if (ret != 0) {
|
|
ret = innobase_read_and_init_auto_inc(&auto_inc);
|
|
|
|
if (ret != 0) {
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("Cannot get table %s auto-inc"
|
|
"counter value in ::info\n",
|
|
ib_table->name);
|
|
auto_inc = 0;
|
|
}
|
|
}
|
|
|
|
stats.auto_increment_value = auto_inc;
|
|
}
|
|
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/**************************************************************************
|
|
Updates index cardinalities of the table, based on 8 random dives into
|
|
each index tree. This does NOT calculate exact statistics on the table. */
|
|
|
|
int
|
|
ha_innobase::analyze(
|
|
/*=================*/
|
|
/* out: returns always 0 (success) */
|
|
THD* thd, /* in: connection thread handle */
|
|
HA_CHECK_OPT* check_opt) /* in: currently ignored */
|
|
{
|
|
/* Simply call ::info() with all the flags */
|
|
info(HA_STATUS_TIME | HA_STATUS_CONST | HA_STATUS_VARIABLE);
|
|
|
|
return(0);
|
|
}
|
|
|
|
/**************************************************************************
|
|
This is mapped to "ALTER TABLE tablename ENGINE=InnoDB", which rebuilds
|
|
the table in MySQL. */
|
|
|
|
int
|
|
ha_innobase::optimize(
|
|
/*==================*/
|
|
THD* thd, /* in: connection thread handle */
|
|
HA_CHECK_OPT* check_opt) /* in: currently ignored */
|
|
{
|
|
return(HA_ADMIN_TRY_ALTER);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Tries to check that an InnoDB table is not corrupted. If corruption is
|
|
noticed, prints to stderr information about it. In case of corruption
|
|
may also assert a failure and crash the server. */
|
|
|
|
int
|
|
ha_innobase::check(
|
|
/*===============*/
|
|
/* out: HA_ADMIN_CORRUPT or
|
|
HA_ADMIN_OK */
|
|
THD* thd, /* in: user thread handle */
|
|
HA_CHECK_OPT* check_opt) /* in: check options, currently
|
|
ignored */
|
|
{
|
|
ulint ret;
|
|
|
|
DBUG_ASSERT(thd == ha_thd());
|
|
ut_a(prebuilt->trx);
|
|
ut_a(prebuilt->trx->magic_n == TRX_MAGIC_N);
|
|
ut_a(prebuilt->trx == thd_to_trx(thd));
|
|
|
|
if (prebuilt->mysql_template == NULL) {
|
|
/* Build the template; we will use a dummy template
|
|
in index scans done in checking */
|
|
|
|
build_template(prebuilt, NULL, table, ROW_MYSQL_WHOLE_ROW);
|
|
}
|
|
|
|
ret = row_check_table_for_mysql(prebuilt);
|
|
|
|
if (ret == DB_SUCCESS) {
|
|
return(HA_ADMIN_OK);
|
|
}
|
|
|
|
return(HA_ADMIN_CORRUPT);
|
|
}
|
|
|
|
/*****************************************************************
|
|
Adds information about free space in the InnoDB tablespace to a table comment
|
|
which is printed out when a user calls SHOW TABLE STATUS. Adds also info on
|
|
foreign keys. */
|
|
|
|
char*
|
|
ha_innobase::update_table_comment(
|
|
/*==============================*/
|
|
/* out: table comment + InnoDB free space +
|
|
info on foreign keys */
|
|
const char* comment)/* in: table comment defined by user */
|
|
{
|
|
uint length = (uint) strlen(comment);
|
|
char* str;
|
|
long flen;
|
|
|
|
/* We do not know if MySQL can call this function before calling
|
|
external_lock(). To be safe, update the thd of the current table
|
|
handle. */
|
|
|
|
if (length > 64000 - 3) {
|
|
return((char*)comment); /* string too long */
|
|
}
|
|
|
|
update_thd(ha_thd());
|
|
|
|
prebuilt->trx->op_info = (char*)"returning table comment";
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
str = NULL;
|
|
|
|
/* output the data to a temporary file */
|
|
|
|
mutex_enter_noninline(&srv_dict_tmpfile_mutex);
|
|
rewind(srv_dict_tmpfile);
|
|
|
|
fprintf(srv_dict_tmpfile, "InnoDB free: %lu kB",
|
|
(ulong) fsp_get_available_space_in_free_extents(
|
|
prebuilt->table->space));
|
|
|
|
dict_print_info_on_foreign_keys(FALSE, srv_dict_tmpfile,
|
|
prebuilt->trx, prebuilt->table);
|
|
flen = ftell(srv_dict_tmpfile);
|
|
if (flen < 0) {
|
|
flen = 0;
|
|
} else if (length + flen + 3 > 64000) {
|
|
flen = 64000 - 3 - length;
|
|
}
|
|
|
|
/* allocate buffer for the full string, and
|
|
read the contents of the temporary file */
|
|
|
|
str = my_malloc(length + flen + 3, MYF(0));
|
|
|
|
if (str) {
|
|
char* pos = str + length;
|
|
if (length) {
|
|
memcpy(str, comment, length);
|
|
*pos++ = ';';
|
|
*pos++ = ' ';
|
|
}
|
|
rewind(srv_dict_tmpfile);
|
|
flen = (uint) fread(pos, 1, flen, srv_dict_tmpfile);
|
|
pos[flen] = 0;
|
|
}
|
|
|
|
mutex_exit_noninline(&srv_dict_tmpfile_mutex);
|
|
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
return(str ? str : (char*) comment);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Gets the foreign key create info for a table stored in InnoDB. */
|
|
|
|
char*
|
|
ha_innobase::get_foreign_key_create_info(void)
|
|
/*==========================================*/
|
|
/* out, own: character string in the form which
|
|
can be inserted to the CREATE TABLE statement,
|
|
MUST be freed with ::free_foreign_key_create_info */
|
|
{
|
|
char* str = 0;
|
|
long flen;
|
|
|
|
ut_a(prebuilt != NULL);
|
|
|
|
/* We do not know if MySQL can call this function before calling
|
|
external_lock(). To be safe, update the thd of the current table
|
|
handle. */
|
|
|
|
update_thd(ha_thd());
|
|
|
|
prebuilt->trx->op_info = (char*)"getting info on foreign keys";
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query,
|
|
release possible adaptive hash latch to avoid
|
|
deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
mutex_enter_noninline(&srv_dict_tmpfile_mutex);
|
|
rewind(srv_dict_tmpfile);
|
|
|
|
/* output the data to a temporary file */
|
|
dict_print_info_on_foreign_keys(TRUE, srv_dict_tmpfile,
|
|
prebuilt->trx, prebuilt->table);
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
flen = ftell(srv_dict_tmpfile);
|
|
if (flen < 0) {
|
|
flen = 0;
|
|
} else if (flen > 64000 - 1) {
|
|
flen = 64000 - 1;
|
|
}
|
|
|
|
/* allocate buffer for the string, and
|
|
read the contents of the temporary file */
|
|
|
|
str = my_malloc(flen + 1, MYF(0));
|
|
|
|
if (str) {
|
|
rewind(srv_dict_tmpfile);
|
|
flen = (uint) fread(str, 1, flen, srv_dict_tmpfile);
|
|
str[flen] = 0;
|
|
}
|
|
|
|
mutex_exit_noninline(&srv_dict_tmpfile_mutex);
|
|
|
|
return(str);
|
|
}
|
|
|
|
|
|
int
|
|
ha_innobase::get_foreign_key_list(THD *thd, List<FOREIGN_KEY_INFO> *f_key_list)
|
|
{
|
|
dict_foreign_t* foreign;
|
|
|
|
DBUG_ENTER("get_foreign_key_list");
|
|
ut_a(prebuilt != NULL);
|
|
update_thd(ha_thd());
|
|
prebuilt->trx->op_info = (char*)"getting list of foreign keys";
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
mutex_enter_noninline(&(dict_sys->mutex));
|
|
foreign = UT_LIST_GET_FIRST(prebuilt->table->foreign_list);
|
|
|
|
while (foreign != NULL) {
|
|
uint i;
|
|
FOREIGN_KEY_INFO f_key_info;
|
|
LEX_STRING *name= 0;
|
|
uint ulen;
|
|
char uname[NAME_LEN+1]; /* Unencoded name */
|
|
char db_name[NAME_LEN+1];
|
|
const char *tmp_buff;
|
|
|
|
tmp_buff= foreign->id;
|
|
i= 0;
|
|
while (tmp_buff[i] != '/')
|
|
i++;
|
|
tmp_buff+= i + 1;
|
|
f_key_info.forein_id= make_lex_string(thd, 0, tmp_buff,
|
|
(uint) strlen(tmp_buff), 1);
|
|
tmp_buff= foreign->referenced_table_name;
|
|
|
|
/* Database name */
|
|
i= 0;
|
|
while (tmp_buff[i] != '/')
|
|
{
|
|
db_name[i]= tmp_buff[i];
|
|
i++;
|
|
}
|
|
db_name[i]= 0;
|
|
ulen= filename_to_tablename(db_name, uname, sizeof(uname));
|
|
f_key_info.referenced_db= make_lex_string(thd, 0, uname, ulen, 1);
|
|
|
|
/* Table name */
|
|
tmp_buff+= i + 1;
|
|
ulen= filename_to_tablename(tmp_buff, uname, sizeof(uname));
|
|
f_key_info.referenced_table= make_lex_string(thd, 0, uname,
|
|
ulen, 1);
|
|
|
|
for (i= 0;;) {
|
|
tmp_buff= foreign->foreign_col_names[i];
|
|
name= make_lex_string(thd, name, tmp_buff,
|
|
(uint) strlen(tmp_buff), 1);
|
|
f_key_info.foreign_fields.push_back(name);
|
|
tmp_buff= foreign->referenced_col_names[i];
|
|
name= make_lex_string(thd, name, tmp_buff,
|
|
(uint) strlen(tmp_buff), 1);
|
|
f_key_info.referenced_fields.push_back(name);
|
|
if (++i >= foreign->n_fields)
|
|
break;
|
|
}
|
|
|
|
ulong length;
|
|
if (foreign->type & DICT_FOREIGN_ON_DELETE_CASCADE)
|
|
{
|
|
length=7;
|
|
tmp_buff= "CASCADE";
|
|
}
|
|
else if (foreign->type & DICT_FOREIGN_ON_DELETE_SET_NULL)
|
|
{
|
|
length=8;
|
|
tmp_buff= "SET NULL";
|
|
}
|
|
else if (foreign->type & DICT_FOREIGN_ON_DELETE_NO_ACTION)
|
|
{
|
|
length=9;
|
|
tmp_buff= "NO ACTION";
|
|
}
|
|
else
|
|
{
|
|
length=8;
|
|
tmp_buff= "RESTRICT";
|
|
}
|
|
f_key_info.delete_method= make_lex_string(thd, f_key_info.delete_method,
|
|
tmp_buff, length, 1);
|
|
|
|
|
|
if (foreign->type & DICT_FOREIGN_ON_UPDATE_CASCADE)
|
|
{
|
|
length=7;
|
|
tmp_buff= "CASCADE";
|
|
}
|
|
else if (foreign->type & DICT_FOREIGN_ON_UPDATE_SET_NULL)
|
|
{
|
|
length=8;
|
|
tmp_buff= "SET NULL";
|
|
}
|
|
else if (foreign->type & DICT_FOREIGN_ON_UPDATE_NO_ACTION)
|
|
{
|
|
length=9;
|
|
tmp_buff= "NO ACTION";
|
|
}
|
|
else
|
|
{
|
|
length=8;
|
|
tmp_buff= "RESTRICT";
|
|
}
|
|
f_key_info.update_method= make_lex_string(thd, f_key_info.update_method,
|
|
tmp_buff, length, 1);
|
|
if (foreign->referenced_index &&
|
|
foreign->referenced_index->name)
|
|
{
|
|
f_key_info.referenced_key_name=
|
|
make_lex_string(thd, f_key_info.referenced_key_name,
|
|
foreign->referenced_index->name,
|
|
strlen(foreign->referenced_index->name), 1);
|
|
}
|
|
|
|
FOREIGN_KEY_INFO *pf_key_info = (FOREIGN_KEY_INFO *)
|
|
thd_memdup(thd, (gptr) &f_key_info, sizeof f_key_info);
|
|
f_key_list->push_back(pf_key_info);
|
|
foreign = UT_LIST_GET_NEXT(foreign_list, foreign);
|
|
}
|
|
mutex_exit_noninline(&(dict_sys->mutex));
|
|
prebuilt->trx->op_info = (char*)"";
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Checks if ALTER TABLE may change the storage engine of the table.
|
|
Changing storage engines is not allowed for tables for which there
|
|
are foreign key constraints (parent or child tables). */
|
|
|
|
bool
|
|
ha_innobase::can_switch_engines(void)
|
|
/*=================================*/
|
|
{
|
|
bool can_switch;
|
|
|
|
DBUG_ENTER("ha_innobase::can_switch_engines");
|
|
|
|
ut_a(prebuilt->trx == thd_to_trx(ha_thd()));
|
|
|
|
prebuilt->trx->op_info =
|
|
"determining if there are foreign key constraints";
|
|
row_mysql_lock_data_dictionary(prebuilt->trx);
|
|
|
|
can_switch = !UT_LIST_GET_FIRST(prebuilt->table->referenced_list)
|
|
&& !UT_LIST_GET_FIRST(prebuilt->table->foreign_list);
|
|
|
|
row_mysql_unlock_data_dictionary(prebuilt->trx);
|
|
prebuilt->trx->op_info = "";
|
|
|
|
DBUG_RETURN(can_switch);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Checks if a table is referenced by a foreign key. The MySQL manual states that
|
|
a REPLACE is either equivalent to an INSERT, or DELETE(s) + INSERT. Only a
|
|
delete is then allowed internally to resolve a duplicate key conflict in
|
|
REPLACE, not an update. */
|
|
|
|
uint
|
|
ha_innobase::referenced_by_foreign_key(void)
|
|
/*========================================*/
|
|
/* out: > 0 if referenced by a FOREIGN KEY */
|
|
{
|
|
if (dict_table_is_referenced_by_foreign_key(prebuilt->table)) {
|
|
|
|
return(1);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Frees the foreign key create info for a table stored in InnoDB, if it is
|
|
non-NULL. */
|
|
|
|
void
|
|
ha_innobase::free_foreign_key_create_info(
|
|
/*======================================*/
|
|
char* str) /* in, own: create info string to free */
|
|
{
|
|
if (str) {
|
|
my_free(str, MYF(0));
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
Tells something additional to the handler about how to do things. */
|
|
|
|
int
|
|
ha_innobase::extra(
|
|
/*===============*/
|
|
/* out: 0 or error number */
|
|
enum ha_extra_function operation)
|
|
/* in: HA_EXTRA_FLUSH or some other flag */
|
|
{
|
|
/* Warning: since it is not sure that MySQL calls external_lock
|
|
before calling this function, the trx field in prebuilt can be
|
|
obsolete! */
|
|
|
|
switch (operation) {
|
|
case HA_EXTRA_FLUSH:
|
|
if (prebuilt->blob_heap) {
|
|
row_mysql_prebuilt_free_blob_heap(prebuilt);
|
|
}
|
|
break;
|
|
case HA_EXTRA_RESET_STATE:
|
|
reset_template(prebuilt);
|
|
break;
|
|
case HA_EXTRA_NO_KEYREAD:
|
|
prebuilt->read_just_key = 0;
|
|
break;
|
|
case HA_EXTRA_KEYREAD:
|
|
prebuilt->read_just_key = 1;
|
|
break;
|
|
case HA_EXTRA_KEYREAD_PRESERVE_FIELDS:
|
|
prebuilt->keep_other_fields_on_keyread = 1;
|
|
break;
|
|
|
|
/* IMPORTANT: prebuilt->trx can be obsolete in
|
|
this method, because it is not sure that MySQL
|
|
calls external_lock before this method with the
|
|
parameters below. We must not invoke update_thd()
|
|
either, because the calling threads may change.
|
|
CAREFUL HERE, OR MEMORY CORRUPTION MAY OCCUR! */
|
|
case HA_EXTRA_IGNORE_DUP_KEY:
|
|
thd_to_trx(ha_thd())->duplicates |= TRX_DUP_IGNORE;
|
|
break;
|
|
case HA_EXTRA_WRITE_CAN_REPLACE:
|
|
thd_to_trx(ha_thd())->duplicates |= TRX_DUP_REPLACE;
|
|
break;
|
|
case HA_EXTRA_WRITE_CANNOT_REPLACE:
|
|
thd_to_trx(ha_thd())->duplicates &= ~TRX_DUP_REPLACE;
|
|
break;
|
|
case HA_EXTRA_NO_IGNORE_DUP_KEY:
|
|
thd_to_trx(ha_thd())->duplicates &=
|
|
~(TRX_DUP_IGNORE | TRX_DUP_REPLACE);
|
|
break;
|
|
default:/* Do nothing */
|
|
;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
int ha_innobase::reset()
|
|
{
|
|
if (prebuilt->blob_heap) {
|
|
row_mysql_prebuilt_free_blob_heap(prebuilt);
|
|
}
|
|
reset_template(prebuilt);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
MySQL calls this function at the start of each SQL statement inside LOCK
|
|
TABLES. Inside LOCK TABLES the ::external_lock method does not work to
|
|
mark SQL statement borders. Note also a special case: if a temporary table
|
|
is created inside LOCK TABLES, MySQL has not called external_lock() at all
|
|
on that table.
|
|
MySQL-5.0 also calls this before each statement in an execution of a stored
|
|
procedure. To make the execution more deterministic for binlogging, MySQL-5.0
|
|
locks all tables involved in a stored procedure with full explicit table
|
|
locks (thd_in_lock_tables(thd) holds in store_lock()) before executing the
|
|
procedure. */
|
|
|
|
int
|
|
ha_innobase::start_stmt(
|
|
/*====================*/
|
|
/* out: 0 or error code */
|
|
THD* thd, /* in: handle to the user thread */
|
|
thr_lock_type lock_type)
|
|
{
|
|
trx_t* trx;
|
|
|
|
update_thd(thd);
|
|
|
|
trx = prebuilt->trx;
|
|
|
|
/* Here we release the search latch and the InnoDB thread FIFO ticket
|
|
if they were reserved. They should have been released already at the
|
|
end of the previous statement, but because inside LOCK TABLES the
|
|
lock count method does not work to mark the end of a SELECT statement,
|
|
that may not be the case. We MUST release the search latch before an
|
|
INSERT, for example. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
prebuilt->sql_stat_start = TRUE;
|
|
prebuilt->hint_need_to_fetch_extra_cols = 0;
|
|
reset_template(prebuilt);
|
|
|
|
if (!prebuilt->mysql_has_locked) {
|
|
/* This handle is for a temporary table created inside
|
|
this same LOCK TABLES; since MySQL does NOT call external_lock
|
|
in this case, we must use x-row locks inside InnoDB to be
|
|
prepared for an update of a row */
|
|
|
|
prebuilt->select_lock_type = LOCK_X;
|
|
} else {
|
|
if (trx->isolation_level != TRX_ISO_SERIALIZABLE
|
|
&& thd_sql_command(thd) == SQLCOM_SELECT
|
|
&& lock_type == TL_READ) {
|
|
|
|
/* For other than temporary tables, we obtain
|
|
no lock for consistent read (plain SELECT). */
|
|
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
} else {
|
|
/* Not a consistent read: restore the
|
|
select_lock_type value. The value of
|
|
stored_select_lock_type was decided in:
|
|
1) ::store_lock(),
|
|
2) ::external_lock(),
|
|
3) ::init_table_handle_for_HANDLER(), and
|
|
4) ::transactional_table_lock(). */
|
|
|
|
prebuilt->select_lock_type =
|
|
prebuilt->stored_select_lock_type;
|
|
}
|
|
}
|
|
|
|
trx->detailed_error[0] = '\0';
|
|
|
|
/* Set the MySQL flag to mark that there is an active transaction */
|
|
if (trx->active_trans == 0) {
|
|
|
|
innobase_register_trx_and_stmt(ht, thd);
|
|
trx->active_trans = 1;
|
|
} else {
|
|
innobase_register_stmt(ht, thd);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Maps a MySQL trx isolation level code to the InnoDB isolation level code */
|
|
inline
|
|
ulint
|
|
innobase_map_isolation_level(
|
|
/*=========================*/
|
|
/* out: InnoDB isolation level */
|
|
enum_tx_isolation iso) /* in: MySQL isolation level code */
|
|
{
|
|
switch(iso) {
|
|
case ISO_REPEATABLE_READ: return(TRX_ISO_REPEATABLE_READ);
|
|
case ISO_READ_COMMITTED: return(TRX_ISO_READ_COMMITTED);
|
|
case ISO_SERIALIZABLE: return(TRX_ISO_SERIALIZABLE);
|
|
case ISO_READ_UNCOMMITTED: return(TRX_ISO_READ_UNCOMMITTED);
|
|
default: ut_a(0); return(0);
|
|
}
|
|
}
|
|
|
|
/**********************************************************************
|
|
As MySQL will execute an external lock for every new table it uses when it
|
|
starts to process an SQL statement (an exception is when MySQL calls
|
|
start_stmt for the handle) we can use this function to store the pointer to
|
|
the THD in the handle. We will also use this function to communicate
|
|
to InnoDB that a new SQL statement has started and that we must store a
|
|
savepoint to our transaction handle, so that we are able to roll back
|
|
the SQL statement in case of an error. */
|
|
|
|
int
|
|
ha_innobase::external_lock(
|
|
/*=======================*/
|
|
/* out: 0 */
|
|
THD* thd, /* in: handle to the user thread */
|
|
int lock_type) /* in: lock type */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("ha_innobase::external_lock");
|
|
DBUG_PRINT("enter",("lock_type: %d", lock_type));
|
|
|
|
update_thd(thd);
|
|
|
|
trx = prebuilt->trx;
|
|
|
|
prebuilt->sql_stat_start = TRUE;
|
|
prebuilt->hint_need_to_fetch_extra_cols = 0;
|
|
|
|
reset_template(prebuilt);
|
|
|
|
if (lock_type == F_WRLCK) {
|
|
|
|
/* If this is a SELECT, then it is in UPDATE TABLE ...
|
|
or SELECT ... FOR UPDATE */
|
|
prebuilt->select_lock_type = LOCK_X;
|
|
prebuilt->stored_select_lock_type = LOCK_X;
|
|
}
|
|
|
|
if (lock_type != F_UNLCK) {
|
|
/* MySQL is setting a new table lock */
|
|
|
|
trx->detailed_error[0] = '\0';
|
|
|
|
/* Set the MySQL flag to mark that there is an active
|
|
transaction */
|
|
if (trx->active_trans == 0) {
|
|
|
|
innobase_register_trx_and_stmt(ht, thd);
|
|
trx->active_trans = 1;
|
|
} else if (trx->n_mysql_tables_in_use == 0) {
|
|
innobase_register_stmt(ht, thd);
|
|
}
|
|
|
|
trx->n_mysql_tables_in_use++;
|
|
prebuilt->mysql_has_locked = TRUE;
|
|
|
|
if (trx->isolation_level == TRX_ISO_SERIALIZABLE
|
|
&& prebuilt->select_lock_type == LOCK_NONE
|
|
&& thd_test_options(thd,
|
|
OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
|
|
/* To get serializable execution, we let InnoDB
|
|
conceptually add 'LOCK IN SHARE MODE' to all SELECTs
|
|
which otherwise would have been consistent reads. An
|
|
exception is consistent reads in the AUTOCOMMIT=1 mode:
|
|
we know that they are read-only transactions, and they
|
|
can be serialized also if performed as consistent
|
|
reads. */
|
|
|
|
prebuilt->select_lock_type = LOCK_S;
|
|
prebuilt->stored_select_lock_type = LOCK_S;
|
|
}
|
|
|
|
/* Starting from 4.1.9, no InnoDB table lock is taken in LOCK
|
|
TABLES if AUTOCOMMIT=1. It does not make much sense to acquire
|
|
an InnoDB table lock if it is released immediately at the end
|
|
of LOCK TABLES, and InnoDB's table locks in that case cause
|
|
VERY easily deadlocks.
|
|
|
|
We do not set InnoDB table locks if user has not explicitly
|
|
requested a table lock. Note that thd_in_lock_tables(thd)
|
|
can hold in some cases, e.g., at the start of a stored
|
|
procedure call (SQLCOM_CALL). */
|
|
|
|
if (prebuilt->select_lock_type != LOCK_NONE) {
|
|
|
|
if (thd_sql_command(thd) == SQLCOM_LOCK_TABLES
|
|
&& THDVAR(thd, table_locks)
|
|
&& thd_test_options(thd, OPTION_NOT_AUTOCOMMIT)
|
|
&& thd_in_lock_tables(thd)) {
|
|
|
|
ulint error = row_lock_table_for_mysql(
|
|
prebuilt, NULL, 0);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
error = convert_error_code_to_mysql(
|
|
(int) error, thd);
|
|
DBUG_RETURN((int) error);
|
|
}
|
|
}
|
|
|
|
trx->mysql_n_tables_locked++;
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* MySQL is releasing a table lock */
|
|
|
|
trx->n_mysql_tables_in_use--;
|
|
prebuilt->mysql_has_locked = FALSE;
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we
|
|
may reserve the kernel mutex, we have to release the search
|
|
system latch first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
/* If the MySQL lock count drops to zero we know that the current SQL
|
|
statement has ended */
|
|
|
|
if (trx->n_mysql_tables_in_use == 0) {
|
|
|
|
trx->mysql_n_tables_locked = 0;
|
|
prebuilt->used_in_HANDLER = FALSE;
|
|
|
|
if (!thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
if (trx->active_trans != 0) {
|
|
innobase_commit(ht, thd, TRUE);
|
|
}
|
|
} else {
|
|
if (trx->isolation_level <= TRX_ISO_READ_COMMITTED
|
|
&& trx->global_read_view) {
|
|
|
|
/* At low transaction isolation levels we let
|
|
each consistent read set its own snapshot */
|
|
|
|
read_view_close_for_mysql(trx);
|
|
}
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/**********************************************************************
|
|
With this function MySQL request a transactional lock to a table when
|
|
user issued query LOCK TABLES..WHERE ENGINE = InnoDB. */
|
|
|
|
int
|
|
ha_innobase::transactional_table_lock(
|
|
/*==================================*/
|
|
/* out: error code */
|
|
THD* thd, /* in: handle to the user thread */
|
|
int lock_type) /* in: lock type */
|
|
{
|
|
trx_t* trx;
|
|
|
|
DBUG_ENTER("ha_innobase::transactional_table_lock");
|
|
DBUG_PRINT("enter",("lock_type: %d", lock_type));
|
|
|
|
/* We do not know if MySQL can call this function before calling
|
|
external_lock(). To be safe, update the thd of the current table
|
|
handle. */
|
|
|
|
update_thd(thd);
|
|
|
|
if (prebuilt->table->ibd_file_missing && !thd_tablespace_op(thd)) {
|
|
ut_print_timestamp(stderr);
|
|
fprintf(stderr, " InnoDB error:\n"
|
|
"MySQL is trying to use a table handle but the .ibd file for\n"
|
|
"table %s does not exist.\n"
|
|
"Have you deleted the .ibd file from the database directory under\n"
|
|
"the MySQL datadir?"
|
|
"See http://dev.mysql.com/doc/refman/5.1/en/innodb-troubleshooting.html\n"
|
|
"how you can resolve the problem.\n",
|
|
prebuilt->table->name);
|
|
DBUG_RETURN(HA_ERR_CRASHED);
|
|
}
|
|
|
|
trx = prebuilt->trx;
|
|
|
|
prebuilt->sql_stat_start = TRUE;
|
|
prebuilt->hint_need_to_fetch_extra_cols = 0;
|
|
|
|
reset_template(prebuilt);
|
|
|
|
if (lock_type == F_WRLCK) {
|
|
prebuilt->select_lock_type = LOCK_X;
|
|
prebuilt->stored_select_lock_type = LOCK_X;
|
|
} else if (lock_type == F_RDLCK) {
|
|
prebuilt->select_lock_type = LOCK_S;
|
|
prebuilt->stored_select_lock_type = LOCK_S;
|
|
} else {
|
|
ut_print_timestamp(stderr);
|
|
fprintf(stderr, " InnoDB error:\n"
|
|
"MySQL is trying to set transactional table lock with corrupted lock type\n"
|
|
"to table %s, lock type %d does not exist.\n",
|
|
prebuilt->table->name, lock_type);
|
|
DBUG_RETURN(HA_ERR_CRASHED);
|
|
}
|
|
|
|
/* MySQL is setting a new transactional table lock */
|
|
|
|
/* Set the MySQL flag to mark that there is an active transaction */
|
|
if (trx->active_trans == 0) {
|
|
|
|
innobase_register_trx_and_stmt(ht, thd);
|
|
trx->active_trans = 1;
|
|
}
|
|
|
|
if (THDVAR(thd, table_locks) && thd_in_lock_tables(thd)) {
|
|
ulint error = DB_SUCCESS;
|
|
|
|
error = row_lock_table_for_mysql(prebuilt, NULL, 0);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
error = convert_error_code_to_mysql((int) error, thd);
|
|
DBUG_RETURN((int) error);
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)) {
|
|
|
|
/* Store the current undo_no of the transaction
|
|
so that we know where to roll back if we have
|
|
to roll back the next SQL statement */
|
|
|
|
trx_mark_sql_stat_end(trx);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Here we export InnoDB status variables to MySQL. */
|
|
static
|
|
int
|
|
innodb_export_status()
|
|
/*==================*/
|
|
{
|
|
if (innodb_inited) {
|
|
srv_export_innodb_status();
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Implements the SHOW INNODB STATUS command. Sends the output of the InnoDB
|
|
Monitor to the client. */
|
|
static
|
|
bool
|
|
innodb_show_status(
|
|
/*===============*/
|
|
handlerton* hton, /* in: the innodb handlerton */
|
|
THD* thd, /* in: the MySQL query thread of the caller */
|
|
stat_print_fn *stat_print)
|
|
{
|
|
trx_t* trx;
|
|
static const char truncated_msg[] = "... truncated...\n";
|
|
const long MAX_STATUS_SIZE = 64000;
|
|
ulint trx_list_start = ULINT_UNDEFINED;
|
|
ulint trx_list_end = ULINT_UNDEFINED;
|
|
|
|
DBUG_ENTER("innodb_show_status");
|
|
|
|
if (hton->state != SHOW_OPTION_YES) {
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
/* We let the InnoDB Monitor to output at most MAX_STATUS_SIZE
|
|
bytes of text. */
|
|
|
|
long flen, usable_len;
|
|
char* str;
|
|
|
|
mutex_enter_noninline(&srv_monitor_file_mutex);
|
|
rewind(srv_monitor_file);
|
|
srv_printf_innodb_monitor(srv_monitor_file,
|
|
&trx_list_start, &trx_list_end);
|
|
flen = ftell(srv_monitor_file);
|
|
os_file_set_eof(srv_monitor_file);
|
|
|
|
if (flen < 0) {
|
|
flen = 0;
|
|
}
|
|
|
|
if (flen > MAX_STATUS_SIZE) {
|
|
usable_len = MAX_STATUS_SIZE;
|
|
} else {
|
|
usable_len = flen;
|
|
}
|
|
|
|
/* allocate buffer for the string, and
|
|
read the contents of the temporary file */
|
|
|
|
if (!(str = my_malloc(usable_len + 1, MYF(0)))) {
|
|
mutex_exit_noninline(&srv_monitor_file_mutex);
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
rewind(srv_monitor_file);
|
|
if (flen < MAX_STATUS_SIZE) {
|
|
/* Display the entire output. */
|
|
flen = (long) fread(str, 1, flen, srv_monitor_file);
|
|
} else if (trx_list_end < (ulint) flen
|
|
&& trx_list_start < trx_list_end
|
|
&& trx_list_start + (flen - trx_list_end)
|
|
< MAX_STATUS_SIZE - sizeof truncated_msg - 1) {
|
|
/* Omit the beginning of the list of active transactions. */
|
|
long len = (long) fread(str, 1, trx_list_start, srv_monitor_file);
|
|
memcpy(str + len, truncated_msg, sizeof truncated_msg - 1);
|
|
len += sizeof truncated_msg - 1;
|
|
usable_len = (MAX_STATUS_SIZE - 1) - len;
|
|
fseek(srv_monitor_file, flen - usable_len, SEEK_SET);
|
|
len += (long) fread(str + len, 1, usable_len, srv_monitor_file);
|
|
flen = len;
|
|
} else {
|
|
/* Omit the end of the output. */
|
|
flen = (long) fread(str, 1, MAX_STATUS_SIZE - 1, srv_monitor_file);
|
|
}
|
|
|
|
mutex_exit_noninline(&srv_monitor_file_mutex);
|
|
|
|
bool result = FALSE;
|
|
|
|
if (stat_print(thd, innobase_hton_name, strlen(innobase_hton_name),
|
|
STRING_WITH_LEN(""), str, flen)) {
|
|
result= TRUE;
|
|
}
|
|
my_free(str, MYF(0));
|
|
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Implements the SHOW MUTEX STATUS command. . */
|
|
static
|
|
bool
|
|
innodb_mutex_show_status(
|
|
/*=====================*/
|
|
handlerton* hton, /* in: the innodb handlerton */
|
|
THD* thd, /* in: the MySQL query thread of the
|
|
caller */
|
|
stat_print_fn* stat_print)
|
|
{
|
|
char buf1[IO_SIZE], buf2[IO_SIZE];
|
|
mutex_t* mutex;
|
|
#ifdef UNIV_DEBUG
|
|
ulint rw_lock_count= 0;
|
|
ulint rw_lock_count_spin_loop= 0;
|
|
ulint rw_lock_count_spin_rounds= 0;
|
|
ulint rw_lock_count_os_wait= 0;
|
|
ulint rw_lock_count_os_yield= 0;
|
|
ulonglong rw_lock_wait_time= 0;
|
|
#endif /* UNIV_DEBUG */
|
|
uint hton_name_len= strlen(innobase_hton_name), buf1len, buf2len;
|
|
DBUG_ENTER("innodb_mutex_show_status");
|
|
|
|
mutex_enter_noninline(&mutex_list_mutex);
|
|
|
|
mutex = UT_LIST_GET_FIRST(mutex_list);
|
|
|
|
while (mutex != NULL) {
|
|
#ifdef UNIV_DEBUG
|
|
if (mutex->mutex_type != 1) {
|
|
if (mutex->count_using > 0) {
|
|
buf1len= my_snprintf(buf1, sizeof(buf1),
|
|
"%s:%s",
|
|
mutex->cmutex_name, mutex->cfile_name);
|
|
buf2len= my_snprintf(buf2, sizeof(buf2),
|
|
"count=%lu, spin_waits=%lu,"
|
|
" spin_rounds=%lu, "
|
|
"os_waits=%lu, os_yields=%lu,"
|
|
" os_wait_times=%lu",
|
|
mutex->count_using,
|
|
mutex->count_spin_loop,
|
|
mutex->count_spin_rounds,
|
|
mutex->count_os_wait,
|
|
mutex->count_os_yield,
|
|
(ulong) (mutex->lspent_time/1000));
|
|
|
|
if (stat_print(thd, innobase_hton_name,
|
|
hton_name_len, buf1, buf1len,
|
|
buf2, buf2len)) {
|
|
mutex_exit_noninline(
|
|
&mutex_list_mutex);
|
|
DBUG_RETURN(1);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
rw_lock_count += mutex->count_using;
|
|
rw_lock_count_spin_loop += mutex->count_spin_loop;
|
|
rw_lock_count_spin_rounds += mutex->count_spin_rounds;
|
|
rw_lock_count_os_wait += mutex->count_os_wait;
|
|
rw_lock_count_os_yield += mutex->count_os_yield;
|
|
rw_lock_wait_time += mutex->lspent_time;
|
|
}
|
|
#else /* UNIV_DEBUG */
|
|
buf1len= my_snprintf(buf1, sizeof(buf1), "%s:%lu",
|
|
mutex->cfile_name, (ulong) mutex->cline);
|
|
buf2len= my_snprintf(buf2, sizeof(buf2), "os_waits=%lu",
|
|
mutex->count_os_wait);
|
|
|
|
if (stat_print(thd, innobase_hton_name,
|
|
hton_name_len, buf1, buf1len,
|
|
buf2, buf2len)) {
|
|
mutex_exit_noninline(&mutex_list_mutex);
|
|
DBUG_RETURN(1);
|
|
}
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
mutex = UT_LIST_GET_NEXT(list, mutex);
|
|
}
|
|
|
|
mutex_exit_noninline(&mutex_list_mutex);
|
|
|
|
#ifdef UNIV_DEBUG
|
|
buf2len= my_snprintf(buf2, sizeof(buf2),
|
|
"count=%lu, spin_waits=%lu, spin_rounds=%lu, "
|
|
"os_waits=%lu, os_yields=%lu, os_wait_times=%lu",
|
|
rw_lock_count, rw_lock_count_spin_loop,
|
|
rw_lock_count_spin_rounds,
|
|
rw_lock_count_os_wait, rw_lock_count_os_yield,
|
|
(ulong) (rw_lock_wait_time/1000));
|
|
|
|
if (stat_print(thd, innobase_hton_name, hton_name_len,
|
|
STRING_WITH_LEN("rw_lock_mutexes"), buf2, buf2len)) {
|
|
DBUG_RETURN(1);
|
|
}
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
static
|
|
bool innobase_show_status(handlerton *hton, THD* thd,
|
|
stat_print_fn* stat_print,
|
|
enum ha_stat_type stat_type)
|
|
{
|
|
switch (stat_type) {
|
|
case HA_ENGINE_STATUS:
|
|
return innodb_show_status(hton, thd, stat_print);
|
|
case HA_ENGINE_MUTEX:
|
|
return innodb_mutex_show_status(hton, thd, stat_print);
|
|
default:
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
Handling the shared INNOBASE_SHARE structure that is needed to provide table
|
|
locking.
|
|
****************************************************************************/
|
|
|
|
static mysql_byte* innobase_get_key(INNOBASE_SHARE* share, uint* length,
|
|
my_bool not_used __attribute__((unused)))
|
|
{
|
|
*length=share->table_name_length;
|
|
|
|
return((mysql_byte*) share->table_name);
|
|
}
|
|
|
|
static INNOBASE_SHARE* get_share(const char* table_name)
|
|
{
|
|
INNOBASE_SHARE *share;
|
|
pthread_mutex_lock(&innobase_share_mutex);
|
|
uint length=(uint) strlen(table_name);
|
|
|
|
if (!(share=(INNOBASE_SHARE*) hash_search(&innobase_open_tables,
|
|
(mysql_byte*) table_name,
|
|
length))) {
|
|
|
|
share = (INNOBASE_SHARE *) my_malloc(sizeof(*share)+length+1,
|
|
MYF(MY_FAE | MY_ZEROFILL));
|
|
|
|
share->table_name_length=length;
|
|
share->table_name=(char*) (share+1);
|
|
strmov(share->table_name,table_name);
|
|
|
|
if (my_hash_insert(&innobase_open_tables,
|
|
(mysql_byte*) share)) {
|
|
pthread_mutex_unlock(&innobase_share_mutex);
|
|
my_free((gptr) share,0);
|
|
|
|
return(0);
|
|
}
|
|
|
|
thr_lock_init(&share->lock);
|
|
pthread_mutex_init(&share->mutex,MY_MUTEX_INIT_FAST);
|
|
}
|
|
|
|
share->use_count++;
|
|
pthread_mutex_unlock(&innobase_share_mutex);
|
|
|
|
return(share);
|
|
}
|
|
|
|
static void free_share(INNOBASE_SHARE* share)
|
|
{
|
|
pthread_mutex_lock(&innobase_share_mutex);
|
|
|
|
if (!--share->use_count) {
|
|
hash_delete(&innobase_open_tables, (mysql_byte*) share);
|
|
thr_lock_delete(&share->lock);
|
|
pthread_mutex_destroy(&share->mutex);
|
|
my_free((gptr) share, MYF(0));
|
|
}
|
|
|
|
pthread_mutex_unlock(&innobase_share_mutex);
|
|
}
|
|
|
|
/*********************************************************************
|
|
Converts a MySQL table lock stored in the 'lock' field of the handle to
|
|
a proper type before storing pointer to the lock into an array of pointers.
|
|
MySQL also calls this if it wants to reset some table locks to a not-locked
|
|
state during the processing of an SQL query. An example is that during a
|
|
SELECT the read lock is released early on the 'const' tables where we only
|
|
fetch one row. MySQL does not call this when it releases all locks at the
|
|
end of an SQL statement. */
|
|
|
|
THR_LOCK_DATA**
|
|
ha_innobase::store_lock(
|
|
/*====================*/
|
|
/* out: pointer to the next
|
|
element in the 'to' array */
|
|
THD* thd, /* in: user thread handle */
|
|
THR_LOCK_DATA** to, /* in: pointer to an array
|
|
of pointers to lock structs;
|
|
pointer to the 'lock' field
|
|
of current handle is stored
|
|
next to this array */
|
|
enum thr_lock_type lock_type) /* in: lock type to store in
|
|
'lock'; this may also be
|
|
TL_IGNORE */
|
|
{
|
|
trx_t* trx;
|
|
|
|
/* Note that trx in this function is NOT necessarily prebuilt->trx
|
|
because we call update_thd() later, in ::external_lock()! Failure to
|
|
understand this caused a serious memory corruption bug in 5.1.11. */
|
|
|
|
trx = check_trx_exists(thd);
|
|
|
|
/* NOTE: MySQL can call this function with lock 'type' TL_IGNORE!
|
|
Be careful to ignore TL_IGNORE if we are going to do something with
|
|
only 'real' locks! */
|
|
|
|
/* If no MySQL table is in use, we need to set the isolation level
|
|
of the transaction. */
|
|
|
|
if (lock_type != TL_IGNORE
|
|
&& trx->n_mysql_tables_in_use == 0) {
|
|
trx->isolation_level = innobase_map_isolation_level(
|
|
(enum_tx_isolation) thd_tx_isolation(thd));
|
|
|
|
if (trx->isolation_level <= TRX_ISO_READ_COMMITTED
|
|
&& trx->global_read_view) {
|
|
|
|
/* At low transaction isolation levels we let
|
|
each consistent read set its own snapshot */
|
|
|
|
read_view_close_for_mysql(trx);
|
|
}
|
|
}
|
|
|
|
DBUG_ASSERT(thd == current_thd);
|
|
const bool in_lock_tables = thd_in_lock_tables(thd);
|
|
const uint sql_command = thd_sql_command(thd);
|
|
|
|
if (sql_command == SQLCOM_DROP_TABLE) {
|
|
|
|
/* MySQL calls this function in DROP TABLE though this table
|
|
handle may belong to another thd that is running a query. Let
|
|
us in that case skip any changes to the prebuilt struct. */
|
|
|
|
} else if ((lock_type == TL_READ && in_lock_tables)
|
|
|| (lock_type == TL_READ_HIGH_PRIORITY && in_lock_tables)
|
|
|| lock_type == TL_READ_WITH_SHARED_LOCKS
|
|
|| lock_type == TL_READ_NO_INSERT
|
|
|| (lock_type != TL_IGNORE
|
|
&& sql_command != SQLCOM_SELECT)) {
|
|
|
|
/* The OR cases above are in this order:
|
|
1) MySQL is doing LOCK TABLES ... READ LOCAL, or we
|
|
are processing a stored procedure or function, or
|
|
2) (we do not know when TL_READ_HIGH_PRIORITY is used), or
|
|
3) this is a SELECT ... IN SHARE MODE, or
|
|
4) we are doing a complex SQL statement like
|
|
INSERT INTO ... SELECT ... and the logical logging (MySQL
|
|
binlog) requires the use of a locking read, or
|
|
MySQL is doing LOCK TABLES ... READ.
|
|
5) we let InnoDB do locking reads for all SQL statements that
|
|
are not simple SELECTs; note that select_lock_type in this
|
|
case may get strengthened in ::external_lock() to LOCK_X.
|
|
Note that we MUST use a locking read in all data modifying
|
|
SQL statements, because otherwise the execution would not be
|
|
serializable, and also the results from the update could be
|
|
unexpected if an obsolete consistent read view would be
|
|
used. */
|
|
|
|
ulint isolation_level;
|
|
|
|
isolation_level = trx->isolation_level;
|
|
|
|
if ((srv_locks_unsafe_for_binlog
|
|
|| isolation_level == TRX_ISO_READ_COMMITTED)
|
|
&& isolation_level != TRX_ISO_SERIALIZABLE
|
|
&& (lock_type == TL_READ || lock_type == TL_READ_NO_INSERT)
|
|
&& (sql_command == SQLCOM_INSERT_SELECT
|
|
|| sql_command == SQLCOM_UPDATE
|
|
|| sql_command == SQLCOM_CREATE_TABLE)) {
|
|
|
|
/* If we either have innobase_locks_unsafe_for_binlog
|
|
option set or this session is using READ COMMITTED
|
|
isolation level and isolation level of the transaction
|
|
is not set to serializable and MySQL is doing
|
|
INSERT INTO...SELECT or UPDATE ... = (SELECT ...) or
|
|
CREATE ... SELECT... without FOR UPDATE or
|
|
IN SHARE MODE in select, then we use consistent
|
|
read for select. */
|
|
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
prebuilt->stored_select_lock_type = LOCK_NONE;
|
|
} else if (sql_command == SQLCOM_CHECKSUM) {
|
|
/* Use consistent read for checksum table */
|
|
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
prebuilt->stored_select_lock_type = LOCK_NONE;
|
|
} else {
|
|
prebuilt->select_lock_type = LOCK_S;
|
|
prebuilt->stored_select_lock_type = LOCK_S;
|
|
}
|
|
|
|
} else if (lock_type != TL_IGNORE) {
|
|
|
|
/* We set possible LOCK_X value in external_lock, not yet
|
|
here even if this would be SELECT ... FOR UPDATE */
|
|
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
prebuilt->stored_select_lock_type = LOCK_NONE;
|
|
}
|
|
|
|
if (lock_type != TL_IGNORE && lock.type == TL_UNLOCK) {
|
|
|
|
/* Starting from 5.0.7, we weaken also the table locks
|
|
set at the start of a MySQL stored procedure call, just like
|
|
we weaken the locks set at the start of an SQL statement.
|
|
MySQL does set in_lock_tables TRUE there, but in reality
|
|
we do not need table locks to make the execution of a
|
|
single transaction stored procedure call deterministic
|
|
(if it does not use a consistent read). */
|
|
|
|
if (lock_type == TL_READ
|
|
&& sql_command == SQLCOM_LOCK_TABLES) {
|
|
/* We come here if MySQL is processing LOCK TABLES
|
|
... READ LOCAL. MyISAM under that table lock type
|
|
reads the table as it was at the time the lock was
|
|
granted (new inserts are allowed, but not seen by the
|
|
reader). To get a similar effect on an InnoDB table,
|
|
we must use LOCK TABLES ... READ. We convert the lock
|
|
type here, so that for InnoDB, READ LOCAL is
|
|
equivalent to READ. This will change the InnoDB
|
|
behavior in mysqldump, so that dumps of InnoDB tables
|
|
are consistent with dumps of MyISAM tables. */
|
|
|
|
lock_type = TL_READ_NO_INSERT;
|
|
}
|
|
|
|
/* If we are not doing a LOCK TABLE, DISCARD/IMPORT
|
|
TABLESPACE or TRUNCATE TABLE then allow multiple
|
|
writers. Note that ALTER TABLE uses a TL_WRITE_ALLOW_READ
|
|
< TL_WRITE_CONCURRENT_INSERT.
|
|
|
|
We especially allow multiple writers if MySQL is at the
|
|
start of a stored procedure call (SQLCOM_CALL) or a
|
|
stored function call (MySQL does have in_lock_tables
|
|
TRUE there). */
|
|
|
|
if ((lock_type >= TL_WRITE_CONCURRENT_INSERT
|
|
&& lock_type <= TL_WRITE)
|
|
&& !(in_lock_tables
|
|
&& sql_command == SQLCOM_LOCK_TABLES)
|
|
&& !thd_tablespace_op(thd)
|
|
&& sql_command != SQLCOM_TRUNCATE
|
|
&& sql_command != SQLCOM_OPTIMIZE
|
|
|
|
#ifdef __WIN__
|
|
/* For alter table on win32 for successful
|
|
operation completion it is used TL_WRITE(=10) lock
|
|
instead of TL_WRITE_ALLOW_READ(=6), however here
|
|
in innodb handler TL_WRITE is lifted to
|
|
TL_WRITE_ALLOW_WRITE, which causes race condition
|
|
when several clients do alter table simultaneously
|
|
(bug #17264). This fix avoids the problem. */
|
|
&& sql_command != SQLCOM_ALTER_TABLE
|
|
#endif
|
|
|
|
&& sql_command != SQLCOM_CREATE_TABLE) {
|
|
|
|
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.
|
|
|
|
We especially allow concurrent inserts if MySQL is at the
|
|
start of a stored procedure call (SQLCOM_CALL)
|
|
(MySQL does have thd_in_lock_tables() TRUE there). */
|
|
|
|
if (lock_type == TL_READ_NO_INSERT
|
|
&& sql_command != SQLCOM_LOCK_TABLES) {
|
|
|
|
lock_type = TL_READ;
|
|
}
|
|
|
|
lock.type = lock_type;
|
|
}
|
|
|
|
*to++= &lock;
|
|
|
|
return(to);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function initializes the auto-inc counter if it has not been
|
|
initialized yet. This function does not change the value of the auto-inc
|
|
counter if it already has been initialized. In parameter ret returns
|
|
the value of the auto-inc counter. */
|
|
|
|
int
|
|
ha_innobase::innobase_read_and_init_auto_inc(
|
|
/*=========================================*/
|
|
/* out: 0 or error code: deadlock or lock wait
|
|
timeout */
|
|
longlong* ret) /* out: auto-inc value */
|
|
{
|
|
longlong auto_inc;
|
|
ulint old_select_lock_type;
|
|
ibool trx_was_not_started = FALSE;
|
|
int error;
|
|
|
|
ut_a(prebuilt);
|
|
ut_a(prebuilt->table);
|
|
|
|
/* Prepare prebuilt->trx in the table handle */
|
|
update_thd(ha_thd());
|
|
|
|
if (prebuilt->trx->conc_state == TRX_NOT_STARTED) {
|
|
trx_was_not_started = TRUE;
|
|
}
|
|
|
|
/* In case MySQL calls this in the middle of a SELECT query, release
|
|
possible adaptive hash latch to avoid deadlocks of threads */
|
|
|
|
trx_search_latch_release_if_reserved(prebuilt->trx);
|
|
|
|
auto_inc = dict_table_autoinc_read(prebuilt->table);
|
|
|
|
if (auto_inc != 0) {
|
|
/* Already initialized */
|
|
*ret = auto_inc;
|
|
|
|
error = 0;
|
|
|
|
goto func_exit_early;
|
|
}
|
|
|
|
error = row_lock_table_autoinc_for_mysql(prebuilt);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
goto func_exit_early;
|
|
}
|
|
|
|
/* Check again if someone has initialized the counter meanwhile */
|
|
auto_inc = dict_table_autoinc_read(prebuilt->table);
|
|
|
|
if (auto_inc != 0) {
|
|
*ret = auto_inc;
|
|
|
|
error = 0;
|
|
|
|
goto func_exit_early;
|
|
}
|
|
|
|
(void) extra(HA_EXTRA_KEYREAD);
|
|
index_init(table->s->next_number_index, 1);
|
|
|
|
/* Starting from 5.0.9, we use a consistent read to read the auto-inc
|
|
column maximum value. This eliminates the spurious deadlocks caused
|
|
by the row X-lock that we previously used. Note the following flaw
|
|
in our algorithm: if some other user meanwhile UPDATEs the auto-inc
|
|
column, our consistent read will not return the largest value. We
|
|
accept this flaw, since the deadlocks were a bigger trouble. */
|
|
|
|
/* Fetch all the columns in the key */
|
|
|
|
prebuilt->hint_need_to_fetch_extra_cols = ROW_RETRIEVE_ALL_COLS;
|
|
|
|
old_select_lock_type = prebuilt->select_lock_type;
|
|
prebuilt->select_lock_type = LOCK_NONE;
|
|
|
|
/* Eliminate an InnoDB error print that happens when we try to SELECT
|
|
from a table when no table has been locked in ::external_lock(). */
|
|
prebuilt->trx->n_mysql_tables_in_use++;
|
|
|
|
error = index_last(table->record[1]);
|
|
|
|
prebuilt->trx->n_mysql_tables_in_use--;
|
|
prebuilt->select_lock_type = old_select_lock_type;
|
|
|
|
if (error) {
|
|
if (error == HA_ERR_END_OF_FILE) {
|
|
/* The table was empty, initialize to 1 */
|
|
auto_inc = 1;
|
|
|
|
error = 0;
|
|
} else {
|
|
/* This should not happen in a consistent read */
|
|
sql_print_error("Consistent read of auto-inc column "
|
|
"returned %lu", (ulong) error);
|
|
auto_inc = -1;
|
|
|
|
goto func_exit;
|
|
}
|
|
} else {
|
|
/* Initialize to max(col) + 1; we use
|
|
'found_next_number_field' below because MySQL in SHOW TABLE
|
|
STATUS does not seem to set 'next_number_field'. The comment
|
|
in table.h says that 'next_number_field' is set when it is
|
|
'active'.
|
|
Since 5.1 MySQL enforces that we announce fields which we will
|
|
read; as we only do a val_*() call, dbug_tmp_use_all_columns()
|
|
with read_set is sufficient. */
|
|
|
|
my_bitmap_map *old_map;
|
|
old_map= dbug_tmp_use_all_columns(table, table->read_set);
|
|
auto_inc = (longlong) table->found_next_number_field->
|
|
val_int_offset(table->s->rec_buff_length) + 1;
|
|
dbug_tmp_restore_column_map(table->read_set, old_map);
|
|
}
|
|
|
|
dict_table_autoinc_initialize(prebuilt->table, auto_inc);
|
|
|
|
func_exit:
|
|
(void) extra(HA_EXTRA_NO_KEYREAD);
|
|
|
|
index_end();
|
|
|
|
*ret = auto_inc;
|
|
|
|
func_exit_early:
|
|
/* Since MySQL does not seem to call autocommit after SHOW TABLE
|
|
STATUS (even if we would register the trx here), we commit our
|
|
transaction here if it was started here. This is to eliminate a
|
|
dangling transaction. If the user had AUTOCOMMIT=0, then SHOW
|
|
TABLE STATUS does leave a dangling transaction if the user does not
|
|
himself call COMMIT. */
|
|
|
|
if (trx_was_not_started) {
|
|
|
|
innobase_commit_low(prebuilt->trx);
|
|
}
|
|
|
|
return(error);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
This function initializes the auto-inc counter if it has not been
|
|
initialized yet. This function does not change the value of the auto-inc
|
|
counter if it already has been initialized. Returns the value of the
|
|
auto-inc counter in *first_value, and ULONGLONG_MAX in *nb_reserved_values (as
|
|
we have a table-level lock). offset, increment, nb_desired_values are ignored.
|
|
*first_value is set to -1 if error (deadlock or lock wait timeout) */
|
|
|
|
void ha_innobase::get_auto_increment(
|
|
/*=================================*/
|
|
ulonglong offset, /* in */
|
|
ulonglong increment, /* in */
|
|
ulonglong nb_desired_values, /* in */
|
|
ulonglong *first_value, /* out */
|
|
ulonglong *nb_reserved_values) /* out */
|
|
{
|
|
longlong nr;
|
|
int error;
|
|
|
|
/* Prepare prebuilt->trx in the table handle */
|
|
update_thd(ha_thd());
|
|
|
|
error = innobase_read_and_init_auto_inc(&nr);
|
|
|
|
if (error) {
|
|
/* This should never happen in the current (5.0.6) code, since
|
|
we call this function only after the counter has been
|
|
initialized. */
|
|
|
|
ut_print_timestamp(stderr);
|
|
sql_print_error("Error %lu in ::get_auto_increment()",
|
|
(ulong) error);
|
|
*first_value= (~(ulonglong) 0);
|
|
return;
|
|
}
|
|
|
|
*first_value= (ulonglong) nr;
|
|
/* table-level autoinc lock reserves up to +inf */
|
|
*nb_reserved_values= ULONGLONG_MAX;
|
|
}
|
|
|
|
/* See comment in handler.h */
|
|
int
|
|
ha_innobase::reset_auto_increment(ulonglong value)
|
|
{
|
|
DBUG_ENTER("ha_innobase::reset_auto_increment");
|
|
|
|
int error;
|
|
|
|
update_thd(ha_thd());
|
|
|
|
error = row_lock_table_autoinc_for_mysql(prebuilt);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
error = convert_error_code_to_mysql(error, user_thd);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
dict_table_autoinc_initialize(prebuilt->table, value);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/* See comment in handler.cc */
|
|
bool
|
|
ha_innobase::get_error_message(int error, String *buf)
|
|
{
|
|
trx_t* trx = check_trx_exists(ha_thd());
|
|
|
|
buf->copy(trx->detailed_error, strlen(trx->detailed_error),
|
|
system_charset_info);
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Compares two 'refs'. A 'ref' is the (internal) primary key value of the row.
|
|
If there is no explicitly declared non-null unique key or a primary key, then
|
|
InnoDB internally uses the row id as the primary key. */
|
|
|
|
int
|
|
ha_innobase::cmp_ref(
|
|
/*=================*/
|
|
/* out: < 0 if ref1 < ref2, 0 if equal, else
|
|
> 0 */
|
|
const mysql_byte* ref1, /* in: an (internal) primary key value in the
|
|
MySQL key value format */
|
|
const mysql_byte* ref2) /* in: an (internal) primary key value in the
|
|
MySQL key value format */
|
|
{
|
|
enum_field_types mysql_type;
|
|
Field* field;
|
|
KEY_PART_INFO* key_part;
|
|
KEY_PART_INFO* key_part_end;
|
|
uint len1;
|
|
uint len2;
|
|
int result;
|
|
|
|
if (prebuilt->clust_index_was_generated) {
|
|
/* The 'ref' is an InnoDB row id */
|
|
|
|
return(memcmp(ref1, ref2, DATA_ROW_ID_LEN));
|
|
}
|
|
|
|
/* Do a type-aware comparison of primary key fields. PK fields
|
|
are always NOT NULL, so no checks for NULL are performed. */
|
|
|
|
key_part = table->key_info[table->s->primary_key].key_part;
|
|
|
|
key_part_end = key_part
|
|
+ table->key_info[table->s->primary_key].key_parts;
|
|
|
|
for (; key_part != key_part_end; ++key_part) {
|
|
field = key_part->field;
|
|
mysql_type = field->type();
|
|
|
|
if (mysql_type == MYSQL_TYPE_TINY_BLOB
|
|
|| mysql_type == MYSQL_TYPE_MEDIUM_BLOB
|
|
|| mysql_type == MYSQL_TYPE_BLOB
|
|
|| mysql_type == MYSQL_TYPE_LONG_BLOB) {
|
|
|
|
/* In the MySQL key value format, a column prefix of
|
|
a BLOB is preceded by a 2-byte length field */
|
|
|
|
len1 = innobase_read_from_2_little_endian(ref1);
|
|
len2 = innobase_read_from_2_little_endian(ref2);
|
|
|
|
ref1 += 2;
|
|
ref2 += 2;
|
|
result = ((Field_blob*)field)->cmp(
|
|
(const char*)ref1, len1,
|
|
(const char*)ref2, len2);
|
|
} else {
|
|
result = field->key_cmp(ref1, ref2);
|
|
}
|
|
|
|
if (result) {
|
|
|
|
return(result);
|
|
}
|
|
|
|
ref1 += key_part->store_length;
|
|
ref2 += key_part->store_length;
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Ask InnoDB if a query to a table can be cached. */
|
|
|
|
my_bool
|
|
ha_innobase::register_query_cache_table(
|
|
/*====================================*/
|
|
/* out: TRUE if query caching
|
|
of the table is permitted */
|
|
THD* thd, /* in: user thread handle */
|
|
char* table_key, /* in: concatenation of database name,
|
|
the null character '\0',
|
|
and the table name */
|
|
uint key_length, /* in: length of the full name, i.e.
|
|
len(dbname) + len(tablename) + 1 */
|
|
qc_engine_callback*
|
|
call_back, /* out: pointer to function for
|
|
checking if query caching
|
|
is permitted */
|
|
ulonglong *engine_data) /* in/out: data to call_back */
|
|
{
|
|
*call_back = innobase_query_caching_of_table_permitted;
|
|
*engine_data = 0;
|
|
return(innobase_query_caching_of_table_permitted(thd, table_key,
|
|
key_length,
|
|
engine_data));
|
|
}
|
|
|
|
char*
|
|
ha_innobase::get_mysql_bin_log_name()
|
|
{
|
|
return(trx_sys_mysql_bin_log_name);
|
|
}
|
|
|
|
ulonglong
|
|
ha_innobase::get_mysql_bin_log_pos()
|
|
{
|
|
/* trx... is ib_longlong, which is a typedef for a 64-bit integer
|
|
(__int64 or longlong) so it's ok to cast it to ulonglong. */
|
|
|
|
return(trx_sys_mysql_bin_log_pos);
|
|
}
|
|
|
|
/**********************************************************************
|
|
This function is used to find the storage length in bytes of the first n
|
|
characters for prefix indexes using a multibyte character set. The function
|
|
finds charset information and returns length of prefix_len characters in the
|
|
index field in bytes.
|
|
|
|
NOTE: the prototype of this function is copied to data0type.c! If you change
|
|
this function, you MUST change also data0type.c! */
|
|
extern "C"
|
|
ulint
|
|
innobase_get_at_most_n_mbchars(
|
|
/*===========================*/
|
|
/* out: number of bytes occupied by the first
|
|
n characters */
|
|
ulint charset_id, /* in: character set id */
|
|
ulint prefix_len, /* in: prefix length in bytes of the index
|
|
(this has to be divided by mbmaxlen to get the
|
|
number of CHARACTERS n in the prefix) */
|
|
ulint data_len, /* in: length of the string in bytes */
|
|
const char* str) /* in: character string */
|
|
{
|
|
ulint char_length; /* character length in bytes */
|
|
ulint n_chars; /* number of characters in prefix */
|
|
CHARSET_INFO* charset; /* charset used in the field */
|
|
|
|
charset = get_charset((uint) charset_id, MYF(MY_WME));
|
|
|
|
ut_ad(charset);
|
|
ut_ad(charset->mbmaxlen);
|
|
|
|
/* Calculate how many characters at most the prefix index contains */
|
|
|
|
n_chars = prefix_len / charset->mbmaxlen;
|
|
|
|
/* If the charset is multi-byte, then we must find the length of the
|
|
first at most n chars in the string. If the string contains less
|
|
characters than n, then we return the length to the end of the last
|
|
character. */
|
|
|
|
if (charset->mbmaxlen > 1) {
|
|
/* my_charpos() returns the byte length of the first n_chars
|
|
characters, or a value bigger than the length of str, if
|
|
there were not enough full characters in str.
|
|
|
|
Why does the code below work:
|
|
Suppose that we are looking for n UTF-8 characters.
|
|
|
|
1) If the string is long enough, then the prefix contains at
|
|
least n complete UTF-8 characters + maybe some extra
|
|
characters + an incomplete UTF-8 character. No problem in
|
|
this case. The function returns the pointer to the
|
|
end of the nth character.
|
|
|
|
2) If the string is not long enough, then the string contains
|
|
the complete value of a column, that is, only complete UTF-8
|
|
characters, and we can store in the column prefix index the
|
|
whole string. */
|
|
|
|
char_length = my_charpos(charset, str,
|
|
str + data_len, (int) n_chars);
|
|
if (char_length > data_len) {
|
|
char_length = data_len;
|
|
}
|
|
} else {
|
|
if (data_len < prefix_len) {
|
|
char_length = data_len;
|
|
} else {
|
|
char_length = prefix_len;
|
|
}
|
|
}
|
|
|
|
return(char_length);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function is used to prepare X/Open XA distributed transaction */
|
|
static
|
|
int
|
|
innobase_xa_prepare(
|
|
/*================*/
|
|
/* out: 0 or error number */
|
|
handlerton *hton,
|
|
THD* thd, /* in: handle to the MySQL thread of the user
|
|
whose XA transaction should be prepared */
|
|
bool all) /* in: TRUE - commit transaction
|
|
FALSE - the current SQL statement ended */
|
|
{
|
|
int error = 0;
|
|
trx_t* trx = check_trx_exists(thd);
|
|
|
|
if (thd_sql_command(thd) != SQLCOM_XA_PREPARE &&
|
|
(all || !thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)))
|
|
{
|
|
|
|
/* For ibbackup to work the order of transactions in binlog
|
|
and InnoDB must be the same. Consider the situation
|
|
|
|
thread1> prepare; write to binlog; ...
|
|
<context switch>
|
|
thread2> prepare; write to binlog; commit
|
|
thread1> ... commit
|
|
|
|
To ensure this will not happen we're taking the mutex on
|
|
prepare, and releasing it on commit.
|
|
|
|
Note: only do it for normal commits, done via ha_commit_trans.
|
|
If 2pc protocol is executed by external transaction
|
|
coordinator, it will be just a regular MySQL client
|
|
executing XA PREPARE and XA COMMIT commands.
|
|
In this case we cannot know how many minutes or hours
|
|
will be between XA PREPARE and XA COMMIT, and we don't want
|
|
to block for undefined period of time.
|
|
*/
|
|
pthread_mutex_lock(&prepare_commit_mutex);
|
|
trx->active_trans = 2;
|
|
}
|
|
|
|
if (!THDVAR(thd, support_xa)) {
|
|
|
|
return(0);
|
|
}
|
|
|
|
thd_get_xid(thd, &trx->xid);
|
|
|
|
/* Release a possible FIFO ticket and search latch. Since we will
|
|
reserve the kernel mutex, we have to release the search system latch
|
|
first to obey the latching order. */
|
|
|
|
innobase_release_stat_resources(trx);
|
|
|
|
if (trx->active_trans == 0 && trx->conc_state != TRX_NOT_STARTED) {
|
|
|
|
sql_print_error("trx->active_trans == 0, but trx->conc_state != "
|
|
"TRX_NOT_STARTED");
|
|
}
|
|
|
|
if (all
|
|
|| (!thd_test_options(thd, OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN))) {
|
|
|
|
/* We were instructed to prepare the whole transaction, or
|
|
this is an SQL statement end and autocommit is on */
|
|
|
|
ut_ad(trx->active_trans);
|
|
|
|
error = (int) trx_prepare_for_mysql(trx);
|
|
} else {
|
|
/* We just mark the SQL statement ended and do not do a
|
|
transaction prepare */
|
|
|
|
if (trx->auto_inc_lock) {
|
|
/* If we had reserved the auto-inc lock for some
|
|
table in this SQL statement we release it now */
|
|
|
|
row_unlock_table_autoinc_for_mysql(trx);
|
|
}
|
|
/* Store the current undo_no of the transaction so that we
|
|
know where to roll back if we have to roll back the next
|
|
SQL statement */
|
|
|
|
trx_mark_sql_stat_end(trx);
|
|
}
|
|
|
|
/* Tell the InnoDB server that there might be work for utility
|
|
threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
return(error);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function is used to recover X/Open XA distributed transactions */
|
|
static
|
|
int
|
|
innobase_xa_recover(
|
|
/*================*/
|
|
/* out: number of prepared transactions
|
|
stored in xid_list */
|
|
handlerton *hton,
|
|
XID* xid_list, /* in/out: prepared transactions */
|
|
uint len) /* in: number of slots in xid_list */
|
|
{
|
|
if (len == 0 || xid_list == NULL) {
|
|
|
|
return(0);
|
|
}
|
|
|
|
return(trx_recover_for_mysql(xid_list, len));
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function is used to commit one X/Open XA distributed transaction
|
|
which is in the prepared state */
|
|
static
|
|
int
|
|
innobase_commit_by_xid(
|
|
/*===================*/
|
|
/* out: 0 or error number */
|
|
handlerton *hton,
|
|
XID* xid) /* in: X/Open XA transaction identification */
|
|
{
|
|
trx_t* trx;
|
|
|
|
trx = trx_get_trx_by_xid(xid);
|
|
|
|
if (trx) {
|
|
innobase_commit_low(trx);
|
|
|
|
return(XA_OK);
|
|
} else {
|
|
return(XAER_NOTA);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function checks that index keys are sensible. */
|
|
static
|
|
int
|
|
innobase_check_index_keys(
|
|
/*======================*/
|
|
/* out: 0 or error number */
|
|
TABLE* table, /* in: MySQL table */
|
|
dict_table_t* innodb_table, /* in: InnoDB table */
|
|
trx_t* trx, /* in: transaction */
|
|
KEY* key_info, /* in: Indexes to be created */
|
|
ulint num_of_keys) /* in: Number of indexes to
|
|
be created */
|
|
{
|
|
Field* field;
|
|
ulint key_num;
|
|
int error = DB_SUCCESS;
|
|
ibool is_unsigned;
|
|
|
|
ut_ad(table && innodb_table && trx && key_info && num_of_keys);
|
|
|
|
for (key_num = 0; key_num < num_of_keys; key_num++) {
|
|
KEY* key;
|
|
|
|
key = &(key_info[key_num]);
|
|
|
|
/* Check that the same index name does not appear
|
|
twice in indexes to be created. */
|
|
|
|
for (ulint i = 0; i < key_num; i++) {
|
|
KEY* key2;
|
|
|
|
key2 = &key_info[i];
|
|
|
|
if (0 == strcmp(key->name, key2->name)) {
|
|
ut_print_timestamp(stderr);
|
|
|
|
fputs(" InnoDB: Error: index ", stderr);
|
|
ut_print_name(stderr, trx, FALSE, key->name);
|
|
fputs(" appears twice in create index\n",
|
|
stderr);
|
|
|
|
error = ER_WRONG_NAME_FOR_INDEX;
|
|
|
|
return(error);
|
|
}
|
|
}
|
|
|
|
/* Check that MySQL does not try to create a column prefix index
|
|
field on an inappropriate data type and that the same colum does
|
|
not appear twice in the index. */
|
|
|
|
for (ulint i = 0; i < key->key_parts; i++) {
|
|
KEY_PART_INFO* key_part1;
|
|
ulint col_type; /* Column type */
|
|
|
|
key_part1 = key->key_part + i;
|
|
|
|
field = key_part1->field;
|
|
|
|
col_type = get_innobase_type_from_mysql_type(
|
|
&is_unsigned, field);
|
|
|
|
if (DATA_BLOB == col_type
|
|
|| (key_part1->length < field->pack_length()
|
|
&& field->type() != MYSQL_TYPE_VARCHAR)
|
|
|| (field->type() == MYSQL_TYPE_VARCHAR
|
|
&& key_part1->length < field->pack_length()
|
|
- ((Field_varstring*)field)->length_bytes)) {
|
|
|
|
if (col_type == DATA_INT
|
|
|| col_type == DATA_FLOAT
|
|
|| col_type == DATA_DOUBLE
|
|
|| col_type == DATA_DECIMAL) {
|
|
fprintf(stderr,
|
|
"InnoDB: error: MySQL is trying to create a column prefix index field\n"
|
|
"InnoDB: on an inappropriate data type. Table name %s, column name %s.\n",
|
|
innodb_table->name,
|
|
field->field_name);
|
|
|
|
error = ER_WRONG_KEY_COLUMN;
|
|
}
|
|
}
|
|
|
|
for (ulint j = 0; j < i; j++) {
|
|
KEY_PART_INFO* key_part2;
|
|
|
|
key_part2 = key->key_part + j;
|
|
|
|
if (0 == strcmp(
|
|
key_part1->field->field_name,
|
|
key_part2->field->field_name)) {
|
|
|
|
ut_print_timestamp(stderr);
|
|
|
|
fputs(" InnoDB: Error: column ",
|
|
stderr);
|
|
|
|
ut_print_name(stderr, trx, FALSE,
|
|
key_part1->field->field_name);
|
|
|
|
fputs(" appears twice in ", stderr);
|
|
|
|
ut_print_name(stderr, trx, FALSE,
|
|
key->name);
|
|
fputs("\n"
|
|
" InnoDB: This is not allowed in InnoDB.\n",
|
|
stderr);
|
|
|
|
error = ER_WRONG_KEY_COLUMN;
|
|
|
|
return(error);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return(error);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Create index field definition for key part */
|
|
static
|
|
void
|
|
innobase_create_index_field_def(
|
|
/*============================*/
|
|
KEY_PART_INFO* key_part, /* in: MySQL key definition */
|
|
mem_heap_t* heap, /* in: memory heap */
|
|
merge_index_field_t* index_field) /* out: index field
|
|
definition for key_part */
|
|
{
|
|
Field* field;
|
|
ibool is_unsigned;
|
|
ulint col_type;
|
|
|
|
DBUG_ENTER("innobase_create_index_field_def");
|
|
|
|
ut_ad(key_part);
|
|
ut_ad(index_field);
|
|
|
|
field = key_part->field;
|
|
ut_a(field);
|
|
|
|
col_type = get_innobase_type_from_mysql_type(&is_unsigned, field);
|
|
|
|
index_field->col_type = col_type;
|
|
|
|
if (DATA_BLOB == col_type
|
|
|| (key_part->length < field->pack_length()
|
|
&& field->type() != MYSQL_TYPE_VARCHAR)
|
|
|| (field->type() == MYSQL_TYPE_VARCHAR
|
|
&& key_part->length < field->pack_length()
|
|
- ((Field_varstring*)field)->length_bytes)) {
|
|
|
|
index_field->prefix_len = key_part->length;
|
|
} else {
|
|
index_field->prefix_len = 0;
|
|
}
|
|
|
|
index_field->field_name = mem_heap_strdup(heap, field->field_name);
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/***********************************************************************
|
|
Create index definition for key */
|
|
static
|
|
merge_index_def_t*
|
|
innobase_create_index_def(
|
|
/*======================*/
|
|
/* out: Index definition */
|
|
KEY* key, /* in: key definition */
|
|
mem_heap_t* heap) /* in: heap where memory is allocated */
|
|
{
|
|
ulint len;
|
|
merge_index_def_t* index;
|
|
ulint n_fields = key->key_parts;
|
|
|
|
DBUG_ENTER("innobase_create_index_def");
|
|
|
|
ut_a(key && heap);
|
|
|
|
index = (merge_index_def_t*) mem_heap_alloc_noninline(
|
|
heap, sizeof(merge_index_def_t));
|
|
|
|
index->fields = (merge_index_field_t*) mem_heap_alloc_noninline(
|
|
heap, n_fields * sizeof *index->fields);
|
|
|
|
index->ind_type = 0;
|
|
index->n_fields = n_fields;
|
|
len = strlen(key->name) + 2;
|
|
index->name = (char *)mem_heap_alloc_noninline(heap, len);
|
|
|
|
--len;
|
|
|
|
if (my_strcasecmp(system_charset_info, key->name, "PRIMARY")) {
|
|
*index->name = TEMP_TABLE_PREFIX;
|
|
memcpy(index->name + 1, key->name, len);
|
|
} else {
|
|
memcpy(index->name, key->name, len);
|
|
}
|
|
|
|
if (key->flags & HA_NOSAME) {
|
|
index->ind_type = index->ind_type | DICT_UNIQUE;
|
|
}
|
|
|
|
if (!my_strcasecmp(system_charset_info, key->name, "PRIMARY")) {
|
|
index->ind_type = index->ind_type | DICT_CLUSTERED;
|
|
}
|
|
|
|
for (ulint i = 0; i < n_fields; i++) {
|
|
innobase_create_index_field_def(&key->key_part[i], heap,
|
|
&index->fields[i]);
|
|
}
|
|
|
|
DBUG_RETURN(index);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Copy index field definition */
|
|
static
|
|
void
|
|
innobase_copy_index_field_def(
|
|
/*==========================*/
|
|
const dict_field_t* field, /* in: definition to copy */
|
|
mem_heap_t* heap, /* in: memory heap */
|
|
merge_index_field_t* index_field) /* out: copied definition */
|
|
{
|
|
DBUG_ENTER("innobase_copy_index_field_def");
|
|
DBUG_ASSERT(field != NULL);
|
|
DBUG_ASSERT(index_field != NULL);
|
|
|
|
index_field->col_type = (field->col->prtype & 0xFFUL);
|
|
index_field->field_name = mem_heap_strdup(heap, field->name);
|
|
index_field->prefix_len = field->prefix_len;
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/***********************************************************************
|
|
Copy index definition for the index */
|
|
static
|
|
merge_index_def_t*
|
|
innobase_copy_index_def(
|
|
/*====================*/
|
|
/* out: Index definition */
|
|
dict_index_t* index, /* in: index definition to copy */
|
|
mem_heap_t* heap) /* in: heap where memory is allocated */
|
|
{
|
|
merge_index_def_t* new_index;
|
|
ulint n_fields;
|
|
ulint i;
|
|
|
|
DBUG_ENTER("innobase_copy_index_def");
|
|
|
|
ut_a(index && heap);
|
|
|
|
if (!(index->type & DICT_CLUSTERED)) {
|
|
/* Note that from the secondary index we take only
|
|
those fields that user defined to be in the index.
|
|
In the internal representation more colums were
|
|
added and those colums are not copied.*/
|
|
|
|
n_fields = index->n_user_defined_cols;
|
|
} else {
|
|
n_fields = index->n_fields;
|
|
}
|
|
|
|
new_index = (merge_index_def_t*) mem_heap_alloc_noninline(
|
|
heap, sizeof(merge_index_def_t));
|
|
|
|
new_index->fields = (merge_index_field_t*) mem_heap_alloc_noninline(
|
|
heap, n_fields * sizeof *new_index->fields);
|
|
|
|
new_index->ind_type = index->type;
|
|
new_index->n_fields = n_fields;
|
|
new_index->name = mem_heap_strdup(heap, index->name);
|
|
|
|
for (i = 0; i < n_fields; i++) {
|
|
innobase_copy_index_field_def(&index->fields[i], heap,
|
|
&new_index->fields[i]);
|
|
}
|
|
|
|
DBUG_RETURN(new_index);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Create an index table where indexes are ordered as follows:
|
|
|
|
IF a primary key is defined for the table THEN
|
|
|
|
1) New primary key
|
|
2) Original unique secondary indexes
|
|
3) New unique secondary indexes
|
|
4) Original secondary indexes
|
|
5) New secondary indexes
|
|
|
|
ELSE
|
|
|
|
1) All new indexes in the order they arrive from MySQL
|
|
|
|
ENDIF
|
|
|
|
*/
|
|
static
|
|
merge_index_def_t**
|
|
innobase_create_key_def(
|
|
/*====================*/
|
|
/* out: key definitions or NULL */
|
|
trx_t* trx, /* in: trx */
|
|
dict_table_t* table, /* in: table definition */
|
|
mem_heap_t* heap, /* in: heap where space for key
|
|
definitions are allocated */
|
|
KEY* key_info, /* in: Indexes to be created */
|
|
ulint* n_keys) /* in/out: Number of indexes to
|
|
be created */
|
|
{
|
|
ulint n_indexes; /* Number of indexes */
|
|
merge_index_def_t** indexdef; /* Index definition */
|
|
merge_index_def_t** indexdefs; /* Index definitions */
|
|
|
|
DBUG_ENTER("innobase_create_key_def");
|
|
|
|
ut_a(trx && table && heap && key_info && n_keys && *n_keys);
|
|
|
|
/* We do not need to count the original primary key */
|
|
n_indexes = *n_keys + UT_LIST_GET_LEN(table->indexes) - 1;
|
|
|
|
indexdef = indexdefs = (merge_index_def_t**) mem_heap_alloc_noninline(
|
|
heap, sizeof(merge_index_def_t*) * n_indexes);
|
|
|
|
/* Primary key if defined is always the first index defined for
|
|
the table */
|
|
|
|
if (!my_strcasecmp(system_charset_info, key_info->name, "PRIMARY")) {
|
|
dict_index_t* index;
|
|
|
|
/* Create the PRIMARY key index definition */
|
|
*indexdef = innobase_create_index_def(key_info, heap);
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
/* Skip the clustered index */
|
|
|
|
index = dict_table_get_next_index_noninline(
|
|
dict_table_get_first_index_noninline(table));
|
|
|
|
/* Copy the definitions of all the secondary indexes */
|
|
|
|
++indexdef; /* Since we've copied the primary key info */
|
|
*n_keys = 1;
|
|
|
|
while (index) {
|
|
|
|
ut_a(!(index->type & DICT_CLUSTERED));
|
|
|
|
*indexdef = innobase_copy_index_def(index, heap);
|
|
index = dict_table_get_next_index_noninline(index);
|
|
|
|
++*n_keys;
|
|
++indexdef;
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
} else {
|
|
ulint i = 0;
|
|
|
|
/* Create definitions for new unique secondary indexes */
|
|
|
|
for (KEY* key = key_info; i < *n_keys; ++key, ++i, ++indexdef) {
|
|
*indexdef = innobase_create_index_def(key, heap);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(indexdefs);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Create a temporary tablename using query id, thread id, and id */
|
|
static
|
|
char*
|
|
innobase_create_temporary_tablename(
|
|
/*================================*/
|
|
/* out: temporary tablename */
|
|
mem_heap_t* heap, /* in: memory heap */
|
|
char id, /* in: identifier */
|
|
const char* table_name) /* in: table name */
|
|
{
|
|
char* name;
|
|
ulint len;
|
|
|
|
len = strlen(table_name) + 3;
|
|
|
|
name = (char*) mem_heap_alloc_noninline(heap, len);
|
|
name[0] = TEMP_TABLE_PREFIX;
|
|
name[1] = id;
|
|
memcpy(name + 2, table_name, len - 2);
|
|
|
|
return(name);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Add a new index to a table */
|
|
|
|
int
|
|
ha_innobase::add_index(
|
|
/*===================*/
|
|
/* out: 0 or error number */
|
|
TABLE* table, /* in: Table where indexes are created */
|
|
KEY* key_info, /* in: Indexes to be created */
|
|
uint num_of_keys) /* in: Number of indexes to be created */
|
|
{
|
|
dict_index_t** index; /* Index to be created */
|
|
dict_table_t* innodb_table; /* InnoDB table in dictionary */
|
|
dict_table_t* indexed_table; /* Table where indexes are created */
|
|
merge_index_def_t** index_defs; /* Index definitions */
|
|
mem_heap_t* heap; /* Heap for index definitions */
|
|
trx_t* trx; /* Transaction */
|
|
trx_t* parent_trx;
|
|
ulint num_of_idx;
|
|
ulint num_created;
|
|
ibool dict_locked = FALSE;
|
|
ibool new_primary;
|
|
ibool new_unique = FALSE;
|
|
ulint error;
|
|
|
|
DBUG_ENTER("ha_innobase::add_index");
|
|
ut_a(table && key_info && num_of_keys);
|
|
|
|
update_thd(current_thd);
|
|
|
|
heap = mem_heap_create_noninline(1024);
|
|
|
|
parent_trx = check_trx_exists(user_thd);
|
|
trx_search_latch_release_if_reserved(parent_trx);
|
|
|
|
trx = parent_trx;
|
|
ut_a(trx);
|
|
|
|
trx_start_if_not_started_noninline(trx);
|
|
|
|
trx->mysql_thd = user_thd;
|
|
trx->mysql_query_str = thd_query(user_thd);
|
|
|
|
if (thd_test_options(user_thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
if (thd_test_options(user_thd, OPTION_RELAXED_UNIQUE_CHECKS)) {
|
|
trx->check_unique_secondary = FALSE;
|
|
}
|
|
|
|
innodb_table = indexed_table
|
|
= dict_table_get(prebuilt->table->name, FALSE);
|
|
|
|
/* Check that index keys are sensible */
|
|
|
|
error = innobase_check_index_keys(
|
|
table, innodb_table, trx, key_info, num_of_keys);
|
|
|
|
if (UNIV_UNLIKELY(error != DB_SUCCESS)) {
|
|
err_exit:
|
|
mem_heap_free_noninline(heap);
|
|
trx_general_rollback_for_mysql(trx, FALSE, NULL);
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, user_thd));
|
|
}
|
|
|
|
/* Create table containing all indexes to be built in this
|
|
alter table add index so that they are in the correct order
|
|
in the table. */
|
|
|
|
num_of_idx = num_of_keys;
|
|
|
|
index_defs = innobase_create_key_def(
|
|
trx, innodb_table, heap, key_info, &num_of_idx);
|
|
|
|
/* If a new primary key is defined for the table we need
|
|
to drop all original secondary indexes from the table. These
|
|
indexes will be rebuilt below. */
|
|
|
|
new_primary = 0 != (DICT_CLUSTERED & index_defs[0]->ind_type);
|
|
|
|
if (new_primary) {
|
|
char* new_table_name = innobase_create_temporary_tablename(
|
|
heap, '1', innodb_table->name);
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
/* Clone table and write UNDO log record */
|
|
indexed_table = row_merge_create_temporary_table(
|
|
new_table_name, innodb_table, trx, &error);
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
if (!indexed_table) {
|
|
|
|
goto err_exit;
|
|
}
|
|
} else if (!trx->dict_redo_list) {
|
|
dict_redo_create_list(trx);
|
|
|
|
ut_a(!trx->sync_cb);
|
|
trx->sync_cb = dict_rename_indexes;
|
|
}
|
|
|
|
/* Allocate memory for dictionary index definitions */
|
|
|
|
index = (dict_index_t**) mem_heap_alloc_noninline(
|
|
heap, num_of_idx * sizeof *index);
|
|
|
|
/* Latch the InnoDB data dictionary exclusively so that no deadlocks
|
|
or lock waits can happen in it during an index create operation.
|
|
Drop table etc. do this latching in row0mysql.c. */
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
num_created = 0;
|
|
dict_locked = TRUE;
|
|
|
|
/* Create the indexes in SYS_INDEXES and load into dictionary.*/
|
|
|
|
for (ulint i = 0; i < num_of_idx && error == DB_SUCCESS; i++) {
|
|
|
|
error = row_merge_create_index(
|
|
trx, &index[i], indexed_table, index_defs[i]);
|
|
|
|
if (index_defs[i]->ind_type & DICT_UNIQUE) {
|
|
|
|
new_unique = TRUE;
|
|
}
|
|
|
|
num_created++;
|
|
}
|
|
|
|
if (error == DB_SUCCESS) {
|
|
/* Raise version number of the table to track this table's
|
|
definition changes.*/
|
|
|
|
indexed_table->version_number++;
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
dict_locked = FALSE;
|
|
|
|
mem_heap_empty_noninline(heap);
|
|
|
|
ut_a(trx->n_active_thrs == 0);
|
|
ut_a(UT_LIST_GET_LEN(trx->signals) == 0);
|
|
|
|
error = row_lock_table_for_merge(trx, innodb_table, LOCK_X);
|
|
}
|
|
|
|
/* Set an exclusive table lock for the new table if a primary
|
|
key is to be built.*/
|
|
if (error == DB_SUCCESS && new_primary) {
|
|
|
|
ut_ad(indexed_table != innodb_table);
|
|
|
|
error = row_lock_table_for_merge(trx, indexed_table, LOCK_X);
|
|
}
|
|
|
|
if (error == DB_SUCCESS) {
|
|
|
|
/* Read clustered index of the table and build indexes
|
|
based on this information using temporary files and merge
|
|
sort.*/
|
|
error = row_build_index_for_mysql(
|
|
trx, innodb_table, indexed_table, index,
|
|
num_of_idx);
|
|
|
|
if (error == DB_SUCCESS && new_primary) {
|
|
row_merge_mark_prebuilt_obsolete(trx, innodb_table);
|
|
}
|
|
}
|
|
|
|
#ifdef UNIV_DEBUG
|
|
/* TODO: At the moment we can't handle the following statement
|
|
in our debugging code below:
|
|
|
|
alter table t drop index b, add index (b);
|
|
|
|
The fix will have to parse the SQL and note that the index
|
|
being added has the same name as the the one being dropped and
|
|
ignore that in the dup index check.*/
|
|
//dict_table_check_for_dup_indexes(prebuilt->table);
|
|
#endif
|
|
|
|
if (dict_locked) {
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
}
|
|
|
|
/* After an error, remove all those index definitions from the
|
|
dictionary which were defined.*/
|
|
|
|
switch (error) {
|
|
case DB_SUCCESS:
|
|
break;
|
|
case DB_DUPLICATE_KEY:
|
|
prebuilt->trx->error_info = NULL;
|
|
prebuilt->trx->error_key_num = trx->error_key_num;
|
|
/* fall through */
|
|
default:
|
|
row_merge_drop_indexes(
|
|
trx, indexed_table, index, num_created);
|
|
if (indexed_table != innodb_table) {
|
|
row_merge_drop_table(trx, indexed_table);
|
|
}
|
|
mem_heap_free_noninline(heap);
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, user_thd));
|
|
}
|
|
|
|
/* If a new primary key was defined for the table and
|
|
there was no error at this point, we can now rename the
|
|
old table as a temporary table, rename the new temporary
|
|
table as a old table and drop the old table. */
|
|
|
|
if (new_primary) {
|
|
char* old_name = mem_heap_strdup(
|
|
heap, innodb_table->name);
|
|
char* tmp_table_name = innobase_create_temporary_tablename(
|
|
heap, '2', innodb_table->name);
|
|
|
|
trx_start_if_not_started_noninline(trx);
|
|
|
|
/* Write entry for UNDO */
|
|
error = row_undo_report_rename_table_dict_operation(
|
|
trx, old_name, indexed_table->name, tmp_table_name);
|
|
|
|
if (error != DB_SUCCESS) {
|
|
|
|
goto func_exit;
|
|
}
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
/* Set the commit flag to FALSE, we will commit the
|
|
transaction ourselves, required for UNDO */
|
|
error = innobase_rename_table(trx, innodb_table->name,
|
|
tmp_table_name, FALSE);
|
|
if (error != DB_SUCCESS) {
|
|
|
|
goto func_exit;
|
|
}
|
|
|
|
error = innobase_rename_table(trx, indexed_table->name,
|
|
old_name, FALSE);
|
|
if (error != DB_SUCCESS) {
|
|
|
|
goto func_exit;
|
|
}
|
|
|
|
row_prebuilt_free(prebuilt);
|
|
prebuilt = row_create_prebuilt(indexed_table);
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
prebuilt->table->n_mysql_handles_opened++;
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
/* Drop the old table if there are no open views
|
|
referring to it. If there such views, we will drop
|
|
the table when we free the prebuilts and there are no
|
|
more references to it. */
|
|
|
|
error = row_merge_drop_table(trx, innodb_table);
|
|
}
|
|
|
|
func_exit:
|
|
/* There might be work for utility threads.*/
|
|
mem_heap_free_noninline(heap);
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
DBUG_RETURN(convert_error_code_to_mysql(error, user_thd));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Drop a index from a table */
|
|
|
|
int
|
|
ha_innobase::prepare_drop_index(
|
|
/*============================*/
|
|
/* out: 0 or error number */
|
|
TABLE* table, /* in: Table where indexes are dropped */
|
|
uint* key_num, /* in: Key nums to be dropped */
|
|
uint num_of_keys) /* in: Number of keys to be dropped */
|
|
{
|
|
trx_t* trx;
|
|
THD* thd;
|
|
ulint error = DB_SUCCESS;
|
|
|
|
DBUG_ENTER("ha_innobase::prepare_drop_index");
|
|
ut_ad(table && key_num && num_of_keys);
|
|
|
|
thd = ha_thd();
|
|
|
|
/* Create a new transaction for prepare index drop if it
|
|
does not exists */
|
|
|
|
trx = check_trx_exists(thd);
|
|
trx_search_latch_release_if_reserved(trx);
|
|
|
|
if (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) {
|
|
trx->check_foreigns = FALSE;
|
|
}
|
|
|
|
if (thd_test_options(thd, OPTION_RELAXED_UNIQUE_CHECKS)) {
|
|
trx->check_unique_secondary = FALSE;
|
|
}
|
|
|
|
/* Test and mark all the indexes to be dropped */
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
for (ulint n_key = 0; n_key < num_of_keys; n_key++) {
|
|
KEY* key;
|
|
dict_index_t* index;
|
|
|
|
key = table->key_info + key_num[n_key];
|
|
ut_a(key);
|
|
|
|
index = NULL;
|
|
|
|
if (key) {
|
|
index = dict_table_get_index_on_name_and_min_id(
|
|
prebuilt->table,
|
|
key->name);
|
|
}
|
|
|
|
if (!index) {
|
|
sql_print_error("InnoDB could not find key n:o %u "
|
|
"with name %s from dict cache for table %s",
|
|
key_num[n_key], key ? key->name : "NULL",
|
|
prebuilt->table->name);
|
|
|
|
error = 1;
|
|
goto func_exit;
|
|
}
|
|
|
|
index->to_be_dropped = TRUE;
|
|
}
|
|
|
|
/* We check for the foreign key constraints after marking the
|
|
candidate indexes for deletion because when we check for an
|
|
equivalent foreign index we don't want to select an index that is
|
|
later deleted. */
|
|
for (ulint n_key = 0; n_key < num_of_keys; n_key++) {
|
|
KEY* key;
|
|
dict_index_t* index;
|
|
|
|
key = table->key_info + key_num[n_key];
|
|
|
|
index = dict_table_get_index_on_name_and_min_id(
|
|
prebuilt->table, key->name);
|
|
|
|
ut_a(index);
|
|
ut_a(index->to_be_dropped);
|
|
|
|
/* If FOREIGN_KEY_CHECK = 1 you may not drop an index
|
|
defined for a foreign key constraint because
|
|
InnoDB requires that both tables contain indexes
|
|
for the constraint. Note that create index id on table
|
|
does a create index and drop index and we can ignore
|
|
here foreign keys because a new index for the foreign
|
|
key has already been created. */
|
|
|
|
if (trx->check_foreigns
|
|
&& thd_sql_command(thd) != SQLCOM_CREATE_INDEX) {
|
|
dict_foreign_t* foreign;
|
|
ibool ok_to_delete = TRUE;
|
|
|
|
/* Check if this index is referenced by some other
|
|
table */
|
|
foreign = dict_table_get_referenced_constraint(
|
|
prebuilt->table, index);
|
|
|
|
if (foreign) {
|
|
|
|
ok_to_delete = FALSE;
|
|
|
|
} else {
|
|
|
|
/* Check if this index references some
|
|
other table */
|
|
foreign = dict_table_get_foreign_constraint(
|
|
prebuilt->table, index);
|
|
|
|
if (foreign) {
|
|
ut_a(foreign->foreign_index == index);
|
|
|
|
/* Search for an equivalent index that
|
|
the foreign key contraint could use
|
|
if this index were to be deleted. */
|
|
if (!dict_table_find_equivalent_index(
|
|
prebuilt->table,
|
|
foreign->foreign_index)) {
|
|
|
|
ok_to_delete = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!ok_to_delete) {
|
|
|
|
trx_set_detailed_error(
|
|
trx,
|
|
"Index needed in foreign key "
|
|
"constraint");
|
|
|
|
trx->error_info = index;
|
|
|
|
FILE* ef = dict_foreign_err_file;
|
|
|
|
error = DB_CANNOT_DROP_FOREIGN_INDEX;
|
|
|
|
mutex_enter_noninline(&dict_foreign_err_mutex);
|
|
rewind(ef);
|
|
ut_print_timestamp(ef);
|
|
|
|
fputs(" Cannot drop index ", ef);
|
|
ut_print_name(ef, trx, FALSE, index->name);
|
|
fputs("\nbecause it is referenced by ", ef);
|
|
ut_print_name(ef, trx, TRUE,
|
|
foreign->foreign_table_name);
|
|
putc('\n', ef);
|
|
mutex_exit_noninline(&dict_foreign_err_mutex);
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
func_exit:
|
|
if (error != DB_SUCCESS) {
|
|
/* Undo our changes since there was some sort of error */
|
|
for (ulint i = 0; i < num_of_keys; i++) {
|
|
KEY* key;
|
|
dict_index_t* index;
|
|
|
|
key = table->key_info + key_num[i];
|
|
ut_a(key);
|
|
|
|
index = dict_table_get_index_on_name_and_min_id(
|
|
prebuilt->table, key->name);
|
|
|
|
if (index) {
|
|
index->to_be_dropped = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
error = convert_error_code_to_mysql(error, thd);
|
|
|
|
DBUG_RETURN((int)error);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Finalize a drop index */
|
|
|
|
int
|
|
ha_innobase::final_drop_index(
|
|
/*==========================*/
|
|
/* out: 0 or error number */
|
|
TABLE* table) /* in: Table where indexes are dropped */
|
|
{
|
|
dict_index_t* index; /* Index to be dropped */
|
|
trx_t* trx; /* Transaction */
|
|
THD* thd;
|
|
|
|
DBUG_ENTER("ha_innobase::final_drop_index");
|
|
ut_ad(table);
|
|
|
|
/* Create a new transaction for final index drop if it does not
|
|
* exits*/
|
|
|
|
thd = ha_thd();
|
|
|
|
trx = check_trx_exists(thd);
|
|
trx_search_latch_release_if_reserved(trx);
|
|
|
|
/* Drop indexes marked to be dropped */
|
|
|
|
row_mysql_lock_data_dictionary(trx);
|
|
|
|
index = dict_table_get_first_index_noninline(prebuilt->table);
|
|
|
|
while (index) {
|
|
dict_index_t* next_index;
|
|
|
|
next_index = dict_table_get_next_index_noninline(index);
|
|
|
|
if (index->to_be_dropped) {
|
|
|
|
row_merge_drop_index(index, prebuilt->table, trx);
|
|
}
|
|
|
|
index = next_index;
|
|
}
|
|
|
|
prebuilt->table->version_number++;
|
|
|
|
#ifdef UNIV_DEBUG
|
|
dict_table_check_for_dup_indexes(prebuilt->table);
|
|
#endif
|
|
|
|
row_mysql_unlock_data_dictionary(trx);
|
|
|
|
/* Flush the log to reduce probability that the .frm files and
|
|
the InnoDB data dictionary get out-of-sync if the user runs
|
|
with innodb_flush_log_at_trx_commit = 0 */
|
|
|
|
log_buffer_flush_to_disk();
|
|
|
|
#ifdef UNIV_DEBUG_INDEX_CREATE
|
|
row_merge_print_table(prebuilt->table);
|
|
#endif
|
|
|
|
/* Tell the InnoDB server that there might be work for
|
|
utility threads: */
|
|
|
|
srv_active_wake_master_thread();
|
|
|
|
trx_commit_for_mysql(trx);
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
This function is used to rollback one X/Open XA distributed transaction
|
|
which is in the prepared state */
|
|
static
|
|
int
|
|
innobase_rollback_by_xid(
|
|
/*=====================*/
|
|
/* out: 0 or error number */
|
|
handlerton *hton,
|
|
XID *xid) /* in: X/Open XA transaction identification */
|
|
{
|
|
trx_t* trx;
|
|
|
|
trx = trx_get_trx_by_xid(xid);
|
|
|
|
if (trx) {
|
|
return(innobase_rollback_trx(trx));
|
|
} else {
|
|
return(XAER_NOTA);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
Create a consistent view for a cursor based on current transaction
|
|
which is created if the corresponding MySQL thread still lacks one.
|
|
This consistent view is then used inside of MySQL when accessing records
|
|
using a cursor. */
|
|
static
|
|
void*
|
|
innobase_create_cursor_view(
|
|
/*========================*/
|
|
/* out: pointer to cursor view or NULL */
|
|
handlerton *hton, /* in: innobase hton */
|
|
THD* thd) /* in: user thread handle */
|
|
{
|
|
return(read_cursor_view_create_for_mysql(check_trx_exists(thd)));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Close the given consistent cursor view of a transaction and restore
|
|
global read view to a transaction read view. Transaction is created if the
|
|
corresponding MySQL thread still lacks one. */
|
|
static
|
|
void
|
|
innobase_close_cursor_view(
|
|
/*=======================*/
|
|
handlerton *hton,
|
|
THD* thd, /* in: user thread handle */
|
|
void* curview)/* in: Consistent read view to be closed */
|
|
{
|
|
read_cursor_view_close_for_mysql(check_trx_exists(thd),
|
|
(cursor_view_t*) curview);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Set the given consistent cursor view to a transaction which is created
|
|
if the corresponding MySQL thread still lacks one. If the given
|
|
consistent cursor view is NULL global read view of a transaction is
|
|
restored to a transaction read view. */
|
|
static
|
|
void
|
|
innobase_set_cursor_view(
|
|
/*=====================*/
|
|
handlerton *hton,
|
|
THD* thd, /* in: user thread handle */
|
|
void* curview)/* in: Consistent cursor view to be set */
|
|
{
|
|
read_cursor_set_for_mysql(check_trx_exists(thd),
|
|
(cursor_view_t*) curview);
|
|
}
|
|
|
|
|
|
bool ha_innobase::check_if_incompatible_data(
|
|
HA_CREATE_INFO* info,
|
|
uint table_changes)
|
|
{
|
|
if (table_changes != IS_EQUAL_YES) {
|
|
|
|
return(COMPATIBLE_DATA_NO);
|
|
}
|
|
|
|
/* Check that auto_increment value was not changed */
|
|
if ((info->used_fields & HA_CREATE_USED_AUTO) &&
|
|
info->auto_increment_value != 0) {
|
|
|
|
return(COMPATIBLE_DATA_NO);
|
|
}
|
|
|
|
/* Check that row format didn't change */
|
|
if ((info->used_fields & HA_CREATE_USED_AUTO) &&
|
|
get_row_type() != info->row_type) {
|
|
|
|
return(COMPATIBLE_DATA_NO);
|
|
}
|
|
|
|
return(COMPATIBLE_DATA_YES);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Fill the dynamic table information_schema.innodb_zip. */
|
|
static
|
|
int
|
|
innobase_stat_zip_fill(
|
|
/*===================*/
|
|
/* out: 0 on success, 1 on failure */
|
|
THD* thd, /* in: thread */
|
|
TABLE_LIST* tables, /* in/out: tables to fill */
|
|
COND* cond, /* in: condition (ignored) */
|
|
ibool reset) /* in: TRUE=reset cumulated counts */
|
|
{
|
|
TABLE* table = (TABLE *) tables->table;
|
|
int status = 0;
|
|
uint y = 0;
|
|
|
|
DBUG_ENTER("innobase_stat_zip_fill");
|
|
|
|
/* Determine log2(PAGE_ZIP_MIN_SIZE / 2 / BUF_BUDDY_LOW). */
|
|
for (uint r = PAGE_ZIP_MIN_SIZE / 2 / BUF_BUDDY_LOW; r >>= 1; y++);
|
|
|
|
mutex_enter_noninline(&buf_pool->mutex);
|
|
|
|
for (uint x = 0; x <= BUF_BUDDY_SIZES; x++) {
|
|
table->field[0]->store(BUF_BUDDY_LOW << x);
|
|
table->field[1]->store(buf_buddy_relocated[x]);
|
|
if (reset) {
|
|
/* This is protected by buf_pool->mutex. */
|
|
buf_buddy_relocated[x] = 0;
|
|
}
|
|
|
|
if (x > y) {
|
|
/* The cumulated counts are not protected by
|
|
any mutex. Thus, some operation in page0zip.c
|
|
could increment a counter between the time we
|
|
read it and clear it. We could introduce
|
|
mutex protection, but it could cause a
|
|
measureable performance hit in page0zip.c. */
|
|
const uint i = x - y;
|
|
table->field[2]->store(page_zip_compress_count[i]);
|
|
table->field[3]->store(page_zip_compress_ok[i]);
|
|
table->field[4]->store(page_zip_decompress_count[i]);
|
|
if (reset) {
|
|
page_zip_compress_count[i] = 0;
|
|
page_zip_compress_ok[i] = 0;
|
|
page_zip_decompress_count[i] = 0;
|
|
}
|
|
} else {
|
|
table->field[2]->store(0);
|
|
table->field[3]->store(0);
|
|
table->field[4]->store(0);
|
|
}
|
|
table->field[5]->store(buf_buddy_used[x]);
|
|
|
|
if (schema_table_store_record(thd, table)) {
|
|
status = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
mutex_exit_noninline(&buf_pool->mutex);
|
|
DBUG_RETURN(status);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Fill the dynamic table information_schema.innodb_zip. */
|
|
static
|
|
int
|
|
innobase_stat_zip_fill(
|
|
/*===================*/
|
|
/* out: 0 on success, 1 on failure */
|
|
THD* thd, /* in: thread */
|
|
TABLE_LIST* tables, /* in/out: tables to fill */
|
|
COND* cond) /* in: condition (ignored) */
|
|
{
|
|
return(innobase_stat_zip_fill(thd, tables, cond, FALSE));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Fill the dynamic table information_schema.innodb_zip_reset. */
|
|
static
|
|
int
|
|
innobase_stat_zip_reset_fill(
|
|
/*=========================*/
|
|
/* out: 0 on success, 1 on failure */
|
|
THD* thd, /* in: thread */
|
|
TABLE_LIST* tables, /* in/out: tables to fill */
|
|
COND* cond) /* in: condition (ignored) */
|
|
{
|
|
return(innobase_stat_zip_fill(thd, tables, cond, TRUE));
|
|
}
|
|
|
|
/* Fields of the dynamic table information_schema.innodb_zip. */
|
|
static ST_FIELD_INFO innobase_stat_zip_fields[] =
|
|
{
|
|
{"SIZE", 5, MYSQL_TYPE_LONG, 0, 0, "Block Size"},
|
|
{"RELOCATED", 21, MYSQL_TYPE_LONG, 0, 0, "Total Number of Relocations"},
|
|
{"COMPRESSED", 21, MYSQL_TYPE_LONG, 0, 0,
|
|
"Total Number of Compressions"},
|
|
{"COMPRESSED_OK", 21, MYSQL_TYPE_LONG, 0, 0,
|
|
"Total Number of Successful Compressions"},
|
|
{"DECOMPRESSED", 21, MYSQL_TYPE_LONG, 0, 0,
|
|
"Total Number of Decompressions"},
|
|
{"USED", 21, MYSQL_TYPE_LONG, 0, 0, "Currently in Use"},
|
|
{0, 0, MYSQL_TYPE_STRING, 0, 0, 0}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Bind the dynamic table information_schema.innodb_zip. */
|
|
static
|
|
int
|
|
innobase_stat_zip_init(
|
|
/*===================*/
|
|
/* out: 0 on success */
|
|
void* p) /* in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("innobase_stat_zip_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innobase_stat_zip_fields;
|
|
schema->fill_table = innobase_stat_zip_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Bind the dynamic table information_schema.innodb_zip_reset. */
|
|
static
|
|
int
|
|
innobase_stat_zip_reset_init(
|
|
/*=========================*/
|
|
/* out: 0 on success */
|
|
void* p) /* in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("innobase_stat_zip_reset_init");
|
|
ST_SCHEMA_TABLE* schema = (ST_SCHEMA_TABLE*) p;
|
|
|
|
schema->fields_info = innobase_stat_zip_fields;
|
|
schema->fill_table = innobase_stat_zip_reset_fill;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Unbind a dynamic table in information_schema. */
|
|
static
|
|
int
|
|
innobase_stat_zip_deinit(
|
|
/*=====================*/
|
|
/* out: 0 on success */
|
|
void* p) /* in/out: table schema object */
|
|
{
|
|
DBUG_ENTER("innobase_stat_zip_deinit");
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
static int show_innodb_vars(THD *thd, SHOW_VAR *var, char *buff)
|
|
{
|
|
innodb_export_status();
|
|
var->type= SHOW_ARRAY;
|
|
var->value= (char *) &innodb_status_variables;
|
|
return 0;
|
|
}
|
|
|
|
static SHOW_VAR innodb_status_variables_export[]= {
|
|
{"Innodb", (char*) &show_innodb_vars, SHOW_FUNC},
|
|
{NullS, NullS, SHOW_LONG}
|
|
};
|
|
|
|
static struct st_mysql_storage_engine innobase_storage_engine=
|
|
{ MYSQL_HANDLERTON_INTERFACE_VERSION };
|
|
|
|
static struct st_mysql_information_schema innobase_stat_zip=
|
|
{ MYSQL_INFORMATION_SCHEMA_INTERFACE_VERSION };
|
|
|
|
/* plugin options */
|
|
static MYSQL_SYSVAR_BOOL(checksums, innobase_use_checksums,
|
|
PLUGIN_VAR_NOCMDARG | PLUGIN_VAR_READONLY,
|
|
"Enable InnoDB checksums validation (enabled by default). "
|
|
"Disable with --skip-innodb-checksums.",
|
|
NULL, NULL, TRUE);
|
|
|
|
static MYSQL_SYSVAR_STR(data_home_dir, innobase_data_home_dir,
|
|
PLUGIN_VAR_READONLY,
|
|
"The common part for InnoDB table spaces.",
|
|
NULL, NULL, NULL);
|
|
|
|
static MYSQL_SYSVAR_BOOL(doublewrite, innobase_use_doublewrite,
|
|
PLUGIN_VAR_NOCMDARG | PLUGIN_VAR_READONLY,
|
|
"Enable InnoDB doublewrite buffer (enabled by default). "
|
|
"Disable with --skip-innodb-doublewrite.",
|
|
NULL, NULL, TRUE);
|
|
|
|
static MYSQL_SYSVAR_ULONG(fast_shutdown, innobase_fast_shutdown,
|
|
PLUGIN_VAR_OPCMDARG,
|
|
"Speeds up the shutdown process of the InnoDB storage engine. Possible "
|
|
"values are 0, 1 (faster)"
|
|
/*
|
|
NetWare can't close unclosed files, can't automatically kill remaining
|
|
threads, etc, so on this OS we disable the crash-like InnoDB shutdown.
|
|
*/
|
|
IF_NETWARE("", " or 2 (fastest - crash-like)")
|
|
".",
|
|
NULL, NULL, 1, 0, IF_NETWARE(1,2), 0);
|
|
|
|
static MYSQL_SYSVAR_BOOL(file_per_table, innobase_file_per_table,
|
|
PLUGIN_VAR_NOCMDARG | PLUGIN_VAR_READONLY,
|
|
"Stores each InnoDB table to an .ibd file in the database dir.",
|
|
NULL, NULL, FALSE);
|
|
|
|
static MYSQL_SYSVAR_ULONG(flush_log_at_trx_commit, srv_flush_log_at_trx_commit,
|
|
PLUGIN_VAR_OPCMDARG,
|
|
"Set to 0 (write and flush once per second),"
|
|
" 1 (write and flush at each commit)"
|
|
" or 2 (write at commit, flush once per second).",
|
|
NULL, NULL, 1, 0, 2, 0);
|
|
|
|
static MYSQL_SYSVAR_STR(flush_method, innobase_unix_file_flush_method,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"With which method to flush data.", NULL, NULL, NULL);
|
|
|
|
static MYSQL_SYSVAR_BOOL(locks_unsafe_for_binlog, innobase_locks_unsafe_for_binlog,
|
|
PLUGIN_VAR_NOCMDARG | PLUGIN_VAR_READONLY,
|
|
"Force InnoDB to not use next-key locking, to use only row-level locking.",
|
|
NULL, NULL, FALSE);
|
|
|
|
#ifdef UNIV_LOG_ARCHIVE
|
|
static MYSQL_SYSVAR_STR(log_arch_dir, innobase_log_arch_dir,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Where full logs should be archived.", NULL, NULL, NULL);
|
|
|
|
static MYSQL_SYSVAR_BOOL(log_archive, innobase_log_archive,
|
|
PLUGIN_VAR_OPCMDARG | PLUGIN_VAR_READONLY,
|
|
"Set to 1 if you want to have logs archived.", NULL, NULL, FALSE);
|
|
#endif /* UNIV_LOG_ARCHIVE */
|
|
|
|
static MYSQL_SYSVAR_STR(log_group_home_dir, innobase_log_group_home_dir,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Path to InnoDB log files.", NULL, NULL, NULL);
|
|
|
|
static MYSQL_SYSVAR_ULONG(max_dirty_pages_pct, srv_max_buf_pool_modified_pct,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Percentage of dirty pages allowed in bufferpool.",
|
|
NULL, NULL, 90, 0, 100, 0);
|
|
|
|
static MYSQL_SYSVAR_ULONG(max_purge_lag, srv_max_purge_lag,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Desired maximum length of the purge queue (0 = no limit)",
|
|
NULL, NULL, 0, 0, ~0L, 0);
|
|
|
|
static MYSQL_SYSVAR_BOOL(rollback_on_timeout, innobase_rollback_on_timeout,
|
|
PLUGIN_VAR_OPCMDARG | PLUGIN_VAR_READONLY,
|
|
"Roll back the complete transaction on lock wait timeout, for 4.x compatibility (disabled by default)",
|
|
NULL, NULL, FALSE);
|
|
|
|
static MYSQL_SYSVAR_BOOL(status_file, innobase_create_status_file,
|
|
PLUGIN_VAR_OPCMDARG | PLUGIN_VAR_NOSYSVAR,
|
|
"Enable SHOW INNODB STATUS output in the innodb_status.<pid> file",
|
|
NULL, NULL, FALSE);
|
|
|
|
static MYSQL_SYSVAR_BOOL(stats_on_metadata, innobase_stats_on_metadata,
|
|
PLUGIN_VAR_OPCMDARG | PLUGIN_VAR_NOSYSVAR,
|
|
"Enable statistics gathering for metadata commands such as SHOW TABLE STATUS (on by default)",
|
|
NULL, NULL, TRUE);
|
|
|
|
static MYSQL_SYSVAR_LONG(additional_mem_pool_size, innobase_additional_mem_pool_size,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Size of a memory pool InnoDB uses to store data dictionary information and other internal data structures.",
|
|
NULL, NULL, 1*1024*1024L, 512*1024L, ~0L, 1024);
|
|
|
|
static MYSQL_SYSVAR_ULONG(autoextend_increment, srv_auto_extend_increment,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Data file autoextend increment in megabytes",
|
|
NULL, NULL, 8L, 1L, 1000L, 0);
|
|
|
|
static MYSQL_SYSVAR_LONGLONG(buffer_pool_size, innobase_buffer_pool_size,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"The size of the memory buffer InnoDB uses to cache data and indexes of its tables.",
|
|
NULL, NULL, 8*1024*1024L, 5*1024*1024L, LONGLONG_MAX, 1024*1024L);
|
|
|
|
static MYSQL_SYSVAR_ULONG(commit_concurrency, srv_commit_concurrency,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Helps in performance tuning in heavily concurrent environments.",
|
|
NULL, NULL, 0, 0, 1000, 0);
|
|
|
|
static MYSQL_SYSVAR_ULONG(concurrency_tickets, srv_n_free_tickets_to_enter,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Number of times a thread is allowed to enter InnoDB within the same SQL query after it has once got the ticket",
|
|
NULL, NULL, 500L, 1L, ~0L, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(file_io_threads, innobase_file_io_threads,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Number of file I/O threads in InnoDB.",
|
|
NULL, NULL, 4, 4, 64, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(force_recovery, innobase_force_recovery,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Helps to save your data in case the disk image of the database becomes corrupt.",
|
|
NULL, NULL, 0, 0, 6, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(lock_wait_timeout, innobase_lock_wait_timeout,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Timeout in seconds an InnoDB transaction may wait for a lock before being rolled back.",
|
|
NULL, NULL, 50, 1, 1024 * 1024 * 1024, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(log_buffer_size, innobase_log_buffer_size,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"The size of the buffer which InnoDB uses to write log to the log files on disk.",
|
|
NULL, NULL, 1024*1024L, 256*1024L, ~0L, 1024);
|
|
|
|
static MYSQL_SYSVAR_LONGLONG(log_file_size, innobase_log_file_size,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Size of each log file in a log group.",
|
|
NULL, NULL, 5*1024*1024L, 1*1024*1024L, LONGLONG_MAX, 1024*1024L);
|
|
|
|
static MYSQL_SYSVAR_LONG(log_files_in_group, innobase_log_files_in_group,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Number of log files in the log group. InnoDB writes to the files in a circular fashion. Value 3 is recommended here.",
|
|
NULL, NULL, 2, 2, 100, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(mirrored_log_groups, innobase_mirrored_log_groups,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Number of identical copies of log groups we keep for the database. Currently this should be set to 1.",
|
|
NULL, NULL, 1, 1, 10, 0);
|
|
|
|
static MYSQL_SYSVAR_LONG(open_files, innobase_open_files,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"How many files at the maximum InnoDB keeps open at the same time.",
|
|
NULL, NULL, 300L, 10L, ~0L, 0);
|
|
|
|
static MYSQL_SYSVAR_ULONG(sync_spin_loops, srv_n_spin_wait_rounds,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Count of spin-loop rounds in InnoDB mutexes",
|
|
NULL, NULL, 20L, 0L, ~0L, 0);
|
|
|
|
static MYSQL_SYSVAR_ULONG(thread_concurrency, srv_thread_concurrency,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Helps in performance tuning in heavily concurrent environments. Sets the maximum number of threads allowed inside InnoDB. Value 0 will disable the thread throttling.",
|
|
NULL, NULL, 8, 0, 1000, 0);
|
|
|
|
static MYSQL_SYSVAR_ULONG(thread_sleep_delay, srv_thread_sleep_delay,
|
|
PLUGIN_VAR_RQCMDARG,
|
|
"Time of innodb thread sleeping before joining InnoDB queue (usec). Value 0 disable a sleep",
|
|
NULL, NULL, 10000L, 0L, ~0L, 0);
|
|
|
|
static MYSQL_SYSVAR_STR(data_file_path, innobase_data_file_path,
|
|
PLUGIN_VAR_RQCMDARG | PLUGIN_VAR_READONLY,
|
|
"Path to individual files and their sizes.",
|
|
NULL, NULL, NULL);
|
|
|
|
static struct st_mysql_sys_var* innobase_system_variables[]= {
|
|
MYSQL_SYSVAR(additional_mem_pool_size),
|
|
MYSQL_SYSVAR(autoextend_increment),
|
|
MYSQL_SYSVAR(buffer_pool_size),
|
|
MYSQL_SYSVAR(checksums),
|
|
MYSQL_SYSVAR(commit_concurrency),
|
|
MYSQL_SYSVAR(concurrency_tickets),
|
|
MYSQL_SYSVAR(data_file_path),
|
|
MYSQL_SYSVAR(data_home_dir),
|
|
MYSQL_SYSVAR(doublewrite),
|
|
MYSQL_SYSVAR(fast_shutdown),
|
|
MYSQL_SYSVAR(file_io_threads),
|
|
MYSQL_SYSVAR(file_per_table),
|
|
MYSQL_SYSVAR(flush_log_at_trx_commit),
|
|
MYSQL_SYSVAR(flush_method),
|
|
MYSQL_SYSVAR(force_recovery),
|
|
MYSQL_SYSVAR(locks_unsafe_for_binlog),
|
|
MYSQL_SYSVAR(lock_wait_timeout),
|
|
#ifdef UNIV_LOG_ARCHIVE
|
|
MYSQL_SYSVAR(log_arch_dir),
|
|
MYSQL_SYSVAR(log_archive),
|
|
#endif /* UNIV_LOG_ARCHIVE */
|
|
MYSQL_SYSVAR(log_buffer_size),
|
|
MYSQL_SYSVAR(log_file_size),
|
|
MYSQL_SYSVAR(log_files_in_group),
|
|
MYSQL_SYSVAR(log_group_home_dir),
|
|
MYSQL_SYSVAR(max_dirty_pages_pct),
|
|
MYSQL_SYSVAR(max_purge_lag),
|
|
MYSQL_SYSVAR(mirrored_log_groups),
|
|
MYSQL_SYSVAR(open_files),
|
|
MYSQL_SYSVAR(rollback_on_timeout),
|
|
MYSQL_SYSVAR(stats_on_metadata),
|
|
MYSQL_SYSVAR(status_file),
|
|
MYSQL_SYSVAR(support_xa),
|
|
MYSQL_SYSVAR(sync_spin_loops),
|
|
MYSQL_SYSVAR(table_locks),
|
|
MYSQL_SYSVAR(thread_concurrency),
|
|
MYSQL_SYSVAR(thread_sleep_delay),
|
|
NULL
|
|
};
|
|
|
|
mysql_declare_plugin(innobase)
|
|
{
|
|
MYSQL_STORAGE_ENGINE_PLUGIN,
|
|
&innobase_storage_engine,
|
|
innobase_hton_name,
|
|
"Innobase OY",
|
|
"Supports transactions, row-level locking, and foreign keys",
|
|
PLUGIN_LICENSE_GPL,
|
|
innobase_init, /* Plugin Init */
|
|
NULL, /* Plugin Deinit */
|
|
0x0100 /* 1.0 */,
|
|
innodb_status_variables_export,/* status variables */
|
|
innobase_system_variables, /* system variables */
|
|
NULL /* reserved */
|
|
},
|
|
{
|
|
MYSQL_INFORMATION_SCHEMA_PLUGIN,
|
|
&innobase_stat_zip,
|
|
"INNODB_ZIP",
|
|
"Innobase Oy",
|
|
"Statistics for the InnoDB compressed buffer pool",
|
|
PLUGIN_LICENSE_GPL,
|
|
innobase_stat_zip_init,
|
|
innobase_stat_zip_deinit,
|
|
0x0100 /* 1.0 */,
|
|
NULL, NULL, NULL
|
|
},
|
|
{
|
|
MYSQL_INFORMATION_SCHEMA_PLUGIN,
|
|
&innobase_stat_zip,
|
|
"INNODB_ZIP_RESET",
|
|
"Innobase Oy",
|
|
"Statistics for the InnoDB compressed buffer pool; reset cumulated counts",
|
|
PLUGIN_LICENSE_GPL,
|
|
innobase_stat_zip_reset_init,
|
|
innobase_stat_zip_deinit,
|
|
0x0100 /* 1.0 */,
|
|
NULL, NULL, NULL
|
|
}
|
|
mysql_declare_plugin_end;
|