mariadb/sql/ha_myisammrg.cc

453 lines
13 KiB
C++
Raw Normal View History

2000-07-31 21:29:14 +02:00
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
2000-07-31 21:29:14 +02:00
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
2000-07-31 21:29:14 +02:00
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2000-07-31 21:29:14 +02:00
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 */
#ifdef __GNUC__
#pragma implementation // gcc: Class implementation
#endif
#include "mysql_priv.h"
#include <m_ctype.h>
#include "ha_myisammrg.h"
#ifndef MASTER
#include "../srclib/myisammrg/myrg_def.h"
2000-07-31 21:29:14 +02:00
#else
#include "../myisammrg/myrg_def.h"
2000-07-31 21:29:14 +02:00
#endif
/*****************************************************************************
** MyISAM MERGE tables
*****************************************************************************/
const char **ha_myisammrg::bas_ext() const
{ static const char *ext[]= { ".MRG", NullS }; return ext; }
int ha_myisammrg::open(const char *name, int mode, uint test_if_locked)
2000-07-31 21:29:14 +02:00
{
char name_buff[FN_REFLEN];
2001-12-02 14:34:01 +02:00
DBUG_PRINT("info", ("ha_myisammrg::open"));
2000-07-31 21:29:14 +02:00
if (!(file=myrg_open(fn_format(name_buff,name,"","",2 | 4), mode,
test_if_locked)))
2001-12-02 14:34:01 +02:00
{
DBUG_PRINT("info", ("ha_myisammrg::open exit %d", my_errno));
2002-11-11 22:43:07 +01:00
return (my_errno ? my_errno : -1);
2001-12-02 14:34:01 +02:00
}
DBUG_PRINT("info", ("ha_myisammrg::open myrg_extrafunc..."))
2002-03-22 22:55:08 +02:00
myrg_extrafunc(file, query_cache_invalidate_by_MyISAM_filename_ref);
2000-07-31 21:29:14 +02:00
if (!(test_if_locked == HA_OPEN_WAIT_IF_LOCKED ||
test_if_locked == HA_OPEN_ABORT_IF_LOCKED))
myrg_extra(file,HA_EXTRA_NO_WAIT_LOCK,0);
2000-07-31 21:29:14 +02:00
info(HA_STATUS_NO_LOCK | HA_STATUS_VARIABLE | HA_STATUS_CONST);
if (!(test_if_locked & HA_OPEN_WAIT_IF_LOCKED))
myrg_extra(file,HA_EXTRA_WAIT_LOCK,0);
2000-09-21 01:58:38 +03:00
if (table->reclength != mean_rec_length && mean_rec_length)
2000-07-31 21:29:14 +02:00
{
DBUG_PRINT("error",("reclength: %d mean_rec_length: %d",
table->reclength, mean_rec_length));
goto err;
2000-07-31 21:29:14 +02:00
}
#if !defined(BIG_TABLES) || SIZEOF_OFF_T == 4
/* Merge table has more than 2G rows */
if (table->crashed)
goto err;
#endif
2002-11-11 22:43:07 +01:00
return (0);
err:
myrg_close(file);
file=0;
2003-01-28 17:42:08 +01:00
return (my_errno= HA_ERR_WRONG_MRG_TABLE_DEF);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::close(void)
{
2002-11-11 22:43:07 +01:00
return myrg_close(file);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::write_row(byte * buf)
{
statistic_increment(ha_write_count,&LOCK_status);
if (table->time_stamp)
update_timestamp(buf+table->time_stamp-1);
if (table->next_number_field && buf == table->record[0])
update_auto_increment();
2002-11-11 22:43:07 +01:00
return myrg_write(file,buf);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::update_row(const byte * old_data, byte * new_data)
{
statistic_increment(ha_update_count,&LOCK_status);
if (table->time_stamp)
update_timestamp(new_data+table->time_stamp-1);
2002-11-11 22:43:07 +01:00
return myrg_update(file,old_data,new_data);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::delete_row(const byte * buf)
{
statistic_increment(ha_delete_count,&LOCK_status);
2002-11-11 22:43:07 +01:00
return myrg_delete(file,buf);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::index_read(byte * buf, const byte * key,
uint key_len, enum ha_rkey_function find_flag)
{
statistic_increment(ha_read_key_count,&LOCK_status);
int error=myrg_rkey(file,buf,active_index, key, key_len, find_flag);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::index_read_idx(byte * buf, uint index, const byte * key,
uint key_len, enum ha_rkey_function find_flag)
{
statistic_increment(ha_read_key_count,&LOCK_status);
int error=myrg_rkey(file,buf,index, key, key_len, find_flag);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::index_read_last(byte * buf, const byte * key, uint key_len)
{
statistic_increment(ha_read_key_count,&LOCK_status);
int error=myrg_rkey(file,buf,active_index, key, key_len,
HA_READ_PREFIX_LAST);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
}
2000-07-31 21:29:14 +02:00
int ha_myisammrg::index_next(byte * buf)
{
statistic_increment(ha_read_next_count,&LOCK_status);
int error=myrg_rnext(file,buf,active_index);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::index_prev(byte * buf)
{
statistic_increment(ha_read_prev_count,&LOCK_status);
int error=myrg_rprev(file,buf, active_index);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
2000-07-31 21:29:14 +02:00
int ha_myisammrg::index_first(byte * buf)
{
statistic_increment(ha_read_first_count,&LOCK_status);
int error=myrg_rfirst(file, buf, active_index);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::index_last(byte * buf)
{
statistic_increment(ha_read_last_count,&LOCK_status);
int error=myrg_rlast(file, buf, active_index);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
2002-11-25 22:18:44 +01:00
int ha_myisammrg::index_next_same(byte * buf,
const byte *key __attribute__((unused)),
uint length __attribute__((unused)))
{
statistic_increment(ha_read_next_count,&LOCK_status);
int error=myrg_rnext_same(file,buf);
table->status=error ? STATUS_NOT_FOUND: 0;
return error;
}
2000-07-31 21:29:14 +02:00
int ha_myisammrg::rnd_init(bool scan)
{
2002-11-11 22:43:07 +01:00
return myrg_extra(file,HA_EXTRA_RESET,0);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::rnd_next(byte *buf)
{
statistic_increment(ha_read_rnd_next_count,&LOCK_status);
int error=myrg_rrnd(file, buf, HA_OFFSET_ERROR);
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::rnd_pos(byte * buf, byte *pos)
{
statistic_increment(ha_read_rnd_count,&LOCK_status);
int error=myrg_rrnd(file, buf, ha_get_ptr(pos,ref_length));
table->status=error ? STATUS_NOT_FOUND: 0;
2002-11-11 22:43:07 +01:00
return error;
2000-07-31 21:29:14 +02:00
}
void ha_myisammrg::position(const byte *record)
{
ulonglong position= myrg_position(file);
ha_store_ptr(ref, ref_length, (my_off_t) position);
}
2002-11-17 18:54:23 +01:00
ha_rows ha_myisammrg::records_in_range(int inx,
const byte *start_key,uint start_key_len,
enum ha_rkey_function start_search_flag,
const byte *end_key,uint end_key_len,
enum ha_rkey_function end_search_flag)
{
return (ha_rows) myrg_records_in_range(file,
inx,
start_key,start_key_len,
start_search_flag,
end_key,end_key_len,
end_search_flag);
}
2000-07-31 21:29:14 +02:00
void ha_myisammrg::info(uint flag)
{
MYMERGE_INFO info;
(void) myrg_status(file,&info,flag);
/*
The following fails if one has not compiled MySQL with -DBIG_TABLES
and one has more than 2^32 rows in the merge tables.
*/
records = (ha_rows) info.records;
deleted = (ha_rows) info.deleted;
#if !defined(BIG_TABLES) || SIZEOF_OFF_T == 4
if ((info.records >= (ulonglong) 1 << 32) ||
(info.deleted >= (ulonglong) 1 << 32))
table->crashed=1;
#endif
2000-07-31 21:29:14 +02:00
data_file_length=info.data_file_length;
errkey = info.errkey;
2003-10-11 13:06:55 +02:00
table->keys_in_use.set_prefix(table->keys);
2000-07-31 21:29:14 +02:00
table->db_options_in_use = info.options;
table->is_view=1;
2000-07-31 21:29:14 +02:00
mean_rec_length=info.reclength;
block_size=0;
update_time=0;
#if SIZEOF_OFF_T > 4
2000-07-31 21:29:14 +02:00
ref_length=6; // Should be big enough
#else
ref_length=4; // Can't be > than my_off_t
#endif
2002-12-10 00:06:05 +01:00
if (flag & HA_STATUS_CONST)
{
if (table->key_parts && info.rec_per_key)
2002-12-10 00:06:05 +01:00
memcpy((char*) table->key_info[0].rec_per_key,
(char*) info.rec_per_key,
sizeof(table->key_info[0].rec_per_key)*table->key_parts);
}
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::extra(enum ha_extra_function operation)
{
/* As this is just a mapping, we don't have to force the underlying
tables to be closed */
if (operation == HA_EXTRA_FORCE_REOPEN ||
operation == HA_EXTRA_PREPARE_FOR_DELETE)
2002-11-11 22:43:07 +01:00
return 0;
return myrg_extra(file,operation,0);
2000-07-31 21:29:14 +02:00
}
/* To be used with WRITE_CACHE, EXTRA_CACHE and BULK_INSERT_BEGIN */
int ha_myisammrg::extra_opt(enum ha_extra_function operation, ulong cache_size)
{
if ((specialflag & SPECIAL_SAFE_MODE) && operation == HA_EXTRA_WRITE_CACHE)
2002-11-11 22:43:07 +01:00
return 0;
return myrg_extra(file, operation, (void*) &cache_size);
}
2000-07-31 21:29:14 +02:00
int ha_myisammrg::reset(void)
{
2002-11-11 22:43:07 +01:00
return myrg_extra(file,HA_EXTRA_RESET,0);
2000-07-31 21:29:14 +02:00
}
int ha_myisammrg::external_lock(THD *thd, int lock_type)
{
2002-11-11 22:43:07 +01:00
return myrg_lock_database(file,lock_type);
}
2000-07-31 21:29:14 +02:00
uint ha_myisammrg::lock_count(void) const
{
2002-11-11 22:43:07 +01:00
return file->tables;
2000-07-31 21:29:14 +02:00
}
THR_LOCK_DATA **ha_myisammrg::store_lock(THD *thd,
THR_LOCK_DATA **to,
enum thr_lock_type lock_type)
{
MYRG_TABLE *open_table;
2000-07-31 21:29:14 +02:00
for (open_table=file->open_tables ;
open_table != file->end_table ;
open_table++)
2000-07-31 21:29:14 +02:00
{
*(to++)= &open_table->table->lock;
if (lock_type != TL_IGNORE && open_table->table->lock.type == TL_UNLOCK)
open_table->table->lock.type=lock_type;
2000-07-31 21:29:14 +02:00
}
2002-11-11 22:43:07 +01:00
return to;
2000-07-31 21:29:14 +02:00
}
/* Find out database name and table name from a filename */
static void split_file_name(const char *file_name,
LEX_STRING *db, LEX_STRING *name)
{
uint dir_length, prefix_length;
char buff[FN_REFLEN];
db->length= 0;
strmake(buff, file_name, sizeof(buff)-1);
dir_length= dirname_length(buff);
if (dir_length > 1)
{
/* Get database */
buff[dir_length-1]= 0; // Remove end '/'
prefix_length= dirname_length(buff);
db->str= (char*) file_name+ prefix_length;
db->length= dir_length - prefix_length -1;
}
name->str= (char*) file_name+ dir_length;
name->length= (uint) (fn_ext(name->str) - name->str);
}
void ha_myisammrg::update_create_info(HA_CREATE_INFO *create_info)
{
2002-11-11 22:43:07 +01:00
DBUG_ENTER("ha_myisammrg::update_create_info");
if (!(create_info->used_fields & HA_CREATE_USED_UNION))
{
MYRG_TABLE *open_table;
THD *thd=current_thd;
create_info->merge_list.next= &create_info->merge_list.first;
create_info->merge_list.elements=0;
for (open_table=file->open_tables ;
open_table != file->end_table ;
open_table++)
{
TABLE_LIST *ptr;
LEX_STRING db, name;
if (!(ptr = (TABLE_LIST *) thd->calloc(sizeof(TABLE_LIST))))
goto err;
split_file_name(open_table->table->filename, &db, &name);
if (!(ptr->real_name= thd->strmake(name.str, name.length)))
goto err;
if (db.length && !(ptr->db= thd->strmake(db.str, db.length)))
goto err;
create_info->merge_list.elements++;
(*create_info->merge_list.next) = (byte*) ptr;
create_info->merge_list.next= (byte**) &ptr->next;
}
*create_info->merge_list.next=0;
}
if (!(create_info->used_fields & HA_CREATE_USED_INSERT_METHOD))
{
create_info->merge_insert_method = file->merge_insert_method;
}
DBUG_VOID_RETURN;
err:
create_info->merge_list.elements=0;
create_info->merge_list.first=0;
DBUG_VOID_RETURN;
}
2000-07-31 21:29:14 +02:00
2000-07-31 21:29:14 +02:00
int ha_myisammrg::create(const char *name, register TABLE *form,
HA_CREATE_INFO *create_info)
{
2000-09-14 02:39:07 +03:00
char buff[FN_REFLEN],**table_names,**pos;
TABLE_LIST *tables= (TABLE_LIST*) create_info->merge_list.first;
THD *thd= current_thd;
2000-09-14 02:39:07 +03:00
DBUG_ENTER("ha_myisammrg::create");
if (!(table_names= (char**) thd->alloc((create_info->merge_list.elements+1)*
sizeof(char*))))
DBUG_RETURN(HA_ERR_OUT_OF_MEM);
2000-09-14 02:39:07 +03:00
for (pos=table_names ; tables ; tables=tables->next)
2002-11-05 00:00:51 +01:00
{
char *table_name;
TABLE **tbl= 0;
2002-11-05 00:00:51 +01:00
if (create_info->options & HA_LEX_CREATE_TMP_TABLE)
tbl= find_temporary_table(thd, tables->db, tables->real_name);
if (!tbl)
2002-11-05 00:00:51 +01:00
{
uint length= my_snprintf(buff,FN_REFLEN,"%s%s/%s",
mysql_real_data_home,
tables->db, tables->real_name);
if (!(table_name= thd->strmake(buff, length)))
2004-02-11 00:06:46 +01:00
DBUG_RETURN(HA_ERR_OUT_OF_MEM);
2002-11-05 00:00:51 +01:00
}
else
table_name=(*tbl)->path;
2002-11-05 00:00:51 +01:00
*pos++= table_name;
}
2000-09-14 02:39:07 +03:00
*pos=0;
DBUG_RETURN(myrg_create(fn_format(buff,name,"","",2+4+16),
(const char **) table_names,
create_info->merge_insert_method,
(my_bool) 0));
2000-07-31 21:29:14 +02:00
}
2001-02-17 19:04:33 +02:00
2001-02-17 19:04:33 +02:00
void ha_myisammrg::append_create_info(String *packet)
{
const char *current_db;
uint db_length;
THD *thd= current_thd;
if (file->merge_insert_method != MERGE_INSERT_DISABLED)
{
packet->append(" INSERT_METHOD=",15);
packet->append(get_type(&merge_insert_method,file->merge_insert_method-1));
}
2001-02-17 19:04:33 +02:00
packet->append(" UNION=(",8);
MYRG_TABLE *open_table,*first;
2001-02-17 19:04:33 +02:00
current_db= table->table_cache_key;
db_length= strlen(current_db);
for (first=open_table=file->open_tables ;
open_table != file->end_table ;
open_table++)
2001-02-17 19:04:33 +02:00
{
LEX_STRING db, name;
split_file_name(open_table->table->filename, &db, &name);
if (open_table != first)
2001-02-17 19:04:33 +02:00
packet->append(',');
/* Report database for mapped table if it isn't in current database */
if (db.length &&
(db_length != db.length ||
strncmp(current_db, db.str, db.length)))
{
append_identifier(thd, packet, db.str, db.length);
packet->append('.');
}
append_identifier(thd, packet, name.str, name.length);
2001-02-17 19:04:33 +02:00
}
packet->append(')');
}