mirror of
https://github.com/MariaDB/server.git
synced 2025-01-17 20:42:30 +01:00
157002b12f
Removing the "external lock" functionality from Maria (as two separate processes wanting to share a table should not only my_lock() the data and index files but also the log files, and share memory (as the latest data is in the page cache), it sounds useless to feature this). Removing the MyISAM logging from Maria (as REDO logging will be done differently). BitKeeper/deleted/.del-maria_log.c~1fb295a18c3f5d4c: Delete: storage/maria/maria_log.c BitKeeper/deleted/.del-ma_log.c~4a44ec11d547772f: Delete: storage/maria/ma_log.c include/maria.h: unneeded storage/maria/Makefile.am: log removed storage/maria/ma_check.c: external locking removed storage/maria/ma_close.c: log removed storage/maria/ma_delete.c: log removed storage/maria/ma_delete_all.c: log removed. Unused var. storage/maria/ma_dynrec.c: external locking removed storage/maria/ma_extra.c: log removed storage/maria/ma_init.c: log removed storage/maria/ma_locking.c: external locking removed, log removed storage/maria/ma_open.c: external locking removed, log removed storage/maria/ma_static.c: log removed storage/maria/ma_statrec.c: external locking removed storage/maria/ma_test2.c: log removed storage/maria/ma_test3.c: log removed storage/maria/ma_update.c: log removed storage/maria/ma_write.c: external locking removed, log removed storage/maria/maria_chk.c: external locking removed storage/maria/maria_def.h: log removed, maria_pid unused. storage/maria/maria_pack.c: fixes for warnings (where pointers are like ulong and so %u is not enough).
298 lines
8.7 KiB
C
298 lines
8.7 KiB
C
/* Copyright (C) 2006 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
/* Functions to handle fixed-length-records */
|
|
|
|
#include "maria_def.h"
|
|
|
|
|
|
int _ma_write_static_record(MARIA_HA *info, const byte *record)
|
|
{
|
|
uchar temp[8]; /* max pointer length */
|
|
if (info->s->state.dellink != HA_OFFSET_ERROR &&
|
|
!info->append_insert_at_end)
|
|
{
|
|
my_off_t filepos=info->s->state.dellink;
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
if (info->s->file_read(info,(char*) &temp[0],info->s->base.rec_reflength,
|
|
info->s->state.dellink+1,
|
|
MYF(MY_NABP)))
|
|
goto err;
|
|
info->s->state.dellink= _ma_rec_pos(info->s,temp);
|
|
info->state->del--;
|
|
info->state->empty-=info->s->base.pack_reclength;
|
|
if (info->s->file_write(info, (char*) record, info->s->base.reclength,
|
|
filepos,
|
|
MYF(MY_NABP)))
|
|
goto err;
|
|
}
|
|
else
|
|
{
|
|
if (info->state->data_file_length > info->s->base.max_data_file_length-
|
|
info->s->base.pack_reclength)
|
|
{
|
|
my_errno=HA_ERR_RECORD_FILE_FULL;
|
|
return(2);
|
|
}
|
|
if (info->opt_flag & WRITE_CACHE_USED)
|
|
{ /* Cash in use */
|
|
if (my_b_write(&info->rec_cache, (byte*) record,
|
|
info->s->base.reclength))
|
|
goto err;
|
|
if (info->s->base.pack_reclength != info->s->base.reclength)
|
|
{
|
|
uint length=info->s->base.pack_reclength - info->s->base.reclength;
|
|
bzero((char*) temp,length);
|
|
if (my_b_write(&info->rec_cache, (byte*) temp,length))
|
|
goto err;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
if (info->s->file_write(info,(char*) record,info->s->base.reclength,
|
|
info->state->data_file_length,
|
|
info->s->write_flag))
|
|
goto err;
|
|
if (info->s->base.pack_reclength != info->s->base.reclength)
|
|
{
|
|
uint length=info->s->base.pack_reclength - info->s->base.reclength;
|
|
bzero((char*) temp,length);
|
|
if (info->s->file_write(info, (byte*) temp,length,
|
|
info->state->data_file_length+
|
|
info->s->base.reclength,
|
|
info->s->write_flag))
|
|
goto err;
|
|
}
|
|
}
|
|
info->state->data_file_length+=info->s->base.pack_reclength;
|
|
info->s->state.split++;
|
|
}
|
|
return 0;
|
|
err:
|
|
return 1;
|
|
}
|
|
|
|
int _ma_update_static_record(MARIA_HA *info, my_off_t pos, const byte *record)
|
|
{
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
return (info->s->file_write(info,
|
|
(char*) record,info->s->base.reclength,
|
|
pos,
|
|
MYF(MY_NABP)) != 0);
|
|
}
|
|
|
|
|
|
int _ma_delete_static_record(MARIA_HA *info)
|
|
{
|
|
uchar temp[9]; /* 1+sizeof(uint32) */
|
|
|
|
info->state->del++;
|
|
info->state->empty+=info->s->base.pack_reclength;
|
|
temp[0]= '\0'; /* Mark that record is deleted */
|
|
_ma_dpointer(info,temp+1,info->s->state.dellink);
|
|
info->s->state.dellink = info->lastpos;
|
|
info->rec_cache.seek_not_done=1;
|
|
return (info->s->file_write(info,(byte*) temp, 1+info->s->rec_reflength,
|
|
info->lastpos, MYF(MY_NABP)) != 0);
|
|
}
|
|
|
|
|
|
int _ma_cmp_static_record(register MARIA_HA *info, register const byte *old)
|
|
{
|
|
DBUG_ENTER("_ma_cmp_static_record");
|
|
|
|
/* We are going to do changes; dont let anybody disturb */
|
|
dont_break(); /* Dont allow SIGHUP or SIGINT */
|
|
|
|
if (info->opt_flag & WRITE_CACHE_USED)
|
|
{
|
|
if (flush_io_cache(&info->rec_cache))
|
|
{
|
|
DBUG_RETURN(-1);
|
|
}
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
}
|
|
|
|
if ((info->opt_flag & READ_CHECK_USED))
|
|
{ /* If check isn't disabled */
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength,
|
|
info->lastpos,
|
|
MYF(MY_NABP)))
|
|
DBUG_RETURN(-1);
|
|
if (memcmp((byte*) info->rec_buff, (byte*) old,
|
|
(uint) info->s->base.reclength))
|
|
{
|
|
DBUG_DUMP("read",old,info->s->base.reclength);
|
|
DBUG_DUMP("disk",info->rec_buff,info->s->base.reclength);
|
|
my_errno=HA_ERR_RECORD_CHANGED; /* Record have changed */
|
|
DBUG_RETURN(1);
|
|
}
|
|
}
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
int _ma_cmp_static_unique(MARIA_HA *info, MARIA_UNIQUEDEF *def,
|
|
const byte *record, my_off_t pos)
|
|
{
|
|
DBUG_ENTER("_ma_cmp_static_unique");
|
|
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
if (info->s->file_read(info, (char*) info->rec_buff, info->s->base.reclength,
|
|
pos, MYF(MY_NABP)))
|
|
DBUG_RETURN(-1);
|
|
DBUG_RETURN(_ma_unique_comp(def, record, info->rec_buff,
|
|
def->null_are_equal));
|
|
}
|
|
|
|
|
|
/* Read a fixed-length-record */
|
|
/* Returns 0 if Ok. */
|
|
/* 1 if record is deleted */
|
|
/* MY_FILE_ERROR on read-error or locking-error */
|
|
|
|
int _ma_read_static_record(register MARIA_HA *info, register my_off_t pos,
|
|
register byte *record)
|
|
{
|
|
int error;
|
|
|
|
if (pos != HA_OFFSET_ERROR)
|
|
{
|
|
if (info->opt_flag & WRITE_CACHE_USED &&
|
|
info->rec_cache.pos_in_file <= pos &&
|
|
flush_io_cache(&info->rec_cache))
|
|
return(-1);
|
|
info->rec_cache.seek_not_done=1; /* We have done a seek */
|
|
|
|
error=info->s->file_read(info,(char*) record,info->s->base.reclength,
|
|
pos,MYF(MY_NABP)) != 0;
|
|
fast_ma_writeinfo(info);
|
|
if (! error)
|
|
{
|
|
if (!*record)
|
|
{
|
|
my_errno=HA_ERR_RECORD_DELETED;
|
|
return(1); /* Record is deleted */
|
|
}
|
|
info->update|= HA_STATE_AKTIV; /* Record is read */
|
|
return(0);
|
|
}
|
|
return(-1); /* Error on read */
|
|
}
|
|
fast_ma_writeinfo(info); /* No such record */
|
|
return(-1);
|
|
}
|
|
|
|
|
|
|
|
int _ma_read_rnd_static_record(MARIA_HA *info, byte *buf,
|
|
register my_off_t filepos,
|
|
my_bool skip_deleted_blocks)
|
|
{
|
|
int locked,error,cache_read;
|
|
uint cache_length;
|
|
MARIA_SHARE *share=info->s;
|
|
DBUG_ENTER("_ma_read_rnd_static_record");
|
|
|
|
cache_read=0;
|
|
cache_length=0;
|
|
if (info->opt_flag & WRITE_CACHE_USED &&
|
|
(info->rec_cache.pos_in_file <= filepos || skip_deleted_blocks) &&
|
|
flush_io_cache(&info->rec_cache))
|
|
DBUG_RETURN(my_errno);
|
|
if (info->opt_flag & READ_CACHE_USED)
|
|
{ /* Cache in use */
|
|
if (filepos == my_b_tell(&info->rec_cache) &&
|
|
(skip_deleted_blocks || !filepos))
|
|
{
|
|
cache_read=1; /* Read record using cache */
|
|
cache_length=(uint) (info->rec_cache.read_end - info->rec_cache.read_pos);
|
|
}
|
|
else
|
|
info->rec_cache.seek_not_done=1; /* Filepos is changed */
|
|
}
|
|
locked=0;
|
|
if (info->lock_type == F_UNLCK)
|
|
{
|
|
if (filepos >= info->state->data_file_length)
|
|
{ /* Test if new records */
|
|
if (_ma_readinfo(info,F_RDLCK,0))
|
|
DBUG_RETURN(my_errno);
|
|
locked=1;
|
|
}
|
|
else
|
|
{ /* We don't nead new info */
|
|
#ifndef UNSAFE_LOCKING
|
|
if ((! cache_read || share->base.reclength > cache_length) &&
|
|
share->tot_locks == 0)
|
|
{ /* record not in cache */
|
|
locked=1;
|
|
}
|
|
#else
|
|
info->tmp_lock_type=F_RDLCK;
|
|
#endif
|
|
}
|
|
}
|
|
if (filepos >= info->state->data_file_length)
|
|
{
|
|
DBUG_PRINT("test",("filepos: %ld (%ld) records: %ld del: %ld",
|
|
filepos/share->base.reclength,filepos,
|
|
info->state->records, info->state->del));
|
|
fast_ma_writeinfo(info);
|
|
DBUG_RETURN(my_errno=HA_ERR_END_OF_FILE);
|
|
}
|
|
info->lastpos= filepos;
|
|
info->nextpos= filepos+share->base.pack_reclength;
|
|
|
|
if (! cache_read) /* No cacheing */
|
|
{
|
|
if ((error= _ma_read_static_record(info,filepos,buf)))
|
|
{
|
|
if (error > 0)
|
|
error=my_errno=HA_ERR_RECORD_DELETED;
|
|
else
|
|
error=my_errno;
|
|
}
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/* Read record with cacheing */
|
|
error=my_b_read(&info->rec_cache,(byte*) buf,share->base.reclength);
|
|
if (info->s->base.pack_reclength != info->s->base.reclength && !error)
|
|
{
|
|
char tmp[8]; /* Skill fill bytes */
|
|
error=my_b_read(&info->rec_cache,(byte*) tmp,
|
|
info->s->base.pack_reclength - info->s->base.reclength);
|
|
}
|
|
if (locked)
|
|
VOID(_ma_writeinfo(info,0)); /* Unlock keyfile */
|
|
if (!error)
|
|
{
|
|
if (!buf[0])
|
|
{ /* Record is removed */
|
|
DBUG_RETURN(my_errno=HA_ERR_RECORD_DELETED);
|
|
}
|
|
/* Found and may be updated */
|
|
info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED;
|
|
DBUG_RETURN(0);
|
|
}
|
|
/* my_errno should be set if rec_cache.error == -1 */
|
|
if (info->rec_cache.error != -1 || my_errno == 0)
|
|
my_errno=HA_ERR_WRONG_IN_RECORD;
|
|
DBUG_RETURN(my_errno); /* Something wrong (EOF?) */
|
|
}
|