24
24
#include <drizzled/definitions.h>
25
#include <drizzled/plugin.h>
25
#include <drizzled/sql_plugin.h>
26
26
#include <drizzled/handler_structs.h>
27
#include <drizzled/message/schema.pb.h>
28
27
#include <drizzled/message/table.pb.h>
29
#include "drizzled/plugin/plugin.h"
30
#include "drizzled/sql_string.h"
31
#include "drizzled/schema_identifier.h"
32
#include "drizzled/table_identifier.h"
33
#include "drizzled/cached_directory.h"
34
#include "drizzled/plugin/monitored_in_transaction.h"
36
#include "drizzled/hash.h"
28
#include <drizzled/registry.h>
49
typedef struct st_hash HASH;
52
typedef drizzle_lex_string LEX_STRING;
40
typedef struct st_mysql_lex_string LEX_STRING;
53
41
typedef bool (stat_print_fn)(Session *session, const char *type, uint32_t type_len,
54
42
const char *file, uint32_t file_len,
55
43
const char *status, uint32_t status_len);
44
enum ha_stat_type { HA_ENGINE_STATUS, HA_ENGINE_LOGS, HA_ENGINE_MUTEX };
57
46
/* Possible flags of a StorageEngine (there can be 32 of them) */
58
47
enum engine_flag_bits {
59
48
HTON_BIT_ALTER_NOT_SUPPORTED, // Engine does not support alter
49
HTON_BIT_CAN_RECREATE, // Delete all is used for truncate
60
50
HTON_BIT_HIDDEN, // Engine does not appear in lists
51
HTON_BIT_FLUSH_AFTER_RENAME,
61
52
HTON_BIT_NOT_USER_SELECTABLE,
62
53
HTON_BIT_TEMPORARY_NOT_SUPPORTED, // Having temporary tables not supported
63
54
HTON_BIT_TEMPORARY_ONLY,
64
55
HTON_BIT_FILE_BASED, // use for check_lowercase_names
65
HTON_BIT_DOES_TRANSACTIONS,
66
HTON_BIT_STATS_RECORDS_IS_EXACT,
68
HTON_BIT_CAN_INDEX_BLOBS,
69
HTON_BIT_PRIMARY_KEY_REQUIRED_FOR_POSITION,
70
HTON_BIT_PRIMARY_KEY_IN_READ_INDEX,
71
HTON_BIT_PARTIAL_COLUMN_READ,
72
HTON_BIT_TABLE_SCAN_ON_INDEX,
73
HTON_BIT_FAST_KEY_READ,
76
HTON_BIT_NO_AUTO_INCREMENT,
77
HTON_BIT_DUPLICATE_POS,
78
HTON_BIT_AUTO_PART_KEY,
79
HTON_BIT_REQUIRE_PRIMARY_KEY,
80
HTON_BIT_REQUIRES_KEY_COLUMNS_FOR_DELETE,
81
HTON_BIT_PRIMARY_KEY_REQUIRED_FOR_DELETE,
82
HTON_BIT_NO_PREFIX_CHAR_KEYS,
83
HTON_BIT_HAS_CHECKSUM,
84
HTON_BIT_SKIP_STORE_LOCK,
85
HTON_BIT_SCHEMA_DICTIONARY,
86
HTON_BIT_FOREIGN_KEYS,
56
HTON_BIT_HAS_DATA_DICTIONARY,
90
60
static const std::bitset<HTON_BIT_SIZE> HTON_NO_FLAGS(0);
91
61
static const std::bitset<HTON_BIT_SIZE> HTON_ALTER_NOT_SUPPORTED(1 << HTON_BIT_ALTER_NOT_SUPPORTED);
62
static const std::bitset<HTON_BIT_SIZE> HTON_CAN_RECREATE(1 << HTON_BIT_CAN_RECREATE);
92
63
static const std::bitset<HTON_BIT_SIZE> HTON_HIDDEN(1 << HTON_BIT_HIDDEN);
64
static const std::bitset<HTON_BIT_SIZE> HTON_FLUSH_AFTER_RENAME(1 << HTON_BIT_FLUSH_AFTER_RENAME);
93
65
static const std::bitset<HTON_BIT_SIZE> HTON_NOT_USER_SELECTABLE(1 << HTON_BIT_NOT_USER_SELECTABLE);
94
66
static const std::bitset<HTON_BIT_SIZE> HTON_TEMPORARY_NOT_SUPPORTED(1 << HTON_BIT_TEMPORARY_NOT_SUPPORTED);
95
67
static const std::bitset<HTON_BIT_SIZE> HTON_TEMPORARY_ONLY(1 << HTON_BIT_TEMPORARY_ONLY);
96
68
static const std::bitset<HTON_BIT_SIZE> HTON_FILE_BASED(1 << HTON_BIT_FILE_BASED);
97
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_DOES_TRANSACTIONS(1 << HTON_BIT_DOES_TRANSACTIONS);
98
static const std::bitset<HTON_BIT_SIZE> HTON_STATS_RECORDS_IS_EXACT(1 << HTON_BIT_STATS_RECORDS_IS_EXACT);
99
static const std::bitset<HTON_BIT_SIZE> HTON_NULL_IN_KEY(1 << HTON_BIT_NULL_IN_KEY);
100
static const std::bitset<HTON_BIT_SIZE> HTON_CAN_INDEX_BLOBS(1 << HTON_BIT_CAN_INDEX_BLOBS);
101
static const std::bitset<HTON_BIT_SIZE> HTON_PRIMARY_KEY_REQUIRED_FOR_POSITION(1 << HTON_BIT_PRIMARY_KEY_REQUIRED_FOR_POSITION);
102
static const std::bitset<HTON_BIT_SIZE> HTON_PRIMARY_KEY_IN_READ_INDEX(1 << HTON_BIT_PRIMARY_KEY_IN_READ_INDEX);
103
static const std::bitset<HTON_BIT_SIZE> HTON_PARTIAL_COLUMN_READ(1 << HTON_BIT_PARTIAL_COLUMN_READ);
104
static const std::bitset<HTON_BIT_SIZE> HTON_TABLE_SCAN_ON_INDEX(1 << HTON_BIT_TABLE_SCAN_ON_INDEX);
105
static const std::bitset<HTON_BIT_SIZE> HTON_FAST_KEY_READ(1 << HTON_BIT_FAST_KEY_READ);
106
static const std::bitset<HTON_BIT_SIZE> HTON_NO_BLOBS(1 << HTON_BIT_NO_BLOBS);
107
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_RECORDS(1 << HTON_BIT_HAS_RECORDS);
108
static const std::bitset<HTON_BIT_SIZE> HTON_NO_AUTO_INCREMENT(1 << HTON_BIT_NO_AUTO_INCREMENT);
109
static const std::bitset<HTON_BIT_SIZE> HTON_DUPLICATE_POS(1 << HTON_BIT_DUPLICATE_POS);
110
static const std::bitset<HTON_BIT_SIZE> HTON_AUTO_PART_KEY(1 << HTON_BIT_AUTO_PART_KEY);
111
static const std::bitset<HTON_BIT_SIZE> HTON_REQUIRE_PRIMARY_KEY(1 << HTON_BIT_REQUIRE_PRIMARY_KEY);
112
static const std::bitset<HTON_BIT_SIZE> HTON_REQUIRES_KEY_COLUMNS_FOR_DELETE(1 << HTON_BIT_REQUIRES_KEY_COLUMNS_FOR_DELETE);
113
static const std::bitset<HTON_BIT_SIZE> HTON_PRIMARY_KEY_REQUIRED_FOR_DELETE(1 << HTON_BIT_PRIMARY_KEY_REQUIRED_FOR_DELETE);
114
static const std::bitset<HTON_BIT_SIZE> HTON_NO_PREFIX_CHAR_KEYS(1 << HTON_BIT_NO_PREFIX_CHAR_KEYS);
115
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_CHECKSUM(1 << HTON_BIT_HAS_CHECKSUM);
116
static const std::bitset<HTON_BIT_SIZE> HTON_SKIP_STORE_LOCK(1 << HTON_BIT_SKIP_STORE_LOCK);
117
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_SCHEMA_DICTIONARY(1 << HTON_BIT_SCHEMA_DICTIONARY);
118
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_FOREIGN_KEYS(1 << HTON_BIT_FOREIGN_KEYS);
69
static const std::bitset<HTON_BIT_SIZE> HTON_HAS_DATA_DICTIONARY(1 << HTON_BIT_HAS_DATA_DICTIONARY);
122
class NamedSavepoint;
127
typedef hash_map<std::string, StorageEngine *> EngineMap;
128
typedef std::vector<StorageEngine *> EngineVector;
130
typedef std::set<std::string> TableNameList;
132
extern const std::string UNKNOWN_STRING;
133
extern const std::string DEFAULT_DEFINITION_FILE_EXT;
72
class TableNameIteratorImplementation;
136
75
StorageEngine is a singleton structure - one instance per storage engine -
137
76
to provide access to storage engine functionality that works on the
138
"global" level (unlike Cursor class that works on a per-table basis)
77
"global" level (unlike handler class that works on a per-table basis)
140
79
usually StorageEngine instance is defined statically in ha_xxx.cc as
142
81
static StorageEngine { ... } xxx_engine;
83
savepoint_*, prepare, recover, and *_by_xid pointers can be 0.
144
class StorageEngine : public Plugin,
145
public MonitoredInTransaction
148
typedef uint64_t Table_flags;
151
const std::bitset<HTON_BIT_SIZE> flags; /* global Cursor flags */
153
static EngineVector &getSchemaEngines();
155
virtual void setTransactionReadWrite(Session& session);
158
* Indicates to a storage engine the start of a
161
virtual void doStartStatement(Session *session)
167
* Indicates to a storage engine the end of
168
* the current SQL statement in the supplied
171
virtual void doEndStatement(Session *session)
177
std::string table_definition_ext;
180
const std::string& getTableDefinitionFileExtension()
182
return table_definition_ext;
88
Name used for storage engine.
90
const std::string name;
91
const bool two_phase_commit;
94
const std::bitset<HTON_BIT_SIZE> flags; /* global handler flags */
96
to store per-savepoint data storage engine is provided with an area
97
of a requested size (0 is ok here).
98
savepoint_offset must be initialized statically to the size of
99
the needed memory to store per-savepoint information.
100
After xxx_init it is changed to be an offset to savepoint storage
101
area and need not be used by storage engine.
102
see binlog_engine and binlog_savepoint_set/rollback for an example.
104
size_t savepoint_offset;
105
size_t orig_savepoint_offset;
186
106
std::vector<std::string> aliases;
108
void setTransactionReadWrite(Session* session);
113
* Implementing classes should override these to provide savepoint
116
virtual int savepoint_set_hook(Session *, void *) { return 0; }
118
virtual int savepoint_rollback_hook(Session *, void *) { return 0; }
120
virtual int savepoint_release_hook(Session *, void *) { return 0; }
189
const std::vector<std::string>& getAliases() const
124
StorageEngine(const std::string name_arg,
125
const std::bitset<HTON_BIT_SIZE> &flags_arg= HTON_NO_FLAGS,
126
size_t savepoint_offset_arg= 0,
127
bool support_2pc= false);
129
virtual ~StorageEngine();
131
static int getTableProto(const char* path,
132
drizzled::message::Table *table_proto);
134
virtual int getTableProtoImplementation(const char* path,
135
drizzled::message::Table *table_proto)
143
each storage engine has it's own memory area (actually a pointer)
144
in the session, for storing per-connection information.
147
session->ha_data[xxx_engine.slot]
149
slot number is initialized by MySQL after xxx_init() is called.
153
inline uint32_t getSlot (void) { return slot; }
154
inline void setSlot (uint32_t value) { slot= value; }
156
const std::vector<std::string>& getAliases()
242
184
return flags.test(flag);
245
// @todo match check_flag interface
246
virtual uint32_t index_flags(enum ha_key_alg) const { return 0; }
247
virtual void startStatement(Session *session)
249
doStartStatement(session);
251
virtual void endStatement(Session *session)
253
doEndStatement(session);
187
void enable() { enabled= true; }
188
void disable() { enabled= false; }
190
std::string getName() const { return name; }
257
* Called during Session::cleanup() for all engines
193
StorageEngine methods:
195
close_connection is only called if
196
session->ha_data[xxx_engine.slot] is non-zero, so even if you don't need
197
this storage area - set it to something, so that MySQL would know
198
this storage engine was accessed in this connection
259
200
virtual int close_connection(Session *)
263
virtual Cursor *create(TableShare &, memory::Root *)= 0;
205
'all' is true if it's a real commit, that makes persistent changes
206
'all' is false if it's not in fact a commit but an end of the
207
statement that is part of the transaction.
208
NOTE 'all' is also false in auto-commit mode where 'end of statement'
209
and 'real commit' mean the same event.
211
virtual int commit(Session *, bool)
216
virtual int rollback(Session *, bool)
222
The void * points to an uninitialized storage area of requested size
223
(see savepoint_offset description)
225
int savepoint_set(Session *session, void *sp)
227
return savepoint_set_hook(session, (unsigned char *)sp+savepoint_offset);
231
The void * points to a storage area, that was earlier passed
232
to the savepoint_set call
234
int savepoint_rollback(Session *session, void *sp)
236
return savepoint_rollback_hook(session,
237
(unsigned char *)sp+savepoint_offset);
240
int savepoint_release(Session *session, void *sp)
242
return savepoint_release_hook(session,
243
(unsigned char *)sp+savepoint_offset);
246
virtual int prepare(Session *, bool) { return 0; }
247
virtual int recover(XID *, uint32_t) { return 0; }
248
virtual int commit_by_xid(XID *) { return 0; }
249
virtual int rollback_by_xid(XID *) { return 0; }
250
virtual handler *create(TableShare *, MEM_ROOT *)= 0;
252
virtual void drop_database(char*) { }
253
virtual int start_consistent_snapshot(Session *) { return 0; }
265
254
virtual bool flush_logs() { return false; }
266
255
virtual bool show_status(Session *, stat_print_fn *, enum ha_stat_type)
260
/* args: current_session, tables, cond */
261
virtual int fill_files_table(Session *, TableList *,
262
Item *) { return 0; }
263
virtual int release_temporary_latches(Session *) { return false; }
272
266
If frm_error() is called then we will use this to find out what file
273
267
extentions exist for the storage engine. This is also used by the default
274
rename_table and delete_table method in Cursor.cc.
268
rename_table and delete_table method in handler.cc.
276
270
For engines that have two file name extentions (separate meta/index file
277
271
and data file), the order of elements is relevant. First element of engine
282
276
virtual const char **bas_ext() const =0;
285
virtual int doCreateTable(Session &session,
287
TableIdentifier &identifier,
288
message::Table &message)= 0;
290
virtual int doRenameTable(Session &session,
291
TableIdentifier &from, TableIdentifier &to)= 0;
295
int renameTable(Session &session, TableIdentifier &from, TableIdentifier &to);
297
// @todo move these to protected
298
virtual void doGetTableNames(CachedDirectory &directory,
299
drizzled::SchemaIdentifier &schema_identifier,
300
TableNameList &set_of_names)= 0;
302
virtual void doGetTableIdentifiers(CachedDirectory &directory,
303
drizzled::SchemaIdentifier &schema_identifier,
304
TableIdentifiers &set_of_identifiers)= 0;
306
virtual int doDropTable(Session &session,
307
TableIdentifier &identifier)= 0;
309
/* Class Methods for operating on plugin */
310
static bool addPlugin(plugin::StorageEngine *engine);
311
static void removePlugin(plugin::StorageEngine *engine);
313
static int getTableDefinition(Session& session,
314
TableIdentifier &identifier,
315
message::Table &table_proto,
316
bool include_temporary_tables= true);
317
static bool doesTableExist(Session &session,
318
TableIdentifier &identifier,
319
bool include_temporary_tables= true);
321
virtual bool doDoesTableExist(Session& session, TableIdentifier &identifier);
323
static plugin::StorageEngine *findByName(const std::string &find_str);
324
static plugin::StorageEngine *findByName(Session& session, const std::string &find_str);
326
static void closeConnection(Session* session);
327
static void dropDatabase(char* path);
328
static bool flushLogs(plugin::StorageEngine *db_type);
329
static int dropTable(Session& session,
330
TableIdentifier &identifier);
331
static int dropTable(Session& session,
332
StorageEngine &engine,
333
TableIdentifier &identifier);
334
static void getTableNames(Session &session, drizzled::SchemaIdentifier& schema_identifier, TableNameList &set_of_names);
335
static void getTableIdentifiers(Session &session, SchemaIdentifier &schema_identifier, TableIdentifiers &set_of_identifiers);
337
// Check to see if any SE objects to creation.
338
static bool canCreateTable(drizzled::TableIdentifier &identifier);
339
virtual bool doCanCreateTable(drizzled::TableIdentifier &identifier)
340
{ (void)identifier; return true; }
342
// @note All schema methods defined here
343
static void getSchemaIdentifiers(Session &session, SchemaIdentifierList &schemas);
344
static bool getSchemaDefinition(TableIdentifier &identifier, message::Schema &proto);
345
static bool getSchemaDefinition(drizzled::SchemaIdentifier &identifier, message::Schema &proto);
346
static bool doesSchemaExist(drizzled::SchemaIdentifier &identifier);
347
static const CHARSET_INFO *getSchemaCollation(drizzled::SchemaIdentifier &identifier);
348
static bool createSchema(const drizzled::message::Schema &schema_message);
349
static bool dropSchema(drizzled::SchemaIdentifier &identifier);
350
static bool alterSchema(const drizzled::message::Schema &schema_message);
352
// @note make private/protected
353
virtual void doGetSchemaIdentifiers(SchemaIdentifierList&)
356
virtual bool doGetSchemaDefinition(drizzled::SchemaIdentifier&, drizzled::message::Schema&)
361
virtual bool doCreateSchema(const drizzled::message::Schema&)
364
virtual bool doAlterSchema(const drizzled::message::Schema&)
367
virtual bool doDropSchema(drizzled::SchemaIdentifier&)
370
static inline const std::string &resolveName(const StorageEngine *engine)
372
return engine == NULL ? UNKNOWN_STRING : engine->getName();
375
static int createTable(Session& session,
376
TableIdentifier &identifier,
377
bool update_create_info,
378
message::Table& table_proto);
380
static void removeLostTemporaryTables(Session &session, const char *directory);
382
Cursor *getCursor(TableShare &share, memory::Root *alloc);
384
uint32_t max_record_length() const
385
{ return std::min((unsigned int)HA_MAX_REC_LENGTH, max_supported_record_length()); }
386
uint32_t max_keys() const
387
{ return std::min((unsigned int)MAX_KEY, max_supported_keys()); }
388
uint32_t max_key_parts() const
389
{ return std::min((unsigned int)MAX_REF_PARTS, max_supported_key_parts()); }
390
uint32_t max_key_length() const
391
{ return std::min((unsigned int)MAX_KEY_LENGTH, max_supported_key_length()); }
392
uint32_t max_key_part_length(void) const
393
{ return std::min((unsigned int)MAX_KEY_LENGTH, max_supported_key_part_length()); }
395
virtual uint32_t max_supported_record_length(void) const
396
{ return HA_MAX_REC_LENGTH; }
397
virtual uint32_t max_supported_keys(void) const { return 0; }
398
virtual uint32_t max_supported_key_parts(void) const { return MAX_REF_PARTS; }
399
virtual uint32_t max_supported_key_length(void) const { return MAX_KEY_LENGTH; }
400
virtual uint32_t max_supported_key_part_length(void) const { return 255; }
402
/* TODO-> Make private */
403
static int deleteDefinitionFromPath(TableIdentifier &identifier);
404
static int renameDefinitionFromPath(TableIdentifier &dest, TableIdentifier &src);
405
static int writeDefinitionFromPath(TableIdentifier &identifier, message::Table &proto);
406
static bool readTableFile(const std::string &path, message::Table &table_message);
410
* The below are simple virtual overrides for the plugin::MonitoredInTransaction
413
virtual bool participatesInSqlTransaction() const
415
return false; /* plugin::StorageEngine is non-transactional in terms of SQL */
417
virtual bool participatesInXaTransaction() const
419
return false; /* plugin::StorageEngine is non-transactional in terms of XA */
421
virtual bool alwaysRegisterForXaTransaction() const
427
} /* namespace plugin */
428
} /* namespace drizzled */
279
virtual int createTableImplementation(Session *session,
280
const char *table_name,
282
HA_CREATE_INFO *create_info,
283
drizzled::message::Table* proto)= 0;
285
virtual int renameTableImplementation(Session* session,
286
const char *from, const char *to);
288
virtual int deleteTableImplementation(Session* session,
289
const std::string table_path);
292
int createTable(Session *session, const char *path, Table *table_arg,
293
HA_CREATE_INFO *create_info,
294
drizzled::message::Table *proto)
296
char name_buff[FN_REFLEN];
297
const char *table_name;
299
table_name= checkLowercaseNames(path, name_buff);
301
setTransactionReadWrite(session);
303
return createTableImplementation(session, table_name, table_arg,
307
int renameTable(Session *session, const char *from, const char *to)
309
setTransactionReadWrite(session);
311
return renameTableImplementation(session, from, to);
314
int deleteTable(Session* session, const std::string table_path)
316
setTransactionReadWrite(session);
318
return deleteTableImplementation(session, table_path);
321
const char *checkLowercaseNames(const char *path, char *tmp_path);
323
virtual TableNameIteratorImplementation* tableNameIterator(const std::string &database)
330
class TableNameIteratorImplementation
335
TableNameIteratorImplementation(const std::string &database) : db(database)
337
virtual ~TableNameIteratorImplementation() {};
339
virtual int next(std::string *name)= 0;
343
class TableNameIterator
346
drizzled::Registry<StorageEngine *>::iterator engine_iter;
347
TableNameIteratorImplementation *current_implementation;
348
TableNameIteratorImplementation *default_implementation;
349
std::string database;
351
TableNameIterator(const std::string &db);
352
~TableNameIterator();
354
int next(std::string *name);
358
StorageEngine *ha_default_storage_engine(Session *session);
359
StorageEngine *ha_resolve_by_name(Session *session, std::string find_str);
361
handler *get_new_handler(TableShare *share, MEM_ROOT *alloc,
362
StorageEngine *db_type);
363
const std::string ha_resolve_storage_engine_name(const StorageEngine *db_type);
430
365
#endif /* DRIZZLED_PLUGIN_STORAGE_ENGINE_H */