mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 13:02:28 +01:00
bbcd542917
Archive fix by Ingo. Innodb compile fix by Monty.
2120 lines
58 KiB
C++
2120 lines
58 KiB
C++
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
|
|
|
|
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; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
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 */
|
|
|
|
|
|
/* Handler-calling-functions */
|
|
|
|
#ifdef __GNUC__
|
|
#pragma implementation // gcc: Class implementation
|
|
#endif
|
|
|
|
#include "mysql_priv.h"
|
|
#include "ha_heap.h"
|
|
#include "ha_myisam.h"
|
|
#include "ha_myisammrg.h"
|
|
#ifdef HAVE_ISAM
|
|
#include "ha_isam.h"
|
|
#include "ha_isammrg.h"
|
|
#endif
|
|
#ifdef HAVE_BERKELEY_DB
|
|
#include "ha_berkeley.h"
|
|
#endif
|
|
#ifdef HAVE_EXAMPLE_DB
|
|
#include "examples/ha_example.h"
|
|
#endif
|
|
#ifdef HAVE_ARCHIVE_DB
|
|
#include "examples/ha_archive.h"
|
|
#endif
|
|
#ifdef HAVE_CSV_DB
|
|
#include "examples/ha_tina.h"
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
#include "ha_innodb.h"
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
#include "ha_ndbcluster.h"
|
|
#endif
|
|
#ifdef HAVE_FEDERATED_DB
|
|
#include "ha_federated.h"
|
|
#endif
|
|
#include <myisampack.h>
|
|
#include <errno.h>
|
|
|
|
/* static functions defined in this file */
|
|
|
|
static int NEAR_F delete_file(const char *name,const char *ext,int extflag);
|
|
|
|
ulong ha_read_count, ha_discover_count;
|
|
|
|
static SHOW_COMP_OPTION have_yes= SHOW_OPTION_YES;
|
|
|
|
struct show_table_type_st sys_table_types[]=
|
|
{
|
|
{"MyISAM", &have_yes,
|
|
"Default engine as of MySQL 3.23 with great performance", DB_TYPE_MYISAM},
|
|
{"HEAP", &have_yes,
|
|
"Alias for MEMORY", DB_TYPE_HEAP},
|
|
{"MEMORY", &have_yes,
|
|
"Hash based, stored in memory, useful for temporary tables", DB_TYPE_HEAP},
|
|
{"MERGE", &have_yes,
|
|
"Collection of identical MyISAM tables", DB_TYPE_MRG_MYISAM},
|
|
{"MRG_MYISAM",&have_yes,
|
|
"Alias for MERGE", DB_TYPE_MRG_MYISAM},
|
|
{"ISAM", &have_isam,
|
|
"Obsolete storage engine, now replaced by MyISAM", DB_TYPE_ISAM},
|
|
{"MRG_ISAM", &have_isam,
|
|
"Obsolete storage engine, now replaced by MERGE", DB_TYPE_MRG_ISAM},
|
|
{"InnoDB", &have_innodb,
|
|
"Supports transactions, row-level locking, and foreign keys", DB_TYPE_INNODB},
|
|
{"INNOBASE", &have_innodb,
|
|
"Alias for INNODB", DB_TYPE_INNODB},
|
|
{"BDB", &have_berkeley_db,
|
|
"Supports transactions and page-level locking", DB_TYPE_BERKELEY_DB},
|
|
{"BERKELEYDB",&have_berkeley_db,
|
|
"Alias for BDB", DB_TYPE_BERKELEY_DB},
|
|
{"NDBCLUSTER", &have_ndbcluster,
|
|
"Clustered, fault-tolerant, memory-based tables", DB_TYPE_NDBCLUSTER},
|
|
{"NDB", &have_ndbcluster,
|
|
"Alias for NDBCLUSTER", DB_TYPE_NDBCLUSTER},
|
|
{"EXAMPLE",&have_example_db,
|
|
"Example storage engine", DB_TYPE_EXAMPLE_DB},
|
|
{"ARCHIVE",&have_archive_db,
|
|
"Archive storage engine", DB_TYPE_ARCHIVE_DB},
|
|
{"CSV",&have_csv_db,
|
|
"CSV storage engine", DB_TYPE_CSV_DB},
|
|
{"FEDERATED",&have_federated_db,
|
|
"Federated MySQL storage engine", DB_TYPE_FEDERATED_DB},
|
|
{NullS, NULL, NullS, DB_TYPE_UNKNOWN}
|
|
};
|
|
|
|
const char *ha_row_type[] = {
|
|
"", "FIXED", "DYNAMIC", "COMPRESSED", "REDUNDANT", "COMPACT", "?","?","?"
|
|
};
|
|
|
|
const char *tx_isolation_names[] =
|
|
{ "READ-UNCOMMITTED", "READ-COMMITTED", "REPEATABLE-READ", "SERIALIZABLE",
|
|
NullS};
|
|
TYPELIB tx_isolation_typelib= {array_elements(tx_isolation_names)-1,"",
|
|
tx_isolation_names, NULL};
|
|
|
|
static TYPELIB known_extensions= {0,"known_exts", NULL, NULL};
|
|
uint known_extensions_id= 0;
|
|
|
|
enum db_type ha_resolve_by_name(const char *name, uint namelen)
|
|
{
|
|
THD *thd= current_thd;
|
|
if (thd && !my_strcasecmp(&my_charset_latin1, name, "DEFAULT")) {
|
|
return (enum db_type) thd->variables.table_type;
|
|
}
|
|
|
|
show_table_type_st *types;
|
|
for (types= sys_table_types; types->type; types++)
|
|
{
|
|
if (!my_strcasecmp(&my_charset_latin1, name, types->type))
|
|
return (enum db_type) types->db_type;
|
|
}
|
|
return DB_TYPE_UNKNOWN;
|
|
}
|
|
|
|
const char *ha_get_storage_engine(enum db_type db_type)
|
|
{
|
|
show_table_type_st *types;
|
|
for (types= sys_table_types; types->type; types++)
|
|
{
|
|
if (db_type == types->db_type)
|
|
return types->type;
|
|
}
|
|
|
|
return "none";
|
|
}
|
|
|
|
/* Use other database handler if databasehandler is not incompiled */
|
|
|
|
enum db_type ha_checktype(enum db_type database_type)
|
|
{
|
|
show_table_type_st *types;
|
|
THD *thd= current_thd;
|
|
for (types= sys_table_types; types->type; types++)
|
|
{
|
|
if ((database_type == types->db_type) &&
|
|
(*types->value == SHOW_OPTION_YES))
|
|
return database_type;
|
|
}
|
|
|
|
switch (database_type) {
|
|
#ifndef NO_HASH
|
|
case DB_TYPE_HASH:
|
|
return (database_type);
|
|
#endif
|
|
case DB_TYPE_MRG_ISAM:
|
|
return (DB_TYPE_MRG_MYISAM);
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ((enum db_type) thd->variables.table_type != DB_TYPE_UNKNOWN ?
|
|
(enum db_type) thd->variables.table_type :
|
|
(enum db_type) global_system_variables.table_type !=
|
|
DB_TYPE_UNKNOWN ?
|
|
(enum db_type) global_system_variables.table_type : DB_TYPE_MYISAM);
|
|
} /* ha_checktype */
|
|
|
|
|
|
handler *get_new_handler(TABLE *table, enum db_type db_type)
|
|
{
|
|
switch (db_type) {
|
|
#ifndef NO_HASH
|
|
case DB_TYPE_HASH:
|
|
return new ha_hash(table);
|
|
#endif
|
|
#ifdef HAVE_ISAM
|
|
case DB_TYPE_MRG_ISAM:
|
|
return new ha_isammrg(table);
|
|
case DB_TYPE_ISAM:
|
|
return new ha_isam(table);
|
|
#else
|
|
case DB_TYPE_MRG_ISAM:
|
|
return new ha_myisammrg(table);
|
|
#endif
|
|
#ifdef HAVE_BERKELEY_DB
|
|
case DB_TYPE_BERKELEY_DB:
|
|
return new ha_berkeley(table);
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
case DB_TYPE_INNODB:
|
|
return new ha_innobase(table);
|
|
#endif
|
|
#ifdef HAVE_EXAMPLE_DB
|
|
case DB_TYPE_EXAMPLE_DB:
|
|
return new ha_example(table);
|
|
#endif
|
|
#ifdef HAVE_ARCHIVE_DB
|
|
case DB_TYPE_ARCHIVE_DB:
|
|
return new ha_archive(table);
|
|
#endif
|
|
#ifdef HAVE_FEDERATED_DB
|
|
case DB_TYPE_FEDERATED_DB:
|
|
return new ha_federated(table);
|
|
#endif
|
|
#ifdef HAVE_CSV_DB
|
|
case DB_TYPE_CSV_DB:
|
|
return new ha_tina(table);
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
case DB_TYPE_NDBCLUSTER:
|
|
return new ha_ndbcluster(table);
|
|
#endif
|
|
case DB_TYPE_HEAP:
|
|
return new ha_heap(table);
|
|
default: // should never happen
|
|
{
|
|
enum db_type def=(enum db_type) current_thd->variables.table_type;
|
|
/* Try first with 'default table type' */
|
|
if (db_type != def)
|
|
return get_new_handler(table, def);
|
|
}
|
|
/* Fall back to MyISAM */
|
|
case DB_TYPE_MYISAM:
|
|
return new ha_myisam(table);
|
|
case DB_TYPE_MRG_MYISAM:
|
|
return new ha_myisammrg(table);
|
|
}
|
|
}
|
|
|
|
bool ha_caching_allowed(THD* thd, char* table_key,
|
|
uint key_length, uint8 cache_type)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (cache_type == HA_CACHE_TBL_ASKTRANSACT)
|
|
return innobase_query_caching_of_table_permitted(thd, table_key, key_length);
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
|
|
/*
|
|
Register handler error messages for use with my_error().
|
|
|
|
SYNOPSIS
|
|
ha_init_errors()
|
|
|
|
RETURN
|
|
0 OK
|
|
!= 0 Error
|
|
*/
|
|
|
|
static int ha_init_errors(void)
|
|
{
|
|
#define SETMSG(nr, msg) errmsgs[(nr) - HA_ERR_FIRST]= (msg)
|
|
const char **errmsgs;
|
|
|
|
/* Allocate a pointer array for the error message strings. */
|
|
/* Zerofill it to avoid uninitialized gaps. */
|
|
if (! (errmsgs= (const char**) my_malloc(HA_ERR_ERRORS * sizeof(char*),
|
|
MYF(MY_WME | MY_ZEROFILL))))
|
|
return 1;
|
|
|
|
/* Set the dedicated error messages. */
|
|
SETMSG(HA_ERR_KEY_NOT_FOUND, ER(ER_KEY_NOT_FOUND));
|
|
SETMSG(HA_ERR_FOUND_DUPP_KEY, ER(ER_DUP_KEY));
|
|
SETMSG(HA_ERR_RECORD_CHANGED, "Update wich is recoverable");
|
|
SETMSG(HA_ERR_WRONG_INDEX, "Wrong index given to function");
|
|
SETMSG(HA_ERR_CRASHED, ER(ER_NOT_KEYFILE));
|
|
SETMSG(HA_ERR_WRONG_IN_RECORD, ER(ER_CRASHED_ON_USAGE));
|
|
SETMSG(HA_ERR_OUT_OF_MEM, "Table handler out of memory");
|
|
SETMSG(HA_ERR_NOT_A_TABLE, "Incorrect file format '%.64s'");
|
|
SETMSG(HA_ERR_WRONG_COMMAND, "Command not supported");
|
|
SETMSG(HA_ERR_OLD_FILE, ER(ER_OLD_KEYFILE));
|
|
SETMSG(HA_ERR_NO_ACTIVE_RECORD, "No record read in update");
|
|
SETMSG(HA_ERR_RECORD_DELETED, "Intern record deleted");
|
|
SETMSG(HA_ERR_RECORD_FILE_FULL, ER(ER_RECORD_FILE_FULL));
|
|
SETMSG(HA_ERR_INDEX_FILE_FULL, "No more room in index file '%.64s'");
|
|
SETMSG(HA_ERR_END_OF_FILE, "End in next/prev/first/last");
|
|
SETMSG(HA_ERR_UNSUPPORTED, ER(ER_ILLEGAL_HA));
|
|
SETMSG(HA_ERR_TO_BIG_ROW, "Too big row");
|
|
SETMSG(HA_WRONG_CREATE_OPTION, "Wrong create option");
|
|
SETMSG(HA_ERR_FOUND_DUPP_UNIQUE, ER(ER_DUP_UNIQUE));
|
|
SETMSG(HA_ERR_UNKNOWN_CHARSET, "Can't open charset");
|
|
SETMSG(HA_ERR_WRONG_MRG_TABLE_DEF, ER(ER_WRONG_MRG_TABLE));
|
|
SETMSG(HA_ERR_CRASHED_ON_REPAIR, ER(ER_CRASHED_ON_REPAIR));
|
|
SETMSG(HA_ERR_CRASHED_ON_USAGE, ER(ER_CRASHED_ON_USAGE));
|
|
SETMSG(HA_ERR_LOCK_WAIT_TIMEOUT, ER(ER_LOCK_WAIT_TIMEOUT));
|
|
SETMSG(HA_ERR_LOCK_TABLE_FULL, ER(ER_LOCK_TABLE_FULL));
|
|
SETMSG(HA_ERR_READ_ONLY_TRANSACTION, ER(ER_READ_ONLY_TRANSACTION));
|
|
SETMSG(HA_ERR_LOCK_DEADLOCK, ER(ER_LOCK_DEADLOCK));
|
|
SETMSG(HA_ERR_CANNOT_ADD_FOREIGN, ER(ER_CANNOT_ADD_FOREIGN));
|
|
SETMSG(HA_ERR_NO_REFERENCED_ROW, ER(ER_NO_REFERENCED_ROW));
|
|
SETMSG(HA_ERR_ROW_IS_REFERENCED, ER(ER_ROW_IS_REFERENCED));
|
|
SETMSG(HA_ERR_NO_SAVEPOINT, "No savepoint with that name");
|
|
SETMSG(HA_ERR_NON_UNIQUE_BLOCK_SIZE, "Non unique key block size");
|
|
SETMSG(HA_ERR_NO_SUCH_TABLE, "No such table: '%.64s'");
|
|
SETMSG(HA_ERR_TABLE_EXIST, ER(ER_TABLE_EXISTS_ERROR));
|
|
SETMSG(HA_ERR_NO_CONNECTION, "Could not connect to storage engine");
|
|
|
|
/* Register the error messages for use with my_error(). */
|
|
return my_error_register(errmsgs, HA_ERR_FIRST, HA_ERR_LAST);
|
|
}
|
|
|
|
|
|
/*
|
|
Unregister handler error messages.
|
|
|
|
SYNOPSIS
|
|
ha_finish_errors()
|
|
|
|
RETURN
|
|
0 OK
|
|
!= 0 Error
|
|
*/
|
|
|
|
static int ha_finish_errors(void)
|
|
{
|
|
const char **errmsgs;
|
|
|
|
/* Allocate a pointer array for the error message strings. */
|
|
if (! (errmsgs= my_error_unregister(HA_ERR_FIRST, HA_ERR_LAST)))
|
|
return 1;
|
|
my_free((gptr) errmsgs, MYF(0));
|
|
return 0;
|
|
}
|
|
|
|
|
|
int ha_init()
|
|
{
|
|
int error= 0;
|
|
if (ha_init_errors())
|
|
return 1;
|
|
#ifdef HAVE_BERKELEY_DB
|
|
if (have_berkeley_db == SHOW_OPTION_YES)
|
|
{
|
|
if (berkeley_init())
|
|
{
|
|
have_berkeley_db= SHOW_OPTION_DISABLED; // If we couldn't use handler
|
|
error= 1;
|
|
}
|
|
else
|
|
opt_using_transactions=1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (have_innodb == SHOW_OPTION_YES)
|
|
{
|
|
if (innobase_init())
|
|
{
|
|
have_innodb= SHOW_OPTION_DISABLED; // If we couldn't use handler
|
|
error= 1;
|
|
}
|
|
else
|
|
opt_using_transactions=1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
{
|
|
if (ndbcluster_init())
|
|
{
|
|
have_ndbcluster= SHOW_OPTION_DISABLED;
|
|
error= 1;
|
|
}
|
|
else
|
|
opt_using_transactions=1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_ARCHIVE_DB
|
|
if (have_archive_db == SHOW_OPTION_YES)
|
|
{
|
|
if (archive_db_init())
|
|
{
|
|
have_archive_db= SHOW_OPTION_DISABLED;
|
|
error= 1;
|
|
}
|
|
}
|
|
#endif
|
|
return error;
|
|
}
|
|
|
|
/* close, flush or restart databases */
|
|
/* Ignore this for other databases than ours */
|
|
|
|
int ha_panic(enum ha_panic_function flag)
|
|
{
|
|
int error=0;
|
|
#ifndef NO_HASH
|
|
error|=h_panic(flag); /* fix hash */
|
|
#endif
|
|
#ifdef HAVE_ISAM
|
|
error|=mrg_panic(flag);
|
|
error|=nisam_panic(flag);
|
|
#endif
|
|
error|=heap_panic(flag);
|
|
error|=mi_panic(flag);
|
|
error|=myrg_panic(flag);
|
|
#ifdef HAVE_BERKELEY_DB
|
|
if (have_berkeley_db == SHOW_OPTION_YES)
|
|
error|=berkeley_end();
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (have_innodb == SHOW_OPTION_YES)
|
|
error|=innobase_end();
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
error|=ndbcluster_end();
|
|
#endif
|
|
#ifdef HAVE_ARCHIVE_DB
|
|
if (have_archive_db == SHOW_OPTION_YES)
|
|
error|= archive_db_end();
|
|
#endif
|
|
if (ha_finish_errors())
|
|
error= 1;
|
|
return error;
|
|
} /* ha_panic */
|
|
|
|
void ha_drop_database(char* path)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (have_innodb == SHOW_OPTION_YES)
|
|
innobase_drop_database(path);
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
ndbcluster_drop_database(path);
|
|
#endif
|
|
}
|
|
|
|
void ha_close_connection(THD* thd)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (have_innodb == SHOW_OPTION_YES)
|
|
innobase_close_connection(thd);
|
|
#endif
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
ndbcluster_close_connection(thd);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
This is used to commit or rollback a single statement depending on the value
|
|
of error. Note that if the autocommit is on, then the following call inside
|
|
InnoDB will commit or rollback the whole transaction (= the statement). The
|
|
autocommit mechanism built into InnoDB is based on counting locks, but if
|
|
the user has used LOCK TABLES then that mechanism does not know to do the
|
|
commit.
|
|
*/
|
|
|
|
int ha_autocommit_or_rollback(THD *thd, int error)
|
|
{
|
|
DBUG_ENTER("ha_autocommit_or_rollback");
|
|
#ifdef USING_TRANSACTIONS
|
|
if (opt_using_transactions)
|
|
{
|
|
if (!error)
|
|
{
|
|
if (ha_commit_stmt(thd))
|
|
error=1;
|
|
}
|
|
else
|
|
(void) ha_rollback_stmt(thd);
|
|
|
|
thd->variables.tx_isolation=thd->session_tx_isolation;
|
|
}
|
|
#endif
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*
|
|
This function is called when MySQL writes the log segment of a
|
|
transaction to the binlog. It is called when the LOCK_log mutex is
|
|
reserved. Here we communicate to transactional table handlers what
|
|
binlog position corresponds to the current transaction. The handler
|
|
can store it and in recovery print to the user, so that the user
|
|
knows from what position in the binlog to start possible
|
|
roll-forward, for example, if the crashed server was a slave in
|
|
replication. This function also calls the commit of the table
|
|
handler, because the order of transactions in the log of the table
|
|
handler must be the same as in the binlog.
|
|
NOTE that to eliminate the bottleneck of the group commit, we do not
|
|
flush the handler log files here, but only later in a call of
|
|
ha_commit_complete().
|
|
|
|
arguments:
|
|
thd: the thread handle of the current connection
|
|
log_file_name: latest binlog file name
|
|
end_offset: the offset in the binlog file up to which we wrote
|
|
return value: 0 if success, 1 if error
|
|
*/
|
|
|
|
int ha_report_binlog_offset_and_commit(THD *thd,
|
|
char *log_file_name,
|
|
my_off_t end_offset)
|
|
{
|
|
int error= 0;
|
|
#ifdef HAVE_INNOBASE_DB
|
|
THD_TRANS *trans;
|
|
trans = &thd->transaction.all;
|
|
if (trans->innodb_active_trans)
|
|
{
|
|
/*
|
|
If we updated some InnoDB tables (innodb_active_trans is true), the
|
|
binlog coords will be reported into InnoDB during the InnoDB commit
|
|
(innobase_report_binlog_offset_and_commit). But if we updated only
|
|
non-InnoDB tables, we need an explicit call to report it.
|
|
*/
|
|
if ((error=innobase_report_binlog_offset_and_commit(thd,
|
|
trans->innobase_tid,
|
|
log_file_name,
|
|
end_offset)))
|
|
{
|
|
my_error(ER_ERROR_DURING_COMMIT, MYF(0), error);
|
|
error=1;
|
|
}
|
|
}
|
|
else if (opt_innodb_safe_binlog) // Don't report if not useful
|
|
innobase_store_binlog_offset_and_flush_log(log_file_name, end_offset);
|
|
#endif
|
|
return error;
|
|
}
|
|
|
|
/*
|
|
Flushes the handler log files (if my.cnf settings do not free us from it)
|
|
after we have called ha_report_binlog_offset_and_commit(). To eliminate
|
|
the bottleneck from the group commit, this should be called when
|
|
LOCK_log has been released in log.cc.
|
|
|
|
arguments:
|
|
thd: the thread handle of the current connection
|
|
return value: always 0
|
|
*/
|
|
|
|
int ha_commit_complete(THD *thd)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
THD_TRANS *trans;
|
|
trans = &thd->transaction.all;
|
|
if (trans->innobase_tid)
|
|
{
|
|
innobase_commit_complete(trans->innobase_tid);
|
|
|
|
trans->innodb_active_trans=0;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
This function should be called when MySQL sends rows of a SELECT result set
|
|
or the EOF mark to the client. It releases a possible adaptive hash index
|
|
S-latch held by thd in InnoDB and also releases a possible InnoDB query
|
|
FIFO ticket to enter InnoDB. To save CPU time, InnoDB allows a thd to
|
|
keep them over several calls of the InnoDB handler interface when a join
|
|
is executed. But when we let the control to pass to the client they have
|
|
to be released because if the application program uses mysql_use_result(),
|
|
it may deadlock on the S-latch if the application on another connection
|
|
performs another SQL query. In MySQL-4.1 this is even more important because
|
|
there a connection can have several SELECT queries open at the same time.
|
|
|
|
arguments:
|
|
thd: the thread handle of the current connection
|
|
return value: always 0
|
|
*/
|
|
|
|
int ha_release_temporary_latches(THD *thd)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
THD_TRANS *trans;
|
|
trans = &thd->transaction.all;
|
|
if (trans->innobase_tid)
|
|
innobase_release_temporary_latches(trans->innobase_tid);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Export statistics for different engines. Currently we use it only for
|
|
InnoDB.
|
|
*/
|
|
|
|
int ha_update_statistics()
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (opt_innodb)
|
|
innodb_export_status();
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
int ha_commit_trans(THD *thd, THD_TRANS* trans)
|
|
{
|
|
int error=0;
|
|
DBUG_ENTER("ha_commit_trans");
|
|
#ifdef USING_TRANSACTIONS
|
|
if (opt_using_transactions)
|
|
{
|
|
bool transaction_commited= 0;
|
|
bool operation_done= 0, need_start_waiters= 0;
|
|
|
|
/* If transaction has done some updates to tables */
|
|
if (trans == &thd->transaction.all && mysql_bin_log.is_open() &&
|
|
my_b_tell(&thd->transaction.trans_log))
|
|
{
|
|
if ((error= wait_if_global_read_lock(thd, 0, 0)))
|
|
{
|
|
/*
|
|
Note that ROLLBACK [TO SAVEPOINT] does not have this test; it's
|
|
because ROLLBACK never updates data, so needn't wait on the lock.
|
|
*/
|
|
my_error(ER_ERROR_DURING_COMMIT, MYF(0), error);
|
|
error= 1;
|
|
}
|
|
else
|
|
need_start_waiters= 1;
|
|
if (mysql_bin_log.is_open())
|
|
{
|
|
mysql_bin_log.write(thd, &thd->transaction.trans_log, 1);
|
|
statistic_increment(binlog_cache_use, &LOCK_status);
|
|
if (thd->transaction.trans_log.disk_writes != 0)
|
|
{
|
|
/*
|
|
We have to do this after addition of trans_log to main binlog since
|
|
this operation can cause flushing of end of trans_log to disk.
|
|
*/
|
|
statistic_increment(binlog_cache_disk_use, &LOCK_status);
|
|
thd->transaction.trans_log.disk_writes= 0;
|
|
}
|
|
reinit_io_cache(&thd->transaction.trans_log,
|
|
WRITE_CACHE, (my_off_t) 0, 0, 1);
|
|
thd->transaction.trans_log.end_of_file= max_binlog_cache_size;
|
|
}
|
|
}
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (trans->ndb_tid)
|
|
{
|
|
if ((error=ndbcluster_commit(thd,trans->ndb_tid)))
|
|
{
|
|
if (error == -1)
|
|
my_message(ER_ERROR_DURING_COMMIT, ER(ER_ERROR_DURING_COMMIT),
|
|
MYF(0));
|
|
error=1;
|
|
}
|
|
if (trans == &thd->transaction.all)
|
|
operation_done= transaction_commited= 1;
|
|
trans->ndb_tid=0;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_BERKELEY_DB
|
|
if (trans->bdb_tid)
|
|
{
|
|
if ((error=berkeley_commit(thd,trans->bdb_tid)))
|
|
{
|
|
my_error(ER_ERROR_DURING_COMMIT, MYF(0), error);
|
|
error=1;
|
|
}
|
|
else
|
|
if (!(thd->options & OPTION_BEGIN))
|
|
transaction_commited= 1;
|
|
trans->bdb_tid=0;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (trans->innobase_tid)
|
|
{
|
|
if ((error=innobase_commit(thd,trans->innobase_tid)))
|
|
{
|
|
my_error(ER_ERROR_DURING_COMMIT, MYF(0), error);
|
|
error=1;
|
|
}
|
|
trans->innodb_active_trans=0;
|
|
if (trans == &thd->transaction.all)
|
|
operation_done= transaction_commited= 1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_QUERY_CACHE
|
|
if (transaction_commited && thd->transaction.changed_tables)
|
|
query_cache.invalidate(thd->transaction.changed_tables);
|
|
#endif /*HAVE_QUERY_CACHE*/
|
|
if (error && trans == &thd->transaction.all && mysql_bin_log.is_open())
|
|
sql_print_error("Got error during commit; Binlog is not up to date!");
|
|
thd->variables.tx_isolation=thd->session_tx_isolation;
|
|
if (operation_done)
|
|
{
|
|
statistic_increment(thd->status_var.ha_commit_count,&LOCK_status);
|
|
thd->transaction.cleanup();
|
|
}
|
|
if (need_start_waiters)
|
|
start_waiting_global_read_lock(thd);
|
|
}
|
|
#endif // using transactions
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
int ha_rollback_trans(THD *thd, THD_TRANS *trans)
|
|
{
|
|
int error=0;
|
|
DBUG_ENTER("ha_rollback_trans");
|
|
#ifdef USING_TRANSACTIONS
|
|
if (opt_using_transactions)
|
|
{
|
|
bool operation_done=0;
|
|
/*
|
|
As rollback can be 30 times slower than insert in InnoDB, and user may
|
|
not know there's rollback (if it's because of a dupl row), better warn.
|
|
*/
|
|
const char *save_proc_info= thd->proc_info;
|
|
thd->proc_info= "Rolling back";
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (trans->ndb_tid)
|
|
{
|
|
if ((error=ndbcluster_rollback(thd, trans->ndb_tid)))
|
|
{
|
|
if (error == -1)
|
|
my_message(ER_ERROR_DURING_ROLLBACK, ER(ER_ERROR_DURING_ROLLBACK),
|
|
MYF(0));
|
|
error=1;
|
|
}
|
|
trans->ndb_tid = 0;
|
|
operation_done=1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_BERKELEY_DB
|
|
if (trans->bdb_tid)
|
|
{
|
|
if ((error=berkeley_rollback(thd, trans->bdb_tid)))
|
|
{
|
|
my_error(ER_ERROR_DURING_ROLLBACK, MYF(0), error);
|
|
error=1;
|
|
}
|
|
trans->bdb_tid=0;
|
|
operation_done=1;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if (trans->innobase_tid)
|
|
{
|
|
if ((error=innobase_rollback(thd, trans->innobase_tid)))
|
|
{
|
|
my_error(ER_ERROR_DURING_ROLLBACK, MYF(0), error);
|
|
error=1;
|
|
}
|
|
trans->innodb_active_trans=0;
|
|
operation_done=1;
|
|
}
|
|
#endif
|
|
if ((trans == &thd->transaction.all) && mysql_bin_log.is_open())
|
|
{
|
|
/*
|
|
Update the binary log with a BEGIN/ROLLBACK block if we have
|
|
cached some queries and we updated some non-transactional
|
|
table. Such cases should be rare (updating a
|
|
non-transactional table inside a transaction...). Count disk
|
|
writes to trans_log in any case.
|
|
*/
|
|
if (my_b_tell(&thd->transaction.trans_log))
|
|
{
|
|
if (unlikely(thd->options & OPTION_STATUS_NO_TRANS_UPDATE))
|
|
mysql_bin_log.write(thd, &thd->transaction.trans_log, 0);
|
|
statistic_increment(binlog_cache_use, &LOCK_status);
|
|
if (thd->transaction.trans_log.disk_writes != 0)
|
|
{
|
|
/*
|
|
We have to do this after addition of trans_log to main binlog since
|
|
this operation can cause flushing of end of trans_log to disk.
|
|
*/
|
|
statistic_increment(binlog_cache_disk_use, &LOCK_status);
|
|
thd->transaction.trans_log.disk_writes= 0;
|
|
}
|
|
}
|
|
/* Flushed or not, empty the binlog cache */
|
|
reinit_io_cache(&thd->transaction.trans_log,
|
|
WRITE_CACHE, (my_off_t) 0, 0, 1);
|
|
thd->transaction.trans_log.end_of_file= max_binlog_cache_size;
|
|
if (operation_done)
|
|
thd->transaction.cleanup();
|
|
}
|
|
thd->variables.tx_isolation=thd->session_tx_isolation;
|
|
if (operation_done)
|
|
statistic_increment(thd->status_var.ha_rollback_count,&LOCK_status);
|
|
thd->proc_info= save_proc_info;
|
|
}
|
|
#endif /* USING_TRANSACTIONS */
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
Rolls the current transaction back to a savepoint.
|
|
Return value: 0 if success, 1 if there was not a savepoint of the given
|
|
name.
|
|
NOTE: how do we handle this (unlikely but legal) case:
|
|
[transaction] + [update to non-trans table] + [rollback to savepoint] ?
|
|
The problem occurs when a savepoint is before the update to the
|
|
non-transactional table. Then when there's a rollback to the savepoint, if we
|
|
simply truncate the binlog cache, we lose the part of the binlog cache where
|
|
the update is. If we want to not lose it, we need to write the SAVEPOINT
|
|
command and the ROLLBACK TO SAVEPOINT command to the binlog cache. The latter
|
|
is easy: it's just write at the end of the binlog cache, but the former
|
|
should be *inserted* to the place where the user called SAVEPOINT. The
|
|
solution is that when the user calls SAVEPOINT, we write it to the binlog
|
|
cache (so no need to later insert it). As transactions are never intermixed
|
|
in the binary log (i.e. they are serialized), we won't have conflicts with
|
|
savepoint names when using mysqlbinlog or in the slave SQL thread.
|
|
Then when ROLLBACK TO SAVEPOINT is called, if we updated some
|
|
non-transactional table, we don't truncate the binlog cache but instead write
|
|
ROLLBACK TO SAVEPOINT to it; otherwise we truncate the binlog cache (which
|
|
will chop the SAVEPOINT command from the binlog cache, which is good as in
|
|
that case there is no need to have it in the binlog).
|
|
*/
|
|
|
|
int ha_rollback_to_savepoint(THD *thd, char *savepoint_name)
|
|
{
|
|
my_off_t binlog_cache_pos=0;
|
|
bool operation_done=0;
|
|
int error=0;
|
|
DBUG_ENTER("ha_rollback_to_savepoint");
|
|
#ifdef USING_TRANSACTIONS
|
|
if (opt_using_transactions)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
/*
|
|
Retrieve the trans_log binlog cache position corresponding to the
|
|
savepoint, and if the rollback is successful inside InnoDB reset the write
|
|
position in the binlog cache to what it was at the savepoint.
|
|
*/
|
|
if ((error=innobase_rollback_to_savepoint(thd, savepoint_name,
|
|
&binlog_cache_pos)))
|
|
{
|
|
my_error(ER_ERROR_DURING_ROLLBACK, MYF(0), error);
|
|
error=1;
|
|
}
|
|
else if (mysql_bin_log.is_open())
|
|
{
|
|
/*
|
|
Write ROLLBACK TO SAVEPOINT to the binlog cache if we have updated some
|
|
non-transactional table. Otherwise, truncate the binlog cache starting
|
|
from the SAVEPOINT command.
|
|
*/
|
|
if (unlikely((thd->options & OPTION_STATUS_NO_TRANS_UPDATE) &&
|
|
my_b_tell(&thd->transaction.trans_log)))
|
|
{
|
|
Query_log_event qinfo(thd, thd->query, thd->query_length, TRUE, FALSE);
|
|
if (mysql_bin_log.write(&qinfo))
|
|
error= 1;
|
|
}
|
|
else
|
|
reinit_io_cache(&thd->transaction.trans_log, WRITE_CACHE,
|
|
binlog_cache_pos, 0, 0);
|
|
}
|
|
operation_done=1;
|
|
#endif
|
|
if (operation_done)
|
|
statistic_increment(thd->status_var.ha_rollback_count,&LOCK_status);
|
|
}
|
|
#endif /* USING_TRANSACTIONS */
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
Sets a transaction savepoint.
|
|
Return value: always 0, that is, succeeds always
|
|
*/
|
|
|
|
int ha_savepoint(THD *thd, char *savepoint_name)
|
|
{
|
|
int error=0;
|
|
DBUG_ENTER("ha_savepoint");
|
|
#ifdef USING_TRANSACTIONS
|
|
if (opt_using_transactions)
|
|
{
|
|
/* Write it to the binary log (see comments of ha_rollback_to_savepoint) */
|
|
if (mysql_bin_log.is_open())
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
innobase_savepoint(thd,savepoint_name,
|
|
my_b_tell(&thd->transaction.trans_log));
|
|
#endif
|
|
Query_log_event qinfo(thd, thd->query, thd->query_length, TRUE, FALSE);
|
|
if (mysql_bin_log.write(&qinfo))
|
|
error= 1;
|
|
}
|
|
#ifdef HAVE_INNOBASE_DB
|
|
else
|
|
innobase_savepoint(thd,savepoint_name,0);
|
|
#endif
|
|
}
|
|
#endif /* USING_TRANSACTIONS */
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
int ha_start_consistent_snapshot(THD *thd)
|
|
{
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if ((have_innodb == SHOW_OPTION_YES) &&
|
|
!innobase_start_trx_and_assign_read_view(thd))
|
|
return 0;
|
|
#endif
|
|
/*
|
|
Same idea as when one wants to CREATE TABLE in one engine which does not
|
|
exist:
|
|
*/
|
|
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN, ER_UNKNOWN_ERROR,
|
|
"This MySQL server does not support any "
|
|
"consistent-read capable storage engine");
|
|
return 0;
|
|
}
|
|
|
|
|
|
bool ha_flush_logs()
|
|
{
|
|
bool result=0;
|
|
#ifdef HAVE_BERKELEY_DB
|
|
if ((have_berkeley_db == SHOW_OPTION_YES) &&
|
|
berkeley_flush_logs())
|
|
result=1;
|
|
#endif
|
|
#ifdef HAVE_INNOBASE_DB
|
|
if ((have_innodb == SHOW_OPTION_YES) &&
|
|
innobase_flush_logs())
|
|
result=1;
|
|
#endif
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
This should return ENOENT if the file doesn't exists.
|
|
The .frm file will be deleted only if we return 0 or ENOENT
|
|
*/
|
|
|
|
int ha_delete_table(enum db_type table_type, const char *path)
|
|
{
|
|
handler *file;
|
|
char tmp_path[FN_REFLEN];
|
|
|
|
/* DB_TYPE_UNKNOWN is used in ALTER TABLE when renaming only .frm files */
|
|
if (table_type == DB_TYPE_UNKNOWN ||
|
|
! (file=get_new_handler((TABLE*) 0, table_type)))
|
|
return ENOENT;
|
|
|
|
if (lower_case_table_names == 2 && !(file->table_flags() & HA_FILE_BASED))
|
|
{
|
|
/* Ensure that table handler get path in lower case */
|
|
strmov(tmp_path, path);
|
|
my_casedn_str(files_charset_info, tmp_path);
|
|
path= tmp_path;
|
|
}
|
|
int error=file->delete_table(path);
|
|
delete file;
|
|
return error;
|
|
}
|
|
|
|
|
|
void ha_store_ptr(byte *buff, uint pack_length, my_off_t pos)
|
|
{
|
|
switch (pack_length) {
|
|
#if SIZEOF_OFF_T > 4
|
|
case 8: mi_int8store(buff,pos); break;
|
|
case 7: mi_int7store(buff,pos); break;
|
|
case 6: mi_int6store(buff,pos); break;
|
|
case 5: mi_int5store(buff,pos); break;
|
|
#endif
|
|
case 4: mi_int4store(buff,pos); break;
|
|
case 3: mi_int3store(buff,pos); break;
|
|
case 2: mi_int2store(buff,(uint) pos); break;
|
|
case 1: buff[0]= (uchar) pos; break;
|
|
}
|
|
return;
|
|
}
|
|
|
|
my_off_t ha_get_ptr(byte *ptr, uint pack_length)
|
|
{
|
|
my_off_t pos;
|
|
switch (pack_length) {
|
|
#if SIZEOF_OFF_T > 4
|
|
case 8:
|
|
pos= (my_off_t) mi_uint8korr(ptr);
|
|
break;
|
|
case 7:
|
|
pos= (my_off_t) mi_uint7korr(ptr);
|
|
break;
|
|
case 6:
|
|
pos= (my_off_t) mi_uint6korr(ptr);
|
|
break;
|
|
case 5:
|
|
pos= (my_off_t) mi_uint5korr(ptr);
|
|
break;
|
|
#endif
|
|
case 4:
|
|
pos= (my_off_t) mi_uint4korr(ptr);
|
|
break;
|
|
case 3:
|
|
pos= (my_off_t) mi_uint3korr(ptr);
|
|
break;
|
|
case 2:
|
|
pos= (my_off_t) mi_uint2korr(ptr);
|
|
break;
|
|
case 1:
|
|
pos= (my_off_t) mi_uint2korr(ptr);
|
|
break;
|
|
default:
|
|
pos=0; // Impossible
|
|
break;
|
|
}
|
|
return pos;
|
|
}
|
|
|
|
/****************************************************************************
|
|
** General handler functions
|
|
****************************************************************************/
|
|
|
|
/* Open database-handler. Try O_RDONLY if can't open as O_RDWR */
|
|
/* Don't wait for locks if not HA_OPEN_WAIT_IF_LOCKED is set */
|
|
|
|
int handler::ha_open(const char *name, int mode, int test_if_locked)
|
|
{
|
|
int error;
|
|
DBUG_ENTER("handler::ha_open");
|
|
DBUG_PRINT("enter",("name: %s db_type: %d db_stat: %d mode: %d lock_test: %d",
|
|
name, table->s->db_type, table->db_stat, mode,
|
|
test_if_locked));
|
|
|
|
if ((error=open(name,mode,test_if_locked)))
|
|
{
|
|
if ((error == EACCES || error == EROFS) && mode == O_RDWR &&
|
|
(table->db_stat & HA_TRY_READ_ONLY))
|
|
{
|
|
table->db_stat|=HA_READ_ONLY;
|
|
error=open(name,O_RDONLY,test_if_locked);
|
|
}
|
|
}
|
|
if (error)
|
|
{
|
|
my_errno=error; /* Safeguard */
|
|
DBUG_PRINT("error",("error: %d errno: %d",error,errno));
|
|
}
|
|
else
|
|
{
|
|
if (table->s->db_options_in_use & HA_OPTION_READ_ONLY_DATA)
|
|
table->db_stat|=HA_READ_ONLY;
|
|
(void) extra(HA_EXTRA_NO_READCHECK); // Not needed in SQL
|
|
|
|
if (!alloc_root_inited(&table->mem_root)) // If temporary table
|
|
ref=(byte*) sql_alloc(ALIGN_SIZE(ref_length)*2);
|
|
else
|
|
ref=(byte*) alloc_root(&table->mem_root, ALIGN_SIZE(ref_length)*2);
|
|
if (!ref)
|
|
{
|
|
close();
|
|
error=HA_ERR_OUT_OF_MEM;
|
|
}
|
|
else
|
|
dupp_ref=ref+ALIGN_SIZE(ref_length);
|
|
}
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/*
|
|
Read first row (only) from a table
|
|
This is never called for InnoDB or BDB tables, as these table types
|
|
has the HA_NOT_EXACT_COUNT set.
|
|
*/
|
|
|
|
int handler::read_first_row(byte * buf, uint primary_key)
|
|
{
|
|
register int error;
|
|
DBUG_ENTER("handler::read_first_row");
|
|
|
|
statistic_increment(current_thd->status_var.ha_read_first_count,&LOCK_status);
|
|
|
|
/*
|
|
If there is very few deleted rows in the table, find the first row by
|
|
scanning the table.
|
|
TODO remove the test for HA_READ_ORDER
|
|
*/
|
|
if (deleted < 10 || primary_key >= MAX_KEY ||
|
|
!(index_flags(primary_key, 0, 0) & HA_READ_ORDER))
|
|
{
|
|
(void) ha_rnd_init(1);
|
|
while ((error= rnd_next(buf)) == HA_ERR_RECORD_DELETED) ;
|
|
(void) ha_rnd_end();
|
|
}
|
|
else
|
|
{
|
|
/* Find the first row through the primary key */
|
|
(void) ha_index_init(primary_key);
|
|
error=index_first(buf);
|
|
(void) ha_index_end();
|
|
}
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
Generate the next auto-increment number based on increment and offset
|
|
|
|
In most cases increment= offset= 1, in which case we get:
|
|
1,2,3,4,5,...
|
|
If increment=10 and offset=5 and previous number is 1, we get:
|
|
1,5,15,25,35,...
|
|
*/
|
|
|
|
inline ulonglong
|
|
next_insert_id(ulonglong nr,struct system_variables *variables)
|
|
{
|
|
nr= (((nr+ variables->auto_increment_increment -
|
|
variables->auto_increment_offset)) /
|
|
(ulonglong) variables->auto_increment_increment);
|
|
return (nr* (ulonglong) variables->auto_increment_increment +
|
|
variables->auto_increment_offset);
|
|
}
|
|
|
|
|
|
/*
|
|
Updates columns with type NEXT_NUMBER if:
|
|
|
|
- If column value is set to NULL (in which case
|
|
auto_increment_field_not_null is 0)
|
|
- If column is set to 0 and (sql_mode & MODE_NO_AUTO_VALUE_ON_ZERO) is not
|
|
set. In the future we will only set NEXT_NUMBER fields if one sets them
|
|
to NULL (or they are not included in the insert list).
|
|
|
|
|
|
There are two different cases when the above is true:
|
|
|
|
- thd->next_insert_id == 0 (This is the normal case)
|
|
In this case we set the set the column for the first row to the value
|
|
next_insert_id(get_auto_increment(column))) which is normally
|
|
max-used-column-value +1.
|
|
|
|
We call get_auto_increment() only for the first row in a multi-row
|
|
statement. For the following rows we generate new numbers based on the
|
|
last used number.
|
|
|
|
- thd->next_insert_id != 0. This happens when we have read a statement
|
|
from the binary log or when one has used SET LAST_INSERT_ID=#.
|
|
|
|
In this case we will set the column to the value of next_insert_id.
|
|
The next row will be given the id
|
|
next_insert_id(next_insert_id)
|
|
|
|
The idea is that generated auto_increment values are predictable and
|
|
independent of the column values in the table. This is needed to be
|
|
able to replicate into a table that already has rows with a higher
|
|
auto-increment value than the one that is inserted.
|
|
|
|
After we have already generated an auto-increment number and the user
|
|
inserts a column with a higher value than the last used one, we will
|
|
start counting from the inserted value.
|
|
|
|
thd->next_insert_id is cleared after it's been used for a statement.
|
|
*/
|
|
|
|
void handler::update_auto_increment()
|
|
{
|
|
ulonglong nr;
|
|
THD *thd= table->in_use;
|
|
struct system_variables *variables= &thd->variables;
|
|
DBUG_ENTER("handler::update_auto_increment");
|
|
|
|
/*
|
|
We must save the previous value to be able to restore it if the
|
|
row was not inserted
|
|
*/
|
|
thd->prev_insert_id= thd->next_insert_id;
|
|
|
|
if ((nr= table->next_number_field->val_int()) != 0 ||
|
|
table->auto_increment_field_not_null &&
|
|
thd->variables.sql_mode & MODE_NO_AUTO_VALUE_ON_ZERO)
|
|
{
|
|
/* Clear flag for next row */
|
|
table->auto_increment_field_not_null= FALSE;
|
|
/* Mark that we didn't generate a new value **/
|
|
auto_increment_column_changed=0;
|
|
|
|
/* Update next_insert_id if we have already generated a value */
|
|
if (thd->clear_next_insert_id && nr >= thd->next_insert_id)
|
|
{
|
|
if (variables->auto_increment_increment != 1)
|
|
nr= next_insert_id(nr, variables);
|
|
else
|
|
nr++;
|
|
thd->next_insert_id= nr;
|
|
DBUG_PRINT("info",("next_insert_id: %lu", (ulong) nr));
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
table->auto_increment_field_not_null= FALSE;
|
|
if (!(nr= thd->next_insert_id))
|
|
{
|
|
nr= get_auto_increment();
|
|
if (variables->auto_increment_increment != 1)
|
|
nr= next_insert_id(nr-1, variables);
|
|
/*
|
|
Update next row based on the found value. This way we don't have to
|
|
call the handler for every generated auto-increment value on a
|
|
multi-row statement
|
|
*/
|
|
thd->next_insert_id= nr;
|
|
}
|
|
|
|
DBUG_PRINT("info",("auto_increment: %lu", (ulong) nr));
|
|
|
|
/* Mark that we should clear next_insert_id before next stmt */
|
|
thd->clear_next_insert_id= 1;
|
|
|
|
if (!table->next_number_field->store((longlong) nr))
|
|
thd->insert_id((ulonglong) nr);
|
|
else
|
|
thd->insert_id(table->next_number_field->val_int());
|
|
|
|
/*
|
|
We can't set next_insert_id if the auto-increment key is not the
|
|
first key part, as there is no guarantee that the first parts will be in
|
|
sequence
|
|
*/
|
|
if (!table->s->next_number_key_offset)
|
|
{
|
|
/*
|
|
Set next insert id to point to next auto-increment value to be able to
|
|
handle multi-row statements
|
|
This works even if auto_increment_increment > 1
|
|
*/
|
|
thd->next_insert_id= next_insert_id(nr, variables);
|
|
}
|
|
else
|
|
thd->next_insert_id= 0;
|
|
|
|
/* Mark that we generated a new value */
|
|
auto_increment_column_changed=1;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
/*
|
|
restore_auto_increment
|
|
|
|
In case of error on write, we restore the last used next_insert_id value
|
|
because the previous value was not used.
|
|
*/
|
|
|
|
void handler::restore_auto_increment()
|
|
{
|
|
THD *thd= table->in_use;
|
|
if (thd->next_insert_id)
|
|
thd->next_insert_id= thd->prev_insert_id;
|
|
}
|
|
|
|
|
|
ulonglong handler::get_auto_increment()
|
|
{
|
|
ulonglong nr;
|
|
int error;
|
|
|
|
(void) extra(HA_EXTRA_KEYREAD);
|
|
index_init(table->s->next_number_index);
|
|
if (!table->s->next_number_key_offset)
|
|
{ // Autoincrement at key-start
|
|
error=index_last(table->record[1]);
|
|
}
|
|
else
|
|
{
|
|
byte key[MAX_KEY_LENGTH];
|
|
key_copy(key, table->record[0],
|
|
table->key_info + table->s->next_number_index,
|
|
table->s->next_number_key_offset);
|
|
error= index_read(table->record[1], key, table->s->next_number_key_offset,
|
|
HA_READ_PREFIX_LAST);
|
|
}
|
|
|
|
if (error)
|
|
nr=1;
|
|
else
|
|
nr= ((ulonglong) table->next_number_field->
|
|
val_int_offset(table->s->rec_buff_length)+1);
|
|
index_end();
|
|
(void) extra(HA_EXTRA_NO_KEYREAD);
|
|
return nr;
|
|
}
|
|
|
|
/* Print error that we got from handler function */
|
|
|
|
void handler::print_error(int error, myf errflag)
|
|
{
|
|
DBUG_ENTER("handler::print_error");
|
|
DBUG_PRINT("enter",("error: %d",error));
|
|
|
|
int textno=ER_GET_ERRNO;
|
|
switch (error) {
|
|
case EACCES:
|
|
textno=ER_OPEN_AS_READONLY;
|
|
break;
|
|
case EAGAIN:
|
|
textno=ER_FILE_USED;
|
|
break;
|
|
case ENOENT:
|
|
textno=ER_FILE_NOT_FOUND;
|
|
break;
|
|
case HA_ERR_KEY_NOT_FOUND:
|
|
case HA_ERR_NO_ACTIVE_RECORD:
|
|
case HA_ERR_END_OF_FILE:
|
|
textno=ER_KEY_NOT_FOUND;
|
|
break;
|
|
case HA_ERR_WRONG_MRG_TABLE_DEF:
|
|
textno=ER_WRONG_MRG_TABLE;
|
|
break;
|
|
case HA_ERR_FOUND_DUPP_KEY:
|
|
{
|
|
uint key_nr=get_dup_key(error);
|
|
if ((int) key_nr >= 0)
|
|
{
|
|
/* Write the dupplicated key in the error message */
|
|
char key[MAX_KEY_LENGTH];
|
|
String str(key,sizeof(key),system_charset_info);
|
|
key_unpack(&str,table,(uint) key_nr);
|
|
uint max_length=MYSQL_ERRMSG_SIZE-(uint) strlen(ER(ER_DUP_ENTRY));
|
|
if (str.length() >= max_length)
|
|
{
|
|
str.length(max_length-4);
|
|
str.append("...");
|
|
}
|
|
my_error(ER_DUP_ENTRY, MYF(0), str.c_ptr(), key_nr+1);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
textno=ER_DUP_KEY;
|
|
break;
|
|
}
|
|
case HA_ERR_NULL_IN_SPATIAL:
|
|
textno= ER_UNKNOWN_ERROR;
|
|
DBUG_VOID_RETURN;
|
|
case HA_ERR_FOUND_DUPP_UNIQUE:
|
|
textno=ER_DUP_UNIQUE;
|
|
break;
|
|
case HA_ERR_RECORD_CHANGED:
|
|
textno=ER_CHECKREAD;
|
|
break;
|
|
case HA_ERR_CRASHED:
|
|
textno=ER_NOT_KEYFILE;
|
|
break;
|
|
case HA_ERR_WRONG_IN_RECORD:
|
|
textno= ER_CRASHED_ON_USAGE;
|
|
break;
|
|
case HA_ERR_CRASHED_ON_USAGE:
|
|
textno=ER_CRASHED_ON_USAGE;
|
|
break;
|
|
case HA_ERR_NOT_A_TABLE:
|
|
textno= error;
|
|
break;
|
|
case HA_ERR_CRASHED_ON_REPAIR:
|
|
textno=ER_CRASHED_ON_REPAIR;
|
|
break;
|
|
case HA_ERR_OUT_OF_MEM:
|
|
my_message(ER_OUT_OF_RESOURCES, ER(ER_OUT_OF_RESOURCES), errflag);
|
|
DBUG_VOID_RETURN;
|
|
case HA_ERR_WRONG_COMMAND:
|
|
textno=ER_ILLEGAL_HA;
|
|
break;
|
|
case HA_ERR_OLD_FILE:
|
|
textno=ER_OLD_KEYFILE;
|
|
break;
|
|
case HA_ERR_UNSUPPORTED:
|
|
textno=ER_UNSUPPORTED_EXTENSION;
|
|
break;
|
|
case HA_ERR_RECORD_FILE_FULL:
|
|
textno=ER_RECORD_FILE_FULL;
|
|
break;
|
|
case HA_ERR_INDEX_FILE_FULL:
|
|
textno= errno;
|
|
break;
|
|
case HA_ERR_LOCK_WAIT_TIMEOUT:
|
|
textno=ER_LOCK_WAIT_TIMEOUT;
|
|
break;
|
|
case HA_ERR_LOCK_TABLE_FULL:
|
|
textno=ER_LOCK_TABLE_FULL;
|
|
break;
|
|
case HA_ERR_LOCK_DEADLOCK:
|
|
textno=ER_LOCK_DEADLOCK;
|
|
break;
|
|
case HA_ERR_READ_ONLY_TRANSACTION:
|
|
textno=ER_READ_ONLY_TRANSACTION;
|
|
break;
|
|
case HA_ERR_CANNOT_ADD_FOREIGN:
|
|
textno=ER_CANNOT_ADD_FOREIGN;
|
|
break;
|
|
case HA_ERR_ROW_IS_REFERENCED:
|
|
textno=ER_ROW_IS_REFERENCED;
|
|
break;
|
|
case HA_ERR_NO_REFERENCED_ROW:
|
|
textno=ER_NO_REFERENCED_ROW;
|
|
break;
|
|
case HA_ERR_NO_SUCH_TABLE:
|
|
{
|
|
/*
|
|
We have to use path to find database name instead of using
|
|
table->table_cache_key because if the table didn't exist, then
|
|
table_cache_key was not set up
|
|
*/
|
|
char *db;
|
|
char buff[FN_REFLEN];
|
|
uint length= dirname_part(buff,table->s->path);
|
|
buff[length-1]=0;
|
|
db=buff+dirname_length(buff);
|
|
my_error(ER_NO_SUCH_TABLE, MYF(0), db, table->alias);
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
/* The error was "unknown" to this function.
|
|
Ask handler if it has got a message for this error */
|
|
bool temporary= FALSE;
|
|
String str;
|
|
temporary= get_error_message(error, &str);
|
|
if (!str.is_empty())
|
|
{
|
|
const char* engine= table_type();
|
|
if (temporary)
|
|
my_error(ER_GET_TEMPORARY_ERRMSG, MYF(0), error, str.ptr(), engine);
|
|
else
|
|
my_error(ER_GET_ERRMSG, MYF(0), error, str.ptr(), engine);
|
|
}
|
|
else
|
|
my_error(ER_GET_ERRNO,errflag,error);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
}
|
|
my_error(textno, errflag, table->alias, error);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/*
|
|
Return an error message specific to this handler
|
|
|
|
SYNOPSIS
|
|
error error code previously returned by handler
|
|
buf Pointer to String where to add error message
|
|
|
|
Returns true if this is a temporary error
|
|
*/
|
|
|
|
bool handler::get_error_message(int error, String* buf)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* Return key if error because of duplicated keys */
|
|
|
|
uint handler::get_dup_key(int error)
|
|
{
|
|
DBUG_ENTER("handler::get_dup_key");
|
|
table->file->errkey = (uint) -1;
|
|
if (error == HA_ERR_FOUND_DUPP_KEY || error == HA_ERR_FOUND_DUPP_UNIQUE ||
|
|
error == HA_ERR_NULL_IN_SPATIAL)
|
|
info(HA_STATUS_ERRKEY | HA_STATUS_NO_LOCK);
|
|
DBUG_RETURN(table->file->errkey);
|
|
}
|
|
|
|
|
|
int handler::delete_table(const char *name)
|
|
{
|
|
int error=0;
|
|
for (const char **ext=bas_ext(); *ext ; ext++)
|
|
{
|
|
if (delete_file(name,*ext,2))
|
|
{
|
|
if ((error=errno) != ENOENT)
|
|
break;
|
|
}
|
|
}
|
|
return error;
|
|
}
|
|
|
|
|
|
int handler::rename_table(const char * from, const char * to)
|
|
{
|
|
DBUG_ENTER("handler::rename_table");
|
|
for (const char **ext=bas_ext(); *ext ; ext++)
|
|
{
|
|
if (rename_file_ext(from,to,*ext))
|
|
DBUG_RETURN(my_errno);
|
|
}
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
/*
|
|
Tell the handler to turn on or off transaction in the handler
|
|
*/
|
|
|
|
int ha_enable_transaction(THD *thd, bool on)
|
|
{
|
|
int error=0;
|
|
|
|
DBUG_ENTER("ha_enable_transaction");
|
|
thd->transaction.on= on;
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
int handler::index_next_same(byte *buf, const byte *key, uint keylen)
|
|
{
|
|
int error;
|
|
if (!(error=index_next(buf)))
|
|
{
|
|
if (key_cmp_if_same(table, key, active_index, keylen))
|
|
{
|
|
table->status=STATUS_NOT_FOUND;
|
|
error=HA_ERR_END_OF_FILE;
|
|
}
|
|
}
|
|
return error;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
** Some general functions that isn't in the handler class
|
|
****************************************************************************/
|
|
|
|
/*
|
|
Initiates table-file and calls apropriate database-creator
|
|
Returns 1 if something got wrong
|
|
*/
|
|
|
|
int ha_create_table(const char *name, HA_CREATE_INFO *create_info,
|
|
bool update_create_info)
|
|
{
|
|
int error;
|
|
TABLE table;
|
|
char name_buff[FN_REFLEN];
|
|
DBUG_ENTER("ha_create_table");
|
|
|
|
if (openfrm(current_thd, name,"",0,(uint) READ_ALL, 0, &table))
|
|
DBUG_RETURN(1);
|
|
if (update_create_info)
|
|
{
|
|
update_create_info_from_table(create_info, &table);
|
|
}
|
|
if (lower_case_table_names == 2 &&
|
|
!(table.file->table_flags() & HA_FILE_BASED))
|
|
{
|
|
/* Ensure that handler gets name in lower case */
|
|
strmov(name_buff, name);
|
|
my_casedn_str(files_charset_info, name_buff);
|
|
name= name_buff;
|
|
}
|
|
|
|
error=table.file->create(name,&table,create_info);
|
|
VOID(closefrm(&table));
|
|
if (error)
|
|
my_error(ER_CANT_CREATE_TABLE, MYF(ME_BELL+ME_WAITTANG), name,error);
|
|
DBUG_RETURN(error != 0);
|
|
}
|
|
|
|
/*
|
|
Try to discover table from engine and
|
|
if found, write the frm file to disk.
|
|
|
|
RETURN VALUES:
|
|
0 : Table existed in engine and created
|
|
on disk if so requested
|
|
1 : Table does not exist
|
|
>1 : error
|
|
|
|
*/
|
|
|
|
int ha_create_table_from_engine(THD* thd,
|
|
const char *db,
|
|
const char *name,
|
|
bool create_if_found)
|
|
{
|
|
int error;
|
|
const void *frmblob;
|
|
uint frmlen;
|
|
char path[FN_REFLEN];
|
|
HA_CREATE_INFO create_info;
|
|
TABLE table;
|
|
DBUG_ENTER("ha_create_table_from_engine");
|
|
DBUG_PRINT("enter", ("name '%s'.'%s' create_if_found: %d",
|
|
db, name, create_if_found));
|
|
|
|
bzero((char*) &create_info,sizeof(create_info));
|
|
|
|
if ((error= ha_discover(thd, db, name, &frmblob, &frmlen)))
|
|
DBUG_RETURN(error);
|
|
/*
|
|
Table exists in handler
|
|
frmblob and frmlen are set
|
|
*/
|
|
|
|
if (create_if_found)
|
|
{
|
|
(void)strxnmov(path,FN_REFLEN,mysql_data_home,"/",db,"/",name,NullS);
|
|
// Save the frm file
|
|
if ((error = writefrm(path, frmblob, frmlen)))
|
|
goto err_end;
|
|
|
|
if (openfrm(thd, path,"",0,(uint) READ_ALL, 0, &table))
|
|
DBUG_RETURN(1);
|
|
|
|
update_create_info_from_table(&create_info, &table);
|
|
create_info.table_options|= HA_CREATE_FROM_ENGINE;
|
|
|
|
if (lower_case_table_names == 2 &&
|
|
!(table.file->table_flags() & HA_FILE_BASED))
|
|
{
|
|
/* Ensure that handler gets name in lower case */
|
|
my_casedn_str(files_charset_info, path);
|
|
}
|
|
|
|
error=table.file->create(path,&table,&create_info);
|
|
VOID(closefrm(&table));
|
|
}
|
|
|
|
err_end:
|
|
my_free((char*) frmblob, MYF(MY_ALLOW_ZERO_PTR));
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
static int NEAR_F delete_file(const char *name,const char *ext,int extflag)
|
|
{
|
|
char buff[FN_REFLEN];
|
|
VOID(fn_format(buff,name,"",ext,extflag | 4));
|
|
return(my_delete_with_symlink(buff,MYF(MY_WME)));
|
|
}
|
|
|
|
void st_ha_check_opt::init()
|
|
{
|
|
flags= sql_flags= 0;
|
|
sort_buffer_size = current_thd->variables.myisam_sort_buff_size;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Key cache handling.
|
|
|
|
This code is only relevant for ISAM/MyISAM tables
|
|
|
|
key_cache->cache may be 0 only in the case where a key cache is not
|
|
initialized or when we where not able to init the key cache in a previous
|
|
call to ha_init_key_cache() (probably out of memory)
|
|
*****************************************************************************/
|
|
|
|
/* Init a key cache if it has not been initied before */
|
|
|
|
|
|
int ha_init_key_cache(const char *name, KEY_CACHE *key_cache)
|
|
{
|
|
DBUG_ENTER("ha_init_key_cache");
|
|
|
|
if (!key_cache->key_cache_inited)
|
|
{
|
|
pthread_mutex_lock(&LOCK_global_system_variables);
|
|
long tmp_buff_size= (long) key_cache->param_buff_size;
|
|
long tmp_block_size= (long) key_cache->param_block_size;
|
|
uint division_limit= key_cache->param_division_limit;
|
|
uint age_threshold= key_cache->param_age_threshold;
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
DBUG_RETURN(!init_key_cache(key_cache,
|
|
tmp_block_size,
|
|
tmp_buff_size,
|
|
division_limit, age_threshold));
|
|
}
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/* Resize key cache */
|
|
|
|
int ha_resize_key_cache(KEY_CACHE *key_cache)
|
|
{
|
|
DBUG_ENTER("ha_resize_key_cache");
|
|
|
|
if (key_cache->key_cache_inited)
|
|
{
|
|
pthread_mutex_lock(&LOCK_global_system_variables);
|
|
long tmp_buff_size= (long) key_cache->param_buff_size;
|
|
long tmp_block_size= (long) key_cache->param_block_size;
|
|
uint division_limit= key_cache->param_division_limit;
|
|
uint age_threshold= key_cache->param_age_threshold;
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
DBUG_RETURN(!resize_key_cache(key_cache, tmp_block_size,
|
|
tmp_buff_size,
|
|
division_limit, age_threshold));
|
|
}
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
/* Change parameters for key cache (like size) */
|
|
|
|
int ha_change_key_cache_param(KEY_CACHE *key_cache)
|
|
{
|
|
if (key_cache->key_cache_inited)
|
|
{
|
|
pthread_mutex_lock(&LOCK_global_system_variables);
|
|
uint division_limit= key_cache->param_division_limit;
|
|
uint age_threshold= key_cache->param_age_threshold;
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
change_key_cache_param(key_cache, division_limit, age_threshold);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Free memory allocated by a key cache */
|
|
|
|
int ha_end_key_cache(KEY_CACHE *key_cache)
|
|
{
|
|
end_key_cache(key_cache, 1); // Can never fail
|
|
return 0;
|
|
}
|
|
|
|
/* Move all tables from one key cache to another one */
|
|
|
|
int ha_change_key_cache(KEY_CACHE *old_key_cache,
|
|
KEY_CACHE *new_key_cache)
|
|
{
|
|
mi_change_key_cache(old_key_cache, new_key_cache);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Try to discover one table from handler(s)
|
|
|
|
RETURN
|
|
0 ok. In this case *frmblob and *frmlen are set
|
|
1 error. frmblob and frmlen may not be set
|
|
*/
|
|
|
|
int ha_discover(THD *thd, const char *db, const char *name,
|
|
const void **frmblob, uint *frmlen)
|
|
{
|
|
int error= 1; // Table does not exist in any handler
|
|
DBUG_ENTER("ha_discover");
|
|
DBUG_PRINT("enter", ("db: %s, name: %s", db, name));
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
error= ndbcluster_discover(thd, db, name, frmblob, frmlen);
|
|
#endif
|
|
if (!error)
|
|
statistic_increment(ha_discover_count,&LOCK_status);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
|
|
/*
|
|
Call this function in order to give the handler the possiblity
|
|
to ask engine if there are any new tables that should be written to disk
|
|
or any dropped tables that need to be removed from disk
|
|
*/
|
|
|
|
int
|
|
ha_find_files(THD *thd,const char *db,const char *path,
|
|
const char *wild, bool dir, List<char> *files)
|
|
{
|
|
int error= 0;
|
|
DBUG_ENTER("ha_find_files");
|
|
DBUG_PRINT("enter", ("db: %s, path: %s, wild: %s, dir: %d",
|
|
db, path, wild, dir));
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
error= ndbcluster_find_files(thd, db, path, wild, dir, files);
|
|
#endif
|
|
DBUG_RETURN(error);
|
|
|
|
|
|
}
|
|
|
|
#ifdef NOT_YET_USED
|
|
|
|
/*
|
|
Ask handler if the table exists in engine
|
|
|
|
RETURN
|
|
0 Table does not exist
|
|
1 Table exists
|
|
# Error code
|
|
|
|
*/
|
|
int ha_table_exists(THD* thd, const char* db, const char* name)
|
|
{
|
|
int error= 2;
|
|
DBUG_ENTER("ha_table_exists");
|
|
DBUG_PRINT("enter", ("db: %s, name: %s", db, name));
|
|
#ifdef HAVE_NDBCLUSTER_DB
|
|
if (have_ndbcluster == SHOW_OPTION_YES)
|
|
error= ndbcluster_table_exists(thd, db, name);
|
|
#endif
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
/*
|
|
Read the first row of a multi-range set.
|
|
|
|
SYNOPSIS
|
|
read_multi_range_first()
|
|
found_range_p Returns a pointer to the element in 'ranges' that
|
|
corresponds to the returned row.
|
|
ranges An array of KEY_MULTI_RANGE range descriptions.
|
|
range_count Number of ranges in 'ranges'.
|
|
sorted If result should be sorted per key.
|
|
buffer A HANDLER_BUFFER for internal handler usage.
|
|
|
|
NOTES
|
|
Record is read into table->record[0].
|
|
*found_range_p returns a valid value only if read_multi_range_first()
|
|
returns 0.
|
|
Sorting is done within each range. If you want an overall sort, enter
|
|
'ranges' with sorted ranges.
|
|
|
|
RETURN
|
|
0 OK, found a row
|
|
HA_ERR_END_OF_FILE No rows in range
|
|
# Error code
|
|
*/
|
|
|
|
int handler::read_multi_range_first(KEY_MULTI_RANGE **found_range_p,
|
|
KEY_MULTI_RANGE *ranges, uint range_count,
|
|
bool sorted, HANDLER_BUFFER *buffer)
|
|
{
|
|
int result= HA_ERR_END_OF_FILE;
|
|
DBUG_ENTER("handler::read_multi_range_first");
|
|
multi_range_sorted= sorted;
|
|
multi_range_buffer= buffer;
|
|
|
|
for (multi_range_curr= ranges, multi_range_end= ranges + range_count;
|
|
multi_range_curr < multi_range_end;
|
|
multi_range_curr++)
|
|
{
|
|
result= read_range_first(multi_range_curr->start_key.length ?
|
|
&multi_range_curr->start_key : 0,
|
|
multi_range_curr->end_key.length ?
|
|
&multi_range_curr->end_key : 0,
|
|
test(multi_range_curr->range_flag & EQ_RANGE),
|
|
multi_range_sorted);
|
|
if (result != HA_ERR_END_OF_FILE)
|
|
break;
|
|
}
|
|
|
|
*found_range_p= multi_range_curr;
|
|
DBUG_PRINT("exit",("result %d", result));
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
|
|
/*
|
|
Read the next row of a multi-range set.
|
|
|
|
SYNOPSIS
|
|
read_multi_range_next()
|
|
found_range_p Returns a pointer to the element in 'ranges' that
|
|
corresponds to the returned row.
|
|
|
|
NOTES
|
|
Record is read into table->record[0].
|
|
*found_range_p returns a valid value only if read_multi_range_next()
|
|
returns 0.
|
|
|
|
RETURN
|
|
0 OK, found a row
|
|
HA_ERR_END_OF_FILE No (more) rows in range
|
|
# Error code
|
|
*/
|
|
|
|
int handler::read_multi_range_next(KEY_MULTI_RANGE **found_range_p)
|
|
{
|
|
int result;
|
|
DBUG_ENTER("handler::read_multi_range_next");
|
|
|
|
/* We should not be called after the last call returned EOF. */
|
|
DBUG_ASSERT(multi_range_curr < multi_range_end);
|
|
|
|
do
|
|
{
|
|
/* Save a call if there can be only one row in range. */
|
|
if (multi_range_curr->range_flag != (UNIQUE_RANGE | EQ_RANGE))
|
|
{
|
|
result= read_range_next();
|
|
|
|
/* On success or non-EOF errors jump to the end. */
|
|
if (result != HA_ERR_END_OF_FILE)
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
We need to set this for the last range only, but checking this
|
|
condition is more expensive than just setting the result code.
|
|
*/
|
|
result= HA_ERR_END_OF_FILE;
|
|
}
|
|
|
|
/* Try the next range(s) until one matches a record. */
|
|
for (multi_range_curr++;
|
|
multi_range_curr < multi_range_end;
|
|
multi_range_curr++)
|
|
{
|
|
result= read_range_first(multi_range_curr->start_key.length ?
|
|
&multi_range_curr->start_key : 0,
|
|
multi_range_curr->end_key.length ?
|
|
&multi_range_curr->end_key : 0,
|
|
test(multi_range_curr->range_flag & EQ_RANGE),
|
|
multi_range_sorted);
|
|
if (result != HA_ERR_END_OF_FILE)
|
|
break;
|
|
}
|
|
}
|
|
while ((result == HA_ERR_END_OF_FILE) &&
|
|
(multi_range_curr < multi_range_end));
|
|
|
|
*found_range_p= multi_range_curr;
|
|
DBUG_PRINT("exit",("handler::read_multi_range_next: result %d", result));
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
|
|
/*
|
|
Read first row between two ranges.
|
|
Store ranges for future calls to read_range_next
|
|
|
|
SYNOPSIS
|
|
read_range_first()
|
|
start_key Start key. Is 0 if no min range
|
|
end_key End key. Is 0 if no max range
|
|
eq_range_arg Set to 1 if start_key == end_key
|
|
sorted Set to 1 if result should be sorted per key
|
|
|
|
NOTES
|
|
Record is read into table->record[0]
|
|
|
|
RETURN
|
|
0 Found row
|
|
HA_ERR_END_OF_FILE No rows in range
|
|
# Error code
|
|
*/
|
|
|
|
int handler::read_range_first(const key_range *start_key,
|
|
const key_range *end_key,
|
|
bool eq_range_arg, bool sorted)
|
|
{
|
|
int result;
|
|
DBUG_ENTER("handler::read_range_first");
|
|
|
|
eq_range= eq_range_arg;
|
|
end_range= 0;
|
|
if (end_key)
|
|
{
|
|
end_range= &save_end_range;
|
|
save_end_range= *end_key;
|
|
key_compare_result_on_equal= ((end_key->flag == HA_READ_BEFORE_KEY) ? 1 :
|
|
(end_key->flag == HA_READ_AFTER_KEY) ? -1 : 0);
|
|
}
|
|
range_key_part= table->key_info[active_index].key_part;
|
|
|
|
if (!start_key) // Read first record
|
|
result= index_first(table->record[0]);
|
|
else
|
|
result= index_read(table->record[0],
|
|
start_key->key,
|
|
start_key->length,
|
|
start_key->flag);
|
|
if (result)
|
|
DBUG_RETURN((result == HA_ERR_KEY_NOT_FOUND)
|
|
? HA_ERR_END_OF_FILE
|
|
: result);
|
|
|
|
DBUG_RETURN (compare_key(end_range) <= 0 ? 0 : HA_ERR_END_OF_FILE);
|
|
}
|
|
|
|
|
|
/*
|
|
Read next row between two ranges.
|
|
|
|
SYNOPSIS
|
|
read_range_next()
|
|
|
|
NOTES
|
|
Record is read into table->record[0]
|
|
|
|
RETURN
|
|
0 Found row
|
|
HA_ERR_END_OF_FILE No rows in range
|
|
# Error code
|
|
*/
|
|
|
|
int handler::read_range_next()
|
|
{
|
|
int result;
|
|
DBUG_ENTER("handler::read_range_next");
|
|
|
|
if (eq_range)
|
|
{
|
|
/* We trust that index_next_same always gives a row in range */
|
|
DBUG_RETURN(index_next_same(table->record[0],
|
|
end_range->key,
|
|
end_range->length));
|
|
}
|
|
result= index_next(table->record[0]);
|
|
if (result)
|
|
DBUG_RETURN(result);
|
|
DBUG_RETURN(compare_key(end_range) <= 0 ? 0 : HA_ERR_END_OF_FILE);
|
|
}
|
|
|
|
|
|
/*
|
|
Compare if found key (in row) is over max-value
|
|
|
|
SYNOPSIS
|
|
compare_key
|
|
range range to compare to row. May be 0 for no range
|
|
|
|
NOTES
|
|
See key.cc::key_cmp() for details
|
|
|
|
RETURN
|
|
The return value is SIGN(key_in_row - range_key):
|
|
|
|
0 Key is equal to range or 'range' == 0 (no range)
|
|
-1 Key is less than range
|
|
1 Key is larger than range
|
|
*/
|
|
|
|
int handler::compare_key(key_range *range)
|
|
{
|
|
int cmp;
|
|
if (!range)
|
|
return 0; // No max range
|
|
cmp= key_cmp(range_key_part, range->key, range->length);
|
|
if (!cmp)
|
|
cmp= key_compare_result_on_equal;
|
|
return cmp;
|
|
}
|
|
|
|
int handler::index_read_idx(byte * buf, uint index, const byte * key,
|
|
uint key_len, enum ha_rkey_function find_flag)
|
|
{
|
|
int error= ha_index_init(index);
|
|
if (!error)
|
|
error= index_read(buf, key, key_len, find_flag);
|
|
if (!error)
|
|
error= ha_index_end();
|
|
return error;
|
|
}
|
|
|
|
|
|
/*
|
|
Returns a list of all known extensions.
|
|
|
|
SYNOPSIS
|
|
ha_known_exts()
|
|
|
|
NOTES
|
|
No mutexes, worst case race is a minor surplus memory allocation
|
|
We have to recreate the extension map if mysqld is restarted (for example
|
|
within libmysqld)
|
|
|
|
RETURN VALUE
|
|
pointer pointer to TYPELIB structure
|
|
*/
|
|
|
|
TYPELIB *ha_known_exts(void)
|
|
{
|
|
if (!known_extensions.type_names || mysys_usage_id != known_extensions_id)
|
|
{
|
|
show_table_type_st *types;
|
|
List<char> found_exts;
|
|
List_iterator_fast<char> it(found_exts);
|
|
const char **ext, *old_ext;
|
|
|
|
known_extensions_id= mysys_usage_id;
|
|
found_exts.push_back((char*) ".db");
|
|
for (types= sys_table_types; types->type; types++)
|
|
{
|
|
if (*types->value == SHOW_OPTION_YES)
|
|
{
|
|
handler *file= get_new_handler(0,(enum db_type) types->db_type);
|
|
for (ext= file->bas_ext(); *ext; ext++)
|
|
{
|
|
while ((old_ext= it++))
|
|
{
|
|
if (!strcmp(old_ext, *ext))
|
|
break;
|
|
}
|
|
if (!old_ext)
|
|
found_exts.push_back((char *) *ext);
|
|
|
|
it.rewind();
|
|
}
|
|
delete file;
|
|
}
|
|
}
|
|
ext= (const char **) my_once_alloc(sizeof(char *)*
|
|
(found_exts.elements+1),
|
|
MYF(MY_WME | MY_FAE));
|
|
|
|
DBUG_ASSERT(ext);
|
|
known_extensions.count= found_exts.elements;
|
|
known_extensions.type_names= ext;
|
|
|
|
while ((old_ext= it++))
|
|
*ext++= old_ext;
|
|
*ext= 0;
|
|
}
|
|
return &known_extensions;
|
|
}
|