75
63
#define CSM_EXT ".CSM" // Meta file
78
static int read_meta_file(int meta_file, ha_rows *rows);
79
static int write_meta_file(int meta_file, ha_rows rows, bool dirty);
66
static TINA_SHARE *get_share(const char *table_name, Table *table);
67
static int free_share(TINA_SHARE *share);
68
static int read_meta_file(File meta_file, ha_rows *rows);
69
static int write_meta_file(File meta_file, ha_rows rows, bool dirty);
71
extern "C" void tina_get_status(void* param, int concurrent_insert);
72
extern "C" void tina_update_status(void* param);
73
extern "C" bool tina_check_status(void* param);
81
75
/* Stuff for shares */
82
76
pthread_mutex_t tina_mutex;
77
static HASH tina_open_tables;
78
static handler *tina_create_handler(handlerton *hton,
84
83
/*****************************************************************************
86
85
*****************************************************************************/
89
If frm_error() is called in table.cc this is called to find out what file
90
extensions exist for this Cursor.
88
Used for sorting chains with qsort().
92
static const char *ha_tina_exts[] = {
98
class Tina : public drizzled::plugin::StorageEngine
100
typedef std::map<string, TinaShare*> TinaMap;
101
TinaMap tina_open_tables;
103
Tina(const string& name_arg)
104
: drizzled::plugin::StorageEngine(name_arg,
105
HTON_TEMPORARY_ONLY |
106
HTON_NO_AUTO_INCREMENT |
107
HTON_SKIP_STORE_LOCK),
112
pthread_mutex_destroy(&tina_mutex);
115
virtual Cursor *create(TableShare &table)
117
return new ha_tina(*this, table);
120
const char **bas_ext() const {
124
int doCreateTable(Session &,
126
const drizzled::TableIdentifier &identifier,
127
drizzled::message::Table&);
129
int doGetTableDefinition(Session& session,
130
const drizzled::TableIdentifier &identifier,
131
drizzled::message::Table &table_message);
133
int doDropTable(Session&, const drizzled::TableIdentifier &identifier);
134
TinaShare *findOpenTable(const string table_name);
135
void addOpenTable(const string &table_name, TinaShare *);
136
void deleteOpenTable(const string &table_name);
139
uint32_t max_keys() const { return 0; }
140
uint32_t max_key_parts() const { return 0; }
141
uint32_t max_key_length() const { return 0; }
142
bool doDoesTableExist(Session& session, const drizzled::TableIdentifier &identifier);
143
int doRenameTable(Session&, const drizzled::TableIdentifier &from, const drizzled::TableIdentifier &to);
145
void doGetTableIdentifiers(drizzled::CachedDirectory &directory,
146
const drizzled::SchemaIdentifier &schema_identifier,
147
drizzled::TableIdentifiers &set_of_identifiers);
150
void Tina::doGetTableIdentifiers(drizzled::CachedDirectory&,
151
const drizzled::SchemaIdentifier&,
152
drizzled::TableIdentifiers&)
156
int Tina::doRenameTable(Session &session,
157
const drizzled::TableIdentifier &from, const drizzled::TableIdentifier &to)
160
for (const char **ext= bas_ext(); *ext ; ext++)
162
if (rename_file_ext(from.getPath().c_str(), to.getPath().c_str(), *ext))
164
if ((error=errno) != ENOENT)
170
session.renameTableMessage(from, to);
175
bool Tina::doDoesTableExist(Session &session, const drizzled::TableIdentifier &identifier)
177
return session.doesTableMessageExist(identifier);
181
int Tina::doDropTable(Session &session,
182
const drizzled::TableIdentifier &identifier)
185
int enoent_or_zero= ENOENT; // Error if no file was deleted
187
for (const char **ext= bas_ext(); *ext ; ext++)
189
std::string full_name= identifier.getPath();
190
full_name.append(*ext);
192
if (internal::my_delete_with_symlink(full_name.c_str(), MYF(0)))
194
if ((error= errno) != ENOENT)
199
enoent_or_zero= 0; // No error for ENOENT
201
error= enoent_or_zero;
204
session.removeTableMessage(identifier);
209
TinaShare *Tina::findOpenTable(const string table_name)
211
TinaMap::iterator find_iter=
212
tina_open_tables.find(table_name);
214
if (find_iter != tina_open_tables.end())
215
return (*find_iter).second;
220
void Tina::addOpenTable(const string &table_name, TinaShare *share)
222
tina_open_tables[table_name]= share;
225
void Tina::deleteOpenTable(const string &table_name)
227
tina_open_tables.erase(table_name);
231
int Tina::doGetTableDefinition(Session &session,
232
const drizzled::TableIdentifier &identifier,
233
drizzled::message::Table &table_message)
235
if (session.getTableMessage(identifier, table_message))
242
static Tina *tina_engine= NULL;
244
static int tina_init_func(drizzled::module::Context &context)
247
tina_engine= new Tina("CSV");
248
context.add(tina_engine);
90
int sort_set (tina_set *a, tina_set *b)
93
We assume that intervals do not intersect. So, it is enought to compare
94
any two points. Here we take start of intervals for comparison.
96
return ( a->begin > b->begin ? 1 : ( a->begin < b->begin ? -1 : 0 ) );
99
static unsigned char* tina_get_key(TINA_SHARE *share, size_t *length,
100
bool not_used __attribute__((unused)))
102
*length=share->table_name_length;
103
return (unsigned char*) share->table_name;
106
static int tina_init_func(void *p)
108
handlerton *tina_hton;
110
tina_hton= (handlerton *)p;
250
111
pthread_mutex_init(&tina_mutex,MY_MUTEX_INIT_FAST);
256
TinaShare::TinaShare(const std::string &table_name_arg) :
257
table_name(table_name_arg),
258
data_file_name(table_name_arg),
260
saved_data_file_length(0),
261
update_file_opened(false),
262
tina_write_opened(false),
267
data_file_name.append(CSV_EXT);
270
TinaShare::~TinaShare()
272
pthread_mutex_destroy(&mutex);
112
(void) hash_init(&tina_open_tables,system_charset_info,32,0,0,
113
(hash_get_key) tina_get_key,0,0);
114
tina_hton->state= SHOW_OPTION_YES;
115
tina_hton->create= tina_create_handler;
116
tina_hton->flags= (HTON_CAN_RECREATE | HTON_SUPPORT_LOG_TABLES |
121
static int tina_done_func(void *p __attribute__((unused)))
123
hash_free(&tina_open_tables);
124
pthread_mutex_destroy(&tina_mutex);
276
131
Simple lock controls.
278
TinaShare *ha_tina::get_share(const std::string &table_name)
133
static TINA_SHARE *get_share(const char *table_name,
134
Table *table __attribute__((unused)))
280
pthread_mutex_lock(&tina_mutex);
282
Tina *a_tina= static_cast<Tina *>(engine);
283
share= a_tina->findOpenTable(table_name);
285
std::string meta_file_name;
137
char meta_file_name[FN_REFLEN];
286
138
struct stat file_stat;
142
pthread_mutex_lock(&tina_mutex);
143
length=(uint) strlen(table_name);
289
146
If share is not present in the hash, create a new share and
290
147
initialize its members.
149
if (!(share=(TINA_SHARE*) hash_search(&tina_open_tables,
150
(unsigned char*) table_name,
294
share= new TinaShare(table_name);
298
pthread_mutex_unlock(&tina_mutex);
302
meta_file_name.assign(table_name);
303
meta_file_name.append(CSM_EXT);
305
if (stat(share->data_file_name.c_str(), &file_stat))
307
pthread_mutex_unlock(&tina_mutex);
153
if (!my_multi_malloc(MYF(MY_WME | MY_ZEROFILL),
154
&share, sizeof(*share),
158
pthread_mutex_unlock(&tina_mutex);
163
share->table_name_length= length;
164
share->table_name= tmp_name;
165
share->crashed= false;
166
share->rows_recorded= 0;
167
share->update_file_opened= false;
168
share->tina_write_opened= false;
169
share->data_file_version= 0;
170
my_stpcpy(share->table_name, table_name);
171
fn_format(share->data_file_name, table_name, "", CSV_EXT,
172
MY_REPLACE_EXT|MY_UNPACK_FILENAME);
173
fn_format(meta_file_name, table_name, "", CSM_EXT,
174
MY_REPLACE_EXT|MY_UNPACK_FILENAME);
176
if (stat(share->data_file_name, &file_stat))
312
178
share->saved_data_file_length= file_stat.st_size;
314
a_tina->addOpenTable(share->table_name, share);
180
if (my_hash_insert(&tina_open_tables, (unsigned char*) share))
182
thr_lock_init(&share->lock);
316
183
pthread_mutex_init(&share->mutex,MY_MUTEX_INIT_FAST);
699
If frm_error() is called in table.cc this is called to find out what file
700
extensions exist for this handler.
702
static const char *ha_tina_exts[] = {
708
const char **ha_tina::bas_ext() const
714
Three functions below are needed to enable concurrent insert functionality
715
for CSV engine. For more details see mysys/thr_lock.c
718
void tina_get_status(void* param,
719
int concurrent_insert __attribute__((unused)))
721
ha_tina *tina= (ha_tina*) param;
725
void tina_update_status(void* param)
727
ha_tina *tina= (ha_tina*) param;
728
tina->update_status();
731
/* this should exist and return 0 for concurrent insert to work */
732
bool tina_check_status(void* param __attribute__((unused)))
738
Save the state of the table
744
This function is used to retrieve the file length. During the lock
745
phase of concurrent insert. For more details see comment to
746
ha_tina::update_status below.
749
void ha_tina::get_status()
751
local_saved_data_file_length= share->saved_data_file_length;
756
Correct the state of the table. Called by unlock routines
757
before the write lock is released.
763
When we employ concurrent insert lock, we save current length of the file
764
during the lock phase. We do not read further saved value, as we don't
765
want to interfere with undergoing concurrent insert. Writers update file
766
length info during unlock with update_status().
769
For log tables concurrent insert works different. The reason is that
770
log tables are always opened and locked. And as they do not unlock
771
tables, the file length after writes should be updated in a different
775
void ha_tina::update_status()
777
/* correct local_saved_data_file_length for writers */
778
share->saved_data_file_length= local_saved_data_file_length;
780
783
Open a database file. Keep in mind that tables are caches, so
781
784
this will not be called for every request. Any sort of positions
782
785
that need to be reset should be kept in the ::extra() call.
784
int ha_tina::doOpen(const TableIdentifier &identifier, int , uint32_t )
787
int ha_tina::open(const char *name, int mode __attribute__((unused)),
788
uint32_t open_options)
786
if (not (share= get_share(identifier.getPath().c_str())))
790
if (!(share= get_share(name, table)))
791
return(HA_ERR_OUT_OF_MEM);
793
if (share->crashed && !(open_options & HA_OPEN_FOR_REPAIR))
792
796
return(HA_ERR_CRASHED_ON_USAGE);
795
799
local_data_file_version= share->data_file_version;
796
if ((data_file= internal::my_open(share->data_file_name.c_str(), O_RDONLY, MYF(0))) == -1)
800
if ((data_file= my_open(share->data_file_name, O_RDONLY, MYF(0))) == -1)
800
Init locking. Pass Cursor object to the locking routines,
804
Init locking. Pass handler object to the locking routines,
801
805
so that they could save/update local_saved_data_file_length value
802
806
during locking. This is needed to enable concurrent inserts.
808
thr_lock_data_init(&share->lock, &lock, (void*) this);
804
809
ref_length=sizeof(off_t);
811
share->lock.get_status= tina_get_status;
812
share->lock.update_status= tina_update_status;
813
share->lock.check_status= tina_check_status;
810
820
Close a database file. We remove ourselves from the shared strucutre.
811
821
If it is empty we destroy it.
1231
internal::my_close(update_temp_file, MYF(0));
1253
my_close(update_temp_file, MYF(0));
1232
1254
share->update_file_opened= false;
1260
Repair CSV table in the case, it is crashed.
1264
session The thread, performing repair
1265
check_opt The options for repair. We do not use it currently.
1268
If the file is empty, change # of rows in the file and complete recovery.
1269
Otherwise, scan the table looking for bad rows. If none were found,
1270
we mark file as a good one and return. If a bad row was encountered,
1271
we truncate the datafile up to the last good row.
1273
TODO: Make repair more clever - it should try to recover subsequent
1274
rows (after the first bad one) as well.
1277
int ha_tina::repair(Session* session,
1278
HA_CHECK_OPT* check_opt __attribute__((unused)))
1280
char repaired_fname[FN_REFLEN];
1284
ha_rows rows_repaired= 0;
1285
off_t write_begin= 0, write_end;
1288
if (!share->saved_data_file_length)
1290
share->rows_recorded= 0;
1294
/* Don't assert in field::val() functions */
1295
table->use_all_columns();
1296
if (!(buf= (unsigned char*) my_malloc(table->s->reclength, MYF(MY_WME))))
1297
return(HA_ERR_OUT_OF_MEM);
1299
/* position buffer to the start of the file */
1300
if (init_data_file())
1301
return(HA_ERR_CRASHED_ON_REPAIR);
1304
Local_saved_data_file_length is initialized during the lock phase.
1305
Sometimes this is not getting executed before ::repair (e.g. for
1306
the log tables). We set it manually here.
1308
local_saved_data_file_length= share->saved_data_file_length;
1309
/* set current position to the beginning of the file */
1310
current_position= next_position= 0;
1312
init_alloc_root(&blobroot, BLOB_MEMROOT_ALLOC_SIZE, 0);
1314
/* Read the file row-by-row. If everything is ok, repair is not needed. */
1315
while (!(rc= find_current_row(buf)))
1317
session_inc_row_count(session);
1319
current_position= next_position;
1322
free_root(&blobroot, MYF(0));
1326
if (rc == HA_ERR_END_OF_FILE)
1329
All rows were read ok until end of file, the file does not need repair.
1330
If rows_recorded != rows_repaired, we should update rows_recorded value
1331
to the current amount of rows.
1333
share->rows_recorded= rows_repaired;
1338
Otherwise we've encountered a bad row => repair is needed.
1339
Let us create a temporary file.
1341
if ((repair_file= my_create(fn_format(repaired_fname, share->table_name,
1343
MY_REPLACE_EXT|MY_UNPACK_FILENAME),
1344
0, O_RDWR | O_TRUNC,MYF(MY_WME))) < 0)
1345
return(HA_ERR_CRASHED_ON_REPAIR);
1347
file_buff->init_buff(data_file);
1350
/* we just truncated the file up to the first bad row. update rows count. */
1351
share->rows_recorded= rows_repaired;
1353
/* write repaired file */
1356
write_end= std::min(file_buff->end(), current_position);
1357
if ((write_end - write_begin) &&
1358
(my_write(repair_file, (unsigned char*)file_buff->ptr(),
1359
write_end - write_begin, MYF_RW)))
1362
write_begin= write_end;
1363
if (write_end== current_position)
1366
file_buff->read_next(); /* shift the buffer */
1370
Close the files and rename repaired file to the datafile.
1371
We have to close the files, as on Windows one cannot rename
1372
a file, which descriptor is still open. EACCES will be returned
1373
when trying to delete the "to"-file in my_rename().
1375
if (my_close(data_file,MYF(0)) || my_close(repair_file, MYF(0)) ||
1376
my_rename(repaired_fname, share->data_file_name, MYF(0)))
1379
/* Open the file again, it should now be repaired */
1380
if ((data_file= my_open(share->data_file_name, O_RDWR|O_APPEND,
1384
/* Set new file size. The file size will be updated by ::update_status() */
1385
local_saved_data_file_length= (size_t) current_position;
1388
share->crashed= false;
1389
return(HA_ADMIN_OK);
1238
1393
DELETE without WHERE calls this
1420
Called by the database to lock the table. Keep in mind that this
1421
is an internal lock.
1423
THR_LOCK_DATA **ha_tina::store_lock(Session *session __attribute__((unused)),
1425
enum thr_lock_type lock_type)
1427
if (lock_type != TL_IGNORE && lock.type == TL_UNLOCK)
1428
lock.type=lock_type;
1265
1434
Create a table. You do not want to leave the table open after a call to
1266
1435
this (the database will call ::open() if it needs to).
1269
int Tina::doCreateTable(Session &session,
1271
const drizzled::TableIdentifier &identifier,
1272
drizzled::message::Table &create_proto)
1438
int ha_tina::create(const char *name, Table *table_arg,
1439
HA_CREATE_INFO *create_info __attribute__((unused)))
1274
1441
char name_buff[FN_REFLEN];
1280
const drizzled::TableShare::Fields fields(table_arg.getShare()->getFields());
1281
for (drizzled::TableShare::Fields::const_iterator iter= fields.begin();
1282
iter != fields.end();
1447
for (Field **field= table_arg->s->field; *field; field++)
1285
if (not *iter) // Historical legacy for NULL array end.
1288
if ((*iter)->real_maybe_null())
1449
if ((*field)->real_maybe_null())
1290
1451
my_error(ER_CHECK_NOT_IMPLEMENTED, MYF(0), "nullable columns");
1291
1452
return(HA_ERR_UNSUPPORTED);
1296
if ((create_file= internal::my_create(internal::fn_format(name_buff, identifier.getPath().c_str(), "", CSM_EXT,
1297
MY_REPLACE_EXT|MY_UNPACK_FILENAME), 0,
1298
O_RDWR | O_TRUNC,MYF(MY_WME))) < 0)
1457
if ((create_file= my_create(fn_format(name_buff, name, "", CSM_EXT,
1458
MY_REPLACE_EXT|MY_UNPACK_FILENAME), 0,
1459
O_RDWR | O_TRUNC,MYF(MY_WME))) < 0)
1301
1462
write_meta_file(create_file, 0, false);
1302
internal::my_close(create_file, MYF(0));
1463
my_close(create_file, MYF(0));
1304
if ((create_file= internal::my_create(internal::fn_format(name_buff, identifier.getPath().c_str(), "", CSV_EXT,
1305
MY_REPLACE_EXT|MY_UNPACK_FILENAME),0,
1306
O_RDWR | O_TRUNC,MYF(MY_WME))) < 0)
1465
if ((create_file= my_create(fn_format(name_buff, name, "", CSV_EXT,
1466
MY_REPLACE_EXT|MY_UNPACK_FILENAME),0,
1467
O_RDWR | O_TRUNC,MYF(MY_WME))) < 0)
1309
internal::my_close(create_file, MYF(0));
1311
session.storeTableMessage(identifier, create_proto);
1317
DRIZZLE_DECLARE_PLUGIN
1470
my_close(create_file, MYF(0));
1475
int ha_tina::check(Session* session,
1476
HA_CHECK_OPT* check_opt __attribute__((unused)))
1480
const char *old_proc_info;
1481
ha_rows count= share->rows_recorded;
1483
old_proc_info= get_session_proc_info(session);
1484
set_session_proc_info(session, "Checking table");
1485
if (!(buf= (unsigned char*) my_malloc(table->s->reclength, MYF(MY_WME))))
1486
return(HA_ERR_OUT_OF_MEM);
1488
/* position buffer to the start of the file */
1489
if (init_data_file())
1490
return(HA_ERR_CRASHED);
1493
Local_saved_data_file_length is initialized during the lock phase.
1494
Check does not use store_lock in certain cases. So, we set it
1497
local_saved_data_file_length= share->saved_data_file_length;
1498
/* set current position to the beginning of the file */
1499
current_position= next_position= 0;
1501
init_alloc_root(&blobroot, BLOB_MEMROOT_ALLOC_SIZE, 0);
1503
/* Read the file row-by-row. If everything is ok, repair is not needed. */
1504
while (!(rc= find_current_row(buf)))
1506
session_inc_row_count(session);
1508
current_position= next_position;
1511
free_root(&blobroot, MYF(0));
1514
set_session_proc_info(session, old_proc_info);
1516
if ((rc != HA_ERR_END_OF_FILE) || count)
1518
share->crashed= true;
1519
return(HA_ADMIN_CORRUPT);
1522
return(HA_ADMIN_OK);
1526
bool ha_tina::check_if_incompatible_data(HA_CREATE_INFO *info __attribute__((unused)),
1527
uint32_t table_changes __attribute__((unused)))
1529
return COMPATIBLE_DATA_YES;
1532
mysql_declare_plugin(csv)
1534
DRIZZLE_STORAGE_ENGINE_PLUGIN,
1322
1537
"Brian Aker, MySQL AB",
1323
1538
"CSV storage engine",
1324
1539
PLUGIN_LICENSE_GPL,
1325
1540
tina_init_func, /* Plugin Init */
1541
tina_done_func, /* Plugin Deinit */
1542
NULL, /* status variables */
1326
1543
NULL, /* system variables */
1327
1544
NULL /* config options */
1329
DRIZZLE_DECLARE_PLUGIN_END;
1546
mysql_declare_plugin_end;