2001-12-06 14:10:51 +02:00
|
|
|
/* Copyright (C) 2000 MySQL 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
|
2006-12-23 20:17:15 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2001-12-06 14:10:51 +02:00
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
2000-07-31 21:29:14 +02:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2001-12-06 14:10:51 +02:00
|
|
|
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 */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* This file includes constants used with all databases */
|
|
|
|
|
|
|
|
#ifndef _my_base_h
|
|
|
|
#define _my_base_h
|
|
|
|
|
|
|
|
#ifndef stdin /* Included first in handler */
|
|
|
|
#define CHSIZE_USED
|
2001-09-14 02:54:33 +03:00
|
|
|
#include <my_global.h>
|
2000-07-31 21:29:14 +02:00
|
|
|
#include <my_dir.h> /* This includes types */
|
|
|
|
#include <my_sys.h>
|
|
|
|
#include <m_string.h>
|
|
|
|
#include <errno.h>
|
2004-03-16 16:35:47 +01:00
|
|
|
|
|
|
|
#ifndef EOVERFLOW
|
|
|
|
#define EOVERFLOW 84
|
|
|
|
#endif
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
#if !defined(USE_MY_FUNC) && !defined(THREAD)
|
|
|
|
#include <my_nosys.h> /* For faster code, after test */
|
|
|
|
#endif /* USE_MY_FUNC */
|
|
|
|
#endif /* stdin */
|
|
|
|
#include <my_list.h>
|
|
|
|
|
|
|
|
/* The following is bits in the flag parameter to ha_open() */
|
|
|
|
|
|
|
|
#define HA_OPEN_ABORT_IF_LOCKED 0 /* default */
|
|
|
|
#define HA_OPEN_WAIT_IF_LOCKED 1
|
|
|
|
#define HA_OPEN_IGNORE_IF_LOCKED 2
|
2000-09-26 00:33:25 +03:00
|
|
|
#define HA_OPEN_TMP_TABLE 4 /* Table is a temp table */
|
|
|
|
#define HA_OPEN_DELAY_KEY_WRITE 8 /* Don't update index */
|
|
|
|
#define HA_OPEN_ABORT_IF_CRASHED 16
|
2000-10-11 00:06:37 +03:00
|
|
|
#define HA_OPEN_FOR_REPAIR 32 /* open even if crashed */
|
2005-12-28 16:05:30 +04:00
|
|
|
#define HA_OPEN_FROM_SQL_LAYER 64
|
2007-02-13 16:33:32 +01:00
|
|
|
#define HA_OPEN_MMAP 128 /* open memory mapped */
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_OPEN_COPY 256 /* Open copy (for repair) */
|
2007-07-25 01:58:12 +03:00
|
|
|
/* Internal temp table, used for temporary results */
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_OPEN_INTERNAL_TABLE 512
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2007-10-11 18:07:40 +03:00
|
|
|
/* The following is parameter to ha_rkey() how to use key */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-10-15 22:40:36 +03:00
|
|
|
/*
|
|
|
|
We define a complete-field prefix of a key value as a prefix where
|
|
|
|
the last included field in the prefix contains the full field, not
|
|
|
|
just some bytes from the start of the field. A partial-field prefix
|
|
|
|
is allowed to contain only a few first bytes from the last included
|
|
|
|
field.
|
2003-02-18 19:57:05 +02:00
|
|
|
|
2003-10-15 22:40:36 +03:00
|
|
|
Below HA_READ_KEY_EXACT, ..., HA_READ_BEFORE_KEY can take a
|
|
|
|
complete-field prefix of a key value as the search
|
|
|
|
key. HA_READ_PREFIX and HA_READ_PREFIX_LAST could also take a
|
|
|
|
partial-field prefix, but currently (4.0.10) they are only used with
|
|
|
|
complete-field prefixes. MySQL uses a padding trick to implement
|
|
|
|
LIKE 'abc%' queries.
|
2003-02-18 19:57:05 +02:00
|
|
|
|
2003-10-15 22:40:36 +03:00
|
|
|
NOTE that in InnoDB HA_READ_PREFIX_LAST will NOT work with a
|
|
|
|
partial-field prefix because InnoDB currently strips spaces from the
|
|
|
|
end of varchar fields!
|
|
|
|
*/
|
2003-02-18 19:57:05 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
enum ha_rkey_function {
|
2002-10-25 14:09:47 +00:00
|
|
|
HA_READ_KEY_EXACT, /* Find first record else error */
|
|
|
|
HA_READ_KEY_OR_NEXT, /* Record or next record */
|
|
|
|
HA_READ_KEY_OR_PREV, /* Record or previous */
|
|
|
|
HA_READ_AFTER_KEY, /* Find next rec. after key-record */
|
|
|
|
HA_READ_BEFORE_KEY, /* Find next rec. before key-record */
|
|
|
|
HA_READ_PREFIX, /* Key which as same prefix */
|
|
|
|
HA_READ_PREFIX_LAST, /* Last key with the same prefix */
|
|
|
|
HA_READ_PREFIX_LAST_OR_PREV, /* Last or prev key with the same prefix */
|
2002-01-05 22:51:42 +02:00
|
|
|
HA_READ_MBR_CONTAIN,
|
|
|
|
HA_READ_MBR_INTERSECT,
|
|
|
|
HA_READ_MBR_WITHIN,
|
|
|
|
HA_READ_MBR_DISJOINT,
|
|
|
|
HA_READ_MBR_EQUAL
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Key algorithm types */
|
|
|
|
|
2002-10-25 14:09:47 +00:00
|
|
|
enum ha_key_alg {
|
2002-04-12 21:35:46 +03:00
|
|
|
HA_KEY_ALG_UNDEF= 0, /* Not specified (old file) */
|
|
|
|
HA_KEY_ALG_BTREE= 1, /* B-tree, default one */
|
|
|
|
HA_KEY_ALG_RTREE= 2, /* R-tree, for spatial searches */
|
|
|
|
HA_KEY_ALG_HASH= 3, /* HASH keys (HEAP tables) */
|
|
|
|
HA_KEY_ALG_FULLTEXT= 4 /* FULLTEXT (MyISAM tables) */
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
2006-12-19 23:20:43 +01:00
|
|
|
/* Storage media types */
|
|
|
|
|
|
|
|
enum ha_storage_media {
|
|
|
|
HA_SM_DEFAULT= 0, /* Not specified (engine default) */
|
|
|
|
HA_SM_DISK= 1, /* DISK storage */
|
|
|
|
HA_SM_MEMORY= 2 /* MAIN MEMORY storage */
|
|
|
|
};
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* The following is parameter to ha_extra() */
|
|
|
|
|
|
|
|
enum ha_extra_function {
|
|
|
|
HA_EXTRA_NORMAL=0, /* Optimize for space (def) */
|
|
|
|
HA_EXTRA_QUICK=1, /* Optimize for speed */
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
HA_EXTRA_NOT_USED=2,
|
2004-04-06 21:35:26 +02:00
|
|
|
HA_EXTRA_CACHE=3, /* Cache record in HA_rrnd() */
|
|
|
|
HA_EXTRA_NO_CACHE=4, /* End caching of records (def) */
|
2000-07-31 21:29:14 +02:00
|
|
|
HA_EXTRA_NO_READCHECK=5, /* No readcheck on update */
|
|
|
|
HA_EXTRA_READCHECK=6, /* Use readcheck (def) */
|
|
|
|
HA_EXTRA_KEYREAD=7, /* Read only key to database */
|
|
|
|
HA_EXTRA_NO_KEYREAD=8, /* Normal read of records (def) */
|
|
|
|
HA_EXTRA_NO_USER_CHANGE=9, /* No user is allowed to write */
|
|
|
|
HA_EXTRA_KEY_CACHE=10,
|
|
|
|
HA_EXTRA_NO_KEY_CACHE=11,
|
|
|
|
HA_EXTRA_WAIT_LOCK=12, /* Wait until file is avalably (def) */
|
|
|
|
HA_EXTRA_NO_WAIT_LOCK=13, /* If file is locked, return quickly */
|
|
|
|
HA_EXTRA_WRITE_CACHE=14, /* Use write cache in ha_write() */
|
|
|
|
HA_EXTRA_FLUSH_CACHE=15, /* flush write_record_cache */
|
|
|
|
HA_EXTRA_NO_KEYS=16, /* Remove all update of keys */
|
|
|
|
HA_EXTRA_KEYREAD_CHANGE_POS=17, /* Keyread, but change pos */
|
|
|
|
/* xxxxchk -r must be used */
|
|
|
|
HA_EXTRA_REMEMBER_POS=18, /* Remember pos for next/prev */
|
|
|
|
HA_EXTRA_RESTORE_POS=19,
|
|
|
|
HA_EXTRA_REINIT_CACHE=20, /* init cache from current record */
|
|
|
|
HA_EXTRA_FORCE_REOPEN=21, /* Datafile have changed on disk */
|
|
|
|
HA_EXTRA_FLUSH, /* Flush tables to disk */
|
2000-10-15 18:45:53 +03:00
|
|
|
HA_EXTRA_NO_ROWS, /* Don't write rows */
|
2000-12-24 15:19:00 +02:00
|
|
|
HA_EXTRA_RESET_STATE, /* Reset positions */
|
|
|
|
HA_EXTRA_IGNORE_DUP_KEY, /* Dup keys don't rollback everything*/
|
2001-02-17 14:19:19 +02:00
|
|
|
HA_EXTRA_NO_IGNORE_DUP_KEY,
|
2007-10-11 18:07:40 +03:00
|
|
|
HA_EXTRA_PREPARE_FOR_DROP,
|
2003-06-12 04:29:02 -07:00
|
|
|
HA_EXTRA_PREPARE_FOR_UPDATE, /* Remove read cache if problems */
|
2004-03-18 16:47:16 +01:00
|
|
|
HA_EXTRA_PRELOAD_BUFFER_SIZE, /* Set buffer size for preloading */
|
|
|
|
/*
|
|
|
|
On-the-fly switching between unique and non-unique key inserting.
|
|
|
|
*/
|
|
|
|
HA_EXTRA_CHANGE_KEY_TO_UNIQUE,
|
2004-05-13 04:50:54 +04:00
|
|
|
HA_EXTRA_CHANGE_KEY_TO_DUP,
|
2004-05-13 01:38:40 +04:00
|
|
|
/*
|
|
|
|
When using HA_EXTRA_KEYREAD, overwrite only key member fields and keep
|
|
|
|
other fields intact. When this is off (by default) InnoDB will use memcpy
|
|
|
|
to overwrite entire row.
|
|
|
|
*/
|
2005-12-01 13:34:48 +01:00
|
|
|
HA_EXTRA_KEYREAD_PRESERVE_FIELDS,
|
2006-01-12 19:51:02 +01:00
|
|
|
HA_EXTRA_MMAP,
|
2006-01-19 05:56:06 +03:00
|
|
|
/*
|
2006-01-12 19:51:02 +01:00
|
|
|
Ignore if the a tuple is not found, continue processing the
|
|
|
|
transaction and ignore that 'row'. Needed for idempotency
|
|
|
|
handling on the slave
|
2006-03-22 15:56:53 +01:00
|
|
|
|
|
|
|
Currently only used by NDB storage engine. Partition handler ignores flag.
|
2006-01-12 19:51:02 +01:00
|
|
|
*/
|
|
|
|
HA_EXTRA_IGNORE_NO_KEY,
|
2006-01-19 05:56:06 +03:00
|
|
|
HA_EXTRA_NO_IGNORE_NO_KEY,
|
|
|
|
/*
|
|
|
|
Mark the table as a log table. For some handlers (e.g. CSV) this results
|
|
|
|
in a special locking for the table.
|
|
|
|
*/
|
2006-07-02 02:12:53 +04:00
|
|
|
HA_EXTRA_MARK_AS_LOG_TABLE,
|
2006-07-02 01:51:10 +04:00
|
|
|
/*
|
|
|
|
Informs handler that write_row() which tries to insert new row into the
|
|
|
|
table and encounters some already existing row with same primary/unique
|
|
|
|
key can replace old row with new row instead of reporting error (basically
|
|
|
|
it informs handler that we do REPLACE instead of simple INSERT).
|
|
|
|
Off by default.
|
|
|
|
*/
|
|
|
|
HA_EXTRA_WRITE_CAN_REPLACE,
|
2007-04-04 12:50:39 +02:00
|
|
|
HA_EXTRA_WRITE_CANNOT_REPLACE,
|
|
|
|
/*
|
|
|
|
Inform handler that delete_row()/update_row() cannot batch deletes/updates
|
|
|
|
and should perform them immediately. This may be needed when table has
|
|
|
|
AFTER DELETE/UPDATE triggers which access to subject table.
|
|
|
|
These flags are reset by the handler::extra(HA_EXTRA_RESET) call.
|
|
|
|
*/
|
|
|
|
HA_EXTRA_DELETE_CANNOT_BATCH,
|
2007-06-28 13:36:26 -07:00
|
|
|
HA_EXTRA_UPDATE_CANNOT_BATCH,
|
|
|
|
/*
|
2007-06-29 13:55:16 -07:00
|
|
|
Inform handler that an "INSERT...ON DUPLICATE KEY UPDATE" will be
|
|
|
|
executed. This condition is unset by HA_EXTRA_NO_IGNORE_DUP_KEY.
|
2007-06-28 13:36:26 -07:00
|
|
|
*/
|
Bug#26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Bug 25038 - Waiting TRUNCATE
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Bug 19627 - temporary merge table locking
Bug 27660 - Falcon: merge table possible
Bug 30273 - merge tables: Can't lock file (errno: 155)
The problems were:
Bug 26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
1. A thread trying to lock a MERGE table performs busy waiting while
REPAIR TABLE or a similar table administration task is ongoing on
one or more of its MyISAM tables.
2. A thread trying to lock a MERGE table performs busy waiting until all
threads that did REPAIR TABLE or similar table administration tasks
on one or more of its MyISAM tables in LOCK TABLES segments do UNLOCK
TABLES. The difference against problem #1 is that the busy waiting
takes place *after* the administration task. It is terminated by
UNLOCK TABLES only.
3. Two FLUSH TABLES within a LOCK TABLES segment can invalidate the
lock. This does *not* require a MERGE table. The first FLUSH TABLES
can be replaced by any statement that requires other threads to
reopen the table. In 5.0 and 5.1 a single FLUSH TABLES can provoke
the problem.
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Trying DML on a MERGE table, which has a child locked and
repaired by another thread, made an infinite loop in the server.
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Locking a MERGE table and its children in parent-child order
and flushing the child deadlocked the server.
Bug 25038 - Waiting TRUNCATE
Truncating a MERGE child, while the MERGE table was in use,
let the truncate fail instead of waiting for the table to
become free.
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Repairing a child of an open MERGE table corrupted the child.
It was necessary to FLUSH the child first.
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Flushing and optimizing locked MERGE children crashed the server.
Bug 19627 - temporary merge table locking
Use of a temporary MERGE table with non-temporary children
could corrupt the children.
Temporary tables are never locked. So we do now prohibit
non-temporary chidlren of a temporary MERGE table.
Bug 27660 - Falcon: merge table possible
It was possible to create a MERGE table with non-MyISAM children.
Bug 30273 - merge tables: Can't lock file (errno: 155)
This was a Windows-only bug. Table administration statements
sometimes failed with "Can't lock file (errno: 155)".
These bugs are fixed by a new implementation of MERGE table open.
When opening a MERGE table in open_tables() we do now add the
child tables to the list of tables to be opened by open_tables()
(the "query_list"). The children are not opened in the handler at
this stage.
After opening the parent, open_tables() opens each child from the
now extended query_list. When the last child is opened, we remove
the children from the query_list again and attach the children to
the parent. This behaves similar to the old open. However it does
not open the MyISAM tables directly, but grabs them from the already
open children.
When closing a MERGE table in close_thread_table() we detach the
children only. Closing of the children is done implicitly because
they are in thd->open_tables.
For more detail see the comment at the top of ha_myisammrg.cc.
Changed from open_ltable() to open_and_lock_tables() in all places
that can be relevant for MERGE tables. The latter can handle tables
added to the list on the fly. When open_ltable() was used in a loop
over a list of tables, the list must be temporarily terminated
after every table for open_and_lock_tables().
table_list->required_type is set to FRMTYPE_TABLE to avoid open of
special tables. Handling of derived tables is suppressed.
These details are handled by the new function
open_n_lock_single_table(), which has nearly the same signature as
open_ltable() and can replace it in most cases.
In reopen_tables() some of the tables open by a thread can be
closed and reopened. When a MERGE child is affected, the parent
must be closed and reopened too. Closing of the parent is forced
before the first child is closed. Reopen happens in the order of
thd->open_tables. MERGE parents do not attach their children
automatically at open. This is done after all tables are reopened.
So all children are open when attaching them.
Special lock handling like mysql_lock_abort() or mysql_lock_remove()
needs to be suppressed for MERGE children or forwarded to the parent.
This depends on the situation. In loops over all open tables one
suppresses child lock handling. When a single table is touched,
forwarding is done.
Behavioral changes:
===================
This patch changes the behavior of temporary MERGE tables.
Temporary MERGE must have temporary children.
The old behavior was wrong. A temporary table is not locked. Hence
even non-temporary children were not locked. See
Bug 19627 - temporary merge table locking.
You cannot change the union list of a non-temporary MERGE table
when LOCK TABLES is in effect. The following does *not* work:
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ...;
LOCK TABLES t1 WRITE, t2 WRITE, m1 WRITE;
ALTER TABLE m1 ... UNION=(t1,t2) ...;
However, you can do this with a temporary MERGE table.
You cannot create a MERGE table with CREATE ... SELECT, neither
as a temporary MERGE table, nor as a non-temporary MERGE table.
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ... SELECT ...;
Gives error message: table is not BASE TABLE.
2007-11-15 20:25:43 +01:00
|
|
|
HA_EXTRA_INSERT_WITH_UPDATE,
|
2007-10-11 18:07:40 +03:00
|
|
|
/* Inform handler that we will do a rename */
|
2007-12-07 22:27:48 +02:00
|
|
|
HA_EXTRA_PREPARE_FOR_RENAME,
|
Bug#26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Bug 25038 - Waiting TRUNCATE
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Bug 19627 - temporary merge table locking
Bug 27660 - Falcon: merge table possible
Bug 30273 - merge tables: Can't lock file (errno: 155)
The problems were:
Bug 26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
1. A thread trying to lock a MERGE table performs busy waiting while
REPAIR TABLE or a similar table administration task is ongoing on
one or more of its MyISAM tables.
2. A thread trying to lock a MERGE table performs busy waiting until all
threads that did REPAIR TABLE or similar table administration tasks
on one or more of its MyISAM tables in LOCK TABLES segments do UNLOCK
TABLES. The difference against problem #1 is that the busy waiting
takes place *after* the administration task. It is terminated by
UNLOCK TABLES only.
3. Two FLUSH TABLES within a LOCK TABLES segment can invalidate the
lock. This does *not* require a MERGE table. The first FLUSH TABLES
can be replaced by any statement that requires other threads to
reopen the table. In 5.0 and 5.1 a single FLUSH TABLES can provoke
the problem.
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Trying DML on a MERGE table, which has a child locked and
repaired by another thread, made an infinite loop in the server.
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Locking a MERGE table and its children in parent-child order
and flushing the child deadlocked the server.
Bug 25038 - Waiting TRUNCATE
Truncating a MERGE child, while the MERGE table was in use,
let the truncate fail instead of waiting for the table to
become free.
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Repairing a child of an open MERGE table corrupted the child.
It was necessary to FLUSH the child first.
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Flushing and optimizing locked MERGE children crashed the server.
Bug 19627 - temporary merge table locking
Use of a temporary MERGE table with non-temporary children
could corrupt the children.
Temporary tables are never locked. So we do now prohibit
non-temporary chidlren of a temporary MERGE table.
Bug 27660 - Falcon: merge table possible
It was possible to create a MERGE table with non-MyISAM children.
Bug 30273 - merge tables: Can't lock file (errno: 155)
This was a Windows-only bug. Table administration statements
sometimes failed with "Can't lock file (errno: 155)".
These bugs are fixed by a new implementation of MERGE table open.
When opening a MERGE table in open_tables() we do now add the
child tables to the list of tables to be opened by open_tables()
(the "query_list"). The children are not opened in the handler at
this stage.
After opening the parent, open_tables() opens each child from the
now extended query_list. When the last child is opened, we remove
the children from the query_list again and attach the children to
the parent. This behaves similar to the old open. However it does
not open the MyISAM tables directly, but grabs them from the already
open children.
When closing a MERGE table in close_thread_table() we detach the
children only. Closing of the children is done implicitly because
they are in thd->open_tables.
For more detail see the comment at the top of ha_myisammrg.cc.
Changed from open_ltable() to open_and_lock_tables() in all places
that can be relevant for MERGE tables. The latter can handle tables
added to the list on the fly. When open_ltable() was used in a loop
over a list of tables, the list must be temporarily terminated
after every table for open_and_lock_tables().
table_list->required_type is set to FRMTYPE_TABLE to avoid open of
special tables. Handling of derived tables is suppressed.
These details are handled by the new function
open_n_lock_single_table(), which has nearly the same signature as
open_ltable() and can replace it in most cases.
In reopen_tables() some of the tables open by a thread can be
closed and reopened. When a MERGE child is affected, the parent
must be closed and reopened too. Closing of the parent is forced
before the first child is closed. Reopen happens in the order of
thd->open_tables. MERGE parents do not attach their children
automatically at open. This is done after all tables are reopened.
So all children are open when attaching them.
Special lock handling like mysql_lock_abort() or mysql_lock_remove()
needs to be suppressed for MERGE children or forwarded to the parent.
This depends on the situation. In loops over all open tables one
suppresses child lock handling. When a single table is touched,
forwarding is done.
Behavioral changes:
===================
This patch changes the behavior of temporary MERGE tables.
Temporary MERGE must have temporary children.
The old behavior was wrong. A temporary table is not locked. Hence
even non-temporary children were not locked. See
Bug 19627 - temporary merge table locking.
You cannot change the union list of a non-temporary MERGE table
when LOCK TABLES is in effect. The following does *not* work:
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ...;
LOCK TABLES t1 WRITE, t2 WRITE, m1 WRITE;
ALTER TABLE m1 ... UNION=(t1,t2) ...;
However, you can do this with a temporary MERGE table.
You cannot create a MERGE table with CREATE ... SELECT, neither
as a temporary MERGE table, nor as a non-temporary MERGE table.
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ... SELECT ...;
Gives error message: table is not BASE TABLE.
2007-11-15 20:25:43 +01:00
|
|
|
/*
|
2009-12-03 02:09:22 +03:00
|
|
|
Special actions for MERGE tables.
|
Bug#26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Bug 25038 - Waiting TRUNCATE
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Bug 19627 - temporary merge table locking
Bug 27660 - Falcon: merge table possible
Bug 30273 - merge tables: Can't lock file (errno: 155)
The problems were:
Bug 26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
1. A thread trying to lock a MERGE table performs busy waiting while
REPAIR TABLE or a similar table administration task is ongoing on
one or more of its MyISAM tables.
2. A thread trying to lock a MERGE table performs busy waiting until all
threads that did REPAIR TABLE or similar table administration tasks
on one or more of its MyISAM tables in LOCK TABLES segments do UNLOCK
TABLES. The difference against problem #1 is that the busy waiting
takes place *after* the administration task. It is terminated by
UNLOCK TABLES only.
3. Two FLUSH TABLES within a LOCK TABLES segment can invalidate the
lock. This does *not* require a MERGE table. The first FLUSH TABLES
can be replaced by any statement that requires other threads to
reopen the table. In 5.0 and 5.1 a single FLUSH TABLES can provoke
the problem.
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Trying DML on a MERGE table, which has a child locked and
repaired by another thread, made an infinite loop in the server.
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Locking a MERGE table and its children in parent-child order
and flushing the child deadlocked the server.
Bug 25038 - Waiting TRUNCATE
Truncating a MERGE child, while the MERGE table was in use,
let the truncate fail instead of waiting for the table to
become free.
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Repairing a child of an open MERGE table corrupted the child.
It was necessary to FLUSH the child first.
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Flushing and optimizing locked MERGE children crashed the server.
Bug 19627 - temporary merge table locking
Use of a temporary MERGE table with non-temporary children
could corrupt the children.
Temporary tables are never locked. So we do now prohibit
non-temporary chidlren of a temporary MERGE table.
Bug 27660 - Falcon: merge table possible
It was possible to create a MERGE table with non-MyISAM children.
Bug 30273 - merge tables: Can't lock file (errno: 155)
This was a Windows-only bug. Table administration statements
sometimes failed with "Can't lock file (errno: 155)".
These bugs are fixed by a new implementation of MERGE table open.
When opening a MERGE table in open_tables() we do now add the
child tables to the list of tables to be opened by open_tables()
(the "query_list"). The children are not opened in the handler at
this stage.
After opening the parent, open_tables() opens each child from the
now extended query_list. When the last child is opened, we remove
the children from the query_list again and attach the children to
the parent. This behaves similar to the old open. However it does
not open the MyISAM tables directly, but grabs them from the already
open children.
When closing a MERGE table in close_thread_table() we detach the
children only. Closing of the children is done implicitly because
they are in thd->open_tables.
For more detail see the comment at the top of ha_myisammrg.cc.
Changed from open_ltable() to open_and_lock_tables() in all places
that can be relevant for MERGE tables. The latter can handle tables
added to the list on the fly. When open_ltable() was used in a loop
over a list of tables, the list must be temporarily terminated
after every table for open_and_lock_tables().
table_list->required_type is set to FRMTYPE_TABLE to avoid open of
special tables. Handling of derived tables is suppressed.
These details are handled by the new function
open_n_lock_single_table(), which has nearly the same signature as
open_ltable() and can replace it in most cases.
In reopen_tables() some of the tables open by a thread can be
closed and reopened. When a MERGE child is affected, the parent
must be closed and reopened too. Closing of the parent is forced
before the first child is closed. Reopen happens in the order of
thd->open_tables. MERGE parents do not attach their children
automatically at open. This is done after all tables are reopened.
So all children are open when attaching them.
Special lock handling like mysql_lock_abort() or mysql_lock_remove()
needs to be suppressed for MERGE children or forwarded to the parent.
This depends on the situation. In loops over all open tables one
suppresses child lock handling. When a single table is touched,
forwarding is done.
Behavioral changes:
===================
This patch changes the behavior of temporary MERGE tables.
Temporary MERGE must have temporary children.
The old behavior was wrong. A temporary table is not locked. Hence
even non-temporary children were not locked. See
Bug 19627 - temporary merge table locking.
You cannot change the union list of a non-temporary MERGE table
when LOCK TABLES is in effect. The following does *not* work:
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ...;
LOCK TABLES t1 WRITE, t2 WRITE, m1 WRITE;
ALTER TABLE m1 ... UNION=(t1,t2) ...;
However, you can do this with a temporary MERGE table.
You cannot create a MERGE table with CREATE ... SELECT, neither
as a temporary MERGE table, nor as a non-temporary MERGE table.
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ... SELECT ...;
Gives error message: table is not BASE TABLE.
2007-11-15 20:25:43 +01:00
|
|
|
*/
|
2009-12-03 02:09:22 +03:00
|
|
|
HA_EXTRA_ADD_CHILDREN_LIST,
|
Bug#26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Bug 25038 - Waiting TRUNCATE
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Bug 19627 - temporary merge table locking
Bug 27660 - Falcon: merge table possible
Bug 30273 - merge tables: Can't lock file (errno: 155)
The problems were:
Bug 26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
1. A thread trying to lock a MERGE table performs busy waiting while
REPAIR TABLE or a similar table administration task is ongoing on
one or more of its MyISAM tables.
2. A thread trying to lock a MERGE table performs busy waiting until all
threads that did REPAIR TABLE or similar table administration tasks
on one or more of its MyISAM tables in LOCK TABLES segments do UNLOCK
TABLES. The difference against problem #1 is that the busy waiting
takes place *after* the administration task. It is terminated by
UNLOCK TABLES only.
3. Two FLUSH TABLES within a LOCK TABLES segment can invalidate the
lock. This does *not* require a MERGE table. The first FLUSH TABLES
can be replaced by any statement that requires other threads to
reopen the table. In 5.0 and 5.1 a single FLUSH TABLES can provoke
the problem.
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Trying DML on a MERGE table, which has a child locked and
repaired by another thread, made an infinite loop in the server.
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Locking a MERGE table and its children in parent-child order
and flushing the child deadlocked the server.
Bug 25038 - Waiting TRUNCATE
Truncating a MERGE child, while the MERGE table was in use,
let the truncate fail instead of waiting for the table to
become free.
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Repairing a child of an open MERGE table corrupted the child.
It was necessary to FLUSH the child first.
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Flushing and optimizing locked MERGE children crashed the server.
Bug 19627 - temporary merge table locking
Use of a temporary MERGE table with non-temporary children
could corrupt the children.
Temporary tables are never locked. So we do now prohibit
non-temporary chidlren of a temporary MERGE table.
Bug 27660 - Falcon: merge table possible
It was possible to create a MERGE table with non-MyISAM children.
Bug 30273 - merge tables: Can't lock file (errno: 155)
This was a Windows-only bug. Table administration statements
sometimes failed with "Can't lock file (errno: 155)".
These bugs are fixed by a new implementation of MERGE table open.
When opening a MERGE table in open_tables() we do now add the
child tables to the list of tables to be opened by open_tables()
(the "query_list"). The children are not opened in the handler at
this stage.
After opening the parent, open_tables() opens each child from the
now extended query_list. When the last child is opened, we remove
the children from the query_list again and attach the children to
the parent. This behaves similar to the old open. However it does
not open the MyISAM tables directly, but grabs them from the already
open children.
When closing a MERGE table in close_thread_table() we detach the
children only. Closing of the children is done implicitly because
they are in thd->open_tables.
For more detail see the comment at the top of ha_myisammrg.cc.
Changed from open_ltable() to open_and_lock_tables() in all places
that can be relevant for MERGE tables. The latter can handle tables
added to the list on the fly. When open_ltable() was used in a loop
over a list of tables, the list must be temporarily terminated
after every table for open_and_lock_tables().
table_list->required_type is set to FRMTYPE_TABLE to avoid open of
special tables. Handling of derived tables is suppressed.
These details are handled by the new function
open_n_lock_single_table(), which has nearly the same signature as
open_ltable() and can replace it in most cases.
In reopen_tables() some of the tables open by a thread can be
closed and reopened. When a MERGE child is affected, the parent
must be closed and reopened too. Closing of the parent is forced
before the first child is closed. Reopen happens in the order of
thd->open_tables. MERGE parents do not attach their children
automatically at open. This is done after all tables are reopened.
So all children are open when attaching them.
Special lock handling like mysql_lock_abort() or mysql_lock_remove()
needs to be suppressed for MERGE children or forwarded to the parent.
This depends on the situation. In loops over all open tables one
suppresses child lock handling. When a single table is touched,
forwarding is done.
Behavioral changes:
===================
This patch changes the behavior of temporary MERGE tables.
Temporary MERGE must have temporary children.
The old behavior was wrong. A temporary table is not locked. Hence
even non-temporary children were not locked. See
Bug 19627 - temporary merge table locking.
You cannot change the union list of a non-temporary MERGE table
when LOCK TABLES is in effect. The following does *not* work:
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ...;
LOCK TABLES t1 WRITE, t2 WRITE, m1 WRITE;
ALTER TABLE m1 ... UNION=(t1,t2) ...;
However, you can do this with a temporary MERGE table.
You cannot create a MERGE table with CREATE ... SELECT, neither
as a temporary MERGE table, nor as a non-temporary MERGE table.
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ... SELECT ...;
Gives error message: table is not BASE TABLE.
2007-11-15 20:25:43 +01:00
|
|
|
HA_EXTRA_ATTACH_CHILDREN,
|
2009-12-03 02:09:22 +03:00
|
|
|
HA_EXTRA_IS_ATTACHED_CHILDREN,
|
Bug#26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Bug 25038 - Waiting TRUNCATE
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Bug 19627 - temporary merge table locking
Bug 27660 - Falcon: merge table possible
Bug 30273 - merge tables: Can't lock file (errno: 155)
The problems were:
Bug 26379 - Combination of FLUSH TABLE and REPAIR TABLE
corrupts a MERGE table
1. A thread trying to lock a MERGE table performs busy waiting while
REPAIR TABLE or a similar table administration task is ongoing on
one or more of its MyISAM tables.
2. A thread trying to lock a MERGE table performs busy waiting until all
threads that did REPAIR TABLE or similar table administration tasks
on one or more of its MyISAM tables in LOCK TABLES segments do UNLOCK
TABLES. The difference against problem #1 is that the busy waiting
takes place *after* the administration task. It is terminated by
UNLOCK TABLES only.
3. Two FLUSH TABLES within a LOCK TABLES segment can invalidate the
lock. This does *not* require a MERGE table. The first FLUSH TABLES
can be replaced by any statement that requires other threads to
reopen the table. In 5.0 and 5.1 a single FLUSH TABLES can provoke
the problem.
Bug 26867 - LOCK TABLES + REPAIR + merge table result in
memory/cpu hogging
Trying DML on a MERGE table, which has a child locked and
repaired by another thread, made an infinite loop in the server.
Bug 26377 - Deadlock with MERGE and FLUSH TABLE
Locking a MERGE table and its children in parent-child order
and flushing the child deadlocked the server.
Bug 25038 - Waiting TRUNCATE
Truncating a MERGE child, while the MERGE table was in use,
let the truncate fail instead of waiting for the table to
become free.
Bug 25700 - merge base tables get corrupted by
optimize/analyze/repair table
Repairing a child of an open MERGE table corrupted the child.
It was necessary to FLUSH the child first.
Bug 30275 - Merge tables: flush tables or unlock tables
causes server to crash
Flushing and optimizing locked MERGE children crashed the server.
Bug 19627 - temporary merge table locking
Use of a temporary MERGE table with non-temporary children
could corrupt the children.
Temporary tables are never locked. So we do now prohibit
non-temporary chidlren of a temporary MERGE table.
Bug 27660 - Falcon: merge table possible
It was possible to create a MERGE table with non-MyISAM children.
Bug 30273 - merge tables: Can't lock file (errno: 155)
This was a Windows-only bug. Table administration statements
sometimes failed with "Can't lock file (errno: 155)".
These bugs are fixed by a new implementation of MERGE table open.
When opening a MERGE table in open_tables() we do now add the
child tables to the list of tables to be opened by open_tables()
(the "query_list"). The children are not opened in the handler at
this stage.
After opening the parent, open_tables() opens each child from the
now extended query_list. When the last child is opened, we remove
the children from the query_list again and attach the children to
the parent. This behaves similar to the old open. However it does
not open the MyISAM tables directly, but grabs them from the already
open children.
When closing a MERGE table in close_thread_table() we detach the
children only. Closing of the children is done implicitly because
they are in thd->open_tables.
For more detail see the comment at the top of ha_myisammrg.cc.
Changed from open_ltable() to open_and_lock_tables() in all places
that can be relevant for MERGE tables. The latter can handle tables
added to the list on the fly. When open_ltable() was used in a loop
over a list of tables, the list must be temporarily terminated
after every table for open_and_lock_tables().
table_list->required_type is set to FRMTYPE_TABLE to avoid open of
special tables. Handling of derived tables is suppressed.
These details are handled by the new function
open_n_lock_single_table(), which has nearly the same signature as
open_ltable() and can replace it in most cases.
In reopen_tables() some of the tables open by a thread can be
closed and reopened. When a MERGE child is affected, the parent
must be closed and reopened too. Closing of the parent is forced
before the first child is closed. Reopen happens in the order of
thd->open_tables. MERGE parents do not attach their children
automatically at open. This is done after all tables are reopened.
So all children are open when attaching them.
Special lock handling like mysql_lock_abort() or mysql_lock_remove()
needs to be suppressed for MERGE children or forwarded to the parent.
This depends on the situation. In loops over all open tables one
suppresses child lock handling. When a single table is touched,
forwarding is done.
Behavioral changes:
===================
This patch changes the behavior of temporary MERGE tables.
Temporary MERGE must have temporary children.
The old behavior was wrong. A temporary table is not locked. Hence
even non-temporary children were not locked. See
Bug 19627 - temporary merge table locking.
You cannot change the union list of a non-temporary MERGE table
when LOCK TABLES is in effect. The following does *not* work:
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ...;
LOCK TABLES t1 WRITE, t2 WRITE, m1 WRITE;
ALTER TABLE m1 ... UNION=(t1,t2) ...;
However, you can do this with a temporary MERGE table.
You cannot create a MERGE table with CREATE ... SELECT, neither
as a temporary MERGE table, nor as a non-temporary MERGE table.
CREATE TABLE m1 ... ENGINE=MRG_MYISAM ... SELECT ...;
Gives error message: table is not BASE TABLE.
2007-11-15 20:25:43 +01:00
|
|
|
HA_EXTRA_DETACH_CHILDREN
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
2007-10-11 18:07:40 +03:00
|
|
|
/* Compatible option, to be deleted in 6.0 */
|
|
|
|
#define HA_EXTRA_PREPARE_FOR_DELETE HA_EXTRA_PREPARE_FOR_DROP
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* The following is parameter to ha_panic() */
|
|
|
|
|
|
|
|
enum ha_panic_function {
|
|
|
|
HA_PANIC_CLOSE, /* Close all databases */
|
|
|
|
HA_PANIC_WRITE, /* Unlock and write status */
|
|
|
|
HA_PANIC_READ /* Lock and read keyinfo */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* The following is parameter to ha_create(); keytypes */
|
|
|
|
|
|
|
|
enum ha_base_keytype {
|
|
|
|
HA_KEYTYPE_END=0,
|
|
|
|
HA_KEYTYPE_TEXT=1, /* Key is sorted as letters */
|
|
|
|
HA_KEYTYPE_BINARY=2, /* Key is sorted as unsigned chars */
|
|
|
|
HA_KEYTYPE_SHORT_INT=3,
|
|
|
|
HA_KEYTYPE_LONG_INT=4,
|
|
|
|
HA_KEYTYPE_FLOAT=5,
|
|
|
|
HA_KEYTYPE_DOUBLE=6,
|
|
|
|
HA_KEYTYPE_NUM=7, /* Not packed num with pre-space */
|
|
|
|
HA_KEYTYPE_USHORT_INT=8,
|
|
|
|
HA_KEYTYPE_ULONG_INT=9,
|
|
|
|
HA_KEYTYPE_LONGLONG=10,
|
|
|
|
HA_KEYTYPE_ULONGLONG=11,
|
|
|
|
HA_KEYTYPE_INT24=12,
|
|
|
|
HA_KEYTYPE_UINT24=13,
|
|
|
|
HA_KEYTYPE_INT8=14,
|
2004-12-18 05:19:21 +02:00
|
|
|
/* Varchar (0-255 bytes) with length packed with 1 byte */
|
|
|
|
HA_KEYTYPE_VARTEXT1=15, /* Key is sorted as letters */
|
|
|
|
HA_KEYTYPE_VARBINARY1=16, /* Key is sorted as unsigned chars */
|
|
|
|
/* Varchar (0-65535 bytes) with length packed with 2 bytes */
|
|
|
|
HA_KEYTYPE_VARTEXT2=17, /* Key is sorted as letters */
|
2004-12-18 06:05:16 +02:00
|
|
|
HA_KEYTYPE_VARBINARY2=18, /* Key is sorted as unsigned chars */
|
2004-12-19 20:25:19 +02:00
|
|
|
HA_KEYTYPE_BIT=19
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#define HA_MAX_KEYTYPE 31 /* Must be log2-1 */
|
|
|
|
|
|
|
|
/* These flags kan be OR:ed to key-flag */
|
|
|
|
|
|
|
|
#define HA_NOSAME 1 /* Set if not dupplicated records */
|
|
|
|
#define HA_PACK_KEY 2 /* Pack string key to previous key */
|
|
|
|
#define HA_AUTO_KEY 16
|
|
|
|
#define HA_BINARY_PACK_KEY 32 /* Packing of all keys to prev key */
|
2002-05-22 18:51:21 +03:00
|
|
|
#define HA_FULLTEXT 128 /* For full-text search */
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_UNIQUE_CHECK 256 /* Check the key for uniqueness */
|
2002-05-22 18:51:21 +03:00
|
|
|
#define HA_SPATIAL 1024 /* For spatial search */
|
2002-01-16 00:42:52 +02:00
|
|
|
#define HA_NULL_ARE_EQUAL 2048 /* NULL in key are cmp as equal */
|
2004-05-12 00:29:52 +03:00
|
|
|
#define HA_GENERATED_KEY 8192 /* Automaticly generated key */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2006-01-12 10:05:07 +01:00
|
|
|
/* The combination of the above can be used for key type comparison. */
|
|
|
|
#define HA_KEYFLAG_MASK (HA_NOSAME | HA_PACK_KEY | HA_AUTO_KEY | \
|
|
|
|
HA_BINARY_PACK_KEY | HA_FULLTEXT | HA_UNIQUE_CHECK | \
|
|
|
|
HA_SPATIAL | HA_NULL_ARE_EQUAL | HA_GENERATED_KEY)
|
|
|
|
|
2009-10-16 19:21:54 +04:00
|
|
|
/*
|
|
|
|
Key contains partial segments.
|
|
|
|
|
|
|
|
This flag is internal to the MySQL server by design. It is not supposed
|
|
|
|
neither to be saved in FRM-files, nor to be passed to storage engines.
|
|
|
|
It is intended to pass information into internal static sort_keys(KEY *,
|
|
|
|
KEY *) function.
|
|
|
|
|
|
|
|
This flag can be calculated -- it's based on key lengths comparison.
|
|
|
|
*/
|
|
|
|
#define HA_KEY_HAS_PART_KEY_SEG 65536
|
2007-10-29 17:34:01 +04:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Automatic bits in key-flag */
|
|
|
|
|
|
|
|
#define HA_SPACE_PACK_USED 4 /* Test for if SPACE_PACK used */
|
|
|
|
#define HA_VAR_LENGTH_KEY 8
|
|
|
|
#define HA_NULL_PART_KEY 64
|
2010-02-20 13:07:32 +03:00
|
|
|
#define HA_USES_COMMENT 4096
|
2005-11-05 13:20:35 +02:00
|
|
|
#define HA_USES_PARSER 16384 /* Fulltext index uses [pre]parser */
|
2006-05-03 15:59:17 +03:00
|
|
|
#define HA_USES_BLOCK_SIZE ((uint) 32768)
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_SORT_ALLOWS_SAME 512 /* Intern bit when sorting records */
|
|
|
|
|
2003-12-12 22:26:58 +02:00
|
|
|
/* These flags can be added to key-seg-flag */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
#define HA_SPACE_PACK 1 /* Pack space in key-seg */
|
2003-12-12 22:26:58 +02:00
|
|
|
#define HA_PART_KEY_SEG 4 /* Used by MySQL for part-key-cols */
|
2004-12-06 02:00:37 +02:00
|
|
|
#define HA_VAR_LENGTH_PART 8
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_NULL_PART 16
|
|
|
|
#define HA_BLOB_PART 32
|
|
|
|
#define HA_SWAP_KEY 64
|
|
|
|
#define HA_REVERSE_SORT 128 /* Sort key in reverse order */
|
2001-10-09 14:53:54 +02:00
|
|
|
#define HA_NO_SORT 256 /* do not bother sorting on this keyseg */
|
2004-12-06 02:00:37 +02:00
|
|
|
/*
|
|
|
|
End space in unique/varchar are considered equal. (Like 'a' and 'a ')
|
|
|
|
Only needed for internal temporary tables.
|
|
|
|
*/
|
|
|
|
#define HA_END_SPACE_ARE_EQUAL 512
|
2004-12-19 20:25:19 +02:00
|
|
|
#define HA_BIT_PART 1024
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* optionbits for database */
|
|
|
|
#define HA_OPTION_PACK_RECORD 1
|
|
|
|
#define HA_OPTION_PACK_KEYS 2
|
|
|
|
#define HA_OPTION_COMPRESS_RECORD 4
|
|
|
|
#define HA_OPTION_LONG_BLOB_PTR 8 /* new ISAM format */
|
|
|
|
#define HA_OPTION_TMP_TABLE 16
|
|
|
|
#define HA_OPTION_CHECKSUM 32
|
|
|
|
#define HA_OPTION_DELAY_KEY_WRITE 64
|
|
|
|
#define HA_OPTION_NO_PACK_KEYS 128 /* Reserved for MySQL */
|
2005-11-25 13:57:13 +03:00
|
|
|
#define HA_OPTION_CREATE_FROM_ENGINE 256
|
2005-12-28 16:05:30 +04:00
|
|
|
#define HA_OPTION_RELIES_ON_SQL_LAYER 512
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_OPTION_NULL_FIELDS 1024
|
|
|
|
#define HA_OPTION_PAGE_CHECKSUM 2048
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_OPTION_TEMP_COMPRESS_RECORD ((uint) 16384) /* set by isamchk */
|
|
|
|
#define HA_OPTION_READ_ONLY_DATA ((uint) 32768) /* Set by isamchk */
|
|
|
|
|
2000-09-26 00:33:25 +03:00
|
|
|
/* Bits in flag to create() */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
#define HA_DONT_TOUCH_DATA 1 /* Don't empty datafile (isamchk) */
|
|
|
|
#define HA_PACK_RECORD 2 /* Request packed record format */
|
|
|
|
#define HA_CREATE_TMP_TABLE 4
|
|
|
|
#define HA_CREATE_CHECKSUM 8
|
2007-07-11 10:49:54 +03:00
|
|
|
#define HA_CREATE_KEEP_FILES 16 /* don't overwrite .MYD and MYI */
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_CREATE_PAGE_CHECKSUM 32
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_CREATE_DELAY_KEY_WRITE 64
|
2005-12-28 16:05:30 +04:00
|
|
|
#define HA_CREATE_RELIES_ON_SQL_LAYER 128
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-11-26 08:54:13 +01:00
|
|
|
/*
|
|
|
|
The following flags (OR-ed) are passed to handler::info() method.
|
|
|
|
The method copies misc handler information out of the storage engine
|
|
|
|
to data structures accessible from MySQL
|
|
|
|
|
|
|
|
Same flags are also passed down to mi_status, myrg_status, etc.
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-11-26 08:54:13 +01:00
|
|
|
/* this one is not used */
|
|
|
|
#define HA_STATUS_POS 1
|
|
|
|
/*
|
|
|
|
assuming the table keeps shared actual copy of the 'info' and
|
|
|
|
local, possibly outdated copy, the following flag means that
|
|
|
|
it should not try to get the actual data (locking the shared structure)
|
|
|
|
slightly outdated version will suffice
|
|
|
|
*/
|
|
|
|
#define HA_STATUS_NO_LOCK 2
|
|
|
|
/* update the time of the last modification (in handler::update_time) */
|
|
|
|
#define HA_STATUS_TIME 4
|
|
|
|
/*
|
|
|
|
update the 'constant' part of the info:
|
|
|
|
handler::max_data_file_length, max_index_file_length, create_time
|
|
|
|
sortkey, ref_length, block_size, data_file_name, index_file_name.
|
|
|
|
handler::table->s->keys_in_use, keys_for_keyread, rec_per_key
|
|
|
|
*/
|
|
|
|
#define HA_STATUS_CONST 8
|
|
|
|
/*
|
|
|
|
update the 'variable' part of the info:
|
|
|
|
handler::records, deleted, data_file_length, index_file_length,
|
|
|
|
delete_length, check_time, mean_rec_length
|
|
|
|
*/
|
|
|
|
#define HA_STATUS_VARIABLE 16
|
|
|
|
/*
|
|
|
|
get the information about the key that caused last duplicate value error
|
|
|
|
update handler::errkey and handler::dupp_ref
|
|
|
|
see handler::get_dup_key()
|
|
|
|
*/
|
|
|
|
#define HA_STATUS_ERRKEY 32
|
|
|
|
/*
|
|
|
|
update handler::auto_increment_value
|
|
|
|
*/
|
|
|
|
#define HA_STATUS_AUTO 64
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2007-10-11 18:07:40 +03:00
|
|
|
/*
|
|
|
|
Errorcodes given by handler functions
|
|
|
|
|
|
|
|
opt_sum_query() assumes these codes are > 1
|
|
|
|
Do not add error numbers before HA_ERR_FIRST.
|
|
|
|
If necessary to add lower numbers, change HA_ERR_FIRST accordingly.
|
|
|
|
*/
|
|
|
|
#define HA_ERR_FIRST 120 /* Copy of first error nr.*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
#define HA_ERR_KEY_NOT_FOUND 120 /* Didn't find key on read or update */
|
|
|
|
#define HA_ERR_FOUND_DUPP_KEY 121 /* Dupplicate key on write */
|
2008-03-28 18:45:03 +02:00
|
|
|
#define HA_ERR_INTERNAL_ERROR 122 /* Internal error */
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_ERR_RECORD_CHANGED 123 /* Uppdate with is recoverable */
|
|
|
|
#define HA_ERR_WRONG_INDEX 124 /* Wrong index given to function */
|
|
|
|
#define HA_ERR_CRASHED 126 /* Indexfile is crashed */
|
|
|
|
#define HA_ERR_WRONG_IN_RECORD 127 /* Record-file is crashed */
|
|
|
|
#define HA_ERR_OUT_OF_MEM 128 /* Record-file is crashed */
|
2003-01-28 17:42:08 +01:00
|
|
|
#define HA_ERR_NOT_A_TABLE 130 /* not a MYI file - no signature */
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_ERR_WRONG_COMMAND 131 /* Command not supported */
|
|
|
|
#define HA_ERR_OLD_FILE 132 /* old databasfile */
|
|
|
|
#define HA_ERR_NO_ACTIVE_RECORD 133 /* No record read in update() */
|
2006-05-09 15:14:29 -04:00
|
|
|
#define HA_ERR_RECORD_DELETED 134 /* A record is not there */
|
2000-07-31 21:29:14 +02:00
|
|
|
#define HA_ERR_RECORD_FILE_FULL 135 /* No more room in file */
|
|
|
|
#define HA_ERR_INDEX_FILE_FULL 136 /* No more room in file */
|
|
|
|
#define HA_ERR_END_OF_FILE 137 /* end in next/prev/first/last */
|
|
|
|
#define HA_ERR_UNSUPPORTED 138 /* unsupported extension used */
|
|
|
|
#define HA_ERR_TO_BIG_ROW 139 /* Too big row */
|
|
|
|
#define HA_WRONG_CREATE_OPTION 140 /* Wrong create option */
|
|
|
|
#define HA_ERR_FOUND_DUPP_UNIQUE 141 /* Dupplicate unique on write */
|
|
|
|
#define HA_ERR_UNKNOWN_CHARSET 142 /* Can't open charset */
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_ERR_WRONG_MRG_TABLE_DEF 143 /* conflicting tables in MERGE */
|
2000-09-26 00:33:25 +03:00
|
|
|
#define HA_ERR_CRASHED_ON_REPAIR 144 /* Last (automatic?) repair failed */
|
2000-11-21 03:43:34 +02:00
|
|
|
#define HA_ERR_CRASHED_ON_USAGE 145 /* Table must be repaired */
|
2002-10-25 14:09:47 +00:00
|
|
|
#define HA_ERR_LOCK_WAIT_TIMEOUT 146
|
2001-03-21 15:34:16 -05:00
|
|
|
#define HA_ERR_LOCK_TABLE_FULL 147
|
2001-05-29 09:29:08 -04:00
|
|
|
#define HA_ERR_READ_ONLY_TRANSACTION 148 /* Updates not allowed */
|
2001-09-17 23:43:53 +03:00
|
|
|
#define HA_ERR_LOCK_DEADLOCK 149
|
2001-10-30 17:38:44 +02:00
|
|
|
#define HA_ERR_CANNOT_ADD_FOREIGN 150 /* Cannot add a foreign key constr. */
|
|
|
|
#define HA_ERR_NO_REFERENCED_ROW 151 /* Cannot add a child row */
|
|
|
|
#define HA_ERR_ROW_IS_REFERENCED 152 /* Cannot delete a parent row */
|
2003-06-15 01:04:28 +03:00
|
|
|
#define HA_ERR_NO_SAVEPOINT 153 /* No savepoint with that name */
|
2003-08-11 22:44:43 +03:00
|
|
|
#define HA_ERR_NON_UNIQUE_BLOCK_SIZE 154 /* Non unique key block size */
|
2004-09-13 14:46:38 +02:00
|
|
|
#define HA_ERR_NO_SUCH_TABLE 155 /* The table does not exist in engine */
|
2004-04-15 09:14:14 +02:00
|
|
|
#define HA_ERR_TABLE_EXIST 156 /* The table existed in storage engine */
|
|
|
|
#define HA_ERR_NO_CONNECTION 157 /* Could not connect to storage engine */
|
2007-10-11 18:07:40 +03:00
|
|
|
/* NULLs are not supported in spatial index */
|
|
|
|
#define HA_ERR_NULL_IN_SPATIAL 158
|
2005-04-07 20:17:37 +02:00
|
|
|
#define HA_ERR_TABLE_DEF_CHANGED 159 /* The table changed in storage engine */
|
2008-03-28 18:45:03 +02:00
|
|
|
/* There's no partition in table for given value */
|
|
|
|
#define HA_ERR_NO_PARTITION_FOUND 160
|
2005-12-22 06:39:02 +01:00
|
|
|
#define HA_ERR_RBR_LOGGING_FAILED 161 /* Row-based binlogging of row failed */
|
2007-10-11 18:07:40 +03:00
|
|
|
#define HA_ERR_DROP_INDEX_FK 162 /* Index needed in foreign key constr */
|
|
|
|
/*
|
|
|
|
Upholding foreign key constraints would lead to a duplicate key error
|
|
|
|
in some other table.
|
|
|
|
*/
|
|
|
|
#define HA_ERR_FOREIGN_DUPLICATE_KEY 163
|
|
|
|
/* The table changed in storage engine */
|
|
|
|
#define HA_ERR_TABLE_NEEDS_UPGRADE 164
|
|
|
|
#define HA_ERR_TABLE_READONLY 165 /* The table is not writable */
|
2005-12-22 06:39:02 +01:00
|
|
|
|
2007-01-29 10:40:26 +01:00
|
|
|
#define HA_ERR_AUTOINC_READ_FAILED 166 /* Failed to get next autoinc value */
|
|
|
|
#define HA_ERR_AUTOINC_ERANGE 167 /* Failed to set row autoinc value */
|
|
|
|
#define HA_ERR_GENERIC 168 /* Generic error */
|
2007-10-11 18:07:40 +03:00
|
|
|
/* row not actually updated: new values same as the old values */
|
|
|
|
#define HA_ERR_RECORD_IS_THE_SAME 169
|
|
|
|
/* It is not possible to log this statement */
|
2007-10-20 18:19:55 +02:00
|
|
|
#define HA_ERR_LOGGING_IMPOSSIBLE 170 /* It is not possible to log this
|
|
|
|
statement */
|
|
|
|
#define HA_ERR_CORRUPT_EVENT 171 /* The event was corrupt, leading to
|
|
|
|
illegal data being read */
|
2007-12-07 22:27:48 +02:00
|
|
|
#define HA_ERR_NEW_FILE 172 /* New file format */
|
2008-02-15 17:58:09 +02:00
|
|
|
#define HA_ERR_ROWS_EVENT_APPLY 173 /* The event could not be processed
|
2008-01-31 14:54:03 +02:00
|
|
|
no other hanlder error happened */
|
2008-03-28 18:45:03 +02:00
|
|
|
#define HA_ERR_INITIALIZATION 174 /* Error during initialization */
|
|
|
|
#define HA_ERR_FILE_TOO_SHORT 175 /* File too short */
|
|
|
|
#define HA_ERR_WRONG_CRC 176 /* Wrong CRC on page */
|
2009-07-24 12:15:06 +05:30
|
|
|
#define HA_ERR_TOO_MANY_CONCURRENT_TRXS 177 /*Too many active concurrent transactions */
|
|
|
|
#define HA_ERR_LAST 177 /* Copy of last error nr */
|
2007-10-11 18:07:40 +03:00
|
|
|
|
|
|
|
/* Number of different errors */
|
2004-12-23 20:11:38 +01:00
|
|
|
#define HA_ERR_ERRORS (HA_ERR_LAST - HA_ERR_FIRST + 1)
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* Other constants */
|
|
|
|
|
|
|
|
#define HA_NAMELEN 64 /* Max length of saved filename */
|
2007-03-17 00:13:25 +01:00
|
|
|
#define NO_SUCH_KEY (~(uint)0) /* used as a key no. */
|
|
|
|
|
|
|
|
typedef ulong key_part_map;
|
|
|
|
#define HA_WHOLE_KEY (~(key_part_map)0)
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* Intern constants in databases */
|
|
|
|
|
|
|
|
/* bits in _search */
|
|
|
|
#define SEARCH_FIND 1
|
|
|
|
#define SEARCH_NO_FIND 2
|
|
|
|
#define SEARCH_SAME 4
|
|
|
|
#define SEARCH_BIGGER 8
|
|
|
|
#define SEARCH_SMALLER 16
|
|
|
|
#define SEARCH_SAVE_BUFF 32
|
|
|
|
#define SEARCH_UPDATE 64
|
|
|
|
#define SEARCH_PREFIX 128
|
|
|
|
#define SEARCH_LAST 256
|
2002-01-05 22:51:42 +02:00
|
|
|
#define MBR_CONTAIN 512
|
|
|
|
#define MBR_INTERSECT 1024
|
|
|
|
#define MBR_WITHIN 2048
|
|
|
|
#define MBR_DISJOINT 4096
|
|
|
|
#define MBR_EQUAL 8192
|
|
|
|
#define MBR_DATA 16384
|
2002-01-16 00:42:52 +02:00
|
|
|
#define SEARCH_NULL_ARE_EQUAL 32768 /* NULL in keys are equal */
|
2003-01-09 02:19:14 +02:00
|
|
|
#define SEARCH_NULL_ARE_NOT_EQUAL 65536 /* NULL in keys are not equal */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* bits in opt_flag */
|
|
|
|
#define QUICK_USED 1
|
|
|
|
#define READ_CACHE_USED 2
|
|
|
|
#define READ_CHECK_USED 4
|
|
|
|
#define KEY_READ_USED 8
|
|
|
|
#define WRITE_CACHE_USED 16
|
2002-10-25 14:09:47 +00:00
|
|
|
#define OPT_NO_ROWS 32
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* bits in update */
|
|
|
|
#define HA_STATE_CHANGED 1 /* Database has changed */
|
|
|
|
#define HA_STATE_AKTIV 2 /* Has a current record */
|
|
|
|
#define HA_STATE_WRITTEN 4 /* Record is written */
|
|
|
|
#define HA_STATE_DELETED 8
|
|
|
|
#define HA_STATE_NEXT_FOUND 16 /* Next found record (record before) */
|
|
|
|
#define HA_STATE_PREV_FOUND 32 /* Prev found record (record after) */
|
|
|
|
#define HA_STATE_NO_KEY 64 /* Last read didn't find record */
|
|
|
|
#define HA_STATE_KEY_CHANGED 128
|
|
|
|
#define HA_STATE_WRITE_AT_END 256 /* set in _ps_find_writepos */
|
|
|
|
#define HA_STATE_BUFF_SAVED 512 /* If current keybuff is info->buff */
|
|
|
|
#define HA_STATE_ROW_CHANGED 1024 /* To invalide ROW cache */
|
|
|
|
#define HA_STATE_EXTEND_BLOCK 2048
|
2006-03-10 15:03:04 +01:00
|
|
|
#define HA_STATE_RNEXT_SAME 4096 /* rnext_same occupied lastkey2 */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-06-24 19:34:55 +02:00
|
|
|
/* myisampack expects no more than 32 field types. */
|
2000-07-31 21:29:14 +02:00
|
|
|
enum en_fieldtype {
|
2001-11-07 00:13:29 +02:00
|
|
|
FIELD_LAST=-1,FIELD_NORMAL,FIELD_SKIP_ENDSPACE,FIELD_SKIP_PRESPACE,
|
|
|
|
FIELD_SKIP_ZERO,FIELD_BLOB,FIELD_CONSTANT,FIELD_INTERVALL,FIELD_ZERO,
|
2006-01-03 17:54:54 +01:00
|
|
|
FIELD_VARCHAR,FIELD_CHECK,
|
|
|
|
FIELD_enum_val_count
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
enum data_file_type {
|
2007-10-11 18:07:40 +03:00
|
|
|
STATIC_RECORD, DYNAMIC_RECORD, COMPRESSED_RECORD, BLOCK_RECORD
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
2004-05-16 14:48:32 +03:00
|
|
|
/* For key ranges */
|
|
|
|
|
2004-12-23 21:45:10 +01:00
|
|
|
#define NO_MIN_RANGE 1
|
|
|
|
#define NO_MAX_RANGE 2
|
|
|
|
#define NEAR_MIN 4
|
|
|
|
#define NEAR_MAX 8
|
|
|
|
#define UNIQUE_RANGE 16
|
|
|
|
#define EQ_RANGE 32
|
|
|
|
#define NULL_RANGE 64
|
|
|
|
#define GEOM_FLAG 128
|
2005-07-18 13:31:02 +02:00
|
|
|
#define SKIP_RANGE 256
|
2004-12-23 21:45:10 +01:00
|
|
|
|
2004-05-16 14:48:32 +03:00
|
|
|
typedef struct st_key_range
|
|
|
|
{
|
WL#3817: Simplify string / memory area types and make things more consistent (first part)
The following type conversions was done:
- Changed byte to uchar
- Changed gptr to uchar*
- Change my_string to char *
- Change my_size_t to size_t
- Change size_s to size_t
Removed declaration of byte, gptr, my_string, my_size_t and size_s.
Following function parameter changes was done:
- All string functions in mysys/strings was changed to use size_t
instead of uint for string lengths.
- All read()/write() functions changed to use size_t (including vio).
- All protocoll functions changed to use size_t instead of uint
- Functions that used a pointer to a string length was changed to use size_t*
- Changed malloc(), free() and related functions from using gptr to use void *
as this requires fewer casts in the code and is more in line with how the
standard functions work.
- Added extra length argument to dirname_part() to return the length of the
created string.
- Changed (at least) following functions to take uchar* as argument:
- db_dump()
- my_net_write()
- net_write_command()
- net_store_data()
- DBUG_DUMP()
- decimal2bin() & bin2decimal()
- Changed my_compress() and my_uncompress() to use size_t. Changed one
argument to my_uncompress() from a pointer to a value as we only return
one value (makes function easier to use).
- Changed type of 'pack_data' argument to packfrm() to avoid casts.
- Changed in readfrm() and writefrom(), ha_discover and handler::discover()
the type for argument 'frmdata' to uchar** to avoid casts.
- Changed most Field functions to use uchar* instead of char* (reduced a lot of
casts).
- Changed field->val_xxx(xxx, new_ptr) to take const pointers.
Other changes:
- Removed a lot of not needed casts
- Added a few new cast required by other changes
- Added some cast to my_multi_malloc() arguments for safety (as string lengths
needs to be uint, not size_t).
- Fixed all calls to hash-get-key functions to use size_t*. (Needed to be done
explicitely as this conflict was often hided by casting the function to
hash_get_key).
- Changed some buffers to memory regions to uchar* to avoid casts.
- Changed some string lengths from uint to size_t.
- Changed field->ptr to be uchar* instead of char*. This allowed us to
get rid of a lot of casts.
- Some changes from true -> TRUE, false -> FALSE, unsigned char -> uchar
- Include zlib.h in some files as we needed declaration of crc32()
- Changed MY_FILE_ERROR to be (size_t) -1.
- Changed many variables to hold the result of my_read() / my_write() to be
size_t. This was needed to properly detect errors (which are
returned as (size_t) -1).
- Removed some very old VMS code
- Changed packfrm()/unpackfrm() to not be depending on uint size
(portability fix)
- Removed windows specific code to restore cursor position as this
causes slowdown on windows and we should not mix read() and pread()
calls anyway as this is not thread safe. Updated function comment to
reflect this. Changed function that depended on original behavior of
my_pwrite() to itself restore the cursor position (one such case).
- Added some missing checking of return value of malloc().
- Changed definition of MOD_PAD_CHAR_TO_FULL_LENGTH to avoid 'long' overflow.
- Changed type of table_def::m_size from my_size_t to ulong to reflect that
m_size is the number of elements in the array, not a string/memory
length.
- Moved THD::max_row_length() to table.cc (as it's not depending on THD).
Inlined max_row_length_blob() into this function.
- More function comments
- Fixed some compiler warnings when compiled without partitions.
- Removed setting of LEX_STRING() arguments in declaration (portability fix).
- Some trivial indentation/variable name changes.
- Some trivial code simplifications:
- Replaced some calls to alloc_root + memcpy to use
strmake_root()/strdup_root().
- Changed some calls from memdup() to strmake() (Safety fix)
- Simpler loops in client-simple.c
2007-05-10 12:59:39 +03:00
|
|
|
const uchar *key;
|
2004-05-16 14:48:32 +03:00
|
|
|
uint length;
|
2007-03-17 00:13:25 +01:00
|
|
|
key_part_map keypart_map;
|
2004-05-16 14:48:32 +03:00
|
|
|
enum ha_rkey_function flag;
|
|
|
|
} key_range;
|
|
|
|
|
2004-12-23 21:45:10 +01:00
|
|
|
typedef struct st_key_multi_range
|
|
|
|
{
|
|
|
|
key_range start_key;
|
|
|
|
key_range end_key;
|
|
|
|
char *ptr; /* Free to use by caller (ptr to row etc) */
|
|
|
|
uint range_flag; /* key range flags see above */
|
|
|
|
} KEY_MULTI_RANGE;
|
|
|
|
|
2004-05-16 14:48:32 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* For number of records */
|
|
|
|
#ifdef BIG_TABLES
|
2002-11-14 12:21:36 +02:00
|
|
|
#define rows2double(A) ulonglong2double(A)
|
2000-07-31 21:29:14 +02:00
|
|
|
typedef my_off_t ha_rows;
|
|
|
|
#else
|
2002-11-14 12:21:36 +02:00
|
|
|
#define rows2double(A) (double) (A)
|
2002-10-25 14:09:47 +00:00
|
|
|
typedef ulong ha_rows;
|
2000-07-31 21:29:14 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define HA_POS_ERROR (~ (ha_rows) 0)
|
|
|
|
#define HA_OFFSET_ERROR (~ (my_off_t) 0)
|
2000-08-15 20:09:37 +03:00
|
|
|
|
|
|
|
#if SYSTEM_SIZEOF_OFF_T == 4
|
|
|
|
#define MAX_FILE_SIZE INT_MAX32
|
|
|
|
#else
|
|
|
|
#define MAX_FILE_SIZE LONGLONG_MAX
|
|
|
|
#endif
|
|
|
|
|
2004-12-18 05:19:21 +02:00
|
|
|
#define HA_VARCHAR_PACKLENGTH(field_length) ((field_length) < 256 ? 1 :2)
|
|
|
|
|
2007-10-11 18:07:40 +03:00
|
|
|
/* invalidator function reference for Query Cache */
|
2010-05-31 12:29:54 -03:00
|
|
|
C_MODE_START
|
2007-10-11 18:07:40 +03:00
|
|
|
typedef void (* invalidator_by_filename)(const char * filename);
|
2010-05-31 12:29:54 -03:00
|
|
|
C_MODE_END
|
2007-10-11 18:07:40 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
#endif /* _my_base_h */
|