diff options
Diffstat (limited to 'isam')
-rw-r--r-- | isam/.cvsignore | 10 | ||||
-rw-r--r-- | isam/ChangeLog | 186 | ||||
-rw-r--r-- | isam/Makefile.am | 51 | ||||
-rw-r--r-- | isam/_cache.c | 92 | ||||
-rw-r--r-- | isam/_dbug.c | 132 | ||||
-rw-r--r-- | isam/_dynrec.c | 1245 | ||||
-rw-r--r-- | isam/_key.c | 239 | ||||
-rw-r--r-- | isam/_locking.c | 345 | ||||
-rw-r--r-- | isam/_packrec.c | 1184 | ||||
-rw-r--r-- | isam/_page.c | 137 | ||||
-rw-r--r-- | isam/_search.c | 889 | ||||
-rw-r--r-- | isam/_statrec.c | 265 | ||||
-rw-r--r-- | isam/changed.c | 35 | ||||
-rw-r--r-- | isam/close.c | 91 | ||||
-rw-r--r-- | isam/create.c | 326 | ||||
-rw-r--r-- | isam/delete.c | 615 | ||||
-rw-r--r-- | isam/extra.c | 258 | ||||
-rw-r--r-- | isam/info.c | 77 | ||||
-rw-r--r-- | isam/isamchk.c | 3448 | ||||
-rw-r--r-- | isam/isamdef.h | 416 | ||||
-rw-r--r-- | isam/isamlog.c | 839 | ||||
-rw-r--r-- | isam/log.c | 156 | ||||
-rwxr-xr-x | isam/make-ccc | 3 | ||||
-rw-r--r-- | isam/open.c | 455 | ||||
-rw-r--r-- | isam/pack_isam.c | 2051 | ||||
-rw-r--r-- | isam/panic.c | 135 | ||||
-rw-r--r-- | isam/range.c | 191 | ||||
-rw-r--r-- | isam/rfirst.c | 34 | ||||
-rw-r--r-- | isam/rkey.c | 63 | ||||
-rw-r--r-- | isam/rlast.c | 34 | ||||
-rw-r--r-- | isam/rnext.c | 66 | ||||
-rw-r--r-- | isam/rprev.c | 63 | ||||
-rw-r--r-- | isam/rrnd.c | 55 | ||||
-rw-r--r-- | isam/rsame.c | 70 | ||||
-rw-r--r-- | isam/rsamepos.c | 59 | ||||
-rw-r--r-- | isam/sort.c | 558 | ||||
-rw-r--r-- | isam/static.c | 45 | ||||
-rw-r--r-- | isam/test1.c | 183 | ||||
-rw-r--r-- | isam/test2.c | 852 | ||||
-rw-r--r-- | isam/test3.c | 479 | ||||
-rwxr-xr-x | isam/test_all | 30 | ||||
-rw-r--r-- | isam/test_all.res | 356 | ||||
-rw-r--r-- | isam/update.c | 119 | ||||
-rw-r--r-- | isam/write.c | 840 |
44 files changed, 17777 insertions, 0 deletions
diff --git a/isam/.cvsignore b/isam/.cvsignore new file mode 100644 index 00000000000..dc55807a96b --- /dev/null +++ b/isam/.cvsignore @@ -0,0 +1,10 @@ +.deps +.libs +Makefile +Makefile.in +isamchk +isamlog +pack_isam +test1 +test2 +test3 diff --git a/isam/ChangeLog b/isam/ChangeLog new file mode 100644 index 00000000000..4a9e3e03954 --- /dev/null +++ b/isam/ChangeLog @@ -0,0 +1,186 @@ +2000-04-26 Michael Widenius <monty@mysql.com> + +* Fixed bug when doing read_next after a delete/insert which balanced key + pages (In this case one internal buffer was wrongly reused) + +1999-11-23 Michael Widenius <monty@monty.pp.sci.fi> + +* Changed prefix from ni_ to nisam_ to avoid problems on MacOS X. + +1999-08-17 Michael Widenius <monty@tik.pp.sci.fi> + +* Changed last parameter to mi_open() to be a bit flag instead of an int. + +1998-10-01 Michael Widenius <monty@monty.pp.sci.fi> + +* Fixed bug in key packing when using some USE_STRCOLL character sets. + +Thu Aug 20 23:17:41 1998 Michael Widenius <monty@bitch.pp.sci.fi> + +* isamchk.c: Sometimes isamchk --sort-table caused isamchk to die. + +1998-06-28 Michael Widenius <monty@monty.pp.sci.fi> + +* Fixed some possible race conditions when using with MySQL and + many reopen/close of the same tables under heavy load! +* Changed isamchk to re-pack records when doing a repair to make it more safer. + +Thu Mar 12 21:44:08 1998 Michael Widenius <monty@monty.pp.sci.fi> + +* Added a safty test to _ni_rec_unpack. + +Wed Nov 26 01:52:55 1997 <monty@monty.pp.sci.fi> + +* Fixed small problem when reading delete-marked records with rkey, rnext and + rprev. In normal applications this should never happen. + +Thu Nov 20 14:01:21 1997 <monty@monty.pp.sci.fi> + +* Fixed range key bug when using compressed key where the first part wasn't + compressed. +* Converted everything to use prototypes. + +Mon Sep 29 13:16:27 1997 <monty@monty.pp.sci.fi> + +* Fixed problem with isamchk and compressed records files with record_reflength + < 4 (Gave wrong key when using isamchk -rq). + +Fri Sep 26 16:06:37 1997 <monty@monty.pp.sci.fi> + +* Fixed bug in range calculation. + +Thu Aug 14 14:44:33 1997 <monty@monty.pp.sci.fi> + +* Removed a couple of unnecessary seeks from 'delete static record' + +Tue Jul 1 22:04:16 1997 <monty@monty.pp.sci.fi> + +* Added checking of 'wrong packed records' when using 'isamchk -e' or + isamchk -ro. + +Fri Feb 7 22:22:28 1997 Michael Widenius <monty@bitch.sci.fi> + +* Fixed use of packed tables with threads (One static variable left) + +Thu Jan 23 09:05:22 1997 Michael Widenius <monty@bitch.sci.fi> + +* Changed create to detect keys > 127 and not pack them. Now one can + define keys with a length of up to (nisam_block_size-18)/2 + by changeing N_MAX_KEY_LENGTH. + +Fri Jan 10 21:01:44 1997 Michael Widenius <monty@bitch.sci.fi> + +* added signed chars as key type. + +Fri Apr 26 14:31:05 1996 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* create.c: All keyfile blocks are now IO_SIZE big (for better keycashing). + +Tue Mar 12 22:42:52 1996 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* isamchk.c: Changed to print info if system table +* write.c: Don't allow more than 1 record in system table. + +Fri Feb 2 16:40:32 1996 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* isamchk.c; Check that delete-link-chain is ok before trying to delete with 'q'. + +Thu Jan 11 13:21:23 1996 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* All same isam files now shares a structure to allow many opens off the same + file + +Sat Nov 25 12:33:53 1995 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* All functions now used my_errno instead of errno + +Mon Oct 23 12:32:02 1995 Michael Widenius <monty@bitch.clinet.fi (Michael Widenius)> + +* isamchk.c: Don't print that database should be fixed on automatic rep. + +Sun Aug 27 20:13:56 1995 Michael Widenius <monty@bitch.analytikerna.se (Michael Widenius)> + +* _dynrec.c added flush_io_cash() if someone did a read when using + WRITE CASHING. + +Thu Apr 20 01:41:24 1995 Michael Widenius (monty@bitch) + +* fixed errno when got error of 'key-not-found' when updateing or + deleting record. + +Tue Jan 17 19:37:48 1995 Michael Widenius (monty@bitch) + +* isamchk can now unpack databases. +* prolinted all files. + +Fri May 27 15:01:06 1994 Michael Widenius (monty@bitch) + +* Don't lock packed databases. + +Sat Apr 16 22:41:23 1994 Michael Widenius (monty@bitch) + +* Added new function read_rsame_with_pos. + +Wed Mar 30 15:52:19 1994 Michael Widenius (monty@bitch) + +* Added creation and recover date to indexfile and isamchk. + +Sat Mar 26 15:03:37 1994 Michael Widenius (monty@bitch) + +* change is_panic() to close all files on ha_panic(write) on systems + (VMS) with can't open one file twice. + +Fri Feb 4 21:09:56 1994 Michael Widenius (monty@bitch) + +* READ_CASH on packed files now makes them mem-mapped if possibly + +Sat Sep 18 14:56:32 1993 Michael Widenius (monty at bitch) + +* changed _search to use pointer to buffer when reading keys. + +Mon Aug 16 19:45:29 1993 Michael Widenius (monty at bitch) + +* isamchk and packisam resolves symbolic links before file is used. + This forces temp-files on same disk as orginal file and rename + of temp-files dosen't destroy symbolic links. + +Mon May 31 18:26:08 1993 Michael Widenius (monty at bitch) + +* Added crc-check of records when packing for safe test if pack ok. + +Tue Mar 2 19:16:00 1993 Michael Widenius (monty@bitch) + +* Added logging of records with ni_log(). + +Fri Jan 29 00:56:58 1993 Michael Widenius (monty@bitch) + +* Fixed bug in _read_rnd_static_record ; A lock was made for + each record even if it was in cash. + +Sun Nov 15 12:51:36 1992 Michael Widenius (monty@bitch) + +* last change breaked _dynrec, when not compileing with dbug. + +Fri Nov 6 03:46:38 1992 Michael Widenius (monty@bitch) + +* Fixed bugg when using packed records and reclength < 8 byte. + +Wed Oct 28 22:23:32 1992 Michael Widenius (monty@bitch) + +* Changed _cash.c to use io_cash to allow use of aioread. + +Fri Oct 23 00:45:53 1992 Michael Widenius (monty@bitch) + +* Added MY_WAIT_IF_FULL to pack_isam. + +Sat Oct 17 14:51:15 1992 Michael Widenius (monty@bitch) + +* Added use of subset of keys (isamchk -k#) + +Mon Oct 5 21:53:18 1992 Michael Widenius (monty@bitch) + +* Remove reloc of database ; Gives only problems with isamchk. + +Mon Aug 17 03:17:09 1992 Michael Widenius (monty@bitch) + +* Changed isam to use io_cash instead of rec_cash diff --git a/isam/Makefile.am b/isam/Makefile.am new file mode 100644 index 00000000000..2d80ba70bf7 --- /dev/null +++ b/isam/Makefile.am @@ -0,0 +1,51 @@ +# Copyright (C) 2000 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 + +INCLUDES = @MT_INCLUDES@ -I$(srcdir)/../include -I../include +LDADD = @CLIENT_EXTRA_LDFLAGS@ libnisam.a ../mysys/libmysys.a \ + ../dbug/libdbug.a ../strings/libmystrings.a +pkglib_LIBRARIES = libnisam.a +bin_PROGRAMS = isamchk isamlog pack_isam +isamchk_DEPENDENCIES= $(LIBRARIES) +isamlog_DEPENDENCIES= $(LIBRARIES) +pack_isam_DEPENDENCIES= $(LIBRARIES) +noinst_PROGRAMS = test1 test2 test3 +noinst_HEADERS = isamdef.h +test1_DEPENDENCIES= $(LIBRARIES) +test2_DEPENDENCIES= $(LIBRARIES) +test3_DEPENDENCIES= $(LIBRARIES) +libnisam_a_SOURCES = open.c extra.c info.c rkey.c rnext.c \ + _search.c _page.c _key.c _locking.c \ + rrnd.c _cache.c _statrec.c _packrec.c \ + _dynrec.c update.c write.c delete.c \ + rprev.c rfirst.c rlast.c rsame.c rsamepos.c \ + panic.c close.c create.c range.c _dbug.c \ + log.c changed.c static.c +isamchk_SOURCES = isamchk.c sort.c +CLEANFILES = test?.IS? isam.log +# Omit dependency for ../mit-pthreads/include/ things +OMIT_DEPENDENCIES = pthread.h stdio.h __stdio.h stdlib.h __stdlib.h math.h\ + __math.h time.h __time.h unistd.h __unistd.h types.h \ + xtypes.h ac-types.h posix.h string.h __string.h \ + errno.h socket.h inet.h dirent.h netdb.h \ + cleanup.h cond.h debug_out.h fd.h kernel.h mutex.h \ + prio_queue.h pthread_attr.h pthread_once.h queue.h\ + sleep.h specific.h version.h pwd.h timers.h uio.h \ + cdefs.h machdep.h signal.h __signal.h util.h wait.h + +# Move to automake rules ? +prolint:; plparse -b -u -hF1 "-width(0,0)" "-format=%f:%l:\s%t:%n\s%m" \ + "-elib(????)" "+elib(?3??)" my.lnt $(nisam_SOURCES) diff --git a/isam/_cache.c b/isam/_cache.c new file mode 100644 index 00000000000..53619126660 --- /dev/null +++ b/isam/_cache.c @@ -0,0 +1,92 @@ +/* Copyright (C) 2000 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 for read record cacheing with nisam */ +/* Used instead of my_b_read() to allow for no-cacheed seeks */ + +#include "isamdef.h" + +#define READING_NEXT 1 +#define READING_HEADER 2 + + /* Copy block from cache if it`s in it. If re_read_if_possibly is */ + /* set read to cache (if after current file-position) else read to */ + /* buff */ + +int _nisam_read_cache(IO_CACHE *info, byte *buff, ulong pos, uint length, + int flag) +{ + uint read_length,in_buff_length; + ulong offset; + char *in_buff_pos; + + if (pos < info->pos_in_file) + { + read_length= (uint) min((ulong) length,(ulong) (info->pos_in_file-pos)); + info->seek_not_done=1; + VOID(my_seek(info->file,pos,MY_SEEK_SET,MYF(0))); + if (my_read(info->file,buff,read_length,MYF(MY_NABP))) + return 1; + if (!(length-=read_length)) + return 0; + pos+=read_length; + buff+=read_length; + } + if ((offset=pos - (ulong) info->pos_in_file) < + (ulong) (info->rc_end - info->rc_request_pos)) + { + in_buff_pos=info->rc_request_pos+(uint) offset; + in_buff_length= min(length,(uint) (info->rc_end-in_buff_pos)); + memcpy(buff,info->rc_request_pos+(uint) offset,(size_t) in_buff_length); + if (!(length-=in_buff_length)) + return 0; + pos+=in_buff_length; + buff+=in_buff_length; + } + else + in_buff_length=0; + if (flag & READING_NEXT) + { + if (pos != ((info)->pos_in_file + + (uint) ((info)->rc_end - (info)->rc_request_pos))) + { + info->pos_in_file=pos; /* Force start here */ + info->rc_pos=info->rc_end=info->rc_request_pos; /* Everything used */ + info->seek_not_done=1; + } + else + info->rc_pos=info->rc_end; /* All block used */ + if (!(*info->read_function)(info,buff,length)) + return 0; + if (!(flag & READING_HEADER) || info->error == -1 || + (uint) info->error+in_buff_length < 3) + return 1; + if (BLOCK_INFO_HEADER_LENGTH < in_buff_length + (uint) info->error) + bzero(buff+info->error,BLOCK_INFO_HEADER_LENGTH - in_buff_length - + (uint) info->error); + return 0; + } + info->seek_not_done=1; + VOID(my_seek(info->file,pos,MY_SEEK_SET,MYF(0))); + if ((read_length=my_read(info->file,buff,length,MYF(0))) == length) + return 0; + if (!(flag & READING_HEADER) || (int) read_length == -1 || + read_length+in_buff_length < 3) + return 1; + bzero(buff+read_length,BLOCK_INFO_HEADER_LENGTH - in_buff_length - + read_length); + return 0; +} /* _nisam_read_cache */ diff --git a/isam/_dbug.c b/isam/_dbug.c new file mode 100644 index 00000000000..644d0e420d6 --- /dev/null +++ b/isam/_dbug.c @@ -0,0 +1,132 @@ +/* Copyright (C) 2000 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 */ + +/* Support rutiner with are using with dbug */ + +#include "isamdef.h" + + /* Print a key in user understandable format */ + +void _nisam_print_key(FILE *stream, register N_KEYSEG *keyseg, const uchar *key) +{ + int flag; + short int s_1; + long int l_1; + float f_1; + double d_1; + uchar *end; + + VOID(fputs("Key: \"",stream)); + flag=0; + for (; keyseg->base.type ;keyseg++) + { + if (flag++) + VOID(putc('-',stream)); + end= (uchar*) key+ keyseg->base.length; + switch (keyseg->base.type) { + case HA_KEYTYPE_BINARY: + if (!(keyseg->base.flag & HA_SPACE_PACK) && keyseg->base.length == 1) + { /* packed binary digit */ + VOID(fprintf(stream,"%d",(uint) *key++)); + break; + } + /* fall through */ + case HA_KEYTYPE_TEXT: + case HA_KEYTYPE_NUM: + if (keyseg->base.flag & HA_SPACE_PACK) + { + VOID(fprintf(stream,"%.*s",(int) *key,key+1)); + key+= (int) *key+1; + } + else + { + VOID(fprintf(stream,"%.*s",(int) keyseg->base.length,key)); + key=end; + } + break; + case HA_KEYTYPE_INT8: + VOID(fprintf(stream,"%d",(int) *((signed char*) key))); + key=end; + break; + case HA_KEYTYPE_SHORT_INT: + shortget(s_1,key); + VOID(fprintf(stream,"%d",(int) s_1)); + key=end; + break; + case HA_KEYTYPE_USHORT_INT: + { + ushort u_1; + ushortget(u_1,key); + VOID(fprintf(stream,"%u",(uint) u_1)); + key=end; + break; + } + case HA_KEYTYPE_LONG_INT: + longget(l_1,key); + VOID(fprintf(stream,"%ld",l_1)); + key=end; + break; + case HA_KEYTYPE_ULONG_INT: + longget(l_1,key); + VOID(fprintf(stream,"%lu",(ulong) l_1)); + key=end; + break; + case HA_KEYTYPE_INT24: + VOID(fprintf(stream,"%d",sint3korr(key))); + key=end; + break; + case HA_KEYTYPE_UINT24: + VOID(fprintf(stream,"%ld",uint3korr(key))); + key=end; + break; + case HA_KEYTYPE_FLOAT: + bmove((byte*) &f_1,(byte*) key,(int) sizeof(float)); + VOID(fprintf(stream,"%g",(double) f_1)); + key=end; + break; + case HA_KEYTYPE_DOUBLE: + doubleget(d_1,key); + VOID(fprintf(stream,"%g",d_1)); + key=end; + break; +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + { + char buff[21]; + longlong tmp; + longlongget(tmp,key); + longlong2str(tmp,buff,-10); + VOID(fprintf(stream,"%s",buff)); + key=end; + break; + } + case HA_KEYTYPE_ULONGLONG: + { + char buff[21]; + longlong tmp; + longlongget(tmp,key); + longlong2str(tmp,buff,10); + VOID(fprintf(stream,"%s",buff)); + key=end; + break; + } +#endif + default: break; /* This never happens */ + } + } + VOID(fputs("\n",stream)); + return; +} /* print_key */ diff --git a/isam/_dynrec.c b/isam/_dynrec.c new file mode 100644 index 00000000000..6d4a491304c --- /dev/null +++ b/isam/_dynrec.c @@ -0,0 +1,1245 @@ +/* Copyright (C) 2000 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 space-packed-records and blobs */ + +#include "isamdef.h" + +/* Enough for comparing if number is zero */ +static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; + +static int write_dynamic_record(N_INFO *info,const byte *record, + uint reclength); +static int _nisam_find_writepos(N_INFO *info,uint reclength,ulong *filepos, + uint *length); +static int update_dynamic_record(N_INFO *info,ulong filepos,byte *record, + uint reclength); +static int delete_dynamic_record(N_INFO *info,ulong filepos, + uint second_read); +static int _nisam_cmp_buffer(File file, const byte *buff, ulong filepos, + uint length); + +#ifdef THREAD +/* Play it safe; We have a small stack when using threads */ +#undef my_alloca +#undef my_afree +#define my_alloca(A) my_malloc((A),MYF(0)) +#define my_afree(A) my_free((A),MYF(0)) +#endif + + /* Interface function from N_INFO */ + +int _nisam_write_dynamic_record(N_INFO *info, const byte *record) +{ + uint reclength=_nisam_rec_pack(info,info->rec_buff,record); + return (write_dynamic_record(info,info->rec_buff,reclength)); +} + +int _nisam_update_dynamic_record(N_INFO *info, ulong pos, const byte *record) +{ + uint length=_nisam_rec_pack(info,info->rec_buff,record); + return (update_dynamic_record(info,pos,info->rec_buff,length)); +} + +int _nisam_write_blob_record(N_INFO *info, const byte *record) +{ + byte *rec_buff; + int error; + uint reclength,extra; + + extra=ALIGN_SIZE(MAX_DYN_BLOCK_HEADER)+N_SPLITT_LENGTH+ + DYN_DELETE_BLOCK_HEADER; + if (!(rec_buff=(byte*) my_alloca(info->s->base.pack_reclength+ + _calc_total_blob_length(info,record)+ + extra))) + return(-1); + reclength=_nisam_rec_pack(info,rec_buff+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER), + record); + error=write_dynamic_record(info,rec_buff+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER), + reclength); + my_afree(rec_buff); + return(error); +} + + +int _nisam_update_blob_record(N_INFO *info, ulong pos, const byte *record) +{ + byte *rec_buff; + int error; + uint reclength,extra; + + extra=ALIGN_SIZE(MAX_DYN_BLOCK_HEADER)+N_SPLITT_LENGTH+ + DYN_DELETE_BLOCK_HEADER; + if (!(rec_buff=(byte*) my_alloca(info->s->base.pack_reclength+ + _calc_total_blob_length(info,record)+ + extra))) + return(-1); + reclength=_nisam_rec_pack(info,rec_buff+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER), + record); + error=update_dynamic_record(info,pos, + rec_buff+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER), + reclength); + my_afree(rec_buff); + return(error); +} + +int _nisam_delete_dynamic_record(N_INFO *info) +{ + return delete_dynamic_record(info,info->lastpos,0); +} + + + /* Write record to data-file */ + +static int write_dynamic_record(N_INFO *info, const byte *record, uint reclength) +{ + int flag; + uint length; + ulong filepos; + DBUG_ENTER("write_dynamic_record"); + + flag=0; + while (reclength) + { + if (_nisam_find_writepos(info,reclength,&filepos,&length)) + goto err; + if (_nisam_write_part_record(info,filepos,length,info->s->state.dellink, + (byte**) &record,&reclength,&flag)) + goto err; + } + + DBUG_RETURN(0); + err: + DBUG_RETURN(1); +} + + + /* Get a block for data ; The given data-area must be used !! */ + +static int _nisam_find_writepos(N_INFO *info, + uint reclength, /* record length */ + ulong *filepos, /* Return file pos */ + uint *length) /* length of block at filepos */ +{ + BLOCK_INFO block_info; + DBUG_ENTER("_nisam_find_writepos"); + + if (info->s->state.dellink != NI_POS_ERROR) + { + *filepos=info->s->state.dellink; + block_info.second_read=0; + info->rec_cache.seek_not_done=1; + if (!(_nisam_get_block_info(&block_info,info->dfile,info->s->state.dellink) & + BLOCK_DELETED)) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(-1); + } + info->s->state.dellink=block_info.next_filepos; + info->s->state.del--; + info->s->state.empty-= block_info.block_len; + *length= block_info.block_len; + } + else + { + if (info->s->state.data_file_length > info->s->base.max_data_file_length) + { + my_errno=HA_ERR_RECORD_FILE_FULL; + DBUG_RETURN(-1); + } + *filepos=info->s->state.data_file_length; /* New block last */ + if ((*length=reclength+3 + test(reclength > 65532)) < + info->s->base.min_block_length) + *length=info->s->base.min_block_length; + info->s->state.data_file_length+= *length; + info->s->state.splitt++; + info->update|=HA_STATE_WRITE_AT_END; + } + DBUG_RETURN(0); +} /* _nisam_find_writepos */ + + + /* Write a block to datafile */ + +int _nisam_write_part_record(N_INFO *info, + ulong filepos, /* points at empty block */ + uint length, /* length of block */ + ulong next_filepos, /* Next empty block */ + byte **record, /* pointer to record ptr */ + uint *reclength, /* length of *record */ + int *flag) /* *flag == 0 if header */ +{ + uint head_length,res_length,extra_length,long_block,del_length; + byte *pos,*record_end; + uchar temp[N_SPLITT_LENGTH+DYN_DELETE_BLOCK_HEADER]; + DBUG_ENTER("_nisam_write_part_record"); + + res_length=extra_length=0; + if (length > *reclength + N_SPLITT_LENGTH) + { /* Splitt big block */ + res_length=length- *reclength - 3 - N_EXTEND_BLOCK_LENGTH; + length-= res_length; /* Use this for first part */ + } + long_block= (length < 65535L && *reclength < 65535L) ? 0 : 1; + if (length-long_block == *reclength+3 || length == *reclength + 4) + { /* Exact what we need */ + temp[0]=(uchar) (1+ *flag); /* 1, or 9 */ + if (long_block) + { + int3store(temp+1,*reclength); + } + else + { + int2store(temp+1,*reclength); + } + head_length=3+long_block; + if (length-long_block == *reclength+4) + { + length--; + temp[0]++; /* 2 or 10 */ + extra_length++; /* One empty */ + } + } + else if (length-long_block < *reclength+5) + { /* To short block */ + if (next_filepos == NI_POS_ERROR) + next_filepos=info->s->state.dellink != NI_POS_ERROR ? + info->s->state.dellink : info->s->state.data_file_length; + if (*flag == 0) /* First block */ + { + head_length=5+4+long_block*2; + temp[0]=4; + if (long_block) + { + int3store(temp+1,*reclength); + int3store(temp+4,length-head_length); + int4store((byte*) temp+7,next_filepos); + } + else + { + int2store(temp+1,*reclength); + int2store(temp+3,length-head_length); + int4store((byte*) temp+5,next_filepos); + } + } + else + { + head_length=3+4+long_block; + temp[0]=12; + if (long_block) + { + int3store(temp+1,length-head_length); + int4store((byte*) temp+4,next_filepos); + } + else + { + int2store(temp+1,length-head_length); + int4store((byte*) temp+3,next_filepos); + } + } + } + else + { /* Block with empty info last */ + head_length=5+long_block*2; + temp[0]= (uchar) (3+ *flag); /* 3 or 11 */ + if (long_block) + { + int3store(temp+1,*reclength); + int3store(temp+4,length-7); + } + else + { + int2store(temp+1,*reclength); + int2store(temp+3,length-5); + } + extra_length= length- *reclength-head_length; + length= *reclength+head_length; /* Write only what is needed */ + } + temp[0]+=(uchar) (long_block*4); + DBUG_DUMP("header",(byte*) temp,head_length); + + /* Make a long block for one write */ + record_end= *record+length-head_length; + del_length=(res_length ? DYN_DELETE_BLOCK_HEADER : 0); + bmove((byte*) (*record-head_length),(byte*) temp,head_length); + memcpy(temp,record_end,(size_t) (extra_length+del_length)); + bzero((byte*) record_end,extra_length); + if (res_length) + { + pos=record_end+extra_length; + pos[0]= '\0'; + int3store(pos+1,res_length); + int4store(pos+4,info->s->state.dellink); + info->s->state.dellink= filepos+length+extra_length; + info->s->state.del++; + info->s->state.empty+=res_length; + info->s->state.splitt++; + } + if (info->opt_flag & WRITE_CACHE_USED && info->update & HA_STATE_WRITE_AT_END) + { + if (my_b_write(&info->rec_cache,(byte*) *record-head_length, + length+extra_length+del_length)) + goto err; + } + else + { + info->rec_cache.seek_not_done=1; + if (my_pwrite(info->dfile,(byte*) *record-head_length,length+extra_length+ + del_length,filepos,MYF(MY_NABP | MY_WAIT_IF_FULL))) + goto err; + } + memcpy(record_end,temp,(size_t) (extra_length+del_length)); + *record=record_end; + *reclength-=(length-head_length); + *flag=8; + + DBUG_RETURN(0); +err: + DBUG_PRINT("exit",("errno: %d",my_errno)); + DBUG_RETURN(1); +} /*_nisam_write_part_record */ + + + /* update record from datafile */ + +static int update_dynamic_record(N_INFO *info, ulong filepos, byte *record, uint reclength) +{ + int flag; + uint error,length; + BLOCK_INFO block_info; + DBUG_ENTER("update_dynamic_record"); + + flag=block_info.second_read=0; + while (reclength > 0) + { + if (filepos != info->s->state.dellink) + { + block_info.next_filepos= NI_POS_ERROR; + if ((error=_nisam_get_block_info(&block_info,info->dfile,filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (!(error & BLOCK_FATAL_ERROR)) + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } + length=(uint) (block_info.filepos-filepos) + block_info.block_len; + } + else + { + if (_nisam_find_writepos(info,reclength,&filepos,&length)) + goto err; + } + if (_nisam_write_part_record(info,filepos,length,block_info.next_filepos, + &record,&reclength,&flag)) + goto err; + if ((filepos=block_info.next_filepos) == NI_POS_ERROR) + filepos=info->s->state.dellink; + } + + if (block_info.next_filepos != NI_POS_ERROR) + if (delete_dynamic_record(info,block_info.next_filepos,1)) + goto err; + DBUG_RETURN(0); +err: + DBUG_RETURN(1); +} + + /* Delete datarecord from database */ + /* info->rec_cache.seek_not_done is updated in cmp_record */ + +static int delete_dynamic_record(N_INFO *info, ulong filepos, uint second_read) +{ + uint length,b_type; + BLOCK_INFO block_info; + DBUG_ENTER("delete_dynamic_record"); + + block_info.second_read=second_read; + do + { + if ((b_type=_nisam_get_block_info(&block_info,info->dfile,filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR) || + (length=(uint) (block_info.filepos-filepos) +block_info.block_len) < + N_MIN_BLOCK_LENGTH) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(1); + } + block_info.header[0]=0; + length=(uint) (block_info.filepos-filepos) +block_info.block_len; + int3store(block_info.header+1,length); + int4store(block_info.header+4,info->s->state.dellink); + if (my_pwrite(info->dfile,(byte*) block_info.header,8,filepos, + MYF(MY_NABP))) + DBUG_RETURN(1); + info->s->state.dellink = filepos; + info->s->state.del++; + info->s->state.empty+=length; + filepos=block_info.next_filepos; + } while (!(b_type & BLOCK_LAST)); + DBUG_RETURN(0); +} + + + /* Pack a record. Return new reclength */ + +uint _nisam_rec_pack(N_INFO *info, register byte *to, register const byte *from) +{ + uint length,new_length,flag,bit,i; + char *pos,*end,*startpos,*packpos; + enum en_fieldtype type; + reg3 N_RECINFO *rec; + N_BLOB *blob; + DBUG_ENTER("_nisam_rec_pack"); + + flag=0 ; bit=1; + startpos=packpos=to; to+= info->s->base.pack_bits; blob=info->blobs; + rec=info->s->rec; + + for (i=info->s->base.fields ; i-- > 0; from+= (rec++)->base.length) + { + length=(uint) rec->base.length; + if ((type = (enum en_fieldtype) rec->base.type) != FIELD_NORMAL) + { + if (type == FIELD_BLOB) + { + if (!blob->length) + flag|=bit; + else + { + char *temp_pos; + memcpy((byte*) to,from,(size_t) length); + memcpy_fixed(&temp_pos,from+length,sizeof(char*)); + memcpy(to+length,temp_pos,(size_t) blob->length); + to+=length+blob->length; + } + blob++; + from+=sizeof(char*); /* Skipp blob-pointer */ + } + else if (type == FIELD_SKIPP_ZERO) + { + if (memcmp((byte*) from,zero_string,length) == 0) + flag|=bit; + else + { + memcpy((byte*) to,from,(size_t) length); to+=length; + } + } + else if (type == FIELD_SKIPP_ENDSPACE || + type == FIELD_SKIPP_PRESPACE) + { + pos= (byte*) from; end= (byte*) from + length; + if (type == FIELD_SKIPP_ENDSPACE) + { /* Pack trailing spaces */ + while (end > from && *(end-1) == ' ') + end--; + } + else + { /* Pack pref-spaces */ + while (pos < end && *pos == ' ') + pos++; + } + new_length=(uint) (end-pos); + if (new_length +1 + test(rec->base.length > 255 && new_length > 127) + < length) + { + if (rec->base.length > 255 && new_length > 127) + { + to[0]=(char) ((new_length & 127)+128); + to[1]=(char) (new_length >> 7); + to+=2; + } + else + *to++= (char) new_length; + memcpy((byte*) to,pos,(size_t) new_length); to+=new_length; + flag|=bit; + } + else + { + memcpy(to,from,(size_t) length); to+=length; + } + } + else if (type == FIELD_ZERO) + continue; /* Don't store this */ + else + { + memcpy(to,from,(size_t) length); to+=length; + continue; /* Normal field */ + } + if ((bit= bit << 1) >= 256) + { + *packpos++ = (char) (uchar) flag; + bit=1; flag=0; + } + } + else + { + memcpy(to,from,(size_t) length); to+=length; + } + } + if (bit != 1) + *packpos= (char) (uchar) flag; + DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos))); + DBUG_RETURN((uint) (to-startpos)); +} /* _nisam_rec_pack */ + + + +/* +** Check if a record was correctly packed. Used only by isamchk +** Returns 0 if record is ok. +*/ + +my_bool _nisam_rec_check(N_INFO *info,const char *from) +{ + uint length,new_length,flag,bit,i; + char *pos,*end,*packpos,*to; + enum en_fieldtype type; + reg3 N_RECINFO *rec; + DBUG_ENTER("_nisam_rec_check"); + + packpos=info->rec_buff; to= info->rec_buff+info->s->base.pack_bits; + rec=info->s->rec; + flag= *packpos; bit=1; + + for (i=info->s->base.fields ; i-- > 0; from+= (rec++)->base.length) + { + length=(uint) rec->base.length; + if ((type = (enum en_fieldtype) rec->base.type) != FIELD_NORMAL) + { + if (type == FIELD_BLOB) + { + uint blob_length= _calc_blob_length(length,from); + if (!blob_length && !(flag & bit)) + goto err; + if (blob_length) + to+=length+ blob_length; + from+=sizeof(char*); + } + else if (type == FIELD_SKIPP_ZERO) + { + if (memcmp((byte*) from,zero_string,length) == 0) + { + if (!(flag & bit)) + goto err; + } + else + to+=length; + } + else if (type == FIELD_SKIPP_ENDSPACE || + type == FIELD_SKIPP_PRESPACE) + { + pos= (byte*) from; end= (byte*) from + length; + if (type == FIELD_SKIPP_ENDSPACE) + { /* Pack trailing spaces */ + while (end > from && *(end-1) == ' ') + end--; + } + else + { /* Pack pre-spaces */ + while (pos < end && *pos == ' ') + pos++; + } + new_length=(uint) (end-pos); + if (new_length +1 + test(rec->base.length > 255 && new_length > 127) + < length) + { + if (!(flag & bit)) + goto err; + if (rec->base.length > 255 && new_length > 127) + { + if (to[0] != (char) ((new_length & 127)+128) || + to[1] != (char) (new_length >> 7)) + goto err; + to+=2; + } + else if (*to++ != (char) new_length) + goto err; + to+=new_length; + } + else + to+=length; + } + else + { + if (type != FIELD_ZERO) + to+=length; /* Not packed field */ + continue; + } + if ((bit= bit << 1) >= 256) + { + flag= *++packpos; + bit=1; + } + } + else + { + to+=length; + } + } + if (bit != 1) + *packpos= (char) (uchar) flag; + if (info->packed_length == (uint) (to - info->rec_buff) && + (bit == 1 || !(flag & ~(bit - 1)))) + DBUG_RETURN(0); + + err: + DBUG_RETURN(1); +} + + + + /* Unpacks a record */ + /* Returns -1 and my_errno =HA_ERR_RECORD_DELETED if reclength isn't */ + /* right. Returns reclength (>0) if ok */ + +uint _nisam_rec_unpack(register N_INFO *info, register byte *to, byte *from, + uint found_length) +{ + uint flag,bit,length,rec_length,min_pack_length; + enum en_fieldtype type; + byte *from_end,*to_end,*packpos; + reg3 N_RECINFO *rec,*end_field; + DBUG_ENTER("_nisam_rec_unpack"); + + to_end=to + info->s->base.reclength; + from_end=from+found_length; + flag= (uchar) *from; bit=1; packpos=from; + if (found_length < info->s->base.min_pack_length) + goto err; + from+= info->s->base.pack_bits; + min_pack_length=info->s->base.min_pack_length - info->s->base.pack_bits; + + for (rec=info->s->rec , end_field=rec+info->s->base.fields ; + rec < end_field ; to+= rec_length, rec++) + { + rec_length=rec->base.length; + if ((type = (enum en_fieldtype) rec->base.type) != FIELD_NORMAL) + { + if (type == FIELD_ZERO) + continue; /* Skipp this */ + if (flag & bit) + { + if (type == FIELD_BLOB) + { + bzero((byte*) to,rec_length+sizeof(char*)); + to+=sizeof(char*); + } + else if (type == FIELD_SKIPP_ZERO) + bzero((byte*) to,rec_length); + else if (type == FIELD_SKIPP_ENDSPACE || + type == FIELD_SKIPP_PRESPACE) + { + if (rec->base.length > 255 && *from & 128) + { + if (from + 1 >= from_end) + goto err; + length= (*from & 127)+ ((uint) (uchar) *(from+1) << 7); from+=2; + } + else + { + if (from == from_end) + goto err; + length= (uchar) *from++; + } + min_pack_length--; + if (length >= rec_length || + min_pack_length + length > (uint) (from_end - from)) + goto err; + if (type == FIELD_SKIPP_ENDSPACE) + { + memcpy(to,(byte*) from,(size_t) length); + bfill((byte*) to+length,rec_length-length,' '); + } + else + { + bfill((byte*) to,rec_length-length,' '); + memcpy(to+rec_length-length,(byte*) from,(size_t) length); + } + from+=length; + } + } + else if (type == FIELD_BLOB) + { + ulong blob_length=_calc_blob_length(rec_length,from); + if ((ulong) (from_end-from) - rec_length < blob_length || + min_pack_length > (uint) (from_end -(from+rec_length+blob_length))) + goto err; + memcpy((byte*) to,(byte*) from,(size_t) rec_length); + from+=rec_length; + /* memcpy crasches alpha egcs 1.1.2 */ + bmove((byte*) to+rec_length,(byte*) &from,sizeof(char*)); + from+=blob_length; + to+=sizeof(char*); + } + else + { + if (type == FIELD_SKIPP_ENDSPACE || type == FIELD_SKIPP_PRESPACE) + min_pack_length--; + if (min_pack_length + rec_length > (uint) (from_end - from)) + goto err; + memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length; + } + if ((bit= bit << 1) >= 256) + { + flag= (uchar) *++packpos; bit=1; + } + } + else + { + if (min_pack_length > (uint) (from_end - from)) + goto err; + min_pack_length-=rec_length; + memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length; + } + } + if (to == to_end && from == from_end && (bit == 1 || !(flag & ~(bit-1)))) + DBUG_RETURN((info->packed_length=found_length)); + err: + my_errno=HA_ERR_RECORD_DELETED; + DBUG_PRINT("error",("to_end: %lx -> %lx from_end: %lx -> %lx", + to,to_end,from,from_end)); + DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length); + DBUG_RETURN(MY_FILE_ERROR); +} /* _nisam_rec_unpack */ + + + /* Calc length of blob. Update info in blobs->length */ + +uint _calc_total_blob_length(N_INFO *info, const byte *record) +{ + uint i,length; + N_BLOB *blob; + + for (i=length=0, blob= info->blobs; i++ < info->s->base.blobs ; blob++) + { + blob->length=_calc_blob_length(blob->pack_length,record + blob->offset); + length+=blob->length; + } + return length; +} + + +uint _calc_blob_length(uint length, const byte *pos) +{ + switch (length) { + case 1: + return (uint) (uchar) *pos; + case 2: + { + short j; shortget(j,pos); + return (uint) (unsigned short) j; + } +#ifdef MSDOS + break; /* skipp microsoft warning */ +#endif + case 3: + return uint3korr(pos); + case 4: + { + long j; longget(j,pos); + return (uint) j; + } +#ifdef MSDOS + break; +#endif + default: + break; + } + return 0; /* Impossible */ +} + + /* Read record from datafile */ + /* Returns 0 if ok, -1 if error */ + +int _nisam_read_dynamic_record(N_INFO *info, ulong filepos, byte *buf) +{ + int flag; + uint b_type,left_length; + byte *to; + BLOCK_INFO block_info; + File file; + DBUG_ENTER("ni_read_dynamic_record"); + + if (filepos != NI_POS_ERROR) + { + LINT_INIT(to); + LINT_INIT(left_length); + file=info->dfile; + block_info.next_filepos=filepos; /* for easyer loop */ + flag=block_info.second_read=0; + do + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file <= block_info.next_filepos && + flush_io_cache(&info->rec_cache)) + goto err; + info->rec_cache.seek_not_done=1; + if ((b_type=_nisam_get_block_info(&block_info,file, + block_info.next_filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) + my_errno=HA_ERR_RECORD_DELETED; + goto err; + } + if (flag == 0) /* First block */ + { + flag=1; + if (block_info.rec_len > (uint) info->s->base.max_pack_length) + goto panic; + if (info->s->base.blobs) + { + if (!(to=fix_rec_buff_for_blob(info,block_info.rec_len))) + goto err; + } + else + to= info->rec_buff; + left_length=block_info.rec_len; + } + if (left_length < block_info.data_len || ! block_info.data_len) + goto panic; /* Wrong linked record */ + if (my_pread(file,(byte*) to,block_info.data_len,block_info.filepos, + MYF(MY_NABP))) + goto panic; + left_length-=block_info.data_len; + to+=block_info.data_len; + } while (left_length); + + info->update|= HA_STATE_AKTIV; /* We have a aktive record */ + VOID(_nisam_writeinfo(info,0)); + DBUG_RETURN(_nisam_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != + MY_FILE_ERROR ? 0 : -1); + } + VOID(_nisam_writeinfo(info,0)); + DBUG_RETURN(-1); /* Wrong data to read */ + +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; +err: + VOID(_nisam_writeinfo(info,0)); + DBUG_RETURN(-1); +} + + +byte *fix_rec_buff_for_blob(N_INFO *info, uint length) +{ + uint extra; + if (! info->rec_buff || length > info->alloced_rec_buff_length) + { + byte *newptr; + extra=ALIGN_SIZE(MAX_DYN_BLOCK_HEADER)+N_SPLITT_LENGTH+ + DYN_DELETE_BLOCK_HEADER; + if (!(newptr=(byte*) my_realloc((gptr) info->rec_alloc,length+extra, + MYF(MY_ALLOW_ZERO_PTR)))) + return newptr; + info->rec_alloc=newptr; + info->rec_buff=newptr+ALIGN_SIZE(DYN_DELETE_BLOCK_HEADER); + info->alloced_rec_buff_length=length; + } + return info->rec_buff; +} + + + /* Compare of record one disk with packed record in memory */ + +int _nisam_cmp_dynamic_record(register N_INFO *info, register const byte *record) +{ + uint flag,reclength,b_type; + ulong filepos; + byte *buffer; + BLOCK_INFO block_info; + DBUG_ENTER("_nisam_cmp_dynamic_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) + { + info->update&= ~HA_STATE_WRITE_AT_END; + if (flush_io_cache(&info->rec_cache)) + DBUG_RETURN(-1); + } + info->rec_cache.seek_not_done=1; + + /* If nobody have touched the database we don't have to test rec */ + + buffer=info->rec_buff; + if ((info->opt_flag & READ_CHECK_USED)) + { /* If check isn't disabled */ + if (info->s->base.blobs) + { + if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+ + _calc_total_blob_length(info,record)))) + DBUG_RETURN(-1); + } + reclength=_nisam_rec_pack(info,buffer,record); + record= buffer; + + filepos=info->lastpos; + flag=block_info.second_read=0; + block_info.next_filepos=filepos; + while (reclength > 0) + { + if ((b_type=_nisam_get_block_info(&block_info,info->dfile, + block_info.next_filepos)) + & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED)) + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + if (flag == 0) /* First block */ + { + flag=1; + if (reclength != block_info.rec_len) + { + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + } else if (reclength < block_info.data_len) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } + reclength-=block_info.data_len; + if (_nisam_cmp_buffer(info->dfile,record,block_info.filepos, + block_info.data_len)) + { + my_errno=HA_ERR_RECORD_CHANGED; + goto err; + } + flag=1; + record+=block_info.data_len; + } + } + my_errno=0; + err: + if (buffer != info->rec_buff) + my_afree((gptr) buffer); + DBUG_RETURN(my_errno); +} + + + /* Compare file to buffert */ + +static int _nisam_cmp_buffer(File file, const byte *buff, ulong filepos, uint length) +{ + uint next_length; + char temp_buff[IO_SIZE*2]; + DBUG_ENTER("_nisam_cmp_buffer"); + + VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); + next_length= IO_SIZE*2 - (uint) (filepos & (IO_SIZE-1)); + + while (length > IO_SIZE*2) + { + if (my_read(file,temp_buff,next_length,MYF(MY_NABP))) + goto err; + if (memcmp((byte*) buff,temp_buff,IO_SIZE)) + DBUG_RETURN(1); + buff+=next_length; + length-= next_length; + next_length=IO_SIZE*2; + } + if (my_read(file,temp_buff,length,MYF(MY_NABP))) + goto err; + DBUG_RETURN(memcmp((byte*) buff,temp_buff,length)); +err: + DBUG_RETURN(1); +} + + +int _nisam_read_rnd_dynamic_record(N_INFO *info, byte *buf, register ulong filepos, int skipp_deleted_blocks) +{ + int flag,info_read,fatal_errcode; + uint left_len,b_type; + byte *to; + BLOCK_INFO block_info; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_read_rnd_dynamic_record"); + + info_read=0; + fatal_errcode= -1; + LINT_INIT(to); + +#ifndef NO_LOCKING + if (info->lock_type == F_UNLCK) + { +#ifndef UNSAFE_LOCKING + if (share->r_locks == 0 && share->w_locks == 0) + { + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + DBUG_RETURN(fatal_errcode); + } +#else + info->tmp_lock_type=F_RDLCK; +#endif + } + else + info_read=1; /* memory-keyinfoblock is ok */ +#endif /* !NO_LOCKING */ + + flag=block_info.second_read=0; + left_len=1; + do + { + if (filepos >= share->state.data_file_length) + { +#ifndef NO_LOCKING + if (!info_read) + { /* Check if changed */ + info_read=1; + info->rec_cache.seek_not_done=1; + if (my_pread(share->kfile,(char*) &share->state.header, + share->state_length, 0L,MYF(MY_NABP))) + goto err; + } + if (filepos >= share->state.data_file_length) +#endif + { + my_errno= HA_ERR_END_OF_FILE; + goto err; + } + } + if (info->opt_flag & READ_CACHE_USED) + { + if (_nisam_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, + sizeof(block_info.header), + test(!flag && skipp_deleted_blocks) | 2)) + goto err; + b_type=_nisam_get_block_info(&block_info,-1,filepos); + } + else + { + if (info->opt_flag & WRITE_CACHE_USED && + info->rec_cache.pos_in_file <= filepos && + flush_io_cache(&info->rec_cache)) + DBUG_RETURN(-1); + info->rec_cache.seek_not_done=1; + b_type=_nisam_get_block_info(&block_info,info->dfile,filepos); + } + + if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if ((b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + && skipp_deleted_blocks) + { + filepos=block_info.filepos+block_info.block_len; + block_info.second_read=0; + continue; /* Search after next_record */ + } + if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + { + my_errno=HA_ERR_RECORD_DELETED; + info->lastpos=block_info.filepos; + info->nextpos=block_info.filepos+block_info.block_len; + fatal_errcode=1; + } + goto err; + } + if (flag == 0) /* First block */ + { + if (block_info.rec_len > (uint) share->base.max_pack_length) + goto panic; + info->lastpos=filepos; + if (share->base.blobs) + { + if (!(to=fix_rec_buff_for_blob(info,block_info.rec_len))) + goto err; + } + else + to= info->rec_buff; + left_len=block_info.rec_len; + } + if (left_len < block_info.data_len) + goto panic; /* Wrong linked record */ + + if (info->opt_flag & READ_CACHE_USED) + { + if (_nisam_read_cache(&info->rec_cache,(byte*) to,block_info.filepos, + block_info.data_len, + test(!flag && skipp_deleted_blocks))) + goto err; + } + else + { + VOID(my_seek(info->dfile,block_info.filepos,MY_SEEK_SET,MYF(0))); + if (my_read(info->dfile,(byte*) to,block_info.data_len,MYF(MY_NABP))) + goto err; + } + if (flag++ == 0) + { + info->nextpos=block_info.filepos+block_info.block_len; + skipp_deleted_blocks=0; + } + left_len-=block_info.data_len; + to+=block_info.data_len; + filepos=block_info.next_filepos; + } while (left_len); + + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + VOID(_nisam_writeinfo(info,0)); + if (_nisam_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) != + MY_FILE_ERROR) + DBUG_RETURN(0); + DBUG_RETURN(fatal_errcode); /* Wrong record */ + +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; /* Something is fatal wrong */ +err: + VOID(_nisam_writeinfo(info,0)); + DBUG_RETURN(fatal_errcode); +} + + + /* Read and process header from a dynamic-record-file */ + +uint _nisam_get_block_info(BLOCK_INFO *info, File file, ulong filepos) +{ + uint return_val=0,length; + uchar *header=info->header; + + if (file >= 0) + { + VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); + if ((length=my_read(file,(char*) header,BLOCK_INFO_HEADER_LENGTH,MYF(0))) + == MY_FILE_ERROR) + return BLOCK_FATAL_ERROR; + if (length != BLOCK_INFO_HEADER_LENGTH) + { /* Test if short block */ + if (length < 3) + { + my_errno=HA_ERR_WRONG_IN_RECORD; /* Garbage */ + return BLOCK_FATAL_ERROR; + } + bzero((byte*) header+length,BLOCK_INFO_HEADER_LENGTH-length); + } + } + DBUG_DUMP("header",(byte*) header,BLOCK_INFO_HEADER_LENGTH); + if (info->second_read) + { + if (info->header[0] <= 8) + return_val=BLOCK_SYNC_ERROR; + } + else + { + if (info->header[0] > 8) + return_val=BLOCK_SYNC_ERROR; + } + info->next_filepos= (ulong) NI_POS_ERROR; /* Dummy ifall no next block */ + + switch (info->header[0]) { + case 0: + if ((info->block_len=(uint) uint3korr(header+1)) < N_MIN_BLOCK_LENGTH) + return BLOCK_FATAL_ERROR; + info->filepos=filepos; + info->next_filepos=uint4korr(header+4); + if (info->next_filepos == (uint32) ~0) /* Fix for 64 bit long */ + info->next_filepos=NI_POS_ERROR; + return return_val | BLOCK_DELETED; /* Deleted block */ + case 1: + info->rec_len=info->data_len=info->block_len=uint2korr(header+1); + info->filepos=filepos+3; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 2: + info->block_len=(info->rec_len=info->data_len=uint2korr(header+1))+1; + info->filepos=filepos+3; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 3: + info->rec_len=info->data_len=uint2korr(header+1); + info->block_len=uint2korr(header+3); + info->filepos=filepos+5; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 4: + info->rec_len=uint2korr(header+1); + info->block_len=info->data_len=uint2korr(header+3); + info->next_filepos=uint4korr(header+5); + info->second_read=1; + info->filepos=filepos+9; + return return_val | BLOCK_FIRST; +#if defined(_MSC_VER) || !defined(__WIN__) + case 5: + info->rec_len=info->data_len=info->block_len=uint3korr(header+1); + info->filepos=filepos+4; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 6: + info->block_len=(info->rec_len=info->data_len=uint3korr(header+1))+1; + info->filepos=filepos+4; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 7: + info->rec_len=info->data_len=uint3korr(header+1); + info->block_len=uint3korr(header+4); + info->filepos=filepos+7; + return return_val | BLOCK_FIRST | BLOCK_LAST; + case 8: + info->rec_len=uint3korr(header+1); + info->block_len=info->data_len=uint3korr(header+4); + info->next_filepos=uint4korr(header+7); + info->second_read=1; + info->filepos=filepos+11; + return return_val | BLOCK_FIRST; +#endif + case 9: + info->data_len=info->block_len=uint2korr(header+1); + info->filepos=filepos+3; + return return_val | BLOCK_LAST; + case 10: + info->block_len=(info->data_len=uint2korr(header+1))+1; + info->filepos=filepos+3; + return return_val | BLOCK_LAST; + case 11: + info->data_len=uint2korr(header+1); + info->block_len=uint2korr(header+3); + info->filepos=filepos+5; + return return_val | BLOCK_LAST; + case 12: + info->data_len=info->block_len=uint2korr(header+1); + info->next_filepos=uint4korr(header+3); + info->second_read=1; + info->filepos=filepos+7; + return return_val; +#if defined(_MSC_VER) || !defined(__WIN__) + case 13: + info->data_len=info->block_len=uint3korr(header+1); + info->filepos=filepos+4; + return return_val | BLOCK_LAST; + case 14: + info->block_len=(info->data_len=uint3korr(header+1))+1; + info->filepos=filepos+4; + return return_val | BLOCK_LAST; + case 15: + info->data_len=uint3korr(header+1); + info->block_len=uint3korr(header+4); + info->filepos=filepos+7; + return return_val | BLOCK_LAST; + case 16: + info->data_len=info->block_len=uint3korr(header+1); + info->next_filepos=uint4korr(header+4); + info->second_read=1; + info->filepos=filepos+8; + return return_val; +#endif + default: + my_errno=HA_ERR_WRONG_IN_RECORD; /* Garbage */ + return BLOCK_ERROR; + } +} diff --git a/isam/_key.c b/isam/_key.c new file mode 100644 index 00000000000..62f080af172 --- /dev/null +++ b/isam/_key.c @@ -0,0 +1,239 @@ +/* Copyright (C) 2000 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 keys */ + +#include "isamdef.h" +#include "m_ctype.h" + +static void _nisam_put_key_in_record(N_INFO *info,uint keynr,byte *record); + + /* Make a intern key from a record */ + /* If ascii key convert according to sortorder */ + /* Ret: Length of key */ + +uint _nisam_make_key(register N_INFO *info, uint keynr, uchar *key, const char *record, ulong filepos) +{ + uint length; + byte *pos,*end; + uchar *start; + reg1 N_KEYSEG *keyseg; + enum ha_base_keytype type; + DBUG_ENTER("_nisam_make_key"); + + start=key; + for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->base.type ;keyseg++) + { + type=(enum ha_base_keytype) keyseg->base.type; + if (keyseg->base.flag & HA_SPACE_PACK) + { + pos= (byte*) record+keyseg->base.start; end=pos+keyseg->base.length; + if (type != HA_KEYTYPE_NUM) + { + while (end > pos && end[-1] == ' ') + end--; + } + else + { + while (pos < end && pos[0] == ' ') + pos++; + } + *key++= (uchar) (length=(uint) (end-pos)); + memcpy((byte*) key,(byte*) pos,(size_t) length); +#ifdef USE_STRCOLL + if (!use_strcoll(default_charset_info)) +#endif + { + if (type == HA_KEYTYPE_TEXT) + case_sort((byte*) key,length); + } + key+=length; + } + else + { + memcpy((byte*) key,(byte*) record+keyseg->base.start, + (size_t) keyseg->base.length); +#ifdef USE_STRCOLL + if (!use_strcoll(default_charset_info)) +#endif + { + if (type == HA_KEYTYPE_TEXT) + case_sort((byte*) key,(uint) keyseg->base.length); + } +#ifdef NAN_TEST + else if (type == HA_KEYTYPE_FLOAT) + { + float nr; + bmove((byte*) &nr,(byte*) key,sizeof(float)); + if (nr == (float) FLT_MAX) + { + nr= (float) FLT_MAX; + bmove((byte*) key,(byte*) &nr,sizeof(float)); + } + } + else if (type == HA_KEYTYPE_DOUBLE) + { + double nr; + bmove((byte*) &nr,(byte*) key,sizeof(double)); + if (nr == DBL_MAX) + { + nr=DBL_MAX; + bmove((byte*) key,(byte*) &nr,sizeof(double)); + } + } +#endif + key+= keyseg->base.length; + } + } + _nisam_dpointer(info,key,filepos); + DBUG_PRINT("exit",("keynr: %d",keynr)); + DBUG_DUMP("key",(byte*) start,(uint) (key-start)+keyseg->base.length); + DBUG_EXECUTE("key",_nisam_print_key(DBUG_FILE,info->s->keyinfo[keynr].seg,start);); + DBUG_RETURN((uint) (key-start)); /* Return keylength */ +} /* _nisam_make_key */ + + + /* Pack a key to intern format from given format (c_rkey) */ + /* if key_length is set returns new length of key */ + +uint _nisam_pack_key(register N_INFO *info, uint keynr, uchar *key, uchar *old, uint key_length) + + + + /* Length of used key */ +{ + int k_length; + uint length; + uchar *pos,*end; + reg1 N_KEYSEG *keyseg; + enum ha_base_keytype type; + DBUG_ENTER("_nisam_pack_key"); + + if ((k_length=(int) key_length) <= 0) + k_length=N_MAX_KEY_BUFF; + + for (keyseg=info->s->keyinfo[keynr].seg ; + keyseg->base.type && k_length >0; + k_length-=keyseg->base.length, old+=keyseg->base.length, keyseg++) + { + length=min((uint) keyseg->base.length,(uint) k_length); + type=(enum ha_base_keytype) keyseg->base.type; + if (keyseg->base.flag & HA_SPACE_PACK) + { + pos=old; end=pos+length; + if (type != HA_KEYTYPE_NUM) + { + while (end > pos && end[-1] == ' ') + end--; + } + else + { + while (pos < end && pos[0] == ' ') + pos++; + } + *key++ = (uchar) (length=(uint) (end-pos)); + memcpy((byte*) key,pos,(size_t) length); + } + else + memcpy((byte*) key,old,(size_t) length); +#ifdef USE_STRCOLL + if (!use_strcoll(default_charset_info)) +#endif + { + if (type == HA_KEYTYPE_TEXT) + case_sort((byte*) key,length); + } + key+= length; + } + if (!keyseg->base.type) + { + if (k_length >= 0) /* Hole key */ + key_length=0; + } + else + { /* Part-key ; fill with null */ + length= (uint) -k_length; /* unused part of last key */ + do + { + length+= (keyseg->base.flag & HA_SPACE_PACK) ? 1 : + keyseg->base.length; + keyseg++; + } while (keyseg->base.type); + bzero((byte*) key,length); + } + DBUG_RETURN(key_length); /* Return part-keylength */ +} /* _nisam_pack_key */ + + + /* Put a key in record */ + /* Used when only-keyread is wanted */ + +static void _nisam_put_key_in_record(register N_INFO *info, uint keynr, byte *record) +{ + uint length; + reg2 byte *key; + byte *pos; + reg1 N_KEYSEG *keyseg; + DBUG_ENTER("_nisam_put_key_in_record"); + + key=(byte*) info->lastkey; + for (keyseg=info->s->keyinfo[keynr].seg ; keyseg->base.type ;keyseg++) + { + if (keyseg->base.flag & HA_SPACE_PACK) + { + length= (uint) (uchar) *key++; + pos= record+keyseg->base.start; + if (keyseg->base.type != (int) HA_KEYTYPE_NUM) + { + memcpy(pos,key,(size_t) length); + bfill(pos+length,keyseg->base.length-length,' '); + } + else + { + bfill(pos,keyseg->base.length-length,' '); + memcpy(pos+keyseg->base.length-length,key,(size_t) length); + } + key+=length; + } + else + { + memcpy(record+keyseg->base.start,(byte*) key, + (size_t) keyseg->base.length); + key+= keyseg->base.length; + } + } + DBUG_VOID_RETURN; +} /* _nisam_put_key_in_record */ + + + /* Here when key reads are used */ + +int _nisam_read_key_record(N_INFO *info, ulong filepos, byte *buf) +{ + VOID(_nisam_writeinfo(info,0)); + if (filepos != NI_POS_ERROR) + { + if (info->lastinx >= 0) + { /* Read only key */ + _nisam_put_key_in_record(info,(uint) info->lastinx,buf); + info->update|= HA_STATE_AKTIV; /* We should find a record */ + return 0; + } + my_errno=HA_ERR_WRONG_INDEX; + return(-1); + } + return(-1); /* Wrong data to read */ +} diff --git a/isam/_locking.c b/isam/_locking.c new file mode 100644 index 00000000000..ca38c611812 --- /dev/null +++ b/isam/_locking.c @@ -0,0 +1,345 @@ +/* Copyright (C) 2000 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 */ + +/* + locking of isam-tables. + reads info from a isam-table. Must be first request before doing any furter + calls to any isamfunktion. Is used to allow many process use the same + isamdatabase. + */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <errno.h> +#endif + + /* lock table by F_UNLCK, F_RDLCK or F_WRLCK */ + +int nisam_lock_database(N_INFO *info, int lock_type) +{ + int error; + uint count; + ISAM_SHARE *share; + uint flag; + DBUG_ENTER("nisam_lock_database"); + + flag=error=0; +#ifndef NO_LOCKING + share=info->s; + if (share->base.options & HA_OPTION_READ_ONLY_DATA || + info->lock_type == lock_type) + DBUG_RETURN(0); + pthread_mutex_lock(&share->intern_lock); + switch (lock_type) { + case F_UNLCK: + if (info->lock_type == F_RDLCK) + count= --share->r_locks; + else + count= --share->w_locks; + if (info->lock_type == F_WRLCK && !share->w_locks && + flush_key_blocks(share->kfile,FLUSH_KEEP)) + error=my_errno; + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + if (end_io_cache(&info->rec_cache)) + error=my_errno; + + if (!count) + { + if (share->changed && !share->w_locks) + { + share->state.process= share->last_process=share->this_process; + share->state.loop= info->last_loop= ++info->this_loop; + share->state.uniq= info->last_uniq= info->this_uniq; + if (my_pwrite(share->kfile,(char*) &share->state.header, + share->state_length,0L,MYF(MY_NABP))) + error=my_errno; + share->changed=0; +#ifdef __WIN__ + if (nisam_flush) + { + _commit(share->kfile); + _commit(info->dfile); + } + else + share->not_flushed=1; +#endif + } + if (share->r_locks) + { /* Only read locks left */ + flag=1; + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + error=my_errno; + } + else if (!share->w_locks) + { /* No more locks */ + flag=1; + if (my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + error=my_errno; + } + } + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + info->lock_type= F_UNLCK; + break; + case F_RDLCK: + if (info->lock_type == F_WRLCK) + { /* Change RW to READONLY */ + if (share->w_locks == 1) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE))) + { + error=my_errno; + break; + } + } + share->w_locks--; + share->r_locks++; + info->lock_type=lock_type; + break; + } + if (!share->r_locks && !share->w_locks) + { + flag=1; +#ifdef HAVE_FCNTL + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF,info->lock_wait)) + { + error=my_errno; + break; + } + if (my_pread(share->kfile, + (char*) &share->state.header,share->state_length,0L, + MYF(MY_NABP))) + { + error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); + my_errno=error; + break; + } +#else + VOID(my_seek(share->kfile,0L,MY_SEEK_SET,MYF(0))); + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF,info->lock_wait)) + { + error=my_errno; + break; + } + if (my_read(share->kfile, + (char*) &share->state.header,share->state_length, + MYF(MY_NABP))) + { + error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF,info->lock_wait)); + my_errno=error; + break; + } +#endif + } + VOID(_nisam_test_if_changed(info)); + share->r_locks++; + info->lock_type=lock_type; + break; + case F_WRLCK: + if (info->lock_type == F_RDLCK) + { /* Change RW to READONLY */ + if (share->r_locks == 1) + { + flag=1; + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF, + MYF(info->lock_wait | MY_SEEK_NOT_DONE))) + { + error=my_errno; + break; + } + share->r_locks--; + share->w_locks++; + info->lock_type=lock_type; + break; + } + } + if (!(share->base.options & HA_OPTION_READ_ONLY_DATA) && !share->w_locks) + { + flag=1; + VOID(my_seek(share->kfile,0L,MY_SEEK_SET,MYF(0))); + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF,info->lock_wait)) + { + error=my_errno; + break; + } + if (!share->r_locks) + { + if (my_read(share->kfile, + (char*) &share->state.header,share->state_length, + MYF(MY_NABP))) + { + error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF,info->lock_wait)); + my_errno=error; + break; + } + } + } + VOID(_nisam_test_if_changed(info)); + info->lock_type=lock_type; + share->w_locks++; + break; + default: + break; /* Impossible */ + } + pthread_mutex_unlock(&share->intern_lock); +#if defined(FULL_LOG) || defined(_lint) + lock_type|=(int) (flag << 8); /* Set bit to set if real lock */ + nisam_log_command(LOG_LOCK,info,(byte*) &lock_type,sizeof(lock_type), + error); +#endif +#endif + DBUG_RETURN(error); +} /* nisam_lock_database */ + + + /* Is used before access to database is granted */ + +int _nisam_readinfo(register N_INFO *info, int lock_type, int check_keybuffer) +{ + ISAM_SHARE *share; + DBUG_ENTER("_nisam_readinfo"); + + share=info->s; + if (info->lock_type == F_UNLCK) + { + if (!share->r_locks && !share->w_locks) + { +#ifndef HAVE_FCNTL + VOID(my_seek(share->kfile,0L,MY_SEEK_SET,MYF(0))); +#endif +#ifndef NO_LOCKING +#ifdef UNSAFE_LOCKING + if ((info->tmp_lock_type=lock_type) != F_RDLCK) +#endif + if (my_lock(share->kfile,lock_type,0L,F_TO_EOF,info->lock_wait)) + DBUG_RETURN(1); +#endif +#ifdef HAVE_FCNTL + if (my_pread(share->kfile, + (char*) &share->state.header,share->state_length,0L, + MYF(MY_NABP))) +#else + if (my_read(share->kfile, + (char*) &share->state.header,share->state_length, + MYF(MY_NABP))) +#endif + { +#ifndef NO_LOCKING + int error=my_errno; + VOID(my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE))); + my_errno=error; +#endif + DBUG_RETURN(1); + } + } + if (check_keybuffer) + VOID(_nisam_test_if_changed(info)); + } + else if (lock_type == F_WRLCK && info->lock_type == F_RDLCK) + { + my_errno=EACCES; /* Not allowed to change */ + DBUG_RETURN(-1); /* when have read_lock() */ + } + DBUG_RETURN(0); +} /* _nisam_readinfo */ + + + /* Every isam-function that uppdates the isam-database must! end */ + /* with this request */ + /* ARGSUSED */ + +int _nisam_writeinfo(register N_INFO *info, uint flags) +{ + int error,olderror; + ISAM_SHARE *share; + DBUG_ENTER("_nisam_writeinfo"); + + error=0; + share=info->s; + if (share->r_locks == 0 && share->w_locks == 0) + { + olderror=my_errno; /* Remember last error */ + if (flags) + { /* Two threads can't be here */ + share->state.process= share->last_process= share->this_process; + share->state.loop= info->last_loop= ++info->this_loop; + share->state.uniq= info->last_uniq= info->this_uniq; + if ((error=my_pwrite(share->kfile,(char*) &share->state.header, + share->state_length,0L,MYF(MY_NABP)) != 0)) + olderror=my_errno; +#ifdef __WIN__ + if (nisam_flush) + { + _commit(share->kfile); + _commit(info->dfile); + } +#endif + } + if (flags != 2) + { +#ifndef NO_LOCKING +#ifdef UNSAFE_LOCKING + if (info->tmp_lock_type != F_RDLCK) +#endif + { + if (my_lock(share->kfile,F_UNLCK,0L,F_TO_EOF, + MYF(MY_WME | MY_SEEK_NOT_DONE)) && !error) + DBUG_RETURN(1); + } + } +#endif + my_errno=olderror; + } + else if (flags) + share->changed= 1; /* Mark keyfile changed */ + DBUG_RETURN(error); +} /* _nisam_writeinfo */ + + + /* Test if someone has changed the database */ + /* (Should be called after readinfo) */ + +int _nisam_test_if_changed(register N_INFO *info) +{ +#ifndef NO_LOCKING + { + ISAM_SHARE *share=info->s; + if (share->state.process != share->last_process || + share->state.loop != info->last_loop || + share->state.uniq != info->last_uniq) + { /* Keyfile has changed */ + if (share->state.process != share->this_process) + VOID(flush_key_blocks(share->kfile,FLUSH_RELEASE)); + share->last_process=share->state.process; + info->last_loop= share->state.loop; + info->last_uniq= share->state.uniq; + info->update|= HA_STATE_WRITTEN; /* Must use file on next */ + info->data_changed= 1; /* For nisam_is_changed */ + return 1; + } + } +#endif + return (!(info->update & HA_STATE_AKTIV) || + (info->update & (HA_STATE_WRITTEN | HA_STATE_DELETED | + HA_STATE_KEY_CHANGED))); +} /* _nisam_test_if_changed */ diff --git a/isam/_packrec.c b/isam/_packrec.c new file mode 100644 index 00000000000..5c387f011ad --- /dev/null +++ b/isam/_packrec.c @@ -0,0 +1,1184 @@ +/* Copyright (C) 2000 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 compressed records */ + +#include "isamdef.h" + +#define IS_CHAR ((uint) 32768) /* Bit if char (not offset) in tree */ + +#if INT_MAX > 65536L +#define BITS_SAVED 32 +#define MAX_QUICK_TABLE_BITS 9 /* Because we may shift in 24 bits */ +#else +#define BITS_SAVED 16 +#define MAX_QUICK_TABLE_BITS 6 +#endif + +#define get_bit(BU) ((BU)->bits ? \ + (BU)->current_byte & ((bit_type) 1 << --(BU)->bits) :\ + (fill_buffer(BU), (BU)->bits= BITS_SAVED-1,\ + (BU)->current_byte & ((bit_type) 1 << (BITS_SAVED-1)))) +#define skipp_to_next_byte(BU) ((BU)->bits&=~7) +#define get_bits(BU,count) (((BU)->bits >= count) ? (((BU)->current_byte >> ((BU)->bits-=count)) & mask[count]) : fill_and_get_bits(BU,count)) + +#define decode_bytes_test_bit(bit) \ + if (low_byte & (1 << (7-bit))) \ + pos++; \ + if (*pos & IS_CHAR) \ + { bits-=(bit+1); break; } \ + pos+= *pos + + + static void read_huff_table(BIT_BUFF *bit_buff,DECODE_TREE *decode_tree, + uint16 **decode_table,byte **intervall_buff, + uint16 *tmp_buff); +static void make_quick_table(uint16 *to_table,uint16 *decode_table, + uint *next_free,uint value,uint bits, + uint max_bits); +static void fill_quick_table(uint16 *table,uint bits, uint max_bits, + uint value); +static uint copy_decode_table(uint16 *to_pos,uint offset, + uint16 *decode_table); +static uint find_longest_bitstream(uint16 *table); +static void (*get_unpack_function(N_RECINFO *rec))(N_RECINFO *field, + BIT_BUFF *buff, + uchar *to, + uchar *end); +static void uf_zerofill_skipp_zero(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_skipp_zero(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_normal(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_endspace_selected(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_endspace_selected(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_endspace(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_endspace(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_prespace_selected(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to, uchar *end); +static void uf_prespace_selected(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_space_prespace(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_prespace(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_zerofill_normal(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_constant(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_intervall(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void uf_zero(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static void decode_bytes(N_RECINFO *rec,BIT_BUFF *bit_buff, + uchar *to,uchar *end); +static uint decode_pos(BIT_BUFF *bit_buff,DECODE_TREE *decode_tree); +static void init_bit_buffer(BIT_BUFF *bit_buff,char *buffer,uint length); +static uint fill_and_get_bits(BIT_BUFF *bit_buff,uint count); +static void fill_buffer(BIT_BUFF *bit_buff); +static uint max_bit(uint value); +#ifdef HAVE_MMAP +static void _nisam_mempack_get_block_info(BLOCK_INFO *info,uint ref_length, + uchar *header); +#endif + +static uint mask[]= +{ + 0x00000000, + 0x00000001, 0x00000003, 0x00000007, 0x0000000f, + 0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, + 0x000001ff, 0x000003ff, 0x000007ff, 0x00000fff, + 0x00001fff, 0x00003fff, 0x00007fff, 0x0000ffff, +#if BITS_SAVED > 16 + 0x0001ffff, 0x0003ffff, 0x0007ffff, 0x000fffff, + 0x001fffff, 0x003fffff, 0x007fffff, 0x00ffffff, + 0x01ffffff, 0x03ffffff, 0x07ffffff, 0x0fffffff, + 0x1fffffff, 0x3fffffff, 0x7fffffff, 0xffffffff, +#endif + }; + + + /* Read all packed info, allocate memory and fix field structs */ + +my_bool _nisam_read_pack_info(N_INFO *info, pbool fix_keys) +{ + File file; + int diff_length; + uint i,trees,huff_tree_bits,rec_reflength,length; + uint16 *decode_table,*tmp_buff; + ulong elements,intervall_length; + char *disk_cache,*intervall_buff; + uchar header[32]; + ISAM_SHARE *share=info->s; + BIT_BUFF bit_buff; + DBUG_ENTER("_nisam_read_pack_info"); + + if (nisam_quick_table_bits < 4) + nisam_quick_table_bits=4; + else if (nisam_quick_table_bits > MAX_QUICK_TABLE_BITS) + nisam_quick_table_bits=MAX_QUICK_TABLE_BITS; + + file=info->dfile; + my_errno=0; + if (my_read(file,(byte*) header,sizeof(header),MYF(MY_NABP))) + { + if (!my_errno) + my_errno=HA_ERR_END_OF_FILE; + DBUG_RETURN(1); + } + if (memcmp((byte*) header,(byte*) nisam_pack_file_magic,4)) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(1); + } + share->pack.header_length=uint4korr(header+4); + share->min_pack_length=(uint) uint4korr(header+8); + share->max_pack_length=(uint) uint4korr(header+12); + set_if_bigger(share->base.pack_reclength,share->max_pack_length); + elements=uint4korr(header+16); + intervall_length=uint4korr(header+20); + trees=uint2korr(header+24); + share->pack.ref_length=header[26]; + rec_reflength=header[27]; + diff_length=(int) rec_reflength - (int) share->base.rec_reflength; + if (fix_keys) + share->rec_reflength=rec_reflength; + share->base.min_block_length=share->min_pack_length+share->pack.ref_length; + + if (!(share->decode_trees=(DECODE_TREE*) + my_malloc((uint) (trees*sizeof(DECODE_TREE)+ + intervall_length*sizeof(byte)), + MYF(MY_WME)))) + DBUG_RETURN(1); + intervall_buff=(byte*) (share->decode_trees+trees); + + length=(uint) (elements*2+trees*(1 << nisam_quick_table_bits)); + if (!(share->decode_tables=(uint16*) + my_malloc((length+512)*sizeof(uint16)+ + (uint) (share->pack.header_length+7), + MYF(MY_WME | MY_ZEROFILL)))) + { + my_free((gptr) share->decode_trees,MYF(0)); + DBUG_RETURN(1); + } + tmp_buff=share->decode_tables+length; + disk_cache=(byte*) (tmp_buff+512); + + if (my_read(file,disk_cache, + (uint) (share->pack.header_length-sizeof(header)), + MYF(MY_NABP))) + { + my_free((gptr) share->decode_trees,MYF(0)); + my_free((gptr) share->decode_tables,MYF(0)); + DBUG_RETURN(1); + } + + huff_tree_bits=max_bit(trees ? trees-1 : 0); + init_bit_buffer(&bit_buff,disk_cache, + (uint) (share->pack.header_length-sizeof(header))); + /* Read new info for each field */ + for (i=0 ; i < share->base.fields ; i++) + { + share->rec[i].base_type=(enum en_fieldtype) get_bits(&bit_buff,4); + share->rec[i].pack_type=(uint) get_bits(&bit_buff,4); + share->rec[i].space_length_bits=get_bits(&bit_buff,4); + share->rec[i].huff_tree=share->decode_trees+(uint) get_bits(&bit_buff, + huff_tree_bits); + share->rec[i].unpack=get_unpack_function(share->rec+i); + } + skipp_to_next_byte(&bit_buff); + decode_table=share->decode_tables; + for (i=0 ; i < trees ; i++) + read_huff_table(&bit_buff,share->decode_trees+i,&decode_table, + &intervall_buff,tmp_buff); + decode_table=(uint16*) + my_realloc((gptr) share->decode_tables, + (uint) ((byte*) decode_table - (byte*) share->decode_tables), + MYF(MY_HOLD_ON_ERROR)); + { + my_ptrdiff_t diff=PTR_BYTE_DIFF(decode_table,share->decode_tables); + share->decode_tables=decode_table; + for (i=0 ; i < trees ; i++) + share->decode_trees[i].table=ADD_TO_PTR(share->decode_trees[i].table, + diff, uint16*); + } + + /* Fix record-ref-length for keys */ + if (fix_keys) + { + for (i=0 ; i < share->base.keys ; i++) + { + share->keyinfo[i].base.keylength+=(uint16) diff_length; + share->keyinfo[i].base.minlength+=(uint16) diff_length; + share->keyinfo[i].base.maxlength+=(uint16) diff_length; + share->keyinfo[i].seg[share->keyinfo[i].base.keysegs].base.length= + (uint16) rec_reflength; + } + } + + if (bit_buff.error || bit_buff.pos < bit_buff.end) + { /* info_length was wrong */ + my_errno=HA_ERR_WRONG_IN_RECORD; + my_free((gptr) share->decode_trees,MYF(0)); + my_free((gptr) share->decode_tables,MYF(0)); + DBUG_RETURN(1); + } + DBUG_RETURN(0); +} + + + /* Read on huff-code-table from datafile */ + +static void read_huff_table(BIT_BUFF *bit_buff, DECODE_TREE *decode_tree, + uint16 **decode_table, byte **intervall_buff, + uint16 *tmp_buff) +{ + uint min_chr,elements,char_bits,offset_bits,size,intervall_length,table_bits, + next_free_offset; + uint16 *ptr,*end; + + LINT_INIT(ptr); + if (!get_bits(bit_buff,1)) + { + min_chr=get_bits(bit_buff,8); + elements=get_bits(bit_buff,9); + char_bits=get_bits(bit_buff,5); + offset_bits=get_bits(bit_buff,5); + intervall_length=0; + ptr=tmp_buff; + } + else + { + min_chr=0; + elements=get_bits(bit_buff,15); + intervall_length=get_bits(bit_buff,16); + char_bits=get_bits(bit_buff,5); + offset_bits=get_bits(bit_buff,5); + decode_tree->quick_table_bits=0; + ptr= *decode_table; + } + size=elements*2-2; + + for (end=ptr+size ; ptr < end ; ptr++) + { + if (get_bit(bit_buff)) + *ptr= (uint16) get_bits(bit_buff,offset_bits); + else + *ptr= (uint16) (IS_CHAR + (get_bits(bit_buff,char_bits) + min_chr)); + } + skipp_to_next_byte(bit_buff); + + decode_tree->table= *decode_table; + decode_tree->intervalls= *intervall_buff; + if (! intervall_length) + { + table_bits=find_longest_bitstream(tmp_buff); + if (table_bits > nisam_quick_table_bits) + table_bits=nisam_quick_table_bits; + next_free_offset= (1 << table_bits); + make_quick_table(*decode_table,tmp_buff,&next_free_offset,0,table_bits, + table_bits); + (*decode_table)+= next_free_offset; + decode_tree->quick_table_bits=table_bits; + } + else + { + (*decode_table)=end; + bit_buff->pos-= bit_buff->bits/8; + memcpy(*intervall_buff,bit_buff->pos,(size_t) intervall_length); + (*intervall_buff)+=intervall_length; + bit_buff->pos+=intervall_length; + bit_buff->bits=0; + } + return; +} + + +static void make_quick_table(uint16 *to_table, uint16 *decode_table, uint *next_free_offset, uint value, uint bits, uint max_bits) +{ + if (!bits--) + { + to_table[value]= (uint16) *next_free_offset; + *next_free_offset=copy_decode_table(to_table, *next_free_offset, + decode_table); + return; + } + if (!(*decode_table & IS_CHAR)) + { + make_quick_table(to_table,decode_table+ *decode_table, + next_free_offset,value,bits,max_bits); + } + else + fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); + decode_table++; + value|= (1 << bits); + if (!(*decode_table & IS_CHAR)) + { + make_quick_table(to_table,decode_table+ *decode_table, + next_free_offset,value,bits,max_bits); + } + else + fill_quick_table(to_table+value,bits,max_bits,(uint) *decode_table); + return; +} + + +static void fill_quick_table(uint16 *table, uint bits, uint max_bits, uint value) +{ + uint16 *end; + value|=(max_bits-bits) << 8; + for (end=table+ (1 << bits) ; + table < end ; + *table++ = (uint16) value | IS_CHAR) ; +} + + +static uint copy_decode_table(uint16 *to_pos, uint offset, uint16 *decode_table) +{ + uint prev_offset; + prev_offset= offset; + + if (!(*decode_table & IS_CHAR)) + { + to_pos[offset]=2; + offset=copy_decode_table(to_pos,offset+2,decode_table+ *decode_table); + } + else + { + to_pos[offset]= *decode_table; + offset+=2; + } + decode_table++; + + if (!(*decode_table & IS_CHAR)) + { + to_pos[prev_offset+1]=(uint16) (offset-prev_offset-1); + offset=copy_decode_table(to_pos,offset,decode_table+ *decode_table); + } + else + to_pos[prev_offset+1]= *decode_table; + return offset; +} + + +static uint find_longest_bitstream(uint16 *table) +{ + uint length=1,length2; + if (!(*table & IS_CHAR)) + length=find_longest_bitstream(table+ *table)+1; + table++; + if (!(*table & IS_CHAR)) + { + length2=find_longest_bitstream(table+ *table)+1; + length=max(length,length2); + } + return length; +} + + + /* Read record from datafile */ + /* Returns length of packed record, -1 if error */ + +int _nisam_read_pack_record(N_INFO *info, ulong filepos, byte *buf) +{ + BLOCK_INFO block_info; + File file; + DBUG_ENTER("_nisam_read_pack_record"); + + if (filepos == NI_POS_ERROR) + DBUG_RETURN(-1); /* _search() didn't find record */ + + file=info->dfile; + if (_nisam_pack_get_block_info(&block_info,info->s->pack.ref_length,file, + filepos)) + goto err; + if (my_read(file,(byte*) info->rec_buff,block_info.rec_len,MYF(MY_NABP))) + goto panic; + info->update|= HA_STATE_AKTIV; + DBUG_RETURN(_nisam_pack_rec_unpack(info,buf,info->rec_buff, + block_info.rec_len)); +panic: + my_errno=HA_ERR_WRONG_IN_RECORD; +err: + DBUG_RETURN(-1); +} + + + +int _nisam_pack_rec_unpack(register N_INFO *info, register byte *to, + byte *from, uint reclength) +{ + byte *end_field; + reg3 N_RECINFO *end; + N_RECINFO *current_field; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_pack_rec_unpack"); + + init_bit_buffer(&info->bit_buff,from,reclength); + + for (current_field=share->rec, end=current_field+share->base.fields ; + current_field < end ; + current_field++,to=end_field) + { + end_field=to+current_field->base.length; + (*current_field->unpack)(current_field,&info->bit_buff,(uchar*) to, + (uchar*) end_field); + } + if (! info->bit_buff.error && + info->bit_buff.pos - info->bit_buff.bits/8 == info->bit_buff.end) + DBUG_RETURN(0); + my_errno=HA_ERR_WRONG_IN_RECORD; + info->update&= ~HA_STATE_AKTIV; + DBUG_RETURN(-1); +} /* _nisam_pack_rec_unpack */ + + + /* Return function to unpack field */ + +static void (*get_unpack_function(N_RECINFO *rec))(N_RECINFO *, BIT_BUFF *, uchar *, uchar *) +{ + switch (rec->base_type) { + case FIELD_SKIPP_ZERO: + if (rec->pack_type & PACK_TYPE_ZERO_FILL) + return &uf_zerofill_skipp_zero; + return &uf_skipp_zero; + case FIELD_NORMAL: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + return &uf_space_normal; + if (rec->pack_type & PACK_TYPE_ZERO_FILL) + return &uf_zerofill_normal; + return &decode_bytes; + case FIELD_SKIPP_ENDSPACE: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + { + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_space_endspace_selected; + return &uf_space_endspace; + } + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_endspace_selected; + return &uf_endspace; + case FIELD_SKIPP_PRESPACE: + if (rec->pack_type & PACK_TYPE_SPACE_FIELDS) + { + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_space_prespace_selected; + return &uf_space_prespace; + } + if (rec->pack_type & PACK_TYPE_SELECTED) + return &uf_prespace_selected; + return &uf_prespace; + case FIELD_CONSTANT: + return &uf_constant; + case FIELD_INTERVALL: + return &uf_intervall; + case FIELD_ZERO: + return &uf_zero; + case FIELD_BLOB: /* Write this sometimes.. */ + case FIELD_LAST: + default: + return 0; /* This should never happend */ + } +} + + /* De different functions to unpack a field */ + +static void uf_zerofill_skipp_zero(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + if (get_bit(bit_buff)) + bzero((char*) to,(uint) (end-to)); + else + { +#ifdef WORDS_BIGENDIAN + bzero((char*) to,rec->space_length_bits); + decode_bytes(rec,bit_buff,to+rec->space_length_bits,end); +#else + end-=rec->space_length_bits; + decode_bytes(rec,bit_buff,to,end); + bzero((byte*) end,rec->space_length_bits); +#endif + } +} + +static void uf_skipp_zero(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + if (get_bit(bit_buff)) + bzero((char*) to,(uint) (end-to)); + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_normal(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_endspace_selected(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } + else + decode_bytes(rec,bit_buff,to,end); + } +} + +static void uf_endspace_selected(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } + else + decode_bytes(rec,bit_buff,to,end); +} + +static void uf_space_endspace(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); + } +} + +static void uf_endspace(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, + uchar *end) +{ + uint spaces; + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + if (to+spaces != end) + decode_bytes(rec,bit_buff,to,end-spaces); + bfill((byte*) end-spaces,spaces,' '); +} + +static void uf_space_prespace_selected(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } + else + decode_bytes(rec,bit_buff,to,end); + } +} + + +static void uf_prespace_selected(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } + else + decode_bytes(rec,bit_buff,to,end); +} + + +static void uf_space_prespace(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if (get_bit(bit_buff)) + bfill((byte*) to,(end-to),' '); + else + { + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); + } +} + +static void uf_prespace(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + uint spaces; + if ((spaces=get_bits(bit_buff,rec->space_length_bits))+to > end) + { + bit_buff->error=1; + return; + } + bfill((byte*) to,spaces,' '); + if (to+spaces != end) + decode_bytes(rec,bit_buff,to+spaces,end); +} + +static void uf_zerofill_normal(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ +#ifdef WORDS_BIGENDIAN + bzero((char*) to,rec->space_length_bits); + decode_bytes(rec,bit_buff,(uchar*) to+rec->space_length_bits,end); +#else + end-=rec->space_length_bits; + decode_bytes(rec,bit_buff,(uchar*) to,end); + bzero((byte*) end,rec->space_length_bits); +#endif +} + +static void uf_constant(N_RECINFO *rec, + BIT_BUFF *bit_buff __attribute__((unused)), + uchar *to, uchar *end) +{ + memcpy(to,rec->huff_tree->intervalls,(size_t) (end-to)); +} + +static void uf_intervall(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + reg1 uint field_length=(uint) (end-to); + memcpy(to,rec->huff_tree->intervalls+field_length*decode_pos(bit_buff, + rec->huff_tree), + (size_t) field_length); +} + + +/*ARGSUSED*/ +static void uf_zero(N_RECINFO *rec __attribute__((unused)), + BIT_BUFF *bit_buff __attribute__((unused)), + uchar *to, uchar *end) +{ + bzero((char*) to,(uint) (end-to)); +} + + + /* Functions to decode of buffer of bits */ + +#if BITS_SAVED == 64 + +static void decode_bytes(rec,bit_buff,to,end) +N_RECINFO *rec; +BIT_BUFF *bit_buff; +uchar *to,*end; +{ + reg1 uint bits,low_byte; + reg3 uint16 *pos; + reg4 uint table_bits,table_and; + DECODE_TREE *decode_tree; + + decode_tree=rec->decode_tree; + bits=bit_buff->bits; /* Save in reg for quicker access */ + table_bits=decode_tree->quick_table_bits; + table_and= (1 << table_bits)-1; + + do + { + if (bits <= 32) + { + if (bit_buff->pos > bit_buff->end+4) + return; /* Can't be right */ + bit_buff->current_byte= (bit_buff->current_byte << 32) + + ((((uint) bit_buff->pos[3])) + + (((uint) bit_buff->pos[2]) << 8) + + (((uint) bit_buff->pos[1]) << 16) + + (((uint) bit_buff->pos[0]) << 24)); + bit_buff->pos+=4; + bits+=32; + } + /* First use info in quick_table */ + low_byte=(uint) (bit_buff->current_byte >> (bits - table_bits)) & table_and; + low_byte=decode_tree->table[low_byte]; + if (low_byte & IS_CHAR) + { + *to++ = (low_byte & 255); /* Found char in quick table */ + bits-= ((low_byte >> 8) & 31); /* Remove bits used */ + } + else + { /* Map through rest of decode-table */ + pos=decode_tree->table+low_byte; + bits-=table_bits; + for (;;) + { + low_byte=(uint) (bit_buff->current_byte >> (bits-8)); + decode_bytes_test_bit(0); + decode_bytes_test_bit(1); + decode_bytes_test_bit(2); + decode_bytes_test_bit(3); + decode_bytes_test_bit(4); + decode_bytes_test_bit(5); + decode_bytes_test_bit(6); + decode_bytes_test_bit(7); + bits-=8; + } + *to++ = *pos; + } + } while (to != end); + + bit_buff->bits=bits; + return; +} + +#else + +static void decode_bytes(N_RECINFO *rec, BIT_BUFF *bit_buff, uchar *to, uchar *end) +{ + reg1 uint bits,low_byte; + reg3 uint16 *pos; + reg4 uint table_bits,table_and; + DECODE_TREE *decode_tree; + + decode_tree=rec->huff_tree; + bits=bit_buff->bits; /* Save in reg for quicker access */ + table_bits=decode_tree->quick_table_bits; + table_and= (1 << table_bits)-1; + + do + { + if (bits < table_bits) + { + if (bit_buff->pos > bit_buff->end+1) + return; /* Can't be right */ +#if BITS_SAVED == 32 + bit_buff->current_byte= (bit_buff->current_byte << 24) + + (((uint) ((uchar) bit_buff->pos[2]))) + + (((uint) ((uchar) bit_buff->pos[1])) << 8) + + (((uint) ((uchar) bit_buff->pos[0])) << 16); + bit_buff->pos+=3; + bits+=24; +#else + if (bits) /* We must have at leasts 9 bits */ + { + bit_buff->current_byte= (bit_buff->current_byte << 8) + + (uint) ((uchar) bit_buff->pos[0]); + bit_buff->pos++; + bits+=8; + } + else + { + bit_buff->current_byte= ((uint) ((uchar) bit_buff->pos[0]) << 8) + + ((uint) ((uchar) bit_buff->pos[1])); + bit_buff->pos+=2; + bits+=16; + } +#endif + } + /* First use info in quick_table */ + low_byte=(bit_buff->current_byte >> (bits - table_bits)) & table_and; + low_byte=decode_tree->table[low_byte]; + if (low_byte & IS_CHAR) + { + *to++ = (low_byte & 255); /* Found char in quick table */ + bits-= ((low_byte >> 8) & 31); /* Remove bits used */ + } + else + { /* Map through rest of decode-table */ + pos=decode_tree->table+low_byte; + bits-=table_bits; + for (;;) + { + if (bits < 8) + { /* We don't need to check end */ +#if BITS_SAVED == 32 + bit_buff->current_byte= (bit_buff->current_byte << 24) + + (((uint) ((uchar) bit_buff->pos[2]))) + + (((uint) ((uchar) bit_buff->pos[1])) << 8) + + (((uint) ((uchar) bit_buff->pos[0])) << 16); + bit_buff->pos+=3; + bits+=24; +#else + bit_buff->current_byte= (bit_buff->current_byte << 8) + + (uint) ((uchar) bit_buff->pos[0]); + bit_buff->pos+=1; + bits+=8; +#endif + } + low_byte=(uint) (bit_buff->current_byte >> (bits-8)); + decode_bytes_test_bit(0); + decode_bytes_test_bit(1); + decode_bytes_test_bit(2); + decode_bytes_test_bit(3); + decode_bytes_test_bit(4); + decode_bytes_test_bit(5); + decode_bytes_test_bit(6); + decode_bytes_test_bit(7); + bits-=8; + } + *to++ = (uchar) *pos; + } + } while (to != end); + + bit_buff->bits=bits; + return; +} +#endif /* BIT_SAVED == 64 */ + + +static uint decode_pos(BIT_BUFF *bit_buff, DECODE_TREE *decode_tree) +{ + uint16 *pos=decode_tree->table; + for (;;) + { + if (get_bit(bit_buff)) + pos++; + if (*pos & IS_CHAR) + return (uint) (*pos & ~IS_CHAR); + pos+= *pos; + } +} + + +int _nisam_read_rnd_pack_record(N_INFO *info, byte *buf, + register ulong filepos, + int skipp_deleted_blocks) +{ + uint b_type; + BLOCK_INFO block_info; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_read_rnd_pack_record"); + + if (filepos >= share->state.data_file_length) + { + my_errno= HA_ERR_END_OF_FILE; + goto err; + } + + if (info->opt_flag & READ_CACHE_USED) + { + if (_nisam_read_cache(&info->rec_cache,(byte*) block_info.header,filepos, + share->pack.ref_length, skipp_deleted_blocks)) + goto err; + b_type=_nisam_pack_get_block_info(&block_info,share->pack.ref_length,-1, + filepos); + } + else + b_type=_nisam_pack_get_block_info(&block_info,share->pack.ref_length, + info->dfile,filepos); + if (b_type) + goto err; +#ifndef DBUG_OFF + if (block_info.rec_len > share->max_pack_length) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } +#endif + if (info->opt_flag & READ_CACHE_USED) + { + if (_nisam_read_cache(&info->rec_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len, + skipp_deleted_blocks)) + goto err; + } + else + { + if (my_read(info->dfile,(byte*) info->rec_buff,block_info.rec_len, + MYF(MY_NABP))) + goto err; + } + info->packed_length=block_info.rec_len; + info->lastpos=filepos; + info->nextpos=block_info.filepos+block_info.rec_len; + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + + DBUG_RETURN (_nisam_pack_rec_unpack(info,buf,info->rec_buff, + block_info.rec_len)); +err: + DBUG_RETURN(-1); +} + + + /* Read and process header from a huff-record-file */ + +uint _nisam_pack_get_block_info(BLOCK_INFO *info, uint ref_length, File file, + ulong filepos) +{ + uchar *header=info->header; + + if (file >= 0) + { + VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0))); + if (my_read(file,(char*) header,ref_length,MYF(MY_NABP))) + return BLOCK_FATAL_ERROR; + } + DBUG_DUMP("header",(byte*) header,ref_length); + + switch (ref_length) { + case 1: + info->rec_len=header[0]; + info->filepos=filepos+1; + break; + case 2: + info->rec_len=uint2korr(header); + info->filepos=filepos+2; + break; + case 3: + info->rec_len=(uint) (uint3korr(header)); + info->filepos=filepos+3; + break; + default: break; + } + return 0; +} + + + /* routines for bit buffer */ + /* Buffer must be 6 byte bigger */ +static void init_bit_buffer(BIT_BUFF *bit_buff, char *buffer, uint length) +{ + bit_buff->pos=(uchar*) buffer; + bit_buff->end=(uchar*) buffer+length; + bit_buff->bits=bit_buff->error=0; + bit_buff->current_byte=0; /* Avoid purify errors */ +} + +static uint fill_and_get_bits(BIT_BUFF *bit_buff, uint count) +{ + uint tmp; + count-=bit_buff->bits; + tmp=(bit_buff->current_byte & mask[bit_buff->bits]) << count; + fill_buffer(bit_buff); + bit_buff->bits=BITS_SAVED - count; + return tmp+(bit_buff->current_byte >> (BITS_SAVED - count)); +} + + /* Fill in empty bit_buff->current_byte from buffer */ + /* Sets bit_buff->error if buffer is exhausted */ + +static void fill_buffer(BIT_BUFF *bit_buff) +{ + if (bit_buff->pos >= bit_buff->end) + { + bit_buff->error= 1; + bit_buff->current_byte=0; + return; + } +#if BITS_SAVED == 64 + bit_buff->current_byte= ((((uint) ((uchar) bit_buff->pos[7]))) + + (((uint) ((uchar) bit_buff->pos[6])) << 8) + + (((uint) ((uchar) bit_buff->pos[5])) << 16) + + (((uint) ((uchar) bit_buff->pos[4])) << 24) + + ((ulonglong) + ((((uint) ((uchar) bit_buff->pos[3]))) + + (((uint) ((uchar) bit_buff->pos[2])) << 8) + + (((uint) ((uchar) bit_buff->pos[1])) << 16) + + (((uint) ((uchar) bit_buff->pos[0])) << 24)) << 32)); + bit_buff->pos+=8; +#else +#if BITS_SAVED == 32 + bit_buff->current_byte= (((uint) ((uchar) bit_buff->pos[3])) + + (((uint) ((uchar) bit_buff->pos[2])) << 8) + + (((uint) ((uchar) bit_buff->pos[1])) << 16) + + (((uint) ((uchar) bit_buff->pos[0])) << 24)); + bit_buff->pos+=4; +#else + bit_buff->current_byte= (uint) (((uint) ((uchar) bit_buff->pos[1]))+ + (((uint) ((uchar) bit_buff->pos[0])) << 8)); + bit_buff->pos+=2; +#endif +#endif +} + + /* Get number of bits neaded to represent value */ + +static uint max_bit(register uint value) +{ + reg2 uint power=1; + + while ((value>>=1)) + power++; + return (power); +} + + +/***************************************************************************** + Some redefined functions to handle files when we are using memmap +*****************************************************************************/ + +#ifdef HAVE_MMAP + +#include <sys/mman.h> + +static int _nisam_read_mempack_record(N_INFO *info,ulong filepos,byte *buf); +static int _nisam_read_rnd_mempack_record(N_INFO*, byte *,ulong, int); + +#ifndef MAP_NORESERVE +#define MAP_NORESERVE 0 /* For irix */ +#endif +#ifndef MAP_FAILED +#define MAP_FAILED -1 +#endif + +my_bool _nisam_memmap_file(N_INFO *info) +{ + byte *file_map; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_memmap_file"); + + if (!info->s->file_map) + { + if (my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)) < + share->state.data_file_length+MEMMAP_EXTRA_MARGIN) + { + DBUG_PRINT("warning",("File isn't extended for memmap")); + DBUG_RETURN(0); + } + file_map=(byte*) + mmap(0,share->state.data_file_length+MEMMAP_EXTRA_MARGIN,PROT_READ, + MAP_SHARED | MAP_NORESERVE,info->dfile,0L); + if (file_map == (byte*) MAP_FAILED) + { + DBUG_PRINT("warning",("mmap failed: errno: %d",errno)); + my_errno=errno; + DBUG_RETURN(0); + } + info->s->file_map=file_map; + } + info->opt_flag|= MEMMAP_USED; + info->read_record=share->read_record=_nisam_read_mempack_record; + share->read_rnd=_nisam_read_rnd_mempack_record; + DBUG_RETURN(1); +} + + +void _nisam_unmap_file(N_INFO *info) +{ + if (info->s->file_map) + (void) munmap((caddr_t) info->s->file_map, + (size_t) info->s->state.data_file_length+ + MEMMAP_EXTRA_MARGIN); +} + + +static void _nisam_mempack_get_block_info(BLOCK_INFO *info, uint ref_length, + uchar *header) +{ + if (ref_length == 1) /* This is most usual */ + info->rec_len=header[0]; + else if (ref_length == 2) + info->rec_len=uint2korr(header); + else + info->rec_len=(uint) (uint3korr(header)); +} + + +static int _nisam_read_mempack_record(N_INFO *info, ulong filepos, byte *buf) +{ + BLOCK_INFO block_info; + ISAM_SHARE *share=info->s; + DBUG_ENTER("ni_read_mempack_record"); + + if (filepos == NI_POS_ERROR) + DBUG_RETURN(-1); /* _search() didn't find record */ + + _nisam_mempack_get_block_info(&block_info,share->pack.ref_length, + (uchar*) share->file_map+filepos); + DBUG_RETURN(_nisam_pack_rec_unpack(info,buf,share->file_map+ + share->pack.ref_length+filepos, + block_info.rec_len)); +} + + +/*ARGSUSED*/ +static int _nisam_read_rnd_mempack_record(N_INFO *info, byte *buf, + register ulong filepos, + int skipp_deleted_blocks + __attribute__((unused))) +{ + BLOCK_INFO block_info; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_read_rnd_mempack_record"); + + if (filepos >= share->state.data_file_length) + { + my_errno=HA_ERR_END_OF_FILE; + goto err; + } + + _nisam_mempack_get_block_info(&block_info,share->pack.ref_length, + (uchar*) share->file_map+filepos); +#ifndef DBUG_OFF + if (block_info.rec_len > info->s->max_pack_length) + { + my_errno=HA_ERR_WRONG_IN_RECORD; + goto err; + } +#endif + info->packed_length=block_info.rec_len; + info->lastpos=filepos; + info->nextpos=filepos+share->pack.ref_length+block_info.rec_len; + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + + DBUG_RETURN (_nisam_pack_rec_unpack(info,buf,share->file_map+ + share->pack.ref_length+filepos, + block_info.rec_len)); +err: + DBUG_RETURN(-1); +} + +#endif /* HAVE_MMAP */ diff --git a/isam/_page.c b/isam/_page.c new file mode 100644 index 00000000000..6f6d632e85d --- /dev/null +++ b/isam/_page.c @@ -0,0 +1,137 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser och skriver nyckelblock */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <errno.h> +#endif + + /* Fetch a key-page in memory */ + +uchar *_nisam_fetch_keypage(register N_INFO *info, N_KEYDEF *keyinfo, + my_off_t page, uchar *buff, int return_buffer) +{ + uchar *tmp; + tmp=(uchar*) key_cache_read(info->s->kfile,page,(byte*) buff, + (uint) keyinfo->base.block_length, + (uint) keyinfo->base.block_length, + return_buffer); + if (tmp == info->buff) + { + info->update|=HA_STATE_BUFF_SAVED; + info->int_pos=(ulong) page; + info->buff_used=1; + } + else + { + info->update&= ~HA_STATE_BUFF_SAVED; + if (tmp) + info->int_pos=(ulong) page; + else + { + info->int_pos=NI_POS_ERROR; + DBUG_PRINT("error",("Got errno: %d from key_cache_read",my_errno)); + my_errno=HA_ERR_CRASHED; + } + } + return tmp; +} /* _nisam_fetch_keypage */ + + + /* Write a key-page on disk */ + +int _nisam_write_keypage(register N_INFO *info, register N_KEYDEF *keyinfo, + my_off_t page, uchar *buff) +{ + reg3 uint length; +#ifndef QQ /* Safety check */ + if (page < info->s->base.keystart || + page+keyinfo->base.block_length > info->s->state.key_file_length || + page & (nisam_block_size-1)) + { + DBUG_PRINT("error",("Trying to write outside key region: %lu", + (long) page)); + my_errno=EINVAL; + return(-1); + } + DBUG_PRINT("page",("write page at: %lu",(long) page,buff)); + DBUG_DUMP("buff",(byte*) buff,getint(buff)); +#endif + + if ((length=keyinfo->base.block_length) > IO_SIZE*2 && + info->s->state.key_file_length != page+length) + length= ((getint(buff)+IO_SIZE-1) & (uint) ~(IO_SIZE-1)); +#ifdef HAVE_purify + { + length=getint(buff); + bzero((byte*) buff+length,keyinfo->base.block_length-length); + length=keyinfo->base.block_length; + } +#endif + return (key_cache_write(info->s->kfile,page,(byte*) buff,length, + (uint) keyinfo->base.block_length, + (int) (info->lock_type != F_UNLCK))); +} /* nisam_write_keypage */ + + + /* Remove page from disk */ + +int _nisam_dispose(register N_INFO *info, N_KEYDEF *keyinfo, my_off_t pos) +{ + uint keynr= (uint) (keyinfo - info->s->keyinfo); + ulong old_link; /* ulong is ok here */ + DBUG_ENTER("_nisam_dispose"); + + old_link=info->s->state.key_del[keynr]; + info->s->state.key_del[keynr]=(ulong) pos; + DBUG_RETURN(key_cache_write(info->s->kfile,pos,(byte*) &old_link, + sizeof(long), + (uint) keyinfo->base.block_length, + (int) (info->lock_type != F_UNLCK))); +} /* _nisam_dispose */ + + + /* Make new page on disk */ + +ulong _nisam_new(register N_INFO *info, N_KEYDEF *keyinfo) +{ + uint keynr= (uint) (keyinfo - info->s->keyinfo); + ulong pos; + DBUG_ENTER("_nisam_new"); + + if ((pos=info->s->state.key_del[keynr]) == NI_POS_ERROR) + { + if (info->s->state.key_file_length >= info->s->base.max_key_file_length) + { + my_errno=HA_ERR_INDEX_FILE_FULL; + DBUG_RETURN(NI_POS_ERROR); + } + pos=info->s->state.key_file_length; + info->s->state.key_file_length+= keyinfo->base.block_length; + } + else + { + if (!key_cache_read(info->s->kfile,pos, + (byte*) &info->s->state.key_del[keynr], + (uint) sizeof(long), + (uint) keyinfo->base.block_length,0)) + pos= NI_POS_ERROR; + } + DBUG_PRINT("exit",("Pos: %d",pos)); + DBUG_RETURN(pos); +} /* _nisam_new */ diff --git a/isam/_search.c b/isam/_search.c new file mode 100644 index 00000000000..3c3f62c3a2a --- /dev/null +++ b/isam/_search.c @@ -0,0 +1,889 @@ +/* Copyright (C) 2000 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 */ + +/* S|ker efter positionen f|r en nyckel samt d{rmedh|rande funktioner */ + +#include "isamdef.h" +#include "m_ctype.h" + +#define CMP(a,b) (a<b ? -1 : a == b ? 0 : 1) + + /* Check index */ + +int _nisam_check_index(N_INFO *info, int inx) +{ + if (inx == -1) /* Use last index */ + inx=info->lastinx; + if (inx >= (int) info->s->state.keys || inx < 0) + { + my_errno=HA_ERR_WRONG_INDEX; + return -1; + } + if (info->lastinx != inx) /* Index changed */ + { + info->lastinx = inx; + info->lastpos = NI_POS_ERROR; + info->update= ((info->update & (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED)) | + HA_STATE_NEXT_FOUND | HA_STATE_PREV_FOUND); + } + if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) + return(-1); + return(inx); +} /* ni_check_index */ + + + /* S|ker reda p} positionen f|r ett record p} basen av en nyckel */ + /* Positionen l{ggs i info->lastpos */ + /* Returns -1 if not found and 1 if search at upper levels */ + +int _nisam_search(register N_INFO *info, register N_KEYDEF *keyinfo, uchar *key, uint key_len, uint nextflag, register ulong pos) +{ + int error,flag; + uint nod_flag; + uchar *keypos,*maxpos; + uchar lastkey[N_MAX_KEY_BUFF],*buff; + DBUG_ENTER("_nisam_search"); + DBUG_PRINT("enter",("pos: %ld nextflag: %d lastpos: %ld", + pos,nextflag,info->lastpos)); + + if (pos == NI_POS_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + info->lastpos= NI_POS_ERROR; + if (!(nextflag & (SEARCH_SMALLER | SEARCH_BIGGER | SEARCH_LAST))) + DBUG_RETURN(-1); /* Not found ; return error */ + DBUG_RETURN(1); /* Search at upper levels */ + } + + if (!(buff=_nisam_fetch_keypage(info,keyinfo,pos,info->buff, + test(!(nextflag & SEARCH_SAVE_BUFF))))) + goto err; + DBUG_DUMP("page",(byte*) buff,getint(buff)); + + flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, + &keypos,lastkey); + nod_flag=test_if_nod(buff); + maxpos=buff+getint(buff)-1; + + if (flag) + { + if ((error=_nisam_search(info,keyinfo,key,key_len,nextflag, + _nisam_kpos(nod_flag,keypos))) <= 0) + DBUG_RETURN(error); + + if (flag >0) + { + if ((nextflag & (SEARCH_SMALLER | SEARCH_LAST)) && + keypos == buff+2+nod_flag) + DBUG_RETURN(1); /* Bigger than key */ + } + else if (nextflag & SEARCH_BIGGER && keypos >= maxpos) + DBUG_RETURN(1); /* Smaller than key */ + } + else + { + if (nextflag & SEARCH_FIND && (!(keyinfo->base.flag & HA_NOSAME) + || key_len) && nod_flag) + { + if ((error=_nisam_search(info,keyinfo,key,key_len,SEARCH_FIND, + _nisam_kpos(nod_flag,keypos))) >= 0 || + my_errno != HA_ERR_KEY_NOT_FOUND) + DBUG_RETURN(error); + info->int_pos= NI_POS_ERROR; /* Buffer not in memory */ + } + } + if (pos != info->int_pos) + { + uchar *old_buff=buff; + if (!(buff=_nisam_fetch_keypage(info,keyinfo,pos,info->buff, + test(!(nextflag & SEARCH_SAVE_BUFF))))) + goto err; + keypos=buff+(keypos-old_buff); + maxpos=buff+(maxpos-old_buff); + } + + if ((nextflag & (SEARCH_SMALLER | SEARCH_LAST)) && flag != 0) + { + keypos=_nisam_get_last_key(info,keyinfo,buff,lastkey,keypos); + if ((nextflag & SEARCH_LAST) && + _nisam_key_cmp(keyinfo->seg, info->lastkey, key, key_len, SEARCH_FIND)) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + goto err; + } + } + + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey)); + VOID(_nisam_move_key(keyinfo,info->lastkey,lastkey)); + info->lastpos=_nisam_dpos(info,nod_flag,keypos); + info->int_keypos=info->buff+ (keypos-buff); + info->int_maxpos=info->buff+ (maxpos-buff); + info->page_changed=0; + info->buff_used= (info->buff != buff); + info->last_search_keypage=info->int_pos; + + DBUG_PRINT("exit",("found key at %ld",info->lastpos)); + DBUG_RETURN(0); +err: + DBUG_PRINT("exit",("Error: %d",my_errno)); + info->lastpos= NI_POS_ERROR; + DBUG_RETURN (-1); +} /* _nisam_search */ + + + /* Search after key in page-block */ + /* If packed key puts smaller or identical key in buff */ + /* ret_pos point to where find or bigger key starts */ + /* ARGSUSED */ + +int _nisam_bin_search(N_INFO *info, register N_KEYDEF *keyinfo, uchar *page, + uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, + uchar *buff __attribute__((unused))) +{ + reg4 int start,mid,end; + int flag; + uint totlength,nod_flag; + DBUG_ENTER("_nisam_bin_search"); + + LINT_INIT(flag); + totlength=keyinfo->base.keylength+(nod_flag=test_if_nod(page)); + start=0; mid=1; + end= (int) ((getint(page)-2-nod_flag)/totlength-1); + DBUG_PRINT("test",("getint: %d end: %d",getint(page),end)); + page+=2+nod_flag; + + while (start != end) + { + mid= (start+end)/2; + if ((flag=_nisam_key_cmp(keyinfo->seg,page+(uint) mid*totlength,key,key_len, + comp_flag)) + >= 0) + end=mid; + else + start=mid+1; + } + if (mid != start) + flag=_nisam_key_cmp(keyinfo->seg,page+(uint) start*totlength,key,key_len, + comp_flag); + if (flag < 0) + start++; /* point at next, bigger key */ + *ret_pos=page+(uint) start*totlength; + DBUG_PRINT("exit",("flag: %d keypos: %d",flag,start)); + DBUG_RETURN(flag); +} /* _nisam_bin_search */ + + + /* Used instead of _nisam_bin_search() when key is packed */ + /* Puts smaller or identical key in buff */ + /* Key is searched sequentially */ + +int _nisam_seq_search(N_INFO *info, register N_KEYDEF *keyinfo, uchar *page, uchar *key, uint key_len, uint comp_flag, uchar **ret_pos, uchar *buff) +{ + int flag; + uint nod_flag,length; + uchar t_buff[N_MAX_KEY_BUFF],*end; + DBUG_ENTER("_nisam_seq_search"); + + LINT_INIT(flag); LINT_INIT(length); + end= page+getint(page); + nod_flag=test_if_nod(page); + page+=2+nod_flag; + *ret_pos=page; + while (page < end) + { + length=(*keyinfo->get_key)(keyinfo,nod_flag,&page,t_buff); + if ((flag=_nisam_key_cmp(keyinfo->seg,t_buff,key,key_len,comp_flag)) >= 0) + break; +#ifdef EXTRA_DEBUG + DBUG_PRINT("loop",("page: %lx key: '%s' flag: %d",page,t_buff,flag)); +#endif + memcpy(buff,t_buff,length); + *ret_pos=page; + } + if (flag == 0) + memcpy(buff,t_buff,length); /* Result is first key */ + DBUG_PRINT("exit",("flag: %d ret_pos: %lx",flag,*ret_pos)); + DBUG_RETURN(flag); +} /* _nisam_seq_search */ + + + /* Get pos to a key_block */ + +ulong _nisam_kpos(uint nod_flag, uchar *after_key) +{ + after_key-=nod_flag; + switch (nod_flag) { + case 3: + return uint3korr(after_key)*512L; + case 2: + return uint2korr(after_key)*512L; + case 1: + return (uint) (*after_key)*512L; + case 0: /* At leaf page */ + default: /* Impossible */ + return(NI_POS_ERROR); + } +} /* _kpos */ + + + /* Save pos to a key_block */ + +void _nisam_kpointer(register N_INFO *info, register uchar *buff, ulong pos) +{ + pos/=512L; + switch (info->s->base.key_reflength) { + case 3: int3store(buff,pos); break; + case 2: int2store(buff,(uint) pos); break; + case 1: buff[0]= (uchar) pos; break; + default: abort(); /* impossible */ + } +} /* _nisam_kpointer */ + + + /* Calc pos to a data-record */ + +ulong _nisam_dpos(N_INFO *info, uint nod_flag, uchar *after_key) +{ + ulong pos; + after_key-=(nod_flag + info->s->rec_reflength); + switch (info->s->rec_reflength) { + case 4: + pos= (ulong) uint4korr(after_key); + break; + case 3: + pos= (ulong) uint3korr(after_key); + break; + case 2: + pos= (ulong) uint2korr(after_key); + break; + default: + pos=0L; /* Shut compiler up */ + } + return (info->s->base.options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? pos : + pos*info->s->base.reclength; +} + + /* save pos to record */ + +void _nisam_dpointer(N_INFO *info, uchar *buff, ulong pos) +{ + if (!(info->s->base.options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + pos/=info->s->base.reclength; + + switch (info->s->rec_reflength) { + case 4: int4store(buff,pos); break; + case 3: int3store(buff,pos); break; + case 2: int2store(buff,(uint) pos); break; + default: abort(); /* Impossible */ + } +} /* _nisam_dpointer */ + + + /* + ** Compare two keys with is bigger + ** Returns <0, 0, >0 acording to with is bigger + ** Key_length specifies length of key to use. Number-keys can't + ** be splitted + ** If flag <> SEARCH_FIND compare also position + */ +int _nisam_key_cmp(register N_KEYSEG *keyseg, register uchar *a, register uchar *b, uint key_length, uint nextflag) +{ + reg4 int flag,length_diff; + int16 s_1,s_2; + int32 l_1,l_2; + uint32 u_1,u_2; + float f_1,f_2; + double d_1,d_2; + reg5 uchar *end; + + if (!(nextflag & (SEARCH_FIND | SEARCH_NO_FIND | SEARCH_LAST)) + || key_length == 0) + key_length=N_MAX_KEY_BUFF*2; + + for ( ; (int) key_length >0 ; key_length-= (keyseg++)->base.length) + { + end= a+ min(keyseg->base.length,key_length); + switch ((enum ha_base_keytype) keyseg->base.type) { + case HA_KEYTYPE_TEXT: /* Ascii; Key is converted */ + case HA_KEYTYPE_BINARY: + if (keyseg->base.flag & HA_SPACE_PACK) + { + uchar *as, *bs; + int length,b_length; + + as=a++; bs=b++; + length= (length_diff= ((int) *as - (b_length= (int) *bs))) < 0 ? + (int) *as : b_length; + end= a+ min(key_length,(uint) length); + +#ifdef USE_STRCOLL + if (use_strcoll(default_charset_info)) { + if (((enum ha_base_keytype) keyseg->base.type) == HA_KEYTYPE_BINARY) + { + while (a < end) + if ((flag= (int) *a++ - (int) *b++)) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + } + else + { + if ((flag = my_strnncoll(default_charset_info, + a, (end-a), b, b_length))) + return (keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag; + b+= (uint) (end-a); + a=end; + } + } + else +#endif + { + while (a < end) + if ((flag= (int) *a++ - (int) *b++)) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + } + if (key_length < (uint) keyseg->base.length) + { /* key_part */ + if (length_diff) + { + if (length_diff < 0 || (uint) *as <= key_length) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? + -length_diff : length_diff); + for (length= (int) key_length-b_length; length-- > 0 ;) + { + if (*a++ != ' ') + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -1 : 1); + } + } + if (nextflag & SEARCH_NO_FIND) /* Find record after key */ + return (nextflag & SEARCH_BIGGER) ? -1 : 1; + return 0; + } + else + { + if (length_diff) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? + -length_diff : length_diff); + } + a=as+ (uint) *as+1 ; b= bs+ b_length+1; /* to next key */ + } + else + { +#ifdef USE_STRCOLL + if (use_strcoll(default_charset_info)) { + if (((enum ha_base_keytype) keyseg->base.type) == HA_KEYTYPE_BINARY) + { + while (a < end) + if ((flag= (int) *a++ - (int) *b++)) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + } + else + { + if ((flag = my_strnncoll(default_charset_info, + a, (end-a), b, (end-a)))) + return (keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag; + b+= (uint) (end-a); + a=end; + } + } + else +#endif + { + while (a < end) + if ((flag= (int) *a++ - (int) *b++)) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + } + } + break; + case HA_KEYTYPE_INT8: + { + int i_1= (int) *((signed char*) a); + int i_2= (int) *((signed char*) b); + if ((flag = CMP(i_1,i_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b++; + break; + } + case HA_KEYTYPE_SHORT_INT: + shortget(s_1,a); + shortget(s_2,b); + if ((flag = CMP(s_1,s_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= 2; /* sizeof(short int); */ + break; + case HA_KEYTYPE_USHORT_INT: + { + uint16 us_1,us_2; + ushortget(us_1,a); + ushortget(us_2,b); + if ((flag = CMP(us_1,us_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+=2; /* sizeof(short int); */ + break; + } + case HA_KEYTYPE_LONG_INT: + longget(l_1,a); + longget(l_2,b); + if ((flag = CMP(l_1,l_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= 4; /* sizeof(long int); */ + break; + case HA_KEYTYPE_ULONG_INT: + ulongget(u_1,a); + ulongget(u_2,b); + if ((flag = CMP(u_1,u_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= 4; /* sizeof(long int); */ + break; + case HA_KEYTYPE_INT24: + l_1=sint3korr(a); + l_2=sint3korr(b); + if ((flag = CMP(l_1,l_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= 3; + break; + case HA_KEYTYPE_UINT24: + l_1=(long) uint3korr(a); + l_2=(long) uint3korr(b); + if ((flag = CMP(l_1,l_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= 3; + break; + case HA_KEYTYPE_FLOAT: + bmove((byte*) &f_1,(byte*) a,(int) sizeof(float)); + bmove((byte*) &f_2,(byte*) b,(int) sizeof(float)); + if ((flag = CMP(f_1,f_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= sizeof(float); + break; + case HA_KEYTYPE_DOUBLE: + doubleget(d_1,a); + doubleget(d_2,b); + if ((flag = CMP(d_1,d_2))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= sizeof(double); + break; + case HA_KEYTYPE_NUM: /* Numeric key */ + { + int swap_flag=keyseg->base.flag & HA_REVERSE_SORT; + if (keyseg->base.flag & HA_SPACE_PACK) + { + int alength,blength; + + if (swap_flag) + swap(uchar*,a,b); + alength= *a++; blength= *b++; + if ((flag=(int) (keyseg->base.length-key_length)) < 0) + flag=0; + if (alength != blength+flag) + { + if ((alength > blength+flag && *a != '-') || + (alength < blength+flag && *b == '-')) + return 1; + else + return -1; + } + if (*a == '-' && *b == '-') + { + swap_flag=1; + swap(uchar*,a,b); + } + end=a+alength; + while (a < end) + if (*a++ != *b++) + { + a--; b--; + if (isdigit((char) *a) && isdigit((char) *b)) + return ((int) *a - (int) *b); + if (*a == '-' || isdigit((char) *b)) + return (-1); + if (*b == '-' || *b++ == ' ' || isdigit((char) *a)) + return (1); + if (*a++ == ' ') + return (-1); + } + } + else + { + for ( ; a < end && *a == ' ' && *b == ' ' ; a++, b++) ; + if (*a == '-' && *b == '-') + swap_flag=1; + if (swap_flag) + { + end=b+(int) (end-a); + swap(uchar*,a,b); + } + while (a < end) + if (*a++ != *b++) + { + a--; b--; + if (isdigit((char) *a) && isdigit((char) *b)) + return ((int) *a - (int) *b); + if (*a == '-' || isdigit((char) *b)) + return (-1); + if (*b == '-' || *b++ == ' ' || isdigit((char) *a)) + return (1); + if (*a++ == ' ') + return -1; + } + } + if (swap_flag) + swap(uchar*,a,b); + break; + } +#ifdef HAVE_LONG_LONG + case HA_KEYTYPE_LONGLONG: + { + longlong ll_a,ll_b; + longlongget(ll_a,a); + longlongget(ll_b,b); + if ((flag = CMP(ll_a,ll_b))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= sizeof(longlong); + break; + } + case HA_KEYTYPE_ULONGLONG: + { + ulonglong ll_a,ll_b; + longlongget(ll_a,a); + longlongget(ll_b,b); + if ((flag = CMP(ll_a,ll_b))) + return ((keyseg->base.flag & HA_REVERSE_SORT) ? -flag : flag); + a= end; + b+= sizeof(ulonglong); + break; + } +#endif + case HA_KEYTYPE_END: /* Ready */ + case HA_KEYTYPE_VARTEXT: /* Impossible */ + case HA_KEYTYPE_VARBINARY: /* Impossible */ + goto end; + } + } +end: + if (!(nextflag & SEARCH_FIND)) + { + if (nextflag & (SEARCH_NO_FIND | SEARCH_LAST)) /* Find record after key */ + return (nextflag & (SEARCH_BIGGER | SEARCH_LAST)) ? -1 : 1; + LINT_INIT(l_1); LINT_INIT(l_2); + switch (keyseg->base.length) { + case 4: + u_1= (ulong) uint4korr(a); + u_2= (ulong) uint4korr(b); + break; + case 3: + u_1= (ulong) uint3korr(a); + u_2= (ulong) uint3korr(b); + break; + case 2: + u_1= (ulong) uint2korr(a); + u_2= (ulong) uint2korr(b); + break; + default: abort(); /* Impossible */ + } + flag = CMP(u_1,u_2); + + if (nextflag & SEARCH_SAME) + return (flag); /* read same */ + if (nextflag & SEARCH_BIGGER) + return (flag <= 0 ? -1 : 1); /* read next */ + return (flag < 0 ? -1 : 1); /* read previous */ + } + return 0; +} /* _nisam_key_cmp */ + + + /* Get key from key-block */ + /* page points at previous key; its advanced to point at next key */ + /* key should contain previous key */ + /* Returns length of found key + pointers */ + /* nod_flag is a flag if we are on nod */ + +uint _nisam_get_key(register N_KEYDEF *keyinfo, uint nod_flag, + register uchar **page, register uchar *key) +{ + reg1 N_KEYSEG *keyseg; + uchar *start,*start_key; + uint length,c_length; + + LINT_INIT(start); + start_key=key; c_length=0; + for (keyseg=keyinfo->seg ; keyseg->base.type ;keyseg++) + { + if (keyseg->base.flag & (HA_SPACE_PACK | HA_PACK_KEY)) + { + start=key; + if (keyseg->base.flag & HA_SPACE_PACK) + key++; + if ((length= *(*page)++) & 128) + { + key+= (c_length=(length & 127)); + if (c_length == 0) /* Same key */ + { + key+= *start; /* Same diff_key as prev */ + length=0; + } + else + { + if (keyseg->base.flag & HA_SPACE_PACK) + length= *(*page)++; + else + length=keyseg->base.length-length+128; /* Rest of key */ + /* Prevent core dumps if wrong data formats */ + if (length > keyseg->base.length) + length=0; + } + } + } + else + length=keyseg->base.length; + memcpy((byte*) key,(byte*) *page,(size_t) length); key+=length; + if (keyseg->base.flag & HA_SPACE_PACK) + *start= (uchar) ((key-start)-1); + *page+=length; + } + length=keyseg->base.length+nod_flag; + bmove((byte*) key,(byte*) *page,length); + *page+=length; + return((uint) (key-start_key)+keyseg->base.length); +} /* _nisam_get_key */ + + + /* same as _nisam_get_key but used with fixed length keys */ + +uint _nisam_get_static_key(register N_KEYDEF *keyinfo, uint nod_flag, register uchar **page, register uchar *key) +{ + memcpy((byte*) key,(byte*) *page, + (size_t) (keyinfo->base.keylength+nod_flag)); + *page+=keyinfo->base.keylength+nod_flag; + return(keyinfo->base.keylength); +} /* _nisam_get_static_key */ + + + /* Get last key from key-block, starting from keypos */ + /* Return pointer to where keystarts */ + +uchar *_nisam_get_last_key(N_INFO *info, N_KEYDEF *keyinfo, uchar *keypos, uchar *lastkey, uchar *endpos) +{ + uint nod_flag; + uchar *lastpos; + + nod_flag=test_if_nod(keypos); + if (! (keyinfo->base.flag & (HA_PACK_KEY | HA_SPACE_PACK_USED))) + { + lastpos=endpos-keyinfo->base.keylength-nod_flag; + if (lastpos > keypos) + bmove((byte*) lastkey,(byte*) lastpos,keyinfo->base.keylength+nod_flag); + } + else + { + lastpos=0 ; keypos+=2+nod_flag; + lastkey[0]=0; + while (keypos < endpos) + { + lastpos=keypos; + VOID(_nisam_get_key(keyinfo,nod_flag,&keypos,lastkey)); + } + } + return lastpos; +} /* _nisam_get_last_key */ + + + /* Calculate length of key */ + +uint _nisam_keylength(N_KEYDEF *keyinfo, register uchar *key) +{ + reg1 N_KEYSEG *keyseg; + uchar *start; + + if (! (keyinfo->base.flag & HA_SPACE_PACK_USED)) + return (keyinfo->base.keylength); + + start=key; + for (keyseg=keyinfo->seg ; keyseg->base.type ; keyseg++) + { + if (keyseg->base.flag & HA_SPACE_PACK) + key+= *key+1; + else + key+= keyseg->base.length; + } + return((uint) (key-start)+keyseg->base.length); +} /* _nisam_keylength */ + + + /* Move a key */ + +uchar *_nisam_move_key(N_KEYDEF *keyinfo, uchar *to, uchar *from) +{ + reg1 uint length; + memcpy((byte*) to, (byte*) from, + (size_t) (length=_nisam_keylength(keyinfo,from))); + return to+length; +} + + /* Find next/previous record with same key */ + /* This can't be used when database is touched after last read */ + +int _nisam_search_next(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *key, uint nextflag, ulong pos) +{ + int error; + uint nod_flag; + uchar lastkey[N_MAX_KEY_BUFF]; + DBUG_ENTER("_nisam_search_next"); + DBUG_PRINT("enter",("nextflag: %d lastpos: %d int_keypos: %lx", + nextflag,info->lastpos,info->int_keypos)); + DBUG_EXECUTE("key",_nisam_print_key(DBUG_FILE,keyinfo->seg,key);); + + if ((nextflag & SEARCH_BIGGER && info->int_keypos >= info->int_maxpos) || + info->int_pos == NI_POS_ERROR || info->page_changed) + DBUG_RETURN(_nisam_search(info,keyinfo,key,0,nextflag | SEARCH_SAVE_BUFF, + pos)); + + if (info->buff_used) + { + if (!_nisam_fetch_keypage(info,keyinfo,info->last_search_keypage, + info->buff,0)) + { + info->lastpos= NI_POS_ERROR; + DBUG_RETURN(-1); + } + info->buff_used=0; + } + + /* Last used buffer is in info->buff */ + + nod_flag=test_if_nod(info->buff); + VOID(_nisam_move_key(keyinfo,lastkey,key)); + + if (nextflag & SEARCH_BIGGER) /* Next key */ + { + ulong tmp_pos=_nisam_kpos(nod_flag,info->int_keypos); + if (tmp_pos != NI_POS_ERROR) + { + if ((error=_nisam_search(info,keyinfo,key,0,nextflag | SEARCH_SAVE_BUFF, + tmp_pos)) <=0) + DBUG_RETURN(error); + } + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&info->int_keypos,lastkey)); + } + else /* Previous key */ + { + info->int_keypos=_nisam_get_last_key(info,keyinfo,info->buff,lastkey, + info->int_keypos); + if (info->int_keypos == info->buff+2) + DBUG_RETURN(_nisam_search(info,keyinfo,key,0,nextflag | SEARCH_SAVE_BUFF, + pos)); + if ((error=_nisam_search(info,keyinfo,key,0,nextflag | SEARCH_SAVE_BUFF, + _nisam_kpos(nod_flag,info->int_keypos))) <= 0) + DBUG_RETURN(error); + } + + info->int_keypos=_nisam_get_last_key(info,keyinfo,info->buff,lastkey, + info->int_keypos); + VOID(_nisam_move_key(keyinfo,info->lastkey,lastkey)); + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&info->int_keypos,info->lastkey)); + info->lastpos=_nisam_dpos(info,nod_flag,info->int_keypos); + DBUG_PRINT("exit",("found key at %d",info->lastpos)); + DBUG_RETURN(0); +} /* _nisam_search_next */ + + + /* S|ker reda p} positionen f|r f|rsta recordet i ett index */ + /* Positionen l{ggs i info->lastpos */ + +int _nisam_search_first(register N_INFO *info, register N_KEYDEF *keyinfo, register ulong pos) +{ + uint nod_flag; + uchar *page; + DBUG_ENTER("_nisam_search_first"); + + if (pos == NI_POS_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; + info->lastpos= NI_POS_ERROR; + DBUG_RETURN(-1); + } + + do + { + if (!_nisam_fetch_keypage(info,keyinfo,pos,info->buff,0)) + { + info->lastpos= NI_POS_ERROR; + DBUG_RETURN(-1); + } + nod_flag=test_if_nod(info->buff); + page=info->buff+2+nod_flag; + } while ((pos=_nisam_kpos(nod_flag,page)) != NI_POS_ERROR); + + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&page,info->lastkey)); + info->int_keypos=page; info->int_maxpos=info->buff+getint(info->buff)-1; + info->lastpos=_nisam_dpos(info,nod_flag,page); + info->page_changed=info->buff_used=0; + info->last_search_keypage=info->int_pos; + + DBUG_PRINT("exit",("found key at %d",info->lastpos)); + DBUG_RETURN(0); +} /* _nisam_search_first */ + + + /* S|ker reda p} positionen f|r sista recordet i ett index */ + /* Positionen l{ggs i info->lastpos */ + +int _nisam_search_last(register N_INFO *info, register N_KEYDEF *keyinfo, register ulong pos) +{ + uint nod_flag; + uchar *buff,*page; + DBUG_ENTER("_nisam_search_last"); + + if (pos == NI_POS_ERROR) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* Didn't find key */ + info->lastpos= NI_POS_ERROR; + DBUG_RETURN(-1); + } + + buff=info->buff; + do + { + if (!_nisam_fetch_keypage(info,keyinfo,pos,buff,0)) + { + info->lastpos= NI_POS_ERROR; + DBUG_RETURN(-1); + } + page= buff+getint(buff); + nod_flag=test_if_nod(buff); + } while ((pos=_nisam_kpos(nod_flag,page)) != NI_POS_ERROR); + + VOID(_nisam_get_last_key(info,keyinfo,buff,info->lastkey,page)); + info->lastpos=_nisam_dpos(info,nod_flag,page); + info->int_keypos=info->int_maxpos=page; + info->page_changed=info->buff_used=0; + info->last_search_keypage=info->int_pos; + + DBUG_PRINT("exit",("found key at %d",info->lastpos)); + DBUG_RETURN(0); +} /* _nisam_search_last */ diff --git a/isam/_statrec.c b/isam/_statrec.c new file mode 100644 index 00000000000..d93f4fe27f5 --- /dev/null +++ b/isam/_statrec.c @@ -0,0 +1,265 @@ +/* Copyright (C) 2000 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 "isamdef.h" + + +int _nisam_write_static_record(N_INFO *info, const byte *record) +{ + uchar temp[4]; /* Not sizeof(long) */ + + if (info->s->state.dellink != NI_POS_ERROR) + { + ulong filepos=info->s->state.dellink; + info->rec_cache.seek_not_done=1; /* We have done a seek */ + VOID(my_seek(info->dfile,info->s->state.dellink+1,MY_SEEK_SET,MYF(0))); + + if (my_read(info->dfile,(char*) &temp[0],sizeof(temp), MYF(MY_NABP))) + goto err; + info->s->state.dellink=uint4korr(temp); + if (info->s->state.dellink == (uint32) ~0) /* Fix for 64 bit long */ + info->s->state.dellink=NI_POS_ERROR; + info->s->state.del--; + info->s->state.empty-=info->s->base.reclength; + VOID(my_seek(info->dfile,filepos,MY_SEEK_SET,MYF(0))); + if (my_write(info->dfile, (char*) record, info->s->base.reclength, + MYF(MY_NABP))) + goto err; + } + else + { + if (info->s->state.data_file_length > info->s->base.max_data_file_length) + { + 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; + } + else + { + info->rec_cache.seek_not_done=1; /* We have done a seek */ + VOID(my_seek(info->dfile,info->s->state.data_file_length, + MY_SEEK_SET,MYF(0))); + if (my_write(info->dfile,(char*) record,info->s->base.reclength, + MYF(MY_NABP | MY_WAIT_IF_FULL))) + goto err; + } + info->s->state.data_file_length+=info->s->base.reclength; + info->s->state.splitt++; + } + return 0; + err: + return 1; +} + +int _nisam_update_static_record(N_INFO *info, ulong pos, const byte *record) +{ + info->rec_cache.seek_not_done=1; /* We have done a seek */ + VOID(my_seek(info->dfile,pos,MY_SEEK_SET,MYF(0))); + return (my_write(info->dfile,(char*) record,info->s->base.reclength, + MYF(MY_NABP)) != 0); +} + + +int _nisam_delete_static_record(N_INFO *info) +{ + uchar temp[5]; /* 1+sizeof(uint32) */ + + info->s->state.del++; + info->s->state.empty+=info->s->base.reclength; + temp[0]= '\0'; /* Mark that record is deleted */ + int4store(temp+1,info->s->state.dellink); + info->s->state.dellink = info->lastpos; + info->rec_cache.seek_not_done=1; + VOID(my_seek(info->dfile,info->lastpos,MY_SEEK_SET,MYF(0))); + return (my_write(info->dfile,(byte*) temp,(uint) sizeof(temp), + MYF(MY_NABP)) != 0); +} + + +int _nisam_cmp_static_record(register N_INFO *info, register const byte *old) +{ + DBUG_ENTER("_nisam_rectest"); + + /* 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 */ + VOID(my_seek(info->dfile,info->lastpos,MY_SEEK_SET,MYF(0))); + if (my_read(info->dfile, (char*) info->rec_buff, info->s->base.reclength, + 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); +} + + /* Read a fixed-length-record */ + /* Returns 0 if Ok. */ + /* 1 if record is deleted */ + /* MY_FILE_ERROR on read-error or locking-error */ + +int _nisam_read_static_record(register N_INFO *info, register ulong pos, + register byte *record) +{ + int error; + + if (pos != NI_POS_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=my_pread(info->dfile,(char*) record,info->s->base.reclength, + pos,MYF(MY_NABP)) != 0; + if (info->s->r_locks == 0 && info->s->w_locks == 0) + VOID(_nisam_writeinfo(info,0)); + if (! error) + { + if (!*record) return(1); /* Record is deleted */ + info->update|= HA_STATE_AKTIV; /* Record is read */ + my_errno=HA_ERR_RECORD_DELETED; + return(0); + } + return(-1); /* Error on read */ + } + VOID(_nisam_writeinfo(info,0)); /* No such record */ + return(-1); +} /* _nisam_read_record */ + + +int _nisam_read_rnd_static_record(N_INFO *info, byte *buf, + register ulong filepos, + int skipp_deleted_blocks) +{ + int locked,error,cache_read; + uint cache_length; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_read_rnd_static_record"); + + cache_read=0; + LINT_INIT(cache_length); + if (info->opt_flag & WRITE_CACHE_USED && + (info->rec_cache.pos_in_file <= filepos || skipp_deleted_blocks) && + flush_io_cache(&info->rec_cache)) + DBUG_RETURN(-1); + if (info->opt_flag & READ_CACHE_USED) + { /* Cash in use */ + if (filepos == my_b_tell(&info->rec_cache) && + (skipp_deleted_blocks || !filepos)) + { + cache_read=1; /* Read record using cache */ + cache_length=(uint) (info->rec_cache.rc_end - info->rec_cache.rc_pos); + } + else + info->rec_cache.seek_not_done=1; /* Filepos is changed */ + } +#ifndef NO_LOCKING + locked=0; + if (info->lock_type == F_UNLCK) + { + if (filepos >= share->state.data_file_length) + { /* Test if new records */ + if (_nisam_readinfo(info,F_RDLCK,0)) + DBUG_RETURN(-1); + locked=1; + } + else + { /* We don't nead new info */ +#ifndef UNSAFE_LOCKING + if ((! cache_read || share->base.reclength > cache_length) && + share->r_locks == 0 && share->w_locks == 0) + { /* record not in cache */ + if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + DBUG_RETURN(-1); + locked=1; + } +#else + info->tmp_lock_type=F_RDLCK; +#endif + } + } +#endif + if (filepos >= share->state.data_file_length) + { +#ifndef NO_LOCKING + DBUG_PRINT("test",("filepos: %ld (%ld) records: %ld del: %ld", + filepos/share->base.reclength,filepos, + share->state.records, share->state.del)); + VOID(_nisam_writeinfo(info,0)); +#endif + my_errno=HA_ERR_END_OF_FILE; + DBUG_RETURN(-1); + } + info->lastpos= filepos; + info->nextpos= filepos+share->base.reclength; + + if (! cache_read) /* No cacheing */ + { + error=_nisam_read_static_record(info,filepos,buf); + if (error > 0) + my_errno=HA_ERR_RECORD_DELETED; + DBUG_RETURN(error); + } + + /* Read record with cacheing */ + error=my_b_read(&info->rec_cache,(byte*) buf,share->base.reclength); + +#ifndef NO_LOCKING + if (locked) + VOID(_nisam_writeinfo(info,0)); /* Unlock keyfile */ +#endif + if (!error) + { + if (!buf[0]) + { /* Record is removed */ + my_errno=HA_ERR_RECORD_DELETED; + DBUG_RETURN(1); + } + /* Found and may be updated */ + info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED; + DBUG_RETURN(0); + } + if (info->rec_cache.error != -1 || my_errno == 0) + my_errno=HA_ERR_WRONG_IN_RECORD; + DBUG_RETURN(-1); /* Something wrong (EOF?) */ +} diff --git a/isam/changed.c b/isam/changed.c new file mode 100644 index 00000000000..4f87a45aa2d --- /dev/null +++ b/isam/changed.c @@ -0,0 +1,35 @@ +/* Copyright (C) 2000 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 */ + +/* Check if somebody has changed table since last check. */ + +#include "isamdef.h" + + /* Return 0 if table isn't changed */ + +int nisam_is_changed(N_INFO *info) +{ + int result; + DBUG_ENTER("nisam_is_changed"); +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) DBUG_RETURN(-1); + VOID(_nisam_writeinfo(info,0)); +#endif + result=(int) info->data_changed; + info->data_changed=0; + DBUG_PRINT("exit",("result: %d",result)); + DBUG_RETURN(result); +} diff --git a/isam/close.c b/isam/close.c new file mode 100644 index 00000000000..6741e7b23f0 --- /dev/null +++ b/isam/close.c @@ -0,0 +1,91 @@ +/* Copyright (C) 2000 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 */ + +/* close a isam-database */ + +#include "isamdef.h" + +int nisam_close(register N_INFO *info) +{ + int error=0,flag; + ISAM_SHARE *share=info->s; + DBUG_ENTER("nisam_close"); + DBUG_PRINT("enter",("base: %lx reopen: %u locks: %u", + info,(uint) share->reopen, + (uint) (share->w_locks+share->r_locks))); + + pthread_mutex_lock(&THR_LOCK_isam); + if (info->lock_type == F_EXTRA_LCK) + info->lock_type=F_UNLCK; /* HA_EXTRA_NO_USER_CHANGE */ + +#ifndef NO_LOCKING + if (info->lock_type != F_UNLCK) + VOID(nisam_lock_database(info,F_UNLCK)); +#else + info->lock_type=F_UNLCK; + share->w_locks--; + if (_nisam_writeinfo(info,test(share->changed))) + error=my_errno; +#endif + pthread_mutex_lock(&share->intern_lock); + + if (share->base.options & HA_OPTION_READ_ONLY_DATA) + share->r_locks--; + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + if (end_io_cache(&info->rec_cache)) + error=my_errno; + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + } + flag= !--share->reopen; + nisam_open_list=list_delete(nisam_open_list,&info->open_list); + pthread_mutex_unlock(&share->intern_lock); + + if (flag) + { + if (share->kfile >= 0 && flush_key_blocks(share->kfile,FLUSH_RELEASE)) + error=my_errno; + if (share->kfile >= 0 && my_close(share->kfile,MYF(0))) + error = my_errno; +#ifdef HAVE_MMAP + _nisam_unmap_file(info); +#endif + if (share->decode_trees) + { + my_free((gptr) share->decode_trees,MYF(0)); + my_free((gptr) share->decode_tables,MYF(0)); + } +#ifdef THREAD + thr_lock_delete(&share->lock); + VOID(pthread_mutex_destroy(&share->intern_lock)); +#endif + my_free((gptr) info->s,MYF(0)); + } + pthread_mutex_unlock(&THR_LOCK_isam); + if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + error = my_errno; + + nisam_log_command(LOG_CLOSE,info,NULL,0,error); + my_free((gptr) info->rec_alloc,MYF(0)); + my_free((gptr) info,MYF(0)); + + if (error) + { + my_errno=error; + DBUG_RETURN(-1); + } + DBUG_RETURN(0); +} /* nisam_close */ diff --git a/isam/create.c b/isam/create.c new file mode 100644 index 00000000000..9b5e9eaece4 --- /dev/null +++ b/isam/create.c @@ -0,0 +1,326 @@ +/* Copyright (C) 2000 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 */ + +/* Skapar en isam-databas */ + +#include "isamdef.h" +#if defined(MSDOS) || defined(__WIN__) +#ifdef __WIN__ +#include <fcntl.h> +#else +#include <process.h> /* Prototype for getpid */ +#endif +#endif + + /* + ** Old options is used when recreating database, from isamchk + ** Note that the minimun reclength that MySQL allows for static rows + ** are 5. (Will be fixed in the next generation) + */ + +int nisam_create(const char *name,uint keys,N_KEYDEF *keyinfo, + N_RECINFO *recinfo, + ulong records,ulong reloc, uint flags,uint old_options, + ulong data_file_length) +{ + register uint i,j; + File dfile,file; + int errpos,save_errno; + uint fields,length,max_key_length,packed,pointer,reclength,min_pack_length, + key_length,info_length,key_segs,options,min_key_length_skipp,max_block, + base_pos; + char buff[max(FN_REFLEN,512)]; + ulong tot_length,pack_reclength; + enum en_fieldtype type; + ISAM_SHARE share; + N_KEYDEF *keydef; + N_KEYSEG *keyseg; + N_RECINFO *rec; + DBUG_ENTER("nisam_create"); + + LINT_INIT(dfile); + pthread_mutex_lock(&THR_LOCK_isam); + errpos=0; + options=0; + base_pos=512; /* Enough for N_STATE_INFO */ + bzero((byte*) &share,sizeof(share)); + if ((file = my_create(fn_format(buff,name,"",N_NAME_IEXT,4),0, + O_RDWR | O_TRUNC,MYF(MY_WME))) < 0) + goto err; + errpos=1; + VOID(fn_format(buff,name,"",N_NAME_DEXT,2+4)); + if (!(flags & HA_DONT_TOUCH_DATA)) + { + if ((dfile = my_create(buff,0,O_RDWR | O_TRUNC,MYF(MY_WME))) < 0) + goto err; + errpos=2; + } + else if (!(old_options & HA_OPTION_TEMP_COMPRESS_RECORD)) + options=old_options & (HA_OPTION_COMPRESS_RECORD | + HA_OPTION_READ_ONLY_DATA | HA_OPTION_PACK_RECORD); + if (reloc > records) + reloc=records; /* Check if wrong parameter */ + + /* Start by checking fields and field-types used */ + reclength=0; + for (rec=recinfo, fields=packed=min_pack_length=0, pack_reclength=0L; + rec->base.type != (int) FIELD_LAST; + rec++,fields++) + { + reclength+=rec->base.length; + if ((type=(enum en_fieldtype) rec->base.type)) + { + packed++; + if (type == FIELD_BLOB) + { + share.base.blobs++; + rec->base.length-= sizeof(char*); /* Don't calc pointer */ + if (pack_reclength != NI_POS_ERROR) + { + if (rec->base.length == 4) + pack_reclength= (ulong) NI_POS_ERROR; + else + pack_reclength+=sizeof(char*)+(1 << (rec->base.length*8)); + } + } + else if (type == FIELD_SKIPP_PRESPACE || + type == FIELD_SKIPP_ENDSPACE) + { + if (pack_reclength != NI_POS_ERROR) + pack_reclength+= rec->base.length > 255 ? 2 : 1; + min_pack_length++; + } + else if (type == FIELD_ZERO) + packed--; + else if (type != FIELD_SKIPP_ZERO) + { + min_pack_length+=rec->base.length; + packed--; /* Not a pack record type */ + } + } + else + min_pack_length+=rec->base.length; + } + if ((packed & 7) == 1) + { /* Bad packing, try to remove a zero-field */ + while (rec != recinfo) + { + rec--; + if (rec->base.type == (int) FIELD_SKIPP_ZERO && rec->base.length == 1) + { + rec->base.type=(int) FIELD_NORMAL; + packed--; + min_pack_length++; + break; + } + } + } + if (packed && !(options & HA_OPTION_COMPRESS_RECORD)) + options|=HA_OPTION_PACK_RECORD; /* Must use packed records */ + + packed=(packed+7)/8; + if (pack_reclength != NI_POS_ERROR) + pack_reclength+= reclength+packed; + min_pack_length+=packed; + + if (options & HA_OPTION_COMPRESS_RECORD) + { + if (data_file_length >= (1L << 24)) + pointer=4; + else if (data_file_length >= (1L << 16)) + pointer=3; + else + pointer=2; + } + else if (((records == 0L && pack_reclength < 255) || + options & HA_OPTION_PACK_RECORD) || + records >= (ulong) 16000000L || + pack_reclength == (ulong) NI_POS_ERROR || + ((options & HA_OPTION_PACK_RECORD) && + pack_reclength+4 >= (ulong) 14000000L/records)) + pointer=4; + else if (records == 0L || records >= (ulong) 65000L || + ((options & HA_OPTION_PACK_RECORD) && + pack_reclength+4 >= (ulong) 60000L/records)) + pointer=3; + else + pointer=2; + + max_block=max_key_length=0; tot_length=key_segs=0; + for (i=0, keydef=keyinfo ; i < keys ; i++ , keydef++) + { + share.state.key_root[i]= share.state.key_del[i]= NI_POS_ERROR; + share.base.rec_per_key[i]= (keydef->base.flag & HA_NOSAME) ? 1L : 0L; + min_key_length_skipp=length=0; + key_length=pointer; + + if (keydef->base.flag & HA_PACK_KEY && + keydef->seg[0].base.length > 127) + keydef->base.flag&= ~HA_PACK_KEY; /* Can't pack long keys */ + if (keydef->base.flag & HA_PACK_KEY) + { + if ((keydef->seg[0].base.flag & HA_SPACE_PACK) && + keydef->seg[0].base.type == (int) HA_KEYTYPE_NUM) + keydef->seg[0].base.flag&= ~HA_SPACE_PACK; + if (!(keydef->seg[0].base.flag & HA_SPACE_PACK)) + length++; + keydef->seg[0].base.flag|=HA_PACK_KEY; /* for easyer intern test */ + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + if (!(keydef->seg[0].base.flag & HA_SPACE_PACK)) + min_key_length_skipp+=keydef->seg[0].base.length; + } + keydef->base.keysegs=0; + for (keyseg=keydef->seg ; keyseg->base.type ; keyseg++) + { + keydef->base.keysegs++; + if (keyseg->base.length > 127) + keyseg->base.flag&= ~(HA_SPACE_PACK | HA_PACK_KEY); + if (keyseg->base.flag & HA_SPACE_PACK) + { + keydef->base.flag |= HA_SPACE_PACK_USED; + options|=HA_OPTION_PACK_KEYS; /* Using packed keys */ + length++; + min_key_length_skipp+=keyseg->base.length; + } + key_length+= keyseg->base.length; + } + bzero((gptr) keyseg,sizeof(keyseg[0])); + keyseg->base.length=(uint16) pointer; /* Last key part is pointer */ + key_segs+=keydef->base.keysegs; + length+=key_length; + keydef->base.block_length=nisam_block_size; + keydef->base.keylength= (uint16) key_length; + keydef->base.minlength= (uint16) (length-min_key_length_skipp); + keydef->base.maxlength= (uint16) length; + + if ((uint) keydef->base.block_length > max_block) + max_block=(uint) keydef->base.block_length; + if (length > max_key_length) + max_key_length= length; + tot_length+= (records/(ulong) (((uint) keydef->base.block_length-5)/ + (length*2)))* + (ulong) keydef->base.block_length; + } + info_length=(uint) (base_pos+sizeof(N_BASE_INFO)+keys*sizeof(N_SAVE_KEYDEF)+ + (keys+key_segs)*sizeof(N_SAVE_KEYSEG)+ + fields*sizeof(N_SAVE_RECINFO)); + + bmove(share.state.header.file_version,(byte*) nisam_file_magic,4); + old_options=options| (old_options & HA_OPTION_TEMP_COMPRESS_RECORD ? + HA_OPTION_COMPRESS_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD: 0); + int2store(share.state.header.options,old_options); + int2store(share.state.header.header_length,info_length); + int2store(share.state.header.state_info_length,sizeof(N_STATE_INFO)); + int2store(share.state.header.base_info_length,sizeof(N_BASE_INFO)); + int2store(share.state.header.base_pos,base_pos); + + share.state.dellink = NI_POS_ERROR; + share.state.process= (ulong) getpid(); + share.state.uniq= (ulong) file; + share.state.loop= 0; + share.state.version= (ulong) time((time_t*) 0); + share.base.options=options; + share.base.rec_reflength=pointer; + share.base.key_reflength=((!tot_length || tot_length > 30000000L) ? 3 : + tot_length > 120000L ? 2 : 1); + share.base.keys= share.state.keys = keys; + share.base.keystart = share.state.key_file_length=MY_ALIGN(info_length, + nisam_block_size); + share.base.max_block=max_block; + share.base.max_key_length=ALIGN_SIZE(max_key_length+4); + share.base.records=records; + share.base.reloc=reloc; + share.base.reclength=reclength; + share.base.pack_reclength=reclength+packed-share.base.blobs*sizeof(char*); + share.base.max_pack_length=pack_reclength; + share.base.min_pack_length=min_pack_length; + share.base.pack_bits=packed; + share.base.fields=fields; + share.base.pack_fields=packed; + share.base.sortkey= (ushort) ~0; + share.base.max_data_file_length= (pointer == 4) ? ~0L : + (options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ? + (1L << (pointer*8)) : + (pointer == 3 && reclength >= 256L) ? NI_POS_ERROR : + ((ulong) reclength * (1L << (pointer*8))); + share.base.max_key_file_length= (share.base.key_reflength == 3 ? + NI_POS_ERROR : + (1L << (share.base.key_reflength*8))*512); + share.base.min_block_length= + (share.base.pack_reclength+3 < N_EXTEND_BLOCK_LENGTH && + ! share.base.blobs) ? + max(share.base.pack_reclength,N_MIN_BLOCK_LENGTH) : + N_EXTEND_BLOCK_LENGTH; + if (! (flags & HA_DONT_TOUCH_DATA)) + share.base.create_time= (long) time((time_t*) 0); + + bzero(buff,base_pos); + if (my_write(file,(char*) &share.state,sizeof(N_STATE_INFO),MYF(MY_NABP)) || + my_write(file,buff,base_pos-sizeof(N_STATE_INFO),MYF(MY_NABP)) || + my_write(file,(char*) &share.base,sizeof(N_BASE_INFO),MYF(MY_NABP))) + goto err; + + for (i=0 ; i < share.base.keys ; i++) + { + if (my_write(file,(char*) &keyinfo[i].base,sizeof(N_SAVE_KEYDEF), + MYF(MY_NABP))) + goto err; + for (j=0 ; j <= keyinfo[i].base.keysegs ; j++) + { + if (my_write(file,(char*) &keyinfo[i].seg[j].base,sizeof(N_SAVE_KEYSEG), + MYF(MY_NABP))) + goto err; + } + } + for (i=0 ; i < share.base.fields ; i++) + if (my_write(file,(char*) &recinfo[i].base, (uint) sizeof(N_SAVE_RECINFO), + MYF(MY_NABP))) + goto err; + + /* Enlarge files */ + if (my_chsize(file,(ulong) share.base.keystart,MYF(0))) + goto err; + + if (! (flags & HA_DONT_TOUCH_DATA)) + { +#ifdef USE_RELOC + if (my_chsize(dfile,share.base.min_pack_length*reloc,MYF(0))) + goto err; +#endif + errpos=1; + if (my_close(dfile,MYF(0))) + goto err; + } + errpos=0; + pthread_mutex_unlock(&THR_LOCK_isam); + if (my_close(file,MYF(0))) + goto err; + DBUG_RETURN(0); + +err: + pthread_mutex_unlock(&THR_LOCK_isam); + save_errno=my_errno; + switch (errpos) { + case 2: + VOID(my_close(dfile,MYF(0))); + /* fall through */ + case 1: + VOID(my_close(file,MYF(0))); + } + my_errno=save_errno; /* R{tt felkod tillbaka */ + DBUG_RETURN(-1); +} /* nisam_create */ diff --git a/isam/delete.c b/isam/delete.c new file mode 100644 index 00000000000..9c66e241147 --- /dev/null +++ b/isam/delete.c @@ -0,0 +1,615 @@ +/* Copyright (C) 2000 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 */ + +/* Tar bort ett record fr}n en isam-databas */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <errno.h> +#endif +#include <assert.h> + +static int d_search(N_INFO *info,N_KEYDEF *keyinfo,uchar *key,ulong page, + uchar *anc_buff); +static int del(N_INFO *info,N_KEYDEF *keyinfo,uchar *key,uchar *anc_buff, + ulong leaf_page,uchar *leaf_buff,uchar *keypos, + ulong next_block,uchar *ret_key); +static int underflow(N_INFO *info,N_KEYDEF *keyinfo,uchar *anc_buff, + ulong leaf_page, uchar *leaf_buff,uchar *keypos); +static uint remove_key(N_KEYDEF *keyinfo,uint nod_flag,uchar *keypos, + uchar *lastkey,uchar *page_end); + + +int nisam_delete(N_INFO *info,const byte *record) +{ + uint i; + uchar *old_key; + int save_errno; + uint32 lastpos; + ISAM_SHARE *share=info->s; + DBUG_ENTER("nisam_delete"); + + /* Test if record is in datafile */ + + if (!(info->update & HA_STATE_AKTIV)) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* No database read */ + DBUG_RETURN(-1); + } + if (share->base.options & HA_OPTION_READ_ONLY_DATA) + { + my_errno=EACCES; + DBUG_RETURN(-1); + } +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_WRLCK,1)) DBUG_RETURN(-1); +#endif + if ((*share->compare_record)(info,record)) + goto err; /* Fel vid kontroll-l{sning */ + + /* Remove all keys from the .ISAM file */ + + old_key=info->lastkey+share->base.max_key_length; + for (i=0 ; i < share->state.keys ; i++ ) + { + VOID(_nisam_make_key(info,i,old_key,record,info->lastpos)); + if (_nisam_ck_delete(info,i,old_key)) goto err; + } + + if ((*share->delete_record)(info)) + goto err; /* Remove record from database */ + + info->update= HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_DELETED; + share->state.records--; + + lastpos= (uint32) info->lastpos; + nisam_log_command(LOG_DELETE,info,(byte*) &lastpos,sizeof(lastpos),0); + VOID(_nisam_writeinfo(info,1)); + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(0); + +err: + save_errno=my_errno; + lastpos= (uint32) info->lastpos; + nisam_log_command(LOG_DELETE,info,(byte*) &lastpos, sizeof(lastpos),0); + VOID(_nisam_writeinfo(info,1)); + info->update|=HA_STATE_WRITTEN; /* Buffer changed */ + allow_break(); /* Allow SIGHUP & SIGINT */ + my_errno=save_errno; + if (save_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_CRASHED; + + DBUG_RETURN(-1); +} /* nisam_delete */ + + + /* Tar bort en nyckel till isam-nyckelfilen */ + +int _nisam_ck_delete(register N_INFO *info, uint keynr, uchar *key) +{ + int error; + uint nod_flag; + ulong old_root; + uchar *root_buff; + N_KEYDEF *keyinfo; + DBUG_ENTER("_nisam_ck_delete"); + + if ((old_root=info->s->state.key_root[keynr]) == NI_POS_ERROR) + { + my_errno=HA_ERR_CRASHED; + DBUG_RETURN(-1); + } + keyinfo=info->s->keyinfo+keynr; + if (!(root_buff= (uchar*) my_alloca((uint) keyinfo->base.block_length+ + N_MAX_KEY_BUFF*2))) + DBUG_RETURN(-1); + if (!_nisam_fetch_keypage(info,keyinfo,old_root,root_buff,0)) + { + error= -1; + goto err; + } + if ((error=d_search(info,keyinfo,key,old_root,root_buff)) >0) + { + if (error == 2) + { + DBUG_PRINT("test",("Enlarging of root when deleting")); + error=_nisam_enlarge_root(info,keynr,key); + } + else + { + error=0; + if (getint(root_buff) <= (nod_flag=test_if_nod(root_buff))+3) + { + if (nod_flag) + info->s->state.key_root[keynr]=_nisam_kpos(nod_flag, + root_buff+2+nod_flag); + else + info->s->state.key_root[keynr]= NI_POS_ERROR; + if (_nisam_dispose(info,keyinfo,old_root)) + error= -1; + } + } + } +err: + my_afree((gptr) root_buff); + DBUG_RETURN(error); +} /* _nisam_ck_delete */ + + + /* Tar bort en nyckel under root */ + /* Returnerar 1 om nuvarande buffert minskade */ + /* Returnerar 2 om nuvarande buffert |kar */ + +static int d_search(register N_INFO *info, register N_KEYDEF *keyinfo, uchar *key, ulong page, uchar *anc_buff) +{ + int flag,ret_value,save_flag; + uint length,nod_flag; + uchar *leaf_buff,*keypos,*next_keypos; + ulong leaf_page,next_block; + uchar lastkey[N_MAX_KEY_BUFF]; + DBUG_ENTER("d_search"); + DBUG_DUMP("page",(byte*) anc_buff,getint(anc_buff)); + + flag=(*keyinfo->bin_search)(info,keyinfo,anc_buff,key,0,SEARCH_SAME,&keypos, + lastkey); + nod_flag=test_if_nod(anc_buff); + + leaf_buff=0; + LINT_INIT(leaf_page); + if (nod_flag) + { + leaf_page=_nisam_kpos(nod_flag,keypos); + if (!(leaf_buff= (uchar*) my_alloca((uint) keyinfo->base.block_length+ + N_MAX_KEY_BUFF*2))) + { + my_errno=ENOMEM; + DBUG_RETURN(-1); + } + if (!_nisam_fetch_keypage(info,keyinfo,leaf_page,leaf_buff,0)) + goto err; + } + + if (flag != 0) + { + if (!nod_flag) + { + my_errno=HA_ERR_CRASHED; /* This should newer happend */ + goto err; + } + save_flag=0; + ret_value=d_search(info,keyinfo,key,leaf_page,leaf_buff); + } + else + { /* Found key */ + next_keypos=keypos; /* Find where next block is */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&next_keypos,lastkey)); + next_block=_nisam_kpos(nod_flag,next_keypos); + length=getint(anc_buff); + length-= remove_key(keyinfo,nod_flag,keypos,lastkey,anc_buff+length); + putint(anc_buff,length,nod_flag); + if (!nod_flag) + { /* On leaf page */ + if (_nisam_write_keypage(info,keyinfo,page,anc_buff)) + DBUG_RETURN(-1); + DBUG_RETURN(length <= (uint) keyinfo->base.block_length/2); + } + save_flag=1; + ret_value=del(info,keyinfo,key,anc_buff,leaf_page,leaf_buff,keypos, + next_block,lastkey); + } + if (ret_value >0) + { + save_flag=1; + if (ret_value == 1) + ret_value= underflow(info,keyinfo,anc_buff,leaf_page,leaf_buff,keypos); + else + { /* This happens only with packed keys */ + DBUG_PRINT("test",("Enlarging of key when deleting")); + VOID(_nisam_get_last_key(info,keyinfo,anc_buff,lastkey,keypos)); + ret_value=_nisam_insert(info,keyinfo,key,anc_buff,keypos,lastkey, + (uchar*) 0,(uchar*) 0,0L); + } + } + if (ret_value == 0 && getint(anc_buff) > keyinfo->base.block_length) + { + save_flag=1; + ret_value=_nisam_splitt_page(info,keyinfo,key,anc_buff,lastkey) | 2; + } + if (save_flag) + ret_value|=_nisam_write_keypage(info,keyinfo,page,anc_buff); + else + { + DBUG_DUMP("page",(byte*) anc_buff,getint(anc_buff)); + } + my_afree((byte*) leaf_buff); + DBUG_RETURN(ret_value); +err: + my_afree((byte*) leaf_buff); + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1); +} /* d_search */ + + + /* Remove a key that has a page-reference */ + +static int del(register N_INFO *info, register N_KEYDEF *keyinfo, uchar *key, + uchar *anc_buff, ulong leaf_page, uchar *leaf_buff, + uchar *keypos, /* Pos to where deleted key was */ + ulong next_block, + uchar *ret_key) /* key before keypos in anc_buff */ +{ + int ret_value,length; + uint a_length,nod_flag; + ulong next_page; + uchar keybuff[N_MAX_KEY_BUFF],*endpos,*next_buff,*key_start; + ISAM_SHARE *share=info->s; + S_PARAM s_temp; + DBUG_ENTER("del"); + DBUG_PRINT("enter",("leaf_page: %ld keypos: %lx",leaf_page,keypos)); + DBUG_DUMP("leaf_buff",(byte*) leaf_buff,getint(leaf_buff)); + + endpos=leaf_buff+getint(leaf_buff); + key_start=_nisam_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos); + + if ((nod_flag=test_if_nod(leaf_buff))) + { + next_page= _nisam_kpos(nod_flag,endpos); + if (!(next_buff= (uchar*) my_alloca((uint) keyinfo->base.block_length+ + N_MAX_KEY_BUFF))) + DBUG_RETURN(-1); + if (!_nisam_fetch_keypage(info,keyinfo,next_page,next_buff,0)) + ret_value= -1; + else + { + DBUG_DUMP("next_page",(byte*) next_buff,getint(next_buff)); + if ((ret_value=del(info,keyinfo,key,anc_buff,next_page,next_buff, + keypos,next_block,ret_key)) >0) + { + endpos=leaf_buff+getint(leaf_buff); + if (ret_value == 1) + { + ret_value=underflow(info,keyinfo,leaf_buff,next_page, + next_buff,endpos); + if (ret_value == 0 && getint(leaf_buff) > keyinfo->base.block_length) + { + ret_value=_nisam_splitt_page(info,keyinfo,key,leaf_buff,ret_key) | 2; + } + } + else + { + DBUG_PRINT("test",("Inserting of key when deleting")); + VOID(_nisam_get_last_key(info,keyinfo,leaf_buff,keybuff,endpos)); + ret_value=_nisam_insert(info,keyinfo,key,leaf_buff,endpos,keybuff, + (uchar*) 0,(uchar*) 0,0L); + } + } + if (_nisam_write_keypage(info,keyinfo,leaf_page,leaf_buff)) + goto err; + } + my_afree((byte*) next_buff); + DBUG_RETURN(ret_value); + } + + /* Remove last key from leaf page */ + + putint(leaf_buff,key_start-leaf_buff,nod_flag); + if (_nisam_write_keypage(info,keyinfo,leaf_page,leaf_buff)) + goto err; + + /* Place last key in ancestor page on deleted key position */ + + a_length=getint(anc_buff); + endpos=anc_buff+a_length; + VOID(_nisam_get_last_key(info,keyinfo,anc_buff,ret_key,keypos)); + length=_nisam_get_pack_key_length(keyinfo,share->base.key_reflength, + keypos == endpos ? (uchar*) 0 : keypos, + ret_key,keybuff,&s_temp); + if (length > 0) + bmove_upp((byte*) endpos+length,(byte*) endpos,(uint) (endpos-keypos)); + else + bmove(keypos,keypos-length, (int) (endpos-keypos)+length); + _nisam_store_key(keyinfo,keypos,&s_temp); + /* Save pointer to next leaf */ + VOID((*keyinfo->get_key)(keyinfo,share->base.key_reflength,&keypos,ret_key)); + _nisam_kpointer(info,keypos - share->base.key_reflength,next_block); + putint(anc_buff,a_length+length,share->base.key_reflength); + + DBUG_RETURN( getint(leaf_buff) <= (uint) keyinfo->base.block_length/2 ); +err: + DBUG_RETURN(-1); +} /* del */ + + + /* Balances adjacent pages if underflow occours */ + +static int underflow(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *anc_buff, + ulong leaf_page, /* Ancestor page and underflow page */ + uchar *leaf_buff, + uchar *keypos) /* Position to pos after key */ +{ + int t_length; + uint length,anc_length,buff_length,leaf_length,p_length,s_length,nod_flag; + ulong next_page; + uchar anc_key[N_MAX_KEY_BUFF],leaf_key[N_MAX_KEY_BUFF], + *buff,*endpos,*next_keypos,*half_pos,*temp_pos; + S_PARAM s_temp; + ISAM_SHARE *share=info->s; + DBUG_ENTER("underflow"); + DBUG_PRINT("enter",("leaf_page: %ld keypos: %lx",leaf_page,keypos)); + DBUG_DUMP("anc_buff",(byte*) anc_buff,getint(anc_buff)); + DBUG_DUMP("leaf_buff",(byte*) leaf_buff,getint(leaf_buff)); + + buff=info->buff; + next_keypos=keypos; + nod_flag=test_if_nod(leaf_buff); + p_length=2+nod_flag; + anc_length=getint(anc_buff); + leaf_length=getint(leaf_buff); + info->page_changed=1; + + if ((keypos < anc_buff+anc_length && (share->rnd++ & 1)) || + keypos == anc_buff+2+share->base.key_reflength) + { /* Use page right of anc-page */ + DBUG_PRINT("test",("use right page")); + + VOID((*keyinfo->get_key)(keyinfo,share->base.key_reflength,&next_keypos, + buff)); + next_page= _nisam_kpos(share->base.key_reflength,next_keypos); + if (!_nisam_fetch_keypage(info,keyinfo,next_page,buff,0)) + goto err; + buff_length=getint(buff); + DBUG_DUMP("next",(byte*) buff,buff_length); + + /* find keys to make a big key-page */ + bmove((byte*) next_keypos-share->base.key_reflength,(byte*) buff+2, + share->base.key_reflength); + VOID(_nisam_get_last_key(info,keyinfo,anc_buff,anc_key,next_keypos)); + VOID(_nisam_get_last_key(info,keyinfo,leaf_buff,leaf_key, + leaf_buff+leaf_length)); + + /* merge pages and put parting key from anc_buff between */ + t_length=(int) _nisam_get_pack_key_length(keyinfo,nod_flag,buff+p_length, + (leaf_length == nod_flag+2 ? + (uchar*) 0 : leaf_key), + anc_key,&s_temp); + length=buff_length-p_length; + endpos=buff+length+leaf_length+t_length; + bmove_upp((byte*) endpos, (byte*) buff+buff_length,length); + memcpy((byte*) buff, (byte*) leaf_buff,(size_t) leaf_length); + _nisam_store_key(keyinfo,buff+leaf_length,&s_temp); + buff_length=(uint) (endpos-buff); + putint(buff,buff_length,nod_flag); + + /* remove key from anc_buff */ + + s_length=remove_key(keyinfo,share->base.key_reflength,keypos,anc_key, + anc_buff+anc_length); + putint(anc_buff,(anc_length-=s_length),share->base.key_reflength); + + if (buff_length <= keyinfo->base.block_length) + { /* Keys in one page */ + memcpy((byte*) leaf_buff,(byte*) buff,(size_t) buff_length); + if (_nisam_dispose(info,keyinfo,next_page)) + goto err; + } + else + { /* Page is full */ + VOID(_nisam_get_last_key(info,keyinfo,anc_buff,anc_key,keypos)); + half_pos=_nisam_find_half_pos(info,keyinfo,buff,leaf_key); + length=(uint) (half_pos-buff); + memcpy((byte*) leaf_buff,(byte*) buff,(size_t) length); + putint(leaf_buff,length,nod_flag); + endpos=anc_buff+anc_length; + + /* Correct new keypointer to leaf_page */ + length=(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key); + _nisam_kpointer(info,leaf_key+length,next_page); + /* Save key in anc_buff */ + t_length=(int) _nisam_get_pack_key_length(keyinfo, + share->base.key_reflength, + keypos == endpos ? + (uchar*) 0 : keypos, + anc_key,leaf_key,&s_temp); + if (t_length >= 0) + bmove_upp((byte*) endpos+t_length,(byte*) endpos, + (uint) (endpos-keypos)); + else + bmove(keypos,keypos-t_length,(uint) (endpos-keypos)+t_length); + _nisam_store_key(keyinfo,keypos,&s_temp); + putint(anc_buff,(anc_length+=t_length),share->base.key_reflength); + + /* Store new page */ + if (nod_flag) + bmove((byte*) buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key)); + t_length=(int) _nisam_get_pack_key_length(keyinfo,nod_flag,(uchar*) 0, + (uchar*) 0, leaf_key,&s_temp); + s_temp.n_length= *half_pos; /* For _nisam_store_key */ + length=(buff+getint(buff))-half_pos; + bmove((byte*) buff+p_length+t_length,(byte*) half_pos,(size_t) length); + _nisam_store_key(keyinfo,buff+p_length,&s_temp); + putint(buff,length+t_length+p_length,nod_flag); + + if (_nisam_write_keypage(info,keyinfo,next_page,buff)) + goto err; + } + if (_nisam_write_keypage(info,keyinfo,leaf_page,leaf_buff)) + goto err; + DBUG_RETURN(anc_length <= (uint) keyinfo->base.block_length/2); + } + + DBUG_PRINT("test",("use left page")); + + keypos=_nisam_get_last_key(info,keyinfo,anc_buff,anc_key,keypos); + next_page= _nisam_kpos(share->base.key_reflength,keypos); + if (!_nisam_fetch_keypage(info,keyinfo,next_page,buff,0)) + goto err; + buff_length=getint(buff); + endpos=buff+buff_length; + DBUG_DUMP("prev",(byte*) buff,buff_length); + + /* find keys to make a big key-page */ + bmove((byte*) next_keypos - share->base.key_reflength,(byte*) leaf_buff+2, + share->base.key_reflength); + next_keypos=keypos; + VOID((*keyinfo->get_key)(keyinfo,share->base.key_reflength,&next_keypos, + anc_key)); + VOID(_nisam_get_last_key(info,keyinfo,buff,leaf_key,endpos)); + + /* merge pages and put parting key from anc_buff between */ + t_length=(int) _nisam_get_pack_key_length(keyinfo,nod_flag, + leaf_buff+p_length, + (leaf_length == nod_flag+2 ? + (uchar*) 0 : leaf_key), + anc_key,&s_temp); + if (t_length >= 0) + bmove((byte*) endpos+t_length,(byte*) leaf_buff+p_length, + (size_t) (leaf_length-p_length)); + else /* We gained space */ + bmove((byte*) endpos,(byte*) leaf_buff+((int) p_length-t_length), + (size_t) (leaf_length-p_length+t_length)); + + _nisam_store_key(keyinfo,endpos,&s_temp); + buff_length=buff_length+leaf_length-p_length+t_length; + putint(buff,buff_length,nod_flag); + + /* remove key from anc_buff */ + s_length=remove_key(keyinfo,share->base.key_reflength,keypos,anc_key, + anc_buff+anc_length); + putint(anc_buff,(anc_length-=s_length),share->base.key_reflength); + + if (buff_length <= keyinfo->base.block_length) + { /* Keys in one page */ + if (_nisam_dispose(info,keyinfo,leaf_page)) + goto err; + } + else + { /* Page is full */ + VOID(_nisam_get_last_key(info,keyinfo,anc_buff,anc_key,keypos)); + endpos=half_pos=_nisam_find_half_pos(info,keyinfo,buff,leaf_key); + + /* Correct new keypointer to leaf_page */ + length=(*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key); + _nisam_kpointer(info,leaf_key+length,leaf_page); + /* Save key in anc_buff */ + DBUG_DUMP("anc_buff",(byte*) anc_buff,anc_length); + DBUG_DUMP("key",(byte*) leaf_key,16); + + temp_pos=anc_buff+anc_length; + t_length=(int) _nisam_get_pack_key_length(keyinfo, + share->base.key_reflength, + keypos == temp_pos ? (uchar*) 0 + : keypos, + anc_key,leaf_key,&s_temp); + if (t_length > 0) + bmove_upp((byte*) temp_pos+t_length,(byte*) temp_pos, + (uint) (temp_pos-keypos)); + else + bmove(keypos,keypos-t_length,(uint) (temp_pos-keypos)+t_length); + _nisam_store_key(keyinfo,keypos,&s_temp); + putint(anc_buff,(anc_length+=t_length),share->base.key_reflength); + + /* Store new page */ + if (nod_flag) + bmove((byte*) leaf_buff+2,(byte*) half_pos-nod_flag,(size_t) nod_flag); + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&half_pos,leaf_key)); + t_length=(int) _nisam_get_pack_key_length(keyinfo,nod_flag, (uchar*) 0, + (uchar*) 0, leaf_key, &s_temp); + s_temp.n_length= *half_pos; /* For _nisam_store_key */ + length=(uint) ((buff+buff_length)-half_pos); + bmove((byte*) leaf_buff+p_length+t_length,(byte*) half_pos, + (size_t) length); + _nisam_store_key(keyinfo,leaf_buff+p_length,&s_temp); + putint(leaf_buff,length+t_length+p_length,nod_flag); + putint(buff,endpos-buff,nod_flag); + if (_nisam_write_keypage(info,keyinfo,leaf_page,leaf_buff)) + goto err; + } + if (_nisam_write_keypage(info,keyinfo,next_page,buff)) + goto err; + DBUG_RETURN(anc_length <= (uint) keyinfo->base.block_length/2); +err: + DBUG_RETURN(-1); +} /* underflow */ + + + /* remove a key from packed buffert */ + /* returns how many chars was removed */ + +static uint remove_key(N_KEYDEF *keyinfo, uint nod_flag, + uchar *keypos, /* Where key starts */ + uchar *lastkey, /* key to be removed */ + uchar *page_end) /* End of page */ +{ + int r_length,s_length,first,diff_flag; + uchar *start; + DBUG_ENTER("remove_key"); + DBUG_PRINT("enter",("keypos: %lx page_end: %lx",keypos,page_end)); + + start=keypos; + if (!(keyinfo->base.flag & (HA_PACK_KEY | HA_SPACE_PACK_USED))) + s_length=(int) (keyinfo->base.keylength+nod_flag); + else + { /* Let keypos point at next key */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey)); + s_length=(keypos-start); + if (keyinfo->base.flag & HA_PACK_KEY) + { + diff_flag= (keyinfo->seg[0].base.flag & HA_SPACE_PACK); + first= *start; + if (keypos != page_end && *keypos & 128 && first != 128) + { /* Referens length */ + if ((r_length= *keypos++ & 127) == 0) + { /* Same key after */ + if (first & 128) + start++; /* Skipp ref length */ + if (diff_flag) + start+= *start+1; /* Skipp key length */ + else + start+=keyinfo->seg[0].base.length- (first & 127); + s_length=(keypos-start); /* Remove pointers and next-key-flag */ + } + else if (! (first & 128)) + { /* Deleted key was not compressed */ + if (diff_flag) + { + *start= (uchar) (r_length+ *keypos); + start+=r_length+1; /* Let ref-part remain */ + s_length=(keypos-start)+1; /* Skipp everything between */ + } + else + { + start+=r_length+1; /* Let ref-part remain */ + s_length=(keypos-start); /* Skipp everything between */ + } + } + else if ((first & 127) < r_length) + { /* mid-part of key is used */ + r_length-=(first & 127); + start++; /* Ref remains the same */ + if (diff_flag) + *start++= (uchar) (*keypos++ + r_length); + start+= r_length; + s_length=(keypos-start); /* Skipp everything between */ + } + } + } + } + bmove((byte*) start,(byte*) start+s_length, + (uint) (page_end-start-s_length)); + DBUG_RETURN((uint) s_length); +} /* remove_key */ diff --git a/isam/extra.c b/isam/extra.c new file mode 100644 index 00000000000..dc9b6ce2d19 --- /dev/null +++ b/isam/extra.c @@ -0,0 +1,258 @@ +/* Copyright (C) 2000 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 */ + +/* Extra functions we want to do with a database */ +/* - Set flags for quicker databasehandler */ +/* - Set databasehandler to normal */ +/* - Reset recordpointers as after open database */ + +#include "isamdef.h" +#ifdef HAVE_MMAP +#include <sys/mman.h> +#endif +#ifdef __WIN__ +#include <errno.h> +#endif + + /* set extra flags for database */ + +int nisam_extra(N_INFO *info, enum ha_extra_function function) +{ + int error=0; + DBUG_ENTER("nisam_extra"); + + switch (function) { + case HA_EXTRA_RESET: + info->lastinx= 0; /* Use first index as def */ + info->int_pos=info->lastpos= NI_POS_ERROR; + info->page_changed=1; + /* Next/prev gives first/last */ + if (info->opt_flag & READ_CACHE_USED) + { + VOID(flush_io_cache(&info->rec_cache)); + reinit_io_cache(&info->rec_cache,READ_CACHE,0, + (pbool) (info->lock_type != F_UNLCK), + (pbool) test(info->update & HA_STATE_ROW_CHANGED)); + } + info->update=((info->update & HA_STATE_CHANGED) | + HA_STATE_NEXT_FOUND | HA_STATE_PREV_FOUND); + break; + case HA_EXTRA_CACHE: +#ifndef NO_LOCKING + if (info->lock_type == F_UNLCK && (info->options & HA_OPTION_PACK_RECORD)) + { + error=1; /* Not possibly if not locked */ + my_errno=EACCES; + break; + } +#endif +#if defined(HAVE_MMAP) && defined(HAVE_MADVICE) + if ((info->options & HA_OPTION_COMPRESS_RECORD)) + { + pthread_mutex_lock(&info->s->intern_lock); + if (_nisam_memmap_file(info)) + { + /* We don't nead MADV_SEQUENTIAL if small file */ + madvise(info->s->file_map,info->s->state.data_file_length, + info->s->state.data_file_length <= RECORD_CACHE_SIZE*16 ? + MADV_RANDOM : MADV_SEQUENTIAL); + pthread_mutex_unlock(&info->s->intern_lock); + break; + } + pthread_mutex_unlock(&info->s->intern_lock); + } +#endif + if (info->opt_flag & WRITE_CACHE_USED) + { + info->opt_flag&= ~WRITE_CACHE_USED; + if ((error=end_io_cache(&info->rec_cache))) + break; + } + if (!(info->opt_flag & + (READ_CACHE_USED | WRITE_CACHE_USED | MEMMAP_USED))) + { + if (!(init_io_cache(&info->rec_cache,info->dfile, + (uint) min(info->s->state.data_file_length+1, + my_default_record_cache_size), + READ_CACHE,0L,(pbool) (info->lock_type != F_UNLCK), + MYF(MY_WAIT_IF_FULL)))) + { + info->opt_flag|=READ_CACHE_USED; + info->update&= ~HA_STATE_ROW_CHANGED; + } + /* info->rec_cache.end_of_file=info->s->state.data_file_length; */ + } + break; + case HA_EXTRA_REINIT_CACHE: + if (info->opt_flag & READ_CACHE_USED) + { + reinit_io_cache(&info->rec_cache,READ_CACHE,info->nextpos, + (pbool) (info->lock_type != F_UNLCK), + (pbool) test(info->update & HA_STATE_ROW_CHANGED)); + info->update&= ~HA_STATE_ROW_CHANGED; + /* info->rec_cache.end_of_file=info->s->state.data_file_length; */ + } + break; + case HA_EXTRA_WRITE_CACHE: +#ifndef NO_LOCKING + if (info->lock_type == F_UNLCK) + { + error=1; /* Not possibly if not locked */ + break; + } +#endif + if (!(info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED))) + if (!(init_io_cache(&info->rec_cache,info->dfile,0, + WRITE_CACHE,info->s->state.data_file_length, + (pbool) (info->lock_type != F_UNLCK), + MYF(MY_WAIT_IF_FULL)))) + { + info->opt_flag|=WRITE_CACHE_USED; + info->update&= ~HA_STATE_ROW_CHANGED; + } + break; + case HA_EXTRA_NO_CACHE: + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error=end_io_cache(&info->rec_cache); + } +#if defined(HAVE_MMAP) && defined(HAVE_MADVICE) + if (info->opt_flag & MEMMAP_USED) + madvise(info->s->file_map,info->s->state.data_file_length,MADV_RANDOM); +#endif + break; + case HA_EXTRA_FLUSH_CACHE: + if (info->opt_flag & WRITE_CACHE_USED) + error=flush_io_cache(&info->rec_cache); + break; + case HA_EXTRA_NO_READCHECK: + info->opt_flag&= ~READ_CHECK_USED; /* No readcheck */ + break; + case HA_EXTRA_READCHECK: + info->opt_flag|= READ_CHECK_USED; + break; + case HA_EXTRA_KEYREAD: /* Read only keys to record */ + case HA_EXTRA_REMEMBER_POS: + info->opt_flag |= REMEMBER_OLD_POS; + bmove((byte*) info->lastkey+info->s->base.max_key_length*2, + (byte*) info->lastkey,info->s->base.max_key_length); + info->save_update= info->update; + info->save_lastinx= info->lastinx; + info->save_lastpos= info->lastpos; + if (function == HA_EXTRA_REMEMBER_POS) + break; + /* fall through */ + case HA_EXTRA_KEYREAD_CHANGE_POS: + info->opt_flag |= KEY_READ_USED; + info->read_record=_nisam_read_key_record; + break; + case HA_EXTRA_NO_KEYREAD: + case HA_EXTRA_RESTORE_POS: + if (info->opt_flag & REMEMBER_OLD_POS) + { + bmove((byte*) info->lastkey, + (byte*) info->lastkey+info->s->base.max_key_length*2, + info->s->base.max_key_length); + info->update= info->save_update | HA_STATE_WRITTEN; + info->lastinx= info->save_lastinx; + info->lastpos= info->save_lastpos; + } + info->read_record= info->s->read_record; + info->opt_flag&= ~(KEY_READ_USED | REMEMBER_OLD_POS); + break; + case HA_EXTRA_NO_USER_CHANGE: /* Database is somehow locked agains changes */ + info->lock_type= F_EXTRA_LCK; /* Simulate as locked */ + break; + case HA_EXTRA_WAIT_LOCK: + info->lock_wait=0; + break; + case HA_EXTRA_NO_WAIT_LOCK: + info->lock_wait=MY_DONT_WAIT; + break; + case HA_EXTRA_NO_KEYS: +#ifndef NO_LOCKING + if (info->lock_type == F_UNLCK) + { + error=1; /* Not possibly if not lock */ + break; + } +#endif + info->s->state.keys=0; + info->s->state.key_file_length=info->s->base.keystart; + info->s->changed=1; /* Update on close */ + break; + case HA_EXTRA_FORCE_REOPEN: + pthread_mutex_lock(&THR_LOCK_isam); + info->s->last_version= 0L; /* Impossible version */ +#ifdef __WIN__ + /* Close the isam and data files as Win32 can't drop an open table */ + if (flush_key_blocks(info->s->kfile,FLUSH_RELEASE)) + error=my_errno; + if (info->opt_flag & (READ_CACHE_USED | WRITE_CACHE_USED)) + { + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + error=end_io_cache(&info->rec_cache); + } + if (info->lock_type != F_UNLCK && ! info->was_locked) + { + info->was_locked=info->lock_type; + if (nisam_lock_database(info,F_UNLCK)) + error=my_errno; + } + if (info->s->kfile >= 0 && my_close(info->s->kfile,MYF(0))) + error=my_errno; + { + LIST *list_element ; + for (list_element=nisam_open_list ; + list_element ; + list_element=list_element->next) + { + N_INFO *tmpinfo=(N_INFO*) list_element->data; + if (tmpinfo->s == info->s) + { + if (tmpinfo->dfile >= 0 && my_close(tmpinfo->dfile,MYF(0))) + error = my_errno; + tmpinfo->dfile=-1; + } + } + } + info->s->kfile=-1; /* Files aren't open anymore */ +#endif + pthread_mutex_unlock(&THR_LOCK_isam); + break; + case HA_EXTRA_FLUSH: +#ifdef __WIN__ + if (info->s->not_flushed) + { + info->s->not_flushed=0; + if (_commit(info->s->kfile)) + error=errno; + if (_commit(info->dfile)) + error=errno; + } + break; +#endif + case HA_EXTRA_NORMAL: /* Theese isn't in use */ + case HA_EXTRA_QUICK: + case HA_EXTRA_KEY_CACHE: + case HA_EXTRA_NO_KEY_CACHE: + default: + break; + } + nisam_log_command(LOG_EXTRA,info,(byte*) &function,sizeof(function),error); + DBUG_RETURN(error); +} /* nisam_extra */ diff --git a/isam/info.c b/isam/info.c new file mode 100644 index 00000000000..43c15af908d --- /dev/null +++ b/isam/info.c @@ -0,0 +1,77 @@ +/* Copyright (C) 2000 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 */ + +/* Ger tillbaka en struct med information om isam-filen */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <sys/stat.h> +#endif + +ulong nisam_position(N_INFO *info) +{ + return info->lastpos; +} + + /* If flag == 1 one only gets pos of last record */ + /* if flag == 2 one get current info (no sync from database */ + +int nisam_info(N_INFO *info, register N_ISAMINFO *x, int flag) +{ + struct stat state; + ISAM_SHARE *share=info->s; + DBUG_ENTER("nisam_info"); + + x->recpos = info->lastpos; + if (flag & (HA_STATUS_TIME | HA_STATUS_CONST | HA_STATUS_VARIABLE | + HA_STATUS_ERRKEY | HA_STATUS_NO_LOCK)) + { +#ifndef NO_LOCKING + if (!(flag & HA_STATUS_NO_LOCK)) + { + pthread_mutex_lock(&share->intern_lock); + VOID(_nisam_readinfo(info,F_RDLCK,0)); + VOID(_nisam_writeinfo(info,0)); + pthread_mutex_unlock(&share->intern_lock); + } +#endif + x->records = share->state.records; + x->deleted = share->state.del; + x->delete_length= share->state.empty; + x->keys = share->state.keys; + x->reclength = share->base.reclength; + x->mean_reclength= share->state.records ? + (share->state.data_file_length-share->state.empty)/share->state.records : + share->min_pack_length; + x->data_file_length=share->state.data_file_length; + x->max_data_file_length=share->base.max_data_file_length; + x->index_file_length=share->state.key_file_length; + x->max_index_file_length=share->base.max_key_file_length; + x->filenr = info->dfile; + x->errkey = info->errkey; + x->dupp_key_pos= info->dupp_key_pos; + x->options = share->base.options; + x->create_time=share->base.create_time; + x->isamchk_time=share->base.isamchk_time; + x->rec_per_key=share->base.rec_per_key; + if ((flag & HA_STATUS_TIME) && !fstat(info->dfile,&state)) + x->update_time=state.st_mtime; + else + x->update_time=0; + x->sortkey= -1; /* No clustering */ + } + DBUG_RETURN(0); +} /* nisam_info */ diff --git a/isam/isamchk.c b/isam/isamchk.c new file mode 100644 index 00000000000..97d190b7115 --- /dev/null +++ b/isam/isamchk.c @@ -0,0 +1,3448 @@ +/* Copyright (C) 2000 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 */ + +/* Descript, check and repair of ISAM tables */ + +#include "isamdef.h" + +#include <m_ctype.h> +#include <stdarg.h> +#include <getopt.h> +#ifdef HAVE_SYS_VADVICE_H +#include <sys/vadvise.h> +#endif +#ifdef HAVE_SYS_MMAN_H +#include <sys/mman.h> +#endif +SET_STACK_SIZE(9000) /* Minimum stack size for program */ + +#define T_VERBOSE 1 +#define T_SILENT 2 +#define T_DESCRIPT 4 +#define T_EXTEND 8 +#define T_INFO 16 +#define T_REP 32 +#define T_OPT 64 /* Not currently used */ +#define T_FORCE_CREATE 128 +#define T_WRITE_LOOP 256 +#define T_UNPACK 512 +#define T_STATISTICS 1024 +#define T_VERY_SILENT 2048 +#define T_SORT_RECORDS 4096 +#define T_SORT_INDEX 8192 +#define T_WAIT_FOREVER 16384 +#define T_REP_BY_SORT 32768 + + +#define O_NEW_INDEX 1 /* Bits set in out_flag */ +#define O_NEW_DATA 2 + +#if defined(_MSC_VER) && !defined(__WIN__) +#define USE_BUFFER_INIT 250L*1024L +#define READ_BUFFER_INIT ((uint) 32768-MALLOC_OVERHEAD) +#define SORT_BUFFER_INIT (uint) (65536L-MALLOC_OVERHEAD) +#define MIN_SORT_BUFFER (1024*16-MALLOC_OVERHEAD) +#else +#define USE_BUFFER_INIT (((1024L*512L-MALLOC_OVERHEAD)/IO_SIZE)*IO_SIZE) +#define READ_BUFFER_INIT (1024L*256L-MALLOC_OVERHEAD) +#define SORT_BUFFER_INIT (2048L*1024L-MALLOC_OVERHEAD) +#define MIN_SORT_BUFFER (4096-MALLOC_OVERHEAD) +#endif + +#define NEAD_MEM ((uint) 10*4*(IO_SIZE+32)+32) /* Nead for recursion */ +#define MAXERR 20 +#define BUFFERS_WHEN_SORTING 16 /* Alloc for sort-key-tree */ +#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE +#define INDEX_TMP_EXT ".TMM" +#define DATA_TMP_EXT ".TMD" +#define MYF_RW MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL) + +#define UPDATE_TIME 1 +#define UPDATE_STAT 2 +#define UPDATE_SORT 4 + +typedef struct st_sort_key_blocks { /* Used when sorting */ + uchar *buff,*end_pos; + uchar lastkey[N_MAX_POSSIBLE_KEY_BUFF]; + uint last_length; + int inited; +} SORT_KEY_BLOCKS; + +typedef struct st_sort_info { + N_INFO *info; + enum data_file_type new_data_file_type; + SORT_KEY_BLOCKS *key_block,*key_block_end; + uint key,find_length; + ulong pos,max_pos,filepos,start_recpos,filelength,dupp,max_records,unique, + buff_length; + my_bool fix_datafile; + char *record,*buff; + N_KEYDEF *keyinfo; + N_KEYSEG *keyseg; +} SORT_INFO; + +enum options {OPT_CHARSETS_DIR=256}; + +static ulong use_buffers=0,read_buffer_length=0,write_buffer_length=0, + sort_buffer_length=0,sort_key_blocks=0,crc=0,unique_count=0; +static uint testflag=0,out_flag=0,warning_printed=0,error_printed=0, + rep_quick=0,verbose=0,opt_follow_links=1; +static uint opt_sort_key=0,total_files=0,max_level=0,max_key=N_MAXKEY; +static ulong keydata=0,totaldata=0,key_blocks=0; +static ulong new_file_pos=0,record_checksum=0,key_file_blocks=0,decode_bits; +static ulong total_records=0,total_deleted=0; +static ulong search_after_block=NI_POS_ERROR; +static byte *record_buff; +static char **defaults_alloc; +static const char *type_names[]= +{ "?","text","binary", "short", "long", "float", + "double","number","unsigned short", + "unsigned long","longlong","ulonglong","int24", + "uint24","int8","?",}, + *packed_txt="packed ", + *diff_txt="stripped ", + *field_pack[]={"","no endspace", "no prespace", + "no zeros", "blob", "constant", "table-lookup", + "always zero","?","?",}; + +static char temp_filename[FN_REFLEN], *isam_file_name, *default_charset; +static IO_CACHE read_cache; +static SORT_INFO sort_info; +static int tmpfile_createflag=O_RDWR | O_TRUNC | O_EXCL; + +static const char *load_default_groups[]= { "isamchk",0 }; + + /* Functions defined in this file */ + +extern int main(int argc,char * *argv); +extern void print_error _VARARGS((const char *fmt,...)); +static void print_warning _VARARGS((const char *fmt,...)); +static void print_info _VARARGS((const char *fmt,...)); +static int nisamchk(char *filename); +static void get_options(int *argc,char * * *argv); +static int chk_del(N_INFO *info,uint testflag); +static int check_k_link(N_INFO *info,uint nr); +static int chk_size(N_INFO *info); +static int chk_key(N_INFO *info); +static int chk_index(N_INFO *info,N_KEYDEF *keyinfo,ulong page,uchar *buff, + ulong *keys,uint level); +static uint isam_key_length(N_INFO *info,N_KEYDEF *keyinfo); +static unsigned long calc_checksum(ulong count); +static int chk_data_link(N_INFO *info,int extend); +static int rep(N_INFO *info,char *name); +static int writekeys(N_INFO *info,byte *buff,ulong filepos); +static void descript(N_INFO *info,char *name); +static int movepoint(N_INFO *info,byte *record,ulong oldpos,ulong newpos, + uint prot_key); +static void lock_memory(void); +static int flush_blocks(File file); +static int sort_records(N_INFO *,my_string,uint,int); +static int sort_index(N_INFO *info,my_string name); +static int sort_record_index(N_INFO *info,N_KEYDEF *keyinfo,ulong page, + uchar *buff,uint sortkey,File new_file); +static int sort_one_index(N_INFO *info,N_KEYDEF *keyinfo,uchar *buff, + File new_file); +static int change_to_newfile(const char * filename,const char * old_ext, + const char * new_ext); +static int lock_file(File file,ulong start,int lock_type,const char* filetype, + const char *filename); +static int filecopy(File to,File from,ulong start,ulong length, + const char * type); + +static void print_version(void); +static int rep_by_sort(N_INFO *info,my_string name); +static int sort_key_read(void *key); +static int sort_get_next_record(void); +static int sort_write_record(void); +static int sort_key_cmp(const void *not_used, const void *a,const void *b); +static int sort_key_write(const void *a); +static ulong get_record_for_key(N_INFO *info,N_KEYDEF *keyinfo, + uchar *key); +static int sort_insert_key(reg1 SORT_KEY_BLOCKS *key_block,uchar *key, + ulong prev_block); +static int sort_delete_record(void); +static void usage(void); +static int flush_pending_blocks(void); +static SORT_KEY_BLOCKS *alloc_key_blocks(uint blocks,uint buffer_length); +static int test_if_almost_full(N_INFO *info); +static int recreate_database(N_INFO **info,char *filename); +static void save_integer(byte *pos,uint pack_length,ulong value); +static int write_data_suffix(N_INFO *info); +static int update_state_info(N_INFO *info,uint update); + + + /* Main program */ + +int main(argc,argv) +int argc; +char **argv; +{ + int error; + MY_INIT(argv[0]); + + get_options(&argc,(char***) &argv); + nisam_quick_table_bits=(uint) decode_bits; + error=0; + while (--argc >= 0) + { + error|= nisamchk(*(argv++)); + VOID(fflush(stdout)); + VOID(fflush(stderr)); + if ((error_printed | warning_printed) && (testflag & T_FORCE_CREATE) && + (!(testflag & (T_REP | T_REP_BY_SORT | T_SORT_RECORDS | + T_SORT_INDEX)))) + { + testflag|=T_REP; + error|=nisamchk(argv[-1]); + testflag&= ~T_REP; + VOID(fflush(stdout)); + VOID(fflush(stderr)); + } + if (argc && (!(testflag & T_SILENT) || testflag & T_INFO)) + { + puts("\n---------\n"); + VOID(fflush(stdout)); + } + } + if (total_files > 1) + { /* Only if descript */ + if (!(testflag & T_SILENT) || testflag & T_INFO) + puts("\n---------\n"); + printf("\nTotal of all %d ISAM-files:\nData records: %8lu Deleted blocks: %8lu\n",total_files,total_records,total_deleted); + } + free_defaults(defaults_alloc); + my_end(testflag & T_INFO ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error); +#ifndef _lint + return 0; /* No compiler warning */ +#endif +} /* main */ + + +static CHANGEABLE_VAR changeable_vars[] = { + { "key_buffer_size",(long*) &use_buffers,(long) USE_BUFFER_INIT, + (long) MALLOC_OVERHEAD, (long) ~0L,(long) MALLOC_OVERHEAD,(long) IO_SIZE }, + { "read_buffer_size", (long*) &read_buffer_length,(long) READ_BUFFER_INIT, + (long) MALLOC_OVERHEAD,(long) ~0L,(long) MALLOC_OVERHEAD,(long) 1L }, + { "write_buffer_size", (long*) &write_buffer_length,(long) READ_BUFFER_INIT, + (long) MALLOC_OVERHEAD,(long) ~0L,(long) MALLOC_OVERHEAD,(long) 1L }, + { "sort_buffer_size",(long*) &sort_buffer_length,(long) SORT_BUFFER_INIT, + (long) (MIN_SORT_BUFFER+MALLOC_OVERHEAD),(long) ~0L, + (long) MALLOC_OVERHEAD,(long) 1L }, + { "sort_key_blocks",(long*) &sort_key_blocks,BUFFERS_WHEN_SORTING,4L,100L,0L, + 1L }, + { "decode_bits",(long*) &decode_bits,9L,4L,17L,0L,1L }, + { NullS,(long*) 0,0L,0L,0L,0L,0L,} }; + + +static struct option long_options[] = +{ + {"analyze", no_argument, 0, 'a'}, + {"character-sets-dir", required_argument, 0, OPT_CHARSETS_DIR}, +#ifndef DBUG_OFF + {"debug", required_argument, 0, '#'}, +#endif + {"default-character-set", required_argument, 0, 'C'}, + {"description", no_argument, 0, 'd'}, + {"extend-check", no_argument, 0, 'e'}, + {"information", no_argument, 0, 'i'}, + {"force", no_argument, 0, 'f'}, + {"help", no_argument, 0, '?'}, + {"keys-used", required_argument, 0, 'k'}, + {"no-symlinks", no_argument, 0, 'l'}, + {"quick", no_argument, 0, 'q'}, + {"recover", no_argument, 0, 'r'}, + {"safe-recover", no_argument, 0, 'o'}, + {"block-search", required_argument, 0, 'b'}, + {"set-variable", required_argument, 0, 'O'}, + {"silent", no_argument, 0, 's'}, + {"sort-index", no_argument, 0, 'S'}, + {"sort-records", required_argument, 0, 'R'}, + {"unpack", no_argument, 0, 'u'}, + {"verbose", no_argument, 0, 'v'}, + {"version", no_argument, 0, 'V'}, + {"wait", no_argument, 0, 'w'}, + {0, 0, 0, 0} +}; + +static void print_version(void) +{ + printf("%s Ver 5.15 for %s at %s\n",my_progname,SYSTEM_TYPE, + MACHINE_TYPE); +} + +static void usage(void) +{ + uint i; + print_version(); + puts("TCX Datakonsult AB, by Monty, for your professional use"); + puts("This software comes with NO WARRANTY: see the PUBLIC for details.\n"); + puts("Description, check and repair of ISAM tables."); + puts("Used without options all tables on the command will be checked for errors"); + printf("Usage: %s [OPTIONS] tables[.ISM]\n", my_progname); + puts("\n\ + -a, --analyze Analyze distribution of keys. Will make some joins in\n\ + MySQL faster.\n\ + -#, --debug=... Output debug log. Often this is 'd:t:o,filename`\n\ + --character-sets-dir=...\n\ + Directory where character sets are\n\ + -C, --default-character-set=...\n\ + Set the default character set\n\ + -d, --description Prints some information about table.\n\ + -e, --extend-check Check the table VERY thoroughly. One need use this\n\ + only in extreme cases as isamchk should normally find\n\ + all errors even without this switch\n\ + -f, --force Overwrite old temporary files.\n\ + If one uses -f when checking tables (running isamchk\n\ + without -r), isamchk will automatically restart with\n\ + -r on any wrong table.\n\ + -?, --help Display this help and exit.\n\ + -i, --information Print statistics information about the table\n\ + -k, --keys-used=# Used with '-r'. Tell ISAM to update only the first\n\ + # keys. This can be used to get faster inserts!\n\ + -l, --no-symlinks Do not follow symbolic links when repairing. Normally\n\ + isamchk repairs the table a symlink points at.\n\ + -q, --quick Used with -r to get a faster repair. (The data file\n\ + isn't touched.) One can give a second '-q' to force\n\ + isamchk to modify the original datafile."); + puts("\ + -r, --recover Can fix almost anything except unique keys that aren't\n\ + unique.\n\ + -o, --safe-recover Uses old recovery method; slower than '-r' but can\n\ + handle a couple of cases that '-r' cannot handle.\n\ + -O, --set-variable var=option\n\ + Change the value of a variable.\n\ + -s, --silent Only print errors. One can use two -s to make isamchk\n\ + very silent\n\ + -S, --sort-index Sort index blocks. This speeds up 'read-next' in\n\ + applications\n\ + -R, --sort-records=#\n\ + Sort records according to an index. This makes your\n\ + data much more localized and may speed up things\n\ + (It may be VERY slow to do a sort the first time!)\n\ + -u, --unpack Unpack file packed with pack_isam.\n\ + -v, --verbose Print more information. This can be used with\n\ + -d and -e. Use many -v for more verbosity!\n\ + -V, --version Print version and exit.\n\ + -w, --wait Wait if table is locked."); + print_defaults("my",load_default_groups); + printf("\nPossible variables for option --set-variable (-O) are:\n"); + for (i=0; changeable_vars[i].name ; i++) + printf("%-20s current value: %lu\n", + changeable_vars[i].name, + *changeable_vars[i].varptr); +} + + /* Check table */ + +static int nisamchk(my_string filename) +{ + int error,lock_type,recreate; + N_INFO *info; + File datafile; + char fixed_name[FN_REFLEN]; + ISAM_SHARE *share; + DBUG_ENTER("nisamchk"); + + out_flag=error=warning_printed=error_printed=recreate=0; + datafile=0; + isam_file_name=filename; /* For error messages */ + if (!(info=nisam_open(filename, + (testflag & T_DESCRIPT) ? O_RDONLY : O_RDWR, + (testflag & T_WAIT_FOREVER) ? HA_OPEN_WAIT_IF_LOCKED : + (testflag & T_DESCRIPT) ? HA_OPEN_IGNORE_IF_LOCKED : + HA_OPEN_ABORT_IF_LOCKED))) + { + /* Avoid twice printing of isam file name */ + error_printed=1; + switch (my_errno) { + case HA_ERR_CRASHED: + print_error("'%s' is not a ISAM-table",filename); + break; + case HA_ERR_OLD_FILE: + print_error("'%s' is a old type of ISAM-table", filename); + break; + case HA_ERR_END_OF_FILE: + print_error("Couldn't read compleat header from '%s'", filename); + break; + case EAGAIN: + print_error("'%s' is locked. Use -w to wait until unlocked",filename); + break; + case ENOENT: + print_error("File '%s' doesn't exist",filename); + break; + case EACCES: + print_error("You don't have permission to use '%s'",filename); + break; + default: + print_error("%d when opening ISAM-table '%s'", + my_errno,filename); + break; + } + DBUG_RETURN(1); + } + share=info->s; + share->base.options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ + share->r_locks=0; + if ((testflag & (T_REP_BY_SORT | T_REP | T_STATISTICS | + T_SORT_RECORDS | T_SORT_INDEX)) && + ((testflag & T_UNPACK && share->data_file_type == COMPRESSED_RECORD) || + share->state_length != sizeof(share->state) || + uint2korr(share->state.header.base_info_length) != + sizeof(share->base) || + (max_key && ! share->state.keys && share->base.keys) || + test_if_almost_full(info) || + info->s->state.header.file_version[3] != nisam_file_magic[3])) + { + if (recreate_database(&info,filename)) + { + VOID(fprintf(stderr, + "ISAM-table '%s' is not fixed because of errors\n", + filename)); + return(-1); + } + recreate=1; + if (!(testflag & (T_REP | T_REP_BY_SORT))) + { + testflag|=T_REP_BY_SORT; /* if only STATISTICS */ + if (!(testflag & T_SILENT)) + printf("- '%s' has old table-format. Recreating index\n",filename); + if (!rep_quick) + rep_quick=1; + } + share=info->s; + share->r_locks=0; + } + + if (testflag & T_DESCRIPT) + { + total_files++; + total_records+=share->state.records; total_deleted+=share->state.del; + descript(info,filename); + } + else + { + if (testflag & (T_REP+T_REP_BY_SORT+T_OPT+T_SORT_RECORDS+T_SORT_INDEX)) + lock_type = F_WRLCK; /* table is changed */ + else + lock_type= F_RDLCK; + if (info->lock_type == F_RDLCK) + info->lock_type=F_UNLCK; /* Read only table */ + if (_nisam_readinfo(info,lock_type,0)) + { + print_error("Can't lock indexfile of '%s', error: %d", + filename,my_errno); + error_printed=0; + goto end2; + } + share->w_locks++; /* Mark (for writeinfo) */ + if (lock_file(info->dfile,0L,lock_type,"datafile of",filename)) + goto end; + info->lock_type= F_EXTRA_LCK; /* Simulate as locked */ + info->tmp_lock_type=lock_type; + datafile=info->dfile; + if (testflag & (T_REP+T_REP_BY_SORT+T_SORT_RECORDS+T_SORT_INDEX)) + { + if (testflag & (T_REP+T_REP_BY_SORT)) + share->state.keys=min(share->base.keys,max_key); + VOID(fn_format(fixed_name,filename,"",N_NAME_IEXT, + 4+ (opt_follow_links ? 16 : 0))); + + if (rep_quick && (error=chk_del(info,testflag & ~T_VERBOSE))) + print_error("Quick-recover aborted; Run recovery without switch 'q'"); + else + { + if (testflag & T_REP_BY_SORT && + (share->state.keys || (rep_quick && !max_key && ! recreate))) + error=rep_by_sort(info,fixed_name); + else if (testflag & (T_REP | T_REP_BY_SORT)) + error=rep(info,fixed_name); + } + if (!error && testflag & T_SORT_RECORDS) + { + if (out_flag & O_NEW_DATA) + { /* Change temp file to org file */ + VOID(lock_file(datafile,0L,F_UNLCK,"datafile of",filename)); + VOID(my_close(datafile,MYF(MY_WME))); /* Close old file */ + VOID(my_close(info->dfile,MYF(MY_WME))); /* Close new file */ + error|=change_to_newfile(fixed_name,N_NAME_DEXT,DATA_TMP_EXT); + if ((info->dfile=my_open(fn_format(temp_filename,fixed_name,"", + N_NAME_DEXT,2+4), + O_RDWR | O_SHARE, + MYF(MY_WME))) <= 0 || + lock_file(info->dfile,0L,F_WRLCK,"datafile",temp_filename)) + error=1; + out_flag&= ~O_NEW_DATA; /* We are using new datafile */ + read_cache.file=info->dfile; + } + if (! error) + error=sort_records(info,fixed_name,opt_sort_key, + test(!(testflag & T_REP))); + datafile=info->dfile; /* This is now locked */ + } + if (!error && testflag & T_SORT_INDEX) + error=sort_index(info,fixed_name); + } + else + { + if (!(testflag & T_SILENT) || testflag & T_INFO) + printf("Checking ISAM file: %s\n",filename); + if (!(testflag & T_SILENT)) + printf("Data records: %7ld Deleted blocks: %7ld\n", + share->state.records,share->state.del); + share->state.keys=min(share->state.keys,max_key); + error =chk_size(info); + error|=chk_del(info,testflag); + error|=chk_key(info); + if (!rep_quick) + { + if (testflag & T_EXTEND) + VOID(init_key_cache(use_buffers,(uint) NEAD_MEM)); + VOID(init_io_cache(&read_cache,datafile,(uint) read_buffer_length, + READ_CACHE,share->pack.header_length,1, + MYF(MY_WME))); + lock_memory(); + error|=chk_data_link(info,testflag & T_EXTEND); + error|=flush_blocks(share->kfile); + VOID(end_io_cache(&read_cache)); + } + } + } +end: + if (!(testflag & T_DESCRIPT)) + { + if (info->update & (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED)) + error|=update_state_info(info, + ((testflag & (T_REP | T_REP_BY_SORT)) ? + UPDATE_TIME | UPDATE_STAT : 0) | + ((testflag & T_SORT_RECORDS) ? + UPDATE_SORT : 0)); + VOID(lock_file(share->kfile,0L,F_UNLCK,"indexfile",filename)); + if (datafile > 0) + VOID(lock_file(datafile,0L,F_UNLCK,"datafile of",filename)); + info->update&= ~(HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + } + share->w_locks--; +end2: + if (datafile && datafile != info->dfile) + VOID(my_close(datafile,MYF(MY_WME))); + if (nisam_close(info)) + { + print_error("%d when closing ISAM-table '%s'",my_errno,filename); + DBUG_RETURN(1); + } + if (error == 0) + { + if (out_flag & O_NEW_DATA) + error|=change_to_newfile(fixed_name,N_NAME_DEXT,DATA_TMP_EXT); + if (out_flag & O_NEW_INDEX) + error|=change_to_newfile(fixed_name,N_NAME_IEXT,INDEX_TMP_EXT); + } + VOID(fflush(stdout)); VOID(fflush(stderr)); + if (error_printed) + { + if (testflag & (T_REP+T_REP_BY_SORT+T_SORT_RECORDS+T_SORT_INDEX)) + VOID(fprintf(stderr, + "ISAM-table '%s' is not fixed because of errors\n", + filename)); + else if (! (error_printed & 2) && !(testflag & T_FORCE_CREATE)) + VOID(fprintf(stderr, + "ISAM-table '%s' is corrupted\nFix it using switch \"-r\" or \"-o\"\n", + filename)); + } + else if (warning_printed && + ! (testflag & (T_REP+T_REP_BY_SORT+T_SORT_RECORDS+T_SORT_INDEX+ + T_FORCE_CREATE))) + VOID(fprintf(stderr, "ISAM-table '%s' is useable but should be fixed\n", + filename)); + VOID(fflush(stderr)); + DBUG_RETURN(error); +} /* nisamchk */ + + + /* Read options */ + +static void get_options(register int *argc,register char ***argv) +{ + int c,option_index=0; + + load_defaults("my",load_default_groups,argc,argv); + defaults_alloc= *argv; + set_all_changeable_vars(changeable_vars); + if (isatty(fileno(stdout))) + testflag|=T_WRITE_LOOP; + while ((c=getopt_long(*argc,*argv,"adeif?lqrosSuvVw#:b:k:O:R:C:", + long_options, &option_index)) != EOF) + { + switch(c) { + case 'a': + testflag|= T_STATISTICS; + break; + case 'C': + default_charset=optarg; + break; + case OPT_CHARSETS_DIR: + charsets_dir = optarg; + break; + case 'b': + search_after_block=strtoul(optarg,NULL,10); + break; + case 's': /* silent */ + if (testflag & T_SILENT) + testflag|=T_VERY_SILENT; + testflag|= T_SILENT; + testflag&= ~T_WRITE_LOOP; + break; + case 'w': + testflag|= T_WAIT_FOREVER; + break; + case 'd': /* description if isam-file */ + testflag|= T_DESCRIPT; + break; + case 'e': /* extend check */ + testflag|= T_EXTEND; + break; + case 'i': + testflag|= T_INFO; + break; + case 'f': + tmpfile_createflag= O_RDWR | O_TRUNC; + testflag|=T_FORCE_CREATE; + break; + case 'k': + max_key= (uint) atoi(optarg); + break; + case 'l': + opt_follow_links=0; + break; + case 'r': /* Repair table */ + testflag= (testflag & ~T_REP) | T_REP_BY_SORT; + break; + case 'o': + testflag= (testflag & ~T_REP_BY_SORT) | T_REP; + my_disable_async_io=1; /* More safety */ + break; + case 'q': + rep_quick++; + break; + case 'u': + testflag|= T_UNPACK | T_REP_BY_SORT; + break; + case 'v': /* Verbose */ + testflag|= T_VERBOSE; + verbose++; + break; + case 'O': + if (set_changeable_var(optarg, changeable_vars)) + { + usage(); + exit(1); + } + break; + case 'R': /* Sort records */ + testflag|= T_SORT_RECORDS; + opt_sort_key=(uint) atoi(optarg)-1; + if (opt_sort_key >= N_MAXKEY) + { + fprintf(stderr, + "The value of the sort key is bigger than max key: %d.\n", + N_MAXKEY); + exit(1); + } + break; + case 'S': /* Sort index */ + testflag|= T_SORT_INDEX; + break; + case '#': + DBUG_PUSH(optarg ? optarg : "d:t:o,/tmp/isamchk.trace"); + break; + case 'V': + print_version(); + exit(0); + case '?': + usage(); + exit(0); + } + } + (*argc)-=optind; + (*argv)+=optind; + if (*argc == 0) + { + usage(); + exit(-1); + } + if ((testflag & T_UNPACK) && (rep_quick || (testflag & T_SORT_RECORDS))) + { + VOID(fprintf(stderr, + "%s: --unpack can't be used with --quick or --sort-records\n", + my_progname)); + exit(1); + } + if (default_charset) + { + if (set_default_charset_by_name(default_charset, MYF(MY_WME))) + exit(1); + } + return; +} /* get options */ + + + /* Check delete links */ + +static int chk_del(info,test_flag) +reg1 N_INFO *info; +uint test_flag; +{ + reg2 ulong i; + uint j,delete_link_length; + ulong empty,next_link; + uchar buff[8]; + DBUG_ENTER("chk_del"); + if (!(test_flag & T_SILENT)) puts("- check delete-chain"); + + record_checksum=0L; + key_file_blocks=info->s->base.keystart; + for (j =0 ; j < info->s->state.keys ; j++) + if (check_k_link(info,j)) + goto wrong; + delete_link_length=(info->s->base.options & HA_OPTION_PACK_RECORD) ? 8 : 5; + + next_link=info->s->state.dellink; + if (info->s->state.del == 0) + { + if (test_flag & T_VERBOSE) + { + puts("No recordlinks"); + } + } + else + { + if (test_flag & T_VERBOSE) + printf("Recordlinks: "); + empty=0; + for (i= info->s->state.del ; i > 0L && next_link != NI_POS_ERROR ; i--) + { + if (test_flag & T_VERBOSE) printf("%10lu",next_link); + if (next_link >= info->s->state.data_file_length) + goto wrong; + if (my_pread(info->dfile,(char*) buff,delete_link_length, + next_link,MYF(MY_NABP))) + { + if (test_flag & T_VERBOSE) puts(""); + print_error("Can't read delete-link at filepos: %lu", + (ulong) next_link); + DBUG_RETURN(1); + } + if (*buff != '\0') + { + if (test_flag & T_VERBOSE) puts(""); + print_error("Record at pos: %lu is not remove-marked", + (ulong) next_link); + goto wrong; + } + if (info->s->base.options & HA_OPTION_PACK_RECORD) + { + next_link=uint4korr(buff+4); + empty+=uint3korr(buff+1); + } + else + { + record_checksum+=next_link; + next_link=uint4korr(buff+1); + empty+=info->s->base.reclength; + } + if (next_link == (uint32) ~0) /* Fix for 64 bit long */ + next_link=NI_POS_ERROR; + } + if (empty != info->s->state.empty) + { + if (test_flag & T_VERBOSE) puts(""); + print_warning("Not used space is supposed to be: %lu but is: %lu", + (ulong) info->s->state.empty,(ulong) empty); + info->s->state.empty=empty; + } + if (i != 0 || next_link != NI_POS_ERROR) + goto wrong; + + if (test_flag & T_VERBOSE) puts("\n"); + } + DBUG_RETURN(0); +wrong: + if (test_flag & T_VERBOSE) puts(""); + print_error("delete-link-chain corrupted"); + DBUG_RETURN(1); +} /* chk_del */ + + + /* Kontrollerar l{nkarna i nyckelfilen */ + +static int check_k_link(info,nr) +register N_INFO *info; +uint nr; +{ + ulong next_link,records; + DBUG_ENTER("check_k_link"); + + if (testflag & T_VERBOSE) + printf("index %2d: ",nr+1); + + next_link=info->s->state.key_del[nr]; + records= (info->s->state.key_file_length / + info->s->keyinfo[nr].base.block_length); + while (next_link != NI_POS_ERROR && records > 0) + { + if (testflag & T_VERBOSE) printf("%10lu",next_link); + if (next_link > info->s->state.key_file_length || + next_link & (info->s->blocksize-1)) + DBUG_RETURN(1); + if (my_pread(info->s->kfile,(char*) &next_link,sizeof(long),next_link, + MYF(MY_NABP))) + DBUG_RETURN(1); + records--; + key_file_blocks+=info->s->keyinfo[nr].base.block_length; + } + if (testflag & T_VERBOSE) + { + if (next_link != NI_POS_ERROR) + printf("%10lu\n",next_link); + else + puts(""); + } + DBUG_RETURN (next_link != NI_POS_ERROR); +} /* check_k_link */ + + + /* Kontrollerar storleken p} filerna */ + +static int chk_size(register N_INFO *info) +{ + int error=0; + register my_off_t skr,size; + DBUG_ENTER("chk_size"); + + if (!(testflag & T_SILENT)) puts("- check file-size"); + + size=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); + if ((skr=(my_off_t) info->s->state.key_file_length) != size) + { + if (skr > size) + { + error=1; + print_error("Size of indexfile is: %-8lu Should be: %lu", + (ulong) size, (ulong) skr); + } + else + print_warning("Size of indexfile is: %-8lu Should be: %lu", + (ulong) size,(ulong) skr); + } + if (!(testflag & T_VERY_SILENT) && + ! (info->s->base.options & HA_OPTION_COMPRESS_RECORD) && + info->s->state.key_file_length > + (ulong) (ulong_to_double(info->s->base.max_key_file_length)*0.9)) + print_warning("Keyfile is almost full, %10lu of %10lu used", + info->s->state.key_file_length, + info->s->base.max_key_file_length-1); + + size=my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + skr=(my_off_t) info->s->state.data_file_length; + if (info->s->base.options & HA_OPTION_COMPRESS_RECORD) + skr+= MEMMAP_EXTRA_MARGIN; +#ifdef USE_RELOC + if (info->data_file_type == STATIC_RECORD && + skr < (my_off_t) info->s->base.reloc*info->s->base.min_pack_length) + skr=(my_off_t) info->s->base.reloc*info->s->base.min_pack_length; +#endif + if (skr != size) + { + info->s->state.data_file_length=(ulong) size; /* Skipp other errors */ + if (skr > size && skr != size + MEMMAP_EXTRA_MARGIN) + { + error=1; + print_error("Size of datafile is: %-8lu Should be: %lu", + (ulong) size,(ulong) skr); + } + else + { + print_warning("Size of datafile is: %-8lu Should be: %lu", + (ulong) size,(ulong) skr); + + } + } + if (!(testflag & T_VERY_SILENT) && + !(info->s->base.options & HA_OPTION_COMPRESS_RECORD) && + info->s->state.data_file_length > + (ulong) (ulong_to_double(info->s->base.max_data_file_length)*0.9)) + print_warning("Datafile is almost full, %10lu of %10lu used", + info->s->state.data_file_length, + info->s->base.max_data_file_length-1); + DBUG_RETURN(error); +} /* chk_size */ + + + /* Kontrollerar nycklarna */ + +static int chk_key(info) +register N_INFO *info; +{ + uint key; + ulong keys,all_keydata,all_totaldata,key_totlength,length, + init_checksum,old_record_checksum; + ISAM_SHARE *share=info->s; + N_KEYDEF *keyinfo; + DBUG_ENTER("chk_key"); + + if (!(testflag & T_SILENT)) puts("- check index reference"); + + all_keydata=all_totaldata=key_totlength=old_record_checksum=0; + init_checksum=record_checksum; + if (!(share->base.options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + old_record_checksum=calc_checksum(share->state.records+share->state.del-1)* + share->base.reclength; + for (key= 0,keyinfo= &share->keyinfo[0]; key < share->state.keys ; + key++,keyinfo++) + { + record_checksum=init_checksum; + unique_count=0L; + if ((!(testflag & T_SILENT)) && share->state.keys >1) + printf ("- check data record references index: %d\n",key+1); + if (share->state.key_root[key] == NI_POS_ERROR && + share->state.records == 0) + continue; + if (!_nisam_fetch_keypage(info,keyinfo,share->state.key_root[key],info->buff, + 0)) + { + print_error("Can't read indexpage from filepos: %lu", + (ulong) share->state.key_root[key]); + DBUG_RETURN(-1); + } + key_file_blocks+=keyinfo->base.block_length; + keys=keydata=totaldata=key_blocks=0; max_level=0; + if (chk_index(info,keyinfo,share->state.key_root[key],info->buff,&keys,1)) + DBUG_RETURN(-1); + if (keys != share->state.records) + { + print_error("Found %lu keys of %lu",(ulong) keys, + (ulong) share->state.records); + DBUG_RETURN(-1); + } + if (!key && (share->base.options & + (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD))) + old_record_checksum=record_checksum; + else if (old_record_checksum != record_checksum) + { + if (key) + print_error("Key %u doesn't point at same records that key 1", + key+1); + else + print_error("Key 1 doesn't point at all records"); + DBUG_RETURN(-1); + } + length=(ulong) isam_key_length(info,keyinfo)*keys + key_blocks*2; + if (testflag & T_INFO && totaldata != 0L && keys != 0L) + printf("Key: %2d: Keyblocks used: %3d%% Packed: %4d%% Max levels: %2d\n", + key+1, + (int) (keydata*100.0/totaldata), + (int) ((long) (length-keydata)*100.0/(double) length), + max_level); + all_keydata+=keydata; all_totaldata+=totaldata; key_totlength+=length; + share->base.rec_per_key[key]= + unique_count ? ((share->state.records+unique_count/2)/ + unique_count) : 1L; + } + if (testflag & T_INFO) + { + if (all_totaldata != 0L && share->state.keys != 1) + printf("Total: Keyblocks used: %3d%% Packed: %4d%%\n\n", + (int) (all_keydata*100.0/all_totaldata), + (int) ((long) (key_totlength-all_keydata)*100.0/ + (double) key_totlength)); + else if (all_totaldata != 0L && share->state.keys) + puts(""); + } + if (key_file_blocks != share->state.key_file_length) + print_warning("Some data are unreferenced in keyfile"); + record_checksum-=init_checksum; /* Remove delete links */ + if (testflag & T_STATISTICS) + DBUG_RETURN(update_state_info(info,UPDATE_STAT)); + DBUG_RETURN(0); +} /* chk_key */ + + + /* Check if index is ok */ + +static int chk_index(info,keyinfo,page,buff,keys,level) +N_INFO *info; +N_KEYDEF *keyinfo; +ulong page,*keys; +uchar *buff; +uint level; +{ + int flag; + uint used_length,comp_flag,nod_flag; + uchar key[N_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*endpos; + ulong next_page,record; + DBUG_ENTER("chk_index"); + DBUG_DUMP("buff",(byte*) buff,getint(buff)); + + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->base.block_length))) + { + print_error("Not Enough memory"); + DBUG_RETURN(-1); + } + + if (keyinfo->base.flag & HA_NOSAME) + comp_flag=SEARCH_FIND; /* Not dupplicates */ + else + comp_flag=SEARCH_SAME; /* Keys in positionorder */ + nod_flag=test_if_nod(buff); + used_length=getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + + keydata+=used_length; totaldata+=keyinfo->base.block_length; /* INFO */ + key_blocks++; + if (level > max_level) + max_level=level; + + if (used_length > keyinfo->base.block_length) + { + print_error("Wrong pageinfo at page: %lu",(ulong) page); + goto err; + } + for ( ;; ) + { + if (nod_flag) + { + next_page=_nisam_kpos(nod_flag,keypos); + if (next_page > info->s->state.key_file_length || + (nod_flag && (next_page & (info->s->blocksize -1)))) + { + my_off_t max_length=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)); + print_error("Wrong pagepointer: %lu at page: %lu", + (ulong) next_page,(ulong) page); + + if (next_page+info->s->blocksize > max_length) + goto err; + info->s->state.key_file_length=(ulong) (max_length & + ~ (my_off_t) + (info->s->blocksize-1)); + } + if (!_nisam_fetch_keypage(info,keyinfo,next_page,temp_buff,0)) + { + print_error("Can't read key from filepos: %lu",(ulong) next_page); + goto err; + } + key_file_blocks+=keyinfo->base.block_length; + if (chk_index(info,keyinfo,next_page,temp_buff,keys,level+1)) + goto err; + } + if (keypos >= endpos || + (*keyinfo->get_key)(keyinfo,nod_flag,&keypos,key) == 0) + break; + if ((*keys)++ && + (flag=_nisam_key_cmp(keyinfo->seg,info->lastkey,key,0,comp_flag)) >=0) + { + DBUG_DUMP("old",(byte*) info->lastkey, + _nisam_keylength(keyinfo,info->lastkey)); + DBUG_DUMP("new",(byte*) key,_nisam_keylength(keyinfo,key)); + + if (comp_flag == SEARCH_FIND && flag == 0) + print_error("Found dupplicated key at page %lu",(ulong) page); + else + print_error("Key in wrong position at page %lu",(ulong) page); + goto err; + } + if (testflag & T_STATISTICS) + { + if (*keys == 1L || + _nisam_key_cmp(keyinfo->seg,info->lastkey,key,0,SEARCH_FIND)) + unique_count++; + } + VOID(_nisam_move_key(keyinfo,(uchar*) info->lastkey,key)); + record= _nisam_dpos(info,nod_flag,keypos); + if (record >= info->s->state.data_file_length) + { + print_error("Found key at page %lu that points to record outside datafile",page); + DBUG_PRINT("test",("page: %lu record: %lu filelength: %lu", + (ulong) page,(ulong) record, + (ulong) info->s->state.data_file_length)); + DBUG_DUMP("key",(byte*) info->lastkey,info->s->base.max_key_length); + goto err; + } + record_checksum+=record; + } + if (keypos != endpos) + { + print_error("Keyblock size at page %lu is not correct. Block length: %d key length: %d",(ulong) page, used_length, (keypos - buff)); + goto err; + } + my_afree((byte*) temp_buff); + DBUG_RETURN(0); + err: + my_afree((byte*) temp_buff); + DBUG_RETURN(1); +} /* chk_index */ + + + /* Calculate a checksum of 1+2+3+4...N = N*(N+1)/2 without overflow */ + +static ulong calc_checksum(count) +ulong count; +{ + ulong sum,a,b; + DBUG_ENTER("calc_checksum"); + + sum=0; + a=count; b=count+1; + if (a & 1) + b>>=1; + else + a>>=1; + while (b) + { + if (b & 1) + sum+=a; + a<<=1; b>>=1; + } + DBUG_PRINT("exit",("sum: %lx",sum)); + DBUG_RETURN(sum); +} /* calc_checksum */ + + + /* Calc length of key in normal isam */ + +static uint isam_key_length(info,keyinfo) +N_INFO *info; +reg1 N_KEYDEF *keyinfo; +{ + uint length; + N_KEYSEG *keyseg; + DBUG_ENTER("isam_key_length"); + + length= info->s->rec_reflength; + for (keyseg=keyinfo->seg ; keyseg->base.type ; keyseg++) + length+= keyseg->base.length; + + DBUG_PRINT("exit",("length: %d",length)); + DBUG_RETURN(length); +} /* key_length */ + + + /* Check that record-link is ok */ + +static int chk_data_link(info,extend) +reg1 N_INFO *info; +int extend; +{ + int error,got_error,flag; + uint key,left_length,b_type; + ulong records,del_blocks,used,empty,pos,splitts,start_recpos, + del_length,link_used,intern_record_checksum,start_block; + byte *record,*to; + N_KEYDEF *keyinfo; + BLOCK_INFO block_info; + DBUG_ENTER("chk_data_link"); + + if (! (info->s->base.options & (HA_OPTION_PACK_RECORD | + HA_OPTION_COMPRESS_RECORD)) && + ! extend) + DBUG_RETURN(0); + + if (!(testflag & T_SILENT)) + { + if (extend) + puts("- check records and index references"); + else + puts("- check record links"); + } + + if (!(record= (byte*) my_alloca(info->s->base.reclength))) + { + print_error("Not Enough memory"); + DBUG_RETURN(-1); + } + records=used=link_used=splitts=del_blocks=del_length= + intern_record_checksum=crc=0L; + LINT_INIT(left_length); LINT_INIT(start_recpos); LINT_INIT(to); + got_error=error=0; + empty=pos=info->s->pack.header_length; + + while (pos < info->s->state.data_file_length) + { + switch (info->s->data_file_type) { + case STATIC_RECORD: + if (my_b_read(&read_cache,(byte*) record,info->s->base.reclength)) + goto err; + start_recpos=pos; + pos+=info->s->base.reclength; + splitts++; + if (*record == '\0') + { + del_blocks++; + del_length+=info->s->base.reclength; + continue; /* Record removed */ + } + used+=info->s->base.reclength; + break; + case DYNAMIC_RECORD: + flag=block_info.second_read=0; + block_info.next_filepos=pos; + do + { + if (_nisam_read_cache(&read_cache,(byte*) block_info.header, + (start_block=block_info.next_filepos), + sizeof(block_info.header),test(! flag) | 2)) + goto err; + b_type=_nisam_get_block_info(&block_info,-1,start_block); + if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR | + BLOCK_FATAL_ERROR)) + { + if (b_type & BLOCK_SYNC_ERROR) + { + if (flag) + { + print_error("Unexpected byte: %d at link: %lu", + (int) block_info.header[0],(ulong) start_block); + goto err2; + } + pos=block_info.filepos+block_info.block_len; + goto next; + } + if (b_type & BLOCK_DELETED) + { + if (block_info.block_len < info->s->base.min_block_length || + block_info.block_len-4 > (uint) info->s->base.max_pack_length) + { + print_error("Deleted block with impossible length %u at %lu", + block_info.block_len,(ulong) pos); + goto err2; + } + del_blocks++; + del_length+=block_info.block_len; + pos=block_info.filepos+block_info.block_len; + splitts++; + goto next; + } + print_error("Wrong bytesec: %d-%d-%d at linkstart: %lu", + block_info.header[0],block_info.header[1], + block_info.header[2],(ulong) start_block); + goto err2; + } + if (info->s->state.data_file_length < block_info.filepos+ + block_info.block_len) + { + print_error("Recordlink that points outside datafile at %lu", + (ulong) pos); + got_error=1; + break; + } + splitts++; + if (!flag++) /* First block */ + { + start_recpos=pos; + pos=block_info.filepos+block_info.block_len; + if (block_info.rec_len > (uint) info->s->base.max_pack_length) + { + print_error("Found too long record at %lu",(ulong) start_recpos); + got_error=1; + break; + } + if (info->s->base.blobs) + { + if (!(to=fix_rec_buff_for_blob(info,block_info.rec_len))) + { + print_error("Not enough memory for blob at %lu", + (ulong) start_recpos); + got_error=1; + break; + } + } + else + to= info->rec_buff; + left_length=block_info.rec_len; + } + if (left_length < block_info.data_len) + { + print_error("Found too long record at %lu",(ulong) start_recpos); + got_error=1; break; + } + if (_nisam_read_cache(&read_cache,(byte*) to,block_info.filepos, + (uint) block_info.data_len, test(flag == 1))) + goto err; + to+=block_info.data_len; + link_used+= block_info.filepos-start_block; + used+= block_info.filepos - start_block + block_info.data_len; + empty+=block_info.block_len-block_info.data_len; + left_length-=block_info.data_len; + if (left_length) + { + if (b_type & BLOCK_LAST) + { + print_error("Record link to short for record at %lu", + (ulong) start_recpos); + got_error=1; + break; + } + if (info->s->state.data_file_length < block_info.next_filepos) + { + print_error("Found next-recordlink that points outside datafile at %lu", + (ulong) block_info.filepos); + got_error=1; + break; + } + } + } while (left_length); + if (! got_error) + { + if (_nisam_rec_unpack(info,record,info->rec_buff,block_info.rec_len) == + MY_FILE_ERROR) + { + print_error("Found wrong record at %lu",(ulong) start_recpos); + got_error=1; + } + if (testflag & (T_EXTEND | T_VERBOSE)) + { + if (_nisam_rec_check(info,record)) + { + print_error("Found wrong packed record at %lu", + (ulong) start_recpos); + got_error=1; + } + } + } + else if (!flag) + pos=block_info.filepos+block_info.block_len; + break; + case COMPRESSED_RECORD: + if (_nisam_read_cache(&read_cache,(byte*) block_info.header,pos, 3,1)) + goto err; + start_recpos=pos; + splitts++; + VOID(_nisam_pack_get_block_info(&block_info,info->s->pack.ref_length,-1, + start_recpos)); + pos=start_recpos+info->s->pack.ref_length+block_info.rec_len; + if (block_info.rec_len < (uint) info->s->min_pack_length || + block_info.rec_len > (uint) info->s->max_pack_length) + { + print_error("Found block with wrong recordlength: %d at %lu", + block_info.rec_len,(ulong) start_recpos); + got_error=1; + break; + } + if (_nisam_read_cache(&read_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len,1)) + goto err; + if (_nisam_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len)) + { + print_error("Found wrong record at %lu",(ulong) start_recpos); + got_error=1; + } + crc^=checksum(record,info->s->base.reclength); + link_used+=info->s->pack.ref_length; + used+=block_info.rec_len+info->s->pack.ref_length; + } + if (! got_error) + { + intern_record_checksum+=start_recpos; + records++; + if (testflag & T_WRITE_LOOP && records % WRITE_COUNT == 0) + { + printf("%lu\r",(ulong) records); VOID(fflush(stdout)); + } + + if (extend) + { + for (key=0,keyinfo= info->s->keyinfo; key<info->s->state.keys; + key++,keyinfo++) + { + VOID(_nisam_make_key(info,key,info->lastkey,record,start_recpos)); + if (_nisam_search(info,keyinfo,info->lastkey,0,SEARCH_SAME, + info->s->state.key_root[key])) + { + print_error("Record at: %10lu Can't find key for index: %2d", + start_recpos,key+1); + if (error++ > MAXERR || !(testflag & T_VERBOSE)) + goto err2; + } + } + } + } + else + { + got_error=0; + if (error++ > MAXERR || !(testflag & T_VERBOSE)) + goto err2; + } + next:; /* Next record */ + } + if (testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + if (records != info->s->state.records) + { + print_error("Record-count is not ok; is %-10lu Should be: %lu", + (ulong) records,(ulong) info->s->state.records); + error=1; + } + else if (record_checksum != intern_record_checksum && info->s->state.keys) + { + print_error("Keypointers and records dosen't match"); + error=1; + } + if (used+empty+del_length != info->s->state.data_file_length) + { + print_warning("Found %lu record-data and %lu unused data and %lu deleted-data\nTotal %lu, Should be: %lu", + (ulong) used,(ulong) empty,(ulong) del_length, + (ulong) (used+empty+del_length), + (ulong) info->s->state.data_file_length); + } + if (del_blocks != info->s->state.del) + { + print_warning("Found %10lu deleted blocks Should be: %lu", + (ulong) del_blocks,(ulong) info->s->state.del); + } + if (splitts != info->s->state.splitt) + { + print_warning("Found %10lu parts Should be: %lu parts", + (ulong) splitts,(ulong) info->s->state.splitt); + } + if ((info->s->base.options & HA_OPTION_COMPRESS_RECORD) && + crc != info->s->state.uniq) + print_warning("Wrong checksum for records; Restore uncompressed table"); + + if (testflag & T_INFO) + { + if (warning_printed || error_printed) + puts(""); + if (used != 0 && ! error_printed) + { + printf("Records:%17lu M.recordlength:%8lu Packed:%14.0f%%\n", + records, (used-link_used)/records, + (info->s->base.blobs ? 0 : + (ulong_to_double(info->s->base.reclength*records)-used)/ + ulong_to_double(info->s->base.reclength*records)*100.0)); + printf("Recordspace used:%8.0f%% Empty space:%11d%% Blocks/Record: %6.2f\n", + (ulong_to_double(used-link_used)/ulong_to_double(used-link_used+empty)*100.0), + (!records ? 100 : (int) (ulong_to_double(del_length+empty)/used*100.0)), + ulong_to_double(splitts - del_blocks) / records); + } + printf("Record blocks:%12lu Delete blocks:%10lu\n", + splitts-del_blocks,del_blocks); + printf("Record data: %12lu Deleted data :%10lu\n", + used-link_used,del_length); + printf("Lost space: %12lu Linkdata: %10lu\n", + empty,link_used); + } + my_afree((gptr) record); + DBUG_RETURN (error); + err: + print_error("got error: %d when reading datafile",my_errno); + err2: + my_afree((gptr) record); + DBUG_RETURN(1); +} /* chk_data_link */ + + + /* Recover old table by reading each record and writing all keys */ + /* Save new datafile-name in temp_filename */ + +static int rep(info,name) +reg1 N_INFO *info; +my_string name; +{ + int error,got_error; + uint i; + ulong start_records,new_header_length,del; + File new_file; + ISAM_SHARE *share=info->s; + DBUG_ENTER("rep"); + + start_records=share->state.records; + new_header_length=(testflag & T_UNPACK) ? 0L : share->pack.header_length; + got_error=1; + new_file= -1; + if (!(testflag & T_SILENT)) + { + printf("- recovering ISAM-table '%s'\n",name); + printf("Data records: %lu\n",(ulong) share->state.records); + } + + VOID(init_key_cache(use_buffers,NEAD_MEM)); + if (init_io_cache(&read_cache,info->dfile,(uint) read_buffer_length, + READ_CACHE,share->pack.header_length,1,MYF(MY_WME))) + goto err; + if (!rep_quick) + if (init_io_cache(&info->rec_cache,-1,(uint) write_buffer_length, + WRITE_CACHE, new_header_length, 1, + MYF(MY_WME | MY_WAIT_IF_FULL))) + goto err; + info->opt_flag|=WRITE_CACHE_USED; + sort_info.start_recpos=0; + sort_info.buff=0; sort_info.buff_length=0; + if (!(sort_info.record=(byte*) my_alloca((uint) share->base.reclength))) + { + print_error("Not Enough memory"); + goto err; + } + + if (!rep_quick) + { + if ((new_file=my_create(fn_format(temp_filename,name,"",DATA_TMP_EXT, + 2+4), + 0,tmpfile_createflag,MYF(0))) < 0) + { + print_error("Can't create new tempfile: '%s'",temp_filename); + goto err; + } + if (filecopy(new_file,info->dfile,0L,new_header_length,"datafile-header")) + goto err; + share->state.dellink= NI_POS_ERROR; + info->rec_cache.file=new_file; + if (testflag & T_UNPACK) + share->base.options&= ~HA_OPTION_COMPRESS_RECORD; + } + + sort_info.info=info; + sort_info.pos=sort_info.max_pos=share->pack.header_length; + sort_info.filepos=new_header_length; + read_cache.end_of_file=sort_info.filelength=(ulong) + my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)); + sort_info.dupp=0; + sort_info.fix_datafile= (my_bool) (! rep_quick); + sort_info.max_records=LONG_MAX; + if ((sort_info.new_data_file_type=share->data_file_type) == + COMPRESSED_RECORD && testflag & T_UNPACK) + { + if (share->base.options & HA_OPTION_PACK_RECORD) + sort_info.new_data_file_type = DYNAMIC_RECORD; + else + sort_info.new_data_file_type = STATIC_RECORD; + } + + del=share->state.del; + share->state.records=share->state.del=share->state.empty= + share->state.splitt=0; + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + for (i=0 ; i < N_MAXKEY ; i++) + share->state.key_del[i]=share->state.key_root[i]= NI_POS_ERROR; + share->state.key_file_length=share->base.keystart; + + lock_memory(); /* Everything is alloced */ + while (!(error=sort_get_next_record())) + { + if (writekeys(info,(byte*) sort_info.record,sort_info.filepos)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY) goto err; + DBUG_DUMP("record",(byte*) sort_info.record,share->base.pack_reclength); + print_info("Dupplicate key %2d for record at %10lu against new record at %10lu",info->errkey+1,sort_info.start_recpos,info->int_pos); + if (testflag & T_VERBOSE) + { + VOID(_nisam_make_key(info,(uint) info->errkey,info->lastkey, + sort_info.record,0L)); + _nisam_print_key(stdout,share->keyinfo[info->errkey].seg,info->lastkey); + } + sort_info.dupp++; + if (rep_quick == 1) + { + error_printed=1; + goto err; + } + continue; + } + if (sort_write_record()) + goto err; + } + if (error > 0 || write_data_suffix(info) || + flush_io_cache(&info->rec_cache) || read_cache.error < 0) + goto err; + + if (testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + if (my_chsize(share->kfile,share->state.key_file_length,MYF(0))) + { + print_warning("Can't change size of indexfile, error: %d",my_errno); + goto err; + } + + if (rep_quick && del+sort_info.dupp != share->state.del) + { + print_error("Couldn't fix table with quick recovery: Found wrong number of deleted records"); + print_error("Run recovery again without -q"); + got_error=1; + goto err; + } + + if (!rep_quick) + { + info->dfile=new_file; + share->state.data_file_length=sort_info.filepos; + share->state.splitt=share->state.records; /* Only hole records */ + out_flag|=O_NEW_DATA; /* Data in new file */ + share->state.version=(ulong) time((time_t*) 0); /* Force reopen */ + } + else + share->state.data_file_length=sort_info.max_pos; + + if (!(testflag & T_SILENT)) + { + if (start_records != share->state.records) + printf("Data records: %lu\n",(ulong) share->state.records); + if (sort_info.dupp) + print_warning("%lu records have been removed",(ulong) sort_info.dupp); + } + + got_error=0; +err: + if (got_error) + { + if (! error_printed) + print_error("%d for record at pos %lu",my_errno, + (ulong) sort_info.start_recpos); + if (new_file >= 0) + { + VOID(my_close(new_file,MYF(0))); + VOID(my_delete(temp_filename,MYF(MY_WME))); + } + } + if (sort_info.record) + { + my_afree(sort_info.record); + } + my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR)); + VOID(end_io_cache(&read_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + VOID(end_io_cache(&info->rec_cache)); + got_error|=flush_blocks(share->kfile); + if (!got_error && testflag & T_UNPACK) + { + share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; + share->pack.header_length=0; + share->data_file_type=sort_info.new_data_file_type; + } + DBUG_RETURN(got_error); +} /* rep */ + + + /* Uppdaterar nyckelfilen i samband med reparation */ + +static int writekeys(register N_INFO *info,byte *buff,ulong filepos) +{ + register uint i; + uchar *key; + DBUG_ENTER("writekeys"); + + key=info->lastkey+info->s->base.max_key_length; + for (i=0 ; i < info->s->state.keys ; i++) + { + VOID(_nisam_make_key(info,i,key,buff,filepos)); + if (_nisam_ck_write(info,i,key)) goto err; + } + DBUG_RETURN(0); + + err: + if (my_errno == HA_ERR_FOUND_DUPP_KEY) + { + info->errkey=(int) i; /* This key was found */ + while ( i-- > 0 ) + { + VOID(_nisam_make_key(info,i,key,buff,filepos)); + if (_nisam_ck_delete(info,i,key)) break; + } + } + DBUG_PRINT("error",("errno: %d",my_errno)); + DBUG_RETURN(-1); +} /* writekeys */ + + + /* Write info about table */ + +static void descript(info,name) +reg1 N_INFO *info; +my_string name; +{ + uint key,field,start,len; + reg3 N_KEYDEF *keyinfo; + reg2 N_KEYSEG *keyseg; + reg4 const char *text; + char buff[40],length[10],*pos,*end; + enum en_fieldtype type; + ISAM_SHARE *share=info->s; + DBUG_ENTER("describe"); + + printf("\nISAM file: %s\n",name); + if (testflag & T_VERBOSE) + { + printf("Isam-version: %d\n",(int) share->state.header.file_version[3]); + if (share->base.create_time) + { + get_date(buff,1,share->base.create_time); + printf("Creation time: %s\n",buff); + } + if (share->base.isamchk_time) + { + get_date(buff,1,share->base.isamchk_time); + printf("Recover time: %s\n",buff); + } + } + printf("Data records: %10lu Deleted blocks: %10lu\n", + share->state.records,share->state.del); + if (testflag & T_SILENT) + DBUG_VOID_RETURN; /* This is enough */ + + if (testflag & T_VERBOSE) + { +#ifdef USE_RELOC + printf("Init-relocation: %10lu\n",share->base.reloc); +#endif + printf("Datafile Parts: %10lu Deleted data: %10lu\n", + share->state.splitt,share->state.empty); + printf("Datafile pointer (bytes):%6d Keyfile pointer (bytes):%6d\n", + share->rec_reflength,share->base.key_reflength); + if (info->s->base.reloc == 1L && info->s->base.records == 1L) + puts("This is a one-record table"); + else + { + if (share->base.max_data_file_length != NI_POS_ERROR || + share->base.max_key_file_length != NI_POS_ERROR) + printf("Max datafile length: %10lu Max keyfile length: %10lu\n", + share->base.max_data_file_length-1, + share->base.max_key_file_length-1); + } + } + + printf("Recordlength: %10d\n",(int) share->base.reclength); + VOID(fputs("Record format: ",stdout)); + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + puts("Compressed"); + else if (share->base.options & HA_OPTION_PACK_RECORD) + puts("Packed"); + else + puts("Fixed length"); + if (share->state.keys != share->base.keys) + printf("Using only %d keys of %d possibly keys\n",share->state.keys, + share->base.keys); + + puts("\ntable description:"); + printf("Key Start Len Index Type"); + if (testflag & T_VERBOSE) + printf(" Root Blocksize Rec/key"); + VOID(putchar('\n')); + + for (key=0, keyinfo= &share->keyinfo[0] ; key < share->base.keys; + key++,keyinfo++) + { + keyseg=keyinfo->seg; + if (keyinfo->base.flag & HA_NOSAME) text="unique "; + else text="multip."; + + pos=buff; + if (keyseg->base.flag & HA_REVERSE_SORT) + *pos++ = '-'; + pos=strmov(pos,type_names[keyseg->base.type]); + *pos++ = ' '; + *pos=0; + if (keyinfo->base.flag & HA_PACK_KEY) + pos=strmov(pos,packed_txt); + if (keyseg->base.flag & HA_SPACE_PACK) + pos=strmov(pos,diff_txt); + printf("%-4d%-6d%-3d %-8s%-21s", + key+1,keyseg->base.start+1,keyseg->base.length,text,buff); + if (testflag & T_VERBOSE) + printf(" %9ld %9d %9ld", + share->state.key_root[key],keyinfo->base.block_length, + share->base.rec_per_key[key]); + VOID(putchar('\n')); + while ((++keyseg)->base.type) + { + pos=buff; + if (keyseg->base.flag & HA_REVERSE_SORT) + *pos++ = '-'; + pos=strmov(pos,type_names[keyseg->base.type]); + *pos++= ' '; + if (keyseg->base.flag & HA_SPACE_PACK) + pos=strmov(pos,diff_txt); + *pos=0; + printf(" %-6d%-3d %-24s\n", + keyseg->base.start+1,keyseg->base.length,buff); + } + } + if (verbose > 1) + { + printf("\nField Start Length Type"); + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + printf(" Huff tree Bits"); + VOID(putchar('\n')); + start=1; + for (field=0 ; field < share->base.fields ; field++) + { + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + type=share->rec[field].base_type; + else + type=(enum en_fieldtype) share->rec[field].base.type; + end=strmov(buff,field_pack[type]); +#ifndef NOT_PACKED_DATABASES + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + { + if (share->rec[field].pack_type & PACK_TYPE_SELECTED) + end=strmov(end,", not_always"); + if (share->rec[field].pack_type & PACK_TYPE_SPACE_FIELDS) + end=strmov(end,", no empty"); + if (share->rec[field].pack_type & PACK_TYPE_ZERO_FILL) + { + sprintf(end,", zerofill(%d)",share->rec[field].space_length_bits); + end=strend(end); + } + } + if (buff[0] == ',') + strmov(buff,buff+2); +#endif + len=(uint) (int2str((long) share->rec[field].base.length,length,10) - + length); + if (type == FIELD_BLOB) + { + length[len]='+'; + VOID(int2str((long) sizeof(char*),length+len+1,10)); + } + printf("%-6d%-6d%-7s%-35s",field+1,start,length,buff); +#ifndef NOT_PACKED_DATABASES + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + { + if (share->rec[field].huff_tree) + printf("%3d %2d", + (uint) (share->rec[field].huff_tree-share->decode_trees)+1, + share->rec[field].huff_tree->quick_table_bits); + } +#endif + VOID(putchar('\n')); + start+=share->rec[field].base.length; + if (type == FIELD_BLOB) + start+=sizeof(char*); + } + } + DBUG_VOID_RETURN; +} /* describe */ + + + /* Change all key-pointers that points to a records */ + +static int movepoint(info,record,oldpos,newpos,prot_key) +register N_INFO *info; +byte *record; +ulong oldpos,newpos; +uint prot_key; +{ + register uint i; + uchar *key; + DBUG_ENTER("movepoint"); + + key=info->lastkey+info->s->base.max_key_length; + for (i=0 ; i < info->s->state.keys; i++) + { + if (i != prot_key) + { + VOID(_nisam_make_key(info,i,key,record,oldpos)); + if (info->s->keyinfo[i].base.flag & HA_NOSAME) + { /* Change pointer direct */ + uint nod_flag; + N_KEYDEF *keyinfo; + keyinfo=info->s->keyinfo+i; + if (_nisam_search(info,keyinfo,key,USE_HOLE_KEY, + (uint) (SEARCH_SAME | SEARCH_SAVE_BUFF), + info->s->state.key_root[i])) + DBUG_RETURN(-1); + nod_flag=test_if_nod(info->buff); + _nisam_dpointer(info,info->int_keypos-nod_flag- + info->s->rec_reflength,newpos); + if (_nisam_write_keypage(info,keyinfo,info->int_pos,info->buff)) + DBUG_RETURN(-1); + } + else + { /* Change old key to new */ + if (_nisam_ck_delete(info,i,key)) + DBUG_RETURN(-1); + VOID(_nisam_make_key(info,i,key,record,newpos)); + if (_nisam_ck_write(info,i,key)) + DBUG_RETURN(-1); + } + } + } + DBUG_RETURN(0); +} /* movepoint */ + + + /* Tell system that we want all memory for our cache */ + +static void lock_memory(void) +{ +#ifdef SUN_OS /* Key-cacheing thrases on sun 4.1 */ + int success; + + success = mlockall(MCL_CURRENT); /* or plock(DATLOCK); */ + + if (geteuid() == 0 && success != 0) + print_warning("Failed to lock memory. errno %d",my_errno); +#endif +} /* lock_memory */ + + + /* Flush all changed blocks to disk */ + +static int flush_blocks(file) +File file; +{ + if (flush_key_blocks(file,FLUSH_RELEASE)) + { + print_error("%d when trying to write bufferts",my_errno); + return(1); + } + end_key_cache(); + return 0; +} /* flush_blocks */ + + + /* Sort records according to one key */ + +static int sort_records(info,name,sort_key,write_info) +register N_INFO *info; +my_string name; +uint sort_key; +int write_info; +{ + int got_error; + uint key; + N_KEYDEF *keyinfo; + File new_file; + uchar *temp_buff; + ulong old_record_count; + ISAM_SHARE *share=info->s; + DBUG_ENTER("sort_records"); + + keyinfo= &share->keyinfo[sort_key]; + got_error=1; + temp_buff=0; record_buff=0; + new_file= -1; + + if (sort_key >= share->state.keys) + { + print_error("Can't sort table '%s' on key %d. It has only %d keys", + name,sort_key+1,share->state.keys); + error_printed=0; + DBUG_RETURN(-1); + } + if (!(testflag & T_SILENT)) + { + printf("- Sorting records in ISAM-table '%s'\n",name); + if (write_info) + printf("Data records: %7lu Deleted: %7lu\n", + share->state.records,share->state.del); + } + if (share->state.key_root[sort_key] == NI_POS_ERROR) + DBUG_RETURN(0); /* Nothing to do */ + + init_key_cache(use_buffers,NEAD_MEM); + if (init_io_cache(&info->rec_cache,-1,(uint) write_buffer_length, + WRITE_CACHE,share->pack.header_length,1, + MYF(MY_WME | MY_WAIT_IF_FULL))) + goto err; + info->opt_flag|=WRITE_CACHE_USED; + + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->base.block_length))) + { + print_error("Not Enough memory"); + goto err; + } + if (!(record_buff=(byte*) my_alloca((uint) share->base.reclength))) + { + print_error("Not Enough memory"); + goto err; + } + if ((new_file=my_create(fn_format(temp_filename,name,"",DATA_TMP_EXT,2+4), + 0,tmpfile_createflag,MYF(0))) <= 0) + { + print_error("Can't create new tempfile: '%s'",temp_filename); + goto err; + } + if (filecopy(new_file,info->dfile,0L,share->pack.header_length, + "datafile-header")) + goto err; + info->rec_cache.file=new_file; /* Use this file for cacheing*/ + + lock_memory(); + for (key=0 ; key < share->state.keys ; key++) + share->keyinfo[key].base.flag|= HA_SORT_ALLOWS_SAME; + + if (my_pread(share->kfile,(byte*) temp_buff, + (uint) keyinfo->base.block_length, + share->state.key_root[sort_key], + MYF(MY_NABP+MY_WME))) + { + print_error("Can't read indexpage from filepos: %lu", + (ulong) share->state.key_root[sort_key]); + goto err; + } + + /* Setup param for sort_write_record */ + bzero((char*) &sort_info,sizeof(sort_info)); + sort_info.info=info; + sort_info.new_data_file_type=share->data_file_type; + sort_info.fix_datafile=1; + sort_info.filepos=share->pack.header_length; + sort_info.record=record_buff; + old_record_count=share->state.records; + share->state.records=0; + + if (sort_record_index(info,keyinfo,share->state.key_root[sort_key],temp_buff, + sort_key,new_file) || + write_data_suffix(info) || + flush_io_cache(&info->rec_cache)) + goto err; + + if (share->state.records != old_record_count) + { + print_error("found %lu of %lu records", + (ulong) share->state.records,(ulong) old_record_count); + goto err; + } + + /* Put same locks as old file */ + if (lock_file(new_file,0L,F_WRLCK,"tempfile",temp_filename)) + goto err; + VOID(lock_file(info->dfile,0L,F_UNLCK,"datafile of",name)); + VOID(my_close(info->dfile,MYF(MY_WME))); + out_flag|=O_NEW_DATA; /* Data in new file */ + + info->dfile=new_file; /* Use new indexfile */ + share->state.del=share->state.empty=0; + share->state.dellink= NI_POS_ERROR; + share->state.data_file_length=sort_info.filepos; + share->state.splitt=share->state.records; /* Only hole records */ + share->state.version=(ulong) time((time_t*) 0); + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + if (testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + got_error=0; + +err: + if (got_error && new_file >= 0) + { + VOID(my_close(new_file,MYF(MY_WME))); + VOID(my_delete(temp_filename,MYF(MY_WME))); + } + if (temp_buff) + { + my_afree((gptr) temp_buff); + } + if (record_buff) + { + my_afree(record_buff); + } + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + VOID(end_io_cache(&info->rec_cache)); + share->base.sortkey=sort_key; + DBUG_RETURN(flush_blocks(share->kfile) | got_error); +} /* sort_records */ + + + /* Sort records recursive using one index */ + +static int sort_record_index(info,keyinfo,page,buff,sort_key,new_file) +N_INFO *info; +N_KEYDEF *keyinfo; +ulong page; +uchar *buff; +uint sort_key; +File new_file; +{ + uint nod_flag,used_length; + uchar *temp_buff,*keypos,*endpos; + ulong next_page,rec_pos; + uchar lastkey[N_MAX_KEY_BUFF]; + DBUG_ENTER("sort_record_index"); + + nod_flag=test_if_nod(buff); + temp_buff=0; + + if (nod_flag) + { + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->base.block_length))) + { + print_error("Not Enough memory"); + DBUG_RETURN(-1); + } + } + used_length=getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + for ( ;; ) + { + _sanity(__FILE__,__LINE__); + if (nod_flag) + { + next_page=_nisam_kpos(nod_flag,keypos); + if (my_pread(info->s->kfile,(byte*) temp_buff, + (uint) keyinfo->base.block_length, next_page, + MYF(MY_NABP+MY_WME))) + { + print_error("Can't read keys from filepos: %lu",(ulong) next_page); + goto err; + } + if (sort_record_index(info,keyinfo,next_page,temp_buff,sort_key, + new_file)) + goto err; + } + _sanity(__FILE__,__LINE__); + if (keypos >= endpos || + (*keyinfo->get_key)(keyinfo,nod_flag,&keypos,lastkey) == 0) + break; + rec_pos= _nisam_dpos(info,nod_flag,keypos); + + if ((*info->s->read_rnd)(info,record_buff,rec_pos,0)) + { + print_error("%d when reading datafile",my_errno); + goto err; + } + if (rec_pos != sort_info.filepos) + { + _nisam_dpointer(info,keypos-nod_flag-info->s->rec_reflength, + sort_info.filepos); + if (movepoint(info,record_buff,rec_pos,sort_info.filepos,sort_key)) + { + print_error("%d when updating key-pointers",my_errno); + goto err; + } + } + if (sort_write_record()) + goto err; + } + bzero((byte*) buff+used_length,keyinfo->base.block_length-used_length); + if (my_pwrite(info->s->kfile,(byte*) buff,(uint) keyinfo->base.block_length, + page,MYF_RW)) + { + print_error("%d when updating keyblock",my_errno); + goto err; + } + if (temp_buff) + my_afree((gptr) temp_buff); + DBUG_RETURN(0); +err: + if (temp_buff) + my_afree((gptr) temp_buff); + DBUG_RETURN(1); +} /* sort_record_index */ + + + /* Sort index for more efficent reads */ + +static int sort_index(info,name) +register N_INFO *info; +my_string name; +{ + reg2 uint key; + reg1 N_KEYDEF *keyinfo; + File new_file; + ulong index_pos[N_MAXKEY]; + DBUG_ENTER("sort_index"); + + if (!(testflag & T_SILENT)) + printf("- Sorting index for ISAM-table '%s'\n",name); + + if ((new_file=my_create(fn_format(temp_filename,name,"",INDEX_TMP_EXT,2+4), + 0,tmpfile_createflag,MYF(0))) <= 0) + { + print_error("Can't create new tempfile: '%s'",temp_filename); + DBUG_RETURN(-1); + } + if (filecopy(new_file,info->s->kfile,0L,(ulong) info->s->base.keystart, + "headerblock")) + goto err; + + new_file_pos=info->s->base.keystart; + for (key= 0,keyinfo= &info->s->keyinfo[0]; key < info->s->state.keys ; + key++,keyinfo++) + { + if (info->s->state.key_root[key] != NI_POS_ERROR) + { + index_pos[key]=new_file_pos; /* Write first block here */ + if (!_nisam_fetch_keypage(info,keyinfo,info->s->state.key_root[key], + info->buff,0)) + { + print_error("Can't read indexpage from filepos: %lu", + (ulong) info->s->state.key_root[key]); + goto err; + } + if (sort_one_index(info,keyinfo,info->buff,new_file)) + goto err; + } + else + index_pos[key]= NI_POS_ERROR; /* No blocks */ + } + + /* Put same locks as old file */ + if (lock_file(new_file,0L,F_WRLCK,"tempfile",temp_filename)) + goto err; + info->s->state.version=(ulong) time((time_t*) 0); + VOID(_nisam_writeinfo(info,1)); /* This unlocks table */ + VOID(my_close(info->s->kfile,MYF(MY_WME))); + out_flag|=O_NEW_INDEX; /* Data in new file */ + + info->s->kfile=new_file; + info->s->state.key_file_length=new_file_pos; + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + for (key=0 ; key < info->s->state.keys ; key++) + { + info->s->state.key_root[key]=index_pos[key]; + info->s->state.key_del[key]= NI_POS_ERROR; + } + DBUG_RETURN(0); + +err: + VOID(my_close(new_file,MYF(MY_WME))); + VOID(my_delete(temp_filename,MYF(MY_WME))); + DBUG_RETURN(-1); +} /* sort_index */ + + + /* Sort records recursive using one index */ + +static int sort_one_index(info,keyinfo,buff,new_file) +N_INFO *info; +N_KEYDEF *keyinfo; +uchar *buff; +File new_file; +{ + uint length,nod_flag,used_length; + uchar *temp_buff,*keypos,*endpos; + ulong new_page_pos,next_page; + DBUG_ENTER("sort_one_index"); + + temp_buff=0; + new_page_pos=new_file_pos; + new_file_pos+=keyinfo->base.block_length; + + if ((nod_flag=test_if_nod(buff))) + { + if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->base.block_length))) + { + print_error("Not Enough memory"); + DBUG_RETURN(-1); + } + + used_length=getint(buff); + keypos=buff+2+nod_flag; + endpos=buff+used_length; + for ( ;; ) + { + if (nod_flag) + { + next_page=_nisam_kpos(nod_flag,keypos); + _nisam_kpointer(info,keypos-nod_flag,new_file_pos); /* Save new pos */ + if (!_nisam_fetch_keypage(info,keyinfo,next_page,temp_buff,0)) + { + print_error("Can't read keys from filepos: %lu", + (ulong) next_page); + goto err; + } + if (sort_one_index(info,keyinfo,temp_buff,new_file)) + goto err; + } + if (keypos >= endpos || + ((*keyinfo->get_key)(keyinfo,nod_flag,&keypos,info->lastkey)) == 0) + break; + } + my_afree((gptr) temp_buff); + } + + /* Fill block with zero and write it to new file */ + + length=getint(buff); + bzero((byte*) buff+length,keyinfo->base.block_length-length); + if (my_pwrite(new_file,(byte*) buff,(uint) keyinfo->base.block_length, + new_page_pos,MYF(MY_NABP | MY_WAIT_IF_FULL))) + { + print_error("Can't write indexblock, error: %d",my_errno); + goto err; + } + DBUG_RETURN(0); +err: + if (temp_buff) + my_afree((gptr) temp_buff); + DBUG_RETURN(1); +} /* sort_one_index */ + + + /* Change to use new file */ + /* Copy stats from old file to new file, deletes orginal and */ + /* changes new file name to old file name */ + +static int change_to_newfile(filename,old_ext,new_ext) +const char *filename,*old_ext,*new_ext; +{ + char old_filename[FN_REFLEN],new_filename[FN_REFLEN]; + + return my_redel(fn_format(old_filename,filename,"",old_ext,2+4), + fn_format(new_filename,filename,"",new_ext,2+4), + MYF(MY_WME+MY_LINK_WARNING)); +} /* change_to_newfile */ + + + /* Locks a hole file */ + /* Gives error-message if file can't be locked */ + +static int lock_file(file,start,lock_type,filetype,filename) +File file; +ulong start; +int lock_type; +const char *filetype,*filename; +{ +#ifndef NO_LOCKING + if (my_lock(file,lock_type,start,F_TO_EOF, + testflag & T_WAIT_FOREVER ? MYF(MY_SEEK_NOT_DONE) : + MYF(MY_SEEK_NOT_DONE | MY_DONT_WAIT))) + { + print_error(" %d when %s %s '%s'",my_errno, + lock_type == F_UNLCK ? "unlocking": "locking", + filetype,filename); + error_printed=2; /* Don't give that data is crashed */ + return 1; + } +#endif + return 0; +} /* lock_file */ + + + /* Copy a block between two files */ + +static int filecopy(File to,File from,ulong start,ulong length, + const char *type) +{ + char tmp_buff[IO_SIZE],*buff; + ulong buff_length; + DBUG_ENTER("filecopy"); + + buff_length=min(write_buffer_length,length); + if (!(buff=my_malloc(buff_length,MYF(0)))) + { + buff=tmp_buff; buff_length=IO_SIZE; + } + + VOID(my_seek(from,start,MY_SEEK_SET,MYF(0))); + while (length > buff_length) + { + if (my_read(from,(byte*) buff,buff_length,MYF(MY_NABP)) || + my_write(to,(byte*) buff,buff_length,MYF(MY_NABP | MY_WAIT_IF_FULL))) + goto err; + length-= buff_length; + } + if (my_read(from,(byte*) buff,(uint) length,MYF(MY_NABP)) || + my_write(to,(byte*) buff,(uint) length,MYF(MY_NABP | MY_WAIT_IF_FULL))) + goto err; + if (buff != tmp_buff) + my_free(buff,MYF(0)); + DBUG_RETURN(0); +err: + if (buff != tmp_buff) + my_free(buff,MYF(0)); + print_error("Can't copy %s to tempfile, error %d",type,my_errno); + DBUG_RETURN(1); +} + + + /* Fix table using sorting */ + /* saves new table in temp_filename */ + +static int rep_by_sort(info,name) +reg1 N_INFO *info; +my_string name; +{ + int got_error; + uint i,length; + ulong start_records,new_header_length,del; + File new_file; + SORT_PARAM sort_param; + ISAM_SHARE *share=info->s; + DBUG_ENTER("rep_by_sort"); + + start_records=share->state.records; + got_error=1; + new_file= -1; + new_header_length=(testflag & T_UNPACK) ? 0 : share->pack.header_length; + if (!(testflag & T_SILENT)) + { + printf("- recovering ISAM-table '%s'\n",name); + printf("Data records: %lu\n",(ulong) start_records); + } + bzero((char*) &sort_info,sizeof(sort_info)); + if (!(sort_info.key_block=alloc_key_blocks((uint) sort_key_blocks, + share->base.max_block)) + || init_io_cache(&read_cache,info->dfile,(uint) read_buffer_length, + READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) || + (! rep_quick && + init_io_cache(&info->rec_cache,info->dfile,(uint) write_buffer_length, + WRITE_CACHE,new_header_length,1, + MYF(MY_WME | MY_WAIT_IF_FULL)))) + goto err; + sort_info.key_block_end=sort_info.key_block+sort_key_blocks; + info->opt_flag|=WRITE_CACHE_USED; + info->rec_cache.file=info->dfile; /* for sort_delete_record */ + + if (!(sort_info.record=(byte*) my_alloca((uint) share->base.reclength))) + { + print_error("Not enough memory for extra record"); + goto err; + } + if (!rep_quick) + { + if ((new_file=my_create(fn_format(temp_filename,name,"",DATA_TMP_EXT, + 2+4), + 0,tmpfile_createflag,MYF(0))) < 0) + { + print_error("Can't create new tempfile: '%s'",temp_filename); + goto err; + } + if (filecopy(new_file,info->dfile,0L,new_header_length,"datafile-header")) + goto err; + if (testflag & T_UNPACK) + share->base.options&= ~HA_OPTION_COMPRESS_RECORD; + share->state.dellink= NI_POS_ERROR; + info->rec_cache.file=new_file; + } + + info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + for (i=0 ; i < N_MAXKEY ; i++) + share->state.key_del[i]=share->state.key_root[i]= NI_POS_ERROR; + share->state.key_file_length=share->base.keystart; + + sort_info.info=info; + if ((sort_info.new_data_file_type=share->data_file_type) == + COMPRESSED_RECORD && testflag & T_UNPACK) + { + if (share->base.options & HA_OPTION_PACK_RECORD) + sort_info.new_data_file_type = DYNAMIC_RECORD; + else + sort_info.new_data_file_type = STATIC_RECORD; + } + + sort_info.filepos=new_header_length; + sort_info.dupp=0; + read_cache.end_of_file=sort_info.filelength= + (ulong) my_seek(read_cache.file,0L,MY_SEEK_END,MYF(0)); + + if (share->data_file_type == DYNAMIC_RECORD) + length=max(share->base.min_pack_length+1,share->base.min_block_length); + else if (share->data_file_type == COMPRESSED_RECORD) + length=share->base.min_block_length; + else + length=share->base.reclength; + sort_param.max_records=sort_info.max_records=sort_info.filelength/length+1; + sort_param.key_cmp=sort_key_cmp; + sort_param.key_write=sort_key_write; + sort_param.key_read=sort_key_read; + sort_param.lock_in_memory=lock_memory; + del=share->state.del; + + for (sort_info.key=0 ; sort_info.key < share->state.keys ; sort_info.key++) + { + if ((!(testflag & T_SILENT))) + printf ("- Fixing index %d\n",sort_info.key+1); + sort_info.max_pos=sort_info.pos=share->pack.header_length; + sort_info.keyinfo=share->keyinfo+sort_info.key; + sort_info.keyseg=sort_info.keyinfo->seg; + sort_info.fix_datafile= (my_bool) (sort_info.key == 0 && ! rep_quick); + sort_info.unique=0; + sort_param.key_length=share->rec_reflength; + for (i=0 ; sort_info.keyseg[i].base.type ; i++) + sort_param.key_length+=sort_info.keyseg[i].base.length+ + (sort_info.keyseg[i].base.flag & HA_SPACE_PACK ? 1 : 0); + share->state.records=share->state.del=share->state.empty=share->state.splitt=0; + + if (_create_index_by_sort(&sort_param, + (pbool) (!(testflag & T_VERBOSE)), + (uint) sort_buffer_length)) + goto err; + /* Set for next loop */ + sort_param.max_records=sort_info.max_records=share->state.records; + share->base.rec_per_key[sort_info.key]= + sort_info.unique ? ((sort_info.max_records+sort_info.unique/2)/ + sort_info.unique) + : 1L; + + if (sort_info.fix_datafile) + { + info->dfile=new_file; + share->state.data_file_length=sort_info.filepos; + share->state.splitt=share->state.records; /* Only hole records */ + share->state.version=(ulong) time((time_t*) 0); + out_flag|=O_NEW_DATA; /* Data in new file */ + read_cache.end_of_file=sort_info.filepos; + if (write_data_suffix(info) || end_io_cache(&info->rec_cache)) + goto err; + share->data_file_type=sort_info.new_data_file_type; + share->pack.header_length=new_header_length; + } + else + share->state.data_file_length=sort_info.max_pos; + + if (flush_pending_blocks()) + goto err; + + read_cache.file=info->dfile; /* re-init read cache */ + reinit_io_cache(&read_cache,READ_CACHE,share->pack.header_length,1,1); + } + + if (testflag & T_WRITE_LOOP) + { + VOID(fputs(" \r",stdout)); VOID(fflush(stdout)); + } + + if (rep_quick && del+sort_info.dupp != share->state.del) + { + print_error("Couldn't fix table with quick recovery: Found wrong number of deleted records"); + print_error("Run recovery again without -q"); + got_error=1; + goto err; + } + + if (rep_quick != 1) + { + ulong skr=share->state.data_file_length+ + (share->base.options & HA_OPTION_COMPRESS_RECORD ? + MEMMAP_EXTRA_MARGIN : 0); +#ifdef USE_RELOC + if (share->data_file_type == STATIC_RECORD && + skr < share->base.reloc*share->base.min_pack_length) + skr=share->base.reloc*share->base.min_pack_length; +#endif + if (skr != sort_info.filelength) + if (my_chsize(info->dfile,skr,MYF(0))) + print_warning("Can't change size of datafile, error: %d",my_errno); + } + if (my_chsize(share->kfile,share->state.key_file_length,MYF(0))) + print_warning("Can't change size of indexfile, error: %d",my_errno); + + if (!(testflag & T_SILENT)) + { + if (start_records != share->state.records) + printf("Data records: %lu\n",(ulong) share->state.records); + if (sort_info.dupp) + print_warning("%lu records have been removed",(ulong) sort_info.dupp); + } + got_error=0; + +err: + if (got_error) + { + if (! error_printed) + print_error("%d when fixing table",my_errno); + if (new_file >= 0) + { + VOID(end_io_cache(&info->rec_cache)); + VOID(my_close(new_file,MYF(0))); + VOID(my_delete(temp_filename,MYF(MY_WME))); + } + } + if (sort_info.key_block) + my_free((gptr) sort_info.key_block,MYF(0)); + if (sort_info.record) + { + my_afree(sort_info.record); + } + VOID(end_io_cache(&read_cache)); + VOID(end_io_cache(&info->rec_cache)); + info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED); + if (!got_error && testflag & T_UNPACK) + { + share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD; + share->pack.header_length=0; + } + DBUG_RETURN(got_error); +} /* rep_by_sort */ + + + /* Read next record and return next key */ + +static int sort_key_read(key) +void *key; +{ + int error; + N_INFO *info; + DBUG_ENTER("sort_key_read"); + + info=sort_info.info; + + if ((error=sort_get_next_record())) + DBUG_RETURN(error); + if (info->s->state.records == sort_info.max_records) + { + print_error("Found too many records; Can`t continue"); + DBUG_RETURN(1); + } + VOID(_nisam_make_key(info,sort_info.key,key,sort_info.record, + sort_info.filepos)); + DBUG_RETURN(sort_write_record()); +} /* sort_key_read */ + + + /* Read next record from file using parameters in sort_info */ + /* Return -1 if end of file, 0 if ok and > 0 if error */ + +static int sort_get_next_record() +{ + int searching; + uint found_record,b_type,left_length; + ulong pos; + byte *to; + BLOCK_INFO block_info; + N_INFO *info; + ISAM_SHARE *share; + DBUG_ENTER("sort_get_next_record"); + + info=sort_info.info; + share=info->s; + switch (share->data_file_type) { + case STATIC_RECORD: + for (;;) + { + if (my_b_read(&read_cache,sort_info.record,share->base.reclength)) + DBUG_RETURN(-1); + sort_info.start_recpos=sort_info.pos; + if (!sort_info.fix_datafile) + sort_info.filepos=sort_info.pos; + sort_info.max_pos=(sort_info.pos+=share->base.reclength); + share->state.splitt++; + if (*sort_info.record) + DBUG_RETURN(0); + if (!sort_info.fix_datafile) + { + share->state.del++; + share->state.empty+=share->base.reclength; + } + } + case DYNAMIC_RECORD: + LINT_INIT(to); + pos=sort_info.pos; + searching=(sort_info.fix_datafile && (testflag & T_EXTEND)); + for (;;) + { + found_record=block_info.second_read= 0; + left_length=1; + do + { + if (pos > sort_info.max_pos) + sort_info.max_pos=pos; + if (found_record && pos == search_after_block) + print_info("Block: %lu used by record at %lu", + search_after_block, + sort_info.start_recpos); + if (_nisam_read_cache(&read_cache,(byte*) block_info.header,pos, + BLOCK_INFO_HEADER_LENGTH, test(! found_record) | 2)) + { + if (found_record) + { + print_info("Can't read whole record at %lu (errno: %d)", + (ulong) sort_info.start_recpos,errno); + goto try_next; + } + DBUG_RETURN(-1); + } + if (searching && ! sort_info.fix_datafile) + { + error_printed=1; + DBUG_RETURN(1); /* Something wrong with data */ + } + if (((b_type=_nisam_get_block_info(&block_info,-1,pos)) & + (BLOCK_ERROR | BLOCK_FATAL_ERROR)) || + ((b_type & BLOCK_FIRST) && + (block_info.rec_len < (uint) share->base.min_pack_length || + block_info.rec_len > (uint) share->base.max_pack_length))) + { + uint i; + if (testflag & T_VERBOSE || searching == 0) + print_info("Wrong bytesec: %3d-%3d-%3d at %10lu; Skipped", + block_info.header[0],block_info.header[1], + block_info.header[2],pos); + if (found_record) + goto try_next; + block_info.second_read=0; + searching=1; + for (i=1 ; i < 11 ; i++) /* Skipp from read string */ + if (block_info.header[i] >= 1 && block_info.header[i] <= 16) + break; + pos+=(ulong) i; + continue; + } + if (block_info.block_len+ (uint) (block_info.filepos-pos) < + share->base.min_block_length || + block_info.block_len-4 > (uint) share->base.max_pack_length) + { + if (!searching) + print_info("Found block with impossible length %u at %lu; Skipped", + block_info.block_len+ (uint) (block_info.filepos-pos), + (ulong) pos); + if (found_record) + goto try_next; + searching=1; + pos++; + block_info.second_read=0; + continue; + } + if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR)) + { + if (!sort_info.fix_datafile && (b_type & BLOCK_DELETED)) + { + share->state.empty+=block_info.block_len; + share->state.del++; + share->state.splitt++; + } + if (found_record) + goto try_next; + /* Check if impossible big deleted block */ + if (block_info.block_len > share->base.max_pack_length +4) + searching=1; + if (searching) + pos++; + else + pos=block_info.filepos+block_info.block_len; + block_info.second_read=0; + continue; + } + + share->state.splitt++; + if (! found_record++) + { + sort_info.find_length=left_length=block_info.rec_len; + sort_info.start_recpos=pos; + if (!sort_info.fix_datafile) + sort_info.filepos=sort_info.start_recpos; + if (sort_info.fix_datafile && (testflag & T_EXTEND)) + sort_info.pos=block_info.filepos+1; + else + sort_info.pos=block_info.filepos+block_info.block_len; + if (share->base.blobs) + { + if (!(to=fix_rec_buff_for_blob(info,block_info.rec_len))) + { + print_error("Not enough memory for blob at %lu", + (ulong) sort_info.start_recpos); + DBUG_RETURN(-1); + } + } + else + to= info->rec_buff; + } + if (left_length < block_info.data_len || ! block_info.data_len) + { + print_info("Found block with too small length at %lu; Skipped", + (ulong) sort_info.start_recpos); + goto try_next; + } + if (block_info.filepos + block_info.data_len > read_cache.end_of_file) + { + print_info("Found block that points outside data file at %lu", + (ulong) sort_info.start_recpos); + goto try_next; + } + if (_nisam_read_cache(&read_cache,to,block_info.filepos, + block_info.data_len, test(found_record == 1))) + { + print_info("Read error for block at: %lu (error: %d); Skipped", + (ulong) block_info.filepos,my_errno); + goto try_next; + } + left_length-=block_info.data_len; + to+=block_info.data_len; + pos=block_info.next_filepos; + if (pos == NI_POS_ERROR && left_length) + { + print_info("Wrong block with wrong total length starting at %lu", + (ulong) sort_info.start_recpos); + goto try_next; + } + if (pos + BLOCK_INFO_HEADER_LENGTH > read_cache.end_of_file) + { + print_info("Found link that points at %lu (outside data file) at %lu", + (ulong) pos,(ulong) sort_info.start_recpos); + goto try_next; + } + } while (left_length); + + if (_nisam_rec_unpack(info,sort_info.record,info->rec_buff, + sort_info.find_length) != MY_FILE_ERROR) + { + if (read_cache.error < 0) + DBUG_RETURN(1); + if ((testflag & (T_EXTEND | T_REP)) || searching) + { + if (_nisam_rec_check(info, sort_info.record)) + { + print_info("Found wrong packed record at %lu", + (ulong) sort_info.start_recpos); + goto try_next; + } + } + DBUG_RETURN(0); + } + try_next: + pos=sort_info.start_recpos+1; + searching=1; + } + case COMPRESSED_RECORD: + for (searching=0 ;; searching=1, sort_info.pos++) + { + if (_nisam_read_cache(&read_cache,(byte*) block_info.header,sort_info.pos, + share->pack.ref_length,1)) + DBUG_RETURN(-1); + if (searching && ! sort_info.fix_datafile) + { + error_printed=1; + DBUG_RETURN(1); /* Something wrong with data */ + } + sort_info.start_recpos=sort_info.pos; + VOID(_nisam_pack_get_block_info(&block_info,share->pack.ref_length,-1, + sort_info.pos)); + if (!block_info.rec_len && + sort_info.pos + MEMMAP_EXTRA_MARGIN == read_cache.end_of_file) + DBUG_RETURN(-1); + if (block_info.rec_len < (uint) share->min_pack_length || + block_info.rec_len > (uint) share->max_pack_length) + { + if (! searching) + print_info("Found block with wrong recordlength: %d at %lu\n", + block_info.rec_len, (ulong) sort_info.pos); + continue; + } + if (_nisam_read_cache(&read_cache,(byte*) info->rec_buff, + block_info.filepos, block_info.rec_len,1)) + { + if (! searching) + print_info("Couldn't read hole record from %lu", + (ulong) sort_info.pos); + continue; + } + if (_nisam_pack_rec_unpack(info,sort_info.record,info->rec_buff, + block_info.rec_len)) + { + if (! searching) + print_info("Found wrong record at %lu",(ulong) sort_info.pos); + continue; + } + if (!sort_info.fix_datafile) + sort_info.filepos=sort_info.pos; + sort_info.max_pos=(sort_info.pos+=share->pack.ref_length+ + block_info.rec_len); + share->state.splitt++; + info->packed_length=block_info.rec_len; + DBUG_RETURN(0); + } + } + DBUG_RETURN(1); /* Impossible */ +} + + + /* Write record to new file */ + +static int sort_write_record() +{ + int flag; + uint block_length,reclength; + byte *from; + uchar *block_buff[3]; + N_INFO *info; + ISAM_SHARE *share; + DBUG_ENTER("sort_write_record"); + + info=sort_info.info; + share=info->s; + if (sort_info.fix_datafile) + { + switch (sort_info.new_data_file_type) { + case STATIC_RECORD: + if (my_b_write(&info->rec_cache,sort_info.record, share->base.reclength)) + { + print_error("%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + sort_info.filepos+=share->base.reclength; + break; + case DYNAMIC_RECORD: + if (! info->blobs) + from=info->rec_buff; + else + { + /* must be sure that local buffer is big enough */ + reclength=info->s->base.pack_reclength+ + _calc_total_blob_length(info,sort_info.record)+ + ALIGN_SIZE(MAX_DYN_BLOCK_HEADER)+N_SPLITT_LENGTH+ + DYN_DELETE_BLOCK_HEADER; + if (sort_info.buff_length < reclength) + { + if (!(sort_info.buff=my_realloc(sort_info.buff, (uint) reclength, + MYF(MY_FREE_ON_ERROR | + MY_ALLOW_ZERO_PTR)))) + DBUG_RETURN(1); + sort_info.buff_length=reclength; + } + from=sort_info.buff+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER); + } + reclength=_nisam_rec_pack(info,from,sort_info.record); + block_length=reclength+ 3 +test(reclength > 65532L); + if (block_length < share->base.min_block_length) + block_length=share->base.min_block_length; + flag=0; + info->update|=HA_STATE_WRITE_AT_END; + if (_nisam_write_part_record(info,0L,block_length,NI_POS_ERROR, + &from,&reclength,&flag)) + { + print_error("%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + sort_info.filepos+=block_length; + break; + case COMPRESSED_RECORD: + reclength=info->packed_length; + save_integer((byte*) block_buff,share->pack.ref_length,reclength); + if (my_b_write(&info->rec_cache,(byte*) block_buff,share->pack.ref_length) + || my_b_write(&info->rec_cache,(byte*) info->rec_buff,reclength)) + { + print_error("%d when writing to datafile",my_errno); + DBUG_RETURN(1); + } + sort_info.filepos+=reclength+share->pack.ref_length; + break; + } + } + share->state.records++; + if (testflag & T_WRITE_LOOP && share->state.records % WRITE_COUNT == 0) + { + printf("%lu\r",(ulong) share->state.records); VOID(fflush(stdout)); + } + DBUG_RETURN(0); +} /* sort_write_record */ + + + /* Compare two keys from _create_index_by_sort */ + +static int sort_key_cmp(const void *not_used __attribute__((unused)), + const void *a,const void *b) +{ + return (_nisam_key_cmp(sort_info.keyseg,*((uchar**) a),*((uchar**) b),0, + SEARCH_SAME)); +} /* sort_key_cmp */ + + +static int sort_key_write(a) +const void *a; +{ + int cmp=sort_info.key_block->inited ? + _nisam_key_cmp(sort_info.keyseg,sort_info.key_block->lastkey,(uchar*) a, + 0,SEARCH_FIND) : -1L; + if ((sort_info.keyinfo->base.flag & HA_NOSAME) && + cmp == 0) + { + sort_info.dupp++; + print_warning("Dupplicate key for record at %10lu against record at %10lu", + sort_info.info->lastpos=get_record_for_key(sort_info.info, + sort_info.keyinfo, + (uchar*) a), + get_record_for_key(sort_info.info,sort_info.keyinfo, + sort_info.key_block->lastkey)); + if (testflag & T_VERBOSE) + _nisam_print_key(stdout,sort_info.keyseg,(uchar*) a); + return(sort_delete_record()); + } + if (cmp) + sort_info.unique++; +#ifndef DBUG_OFF + if (cmp > 0) + { + print_error("Fatal intern error: Keys are not in order from sort"); + return(1); + } +#endif + return (sort_insert_key(sort_info.key_block,(uchar*) a,NI_POS_ERROR)); +} /* sort_key_write */ + + + /* get pointer to record from a key */ + +static ulong get_record_for_key(info,keyinfo,key) +N_INFO *info; +N_KEYDEF *keyinfo; +uchar *key; +{ + return _nisam_dpos(info,0,key+_nisam_keylength(keyinfo,key)); +} /* get_record_for_key */ + + + /* Insert a key in sort-key-blocks */ + +static int sort_insert_key(key_block,key,prev_block) +reg1 SORT_KEY_BLOCKS *key_block; +uchar *key; +ulong prev_block; +{ + uint a_length,t_length,nod_flag; + ulong filepos; + uchar *anc_buff,*lastkey; + S_PARAM s_temp; + N_INFO *info; + DBUG_ENTER("sort_insert_key"); + + anc_buff=key_block->buff; + info=sort_info.info; + lastkey=key_block->lastkey; + nod_flag= (key_block == sort_info.key_block ? 0 : + sort_info.info->s->base.key_reflength); + + if (!key_block->inited) + { + key_block->inited=1; + if (key_block == sort_info.key_block_end) + { + print_error("To many keyblocklevels; Try increasing sort_key_blocks"); + DBUG_RETURN(1); + } + a_length=2+nod_flag; + key_block->end_pos=anc_buff+2; + lastkey=0; /* No previous key in block */ + } + else + a_length=getint(anc_buff); + + /* Save pointer to previous block */ + if (nod_flag) + _nisam_kpointer(info,key_block->end_pos,prev_block); + + t_length=_nisam_get_pack_key_length(sort_info.keyinfo,nod_flag, + (uchar*) 0,lastkey,key,&s_temp); + _nisam_store_key(sort_info.keyinfo,key_block->end_pos+nod_flag,&s_temp); + a_length+=t_length; + putint(anc_buff,a_length,nod_flag); + key_block->end_pos+=t_length; + if (a_length <= sort_info.keyinfo->base.block_length) + { + VOID(_nisam_move_key(sort_info.keyinfo,key_block->lastkey,key)); + key_block->last_length=a_length-t_length; + DBUG_RETURN(0); + } + + /* Fill block with end-zero and write filled block */ + putint(anc_buff,key_block->last_length,nod_flag); + bzero((byte*) anc_buff+key_block->last_length, + sort_info.keyinfo->base.block_length- key_block->last_length); + if ((filepos=_nisam_new(info,sort_info.keyinfo)) == NI_POS_ERROR) + return 1; + if (my_pwrite(info->s->kfile,(byte*) anc_buff, + (uint) sort_info.keyinfo->base.block_length,filepos,MYF_RW)) + DBUG_RETURN(1); + DBUG_DUMP("buff",(byte*) anc_buff,getint(anc_buff)); + + /* Write separator-key to block in next level */ + if (sort_insert_key(key_block+1,key_block->lastkey,filepos)) + DBUG_RETURN(1); + + /* clear old block and write new key in it */ + key_block->inited=0; + DBUG_RETURN(sort_insert_key(key_block,key,prev_block)); +} /* sort_insert_key */ + + + /* Delete record when we found a dupplicated key */ + +static int sort_delete_record() +{ + uint i; + int old_file,error; + uchar *key; + N_INFO *info; + DBUG_ENTER("sort_delete_record"); + + if (rep_quick == 1) + { + VOID(fputs("Quick-recover aborted; Run recovery without switch 'q' or with switch -qq\n",stderr)); + error_printed=1; + DBUG_RETURN(1); + } + info=sort_info.info; + if (info->s->base.options & HA_OPTION_COMPRESS_RECORD) + { + VOID(fputs("Recover aborted; Can't run standard recovery on compressed tables\nwith errors in data-file\nUse switch '--safe-recover' to fix it\n",stderr)); + error_printed=1; + DBUG_RETURN(1); + } + + old_file=info->dfile; + info->dfile=info->rec_cache.file; + if (sort_info.key) + { + key=info->lastkey+info->s->base.max_key_length; + if ((*info->s->read_rnd)(info,sort_info.record,info->lastpos,0) < 0) + { + print_error("Can't read record to be removed"); + info->dfile=old_file; + DBUG_RETURN(1); + } + + for (i=0 ; i < sort_info.key ; i++) + { + VOID(_nisam_make_key(info,i,key,sort_info.record,info->lastpos)); + if (_nisam_ck_delete(info,i,key)) + { + print_error("Can't delete key %d from record to be removed",i+1); + info->dfile=old_file; + DBUG_RETURN(1); + } + } + } + error=flush_io_cache(&info->rec_cache) || (*info->s->delete_record)(info); + info->dfile=old_file; /* restore actual value */ + info->s->state.records--; + DBUG_RETURN(error); +} /* sort_delete_record */ + + + /* Fix all pending blocks and flush everything to disk */ + +static int flush_pending_blocks() +{ + uint nod_flag,length; + ulong filepos; + N_INFO *info; + SORT_KEY_BLOCKS *key_block; + DBUG_ENTER("flush_pending_blocks"); + + filepos= NI_POS_ERROR; /* if empty file */ + info=sort_info.info; + nod_flag=0; + for (key_block=sort_info.key_block ; key_block->inited ; key_block++) + { + key_block->inited=0; + length=getint(key_block->buff); + if (nod_flag) + _nisam_kpointer(info,key_block->end_pos,filepos); + if ((filepos=_nisam_new(info,sort_info.keyinfo)) == NI_POS_ERROR) + DBUG_RETURN(1); + bzero((byte*) key_block->buff+length, + sort_info.keyinfo->base.block_length-length); + if (my_pwrite(info->s->kfile,(byte*) key_block->buff, + (uint) sort_info.keyinfo->base.block_length,filepos,MYF_RW)) + DBUG_RETURN(1); + DBUG_DUMP("buff",(byte*) key_block->buff,length); + nod_flag=1; + } + info->s->state.key_root[sort_info.key]=filepos; /* Last is root for tree */ + DBUG_RETURN(0); +} /* flush_pending_blocks */ + + + /* alloc space and pointers for key_blocks */ + +static SORT_KEY_BLOCKS *alloc_key_blocks(blocks,buffer_length) +uint blocks,buffer_length; +{ + reg1 uint i; + SORT_KEY_BLOCKS *block; + DBUG_ENTER("alloc_key_blocks"); + + if (!(block=(SORT_KEY_BLOCKS*) my_malloc((sizeof(SORT_KEY_BLOCKS)+ + buffer_length+IO_SIZE)*blocks, + MYF(0)))) + { + print_error("Not Enough memory for sort-key-blocks"); + return(0); + } + for (i=0 ; i < blocks ; i++) + { + block[i].inited=0; + block[i].buff=(uchar*) (block+blocks)+(buffer_length+IO_SIZE)*i; + } + DBUG_RETURN(block); +} /* alloc_key_blocks */ + + + /* print warnings and errors */ + /* VARARGS */ + +static void print_info(const char * fmt,...) +{ + va_list args; + + va_start(args,fmt); + VOID(vfprintf(stdout, fmt, args)); + VOID(fputc('\n',stdout)); + va_end(args); + return; +} + +/* VARARGS */ + +static void print_warning(const char * fmt,...) +{ + va_list args; + DBUG_ENTER("print_warning"); + + if (!warning_printed && !error_printed) + { + fflush(stdout); + if (testflag & T_SILENT) + fprintf(stderr,"%s: ISAM file %s\n",my_progname,isam_file_name); + } + warning_printed=1; + va_start(args,fmt); + fprintf(stderr,"%s: warning: ",my_progname); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + va_end(args); + DBUG_VOID_RETURN; +} + +/* VARARGS */ + +void print_error(const char *fmt,...) +{ + va_list args; + DBUG_ENTER("print_error"); + DBUG_PRINT("enter",("format: %s",fmt)); + + if (!warning_printed && !error_printed) + { + fflush(stdout); + if (testflag & T_SILENT) + fprintf(stderr,"%s: ISAM file %s\n",my_progname,isam_file_name); + } + error_printed|=1; + va_start(args,fmt); + fprintf(stderr,"%s: error: ",my_progname); + VOID(vfprintf(stderr, fmt, args)); + VOID(fputc('\n',stderr)); + va_end(args); + DBUG_VOID_RETURN; +} + + /* Check if file is almost full */ + +static int test_if_almost_full(info) +N_INFO *info; +{ + double diff= 0.9; + if (info->s->base.options & HA_OPTION_COMPRESS_RECORD) + { /* Fix problem with pack_isam */ + diff=1.0; + if (info->s->base.rec_reflength == 4) + info->s->base.max_data_file_length= (uint32) ~0L; + else + info->s->base.max_data_file_length= + 1L << (info->s->base.rec_reflength); + } + return (my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0)) > + (ulong) (info->s->base.max_key_file_length*diff) || + my_seek(info->dfile,0L,MY_SEEK_END,MYF(0)) > + (ulong) (info->s->base.max_data_file_length*diff)); +} + + /* Recreate table with bigger more alloced record-data */ + +static int recreate_database(org_info,filename) +N_INFO **org_info; +char *filename; +{ + int error; + N_INFO info; + ISAM_SHARE share; + N_KEYDEF *keyinfo; + N_RECINFO *recinfo,*rec,*end; + uint unpack; + ulong max_records; + char name[FN_REFLEN]; + + error=1; /* Default error */ + info= **org_info; + share= *(*org_info)->s; + unpack= (share.base.options & HA_OPTION_COMPRESS_RECORD) && + (testflag & T_UNPACK); + if (!(keyinfo=(N_KEYDEF*) my_alloca(sizeof(N_KEYDEF)*share.base.keys))) + return 0; + memcpy((byte*) keyinfo,(byte*) share.keyinfo, + (size_t) (sizeof(N_KEYDEF)*share.base.keys)); + if (!(recinfo=(N_RECINFO*) + my_alloca(sizeof(N_RECINFO)*(share.base.fields+1)))) + { + my_afree((gptr) keyinfo); + return 1; + } + memcpy((byte*) recinfo,(byte*) share.rec, + (size_t) (sizeof(N_RECINFO)*(share.base.fields+1))); + for (rec=recinfo,end=recinfo+share.base.fields; rec != end ; rec++) + { + if (rec->base.type == (int) FIELD_BLOB) + rec->base.length+=sizeof(char*); + else if (unpack && !(share.base.options & HA_OPTION_PACK_RECORD)) + rec->base.type=(int) FIELD_NORMAL; + } + + if (share.base.options & HA_OPTION_COMPRESS_RECORD) + share.base.records=max_records=share.state.records; + else if (share.base.min_pack_length) + max_records=(ulong) (my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)) / + (ulong) share.base.min_pack_length); + else + max_records=0; + unpack= (share.base.options & HA_OPTION_COMPRESS_RECORD) && + (testflag & T_UNPACK); + share.base.options&= ~HA_OPTION_TEMP_COMPRESS_RECORD; + VOID(nisam_close(*org_info)); + if (nisam_create(fn_format(name,filename,"",N_NAME_IEXT, + 4+ (opt_follow_links ? 16 : 0)), + share.base.keys,keyinfo,recinfo, + max(max_records,share.base.records),share.base.reloc, + HA_DONT_TOUCH_DATA, + share.base.options | + (unpack ? HA_OPTION_TEMP_COMPRESS_RECORD + : 0), + (ulong) my_seek(info.dfile,0L,MY_SEEK_END,MYF(0)))) + { + print_error("Got error %d when trying to recreate indexfile",my_errno); + goto end; + } + *org_info=nisam_open(name,O_RDWR, + (testflag & T_WAIT_FOREVER) ? HA_OPEN_WAIT_IF_LOCKED : + (testflag & T_DESCRIPT) ? HA_OPEN_IGNORE_IF_LOCKED : + HA_OPEN_ABORT_IF_LOCKED); + if (!*org_info) + { + print_error("Got error %d when trying to open re-created indexfile", + my_errno); + goto end; + } + /* We are modifing */ + (*org_info)->s->base.options&= ~HA_OPTION_READ_ONLY_DATA; + VOID(_nisam_readinfo(*org_info,F_WRLCK,0)); + (*org_info)->s->state.records=share.state.records; + if (share.base.create_time) + (*org_info)->s->base.create_time=share.base.create_time; + (*org_info)->s->state.uniq=(*org_info)->this_uniq= + share.state.uniq; + (*org_info)->s->state.del=share.state.del; + (*org_info)->s->state.dellink=share.state.dellink; + (*org_info)->s->state.empty=share.state.empty; + (*org_info)->s->state.data_file_length=share.state.data_file_length; + if (update_state_info(*org_info,UPDATE_TIME | UPDATE_STAT)) + goto end; + error=0; +end: + my_afree((gptr) keyinfo); + my_afree((gptr) recinfo); + return error; +} + + /* Store long in 1,2,3 or 4 bytes */ + +static void save_integer(pos,pack_length,value) +byte *pos; +uint pack_length; +ulong value; +{ + switch (pack_length) { + case 4: int4store(pos,value); break; + case 3: int3store(pos,value); break; + case 2: int2store(pos,(uint) value); break; + case 1: pos[0]= (char) (uchar) value; break; + default: break; + } + return; +} + + /* write suffix to data file if neaded */ + +static int write_data_suffix(info) +N_INFO *info; +{ + if (info->s->base.options & HA_OPTION_COMPRESS_RECORD && + sort_info.fix_datafile) + { + char buff[MEMMAP_EXTRA_MARGIN]; + bzero(buff,sizeof(buff)); + if (my_b_write(&info->rec_cache,buff,sizeof(buff))) + { + print_error("%d when writing to datafile",my_errno); + return 1; + } + read_cache.end_of_file+=sizeof(buff); + } + return 0; +} + + + /* Update state and isamchk_time of indexfile */ + +static int update_state_info(info,update) +N_INFO *info; +uint update; +{ + ISAM_SHARE *share=info->s; + uint base_pos=uint2korr(info->s->state.header.base_pos); + + if (update & (UPDATE_STAT | UPDATE_SORT | UPDATE_TIME)) + { + if (offsetof(N_BASE_INFO,rec_per_key) > + uint2korr(share->state.header.base_info_length)) + { + VOID(fputs("Internal error: Trying to change base of old table\n", + stderr)); + } + else + { + if (update & UPDATE_TIME) + { + share->base.isamchk_time= (long) time((time_t*) 0); + if (!share->base.create_time) + share->base.create_time=share->base.isamchk_time; + if (my_pwrite(share->kfile,(gptr) &share->base.create_time, + sizeof(long)*2, + base_pos+offsetof(N_BASE_INFO,create_time), + MYF(MY_NABP))) + goto err; + } + if (update & (UPDATE_STAT | UPDATE_SORT)) + { + if (my_pwrite(share->kfile,(gptr) share->base.rec_per_key, + sizeof(long)*share->state.keys+sizeof(uint), + base_pos+offsetof(N_BASE_INFO,rec_per_key), + MYF(MY_NABP))) + goto err; + } + } + } + { /* Force update of status */ + int error; + uint r_locks=share->r_locks,w_locks=share->w_locks; + share->r_locks=share->w_locks=0; + error=_nisam_writeinfo(info,2); + share->r_locks=r_locks; share->w_locks=w_locks; + if (!error) + return 0; + } +err: + print_error("%d when updateing keyfile",my_errno); + return 1; +} diff --git a/isam/isamdef.h b/isam/isamdef.h new file mode 100644 index 00000000000..da08b5f6a14 --- /dev/null +++ b/isam/isamdef.h @@ -0,0 +1,416 @@ +/* Copyright (C) 2000 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 */ + +/* Denna fil includeras i alla isam-filer */ + +#define ISAM_LIBRARY +#include <nisam.h> /* Structs & some defines */ +#ifdef THREAD +#include <my_pthread.h> +#include <thr_lock.h> +#else +#include <my_no_pthread.h> +#endif + +#ifdef my_write +#undef my_write /* We want test if disk full */ +#endif +#undef HA_SORT_ALLOWS_SAME +#define HA_SORT_ALLOWS_SAME 128 /* Can't be > 128 in NISAM */ + +#ifdef __WATCOMC__ +#pragma pack(2) +#define uint uint16 /* Same format as in MSDOS */ +#endif +#ifdef __ZTC__ +#pragma ZTC align 2 +#define uint uint16 /* Same format as in MSDOS */ +#endif +#if defined(__WIN__) && defined(_MSC_VER) +#pragma pack(push,isamdef,2) +#define uint uint16 +#endif + +typedef struct st_state_info +{ + struct { /* Fileheader */ + uchar file_version[4]; + uchar options[2]; + uchar header_length[2]; + uchar state_info_length[2]; + uchar base_info_length[2]; + uchar base_pos[2]; + uchar not_used[2]; + } header; + + ulong records; /* Antal record i databasen */ + ulong del; /* Antalet borttagna poster */ + ulong dellink; /* L{nk till n{sta borttagna */ + ulong key_file_length; + ulong data_file_length; + ulong splitt; /* Antal splittrade delar */ + ulong empty; /* Outnyttjat utrymme */ + ulong process; /* Vem som senast uppdatera */ + ulong loop; /* not used anymore */ + ulong uniq; /* Unik nr i denna process */ + ulong key_root[N_MAXKEY]; /* Pekare till rootblocken */ + ulong key_del[N_MAXKEY]; /* Del-l{nkar f|r n-block */ + ulong sec_index_changed; /* Updated when new sec_index */ + ulong sec_index_used; /* 1 bit for each sec index in use */ + ulong version; /* timestamp of create */ + uint keys; /* Keys in use for database */ +} N_STATE_INFO; + + +typedef struct st_base_info +{ + ulong keystart; /* Var nycklarna b|rjar */ + ulong records,reloc; /* Parameter vid skapandet */ + ulong max_pack_length; /* Max possibly length of packed rec.*/ + ulong max_data_file_length; + ulong max_key_file_length; + uint reclength; /* length of unpacked record */ + uint options; /* Options used */ + uint pack_reclength; /* Length of full packed rec. */ + uint min_pack_length; + uint min_block_length; + uint rec_reflength; /* = 2 or 3 or 4 */ + uint key_reflength; /* = 2 or 3 or 4 */ + uint keys; /* Keys defined for database */ + uint blobs; /* Number of blobs */ + uint max_block; /* Max blockl{ngd anv{nd */ + uint max_key_length; /* L{ngsta nyckel-l{ngden */ + uint fields, /* Antal f{lt i databasen */ + pack_fields, /* Packade f{lt i databasen */ + pack_bits; /* Length of packed bits */ + time_t create_time; /* Time when created database */ + time_t isamchk_time; /* Time for last recover */ + ulong rec_per_key[N_MAXKEY]; /* for sql optimizing */ + uint sortkey; /* sorted by this key */ +} N_BASE_INFO; + + +#ifdef __ZTC__ +#pragma ZTC align +#undef uint +#endif +#ifdef __WATCOMC__ +#pragma pack() +#undef uint +#endif +#if defined(__WIN__) && defined(_MSC_VER) +#pragma pack(pop,isamdef) +#undef uint +#endif + + /* Structs used intern in database */ + +typedef struct st_n_blob /* Info of record */ +{ + uint offset; /* Offset to blob in record */ + uint pack_length; /* Type of packed length */ + uint length; /* Calc:ed for each record */ +} N_BLOB; + + +typedef struct st_isam_pack { + ulong header_length; + uint ref_length; +} N_PACK; + + +typedef struct st_isam_share { /* Shared between opens */ + N_STATE_INFO state; + N_BASE_INFO base; + N_KEYDEF *keyinfo; /* Nyckelinfo */ + N_RECINFO *rec; /* Pointer till recdata */ + N_PACK pack; /* Data about packed records */ + N_BLOB *blobs; /* Pointer to blobs */ + char *filename; /* Name of indexfile */ + byte *file_map; /* mem-map of file if possible */ + ulong this_process; /* processid */ + ulong last_process; /* For table-change-check */ + ulong last_version; /* Version on start */ + uint rec_reflength; /* rec_reflength in use now */ + int kfile; /* Shared keyfile */ + int mode; /* mode of file on open */ + int reopen; /* How many times reopened */ + uint state_length; + uint w_locks,r_locks; /* Number of read/write locks */ + uint min_pack_length; /* Theese is used by packed data */ + uint max_pack_length; + uint blocksize; /* blocksize of keyfile */ + my_bool changed,not_flushed; /* If changed since lock */ + int rnd; /* rnd-counter */ + DECODE_TREE *decode_trees; + uint16 *decode_tables; + enum data_file_type data_file_type; + int (*read_record)(struct st_isam_info*, ulong, byte*); + int (*write_record)(struct st_isam_info*, const byte*); + int (*update_record)(struct st_isam_info*, ulong, const byte*); + int (*delete_record)(struct st_isam_info*); + int (*read_rnd)(struct st_isam_info*, byte*, ulong, int); + int (*compare_record)(struct st_isam_info*, const byte *); +#ifdef THREAD + THR_LOCK lock; + pthread_mutex_t intern_lock; /* Locking for use with _locking */ +#endif +} ISAM_SHARE; + + +typedef uint bit_type; + +typedef struct st_bit_buff { /* Used for packing of record */ + bit_type current_byte; + uint bits; + uchar *pos,*end; + uint error; +} BIT_BUFF; + + +typedef struct st_isam_info { + ISAM_SHARE *s; /* Shared between open:s */ + N_BLOB *blobs; /* Pointer to blobs */ + int dfile; /* The datafile */ + BIT_BUFF bit_buff; + uint options; + uint opt_flag; /* Optim. for space/speed */ + uint update; /* If file changed since open */ + char *filename; /* parameter to open filename */ + ulong this_uniq; /* uniq filenumber or thread */ + ulong last_uniq; /* last uniq number */ + ulong this_loop; /* counter for this open */ + ulong last_loop; /* last used counter */ + ulong lastpos, /* Last record position */ + nextpos; /* Position to next record */ + ulong int_pos; /* Intern variabel */ + ulong dupp_key_pos; /* Position to record with dupp key */ + ulong last_search_keypage; + ulong save_lastpos; + uint packed_length; /* Length of found, packed record */ + uint alloced_rec_buff_length; /* Max recordlength malloced */ + uchar *buff, /* Temp area for key */ + *lastkey; /* Last used search key */ + byte *rec_buff, /* Tempbuff for recordpack */ + *rec_alloc; /* Malloced area for record */ + uchar *int_keypos, /* Intern variabel */ + *int_maxpos; /* Intern variabel */ + int lastinx; /* Last used index */ + int errkey; /* Got last error on this key */ + uint data_changed; /* Somebody has changed data */ + int lock_type; /* How database was locked */ + int tmp_lock_type; /* When locked by readinfo */ + int was_locked; /* Was locked in panic */ + myf lock_wait; /* is 0 or MY_DONT_WAIT */ + my_bool page_changed; + my_bool buff_used; + uint save_update; /* When using KEY_READ */ + int save_lastinx; + int (*read_record)(struct st_isam_info*, ulong, byte*); + LIST open_list; + IO_CACHE rec_cache; /* When cacheing records */ +#ifdef THREAD + THR_LOCK_DATA lock; +#endif +} N_INFO; + + + /* Some defines used by isam-funktions */ + +#define USE_HOLE_KEY 0 /* Use hole key in _nisam_search() */ +#define F_EXTRA_LCK -1 + + /* bits in opt_flag */ +#define MEMMAP_USED 32 +#define REMEMBER_OLD_POS 64 + +#define getint(x) ((uint) (uint16) *((int16*) (x)) & 32767) +#define putint(x,y,nod) (*((uint16*) (x))= ((nod ? (uint16) 32768 : 0)+(uint16) (y))) +#ifdef WORDS_BIGENDIAN +#define test_if_nod(x) (x[0] & 128 ? info->s->base.key_reflength : 0) +#else +#define test_if_nod(x) (x[1] & 128 ? info->s->base.key_reflength : 0) +#endif + +#define N_MIN_BLOCK_LENGTH 8 /* Because of delete-link */ +#define N_EXTEND_BLOCK_LENGTH 20 /* Don't use to small record-blocks */ +#define N_SPLITT_LENGTH ((N_EXTEND_BLOCK_LENGTH+3)*2) +#define MAX_DYN_BLOCK_HEADER 11 /* Max prefix of record-block */ +#define DYN_DELETE_BLOCK_HEADER 8 /* length of delete-block-header */ +#define MEMMAP_EXTRA_MARGIN 7 /* Write this as a suffix for file */ +#define INDEX_BLOCK_MARGIN 16 /* Safety margin for .ISM tables */ + +#define PACK_TYPE_SELECTED 1 /* Bits in field->pack_type */ +#define PACK_TYPE_SPACE_FIELDS 2 +#define PACK_TYPE_ZERO_FILL 4 + +#ifdef THREAD +extern pthread_mutex_t THR_LOCK_isam; +#endif + + /* Some extern variables */ + +extern LIST *nisam_open_list; +extern uchar NEAR nisam_file_magic[],NEAR nisam_pack_file_magic[]; +extern uint NEAR nisam_read_vec[],nisam_quick_table_bits; +extern File nisam_log_file; + + /* This is used by _nisam_get_pack_key_length och _nisam_store_key */ + +typedef struct st_s_param +{ + uint ref_length,key_length, + n_ref_length, + n_length, + totlength, + part_of_prev_key,prev_length; + uchar *key, *prev_key; +} S_PARAM; + + /* Prototypes for intern functions */ + +extern int _nisam_read_dynamic_record(N_INFO *info,ulong filepos,byte *buf); +extern int _nisam_write_dynamic_record(N_INFO*, const byte*); +extern int _nisam_update_dynamic_record(N_INFO*, ulong, const byte*); +extern int _nisam_delete_dynamic_record(N_INFO *info); +extern int _nisam_cmp_dynamic_record(N_INFO *info,const byte *record); +extern int _nisam_read_rnd_dynamic_record(N_INFO *, byte *,ulong, int); +extern int _nisam_write_blob_record(N_INFO*, const byte*); +extern int _nisam_update_blob_record(N_INFO*, ulong, const byte*); +extern int _nisam_read_static_record(N_INFO *info,ulong filepos,byte *buf); +extern int _nisam_write_static_record(N_INFO*, const byte*); +extern int _nisam_update_static_record(N_INFO*, ulong, const byte*); +extern int _nisam_delete_static_record(N_INFO *info); +extern int _nisam_cmp_static_record(N_INFO *info,const byte *record); +extern int _nisam_read_rnd_static_record(N_INFO*, byte *,ulong, int); +extern int _nisam_ck_write(N_INFO *info,uint keynr,uchar *key); +extern int _nisam_enlarge_root(N_INFO *info,uint keynr,uchar *key); +extern int _nisam_insert(N_INFO *info,N_KEYDEF *keyinfo,uchar *key, + uchar *anc_buff,uchar *key_pos,uchar *key_buff, + uchar *father_buff, uchar *father_keypos, + ulong father_page); +extern int _nisam_splitt_page(N_INFO *info,N_KEYDEF *keyinfo,uchar *key, + uchar *buff,uchar *key_buff); +extern uchar *_nisam_find_half_pos(N_INFO *info,N_KEYDEF *keyinfo,uchar *page, + uchar *key); +extern uint _nisam_get_pack_key_length(N_KEYDEF *keyinfo,uint nod_flag, + uchar *key_pos,uchar *key_buff, + uchar *key, S_PARAM *s_temp); +extern void _nisam_store_key(N_KEYDEF *keyinfo,uchar *key_pos, + S_PARAM *s_temp); +extern int _nisam_ck_delete(N_INFO *info,uint keynr,uchar *key); +extern int _nisam_readinfo(N_INFO *info,int lock_flag,int check_keybuffer); +extern int _nisam_writeinfo(N_INFO *info, uint flag); +extern int _nisam_test_if_changed(N_INFO *info); +extern int _nisam_check_index(N_INFO *info,int inx); +extern int _nisam_search(N_INFO *info,N_KEYDEF *keyinfo,uchar *key,uint key_len,uint nextflag,ulong pos); +extern int _nisam_bin_search(struct st_isam_info *info,N_KEYDEF *keyinfo,uchar *page,uchar *key,uint key_len,uint comp_flag,uchar * *ret_pos,uchar *buff); +extern int _nisam_seq_search(N_INFO *info,N_KEYDEF *keyinfo,uchar *page,uchar *key,uint key_len,uint comp_flag,uchar * *ret_pos,uchar *buff); +extern ulong _nisam_kpos(uint nod_flag,uchar *after_key); +extern void _nisam_kpointer(N_INFO *info,uchar *buff,ulong pos); +extern ulong _nisam_dpos(N_INFO *info, uint nod_flag,uchar *after_key); +extern void _nisam_dpointer(N_INFO *info, uchar *buff,ulong pos); +extern int _nisam_key_cmp(N_KEYSEG *keyseg,uchar *a,uchar *b, + uint key_length,uint nextflag); +extern uint _nisam_get_key(N_KEYDEF *keyinfo,uint nod_flag,uchar * *page,uchar *key); +extern uint _nisam_get_static_key(N_KEYDEF *keyinfo,uint nod_flag,uchar * *page,uchar *key); +extern uchar *_nisam_get_last_key(N_INFO *info,N_KEYDEF *keyinfo,uchar *keypos,uchar *lastkey,uchar *endpos); +extern uint _nisam_keylength(N_KEYDEF *keyinfo,uchar *key); +extern uchar *_nisam_move_key(N_KEYDEF *keyinfo,uchar *to,uchar *from); +extern int _nisam_search_next(N_INFO *info,N_KEYDEF *keyinfo,uchar *key,uint nextflag,ulong pos); +extern int _nisam_search_first(N_INFO *info,N_KEYDEF *keyinfo,ulong pos); +extern int _nisam_search_last(N_INFO *info,N_KEYDEF *keyinfo,ulong pos); +extern uchar *_nisam_fetch_keypage(N_INFO *info,N_KEYDEF *keyinfo,my_off_t page, + uchar *buff,int return_buffer); +extern int _nisam_write_keypage(N_INFO *info,N_KEYDEF *keyinfo,my_off_t page, + uchar *buff); +extern int _nisam_dispose(N_INFO *info,N_KEYDEF *keyinfo,my_off_t pos); +extern ulong _nisam_new(N_INFO *info,N_KEYDEF *keyinfo); +extern uint _nisam_make_key(N_INFO *info,uint keynr,uchar *key, + const char *record,ulong filepos); +extern uint _nisam_pack_key(N_INFO *info,uint keynr,uchar *key,uchar *old,uint key_length); +extern int _nisam_read_key_record(N_INFO *info,ulong filepos,byte *buf); +extern int _nisam_read_cache(IO_CACHE *info,byte *buff,ulong pos, + uint length,int re_read_if_possibly); +extern byte *fix_rec_buff_for_blob(N_INFO *info,uint blob_length); +extern uint _nisam_rec_unpack(N_INFO *info,byte *to,byte *from, + uint reclength); +my_bool _nisam_rec_check(N_INFO *info,const char *from); +extern int _nisam_write_part_record(N_INFO *info,ulong filepos,uint length, + ulong next_filepos,byte **record, + uint *reclength,int *flag); +extern void _nisam_print_key(FILE *stream,N_KEYSEG *keyseg,const uchar *key); +extern my_bool _nisam_read_pack_info(N_INFO *info,pbool fix_keys); +extern int _nisam_read_pack_record(N_INFO *info,ulong filepos,byte *buf); +extern int _nisam_read_rnd_pack_record(N_INFO*, byte *,ulong, int); +extern int _nisam_pack_rec_unpack(N_INFO *info,byte *to,byte *from, + uint reclength); + +typedef struct st_sortinfo { + uint key_length; + ulong max_records; + int (*key_cmp)(const void *, const void *, const void *); + int (*key_read)(void *buff); + int (*key_write)(const void *buff); + void (*lock_in_memory)(void); +} SORT_PARAM; + +int _create_index_by_sort(SORT_PARAM *info,pbool no_messages, + uint sortbuff_size); + +#define BLOCK_INFO_HEADER_LENGTH 11 + +typedef struct st_block_info { /* Parameter to _nisam_get_block_info */ + uchar header[BLOCK_INFO_HEADER_LENGTH]; + uint rec_len; + uint data_len; + uint block_len; + ulong filepos; /* Must be ulong on Alpha! */ + ulong next_filepos; + uint second_read; +} BLOCK_INFO; + + /* bits in return from _nisam_get_block_info */ + +#define BLOCK_FIRST 1 +#define BLOCK_LAST 2 +#define BLOCK_DELETED 4 +#define BLOCK_ERROR 8 /* Wrong data */ +#define BLOCK_SYNC_ERROR 16 /* Right data at wrong place */ +#define BLOCK_FATAL_ERROR 32 /* hardware-error */ + +enum nisam_log_commands { + LOG_OPEN,LOG_WRITE,LOG_UPDATE,LOG_DELETE,LOG_CLOSE,LOG_EXTRA,LOG_LOCK +}; + +#define nisam_log_simple(a,b,c,d) if (nisam_log_file >= 0) _nisam_log(a,b,c,d) +#define nisam_log_command(a,b,c,d,e) if (nisam_log_file >= 0) _nisam_log_command(a,b,c,d,e) +#define nisam_log_record(a,b,c,d,e) if (nisam_log_file >= 0) _nisam_log_record(a,b,c,d,e) + +extern uint _nisam_get_block_info(BLOCK_INFO *,File, ulong); +extern uint _nisam_rec_pack(N_INFO *info,byte *to,const byte *from); +extern uint _nisam_pack_get_block_info(BLOCK_INFO *, uint, File, ulong); +extern uint _calc_total_blob_length(N_INFO *info,const byte *record); +extern void _nisam_log(enum nisam_log_commands command,N_INFO *info, + const byte *buffert,uint length); +extern void _nisam_log_command(enum nisam_log_commands command, + N_INFO *info, const byte *buffert, + uint length, int result); +extern void _nisam_log_record(enum nisam_log_commands command,N_INFO *info, + const byte *record,ulong filepos, + int result); +extern my_bool _nisam_memmap_file(N_INFO *info); +extern void _nisam_unmap_file(N_INFO *info); diff --git a/isam/isamlog.c b/isam/isamlog.c new file mode 100644 index 00000000000..ddeea8a267d --- /dev/null +++ b/isam/isamlog.c @@ -0,0 +1,839 @@ +/* Copyright (C) 2000 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 */ + +/* write whats in isam.log */ + +#ifndef USE_MY_FUNC +#define USE_MY_FUNC +#endif + +#include "isamdef.h" +#include <my_tree.h> +#include <stdarg.h> +#ifdef HAVE_GETRUSAGE +#include <sys/resource.h> +#endif + +#define FILENAME(A) (A ? A->show_name : "Unknown") + +struct file_info { + long process; + int filenr,id; + my_string name,show_name,record; + N_INFO *isam; + bool closed,used; + ulong accessed; +}; + +struct test_if_open_param { + my_string name; + int max_id; +}; + +struct st_access_param +{ + ulong min_accessed; + struct file_info *found; +}; + +#define NO_FILEPOS (ulong) ~0L + +extern int main(int argc,char * *argv); +static void get_options(int *argc,char ***argv); +static int examine_log(my_string file_name,char **table_names); +static int read_string(IO_CACHE *file,gptr *to,uint length); +static int file_info_compare(void *a,void *b); +static int test_if_open(struct file_info *key,element_count count, + struct test_if_open_param *param); +static void fix_blob_pointers(N_INFO *isam,byte *record); +static uint set_maximum_open_files(uint); +static int test_when_accessed(struct file_info *key,element_count count, + struct st_access_param *access_param); +static void file_info_free(struct file_info *info); +static int close_some_file(TREE *tree); +static int reopen_closed_file(TREE *tree,struct file_info *file_info); +static int find_record_with_key(struct file_info *file_info,byte *record); +static void printf_log(const char *str,...); +static bool cmp_filename(struct file_info *file_info,my_string name); + +static uint verbose=0,update=0,test_info=0,max_files=0,re_open_count=0, + recover=0,prefix_remove=0; +static my_string log_filename=0,filepath=0,write_filename=0,record_pos_file=0; +static ulong com_count[10][3],number_of_commands=(ulong) ~0L,start_offset=0, + record_pos= NO_FILEPOS,isamlog_filepos,isamlog_process; +static const char *command_name[]= +{"open","write","update","delete","close","extra","lock","re-open",NullS}; + + +int main(argc,argv) +int argc; +char **argv; +{ + int error,i,first; + ulong total_count,total_error,total_recover; + MY_INIT(argv[0]); + + log_filename=nisam_log_filename; + get_options(&argc,&argv); + /* Nr of isam-files */ + max_files=(set_maximum_open_files(min(max_files,8))-6)/2; + + if (update) + printf("Trying to %s isamfiles according to log '%s'\n", + (recover ? "recover" : "update"),log_filename); + error= examine_log(log_filename,argv); + if (update && ! error) + puts("isamfile:s updated successfully"); + total_count=total_error=total_recover=0; + for (i=first=0 ; command_name[i] ; i++) + { + if (com_count[i][0]) + { + if (!first++) + { + if (verbose || update) + puts(""); + puts("Commands Used count Errors Recover errors"); + } + printf("%-12s%9ld%10ld%17ld\n",command_name[i],com_count[i][0], + com_count[i][1],com_count[i][2]); + total_count+=com_count[i][0]; + total_error+=com_count[i][1]; + total_recover+=com_count[i][2]; + } + } + if (total_count) + printf("%-12s%9ld%10ld%17ld\n","Total",total_count,total_error, + total_recover); + if (re_open_count) + printf("Had to do %d re-open because of too few possibly open files\n", + re_open_count); + VOID(nisam_panic(HA_PANIC_CLOSE)); + my_end(test_info ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error); + return 0; /* No compiler warning */ +} /* main */ + + +static void get_options(argc,argv) +register int *argc; +register char ***argv; +{ + int help,version; + const char *usage; + char *pos, option; + + help=0; + usage="Usage: %s [-?iruvIV] [-c #] [-f #] [-F filepath/] [-o #] [-R file recordpos] [-w write_file] [log-filename [table ...]] \n"; + pos= (char*) ""; + + while (--*argc > 0 && *(pos = *(++*argv)) == '-' ) { + while (*++pos) + { + version=0; + switch((option=*pos)) { + case '#': + DBUG_PUSH (++pos); + pos= (char*) " "; /* Skipp rest of arg */ + break; + case 'c': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + number_of_commands=(ulong) atol(pos); + pos= (char*) " "; + break; + case 'u': + update=1; + break; + case 'f': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + max_files=(uint) atoi(pos); + pos= (char*) " "; + break; + case 'i': + test_info=1; + break; + case 'o': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + start_offset=(ulong) atol(pos); + pos= (char*) " "; + break; + case 'p': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + prefix_remove=atoi(pos); + break; + case 'r': + update=1; + recover++; + break; + case 'R': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + record_pos_file=pos; + if (!--*argc) + goto err; + record_pos=(ulong) atol(*(++*argv)); + pos= (char*) " "; + break; + case 'v': + verbose++; + break; + case 'w': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + write_filename=pos; + pos= (char*) " "; + break; + case 'F': + if (! *++pos) + { + if (!--*argc) + goto err; + else + pos= *(++*argv); + } + filepath=pos; + pos= (char*) " "; + break; + case 'V': + version=1; + /* Fall through */ + case 'I': + case '?': + printf("%s Ver 3.1 for %s at %s\n",my_progname,SYSTEM_TYPE, + MACHINE_TYPE); + puts("TCX Datakonsult AB, by Monty, for your professional use\n"); + if (version) + break; + puts("Write info about whats in a nisam log file."); + printf("If no file name is given %s is used\n",log_filename); + puts(""); + printf(usage,my_progname); + puts(""); + puts("Options: -? or -I \"Info\" -V \"version\" -c \"do only # commands\""); + puts(" -f \"max open files\" -F \"filepath\" -i \"extra info\""); + puts(" -o \"offset\" -p # \"remove # components from path\""); + puts(" -r \"recover\" -R \"file recordposition\""); + puts(" -u \"update\" -v \"verbose\" -w \"write file\""); + puts("\nOne can give a second and a third '-v' for more verbose."); + puts("Normaly one does a update (-u)."); + puts("If a recover is done all writes and all possibly updates and deletes is done\nand errors are only counted."); + puts("If one gives table names as arguments only these tables will be updated\n"); + help=1; + break; + default: + printf("illegal option: \"-%c\"\n",*pos); + break; + } + } + } + if (! *argc) + { + if (help) + exit(0); + (*argv)++; + } + if (*argc >= 1) + { + log_filename=pos; + (*argc)--; + (*argv)++; + } + return; + err: + VOID(fprintf(stderr,"option \"%c\" used without or with wrong argument\n", + option)); + exit(1); +} + + +static int examine_log(my_string file_name, char **table_names) +{ + uint command,result,files_open; + ulong access_time,length; + uint32 filepos; + int lock_command,ni_result; + char isam_file_name[FN_REFLEN]; + uchar head[20]; + gptr buff; + struct test_if_open_param open_param; + IO_CACHE cache; + File file; + FILE *write_file; + enum ha_extra_function extra_command; + TREE tree; + struct file_info file_info,*curr_file_info; + DBUG_ENTER("examine_log"); + + if ((file=my_open(file_name,O_RDONLY,MYF(MY_WME))) < 0) + DBUG_RETURN(1); + write_file=0; + if (write_filename) + { + if (!(write_file=my_fopen(write_filename,O_WRONLY,MYF(MY_WME)))) + { + my_close(file,MYF(0)); + DBUG_RETURN(1); + } + } + + init_io_cache(&cache,file,0,READ_CACHE,start_offset,0,MYF(0)); + bzero((gptr) com_count,sizeof(com_count)); + init_tree(&tree,0,sizeof(file_info),(qsort_cmp) file_info_compare,0, + (void(*)(void*)) file_info_free); + VOID(init_key_cache(KEY_CACHE_SIZE,(uint) (10*4*(IO_SIZE+MALLOC_OVERHEAD)))); + + files_open=0; access_time=0; + while (access_time++ != number_of_commands && + !my_b_read(&cache,(byte*) head,9)) + { + isamlog_filepos=my_b_tell(&cache)-9L; + file_info.filenr=uint2korr(head+1); + isamlog_process=file_info.process=(long) uint4korr(head+3); + result=uint2korr(head+7); + if ((curr_file_info=(struct file_info*) tree_search(&tree,&file_info))) + { + curr_file_info->accessed=access_time; + if (update && curr_file_info->used && curr_file_info->closed) + { + if (reopen_closed_file(&tree,curr_file_info)) + { + command=sizeof(com_count)/sizeof(com_count[0][0])/3; + result=0; + goto com_err; + } + } + } + command=(uint) head[0]; + if (command < sizeof(com_count)/sizeof(com_count[0][0])/3 && + (!curr_file_info || curr_file_info->used)) + { + com_count[command][0]++; + if (result) + com_count[command][1]++; + } + switch ((enum nisam_log_commands) command) { + case LOG_OPEN: + com_count[command][0]--; /* Must be counted explicite */ + if (result) + com_count[command][1]--; + + if (curr_file_info) + printf("\nWarning: %s is opened twice with same process and filenumber\n", + curr_file_info->show_name); + if (my_b_read(&cache,(byte*) head,2)) + goto err; + file_info.name=0; + file_info.show_name=0; + file_info.record=0; + if (read_string(&cache,(gptr*) &file_info.name,(uint) uint2korr(head))) + goto err; + { + uint i; + char *pos=file_info.name,*to; + for (i=0 ; i < prefix_remove ; i++) + { + char *next; + if (!(next=strchr(pos,FN_LIBCHAR))) + break; + pos=next+1; + } + to=isam_file_name; + if (filepath) + { + strmov(isam_file_name,filepath); + convert_dirname(isam_file_name); + to=strend(isam_file_name); + } + strmov(to,pos); + fn_ext(isam_file_name)[0]=0; /* Remove extension */ + } + open_param.name=file_info.name; + open_param.max_id=0; + VOID(tree_walk(&tree,(tree_walk_action) test_if_open,(void*) &open_param, + left_root_right)); + file_info.id=open_param.max_id+1; + file_info.show_name=my_memdup(isam_file_name, + (uint) strlen(isam_file_name)+6, + MYF(MY_WME)); + if (file_info.id > 1) + sprintf(strend(file_info.show_name),"<%d>",file_info.id); + file_info.closed=1; + file_info.accessed=access_time; + file_info.used=1; + if (table_names[0]) + { + char **name; + file_info.used=0; + for (name=table_names ; *name ; name++) + { + if (!strcmp(*name,isam_file_name)) + file_info.used=1; /* Update/log only this */ + } + } + if (update && file_info.used) + { + if (files_open >= max_files) + { + if (close_some_file(&tree)) + goto com_err; + files_open--; + } + if (!(file_info.isam= nisam_open(isam_file_name,O_RDWR, + HA_OPEN_WAIT_IF_LOCKED))) + goto com_err; + if (!(file_info.record=my_malloc(file_info.isam->s->base.reclength, + MYF(MY_WME)))) + goto end; + files_open++; + file_info.closed=0; + } + VOID(tree_insert(&tree,(gptr) &file_info,0)); + if (file_info.used) + { + if (verbose && !record_pos_file) + printf_log("%s: open",file_info.show_name); + com_count[command][0]++; + if (result) + com_count[command][1]++; + } + break; + case LOG_CLOSE: + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s -> %d",FILENAME(curr_file_info), + command_name[command],result); + if (curr_file_info) + { + if (!curr_file_info->closed) + files_open--; + file_info_free(curr_file_info); + VOID(tree_delete(&tree,(gptr) curr_file_info)); + } + break; + case LOG_EXTRA: + if (my_b_read(&cache,(byte*) head,sizeof(extra_command))) + goto err; + memcpy_fixed(&extra_command,head,sizeof(extra_command)); + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s(%d) -> %d",FILENAME(curr_file_info), + command_name[command], extra_command,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (nisam_extra(curr_file_info->isam,extra_command) != (int) result) + { + VOID(fprintf(stderr, + "Warning: error %d, expected %d on command %s at %lx\n", + my_errno,result,command_name[command],isamlog_filepos)); + } + } + break; + case LOG_DELETE: + if (my_b_read(&cache,(byte*) head,sizeof(filepos))) + goto err; + memcpy_fixed(&filepos,head,sizeof(filepos)); + if (verbose && (!record_pos_file || + ((record_pos == filepos || record_pos == NO_FILEPOS) && + !cmp_filename(curr_file_info,record_pos_file))) && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s at %ld -> %d",FILENAME(curr_file_info), + command_name[command],(long) filepos,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (nisam_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) + { + if (!recover) + goto com_err; + com_count[command][2]++; /* Mark error */ + } + ni_result=nisam_delete(curr_file_info->isam,curr_file_info->record); + if ((ni_result == 0 && result) || + (ni_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (ni_result) + com_count[command][2]++; /* Mark error */ + } + } + break; + case LOG_WRITE: + case LOG_UPDATE: + if (my_b_read(&cache,(byte*) head,8)) + goto err; + filepos=uint4korr(head); + length=uint4korr(head+4); + buff=0; + if (read_string(&cache,&buff,(uint) length)) + goto err; + if ((!record_pos_file || + ((record_pos == filepos || record_pos == NO_FILEPOS) && + !cmp_filename(curr_file_info,record_pos_file))) && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + { + if (write_file && + (my_fwrite(write_file,buff,length,MYF(MY_WAIT_IF_FULL | MY_NABP)))) + goto end; + if (verbose) + printf_log("%s: %s at %ld, length=%ld -> %d", + FILENAME(curr_file_info), + command_name[command], filepos,length,result); + } + if (update && curr_file_info && !curr_file_info->closed) + { + if (curr_file_info->isam->s->base.blobs) + fix_blob_pointers(curr_file_info->isam,buff); + if ((enum nisam_log_commands) command == LOG_UPDATE) + { + if (nisam_rrnd(curr_file_info->isam,curr_file_info->record,filepos)) + { + if (!recover) + { + result=0; + goto com_err; + } + if (recover == 1 || result || + find_record_with_key(curr_file_info,buff)) + { + com_count[command][2]++; /* Mark error */ + break; + } + } + ni_result=nisam_update(curr_file_info->isam,curr_file_info->record, + buff); + if ((ni_result == 0 && result) || + (ni_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (ni_result) + com_count[command][2]++; /* Mark error */ + } + } + else + { + ni_result=nisam_write(curr_file_info->isam,buff); + if ((ni_result == 0 && result) || + (ni_result && (uint) my_errno != result)) + { + if (!recover) + goto com_err; + if (ni_result) + com_count[command][2]++; /* Mark error */ + } + if (! recover && filepos != curr_file_info->isam->lastpos) + { + printf("Warning: Wrote at position: %ld, should have been %ld", + curr_file_info->isam->lastpos,(long) filepos); + goto com_err; + } + } + } + my_free(buff,MYF(0)); + break; + case LOG_LOCK: + if (my_b_read(&cache,(byte*) head,sizeof(lock_command))) + goto err; + memcpy_fixed(&lock_command,head,sizeof(lock_command)); + if (verbose && !record_pos_file && + (!table_names[0] || (curr_file_info && curr_file_info->used))) + printf_log("%s: %s(%d) -> %d\n",FILENAME(curr_file_info), + command_name[command],lock_command,result); + if (update && curr_file_info && !curr_file_info->closed) + { + if (nisam_lock_database(curr_file_info->isam,lock_command) != + (int) result) + goto com_err; + } + break; + default: + VOID(fprintf(stderr, + "Error: found unknown command %d in logfile, aborted\n", + command)); + goto end; + } + } + end_key_cache(); + delete_tree(&tree); + VOID(end_io_cache(&cache)); + VOID(my_close(file,MYF(0))); + if (write_file && my_fclose(write_file,MYF(MY_WME))) + DBUG_RETURN(1); + DBUG_RETURN(0); + + err: + VOID(fprintf(stderr,"Got error %d when reading from logfile\n",my_errno)); + goto end; + com_err: + VOID(fprintf(stderr,"Got error %d, expected %d on command %s at %lx\n", + my_errno,result,command_name[command],isamlog_filepos)); + end: + end_key_cache(); + delete_tree(&tree); + VOID(end_io_cache(&cache)); + VOID(my_close(file,MYF(0))); + if (write_file) + VOID(my_fclose(write_file,MYF(MY_WME))); + DBUG_RETURN(1); +} + + +static int read_string(file,to,length) +IO_CACHE *file; +reg1 gptr *to; +reg2 uint length; +{ + DBUG_ENTER("read_string"); + + if (*to) + my_free((gptr) *to,MYF(0)); + if (!(*to= (gptr) my_malloc(length+1,MYF(MY_WME))) || + my_b_read(file,(byte*) *to,length)) + { + if (*to) + my_free(*to,MYF(0)); + *to= 0; + DBUG_RETURN(1); + } + *((char*) *to+length)= '\0'; + DBUG_RETURN (0); +} /* read_string */ + + +static int file_info_compare(a,b) +void *a; +void *b; +{ + long lint; + + if ((lint=((struct file_info*) a)->process - + ((struct file_info*) b)->process)) + return lint < 0L ? -1 : 1; + return ((struct file_info*) a)->filenr - ((struct file_info*) b)->filenr; +} + + /* ARGSUSED */ + +static int test_if_open (key,count,param) +struct file_info *key; +element_count count __attribute__((unused)); +struct test_if_open_param *param; +{ + if (!strcmp(key->name,param->name) && key->id > param->max_id) + param->max_id=key->id; + return 0; +} + + +static void fix_blob_pointers(info,record) +N_INFO *info; +byte *record; +{ + byte *pos; + N_BLOB *blob,*end; + + pos=record+info->s->base.reclength; + for (end=info->blobs+info->s->base.blobs, blob= info->blobs; + blob != end ; + blob++) + { + bmove(record+blob->offset+blob->pack_length,&pos,sizeof(char*)); + pos+=_calc_blob_length(blob->pack_length,record+blob->offset); + } +} + +static uint set_maximum_open_files(maximum_files) +uint maximum_files; +{ +#if defined(HAVE_GETRUSAGE) && defined(RLIMIT_NOFILE) + struct rlimit rlimit; + int old_max; + + if (maximum_files > MY_NFILE) + maximum_files=MY_NFILE; /* Don't crash my_open */ + + if (!getrlimit(RLIMIT_NOFILE,&rlimit)) + { + old_max=rlimit.rlim_max; + if (maximum_files && (int) maximum_files > old_max) + rlimit.rlim_max=maximum_files; + rlimit.rlim_cur=rlimit.rlim_max; + if (setrlimit(RLIMIT_NOFILE,&rlimit)) + { + if (old_max != (int) maximum_files) + { /* Set as much as we can */ + rlimit.rlim_max=rlimit.rlim_cur=old_max; + setrlimit(RLIMIT_NOFILE,&rlimit); + } + } + getrlimit(RLIMIT_NOFILE,&rlimit); /* Read if broken setrlimit */ + if (maximum_files && maximum_files < rlimit.rlim_cur) + VOID(fprintf(stderr,"Warning: Error from setrlimit: Max open files is %d\n",old_max)); + return rlimit.rlim_cur; + } +#endif + return min(maximum_files,MY_NFILE); +} + + /* close the file with hasn't been accessed for the longest time */ + /* ARGSUSED */ + +static int test_when_accessed (key,count,access_param) +struct file_info *key; +element_count count __attribute__((unused)); +struct st_access_param *access_param; +{ + if (key->accessed < access_param->min_accessed && ! key->closed) + { + access_param->min_accessed=key->accessed; + access_param->found=key; + } + return 0; +} + + +static void file_info_free(fileinfo) +struct file_info *fileinfo; +{ + if (update) + { + if (!fileinfo->closed) + VOID(nisam_close(fileinfo->isam)); + if (fileinfo->record) + my_free(fileinfo->record,MYF(0)); + } + my_free(fileinfo->name,MYF(0)); + my_free(fileinfo->show_name,MYF(0)); +} + + + +static int close_some_file(tree) +TREE *tree; +{ + struct st_access_param access_param; + + access_param.min_accessed=LONG_MAX; + access_param.found=0; + + VOID(tree_walk(tree,(tree_walk_action) test_when_accessed, + (void*) &access_param,left_root_right)); + if (!access_param.found) + return 1; /* No open file that is possibly to close */ + if (nisam_close(access_param.found->isam)) + return 1; + access_param.found->closed=1; + return 0; +} + + +static int reopen_closed_file(tree,fileinfo) +TREE *tree; +struct file_info *fileinfo; +{ + char name[FN_REFLEN]; + if (close_some_file(tree)) + return 1; /* No file to close */ + strmov(name,fileinfo->show_name); + if (fileinfo->id > 1) + *strrchr(name,'<')='\0'; /* Remove "<id>" */ + + if (!(fileinfo->isam= nisam_open(name,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) + return 1; + fileinfo->closed=0; + re_open_count++; + return 0; +} + + /* Try to find record with uniq key */ + +static int find_record_with_key(file_info,record) +struct file_info *file_info; +byte *record; +{ + uint key; + N_INFO *info=file_info->isam; + uchar tmp_key[N_MAX_KEY_BUFF]; + + for (key=0 ; key < info->s->state.keys ; key++) + { + if (info->s->keyinfo[key].base.flag & HA_NOSAME) + { + VOID(_nisam_make_key(info,key,tmp_key,record,0L)); + return nisam_rkey(info,file_info->record,(int) key,(char*) tmp_key,0, + HA_READ_KEY_EXACT); + } + } + return 1; +} + + +static void printf_log(const char *format,...) +{ + va_list args; + va_start(args,format); + if (verbose > 2) + printf("%9ld:",isamlog_filepos); + if (verbose > 1) + printf("%5ld ",isamlog_process); /* Write process number */ + (void) vprintf((char*) format,args); + putchar('\n'); + va_end(args); +} + + +static bool cmp_filename(file_info,name) +struct file_info *file_info; +my_string name; +{ + if (!file_info) + return 1; + return strcmp(file_info->name,name) ? 1 : 0; +} diff --git a/isam/log.c b/isam/log.c new file mode 100644 index 00000000000..a95b53b5110 --- /dev/null +++ b/isam/log.c @@ -0,0 +1,156 @@ +/* Copyright (C) 2000 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 */ + +/* Logging of isamcommands and records on logfile */ + +#include "isamdef.h" +#if defined(MSDOS) || defined(__WIN__) +#include <errno.h> +#include <fcntl.h> +#ifndef __WIN__ +#include <process.h> +#endif +#endif +#ifdef VMS +#include <processes.h> +#endif + +#ifdef THREAD +#undef GETPID +#define GETPID() (log_type == 1 ? getpid() : (long) my_thread_id()); +#else +#define GETPID() getpid() +#endif + + /* Activate logging if flag is 1 and reset logging if flag is 0 */ + +static int log_type=0; + +int nisam_log(int activate_log) +{ + int error=0; + char buff[FN_REFLEN]; + DBUG_ENTER("nisam_log"); + + log_type=activate_log; + if (activate_log) + { + if (nisam_log_file < 0) + { + if ((nisam_log_file = my_create(fn_format(buff,nisam_log_filename, + "",".log",4), + 0,(O_RDWR | O_BINARY | O_APPEND),MYF(0))) + < 0) + DBUG_RETURN(1); + } + } + else if (nisam_log_file >= 0) + { + error=my_close(nisam_log_file,MYF(0)); + nisam_log_file= -1; + } + DBUG_RETURN(error); +} + + + /* Logging of records and commands on logfile */ + /* All logs starts with command(1) dfile(2) process(4) result(2) */ + +void _nisam_log(enum nisam_log_commands command, N_INFO *info, const byte *buffert, uint length) +{ + char buff[11]; + int error,old_errno; + ulong pid=(ulong) GETPID(); + old_errno=my_errno; + bzero(buff,sizeof(buff)); + buff[0]=(char) command; + int2store(buff+1,info->dfile); + int4store(buff+3,pid); + int2store(buff+9,length); + + pthread_mutex_lock(&THR_LOCK_isam); + error=my_lock(nisam_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(nisam_log_file,buff,sizeof(buff),MYF(0))); + VOID(my_write(nisam_log_file,buffert,length,MYF(0))); + if (!error) + error=my_lock(nisam_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_isam); + my_errno=old_errno; +} + + +void _nisam_log_command(enum nisam_log_commands command, N_INFO *info, const byte *buffert, uint length, int result) +{ + char buff[9]; + int error,old_errno; + ulong pid=(ulong) GETPID(); + + old_errno=my_errno; + buff[0]=(char) command; + int2store(buff+1,info->dfile); + int4store(buff+3,pid); + int2store(buff+7,result); + pthread_mutex_lock(&THR_LOCK_isam); + error=my_lock(nisam_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(nisam_log_file,buff,sizeof(buff),MYF(0))); + if (buffert) + VOID(my_write(nisam_log_file,buffert,length,MYF(0))); + if (!error) + error=my_lock(nisam_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_isam); + my_errno=old_errno; +} + + +void _nisam_log_record(enum nisam_log_commands command, N_INFO *info, const byte *record, ulong filepos, int result) +{ + char buff[17],*pos; + int error,old_errno; + uint length; + ulong pid=(ulong) GETPID(); + + old_errno=my_errno; + if (!info->s->base.blobs) + length=info->s->base.reclength; + else + length=info->s->base.reclength+ _calc_total_blob_length(info,record); + buff[0]=(char) command; + int2store(buff+1,info->dfile); + int4store(buff+3,pid); + int2store(buff+7,result); + int4store(buff+9,filepos); + int4store(buff+13,length); + pthread_mutex_lock(&THR_LOCK_isam); + error=my_lock(nisam_log_file,F_WRLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + VOID(my_write(nisam_log_file,buff,sizeof(buff),MYF(0))); + VOID(my_write(nisam_log_file,(byte*) record,info->s->base.reclength,MYF(0))); + if (info->s->base.blobs) + { + N_BLOB *blob,*end; + + for (end=info->blobs+info->s->base.blobs, blob= info->blobs; + blob != end ; + blob++) + { + bmove(&pos,record+blob->offset+blob->pack_length,sizeof(char*)); + VOID(my_write(nisam_log_file,pos,blob->length,MYF(0))); + } + } + if (!error) + error=my_lock(nisam_log_file,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)); + pthread_mutex_unlock(&THR_LOCK_isam); + my_errno=old_errno; +} diff --git a/isam/make-ccc b/isam/make-ccc new file mode 100755 index 00000000000..d9a95dbc14b --- /dev/null +++ b/isam/make-ccc @@ -0,0 +1,3 @@ +ccc -DHAVE_CONFIG_H -I. -I. -I.. -I./../include -I../include -DDBUG_OFF -fast -O3 -fomit-frame-pointer -c _cache.c _dbug.c _dynrec.c _key.c _locking.c _packrec.c _page.c _search.c _statrec.c changed.c close.c create.c delete.c extra.c info.c log.c open.c panic.c range.c rfirst.c rkey.c rlast.c rnext.c rprev.c rrnd.c rsame.c rsamepos.c static.c update.c write.c +rm libnisam.a +ar -cr libnisam.a _cache.o diff --git a/isam/open.c b/isam/open.c new file mode 100644 index 00000000000..8a6e0eb5814 --- /dev/null +++ b/isam/open.c @@ -0,0 +1,455 @@ +/* Copyright (C) 2000 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 */ + +/* open a isam-database */ + +#include "isamdef.h" +#if defined(MSDOS) || defined(__WIN__) +#ifdef __WIN__ +#include <fcntl.h> +#else +#include <process.h> /* Prototype for getpid */ +#endif +#endif +#ifdef VMS +#include "static.c" +#endif + +static void setup_functions(ISAM_SHARE *info); +static void setup_key_functions(N_KEYDEF *keyinfo); + +#define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); \ + pos+=size;} + +/****************************************************************************** +** Return the shared struct if the table is already open. +** In MySQL the server will handle version issues. +******************************************************************************/ + +static N_INFO *test_if_reopen(char *filename) +{ + LIST *pos; + + for (pos=nisam_open_list ; pos ; pos=pos->next) + { + N_INFO *info=(N_INFO*) pos->data; + ISAM_SHARE *share=info->s; + if (!strcmp(share->filename,filename) && share->last_version) + return info; + } + return 0; +} + + +/****************************************************************************** + open a isam database. + By default exit with error if database is locked + if handle_locking & HA_OPEN_WAIT_IF_LOCKED then wait if database is locked + if handle_locking & HA_OPEN_IGNORE_IF_LOCKED then continue, but count-vars + in st_i_info may be wrong. count-vars are automaticly fixed after next + isam request. +******************************************************************************/ + + +N_INFO *nisam_open(const char *name, int mode, uint handle_locking) +{ + int lock_error,kfile,open_mode,save_errno; + uint i,j,len,errpos,head_length,base_pos,offset,info_length,extra; + char name_buff[FN_REFLEN],*disk_cache,*disk_pos; + N_INFO info,*m_info,*old_info; + ISAM_SHARE share_buff,*share; + DBUG_ENTER("nisam_open"); + + LINT_INIT(m_info); + kfile= -1; + lock_error=1; + errpos=0; + head_length=sizeof(share_buff.state.header); + bzero((byte*) &info,sizeof(info)); + + VOID(fn_format(name_buff,name,"",N_NAME_IEXT,4+16+32)); + pthread_mutex_lock(&THR_LOCK_isam); + if (!(old_info=test_if_reopen(name_buff))) + { + share= &share_buff; + bzero((gptr) &share_buff,sizeof(share_buff)); + + if ((kfile=my_open(name_buff,(open_mode=O_RDWR) | O_SHARE,MYF(0))) < 0) + { + if ((errno != EROFS && errno != EACCES) || + mode != O_RDONLY || + (kfile=my_open(name_buff,(open_mode=O_RDONLY) | O_SHARE,MYF(0))) < 0) + goto err; + } + errpos=1; + if (my_read(kfile,(char*) share->state.header.file_version,head_length, + MYF(MY_NABP))) + goto err; + + if (memcmp((byte*) share->state.header.file_version, + (byte*) nisam_file_magic, 3) || + share->state.header.file_version[3] == 0 || + (uchar) share->state.header.file_version[3] > 3) + { + DBUG_PRINT("error",("Wrong header in %s",name_buff)); + DBUG_DUMP("error_dump",(char*) share->state.header.file_version, + head_length); + my_errno=HA_ERR_CRASHED; + goto err; + } + if (uint2korr(share->state.header.options) & + ~(HA_OPTION_PACK_RECORD | HA_OPTION_PACK_KEYS | + HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA | + HA_OPTION_TEMP_COMPRESS_RECORD)) + { + DBUG_PRINT("error",("wrong options: 0x%lx", + uint2korr(share->state.header.options))); + my_errno=HA_ERR_OLD_FILE; + goto err; + } + info_length=uint2korr(share->state.header.header_length); + base_pos=uint2korr(share->state.header.base_pos); + if (!(disk_cache=(char*) my_alloca(info_length))) + { + my_errno=ENOMEM; + goto err; + } + errpos=2; + + VOID(my_seek(kfile,0L,MY_SEEK_SET,MYF(0))); +#ifndef NO_LOCKING + if (!(handle_locking & HA_OPEN_TMP_TABLE)) + { + if ((lock_error=my_lock(kfile,F_RDLCK,0L,F_TO_EOF, + MYF(handle_locking & HA_OPEN_WAIT_IF_LOCKED ? + 0 : MY_DONT_WAIT))) && + !(handle_locking & HA_OPEN_IGNORE_IF_LOCKED)) + goto err; + } +#endif + errpos=3; + if (my_read(kfile,disk_cache,info_length,MYF(MY_NABP))) + goto err; + len=uint2korr(share->state.header.state_info_length); + if (len != sizeof(N_STATE_INFO)) + { + DBUG_PRINT("warning", + ("saved_state_info_length: %d base_info_length: %d", + len,sizeof(N_STATE_INFO))); + } + if (len > sizeof(N_STATE_INFO)) + len=sizeof(N_STATE_INFO); + share->state_length=len; + memcpy(&share->state.header.file_version[0],disk_cache,(size_t) len); + len=uint2korr(share->state.header.base_info_length); + if (len != sizeof(N_BASE_INFO)) + { + DBUG_PRINT("warning",("saved_base_info_length: %d base_info_length: %d", + len,sizeof(N_BASE_INFO))); + if (len <= offsetof(N_BASE_INFO,sortkey)) + share->base.sortkey=(ushort) ~0; + } + memcpy((char*) (byte*) &share->base,disk_cache+base_pos, + (size_t) min(len,sizeof(N_BASE_INFO))); + disk_pos=disk_cache+base_pos+len; + share->base.options=uint2korr(share->state.header.options); + if (share->base.max_key_length > N_MAX_KEY_BUFF) + { + my_errno=HA_ERR_UNSUPPORTED; + goto err; + } + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + share->base.max_key_length+=2; /* For safety */ + + if (!my_multi_malloc(MY_WME, + &share,sizeof(*share), + &share->keyinfo,share->base.keys*sizeof(N_KEYDEF), + &share->rec,(share->base.fields+1)*sizeof(N_RECINFO), + &share->blobs,sizeof(N_BLOB)*share->base.blobs, + &share->filename,strlen(name_buff)+1, + NullS)) + goto err; + errpos=4; + *share=share_buff; + strmov(share->filename,name_buff); + + /* Fix key in used if old nisam-database */ + if (share->state_length <= offsetof(N_STATE_INFO,keys)) + share->state.keys=share->base.keys; + + share->blocksize=min(IO_SIZE,nisam_block_size); + for (i=0 ; i < share->base.keys ; i++) + { + get_next_element(&share->keyinfo[i].base,disk_pos,sizeof(N_SAVE_KEYDEF)); + setup_key_functions(share->keyinfo+i); + set_if_smaller(share->blocksize,share->keyinfo[i].base.block_length); + for (j=0 ; j <= share->keyinfo[i].base.keysegs ; j++) + { + get_next_element(&share->keyinfo[i].seg[j],disk_pos, + sizeof(N_SAVE_KEYSEG)); + } + } + if (!share->blocksize) + { + my_errno=HA_ERR_CRASHED; + goto err; + } + + for (i=j=offset=0 ; i < share->base.fields ; i++) + { + get_next_element(&share->rec[i].base,disk_pos,sizeof(N_SAVE_RECINFO)); +#ifndef NOT_PACKED_DATABASES + share->rec[i].pack_type=0; + share->rec[i].huff_tree=0; +#endif + if (share->rec[i].base.type == (int) FIELD_BLOB) + { + share->blobs[j].pack_length=share->rec[i].base.length; + share->blobs[j].offset=offset; + j++; + offset+=sizeof(char*); + } + offset+=share->rec[i].base.length; + } + share->rec[i].base.type=(int) FIELD_LAST; + +#ifndef NO_LOCKING + if (! lock_error) + { + VOID(my_lock(kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE))); + lock_error=1; /* Database unlocked */ + } +#endif + + if ((info.dfile=my_open(fn_format(name_buff,name,"",N_NAME_DEXT,2+4), + mode | O_SHARE, + MYF(MY_WME))) < 0) + goto err; + errpos=5; + + share->kfile=kfile; + share->mode=open_mode; + share->this_process=(ulong) getpid(); + share->rnd= (int) share->this_process; /* rnd-counter for splitts */ +#ifndef DBUG_OFF + share->rnd=0; /* To make things repeatable */ +#endif + share->last_process= share->state.process; + if (!(share->last_version=share->state.version)) + share->last_version=1; /* Safety */ + share->rec_reflength=share->base.rec_reflength; /* May be changed */ + + share->data_file_type=STATIC_RECORD; + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + { + share->data_file_type = COMPRESSED_RECORD; + share->base.options|= HA_OPTION_READ_ONLY_DATA; + info.s=share; + if (_nisam_read_pack_info(&info, + (pbool) test(!(share->base.options & + (HA_OPTION_PACK_RECORD | + HA_OPTION_TEMP_COMPRESS_RECORD))))) + goto err; + } + else if (share->base.options & HA_OPTION_PACK_RECORD) + share->data_file_type = DYNAMIC_RECORD; + my_afree((gptr) disk_cache); + setup_functions(share); +#ifdef THREAD + thr_lock_init(&share->lock); + VOID(pthread_mutex_init(&share->intern_lock,NULL)); +#endif + } + else + { + share= old_info->s; + if (mode == O_RDWR && share->mode == O_RDONLY) + { + my_errno=EACCES; /* Can't open in write mode*/ + goto err; + } + if ((info.dfile=my_open(fn_format(name_buff,old_info->filename,"", + N_NAME_DEXT,2+4), + mode | O_SHARE,MYF(MY_WME))) < 0) + { + my_errno=errno; + goto err; + } + errpos=5; + } + + /* alloc and set up private structure parts */ + if (!my_multi_malloc(MY_WME, + &m_info,sizeof(N_INFO), + &info.blobs,sizeof(N_BLOB)*share->base.blobs, + &info.buff,(share->base.max_block*2+ + share->base.max_key_length), + &info.lastkey,share->base.max_key_length*3+1, + &info.filename,strlen(name)+1, + NullS)) + goto err; + errpos=6; + strmov(info.filename,name); + memcpy(info.blobs,share->blobs,sizeof(N_BLOB)*share->base.blobs); + + info.s=share; + info.lastpos= NI_POS_ERROR; + info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND); + info.opt_flag=READ_CHECK_USED; + info.alloced_rec_buff_length=share->base.pack_reclength; + info.this_uniq= (ulong) info.dfile; /* Uniq number in process */ + info.this_loop=0; /* Update counter */ + info.last_uniq= share->state.uniq; + info.last_loop= share->state.loop; + info.options=share->base.options | + (mode == O_RDONLY ? HA_OPTION_READ_ONLY_DATA : 0); + info.lock_type=F_UNLCK; + info.errkey= -1; + pthread_mutex_lock(&share->intern_lock); + info.read_record=share->read_record; + share->reopen++; + if (share->base.options & HA_OPTION_READ_ONLY_DATA) + { + info.lock_type=F_RDLCK; + share->r_locks++; + info.this_uniq=share->state.uniq; /* Row checksum */ + } +#ifndef NO_LOCKING + if (handle_locking & HA_OPEN_TMP_TABLE) +#endif + { + share->w_locks++; /* We don't have to update status */ + info.lock_type=F_WRLCK; + } + pthread_mutex_unlock(&share->intern_lock); + + /* Allocate buffer for one record */ + + extra=0; + if (share->base.options & HA_OPTION_PACK_RECORD) + extra=ALIGN_SIZE(MAX_DYN_BLOCK_HEADER)+N_SPLITT_LENGTH+ + DYN_DELETE_BLOCK_HEADER; + if (!(info.rec_alloc=(byte*) my_malloc(share->base.pack_reclength+extra+ + 6, + MYF(MY_WME | MY_ZEROFILL)))) + goto err; + if (extra) + info.rec_buff=info.rec_alloc+ALIGN_SIZE(MAX_DYN_BLOCK_HEADER); + else + info.rec_buff=info.rec_alloc; + + *m_info=info; +#ifdef THREAD + thr_lock_data_init(&share->lock,&m_info->lock,NULL); +#endif + + m_info->open_list.data=(void*) m_info; + nisam_open_list=list_add(nisam_open_list,&m_info->open_list); + + pthread_mutex_unlock(&THR_LOCK_isam); + nisam_log_simple(LOG_OPEN,m_info,share->filename, + (uint) strlen(share->filename)); + DBUG_RETURN(m_info); + +err: + save_errno=my_errno ? my_errno : HA_ERR_END_OF_FILE; + switch (errpos) { + case 6: + my_free((gptr) m_info,MYF(0)); + /* fall through */ + case 5: + VOID(my_close(info.dfile,MYF(0))); + if (old_info) + break; /* Don't remove open table */ + /* fall through */ + case 4: + my_free((gptr) share,MYF(0)); + /* fall through */ + case 3: +#ifndef NO_LOCKING + if (! lock_error) + VOID(my_lock(kfile, F_UNLCK, 0L, F_TO_EOF, MYF(MY_SEEK_NOT_DONE))); +#endif + /* fall through */ + case 2: + my_afree((gptr) disk_cache); + /* fall through */ + case 1: + VOID(my_close(kfile,MYF(0))); + /* fall through */ + case 0: + default: + break; + } + pthread_mutex_unlock(&THR_LOCK_isam); + my_errno=save_errno; + DBUG_RETURN (NULL); +} /* nisam_open */ + + + /* Set up functions in structs */ + +static void setup_functions(register ISAM_SHARE *share) +{ + if (share->base.options & HA_OPTION_COMPRESS_RECORD) + { + share->read_record=_nisam_read_pack_record; + share->read_rnd=_nisam_read_rnd_pack_record; + } + else if (share->base.options & HA_OPTION_PACK_RECORD) + { + share->read_record=_nisam_read_dynamic_record; + share->read_rnd=_nisam_read_rnd_dynamic_record; + share->delete_record=_nisam_delete_dynamic_record; + share->compare_record=_nisam_cmp_dynamic_record; + if (share->base.blobs) + { + share->update_record=_nisam_update_blob_record; + share->write_record=_nisam_write_blob_record; + } + else + { + share->write_record=_nisam_write_dynamic_record; + share->update_record=_nisam_update_dynamic_record; + } + } + else + { + share->read_record=_nisam_read_static_record; + share->read_rnd=_nisam_read_rnd_static_record; + share->delete_record=_nisam_delete_static_record; + share->compare_record=_nisam_cmp_static_record; + share->update_record=_nisam_update_static_record; + share->write_record=_nisam_write_static_record; + } + return; +} + + +static void setup_key_functions(register N_KEYDEF *keyinfo) +{ + if (keyinfo->base.flag & (HA_PACK_KEY | HA_SPACE_PACK_USED)) + { + keyinfo->bin_search=_nisam_seq_search; + keyinfo->get_key=_nisam_get_key; + } + else + { + keyinfo->bin_search=_nisam_bin_search; + keyinfo->get_key=_nisam_get_static_key; + } + return; +} diff --git a/isam/pack_isam.c b/isam/pack_isam.c new file mode 100644 index 00000000000..9193f52a863 --- /dev/null +++ b/isam/pack_isam.c @@ -0,0 +1,2051 @@ +/* Copyright (C) 1979-1999 TcX AB & Monty Program KB & Detron HB + + This software is distributed with NO WARRANTY OF ANY KIND. No author or + distributor accepts any responsibility for the consequences of using it, or + for whether it serves any particular purpose or works at all, unless he or + she says so in writing. Refer to the Free Public License (the "License") + for full details. + Every copy of this file must include a copy of the License, normally in a + plain ASCII text file named PUBLIC. The License grants you the right to + copy, modify and redistribute this file, but only under certain conditions + described in the License. Among other things, the License requires that + the copyright notice and this notice be preserved on all copies. */ + +/* Pack isam file*/ + +#ifndef USE_MY_FUNC +#define USE_MY_FUNC /* We nead at least my_malloc */ +#endif + +#include "isamdef.h" +#include <queues.h> +#include <my_tree.h> +#include "mysys_err.h" +#ifdef MSDOS +#include <io.h> +#endif +#ifndef __GNU_LIBRARY__ +#define __GNU_LIBRARY__ /* Skipp warnings in getopt.h */ +#endif +#include <getopt.h> + +#if INT_MAX > 32767 +#define BITS_SAVED 32 +#else +#define BITS_SAVED 16 +#endif + +#define IS_OFFSET ((uint) 32768) /* Bit if offset or char in tree */ +#define HEAD_LENGTH 32 +#define ALLOWED_JOIN_DIFF 256 /* Diff allowed to join trees */ + +#define DATA_TMP_EXT ".TMD" +#define OLD_EXT ".OLD" +#define WRITE_COUNT MY_HOW_OFTEN_TO_WRITE + +#ifdef __WIN__ +static double ulonglong2double(ulonglong value) +{ + longlong nr=(longlong) value; + if (nr >= 0) + return (double) nr; + return (18446744073709551616.0 + (double) nr); +} + +#if SIZEOF_OFF_T > 4 +#define my_off_t2double(A) ulonglong2double(A) +#else +#define my_off_t2double(A) ((double) (A)) +#endif /* SIZEOF_OFF_T > 4 */ +#endif + +struct st_file_buffer { + File file; + char *buffer,*pos,*end; + my_off_t pos_in_file; + int bits; + uint byte; +}; + +struct st_huff_tree; +struct st_huff_element; + +typedef struct st_huff_counts { + uint field_length,max_zero_fill; + uint pack_type; + uint max_end_space,max_pre_space,length_bits,min_space; + enum en_fieldtype field_type; + struct st_huff_tree *tree; /* Tree for field */ + my_off_t counts[256]; + my_off_t end_space[8]; + my_off_t pre_space[8]; + my_off_t tot_end_space,tot_pre_space,zero_fields,empty_fields,bytes_packed; + TREE int_tree; + byte *tree_buff; + byte *tree_pos; +} HUFF_COUNTS; + +typedef struct st_huff_element HUFF_ELEMENT; + +struct st_huff_element { + my_off_t count; + union un_element { + struct st_nod { + HUFF_ELEMENT *left,*right; + } nod; + struct st_leaf { + HUFF_ELEMENT *null; + uint element_nr; /* Number of element */ + } leaf; + } a; +}; + + +typedef struct st_huff_tree { + HUFF_ELEMENT *root,*element_buffer; + HUFF_COUNTS *counts; + uint tree_number; + uint elements; + my_off_t bytes_packed; + uint tree_pack_length; + uint min_chr,max_chr,char_bits,offset_bits,max_offset,height; + ulong *code; + uchar *code_len; +} HUFF_TREE; + + +typedef struct st_isam_mrg { + N_INFO **file,**current,**end; + uint count; + uint min_pack_length; /* Theese is used by packed data */ + uint max_pack_length; + uint ref_length; + my_off_t records; +} MRG_INFO; + + +extern int main(int argc,char * *argv); +static void get_options(int *argc,char ***argv); +static N_INFO *open_isam_file(char *name,int mode); +static bool open_isam_files(MRG_INFO *mrg,char **names,uint count); +static int compress(MRG_INFO *file,char *join_name); +static HUFF_COUNTS *init_huff_count(N_INFO *info,my_off_t records); +static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, + uint trees, + HUFF_COUNTS *huff_counts, + uint fields); +static int compare_tree(const uchar *s,const uchar *t); +static int get_statistic(MRG_INFO *mrg,HUFF_COUNTS *huff_counts); +static void check_counts(HUFF_COUNTS *huff_counts,uint trees, + my_off_t records); +static int test_space_compress(HUFF_COUNTS *huff_counts,my_off_t records, + uint max_space_length,my_off_t *space_counts, + my_off_t tot_space_count, + enum en_fieldtype field_type); +static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts,uint trees); +static int make_huff_tree(HUFF_TREE *tree,HUFF_COUNTS *huff_counts); +static int compare_huff_elements(void *not_used, byte *a,byte *b); +static int save_counts_in_queue(byte *key,element_count count, + HUFF_TREE *tree); +static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts,uint flag); +static uint join_same_trees(HUFF_COUNTS *huff_counts,uint trees); +static int make_huff_decode_table(HUFF_TREE *huff_tree,uint trees); +static void make_traverse_code_tree(HUFF_TREE *huff_tree, + HUFF_ELEMENT *element,uint size, + ulong code); +static int write_header(MRG_INFO *isam_file, uint header_length,uint trees, + my_off_t tot_elements,my_off_t filelength); +static void write_field_info(HUFF_COUNTS *counts, uint fields,uint trees); +static my_off_t write_huff_tree(HUFF_TREE *huff_tree,uint trees); +static uint *make_offset_code_tree(HUFF_TREE *huff_tree, + HUFF_ELEMENT *element, + uint *offset); +static uint max_bit(uint value); +static int compress_isam_file(MRG_INFO *file,HUFF_COUNTS *huff_counts); +static char *make_new_name(char *new_name,char *old_name); +static char *make_old_name(char *new_name,char *old_name); +static void init_file_buffer(File file,pbool read_buffer); +static int flush_buffer(uint neaded_length); +static void end_file_buffer(void); +static void write_bits(ulong value,uint bits); +static void flush_bits(void); +static void save_integer(byte *pos,uint pack_length,my_off_t value); +static void save_state(N_INFO *isam_file,MRG_INFO *mrg,my_off_t new_length, + ulong crc); +static int save_state_mrg(File file,MRG_INFO *isam_file,my_off_t new_length, + ulong crc); +static int mrg_close(MRG_INFO *mrg); +static int mrg_rrnd(MRG_INFO *info,byte *buf); +static void mrg_reset(MRG_INFO *mrg); + + +static int backup=0,error_on_write=0,test_only=0,verbose=0,silent=0, + write_loop=0,force_pack=0,opt_wait=0,isamchk_neaded=0; +static int tmpfile_createflag=O_RDWR | O_TRUNC | O_EXCL; +static uint tree_buff_length=8196-MALLOC_OVERHEAD,force_pack_ref_length; +static char tmp_dir[FN_REFLEN]={0},*join_table; +static my_off_t intervall_length; +static ulong crc; +static struct st_file_buffer file_buffer; +static QUEUE queue; +static HUFF_COUNTS *global_count; +static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}; +static const char *load_default_groups[]= { "pack_isam",0 }; + + /* The main program */ + +int main(int argc, char **argv) +{ + int error,ok; + MRG_INFO merge; + MY_INIT(argv[0]); + + load_defaults("my",load_default_groups,&argc,&argv); + get_options(&argc,&argv); + + error=ok=isamchk_neaded=0; + if (join_table) + { /* Join files into one */ + if (open_isam_files(&merge,argv,(uint) argc) || + compress(&merge,join_table)) + error=1; + } + else while (argc--) + { + N_INFO *isam_file; + if (!(isam_file=open_isam_file(*argv++,O_RDWR))) + error=1; + else + { + merge.file= &isam_file; + merge.current=0; + merge.count=1; + if (compress(&merge,0)) + error=1; + else + ok=1; + } + } + if (ok && isamchk_neaded && !silent) + puts("Remember to run isamchk -rq on compressed databases"); + VOID(fflush(stdout)); VOID(fflush(stderr)); + my_end(verbose ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR); + exit(error ? 2 : 0); +#ifndef _lint + return 0; /* No compiler warning */ +#endif +} + + +static struct option long_options[] = +{ + {"backup", no_argument, 0, 'b'}, + {"debug", optional_argument, 0, '#'}, + {"force", no_argument, 0, 'f'}, + {"join", required_argument, 0, 'j'}, + {"help", no_argument, 0, '?'}, + {"packlength",required_argument, 0, 'p'}, + {"silent", no_argument, 0, 's'}, + {"tmpdir", required_argument, 0, 'T'}, + {"test", no_argument, 0, 't'}, + {"verbose", no_argument, 0, 'v'}, + {"version", no_argument, 0, 'V'}, + {"wait", no_argument, 0, 'w'}, + {0, 0, 0, 0} +}; + +static void print_version(void) +{ + printf("%s Ver 5.8 for %s on %s\n",my_progname,SYSTEM_TYPE,MACHINE_TYPE); +} + +static void usage(void) +{ + print_version(); + puts("Copyright (C) 1994-2000 TcX AB & Monty Program KB & Detron HB."); + puts("This is not free software. You must have a licence to use this program"); + puts("This software comes with ABSOLUTELY NO WARRANTY\n"); + puts("Pack a ISAM-table to take much smaller space"); + puts("Keys are not updated, so you must run isamchk -rq on any table"); + puts("that has keys after you have compressed it"); + puts("You should give the .ISM file as the filename argument"); + + printf("\nUsage: %s [OPTIONS] filename...\n", my_progname); + puts("\n\ + -b, --backup Make a backup of the table as table_name.OLD\n\ + -f, --force Force packing of table even if it's gets bigger or\n\ + tempfile exists.\n\ + -j, --join='new_table_name'\n\ + Join all given tables into 'new_table_name'.\n\ + All tables MUST have the identical layout.\n\ + -p, --packlength=# Force storage size of recordlength (1,2 or 3)\n\ + -s, --silent Be more silent.\n\ + -t, --test Don't pack table, only test packing it\n\ + -v, --verbose Write info about progress and packing result\n\ + -w, --wait Wait and retry if table is in use\n\ + -T, --tmpdir=# Use temporary directory to store temporary table\n\ + -#, --debug=... output debug log. Often this is 'd:t:o,filename`\n\ + -?, --help display this help and exit\n\ + -V, --version output version information and exit\n"); + print_defaults("my",load_default_groups); +}; + + /* reads options */ + /* Initiates DEBUG - but no debugging here ! */ + +static void get_options(int *argc,char ***argv) +{ + int c,option_index=0; + uint length; + + my_progname= argv[0][0]; + if (isatty(fileno(stdout))) + write_loop=1; + + while ((c=getopt_long(*argc,*argv,"bfj:p:stvwT:#::?V",long_options, + &option_index)) != EOF) + { + switch(c) { + case 'b': + backup=1; + break; + case 'f': + force_pack=1; + tmpfile_createflag=O_RDWR | O_TRUNC; + break; + case 'j': + join_table=optarg; + break; + case 'p': + force_pack_ref_length=(uint) atoi(optarg); + if (force_pack_ref_length > 3) + force_pack_ref_length=0; + break; + case 's': + write_loop=verbose=0; silent=1; + break; + case 't': + test_only=verbose=1; + break; + case 'T': + length=(uint) (strmov(tmp_dir,optarg)-tmp_dir); + if (length != dirname_length(tmp_dir)) + { + tmp_dir[length]=FN_LIBCHAR; + tmp_dir[length+1]=0; + } + break; + case 'v': + verbose=1; silent=0; + break; + case 'w': + opt_wait=1; + break; + case '#': + DBUG_PUSH(optarg ? optarg : "d:t:o"); + break; + case 'V': print_version(); exit(0); + case 'I': + case '?': + usage(); + exit(0); + default: + fprintf(stderr,"%s: Illegal option: -%c\n",my_progname,opterr); + usage(); + exit(1); + } + } + (*argc)-=optind; + (*argv)+=optind; + if (!*argc) + { + usage(); + exit(1); + } + if (join_table) + { + backup=0; /* Not needed */ + tmp_dir[0]=0; + } + return; +} + + +static N_INFO *open_isam_file(char *name,int mode) +{ + N_INFO *isam_file; + ISAM_SHARE *share; + DBUG_ENTER("open_isam_file"); + + if (!(isam_file=nisam_open(name,mode,(opt_wait ? HA_OPEN_WAIT_IF_LOCKED : + HA_OPEN_ABORT_IF_LOCKED)))) + { + VOID(fprintf(stderr,"%s gave error %d on open\n",name,my_errno)); + DBUG_RETURN(0); + } + share=isam_file->s; + if (share->base.blobs) + { + VOID(fprintf(stderr,"%s has blobs, can't pack it\n",name)); + VOID(nisam_close(isam_file)); + DBUG_RETURN(0); + } + if (share->base.options & HA_OPTION_COMPRESS_RECORD && !join_table) + { + if (!force_pack) + { + VOID(fprintf(stderr,"%s is already compressed\n",name)); + VOID(nisam_close(isam_file)); + DBUG_RETURN(0); + } + if (verbose) + puts("Recompressing already compressed table"); + share->base.options&= ~HA_OPTION_READ_ONLY_DATA; /* We are modifing it */ + } + if (! force_pack && share->state.records != 0 && + (share->state.records <= 1 || + share->state.data_file_length < 1024) && ! join_table) + { + VOID(fprintf(stderr,"%s is too small to compress\n",name)); + VOID(nisam_close(isam_file)); + DBUG_RETURN(0); + } + VOID(nisam_lock_database(isam_file,F_WRLCK)); + DBUG_RETURN(isam_file); +} + + +static bool open_isam_files(MRG_INFO *mrg,char **names,uint count) +{ + uint i,j; + mrg->count=0; + mrg->current=0; + mrg->file=(N_INFO**) my_malloc(sizeof(N_INFO*)*count,MYF(MY_FAE)); + for (i=0; i < count ; i++) + { + if (!(mrg->file[i]=open_isam_file(names[i],O_RDONLY))) + goto error; + } + /* Check that files are identical */ + for (j=0 ; j < count-1 ; j++) + { + N_RECINFO *m1,*m2,*end; + if (mrg->file[j]->s->base.reclength != mrg->file[j+1]->s->base.reclength || + mrg->file[j]->s->base.fields != mrg->file[j+1]->s->base.fields) + goto diff_file; + m1=mrg->file[j]->s->rec; + end=m1+mrg->file[j]->s->base.fields; + m2=mrg->file[j+1]->s->rec; + for ( ; m1 != end ; m1++,m2++) + { + if ((m1->base.type != m2->base.type && ! force_pack) || + m1->base.length != m2->base.length) + goto diff_file; + } + } + mrg->count=count; + return 0; + + diff_file: + fprintf(stderr,"%s: Tables '%s' and '%s' are not identical\n", + my_progname,names[j],names[j+1]); + error: + while (i--) + nisam_close(mrg->file[i]); + return 1; +} + + +static int compress(MRG_INFO *mrg,char *result_table) +{ + int error; + File new_file,join_isam_file; + N_INFO *isam_file; + ISAM_SHARE *share; + char org_name[FN_REFLEN],new_name[FN_REFLEN],temp_name[FN_REFLEN]; + uint i,header_length,fields,trees,used_trees; + my_off_t old_length,new_length,tot_elements; + HUFF_COUNTS *huff_counts; + HUFF_TREE *huff_trees; + DBUG_ENTER("compress"); + + isam_file=mrg->file[0]; /* Take this as an example */ + share=isam_file->s; + new_file=join_isam_file= -1; + trees=fields=0; + huff_trees=0; + huff_counts=0; + + /* Create temporary or join file */ + + if (backup) + VOID(fn_format(org_name,isam_file->filename,"",N_NAME_DEXT,2)); + else + VOID(fn_format(org_name,isam_file->filename,"",N_NAME_DEXT,2+4+16)); + if (!test_only && result_table) + { + /* Make a new indexfile based on first file in list */ + uint length; + char *buff; + strmov(org_name,result_table); /* Fix error messages */ + VOID(fn_format(new_name,result_table,"",N_NAME_IEXT,2)); + if ((join_isam_file=my_create(new_name,0,tmpfile_createflag,MYF(MY_WME))) + < 0) + goto err; + length=share->base.keystart; + if (!(buff=my_malloc(length,MYF(MY_WME)))) + goto err; + if (my_pread(share->kfile,buff,length,0L,MYF(MY_WME | MY_NABP)) || + my_write(join_isam_file,buff,length, + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL))) + { + my_free(buff,MYF(0)); + goto err; + } + my_free(buff,MYF(0)); + VOID(fn_format(new_name,result_table,"",N_NAME_DEXT,2)); + } + else if (!tmp_dir[0]) + VOID(make_new_name(new_name,org_name)); + else + VOID(fn_format(new_name,org_name,tmp_dir,DATA_TMP_EXT,1+2+4)); + if (!test_only && + (new_file=my_create(new_name,0,tmpfile_createflag,MYF(MY_WME))) < 0) + goto err; + + /* Start calculating statistics */ + + mrg->records=0; + for (i=0 ; i < mrg->count ; i++) + mrg->records+=mrg->file[i]->s->state.records; + if (write_loop || verbose) + { + printf("Compressing %s: (%lu records)\n", + result_table ? new_name : org_name,(ulong) mrg->records); + } + trees=fields=share->base.fields; + huff_counts=init_huff_count(isam_file,mrg->records); + QUICK_SAFEMALLOC; + if (write_loop || verbose) + printf("- Calculating statistics\n"); + if (get_statistic(mrg,huff_counts)) + goto err; + NORMAL_SAFEMALLOC; + old_length=0; + for (i=0; i < mrg->count ; i++) + old_length+= (mrg->file[i]->s->state.data_file_length - + mrg->file[i]->s->state.empty); + + if (init_queue(&queue,256,0,0,compare_huff_elements,0)) + goto err; + check_counts(huff_counts,fields,mrg->records); + huff_trees=make_huff_trees(huff_counts,trees); + if ((int) (used_trees=join_same_trees(huff_counts,trees)) < 0) + goto err; + if (make_huff_decode_table(huff_trees,fields)) + goto err; + + init_file_buffer(new_file,0); + file_buffer.pos_in_file=HEAD_LENGTH; + if (! test_only) + VOID(my_seek(new_file,file_buffer.pos_in_file,MY_SEEK_SET,MYF(0))); + + write_field_info(huff_counts,fields,used_trees); + if (!(tot_elements=write_huff_tree(huff_trees,trees))) + goto err; + header_length=(uint) file_buffer.pos_in_file+ + (uint) (file_buffer.pos-file_buffer.buffer); + + /* Compress file */ + if (write_loop || verbose) + printf("- Compressing file\n"); + error=compress_isam_file(mrg,huff_counts); + new_length=file_buffer.pos_in_file; + if (!error && !test_only) + { + char buff[MEMMAP_EXTRA_MARGIN]; /* End marginal for memmap */ + bzero(buff,sizeof(buff)); + error=my_write(file_buffer.file,buff,sizeof(buff), + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; + } + if (!error) + error=write_header(mrg,header_length,used_trees,tot_elements, + new_length); + end_file_buffer(); + + if (verbose && mrg->records) + printf("Min record length: %6d Max length: %6d Mean total length: %6d\n", + mrg->min_pack_length,mrg->max_pack_length, + (long) new_length/mrg->records); + + if (!test_only) + { + error|=my_close(new_file,MYF(MY_WME)); + if (!result_table) + { + error|=my_close(isam_file->dfile,MYF(MY_WME)); + isam_file->dfile= -1; /* Tell nisam_close file is closed */ + } + } + + free_counts_and_tree_and_queue(huff_trees,trees,huff_counts,fields); + if (! test_only && ! error) + { + if (result_table) + { + error=save_state_mrg(join_isam_file,mrg,new_length,crc); + } + else + { + if (backup) + { + if (my_rename(org_name,make_old_name(temp_name,isam_file->filename), + MYF(MY_WME))) + error=1; + else + { + if (tmp_dir[0]) + { + if (!(error=my_copy(new_name,org_name,MYF(MY_WME)))) + VOID(my_delete(new_name,MYF(MY_WME))); + } + else + error=my_rename(new_name,org_name,MYF(MY_WME)); + if (!error) + VOID(my_copystat(temp_name,org_name,MYF(MY_COPYTIME))); + } + } + else + { + if (tmp_dir[0]) + { + + if (!(error=my_copy(new_name,org_name, + MYF(MY_WME | MY_HOLD_ORIGINAL_MODES + | MY_COPYTIME)))) + VOID(my_delete(new_name,MYF(MY_WME))); + } + else + error=my_redel(org_name,new_name,MYF(MY_WME | MY_COPYTIME)); + } + if (! error) + save_state(isam_file,mrg,new_length,crc); + } + } + error|=mrg_close(mrg); + if (join_isam_file >= 0) + error|=my_close(join_isam_file,MYF(MY_WME)); + if (error) + { + VOID(fprintf(stderr,"Aborting: %s is not compressed\n",org_name)); + DBUG_RETURN(-1); + } + if (write_loop || verbose) + { + if (old_length) + printf("%.4g%% \n", + my_off_t2double(old_length-new_length)*100.0/ + my_off_t2double(old_length)); + else + puts("Empty file saved in compressed format"); + } + DBUG_RETURN(0); + + err: + free_counts_and_tree_and_queue(huff_trees,trees,huff_counts,fields); + if (new_file >= 0) + VOID(my_close(new_file,MYF(0))); + if (join_isam_file >= 0) + VOID(my_close(join_isam_file,MYF(0))); + mrg_close(mrg); + VOID(fprintf(stderr,"Aborted: %s is not compressed\n",org_name)); + DBUG_RETURN(-1); +} + + /* Init a huff_count-struct for each field and init it */ + +static HUFF_COUNTS *init_huff_count(N_INFO *info,my_off_t records) +{ + reg2 uint i; + reg1 HUFF_COUNTS *count; + if ((count = (HUFF_COUNTS*) my_malloc(info->s->base.fields*sizeof(HUFF_COUNTS), + MYF(MY_ZEROFILL | MY_WME)))) + { + for (i=0 ; i < info->s->base.fields ; i++) + { + enum en_fieldtype type; + count[i].field_length=info->s->rec[i].base.length; + type= count[i].field_type= (enum en_fieldtype) info->s->rec[i].base.type; + if (type == FIELD_INTERVALL || + type == FIELD_CONSTANT || + type == FIELD_ZERO) + type = FIELD_NORMAL; + if (count[i].field_length <= 8 && + (type == FIELD_NORMAL || + type == FIELD_SKIPP_ZERO)) + count[i].max_zero_fill= count[i].field_length; + init_tree(&count[i].int_tree,0,-1,(qsort_cmp) compare_tree,0,NULL); + if (records) + count[i].tree_pos=count[i].tree_buff = + my_malloc(count[i].field_length > 1 ? tree_buff_length : 2, + MYF(MY_WME)); + } + } + return count; +} + + + /* Free memory used by counts and trees */ + +static void free_counts_and_tree_and_queue(HUFF_TREE *huff_trees, uint trees, HUFF_COUNTS *huff_counts, uint fields) +{ + register uint i; + + if (huff_trees) + { + for (i=0 ; i < trees ; i++) + { + if (huff_trees[i].element_buffer) + my_free((gptr) huff_trees[i].element_buffer,MYF(0)); + if (huff_trees[i].code) + my_free((gptr) huff_trees[i].code,MYF(0)); + } + my_free((gptr) huff_trees,MYF(0)); + } + if (huff_counts) + { + for (i=0 ; i < fields ; i++) + { + if (huff_counts[i].tree_buff) + { + my_free((gptr) huff_counts[i].tree_buff,MYF(0)); + delete_tree(&huff_counts[i].int_tree); + } + } + my_free((gptr) huff_counts,MYF(0)); + } + delete_queue(&queue); /* This is safe to free */ + return; +} + + /* Read through old file and gather some statistics */ + +static int get_statistic(MRG_INFO *mrg,HUFF_COUNTS *huff_counts) +{ + int error; + uint length,reclength; + byte *record,*pos,*next_pos,*end_pos,*start_pos; + my_off_t record_count; + HUFF_COUNTS *count,*end_count; + TREE_ELEMENT *element; + DBUG_ENTER("get_statistic"); + + reclength=mrg->file[0]->s->base.reclength; + record=(byte*) my_alloca(reclength); + end_count=huff_counts+mrg->file[0]->s->base.fields; + record_count=crc=0; + + mrg_reset(mrg); + while ((error=mrg_rrnd(mrg,record)) >= 0) + { + if (! error) + { + crc^=checksum(record,reclength); + for (pos=record,count=huff_counts ; + count < end_count ; + count++, + pos=next_pos) + { + next_pos=end_pos=(start_pos=pos)+count->field_length; + + /* Put value in tree if there is room for it */ + if (count->tree_buff) + { + global_count=count; + if (!(element=tree_insert(&count->int_tree,pos,0)) || + element->count == 1 && + count->tree_buff + tree_buff_length < + count->tree_pos + count->field_length || + count->field_length == 1 && + count->int_tree.elements_in_tree > 1) + { + delete_tree(&count->int_tree); + my_free(count->tree_buff,MYF(0)); + count->tree_buff=0; + } + else + { + if (element->count == 1) + { /* New element */ + memcpy(count->tree_pos,pos,(size_t) count->field_length); + tree_set_pointer(element,count->tree_pos); + count->tree_pos+=count->field_length; + } + } + } + + /* Save character counters and space-counts and zero-field-counts */ + if (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIPP_ENDSPACE) + { + for ( ; end_pos > pos ; end_pos--) + if (end_pos[-1] != ' ') + break; + if (end_pos == pos) + { + count->empty_fields++; + count->max_zero_fill=0; + continue; + } + length= (uint) (next_pos-end_pos); + count->tot_end_space+=length; + if (length < 8) + count->end_space[length]++; + if (count->max_end_space < length) + count->max_end_space = length; + } + if (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIPP_PRESPACE) + { + for (pos=start_pos; pos < end_pos ; pos++) + if (pos[0] != ' ') + break; + if (end_pos == pos) + { + count->empty_fields++; + count->max_zero_fill=0; + continue; + } + length= (uint) (pos-start_pos); + count->tot_pre_space+=length; + if (length < 8) + count->pre_space[length]++; + if (count->max_pre_space < length) + count->max_pre_space = length; + } + if (count->field_length <= 8 && + (count->field_type == FIELD_NORMAL || + count->field_type == FIELD_SKIPP_ZERO)) + { + uint i; + if (!memcmp((byte*) start_pos,zero_string,count->field_length)) + { + count->zero_fields++; + continue; + } +#ifdef BYTE_ORDER_HIGH_FIRST + for (i =0 ; i < count->max_zero_fill && ! start_pos[i] ; i++) ; + if (i < count->max_zero_fill) + count->max_zero_fill=i; +#else + for (i =0 ; i < count->max_zero_fill && ! end_pos[-1 - (int) i] ; i++) ; + if (i < count->max_zero_fill) + count->max_zero_fill=i; +#endif + } + for (pos=start_pos ; pos < end_pos ; pos++) + count->counts[(uchar) *pos]++; + } + record_count++; + if (write_loop && record_count % WRITE_COUNT == 0) + { + printf("%lu\r",(ulong) record_count); VOID(fflush(stdout)); + } + } + } + if (write_loop) + { + printf(" \r"); VOID(fflush(stdout)); + } + mrg->records=record_count; + my_afree((gptr) record); + DBUG_RETURN(0); +} + +static int compare_huff_elements(void *not_used, byte *a, byte *b) +{ + return *((my_off_t*) a) < *((my_off_t*) b) ? -1 : + (*((my_off_t*) a) == *((my_off_t*) b) ? 0 : 1); +} + + /* Check each tree if we should use pre-space-compress, end-space- + compress, empty-field-compress or zero-field-compress */ + +static void check_counts(HUFF_COUNTS *huff_counts, uint trees, my_off_t records) +{ + uint space_fields,fill_zero_fields,field_count[(int) FIELD_ZERO+1]; + my_off_t old_length,new_length,length; + DBUG_ENTER("check_counts"); + + bzero((gptr) field_count,sizeof(field_count)); + space_fields=fill_zero_fields=0; + + for (; trees-- ; huff_counts++) + { + huff_counts->field_type=FIELD_NORMAL; + huff_counts->pack_type=0; + + if (huff_counts->zero_fields || ! records) + { + my_off_t old_space_count; + if (huff_counts->zero_fields == records) + { + huff_counts->field_type= FIELD_ZERO; + huff_counts->bytes_packed=0; + huff_counts->counts[0]=0; + goto found_pack; + } + old_space_count=huff_counts->counts[' ']; + huff_counts->counts[' ']+=huff_counts->tot_end_space+ + huff_counts->tot_pre_space + + huff_counts->empty_fields * huff_counts->field_length; + old_length=calc_packed_length(huff_counts,0)+records/8; + length=huff_counts->zero_fields*huff_counts->field_length; + huff_counts->counts[0]+=length; + new_length=calc_packed_length(huff_counts,0); + if (old_length < new_length && huff_counts->field_length > 1) + { + huff_counts->field_type=FIELD_SKIPP_ZERO; + huff_counts->counts[0]-=length; + huff_counts->bytes_packed=old_length- records/8; + goto found_pack; + } + huff_counts->counts[' ']=old_space_count; + } + huff_counts->bytes_packed=calc_packed_length(huff_counts,0); + if (huff_counts->empty_fields) + { + if (huff_counts->field_length > 2 && + huff_counts->empty_fields + (records - huff_counts->empty_fields)* + (1+max_bit(max(huff_counts->max_pre_space, + huff_counts->max_end_space))) < + records * max_bit(huff_counts->field_length)) + { + huff_counts->pack_type |= PACK_TYPE_SPACE_FIELDS; + } + else + { + length=huff_counts->empty_fields*huff_counts->field_length; + if (huff_counts->tot_end_space || ! huff_counts->tot_pre_space) + { + huff_counts->tot_end_space+=length; + huff_counts->max_end_space=huff_counts->field_length; + if (huff_counts->field_length < 8) + huff_counts->end_space[huff_counts->field_length]+= + huff_counts->empty_fields; + } + else + { + huff_counts->tot_pre_space+=length; + huff_counts->max_pre_space=huff_counts->field_length; + if (huff_counts->field_length < 8) + huff_counts->pre_space[huff_counts->field_length]+= + huff_counts->empty_fields; + } + } + } + if (huff_counts->tot_end_space) + { + huff_counts->counts[' ']+=huff_counts->tot_pre_space; + if (test_space_compress(huff_counts,records,huff_counts->max_end_space, + huff_counts->end_space, + huff_counts->tot_end_space,FIELD_SKIPP_ENDSPACE)) + goto found_pack; + huff_counts->counts[' ']-=huff_counts->tot_pre_space; + } + if (huff_counts->tot_pre_space) + { + if (test_space_compress(huff_counts,records,huff_counts->max_pre_space, + huff_counts->pre_space, + huff_counts->tot_pre_space,FIELD_SKIPP_PRESPACE)) + goto found_pack; + } + + found_pack: /* Found field-packing */ + + /* Test if we can use zero-fill */ + + if (huff_counts->max_zero_fill && + (huff_counts->field_type == FIELD_NORMAL || + huff_counts->field_type == FIELD_SKIPP_ZERO)) + { + huff_counts->counts[0]-=huff_counts->max_zero_fill* + (huff_counts->field_type == FIELD_SKIPP_ZERO ? + records - huff_counts->zero_fields : records); + huff_counts->pack_type|=PACK_TYPE_ZERO_FILL; + huff_counts->bytes_packed=calc_packed_length(huff_counts,0); + } + + /* Test if intervall-field is better */ + + if (huff_counts->tree_buff) + { + HUFF_TREE tree; + + tree.element_buffer=0; + if (!make_huff_tree(&tree,huff_counts) && + tree.bytes_packed+tree.tree_pack_length < huff_counts->bytes_packed) + { + if (tree.elements == 1) + huff_counts->field_type=FIELD_CONSTANT; + else + huff_counts->field_type=FIELD_INTERVALL; + huff_counts->pack_type=0; + } + else + { + my_free((gptr) huff_counts->tree_buff,MYF(0)); + delete_tree(&huff_counts->int_tree); + huff_counts->tree_buff=0; + } + if (tree.element_buffer) + my_free((gptr) tree.element_buffer,MYF(0)); + } + if (huff_counts->pack_type & PACK_TYPE_SPACE_FIELDS) + space_fields++; + if (huff_counts->pack_type & PACK_TYPE_ZERO_FILL) + fill_zero_fields++; + field_count[huff_counts->field_type]++; + } + if (verbose) + printf("\nnormal: %3d empty-space: %3d empty-zero: %3d empty-fill: %3d\npre-space: %3d end-space: %3d table-lookup: %3d zero: %3d\n", + field_count[FIELD_NORMAL],space_fields, + field_count[FIELD_SKIPP_ZERO],fill_zero_fields, + field_count[FIELD_SKIPP_PRESPACE], + field_count[FIELD_SKIPP_ENDSPACE], + field_count[FIELD_INTERVALL], + field_count[FIELD_ZERO]); + DBUG_VOID_RETURN; +} + + /* Test if we can use space-compression and empty-field-compression */ + +static int +test_space_compress(HUFF_COUNTS *huff_counts, my_off_t records, + uint max_space_length, my_off_t *space_counts, + my_off_t tot_space_count, enum en_fieldtype field_type) +{ + int min_pos; + uint length_bits,i; + my_off_t space_count,min_space_count,min_pack,new_length,skipp; + + length_bits=max_bit(max_space_length); + + /* Default no end_space-packing */ + space_count=huff_counts->counts[(uint) ' ']; + min_space_count= (huff_counts->counts[(uint) ' ']+= tot_space_count); + min_pack=calc_packed_length(huff_counts,0); + min_pos= -2; + huff_counts->counts[(uint) ' ']=space_count; + + /* Test with allways space-count */ + new_length=huff_counts->bytes_packed+length_bits*records/8; + if (new_length+1 < min_pack) + { + min_pos= -1; + min_pack=new_length; + min_space_count=space_count; + } + /* Test with length-flag */ + for (skipp=0L, i=0 ; i < 8 ; i++) + { + if (space_counts[i]) + { + if (i) + huff_counts->counts[(uint) ' ']+=space_counts[i]; + skipp+=huff_counts->pre_space[i]; + new_length=calc_packed_length(huff_counts,0)+ + (records+(records-skipp)*(1+length_bits))/8; + if (new_length < min_pack) + { + min_pos=(int) i; + min_pack=new_length; + min_space_count=huff_counts->counts[(uint) ' ']; + } + } + } + + huff_counts->counts[(uint) ' ']=min_space_count; + huff_counts->bytes_packed=min_pack; + switch (min_pos) { + case -2: + return(0); /* No space-compress */ + case -1: /* Always space-count */ + huff_counts->field_type=field_type; + huff_counts->min_space=0; + huff_counts->length_bits=max_bit(max_space_length); + break; + default: + huff_counts->field_type=field_type; + huff_counts->min_space=(uint) min_pos; + huff_counts->pack_type|=PACK_TYPE_SELECTED; + huff_counts->length_bits=max_bit(max_space_length); + break; + } + return(1); /* Using space-compress */ +} + + + /* Make a huff_tree of each huff_count */ + +static HUFF_TREE* make_huff_trees(HUFF_COUNTS *huff_counts, uint trees) +{ + uint tree; + HUFF_TREE *huff_tree; + DBUG_ENTER("make_huff_trees"); + + if (!(huff_tree=(HUFF_TREE*) my_malloc(trees*sizeof(HUFF_TREE), + MYF(MY_WME | MY_ZEROFILL)))) + DBUG_RETURN(0); + + for (tree=0 ; tree < trees ; tree++) + { + if (make_huff_tree(huff_tree+tree,huff_counts+tree)) + { + while (tree--) + my_free((gptr) huff_tree[tree].element_buffer,MYF(0)); + my_free((gptr) huff_tree,MYF(0)); + DBUG_RETURN(0); + } + } + DBUG_RETURN(huff_tree); +} + + /* Update huff_tree according to huff_counts->counts or + huff_counts->tree_buff */ + +static int make_huff_tree(HUFF_TREE *huff_tree, HUFF_COUNTS *huff_counts) +{ + uint i,found,bits_packed,first,last; + my_off_t bytes_packed; + HUFF_ELEMENT *a,*b,*new; + + first=last=0; + if (huff_counts->tree_buff) + { + found= (uint) (huff_counts->tree_pos - huff_counts->tree_buff) / + huff_counts->field_length; + first=0; last=found-1; + } + else + { + for (i=found=0 ; i < 256 ; i++) + { + if (huff_counts->counts[i]) + { + if (! found++) + first=i; + last=i; + } + } + if (found < 2) + found=2; + } + + if (queue.max_elements < found) + { + delete_queue(&queue); + if (init_queue(&queue,found,0,0,compare_huff_elements,0)) + return -1; + } + + if (!huff_tree->element_buffer) + { + if (!(huff_tree->element_buffer= + (HUFF_ELEMENT*) my_malloc(found*2*sizeof(HUFF_ELEMENT),MYF(MY_WME)))) + return 1; + } + else + { + HUFF_ELEMENT *temp; + if (!(temp= + (HUFF_ELEMENT*) my_realloc((gptr) huff_tree->element_buffer, + found*2*sizeof(HUFF_ELEMENT), + MYF(MY_WME)))) + return 1; + huff_tree->element_buffer=temp; + } + + huff_counts->tree=huff_tree; + huff_tree->counts=huff_counts; + huff_tree->min_chr=first; + huff_tree->max_chr=last; + huff_tree->char_bits=max_bit(last-first); + huff_tree->offset_bits=max_bit(found-1)+1; + + if (huff_counts->tree_buff) + { + huff_tree->elements=0; + tree_walk(&huff_counts->int_tree, + (int (*)(void*, element_count,void*)) save_counts_in_queue, + (gptr) huff_tree, left_root_right); + huff_tree->tree_pack_length=(1+15+16+5+5+ + (huff_tree->char_bits+1)*found+ + (huff_tree->offset_bits+1)* + (found-2)+7)/8 + + (uint) (huff_tree->counts->tree_pos- + huff_tree->counts->tree_buff); + } + else + { + huff_tree->elements=found; + huff_tree->tree_pack_length=(9+9+5+5+ + (huff_tree->char_bits+1)*found+ + (huff_tree->offset_bits+1)* + (found-2)+7)/8; + + for (i=first, found=0 ; i <= last ; i++) + { + if (huff_counts->counts[i]) + { + new=huff_tree->element_buffer+(found++); + new->count=huff_counts->counts[i]; + new->a.leaf.null=0; + new->a.leaf.element_nr=i; + queue.root[found]=(byte*) new; + } + } + while (found < 2) + { /* Our huff_trees request at least 2 elements */ + new=huff_tree->element_buffer+(found++); + new->count=0; + new->a.leaf.null=0; + if (last) + new->a.leaf.element_nr=huff_tree->min_chr=last-1; + else + new->a.leaf.element_nr=huff_tree->max_chr=last+1; + queue.root[found]=(byte*) new; + } + } + queue.elements=found; + + for (i=found/2 ; i > 0 ; i--) + _downheap(&queue,i); + bytes_packed=0; bits_packed=0; + for (i=1 ; i < found ; i++) + { + a=(HUFF_ELEMENT*) queue_remove(&queue,0); + b=(HUFF_ELEMENT*) queue.root[1]; + new=huff_tree->element_buffer+found+i; + new->count=a->count+b->count; + bits_packed+=(uint) (new->count & 7); + bytes_packed+=new->count/8; + new->a.nod.left=a; /* lesser in left */ + new->a.nod.right=b; + queue.root[1]=(byte*) new; + queue_replaced(&queue); + } + huff_tree->root=(HUFF_ELEMENT*) queue.root[1]; + huff_tree->bytes_packed=bytes_packed+(bits_packed+7)/8; + return 0; +} + +static int compare_tree(register const uchar *s, register const uchar *t) +{ + uint length; + for (length=global_count->field_length; length-- ;) + if (*s++ != *t++) + return (int) s[-1] - (int) t[-1]; + return 0; +} + + /* Used by make_huff_tree to save intervall-counts in queue */ + +static int save_counts_in_queue(byte *key, element_count count, HUFF_TREE *tree) +{ + HUFF_ELEMENT *new; + + new=tree->element_buffer+(tree->elements++); + new->count=count; + new->a.leaf.null=0; + new->a.leaf.element_nr= (uint) (key- tree->counts->tree_buff) / + tree->counts->field_length; + queue.root[tree->elements]=(byte*) new; + return 0; +} + + + /* Calculate length of file if given counts should be used */ + /* Its actually a faster version of make_huff_tree */ + +static my_off_t calc_packed_length(HUFF_COUNTS *huff_counts, uint add_tree_lenght) +{ + uint i,found,bits_packed,first,last; + my_off_t bytes_packed; + HUFF_ELEMENT element_buffer[256]; + DBUG_ENTER("calc_packed_length"); + + first=last=0; + for (i=found=0 ; i < 256 ; i++) + { + if (huff_counts->counts[i]) + { + if (! found++) + first=i; + last=i; + queue.root[found]=(byte*) &huff_counts->counts[i]; + } + } + if (!found) + DBUG_RETURN(0); /* Empty tree */ + if (found < 2) + queue.root[++found]=(byte*) &huff_counts->counts[last ? 0 : 1]; + + queue.elements=found; + + bytes_packed=0; bits_packed=0; + if (add_tree_lenght) + bytes_packed=(8+9+5+5+(max_bit(last-first)+1)*found+ + (max_bit(found-1)+1+1)*(found-2) +7)/8; + for (i=(found+1)/2 ; i > 0 ; i--) + _downheap(&queue,i); + for (i=0 ; i < found-1 ; i++) + { + HUFF_ELEMENT *a,*b,*new; + a=(HUFF_ELEMENT*) queue_remove(&queue,0); + b=(HUFF_ELEMENT*) queue.root[1]; + new=element_buffer+i; + new->count=a->count+b->count; + bits_packed+=(uint) (new->count & 7); + bytes_packed+=new->count/8; + queue.root[1]=(byte*) new; + queue_replaced(&queue); + } + DBUG_RETURN(bytes_packed+(bits_packed+7)/8); +} + + + /* Remove trees that don't give any compression */ + +static uint join_same_trees(HUFF_COUNTS *huff_counts, uint trees) +{ + uint k,tree_number; + HUFF_COUNTS count,*i,*j,*last_count; + + last_count=huff_counts+trees; + for (tree_number=0, i=huff_counts ; i < last_count ; i++) + { + if (!i->tree->tree_number) + { + i->tree->tree_number= ++tree_number; + if (i->tree_buff) + continue; /* Don't join intervall */ + for (j=i+1 ; j < last_count ; j++) + { + if (! j->tree->tree_number && ! j->tree_buff) + { + for (k=0 ; k < 256 ; k++) + count.counts[k]=i->counts[k]+j->counts[k]; + if (calc_packed_length(&count,1) <= + i->tree->bytes_packed + j->tree->bytes_packed+ + i->tree->tree_pack_length+j->tree->tree_pack_length+ + ALLOWED_JOIN_DIFF) + { + memcpy((byte*) i->counts,(byte*) count.counts, + sizeof(count.counts[0])*256); + my_free((gptr) j->tree->element_buffer,MYF(0)); + j->tree->element_buffer=0; + j->tree=i->tree; + bmove((byte*) i->counts,(byte*) count.counts, + sizeof(count.counts[0])*256); + if (make_huff_tree(i->tree,i)) + return (uint) -1; + } + } + } + } + } + if (verbose) + printf("Original trees: %d After join: %d\n",trees,tree_number); + return tree_number; /* Return trees left */ +} + + + /* Fill in huff_tree decode tables */ + +static int make_huff_decode_table(HUFF_TREE *huff_tree, uint trees) +{ + uint elements; + for ( ; trees-- ; huff_tree++) + { + if (huff_tree->tree_number > 0) + { + elements=huff_tree->counts->tree_buff ? huff_tree->elements : 256; + if (!(huff_tree->code = + (ulong*) my_malloc(elements* + (sizeof(ulong)+sizeof(uchar)), + MYF(MY_WME | MY_ZEROFILL)))) + return 1; + huff_tree->code_len=(uchar*) (huff_tree->code+elements); + make_traverse_code_tree(huff_tree,huff_tree->root,32,0); + } + } + return 0; +} + + +static void make_traverse_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, + uint size, ulong code) +{ + uint chr; + if (!element->a.leaf.null) + { + chr=element->a.leaf.element_nr; + huff_tree->code_len[chr]=(uchar) (32-size); + huff_tree->code[chr]= (code >> size); + if (huff_tree->height < 32-size) + huff_tree->height= 32-size; + } + else + { + size--; + make_traverse_code_tree(huff_tree,element->a.nod.left,size,code); + make_traverse_code_tree(huff_tree,element->a.nod.right,size, + code+((ulong) 1L << size)); + } + return; +} + + + /* Write header to new packed data file */ + +static int write_header(MRG_INFO *mrg,uint head_length,uint trees, + my_off_t tot_elements,my_off_t filelength) +{ + byte *buff=file_buffer.pos; + + bzero(buff,HEAD_LENGTH); + memcpy(buff,nisam_pack_file_magic,4); + int4store(buff+4,head_length); + int4store(buff+8, mrg->min_pack_length); + int4store(buff+12,mrg->max_pack_length); + int4store(buff+16,tot_elements); + int4store(buff+20,intervall_length); + int2store(buff+24,trees); + buff[26]=(char) mrg->ref_length; + /* Save record pointer length */ + buff[27]= (uchar) (filelength >= (1L << 24) ? 4 : + filelength >= (1L << 16) ? 3 : 2); + if (test_only) + return 0; + VOID(my_seek(file_buffer.file,0L,MY_SEEK_SET,MYF(0))); + return my_write(file_buffer.file,file_buffer.pos,HEAD_LENGTH, + MYF(MY_WME | MY_NABP | MY_WAIT_IF_FULL)) != 0; +} + + /* Write fieldinfo to new packed file */ + +static void write_field_info(HUFF_COUNTS *counts, uint fields, uint trees) +{ + reg1 uint i; + uint huff_tree_bits; + huff_tree_bits=max_bit(trees ? trees-1 : 0); + + for (i=0 ; i++ < fields ; counts++) + { + write_bits((ulong) (int) counts->field_type,4); + write_bits(counts->pack_type,4); + if (counts->pack_type & PACK_TYPE_ZERO_FILL) + write_bits(counts->max_zero_fill,4); + else + write_bits(counts->length_bits,4); + write_bits((ulong) counts->tree->tree_number-1,huff_tree_bits); + } + flush_bits(); + return; +} + + /* Write all huff_trees to new datafile. Return tot count of + elements in all trees + Returns 0 on error */ + +static my_off_t write_huff_tree(HUFF_TREE *huff_tree, uint trees) +{ + uint i,int_length; + uint *packed_tree,*offset,length; + my_off_t elements; + + for (i=length=0 ; i < trees ; i++) + if (huff_tree[i].tree_number > 0 && huff_tree[i].elements > length) + length=huff_tree[i].elements; + if (!(packed_tree=(uint*) my_alloca(sizeof(uint)*length*2))) + { + my_error(EE_OUTOFMEMORY,MYF(ME_BELL),sizeof(uint)*length*2); + return 0; + } + + intervall_length=0; + for (elements=0; trees-- ; huff_tree++) + { + if (huff_tree->tree_number == 0) + continue; /* Deleted tree */ + elements+=huff_tree->elements; + huff_tree->max_offset=2; + if (huff_tree->elements <= 1) + offset=packed_tree; + else + offset=make_offset_code_tree(huff_tree,huff_tree->root,packed_tree); + huff_tree->offset_bits=max_bit(huff_tree->max_offset); + if (huff_tree->max_offset >= IS_OFFSET) + { /* This should be impossible */ + VOID(fprintf(stderr,"Tree offset got too big: %d, aborted\n", + huff_tree->max_offset)); + my_afree((gptr) packed_tree); + return 0; + } + +#ifdef EXTRA_DBUG + printf("pos: %d elements: %d tree-elements: %d char_bits: %d\n", + (uint) (file_buffer.pos-file_buffer.buffer), + huff_tree->elements, (offset-packed_tree),huff_tree->char_bits); +#endif + if (!huff_tree->counts->tree_buff) + { + write_bits(0,1); + write_bits(huff_tree->min_chr,8); + write_bits(huff_tree->elements,9); + write_bits(huff_tree->char_bits,5); + write_bits(huff_tree->offset_bits,5); + int_length=0; + } + else + { + int_length=(uint) (huff_tree->counts->tree_pos - + huff_tree->counts->tree_buff); + write_bits(1,1); + write_bits(huff_tree->elements,15); + write_bits(int_length,16); + write_bits(huff_tree->char_bits,5); + write_bits(huff_tree->offset_bits,5); + intervall_length+=int_length; + } + length=(uint) (offset-packed_tree); + if (length != huff_tree->elements*2-2) + printf("error: Huff-tree-length: %d != calc_length: %d\n", + length,huff_tree->elements*2-2); + + for (i=0 ; i < length ; i++) + { + if (packed_tree[i] & IS_OFFSET) + write_bits(packed_tree[i] - IS_OFFSET+ ((ulong) 1L << huff_tree->offset_bits), + huff_tree->offset_bits+1); + else + write_bits(packed_tree[i]-huff_tree->min_chr,huff_tree->char_bits+1); + } + flush_bits(); + if (huff_tree->counts->tree_buff) + { + for (i=0 ; i < int_length ; i++) + write_bits((uint) (uchar) huff_tree->counts->tree_buff[i],8); + } + flush_bits(); + } + my_afree((gptr) packed_tree); + return elements; +} + + +static uint *make_offset_code_tree(HUFF_TREE *huff_tree, HUFF_ELEMENT *element, + uint *offset) +{ + uint *prev_offset; + + prev_offset= offset; + if (!element->a.nod.left->a.leaf.null) + { + offset[0] =(uint) element->a.nod.left->a.leaf.element_nr; + offset+=2; + } + else + { + prev_offset[0]= IS_OFFSET+2; + offset=make_offset_code_tree(huff_tree,element->a.nod.left,offset+2); + } + if (!element->a.nod.right->a.leaf.null) + { + prev_offset[1]=element->a.nod.right->a.leaf.element_nr; + return offset; + } + else + { + uint temp=(uint) (offset-prev_offset-1); + prev_offset[1]= IS_OFFSET+ temp; + if (huff_tree->max_offset < temp) + huff_tree->max_offset = temp; + return make_offset_code_tree(huff_tree,element->a.nod.right,offset); + } +} + + /* Get number of bits neaded to represent value */ + +static uint max_bit(register uint value) +{ + reg2 uint power=1; + + while ((value>>=1)) + power++; + return (power); +} + + +static int compress_isam_file(MRG_INFO *mrg, HUFF_COUNTS *huff_counts) +{ + int error; + uint i,max_calc_length,pack_ref_length,min_record_length,max_record_length, + intervall,field_length; + my_off_t record_count,max_allowed_length; + ulong length; + byte *record,*pos,*end_pos,*record_pos,*start_pos; + HUFF_COUNTS *count,*end_count; + HUFF_TREE *tree; + N_INFO *isam_file=mrg->file[0]; + DBUG_ENTER("compress_isam_file"); + + if (!(record=(byte*) my_alloca(isam_file->s->base.reclength))) + return -1; + end_count=huff_counts+isam_file->s->base.fields; + min_record_length= (uint) ~0; + max_record_length=0; + + for (i=max_calc_length=0 ; i < isam_file->s->base.fields ; i++) + { + if (!(huff_counts[i].pack_type & PACK_TYPE_ZERO_FILL)) + huff_counts[i].max_zero_fill=0; + if (huff_counts[i].field_type == FIELD_CONSTANT || + huff_counts[i].field_type == FIELD_ZERO) + continue; + if (huff_counts[i].field_type == FIELD_INTERVALL) + max_calc_length+=huff_counts[i].tree->height; + else + max_calc_length+= + (huff_counts[i].field_length - huff_counts[i].max_zero_fill)* + huff_counts[i].tree->height+huff_counts[i].length_bits; + } + max_calc_length/=8; + if (max_calc_length <= 255) + pack_ref_length=1; + else if (max_calc_length <= 65535) + pack_ref_length=2; + else + pack_ref_length=3; + if (force_pack_ref_length) + pack_ref_length=force_pack_ref_length; + max_allowed_length= 1L << (pack_ref_length*8); + record_count=0; + + mrg_reset(mrg); + while ((error=mrg_rrnd(mrg,record)) >= 0) + { + if (! error) + { + if (flush_buffer(max_calc_length+pack_ref_length)) + break; + record_pos=file_buffer.pos; + file_buffer.pos+=pack_ref_length; + for (start_pos=record, count= huff_counts; count < end_count ; count++) + { + end_pos=start_pos+(field_length=count->field_length); + tree=count->tree; + + if (count->pack_type & PACK_TYPE_SPACE_FIELDS) + { + for (pos=start_pos ; *pos == ' ' && pos < end_pos; pos++) ; + if (pos == end_pos) + { + write_bits(1,1); + start_pos=end_pos; + continue; + } + write_bits(0,1); + } + +#ifdef BYTE_ORDER_HIGH_FIRST + start_pos+=count->max_zero_fill; +#else + end_pos-=count->max_zero_fill; +#endif + field_length-=count->max_zero_fill; + + switch(count->field_type) { + case FIELD_SKIPP_ZERO: + if (!memcmp((byte*) start_pos,zero_string,field_length)) + { + write_bits(1,1); + start_pos=end_pos; + break; + } + write_bits(0,1); + /* Fall through */ + case FIELD_NORMAL: + for ( ; start_pos < end_pos ; start_pos++) + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + break; + case FIELD_SKIPP_ENDSPACE: + for (pos=end_pos ; pos > start_pos && pos[-1] == ' ' ; pos--) ; + length=(uint) (end_pos-pos); + if (count->pack_type & PACK_TYPE_SELECTED) + { + if (length > count->min_space) + { + write_bits(1,1); + write_bits(length,count->length_bits); + } + else + { + write_bits(0,1); + pos=end_pos; + } + } + else + write_bits(length,count->length_bits); + for ( ; start_pos < pos ; start_pos++) + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + start_pos=end_pos; + break; + case FIELD_SKIPP_PRESPACE: + for (pos=start_pos ; pos < end_pos && pos[0] == ' ' ; pos++) ; + length=(uint) (pos-start_pos); + if (count->pack_type & PACK_TYPE_SELECTED) + { + if (length > count->min_space) + { + write_bits(1,1); + write_bits(length,count->length_bits); + } + else + { + pos=start_pos; + write_bits(0,1); + } + } + else + write_bits(length,count->length_bits); + for (start_pos=pos ; start_pos < end_pos ; start_pos++) + write_bits(tree->code[(uchar) *start_pos], + (uint) tree->code_len[(uchar) *start_pos]); + break; + case FIELD_CONSTANT: + case FIELD_ZERO: + start_pos=end_pos; + break; + case FIELD_INTERVALL: + global_count=count; + pos=(byte*) tree_search(&count->int_tree,start_pos); + intervall=(uint) (pos - count->tree_buff)/field_length; + write_bits(tree->code[intervall],(uint) tree->code_len[intervall]); + start_pos=end_pos; + break; + case FIELD_BLOB: + VOID(fprintf(stderr,"Can't pack files with blobs. Aborting\n")); + DBUG_RETURN(1); + case FIELD_LAST: + case FIELD_VARCHAR: + case FIELD_CHECK: + abort(); /* Impossible */ + } +#ifndef BYTE_ORDER_HIGH_FIRST + start_pos+=count->max_zero_fill; +#endif + } + flush_bits(); + length=(ulong) (file_buffer.pos-record_pos)-pack_ref_length; + save_integer(record_pos,pack_ref_length,length); + if (length < (ulong) min_record_length) + min_record_length=(uint) length; + if (length > (ulong) max_record_length) + { + max_record_length=(uint) length; + if (max_record_length >= max_allowed_length) + { + fprintf(stderr, + "Error: Found record with packed-length: %d, max is: %d\n", + max_record_length,max_allowed_length); + error=1; + break; + } + } + if (write_loop && ++record_count % WRITE_COUNT == 0) + { + printf("%lu\r",(ulong) record_count); VOID(fflush(stdout)); + } + } + else if (my_errno != HA_ERR_RECORD_DELETED) + break; + } + if (error < 0) + { + error=0; + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%s: Got error %d reading records\n",my_progname,my_errno); + error= 1; + } + } + + my_afree((gptr) record); + mrg->ref_length=pack_ref_length; + mrg->min_pack_length=max_record_length ? min_record_length : 0; + mrg->max_pack_length=max_record_length; + if (verbose && max_record_length && + max_record_length < max_allowed_length/256) + printf("Record-length is %d bytes, could have been %d bytes\nYou can change this by using -p=%d next time you pack this file\n", + pack_ref_length, + max_record_length/256+1, + max_record_length/256+1); + DBUG_RETURN(error || error_on_write || flush_buffer((uint) ~0)); +} + + +static char *make_new_name(char *new_name, char *old_name) +{ + return fn_format(new_name,old_name,"",DATA_TMP_EXT,2+4); +} + +static char *make_old_name(char *new_name, char *old_name) +{ + return fn_format(new_name,old_name,"",OLD_EXT,2+4); +} + + /* rutines for bit writing buffer */ + +static void init_file_buffer(File file, pbool read_buffer) +{ + file_buffer.file=file; + file_buffer.buffer=my_malloc(ALIGN_SIZE(RECORD_CACHE_SIZE),MYF(MY_WME)); + file_buffer.end=file_buffer.buffer+ALIGN_SIZE(RECORD_CACHE_SIZE)-4; + file_buffer.pos_in_file=0; + error_on_write=0; + if (read_buffer) + { + + file_buffer.pos=file_buffer.end; + file_buffer.bits=0; + } + else + { + file_buffer.pos=file_buffer.buffer; + file_buffer.bits=BITS_SAVED; + } + file_buffer.byte=0; +} + + +static int flush_buffer(uint neaded_length) +{ + uint length; + if ((uint) (file_buffer.end - file_buffer.pos) > neaded_length) + return 0; + length=(uint) (file_buffer.pos-file_buffer.buffer); + file_buffer.pos=file_buffer.buffer; + file_buffer.pos_in_file+=length; + if (test_only) + return 0; + return (error_on_write|=test(my_write(file_buffer.file,file_buffer.buffer, + length, + MYF(MY_WME | MY_NABP | + MY_WAIT_IF_FULL)))); +} + +static void end_file_buffer(void) +{ + my_free((gptr) file_buffer.buffer,MYF(0)); +} + + /* output `bits` low bits of `value' */ + +static void write_bits (register ulong value, register uint bits) +{ + if ((file_buffer.bits-=(int) bits) >= 0) + { + file_buffer.byte|=value << file_buffer.bits; + } + else + { + reg3 uint byte_buff; + bits= (uint) -file_buffer.bits; + byte_buff=file_buffer.byte | (uint) (value >> bits); +#if BITS_SAVED == 32 + *file_buffer.pos++= (byte) (byte_buff >> 24) ; + *file_buffer.pos++= (byte) (byte_buff >> 16) ; +#endif + *file_buffer.pos++= (byte) (byte_buff >> 8) ; + *file_buffer.pos++= (byte) byte_buff; + + value&=((ulong) 1L << bits)-1; +#if BITS_SAVED == 16 + if (bits >= sizeof(uint)) + { + bits-=8; + *file_buffer.pos++= (uchar) (value >> bits); + value&= ((ulong) 1L << bits)-1; + if (bits >= sizeof(uint)) + { + bits-=8; + *file_buffer.pos++= (uchar) (value >> bits); + value&= ((ulong) 1L << bits)-1; + } + } +#endif + if (file_buffer.pos >= file_buffer.end) + VOID(flush_buffer((uint) ~0)); + file_buffer.bits=(int) (BITS_SAVED - bits); + file_buffer.byte=(uint) (value << (BITS_SAVED - bits)); + } + return; +} + + /* Flush bits in bit_buffer to buffer */ + +static void flush_bits (void) +{ + uint bits,byte_buff; + + bits=(file_buffer.bits) & ~7; + byte_buff = file_buffer.byte >> bits; + bits=BITS_SAVED - bits; + while (bits > 0) + { + bits-=8; + *file_buffer.pos++= (byte) (uchar) (byte_buff >> bits) ; + } + file_buffer.bits=BITS_SAVED; + file_buffer.byte=0; + return; +} + + /* Store long in 1,2,3,4 or 5 bytes */ + +static void save_integer(byte *pos, uint pack_length, my_off_t value) +{ + switch (pack_length) { + case 5: int5store(pos,(ulonglong) value); break; + default: int4store(pos,(ulong) value); break; + case 3: int3store(pos,(ulong) value); break; + case 2: int2store(pos,(uint) value); break; + case 1: pos[0]= (byte) (uchar) value; break; + } + return; +} + + +/**************************************************************************** +** functions to handle the joined files +****************************************************************************/ + +static void save_state(N_INFO *isam_file,MRG_INFO *mrg,my_off_t new_length, + ulong crc) +{ + ISAM_SHARE *share=isam_file->s; + uint options=uint2korr(share->state.header.options); + DBUG_ENTER("save_state"); + + options|= HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA; + int2store(share->state.header.options,options); + + share->state.data_file_length=(ulong) new_length; + share->state.del=share->state.empty=0; + share->state.dellink= (ulong) NI_POS_ERROR; + share->state.splitt=(ulong) mrg->records; + share->state.version=(ulong) time((time_t*) 0); + share->state.keys=0; + share->state.key_file_length=share->base.keystart; + + isam_file->update|=(HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + isam_file->this_uniq=crc; /* Save crc here */ + share->changed=1; /* Force write of header */ + VOID(my_chsize(share->kfile,share->state.key_file_length, + MYF(0))); + if (share->state.keys != share->base.keys) + isamchk_neaded=1; + DBUG_VOID_RETURN; +} + + +static int save_state_mrg(File file,MRG_INFO *mrg,my_off_t new_length, + ulong crc) +{ + N_STATE_INFO state; + N_INFO *isam_file=mrg->file[0]; + uint options; + DBUG_ENTER("save_state_mrg"); + + memcpy(&state,&isam_file->s->state,sizeof(state)); + options= (uint2korr(state.header.options) | HA_OPTION_COMPRESS_RECORD | + HA_OPTION_READ_ONLY_DATA); + int2store(state.header.options,options); + state.data_file_length=(ulong) new_length; + state.del=state.empty=0; + state.dellink= (ulong) NI_POS_ERROR; + state.records=state.splitt=(ulong) mrg->records; + state.version=(ulong) time((time_t*) 0); + state.keys=0; + state.key_file_length=isam_file->s->base.keystart; + state.uniq=crc; + if (state.keys != isam_file->s->base.keys) + isamchk_neaded=1; + DBUG_RETURN (my_pwrite(file,(char*) &state.header, + isam_file->s->state_length,0L, + MYF(MY_NABP | MY_WME)) != 0); +} + + +/* reset for mrg_rrnd */ + +static void mrg_reset(MRG_INFO *mrg) +{ + if (mrg->current) + { + nisam_extra(*mrg->current,HA_EXTRA_NO_CACHE); + mrg->current=0; + } +} + +static int mrg_rrnd(MRG_INFO *info,byte *buf) +{ + int error; + N_INFO *isam_info; + my_off_t filepos; + + if (!info->current) + { + isam_info= *(info->current=info->file); + info->end=info->current+info->count; + nisam_extra(isam_info,HA_EXTRA_CACHE); + nisam_extra(isam_info,HA_EXTRA_RESET); + filepos=isam_info->s->pack.header_length; + } + else + { + isam_info= *info->current; + filepos= isam_info->nextpos; + } + + for (;;) + { + isam_info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if ((error=(*isam_info->s->read_rnd)(isam_info,(byte*) buf, + (ulong) filepos, 1)) >= 0 || + my_errno != HA_ERR_END_OF_FILE) + return (error); + nisam_extra(isam_info,HA_EXTRA_NO_CACHE); + if (info->current+1 == info->end) + return(-1); + info->current++; + isam_info= *info->current; + filepos=isam_info->s->pack.header_length; + nisam_extra(isam_info,HA_EXTRA_CACHE); + nisam_extra(isam_info,HA_EXTRA_RESET); + } +} + + +static int mrg_close(MRG_INFO *mrg) +{ + uint i; + int error=0; + for (i=0 ; i < mrg->count ; i++) + error|=nisam_close(mrg->file[i]); + return error; +} diff --git a/isam/panic.c b/isam/panic.c new file mode 100644 index 00000000000..52a5d1eb3b6 --- /dev/null +++ b/isam/panic.c @@ -0,0 +1,135 @@ +/* Copyright (C) 2000 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 */ + +#include "isamdef.h" + + /* if flag == HA_PANIC_CLOSE then all misam files are closed */ + /* if flag == HA_PANIC_WRITE then all misam files are unlocked and + all changed data in single user misam is written to file */ + /* if flag == HA_PANIC_READ then all misam files that was locked when + nisam_panic(HA_PANIC_WRITE) was done is locked. A ni_readinfo() is + done for all single user files to get changes in database */ + + +int nisam_panic(enum ha_panic_function flag) +{ + int error=0; + LIST *list_element,*next_open; + N_INFO *info; + DBUG_ENTER("nisam_panic"); + + pthread_mutex_lock(&THR_LOCK_isam); + for (list_element=nisam_open_list ; list_element ; list_element=next_open) + { + next_open=list_element->next; /* Save if close */ + info=(N_INFO*) list_element->data; + switch (flag) { + case HA_PANIC_CLOSE: + pthread_mutex_unlock(&THR_LOCK_isam); /* Not exactly right... */ + if (nisam_close(info)) + error=my_errno; + pthread_mutex_lock(&THR_LOCK_isam); + break; + case HA_PANIC_WRITE: /* Do this to free databases */ +#ifdef CANT_OPEN_FILES_TWICE + if (info->s->base.options & HA_OPTION_READ_ONLY_DATA) + break; +#endif + if (flush_key_blocks(info->s->kfile,FLUSH_RELEASE)) + error=my_errno; + if (info->opt_flag & WRITE_CACHE_USED) + if (flush_io_cache(&info->rec_cache)) + error=my_errno; + if (info->opt_flag & READ_CACHE_USED) + { + if (flush_io_cache(&info->rec_cache)) + error=my_errno; + reinit_io_cache(&info->rec_cache,READ_CACHE,0, + (pbool) (info->lock_type != F_UNLCK),1); + } +#ifndef NO_LOCKING + if (info->lock_type != F_UNLCK && ! info->was_locked) + { + info->was_locked=info->lock_type; + if (nisam_lock_database(info,F_UNLCK)) + error=my_errno; + } +#else + { + int save_status=info->s->w_locks; /* Only w_locks! */ + info->s->w_locks=0; + if (_nisam_writeinfo(info, test(info->update & HA_STATE_CHANGED))) + error=my_errno; + info->s->w_locks=save_status; + info->update&= ~HA_STATE_CHANGED; /* Not changed */ + } +#endif /* NO_LOCKING */ +#ifdef CANT_OPEN_FILES_TWICE + if (info->s->kfile >= 0 && my_close(info->s->kfile,MYF(0))) + error = my_errno; + if (info->dfile >= 0 && my_close(info->dfile,MYF(0))) + error = my_errno; + info->s->kfile=info->dfile= -1; /* Files aren't open anymore */ + break; +#endif + case HA_PANIC_READ: /* Restore to before WRITE */ +#ifdef CANT_OPEN_FILES_TWICE + { /* Open closed files */ + char name_buff[FN_REFLEN]; + if (info->s->kfile < 0) + if ((info->s->kfile= my_open(fn_format(name_buff,info->filename,"", + N_NAME_IEXT,4),info->mode, + MYF(MY_WME))) < 0) + error = my_errno; + if (info->dfile < 0) + { + if ((info->dfile= my_open(fn_format(name_buff,info->filename,"", + N_NAME_DEXT,4),info->mode, + MYF(MY_WME))) < 0) + error = my_errno; + info->rec_cache.file=info->dfile; + } + } +#endif +#ifndef NO_LOCKING + if (info->was_locked) + { + if (nisam_lock_database(info, info->was_locked)) + error=my_errno; + info->was_locked=0; + } +#else + { + int lock_type,w_locks; + lock_type=info->lock_type ; w_locks=info->s->w_locks; + info->lock_type=0; info->s->w_locks=0; + if (_nisam_readinfo(info,0,1)) /* Read changed data */ + error=my_errno; + info->lock_type=lock_type; info->s->w_locks=w_locks; + } + /* Don't use buffer when doing next */ + info->update|=HA_STATE_WRITTEN; +#endif /* NO_LOCKING */ + break; + } + } + if (flag == HA_PANIC_CLOSE) + VOID(nisam_log(0)); /* Close log if neaded */ + pthread_mutex_unlock(&THR_LOCK_isam); + if (!error) DBUG_RETURN(0); + my_errno=error; + DBUG_RETURN(-1); +} /* nisam_panic */ diff --git a/isam/range.c b/isam/range.c new file mode 100644 index 00000000000..5594991cfc3 --- /dev/null +++ b/isam/range.c @@ -0,0 +1,191 @@ +/* Copyright (C) 2000 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 */ + +/* + Gives a approximated number of how many records there is between two keys. + Used when optimizing querries. + */ + +#include "isamdef.h" + +static ulong _nisam_record_pos(N_INFO *info,const byte *key,uint key_len, + enum ha_rkey_function search_flag); +static double _nisam_search_pos(N_INFO *info,N_KEYDEF *keyinfo,uchar *key, + uint key_len,uint nextflag,ulong pos); +static uint _nisam_keynr(N_INFO *info,N_KEYDEF *keyinfo,uchar *page, + uchar *keypos,uint *ret_max_key); + + + /* If start_key = 0 assume read from start */ + /* If end_key = 0 assume read to end */ + /* Returns NI_POS_ERROR on error */ + +ulong nisam_records_in_range(N_INFO *info, 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) +{ + ulong start_pos,end_pos; + DBUG_ENTER("nisam_records_in_range"); + + if ((inx = _nisam_check_index(info,inx)) < 0) + DBUG_RETURN(NI_POS_ERROR); + +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) + DBUG_RETURN(NI_POS_ERROR); +#endif + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + start_pos= (start_key ? + _nisam_record_pos(info,start_key,start_key_len,start_search_flag) : + 0L); + end_pos= (end_key ? + _nisam_record_pos(info,end_key,end_key_len,end_search_flag) : + info->s->state.records+1L); + VOID(_nisam_writeinfo(info,0)); + if (start_pos == NI_POS_ERROR || end_pos == NI_POS_ERROR) + DBUG_RETURN(NI_POS_ERROR); + DBUG_PRINT("info",("records: %ld",end_pos-start_pos)); + DBUG_RETURN(end_pos < start_pos ? 0L : + (end_pos == start_pos ? 1L : end_pos-start_pos)); +} + + + /* Find relative position (in records) for key in index-tree */ + +static ulong _nisam_record_pos(N_INFO *info, const byte *key, uint key_len, + enum ha_rkey_function search_flag) +{ + uint inx=(uint) info->lastinx; + N_KEYDEF *keyinfo=info->s->keyinfo+inx; + uchar *key_buff; + double pos; + + DBUG_ENTER("_nisam_record_pos"); + DBUG_PRINT("enter",("search_flag: %d",search_flag)); + + if (key_len >= (keyinfo->base.keylength-info->s->rec_reflength) + && !(keyinfo->base.flag & HA_SPACE_PACK_USED)) + key_len=USE_HOLE_KEY; + key_buff=info->lastkey+info->s->base.max_key_length; + key_len=_nisam_pack_key(info,inx,key_buff,(uchar*) key,key_len); + DBUG_EXECUTE("key",_nisam_print_key(DBUG_FILE,keyinfo->seg, + (uchar*) key_buff);); + pos=_nisam_search_pos(info,keyinfo,key_buff,key_len, + nisam_read_vec[search_flag] | SEARCH_SAVE_BUFF, + info->s->state.key_root[inx]); + if (pos >= 0.0) + { + DBUG_PRINT("exit",("pos: %ld",(ulong) (pos*info->s->state.records))); + DBUG_RETURN((ulong) (pos*info->s->state.records+0.5)); + } + DBUG_RETURN(NI_POS_ERROR); +} + + + /* This is a modified version of _nisam_search */ + /* Returns offset for key in indextable (decimal 0.0 <= x <= 1.0) */ + +static double _nisam_search_pos(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *key, uint key_len, uint nextflag, + register ulong pos) +{ + int flag; + uint nod_flag,keynr,max_keynr; + uchar *keypos,*buff; + double offset; + DBUG_ENTER("_nisam_search_pos"); + + if (pos == NI_POS_ERROR) + DBUG_RETURN(0.5); + + if (!(buff=_nisam_fetch_keypage(info,keyinfo,pos,info->buff,1))) + goto err; + flag=(*keyinfo->bin_search)(info,keyinfo,buff,key,key_len,nextflag, + &keypos,info->lastkey); + nod_flag=test_if_nod(buff); + keynr=_nisam_keynr(info,keyinfo,buff,keypos,&max_keynr); + + if (flag) + { + /* + ** Didn't found match. keypos points at next (bigger) key + * Try to find a smaller, better matching key. + ** Matches keynr + [0-1] + */ + if ((offset=_nisam_search_pos(info,keyinfo,key,key_len,nextflag, + _nisam_kpos(nod_flag,keypos))) < 0) + DBUG_RETURN(offset); + } + else + { + /* + ** Found match. Keypos points at the start of the found key + ** Matches keynr+1 + */ + offset=1.0; /* Matches keynr+1 */ + if (nextflag & SEARCH_FIND && (!(keyinfo->base.flag & HA_NOSAME) + || key_len) && nod_flag) + { + /* + ** There may be identical keys in the tree. Try to match on of those. + ** Matches keynr + [0-1] + */ + if ((offset=_nisam_search_pos(info,keyinfo,key,key_len,SEARCH_FIND, + _nisam_kpos(nod_flag,keypos))) < 0) + DBUG_RETURN(offset); /* Read error */ + } + } + DBUG_PRINT("info",("keynr: %d offset: %g max_keynr: %d nod: %d flag: %d", + keynr,offset,max_keynr,nod_flag,flag)); + DBUG_RETURN((keynr+offset)/(max_keynr+1)); +err: + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1.0); +} + + + /* Get keynummer of current key and max number of keys in nod */ + +static uint _nisam_keynr(N_INFO *info, register N_KEYDEF *keyinfo, uchar *page, uchar *keypos, uint *ret_max_key) +{ + uint nod_flag,keynr,max_key; + uchar t_buff[N_MAX_KEY_BUFF],*end; + + end= page+getint(page); + nod_flag=test_if_nod(page); + page+=2+nod_flag; + + if (!(keyinfo->base.flag & + (HA_PACK_KEY | HA_SPACE_PACK | HA_SPACE_PACK_USED))) + { + *ret_max_key= (uint) (end-page)/(keyinfo->base.keylength+nod_flag); + return (uint) (keypos-page)/(keyinfo->base.keylength+nod_flag); + } + + max_key=keynr=0; + while (page < end) + { + t_buff[0]=0; /* Don't move packed key */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&page,t_buff)); + max_key++; + if (page == keypos) + keynr=max_key; + } + *ret_max_key=max_key; + return(keynr); +} diff --git a/isam/rfirst.c b/isam/rfirst.c new file mode 100644 index 00000000000..82fd3994bdf --- /dev/null +++ b/isam/rfirst.c @@ -0,0 +1,34 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser f|rsta posten som har samma isam-nyckel */ + +#include "isamdef.h" + + /* + L{ser f|rsta posten med samma isamnyckel som f|reg}ende l{sning. + Man kan ha gjort write, update eller delete p} f|reg}ende post. + OBS! [ven om man {ndrade isamnyckeln p} f|reg}ende post l{ses + posten i avseende p} f|reg}ende isam-nyckel-l{sning !! + */ + +int nisam_rfirst(N_INFO *info, byte *buf, int inx) +{ + DBUG_ENTER("nisam_rfirst"); + info->lastpos= NI_POS_ERROR; + info->update|= HA_STATE_PREV_FOUND; + DBUG_RETURN(nisam_rnext(info,buf,inx)); +} /* nisam_rfirst */ diff --git a/isam/rkey.c b/isam/rkey.c new file mode 100644 index 00000000000..8f1f2f11ab5 --- /dev/null +++ b/isam/rkey.c @@ -0,0 +1,63 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser p} basen av en isam_nyckel */ + +#include "isamdef.h" + + + /* Read a record using key */ + /* Ordinary search_flag is 0 ; Give error if no record with key */ + +int nisam_rkey(N_INFO *info, byte *buf, int inx, const byte *key, uint key_len, enum ha_rkey_function search_flag) +{ + uchar *key_buff; + ISAM_SHARE *share=info->s; + DBUG_ENTER("nisam_rkey"); + DBUG_PRINT("enter",("base: %lx inx: %d search_flag: %d", + info,inx,search_flag)); + + if ((inx = _nisam_check_index(info,inx)) < 0) + DBUG_RETURN(-1); + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if (key_len >= (share->keyinfo[inx].base.keylength - share->rec_reflength) + && !(info->s->keyinfo[inx].base.flag & HA_SPACE_PACK_USED)) + key_len=USE_HOLE_KEY; + key_buff=info->lastkey+info->s->base.max_key_length; + key_len=_nisam_pack_key(info,(uint) inx,key_buff,(uchar*) key,key_len); + DBUG_EXECUTE("key",_nisam_print_key(DBUG_FILE,share->keyinfo[inx].seg, + (uchar*) key);); + +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) + goto err; +#endif + + VOID(_nisam_search(info,info->s->keyinfo+inx,key_buff,key_len, + nisam_read_vec[search_flag],info->s->state.key_root[inx])); + if ((*info->read_record)(info,info->lastpos,buf) >= 0) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + + info->lastpos = NI_POS_ERROR; /* Didn't find key */ + VOID(_nisam_move_key(info->s->keyinfo+inx,info->lastkey,key_buff)); + if (search_flag == HA_READ_AFTER_KEY) + info->update|=HA_STATE_NEXT_FOUND; /* Previous gives last row */ +err: + DBUG_RETURN(-1); +} /* nisam_rkey */ diff --git a/isam/rlast.c b/isam/rlast.c new file mode 100644 index 00000000000..df2b1bc39af --- /dev/null +++ b/isam/rlast.c @@ -0,0 +1,34 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser sista posten som har samma isam-nyckel */ + +#include "isamdef.h" + + /* + L{ser sista posten med samma isamnyckel som f|reg}ende l{sning. + Man kan ha gjort write, update eller delete p} f|reg}ende post. + OBS! [ven om man {ndrade isamnyckeln p} f|reg}ende post l{ses + posten i avseende p} f|reg}ende isam-nyckel-l{sning !! + */ + +int nisam_rlast(N_INFO *info, byte *buf, int inx) +{ + DBUG_ENTER("nisam_rlast"); + info->lastpos= NI_POS_ERROR; + info->update|= HA_STATE_NEXT_FOUND; + DBUG_RETURN(nisam_rprev(info,buf,inx)); +} /* nisam_rlast */ diff --git a/isam/rnext.c b/isam/rnext.c new file mode 100644 index 00000000000..ebe00fc2fa1 --- /dev/null +++ b/isam/rnext.c @@ -0,0 +1,66 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser n{sta post med samma isam-nyckel */ + +#include "isamdef.h" + + /* + L{ser n{sta post med samma isamnyckel som f|reg}ende l{sning. + Man kan ha gjort write, update eller delete p} f|reg}ende post. + OBS! [ven om man {ndrade isamnyckeln p} f|reg}ende post l{ses + posten i avseende p} f|reg}ende isam-nyckel-l{sning !! + */ + +int nisam_rnext(N_INFO *info, byte *buf, int inx) +{ + int error; + uint flag; + DBUG_ENTER("nisam_rnext"); + + if ((inx = _nisam_check_index(info,inx)) < 0) + DBUG_RETURN(-1); + flag=SEARCH_BIGGER; /* Read next */ + if (info->lastpos == NI_POS_ERROR && info->update & HA_STATE_PREV_FOUND) + flag=0; /* Read first */ + +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) DBUG_RETURN(-1); +#endif + if (!flag) + error=_nisam_search_first(info,info->s->keyinfo+inx, + info->s->state.key_root[inx]); + else if (_nisam_test_if_changed(info) == 0) + error=_nisam_search_next(info,info->s->keyinfo+inx,info->lastkey,flag, + info->s->state.key_root[inx]); + else + error=_nisam_search(info,info->s->keyinfo+inx,info->lastkey,0,flag, + info->s->state.key_root[inx]); + + /* Don't clear if database-changed */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | + HA_STATE_BUFF_SAVED); + info->update|= HA_STATE_NEXT_FOUND; + + if (error && my_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_END_OF_FILE; + if ((*info->read_record)(info,info->lastpos,buf) >=0) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + DBUG_RETURN(-1); +} /* nisam_rnext */ diff --git a/isam/rprev.c b/isam/rprev.c new file mode 100644 index 00000000000..18b1e31502c --- /dev/null +++ b/isam/rprev.c @@ -0,0 +1,63 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser f|reg}ende post med samma isam-nyckel */ + +#include "isamdef.h" + + /* + L{ser f|reg}ende post med samma isamnyckel som f|reg}ende l{sning. + Man kan ha gjort write, update eller delete p} f|reg}ende post. + OBS! [ven om man {ndrade isamnyckeln p} f|reg}ende post l{ses + posten i avseende p} f|reg}ende isam-nyckel-l{sning !! + */ + +int nisam_rprev(N_INFO *info, byte *buf, int inx) +{ + int error; + register uint flag; + DBUG_ENTER("nisam_rprev"); + + if ((inx = _nisam_check_index(info,inx)) < 0) + DBUG_RETURN(-1); + flag=SEARCH_SMALLER; /* Read previous */ + if (info->lastpos == NI_POS_ERROR && info->update & HA_STATE_NEXT_FOUND) + flag=0; /* Read last */ + +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) DBUG_RETURN(-1); +#endif + if (!flag) + error=_nisam_search_last(info,info->s->keyinfo+inx,info->s->state.key_root[inx]); + else if (_nisam_test_if_changed(info) == 0) + error=_nisam_search_next(info,info->s->keyinfo+inx,info->lastkey,flag, + info->s->state.key_root[inx]); + else + error=_nisam_search(info,info->s->keyinfo+inx,info->lastkey,0,flag, + info->s->state.key_root[inx]); + + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | + HA_STATE_BUFF_SAVED); + info->update|= HA_STATE_PREV_FOUND; + if (error && my_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_END_OF_FILE; + if ((*info->read_record)(info,info->lastpos,buf) >=0) + { + info->update|= HA_STATE_AKTIV; /* Record is read */ + DBUG_RETURN(0); + } + DBUG_RETURN(-1); +} /* nisam_rprev */ diff --git a/isam/rrnd.c b/isam/rrnd.c new file mode 100644 index 00000000000..7fd197a6d58 --- /dev/null +++ b/isam/rrnd.c @@ -0,0 +1,55 @@ +/* Copyright (C) 2000 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 */ + +/* Read a record with random-access. The position to the record must + get by N_INFO. The next record can be read with pos= -1 */ + + +#include "isamdef.h" + +/* + If filepos == NI_POS_ERROR, read next + Returns: + 0 = Ok. + 1 = Row was deleted + -1 = EOF (check errno to verify) +*/ + +int nisam_rrnd(N_INFO *info, byte *buf, register ulong filepos) +{ + int skipp_deleted_blocks; + DBUG_ENTER("nisam_rrnd"); + + skipp_deleted_blocks=0; + + if (filepos == NI_POS_ERROR) + { + skipp_deleted_blocks=1; + if (info->lastpos == NI_POS_ERROR) /* First read ? */ + filepos= info->s->pack.header_length; /* Read first record */ + else + filepos= info->nextpos; + } + + info->lastinx= -1; /* Can't forward or backward */ + /* Init all but update-flag */ + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + if (info->opt_flag & WRITE_CACHE_USED && flush_io_cache(&info->rec_cache)) + DBUG_RETURN(my_errno); + + DBUG_RETURN ((*info->s->read_rnd)(info,buf,filepos,skipp_deleted_blocks)); +} diff --git a/isam/rsame.c b/isam/rsame.c new file mode 100644 index 00000000000..fe617cf258c --- /dev/null +++ b/isam/rsame.c @@ -0,0 +1,70 @@ +/* Copyright (C) 2000 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 */ + +/* L{ser nuvarande record med direktl{sning */ +/* Klarar b}de poster l{sta med nyckel och rrnd. */ + +#include "isamdef.h" + + /* Funktionen ger som resultat: + 0 = Ok. + 1 = Posten borttagen + -1 = EOF (eller motsvarande: se errno) */ + + +int nisam_rsame(N_INFO *info, byte *record, int inx) + + + /* If inx >= 0 find record using key */ +{ + DBUG_ENTER("nisam_rsame"); + + if (inx >= (int) info->s->state.keys || inx < -1) + { + my_errno=HA_ERR_WRONG_INDEX; + DBUG_RETURN(-1); + } + if (info->lastpos == NI_POS_ERROR || info->update & HA_STATE_DELETED) + { + my_errno=HA_ERR_KEY_NOT_FOUND; /* No current record */ + DBUG_RETURN(-1); + } + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + + /* L{s record fr}n datafilen */ + +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_RDLCK,1)) + DBUG_RETURN(-1); +#endif + + if (inx >= 0) + { + info->lastinx=inx; + VOID(_nisam_make_key(info,(uint) inx,info->lastkey,record,info->lastpos)); + VOID(_nisam_search(info,info->s->keyinfo+inx,info->lastkey,0,SEARCH_SAME, + info->s->state.key_root[inx])); + } + + if ((*info->read_record)(info,info->lastpos,record) == 0) + DBUG_RETURN(0); + if (my_errno == HA_ERR_RECORD_DELETED) + { + my_errno=HA_ERR_KEY_NOT_FOUND; + DBUG_RETURN(1); + } + DBUG_RETURN(-1); +} /* nisam_rsame */ diff --git a/isam/rsamepos.c b/isam/rsamepos.c new file mode 100644 index 00000000000..500dfc60e38 --- /dev/null +++ b/isam/rsamepos.c @@ -0,0 +1,59 @@ +/* Copyright (C) 2000 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 */ + +/* read record through position and fix key-position */ +/* As nisam_rsame but supply a position */ + +#include "isamdef.h" + + + /* + ** If inx >= 0 update index pointer + ** Returns one of the following values: + ** 0 = Ok. + ** 1 = Record deleted + ** -1 = EOF (or something similar. More information in my_errno) + */ + +int nisam_rsame_with_pos(N_INFO *info, byte *record, int inx, ulong filepos) +{ + DBUG_ENTER("nisam_rsame_with_pos"); + + if (inx >= (int) info->s->state.keys || inx < -1) + { + my_errno=HA_ERR_WRONG_INDEX; + DBUG_RETURN(-1); + } + + info->update&= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED); + if ((*info->s->read_rnd)(info,record,filepos,0)) + { + if (my_errno == HA_ERR_RECORD_DELETED) + { + my_errno=HA_ERR_KEY_NOT_FOUND; + DBUG_RETURN(1); + } + DBUG_RETURN(-1); + } + info->lastpos=filepos; + info->lastinx=inx; + if (inx >= 0) + { + VOID(_nisam_make_key(info,(uint) inx,info->lastkey,record,info->lastpos)); + info->update|=HA_STATE_KEY_CHANGED; /* Don't use indexposition */ + } + DBUG_RETURN(0); +} /* nisam_rsame_pos */ diff --git a/isam/sort.c b/isam/sort.c new file mode 100644 index 00000000000..72c4c7564f8 --- /dev/null +++ b/isam/sort.c @@ -0,0 +1,558 @@ +/* Copyright (C) 2000 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 */ + +/* + Creates a index for a database by reading keys, sorting them and outputing + them in sorted order through SORT_INFO functions. +*/ + +#include "isamdef.h" +#if defined(MSDOS) || defined(__WIN__) +#include <fcntl.h> +#else +#include <stddef.h> +#endif +#include <queues.h> + + /* static variabels */ + +#define MERGEBUFF 15 +#define MERGEBUFF2 31 +#define MIN_SORT_MEMORY (4096-MALLOC_OVERHEAD) +#define MYF_RW MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL) + +typedef struct st_buffpek { /* Struktur om sorteringsbuffrarna */ + my_off_t file_pos; /* Position var bufferten finns */ + ulong count; /* Antal nycklar i bufferten */ + uchar *base,*key; /* Pekare inom sort_key - indexdel */ + uint mem_count; /* Antal nycklar kvar i minnet */ + uint max_keys; /* Max keys in buffert */ +} BUFFPEK; + +extern void print_error _VARARGS((const char *fmt,...)); + + /* functions defined in this file */ + +static ulong NEAR_F find_all_keys(SORT_PARAM *info,uint keys, + uchar * *sort_keys, + BUFFPEK *buffpek,int *maxbuffer, + FILE **tempfile, my_string tempname); +static int NEAR_F write_keys(SORT_PARAM *info,uchar * *sort_keys, + uint count, BUFFPEK *buffpek,FILE **tempfile, + my_string tempname); +static int NEAR_F write_index(SORT_PARAM *info,uchar * *sort_keys, + uint count); +static int NEAR_F merge_many_buff(SORT_PARAM *info,uint keys, + uchar * *sort_keys, + BUFFPEK *buffpek,int *maxbuffer, + FILE * *t_file, my_string tempname); +static uint NEAR_F read_to_buffer(FILE *fromfile,BUFFPEK *buffpek, + uint sort_length); +static int NEAR_F merge_buffers(SORT_PARAM *info,uint keys,FILE *from_file, + FILE *to_file, uchar * *sort_keys, + BUFFPEK *lastbuff,BUFFPEK *Fb, + BUFFPEK *Tb); +static int NEAR_F merge_index(SORT_PARAM *,uint,uchar **,BUFFPEK *, int, + FILE *); +static char **make_char_array(uint fields,uint length,myf my_flag); +static FILE *opentemp(my_string name); +static void closetemp(char *name,FILE *stream); + + + /* Creates a index of sorted keys */ + /* Returns 0 if everything went ok */ + +int _create_index_by_sort(info,no_messages,sortbuff_size) +SORT_PARAM *info; +pbool no_messages; +uint sortbuff_size; +{ + int error,maxbuffer,skr; + uint memavl,old_memavl,keys,sort_length; + BUFFPEK *buffpek; + char tempname[FN_REFLEN]; + ulong records; + uchar **sort_keys; + FILE *tempfile; + DBUG_ENTER("_create_index_by_sort"); + + tempfile=0; buffpek= (BUFFPEK *) NULL; sort_keys= (uchar **) NULL; error= 1; + maxbuffer=1; + + memavl=max(sortbuff_size,MIN_SORT_MEMORY); + records= info->max_records; + sort_length= info->key_length; + LINT_INIT(keys); + + while (memavl >= MIN_SORT_MEMORY) + { + if ((records+1)*(sort_length+sizeof(char*)) < (ulong) memavl) + keys= records+1; + else + do + { + skr=maxbuffer; + if (memavl < sizeof(BUFFPEK)*(uint) maxbuffer || + (keys=(memavl-sizeof(BUFFPEK)*(uint) maxbuffer)/ + (sort_length+sizeof(char*))) <= 1) + { + print_error("Sortbuffer to small"); + goto err; + } + } + while ((maxbuffer= (int) (records/(keys-1)+1)) != skr); + + if ((sort_keys= (uchar **) make_char_array(keys,sort_length,MYF(0)))) + { + if ((buffpek = (BUFFPEK*) my_malloc((uint) (sizeof(BUFFPEK)* + (uint) maxbuffer), + MYF(0)))) + break; + else + my_free((gptr) sort_keys,MYF(0)); + } + old_memavl=memavl; + if ((memavl=memavl/4*3) < MIN_SORT_MEMORY && old_memavl > MIN_SORT_MEMORY) + memavl=MIN_SORT_MEMORY; + } + if (memavl < MIN_SORT_MEMORY) + { + print_error("Sortbuffer to small"); + goto err; + } + (*info->lock_in_memory)(); /* Everything is allocated */ + + if (!no_messages) + printf(" - Searching for keys, allocating buffer for %d keys\n",keys); + + if ((records=find_all_keys(info,keys,sort_keys,buffpek,&maxbuffer,&tempfile, + tempname)) + == (ulong) -1) + goto err; + if (maxbuffer == 0) + { + if (!no_messages) + printf(" - Dumping %lu keys\n",records); + if (write_index(info,sort_keys,(uint) records)) + goto err; + } + else + { + keys=(keys*(sort_length+sizeof(char*)))/sort_length; + if (maxbuffer >= MERGEBUFF2) + { + if (!no_messages) + printf(" - Merging %lu keys\n",records); + if (merge_many_buff(info,keys,sort_keys,buffpek,&maxbuffer,&tempfile, + tempname)) + goto err; + } + if (!no_messages) + puts(" - Last merge and dumping keys"); + if (merge_index(info,keys,sort_keys,buffpek,maxbuffer,tempfile)) + goto err; + } + error =0; + +err: + if (sort_keys) + my_free((gptr) sort_keys,MYF(0)); + if (buffpek) + my_free((gptr) buffpek,MYF(0)); + if (tempfile) + closetemp(tempname,tempfile); + + DBUG_RETURN(error ? -1 : 0); +} /* _create_index_by_sort */ + + + /* Search after all keys and place them in a temp. file */ + +static ulong NEAR_F find_all_keys(info,keys,sort_keys,buffpek,maxbuffer, + tempfile,tempname) +SORT_PARAM *info; +uint keys; +uchar **sort_keys; +BUFFPEK *buffpek; +int *maxbuffer; +FILE **tempfile; +my_string tempname; +{ + int error; + uint index,indexpos; + DBUG_ENTER("find_all_keys"); + + index=indexpos=error=0; + + while (!(error=(*info->key_read)(sort_keys[index]))) + { + if ((uint) ++index == keys) + { + if (indexpos >= (uint) *maxbuffer || + write_keys(info,sort_keys,index-1,buffpek+indexpos,tempfile, + tempname)) + DBUG_RETURN(NI_POS_ERROR); + memcpy(sort_keys[0],sort_keys[index-1],(size_t) info->key_length); + index=1; indexpos++; + } + } + if (error > 0) + DBUG_RETURN(NI_POS_ERROR); /* Aborted by get_key */ + if (indexpos) + if (indexpos >= (uint) *maxbuffer || + write_keys(info,sort_keys,index,buffpek+indexpos,tempfile,tempname)) + DBUG_RETURN(NI_POS_ERROR); + *maxbuffer=(int) indexpos; + DBUG_RETURN(indexpos*(keys-1)+index); +} /* find_all_keys */ + + + /* Write all keys in memory to file for later merge */ + +static int NEAR_F write_keys(info,sort_keys,count,buffpek,tempfile,tempname) +SORT_PARAM *info; +reg1 uchar **sort_keys; +uint count; +BUFFPEK *buffpek; +reg2 FILE **tempfile; +my_string tempname; +{ + DBUG_ENTER("write_keys"); + + qsort2((byte*) sort_keys,count,sizeof(byte*),(qsort2_cmp) info->key_cmp, + NullS); + if (! *tempfile && ! (*tempfile=opentemp(tempname))) + DBUG_RETURN(1); + buffpek->file_pos=my_ftell(*tempfile,MYF(0)); + buffpek->count=count; + while (count--) + if (my_fwrite(*tempfile,(byte*)*sort_keys++,info->key_length,MYF_RW)) + DBUG_RETURN(1); + DBUG_RETURN(0); +} /* write_keys */ + + + /* Write index */ + +static int NEAR_F write_index(info,sort_keys,count) +SORT_PARAM *info; +reg1 uchar **sort_keys; +reg2 uint count; +{ + DBUG_ENTER("write_index"); + + qsort2((gptr) sort_keys,(size_t) count,sizeof(byte*), + (qsort2_cmp) info->key_cmp, NullS); + while (count--) + if ((*info->key_write)(*sort_keys++)) + DBUG_RETURN(-1); + DBUG_RETURN(0); +} /* write_index */ + + + /* Merge buffers to make < MERGEBUFF2 buffers */ + +static int NEAR_F merge_many_buff(info,keys,sort_keys,buffpek,maxbuffer,t_file, + t_name) +SORT_PARAM *info; +uint keys; +uchar **sort_keys; +int *maxbuffer; +BUFFPEK *buffpek; +FILE **t_file; +my_string t_name; +{ + register int i; + FILE *from_file,*to_file,*temp; + FILE *t_file2; + char t_name2[FN_REFLEN]; + BUFFPEK *lastbuff; + DBUG_ENTER("merge_many_buff"); + + if (!(t_file2=opentemp(t_name2))) + DBUG_RETURN(1); + + from_file= *t_file ; to_file= t_file2; + while (*maxbuffer >= MERGEBUFF2) + { + lastbuff=buffpek; + for (i=0 ; i <= *maxbuffer-MERGEBUFF*3/2 ; i+=MERGEBUFF) + { + if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, + buffpek+i,buffpek+i+MERGEBUFF-1)) + break; + } + if (merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff++, + buffpek+i,buffpek+ *maxbuffer)) + break; + *maxbuffer= (int) (lastbuff-buffpek)-1; + temp=from_file; from_file=to_file; to_file=temp; + VOID(my_fseek(to_file,0L,MY_SEEK_SET,MYF(0))); + } + if (to_file == *t_file) + { + closetemp(t_name,to_file); + *t_file=t_file2; + VOID(strmov(t_name,t_name2)); + } + else closetemp(t_name2,to_file); + + DBUG_RETURN(*maxbuffer >= MERGEBUFF2); /* Return 1 if interrupted */ +} /* merge_many_buff */ + + + /* Read data to buffer */ + /* This returns (uint) -1 if something goes wrong */ + +static uint NEAR_F read_to_buffer(fromfile,buffpek,sort_length) +FILE *fromfile; +BUFFPEK *buffpek; +uint sort_length; +{ + register uint count; + uint length; + + if ((count=(uint) min((ulong) buffpek->max_keys,buffpek->count))) + { + VOID(my_fseek(fromfile,buffpek->file_pos,MY_SEEK_SET,MYF(0))); + if (my_fread(fromfile,(byte*) buffpek->base, + (length= sort_length*count),MYF_RW)) + return((uint) -1); + buffpek->key=buffpek->base; + buffpek->file_pos+= length; /* New filepos */ + buffpek->count-= count; + buffpek->mem_count= count; + } + return (count*sort_length); +} /* read_to_buffer */ + + + /* Merge buffers to one buffer */ + /* If to_file == 0 then use info->key_write */ + +static int NEAR_F merge_buffers(info,keys,from_file,to_file,sort_keys,lastbuff, + Fb,Tb) +SORT_PARAM *info; +uint keys; +FILE *from_file,*to_file; +uchar **sort_keys; +BUFFPEK *lastbuff,*Fb,*Tb; +{ + int error; + uint sort_length,maxcount; + ulong count; + my_off_t to_start_filepos; + uchar *strpos; + BUFFPEK *buffpek,**refpek; + QUEUE queue; + DBUG_ENTER("merge_buffers"); + + count=error=0; + maxcount=keys/((uint) (Tb-Fb) +1); + sort_length=info->key_length; + + LINT_INIT(to_start_filepos); + if (to_file) + to_start_filepos=my_ftell(to_file,MYF(0)); + strpos=(uchar*) sort_keys; + + if (init_queue(&queue,(uint) (Tb-Fb)+1,offsetof(BUFFPEK,key),0, + (int (*)(void *, byte *,byte *)) info->key_cmp,0)) + DBUG_RETURN(1); + + for (buffpek= Fb ; buffpek <= Tb && error != -1 ; buffpek++) + { + count+= buffpek->count; + buffpek->base= strpos; + buffpek->max_keys=maxcount; + strpos+= (uint) (error=(int) read_to_buffer(from_file,buffpek, + sort_length)); + queue_insert(&queue,(void*) buffpek); + } + if (error == -1) + goto err; + + while (queue.elements > 1) + { + for (;;) + { + buffpek=(BUFFPEK*) queue_top(&queue); + if (to_file) + { + if (my_fwrite(to_file,(byte*) buffpek->key,(uint) sort_length, + MYF_RW | MY_WAIT_IF_FULL)) + { + error=1; goto err; + } + } + else + { + if ((*info->key_write)((void*) buffpek->key)) + { + error=1; goto err; + } + } + buffpek->key+=sort_length; + if (! --buffpek->mem_count) + { + if (!(error=(int) read_to_buffer(from_file,buffpek,sort_length))) + { + uchar *base=buffpek->base; + uint max_keys=buffpek->max_keys; + + VOID(queue_remove(&queue,0)); + + /* Put room used by buffer to use in other buffer */ + for (refpek= (BUFFPEK**) &queue_top(&queue); + refpek <= (BUFFPEK**) &queue_end(&queue); + refpek++) + { + buffpek= *refpek; + if (buffpek->base+buffpek->max_keys*sort_length == base) + { + buffpek->max_keys+=max_keys; + break; + } + else if (base+max_keys*sort_length == buffpek->base) + { + buffpek->base=base; + buffpek->max_keys+=max_keys; + break; + } + } + break; /* One buffer have been removed */ + } + } + queue_replaced(&queue); /* Top element has been replaced */ + } + } + buffpek=(BUFFPEK*) queue_top(&queue); + buffpek->base=(uchar *) sort_keys; + buffpek->max_keys=keys; + do + { + if (to_file) + { + if (my_fwrite(to_file,(byte*) buffpek->key, + (uint) (sort_length*buffpek->mem_count), + MYF_RW | MY_WAIT_IF_FULL)) + { + error=1; goto err; + } + } + else + { + register uchar *end; + strpos= buffpek->key; + for (end=strpos+buffpek->mem_count*sort_length; + strpos != end ; + strpos+=sort_length) + { + if ((*info->key_write)((void*) strpos)) + { + error=1; goto err; + } + } + } + } + while ((error=(int) read_to_buffer(from_file,buffpek,sort_length)) != -1 && + error != 0); + + lastbuff->count=count; + if (to_file) + lastbuff->file_pos=to_start_filepos; /* New block starts here */ +err: + delete_queue(&queue); + DBUG_RETURN(error); +} /* merge_buffers */ + + + /* Do a merge to output-file (save only positions) */ + +static int NEAR_F merge_index(info,keys,sort_keys,buffpek,maxbuffer,tempfile) +SORT_PARAM *info; +uint keys; +uchar **sort_keys; +BUFFPEK *buffpek; +int maxbuffer; +FILE *tempfile; +{ + DBUG_ENTER("merge_index"); + if (merge_buffers(info,keys,tempfile,(FILE*) 0,sort_keys,buffpek,buffpek, + buffpek+maxbuffer)) + DBUG_RETURN(1); + DBUG_RETURN(0); +} /* merge_index */ + + + /* Make a pointer of arrays to keys */ + +static char **make_char_array(fields,length,my_flag) +register uint fields; +uint length; +myf my_flag; +{ + register char **pos; + char **old_pos,*char_pos; + DBUG_ENTER("make_char_array"); + + if ((old_pos= (char**) my_malloc( fields*(length+sizeof(char*)), my_flag))) + { + pos=old_pos; char_pos=((char*) (pos+fields)) -length; + while (fields--) + *(pos++) = (char_pos+= length); + } + + DBUG_RETURN(old_pos); +} /* make_char_array */ + + + /* |ppnar en tempor{rfil som kommer att raderas efter anv{nding */ + +static FILE *opentemp(name) +my_string name; +{ + FILE *stream; + reg1 my_string str_pos; + DBUG_ENTER("opentemp"); + + if (!(str_pos=my_tempnam(NullS,"ST",MYF(MY_WME)))) + DBUG_RETURN(0); + VOID(strmov(name,str_pos)); + (*free)(str_pos); /* Inte via vanliga malloc */ + + stream=my_fopen(name,(int) (O_RDWR | FILE_BINARY | O_CREAT | O_TEMPORARY), + MYF(MY_WME)); +#if O_TEMPORARY == 0 && !defined(CANT_DELETE_OPEN_FILES) + VOID(my_delete(name,MYF(MY_WME | ME_NOINPUT))); +#endif + DBUG_PRINT("exit",("stream: %lx",stream)); + DBUG_RETURN (stream); +} /* opentemp */ + + +static void closetemp(char *name __attribute__((unused)) ,FILE *stream) +{ + DBUG_ENTER("closetemp"); + + if (stream) + VOID(my_fclose(stream,MYF(MY_WME))); +#ifdef CANT_DELETE_OPEN_FILES + if (name) + VOID(my_delete(name,MYF(MY_WME))); +#endif + DBUG_VOID_RETURN; +} /* closetemp */ diff --git a/isam/static.c b/isam/static.c new file mode 100644 index 00000000000..941c4defea2 --- /dev/null +++ b/isam/static.c @@ -0,0 +1,45 @@ +/* Copyright (C) 2000 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 */ + +/* + Static variables for pisam library. All definied here for easy making of + a shared library +*/ + +#ifndef _global_h +#include "isamdef.h" +#endif + +LIST *nisam_open_list=0; +uchar NEAR nisam_file_magic[]= +{ (uchar) 254, (uchar) 254,'\005', '\002', }; +uchar NEAR nisam_pack_file_magic[]= +{ (uchar) 254, (uchar) 254,'\006', '\001', }; +my_string nisam_log_filename= (char*) "isam.log"; +File nisam_log_file= -1; +uint nisam_quick_table_bits=9; +uint nisam_block_size=1024; /* Best by test */ +my_bool nisam_flush=0; + +/* read_vec[] is used for converting between P_READ_KEY.. and SEARCH_ */ +/* Position is , == , >= , <= , > , < */ + +uint NEAR nisam_read_vec[]= +{ + SEARCH_FIND, SEARCH_FIND | SEARCH_BIGGER, SEARCH_FIND | SEARCH_SMALLER, + SEARCH_NO_FIND | SEARCH_BIGGER, SEARCH_NO_FIND | SEARCH_SMALLER, + SEARCH_FIND, SEARCH_LAST +}; diff --git a/isam/test1.c b/isam/test1.c new file mode 100644 index 00000000000..1ec5d8b0318 --- /dev/null +++ b/isam/test1.c @@ -0,0 +1,183 @@ +/* Copyright (C) 2000 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 */ + +#include "isamdef.h" + +static void get_options(int argc, char *argv[]); + +static int rec_pointer_size=0,verbose=0,remove_ant=0,pack_keys=1,flags[50], + packed_field=FIELD_SKIPP_PRESPACE; + +int main(argc,argv) +int argc; +char *argv[]; +{ + N_INFO *file; + int i,j,error,deleted,found; + char record[128],key[32],*filename,read_record[128]; + N_KEYDEF keyinfo[10]; + N_RECINFO recinfo[10]; + MY_INIT(argv[0]); + + filename= (char*) "test1"; + my_init(); + get_options(argc,argv); + keyinfo[0].seg[0].base.type=HA_KEYTYPE_NUM; + keyinfo[0].seg[0].base.flag=(uint8) (pack_keys ? + HA_PACK_KEY | HA_SPACE_PACK : 0); + keyinfo[0].seg[0].base.start=0; + keyinfo[0].seg[0].base.length=6; + keyinfo[0].seg[1].base.type=HA_KEYTYPE_END; + keyinfo[0].base.flag = (uint8) (pack_keys ? + HA_NOSAME | HA_PACK_KEY : HA_NOSAME); + + recinfo[0].base.type=packed_field; recinfo[0].base.length=6; + recinfo[1].base.type=FIELD_NORMAL; recinfo[1].base.length=24; + recinfo[2].base.type=FIELD_LAST; + + deleted=0; + bzero((byte*) flags,sizeof(flags)); + + printf("- Creating isam-file\n"); + if (nisam_create(filename,1,keyinfo,recinfo, + (ulong) (rec_pointer_size ? (1L << (rec_pointer_size*8))/40 : + 0),10l,0,0,0L)) + goto err; + if (!(file=nisam_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + printf("- Writing key:s\n"); + strmov(record," ..... key"); strappend(record,30,' '); + + my_errno=0; + for (i=49 ; i>=1 ; i-=2 ) + { + j=i%25 +1; + sprintf(key,"%6d",j); + bmove(record,key,6); + error=nisam_write(file,record); + flags[j]=1; + if (verbose || error) + printf("J= %2d nisam_write: %d errno: %d\n", j,error,my_errno); + } + if (nisam_close(file)) goto err; + printf("- Reopening file\n"); + if (!(file=nisam_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) goto err; + printf("- Removing keys\n"); + for (i=1 ; i<=10 ; i++) + { + if (i == remove_ant) { VOID(nisam_close(file)) ; exit(0) ; } + sprintf(key,"%6d",(j=(int) ((rand() & 32767)/32767.*25))); + my_errno=0; + if ((error = nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT))) + { + if (verbose || (flags[j] == 1 || + (error && my_errno != HA_ERR_KEY_NOT_FOUND))) + printf("key: %s nisam_rkey: %3d errno: %3d\n",key,error,my_errno); + } + else + { + error=nisam_delete(file,read_record); + if (verbose || error) + printf("key: %s nisam_delete: %3d errno: %3d\n",key,error,my_errno); + flags[j]=0; + if (! error) + deleted++; + } + } + printf("- Reading records with key\n"); + for (i=1 ; i<=25 ; i++) + { + sprintf(key,"%6d",i); + bmove(record,key,6); + my_errno=0; + error=nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT); + if (verbose || + (error == 0 && flags[i] != 1) || + (error && (flags[i] != 0 || my_errno != HA_ERR_KEY_NOT_FOUND))) + { + printf("key: %s nisam_rkey: %3d errno: %3d record: %s\n", + key,error,my_errno,record+1); + } + } + + printf("- Reading records with position\n"); + for (i=1,found=0 ; i <= 30 ; i++) + { + my_errno=0; + if ((error=nisam_rrnd(file,read_record,i == 1 ? 0L : NI_POS_ERROR)) == -1) + { + if (found != 25-deleted) + printf("Found only %d of %d records\n",found,25-deleted); + break; + } + if (!error) + found++; + if (verbose || (error != 0 && error != 1)) + { + printf("pos: %2d nisam_rrnd: %3d errno: %3d record: %s\n", + i-1,error,my_errno,read_record+1); + } + } + if (nisam_close(file)) goto err; + my_end(MY_CHECK_ERROR); + + exit(0); +err: + printf("got error: %3d when using nisam-database\n",my_errno); + exit(1); + return 0; /* skipp warning */ +} /* main */ + + + /* l{ser optioner */ + /* OBS! intierar endast DEBUG - ingen debuggning h{r ! */ + +static void get_options(argc,argv) +int argc; +char *argv[]; +{ + char *pos; + + while (--argc >0 && *(pos = *(++argv)) == '-' ) { + switch(*++pos) { + case 'R': /* Length of record pointer */ + rec_pointer_size=atoi(++pos); + if (rec_pointer_size > 3) + rec_pointer_size=0; + break; + case 'P': + pack_keys=0; /* Don't use packed key */ + break; + case 'S': + packed_field=FIELD_NORMAL; /* static-size record*/ + break; + case 'v': /* verbose */ + verbose=1; + break; + case 'm': + remove_ant=atoi(++pos); + break; + case 'V': + printf("isamtest1 Ver 1.0 \n"); + exit(0); + case '#': + DEBUGGER_ON; + DBUG_PUSH (++pos); + break; + } + } + return; +} /* get options */ diff --git a/isam/test2.c b/isam/test2.c new file mode 100644 index 00000000000..6ed041ad8c5 --- /dev/null +++ b/isam/test2.c @@ -0,0 +1,852 @@ +/* Copyright (C) 2000 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 */ + +/* Test av isam-databas: stor test */ + +#ifndef USE_MY_FUNC /* We want to be able to dbug this !! */ +#define USE_MY_FUNC +#endif +#ifdef DBUG_OFF +#undef DBUG_OFF +#endif +#ifndef SAFEMALLOC +#define SAFEMALLOC +#endif + +#include "isamdef.h" + +#define STANDAR_LENGTH 37 +#define NISAM_KEYS 6 +#if !defined(MSDOS) && !defined(labs) +#define labs(a) abs(a) +#endif + +static void get_options(int argc, char *argv[]); +static uint rnd(uint max_value); +static void fix_length(byte *record,uint length); +static void put_blob_in_record(char *blob_pos,char **blob_buffer); +static void copy_key(struct st_isam_info *info,uint inx, + uchar *record,uchar *key); + +static int verbose=0,testflag=0,pack_type=HA_SPACE_PACK, + first_key=0,async_io=0,key_cacheing=0,write_cacheing=0,locking=0, + rec_pointer_size=0,pack_fields=1,use_log=0; +static uint keys=NISAM_KEYS,recant=1000; +static uint use_blob=0; +static uint16 key1[1000],key3[5000]; +static char record[300],record2[300],key[100],key2[100], + read_record[300],read_record2[300],read_record3[300]; + + + /* Test program */ + +int main(argc,argv) +int argc; +char *argv[]; +{ + uint i; + int j,n1,n2,n3,error,k; + uint write_count,update,dupp_keys,delete,start,length,blob_pos, + reclength,ant; + ulong lastpos,range_records,records; + N_INFO *file; + N_KEYDEF keyinfo[10]; + N_RECINFO recinfo[10]; + N_ISAMINFO info; + char *filename,*blob_buffer; + MY_INIT(argv[0]); + + filename= (char*) "test2.ISM"; + get_options(argc,argv); + if (! async_io) + my_disable_async_io=1; + + reclength=STANDAR_LENGTH+60+(use_blob ? 8 : 0); + blob_pos=STANDAR_LENGTH+60; + keyinfo[0].seg[0].base.start=0; + keyinfo[0].seg[0].base.length=6; + keyinfo[0].seg[0].base.type=HA_KEYTYPE_TEXT; + keyinfo[0].seg[0].base.flag=(uint8) pack_type; + keyinfo[0].seg[1].base.type=0; + keyinfo[0].base.flag = (uint8) (pack_type ? HA_PACK_KEY : 0); + keyinfo[1].seg[0].base.start=7; + keyinfo[1].seg[0].base.length=6; + keyinfo[1].seg[0].base.type=HA_KEYTYPE_BINARY; + keyinfo[1].seg[0].base.flag=0; + keyinfo[1].seg[1].base.start=0; /* Tv}delad nyckel */ + keyinfo[1].seg[1].base.length=6; + keyinfo[1].seg[1].base.type=HA_KEYTYPE_NUM; + keyinfo[1].seg[1].base.flag=HA_REVERSE_SORT; + keyinfo[1].seg[2].base.type=0; + keyinfo[1].base.flag =0; + keyinfo[2].seg[0].base.start=12; + keyinfo[2].seg[0].base.length=8; + keyinfo[2].seg[0].base.type=HA_KEYTYPE_BINARY; + keyinfo[2].seg[0].base.flag=HA_REVERSE_SORT; + keyinfo[2].seg[1].base.type=0; + keyinfo[2].base.flag =HA_NOSAME; + keyinfo[3].seg[0].base.start=0; + keyinfo[3].seg[0].base.length=reclength-(use_blob ? 8 : 0); + keyinfo[3].seg[0].base.type=HA_KEYTYPE_TEXT; + keyinfo[3].seg[0].base.flag=(uint8) pack_type; + keyinfo[3].seg[1].base.type=0; + keyinfo[3].base.flag = (uint8) (pack_type ? HA_PACK_KEY : 0); + keyinfo[4].seg[0].base.start=0; + keyinfo[4].seg[0].base.length=5; + keyinfo[4].seg[0].base.type=HA_KEYTYPE_TEXT; + keyinfo[4].seg[0].base.flag=0; + keyinfo[4].seg[1].base.type=0; + keyinfo[4].base.flag = (uint8) (pack_type ? HA_PACK_KEY : 0); + keyinfo[5].seg[0].base.start=0; + keyinfo[5].seg[0].base.length=4; + keyinfo[5].seg[0].base.type=HA_KEYTYPE_TEXT; + keyinfo[5].seg[0].base.flag=(uint8) pack_type; + keyinfo[5].seg[1].base.type=0; + keyinfo[5].base.flag = (uint8) (pack_type ? HA_PACK_KEY : 0); + + recinfo[0].base.type=pack_fields ? FIELD_SKIPP_PRESPACE : 0; + recinfo[0].base.length=7; + recinfo[1].base.type=pack_fields ? FIELD_SKIPP_PRESPACE : 0; + recinfo[1].base.length=5; + recinfo[2].base.type=pack_fields ? FIELD_SKIPP_PRESPACE : 0; + recinfo[2].base.length=9; + recinfo[3].base.type=FIELD_NORMAL; + recinfo[3].base.length=STANDAR_LENGTH-7-5-9-4; + recinfo[4].base.type=pack_fields ? FIELD_SKIPP_ZERO : 0; + recinfo[4].base.length=4; + recinfo[5].base.type=pack_fields ? FIELD_SKIPP_ENDSPACE : 0; + recinfo[5].base.length=60; + if (use_blob) + { + recinfo[6].base.type=FIELD_BLOB; + recinfo[6].base.length=4+sizeof(char*); /* 4 byte ptr, 4 byte length */ + recinfo[7].base.type= FIELD_LAST; + } + else + recinfo[6].base.type= FIELD_LAST; + + write_count=update=dupp_keys=delete=0; + blob_buffer=0; + + for (i=999 ; i>0 ; i--) key1[i]=0; + for (i=4999 ; i>0 ; i--) key3[i]=0; + + printf("- Creating isam-file\n"); + /* DBUG_PUSH(""); */ + my_delete(filename,MYF(0)); /* Remove old locks under gdb */ + file= 0; + if (nisam_create(filename,keys,&keyinfo[first_key],&recinfo[0], + (ulong) (rec_pointer_size ? (1L << (rec_pointer_size*8))/ + reclength : 0),100l,0,0,0L)) + goto err; + if (use_log) + nisam_log(1); + if (!(file=nisam_open(filename,2,HA_OPEN_ABORT_IF_LOCKED))) + goto err; + printf("- Writing key:s\n"); + if (key_cacheing) + init_key_cache(IO_SIZE*16,(uint) IO_SIZE*4*10); /* Use a small cache */ + if (locking) + nisam_lock_database(file,F_WRLCK); + if (write_cacheing) + nisam_extra(file,HA_EXTRA_WRITE_CACHE); + + for (i=0 ; i < recant ; i++) + { + n1=rnd(1000); n2=rnd(100); n3=rnd(5000); + sprintf(record,"%6d:%4d:%8d:Pos: %4d ",n1,n2,n3,write_count); + longstore(record+STANDAR_LENGTH-4,(long) i); + fix_length(record,(uint) STANDAR_LENGTH+rnd(60)); + put_blob_in_record(record+blob_pos,&blob_buffer); + DBUG_PRINT("test",("record: %d",i)); + + if (nisam_write(file,record)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY || key3[n3] == 0) + { + printf("Error: %d in write at record: %d\n",my_errno,i); + goto err; + } + if (verbose) printf(" Double key: %d\n",n3); + } + else + { + if (key3[n3] == 1 && first_key <3 && first_key+keys >= 3) + { + printf("Error: Didn't get error when writing second key: '%8d'\n",n3); + goto err; + } + write_count++; key1[n1]++; key3[n3]=1; + } + + /* Check if we can find key without flushing database */ + if (i == recant/2) + { + for (j=rnd(1000) ; j>0 && key1[j] == 0 ; j--) ; + if (!j) + for (j=999 ; j>0 && key1[j] == 0 ; j--) ; + sprintf(key,"%6d",j); + if (nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("Test in loop: Can't find key: \"%s\"\n",key); + goto err; + } + } + } + if (testflag==1) goto end; + + if (write_cacheing) + if (nisam_extra(file,HA_EXTRA_NO_CACHE)) + { + puts("got error from nisam_extra(HA_EXTRA_NO_CACHE)"); + goto end; + } + + printf("- Delete\n"); + for (i=0 ; i<recant/10 ; i++) + { + for (j=rnd(1000) ; j>0 && key1[j] == 0 ; j--) ; + if (j != 0) + { + sprintf(key,"%6d",j); + if (nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("can't find key1: \"%s\"\n",key); + goto err; + } + if (nisam_delete(file,read_record)) + { + printf("error: %d; can't delete record: \"%s\"\n", my_errno,read_record); + goto err; + } + delete++; + key1[atoi(read_record+keyinfo[0].seg[0].base.start)]--; + key3[atoi(read_record+keyinfo[2].seg[0].base.start)]=0; + } + } + if (testflag==2) goto end; + + printf("- Update\n"); + for (i=0 ; i<recant/10 ; i++) + { + n1=rnd(1000); n2=rnd(100); n3=rnd(5000); + sprintf(record2,"%6d:%4d:%8d:XXX: %4d ",n1,n2,n3,update); + longstore(record2+STANDAR_LENGTH-4,(long) i); + fix_length(record2,(uint) STANDAR_LENGTH+rnd(60)); + + for (j=rnd(1000) ; j>0 && key1[j] == 0 ; j--) ; + if (j != 0) + { + sprintf(key,"%6d",j); + if (nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) + { + printf("can't find key1: \"%s\"\n",key); + goto err; + } + if (use_blob) + { + if (i & 1) + put_blob_in_record(record+blob_pos,&blob_buffer); + else + bmove(record+blob_pos,read_record+blob_pos,8); + } + if (nisam_update(file,read_record,record2)) + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY || key3[n3] == 0) + { + printf("error: %d; can't uppdate:\nFrom: \"%s\"\nTo: \"%s\"\n", + my_errno,read_record,record2); + goto err; + } + if (verbose) + printf("Double key when tryed to uppdate:\nFrom: \"%s\"\nTo: \"%s\"\n",record,record2); + } + else + { + key1[atoi(read_record+keyinfo[0].seg[0].base.start)]--; + key3[atoi(read_record+keyinfo[2].seg[0].base.start)]=0; + key1[n1]++; key3[n3]=1; + update++; + } + } + } + if (testflag==3) goto end; + + printf("- Same key: first - next -> last - prev -> first\n"); + DBUG_PRINT("progpos",("first - next -> last - prev -> first")); + for (i=999, dupp_keys=j=0 ; i>0 ; i--) + { + if (key1[i] >dupp_keys) { dupp_keys=key1[i]; j=i; } + } + sprintf(key,"%6d",j); + if (verbose) printf(" Using key: \"%s\" Keys: %d\n",key,dupp_keys); + if (nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (nisam_rsame(file,read_record2,-1)) goto err; + if (memcmp(read_record,read_record2,reclength) != 0) + { + printf("nisam_rsame didn't find same record\n"); + goto end; + } + nisam_info(file,&info,0); + if (nisam_rfirst(file,read_record2,0) || + nisam_rsame_with_pos(file,read_record2,0,info.recpos) || + memcmp(read_record,read_record2,reclength) != 0) + { + printf("nisam_rsame_with_pos didn't find same record\n"); + goto end; + } + { + int skr=nisam_rnext(file,read_record2,0); + if ((skr && my_errno != HA_ERR_END_OF_FILE) || + nisam_rprev(file,read_record2,-1) || + memcmp(read_record,read_record2,reclength) != 0) + { + printf("nisam_rsame_with_pos lost position\n"); + goto end; + } + } + ant=1; + start=keyinfo[0].seg[0].base.start; length=keyinfo[0].seg[0].base.length; + while (nisam_rnext(file,read_record2,0) == 0 && + memcmp(read_record2+start,key,length) == 0) ant++; + if (ant != dupp_keys) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys); + goto end; + } + ant=0; + while (nisam_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys) + { + printf("prev: I can only find: %d records of %d\n",ant,dupp_keys); + goto end; + } + + printf("- All keys: first - next -> last - prev -> first\n"); + DBUG_PRINT("progpos",("All keys: first - next -> last - prev -> first")); + ant=1; + if (nisam_rfirst(file,read_record,0)) + { + printf("Can't find first record\n"); + goto end; + } + while (nisam_rnext(file,read_record3,0) == 0 && ant < write_count+10) + ant++; + if (ant != write_count - delete) + { + printf("next: I found: %d records of %d\n",ant,write_count - delete); + goto end; + } + if (nisam_rlast(file,read_record2,0) || + bcmp(read_record2,read_record3,reclength)) + { + printf("Can't find last record\n"); + DBUG_DUMP("record2",(byte*) read_record2,reclength); + DBUG_DUMP("record3",(byte*) read_record3,reclength); + goto end; + } + ant=1; + while (nisam_rprev(file,read_record3,0) == 0 && ant < write_count+10) + ant++; + if (ant != write_count - delete) + { + printf("prev: I found: %d records of %d\n",ant,write_count); + goto end; + } + if (bcmp(read_record,read_record3,reclength)) + { + printf("Can't find first record\n"); + goto end; + } + + printf("- Test if: Read first - next - prev - prev - next == first\n"); + DBUG_PRINT("progpos",("- Read first - next - prev - prev - next == first")); + if (nisam_rfirst(file,read_record,0) || + nisam_rnext(file,read_record3,0) || + nisam_rprev(file,read_record3,0) || + nisam_rprev(file,read_record3,0) == 0 || + nisam_rnext(file,read_record3,0)) + goto err; + if (bcmp(read_record,read_record3,reclength) != 0) + printf("Can't find first record\n"); + + printf("- Test if: Read last - prev - next - next - prev == last\n"); + DBUG_PRINT("progpos",("Read last - prev - next - next - prev == last")); + if (nisam_rlast(file,read_record2,0) || + nisam_rprev(file,read_record3,0) || + nisam_rnext(file,read_record3,0) || + nisam_rnext(file,read_record3,0) == 0 || + nisam_rprev(file,read_record3,0)) + goto err; + if (bcmp(read_record2,read_record3,reclength)) + printf("Can't find last record\n"); + + puts("- Test read key-part"); + strmov(key2,key); + for(i=strlen(key2) ; i-- > 1 ;) + { + key2[i]=0; + if (nisam_rkey(file,read_record,0,key2,(uint) i,HA_READ_KEY_EXACT)) goto err; + if (bcmp(read_record+start,key,(uint) i)) + { + puts("Didn't find right record"); + goto end; + } + } + if (dupp_keys > 2) + { + printf("- Read key (first) - next - delete - next -> last\n"); + DBUG_PRINT("progpos",("first - next - delete - next -> last")); + if (nisam_rkey(file,read_record,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (nisam_rnext(file,read_record3,0)) goto err; + if (nisam_delete(file,read_record3)) goto err; + delete++; + ant=1; + while (nisam_rnext(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-1) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-1); + goto end; + } + } + if (dupp_keys>4) + { + printf("- Read last of key - prev - delete - prev -> first\n"); + DBUG_PRINT("progpos",("last - prev - delete - prev -> first")); + if (nisam_rprev(file,read_record3,0)) goto err; + if (nisam_rprev(file,read_record3,0)) goto err; + if (nisam_delete(file,read_record3)) goto err; + delete++; + ant=1; + while (nisam_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-2) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-2); + goto end; + } + } + if (dupp_keys > 6) + { + printf("- Read first - delete - next -> last\n"); + DBUG_PRINT("progpos",("first - delete - next -> last")); + if (nisam_rkey(file,read_record3,0,key,0,HA_READ_KEY_EXACT)) goto err; + if (nisam_delete(file,read_record3)) goto err; + delete++; + ant=1; + if (nisam_rnext(file,read_record,0)) + goto err; /* Skall finnas poster */ + while (nisam_rnext(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-3) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-3); + goto end; + } + + printf("- Read last - delete - prev -> first\n"); + DBUG_PRINT("progpos",("last - delete - prev -> first")); + if (nisam_rprev(file,read_record3,0)) goto err; + if (nisam_delete(file,read_record3)) goto err; + delete++; + ant=0; + while (nisam_rprev(file,read_record3,0) == 0 && + bcmp(read_record3+start,key,length) == 0) ant++; + if (ant != dupp_keys-4) + { + printf("next: I can only find: %d keys of %d\n",ant,dupp_keys-4); + goto end; + } + } + + puts("- Test if: Read rrnd - same"); + DBUG_PRINT("progpos",("Read rrnd - same")); + for (i=0 ; i < write_count ; i++) + { + if (nisam_rrnd(file,read_record,i == 0 ? 0L : NI_POS_ERROR) == 0) + break; + } + if (i == write_count) + goto err; + + bmove(read_record2,read_record,reclength); + for (i=2 ; i-- > 0 ;) + { + if (nisam_rsame(file,read_record2,(int) i)) goto err; + if (bcmp(read_record,read_record2,reclength) != 0) + { + printf("is_rsame didn't find same record\n"); + goto end; + } + } + puts("- Test nisam_records_in_range"); + nisam_info(file,&info,HA_STATUS_VARIABLE); + for (i=0 ; i < info.keys ; i++) + { + if (nisam_rfirst(file,read_record,(int) i) || + nisam_rlast(file,read_record2,(int) i)) + goto err; + copy_key(file,(uint) i,(uchar*) read_record,(uchar*) key); + copy_key(file,(uint) i,(uchar*) read_record2,(uchar*) key2); + range_records=nisam_records_in_range(file,(int) i,key,0,HA_READ_KEY_EXACT, + key2,0,HA_READ_AFTER_KEY); + if (range_records < info.records*8/10 || + range_records > info.records*12/10) + { + printf("ni_records_range returned %lu; Should be about %lu\n", + range_records,info.records); + goto end; + } + if (verbose) + { + printf("ni_records_range returned %ld; Exact is %ld (diff: %4.2g %%)\n", + range_records,info.records, + labs((long) range_records - (long) info.records)*100.0/ + info.records); + + } + } + for (i=0 ; i < 5 ; i++) + { + for (j=rnd(1000) ; j>0 && key1[j] == 0 ; j--) ; + for (k=rnd(1000) ; k>0 && key1[k] == 0 ; k--) ; + if (j != 0 && k != 0) + { + if (j > k) + swap(int,j,k); + sprintf(key,"%6d",j); + sprintf(key2,"%6d",k); + range_records=nisam_records_in_range(file,0,key,0,HA_READ_AFTER_KEY, + key2,0,HA_READ_BEFORE_KEY); + records=0; + for (j++ ; j < k ; j++) + records+=key1[j]; + if ((long) range_records < (long) records*7/10-2 || + (long) range_records > (long) records*13/10+2) + { + printf("ni_records_range returned %ld; Should be about %ld\n", + range_records,records); + goto end; + } + if (verbose && records) + { + printf("ni_records_range returned %ld; Exact is %ld (diff: %4.2g %%)\n", + range_records,records, + labs((long) range_records-(long) records)*100.0/records); + + } + } + } + + printf("- nisam_info\n"); + nisam_info(file,&info,0); + if (info.records != write_count-delete || info.deleted > delete + update + || info.keys != keys) + { + puts("Wrong info from nisam_info"); + printf("Got: records: %ld delete: %ld i_keys: %d\n", + info.records,info.deleted,info.keys); + } + if (verbose) + { + char buff[80]; + get_date(buff,3,info.create_time); + printf("info: Created %s\n",buff); + get_date(buff,3,info.isamchk_time); + printf("info: checked %s\n",buff); + get_date(buff,3,info.update_time); + printf("info: Modified %s\n",buff); + } + + nisam_panic(HA_PANIC_WRITE); + nisam_panic(HA_PANIC_READ); + if (nisam_is_changed(file)) + puts("Warning: nisam_is_changed reported that datafile was changed"); + + printf("- nisam_extra(CACHE) + nisam_rrnd.... + nisam_extra(NO_CACHE)\n"); + if (nisam_extra(file,HA_EXTRA_RESET) || nisam_extra(file,HA_EXTRA_CACHE)) + { + if (locking || (!use_blob && !pack_fields)) + { + puts("got error from nisam_extra(HA_EXTRA_CACHE)"); + goto end; + } + } + ant=0; + while ((error=nisam_rrnd(file,record,NI_POS_ERROR)) >= 0 && + ant < write_count + 10) + ant+= error ? 0 : 1; + if (ant != write_count-delete) + { + printf("rrnd with cache: I can only find: %d records of %d\n", + ant,write_count-delete); + goto end; + } + if (nisam_extra(file,HA_EXTRA_NO_CACHE)) + { + puts("got error from nisam_extra(HA_EXTRA_NO_CACHE)"); + goto end; + } + + if (testflag == 4) goto end; + + printf("- Removing keys\n"); + lastpos = NI_POS_ERROR; + /* DBUG_POP(); */ + nisam_extra(file,HA_EXTRA_RESET); + while ((error=nisam_rrnd(file,read_record,NI_POS_ERROR)) >=0) + { + nisam_info(file,&info,1); + if (lastpos >= info.recpos && lastpos != NI_POS_ERROR) + { + printf("nisam_rrnd didn't advance filepointer; old: %ld, new: %ld\n", + lastpos,info.recpos); + goto err; + } + lastpos=info.recpos; + if (error == 0) + { + if (nisam_rsame(file,read_record,-1)) + { + printf("can't find record %lx\n",info.recpos); + goto err; + } + if (use_blob) + { + ulong blob_length,pos; + uchar *ptr; + longget(blob_length,read_record+blob_pos+4); + ptr=(uchar*) blob_length; + longget(blob_length,read_record+blob_pos); + for (pos=0 ; pos < blob_length ; pos++) + { + if (ptr[pos] != (uchar) (blob_length+pos)) + { + printf("found blob with wrong info at %ld\n",lastpos); + use_blob=0; + break; + } + } + } + if (nisam_delete(file,read_record)) + { + printf("can't delete record: %s\n",read_record); + goto err; + } + delete++; + } + } + if (my_errno != HA_ERR_END_OF_FILE && my_errno != HA_ERR_RECORD_DELETED) + printf("error: %d from nisam_rrnd\n",my_errno); + if (write_count != delete) + { + printf("Deleted only %d of %d records\n",write_count,delete); + goto err; + } +end: + if (nisam_close(file)) + goto err; + nisam_panic(HA_PANIC_CLOSE); /* Should close log */ + printf("\nFollowing test have been made:\n"); + printf("Write records: %d\nUpdate records: %d\nSame-key-read: %d\nDelete records: %d\n", write_count,update,dupp_keys,delete); + if (rec_pointer_size) + printf("Record pointer size: %d\n",rec_pointer_size); + if (key_cacheing) + puts("Key cacheing used"); + if (write_cacheing) + puts("Write cacheing used"); + if (async_io && locking) + puts("Asyncron io with locking used"); + else if (locking) + puts("Locking used"); + if (use_blob) + puts("blobs used"); + end_key_cache(); + if (blob_buffer) + my_free(blob_buffer,MYF(0)); + my_end(MY_CHECK_ERROR | MY_GIVE_INFO); + return(0); +err: + printf("got error: %d when using NISAM-database\n",my_errno); + if (file) + VOID(nisam_close(file)); + return(1); +} /* main */ + + + /* l{ser optioner */ + /* OBS! intierar endast DEBUG - ingen debuggning h{r ! */ + +static void get_options(argc,argv) +int argc; +char *argv[]; +{ + char *pos,*progname; + DEBUGGER_OFF; + + progname= argv[0]; + + while (--argc >0 && *(pos = *(++argv)) == '-' ) { + switch(*++pos) { + case 'b': + if (*++pos) + nisam_block_size= MY_ALIGN(atoi(pos),512); + set_if_bigger(nisam_block_size,8192); /* Max block size */ + set_if_smaller(nisam_block_size,1024); + break; + case 'B': + use_blob=1; + break; + case 'K': /* Use key cacheing */ + key_cacheing=1; + break; + case 'W': /* Use write cacheing */ + write_cacheing=1; + if (*++pos) + my_default_record_cache_size=atoi(pos); + break; + case 'i': + if (*++pos) + srand(atoi(pos)); + break; + case 'l': + use_log=1; + break; + case 'L': + locking=1; + break; + case 'A': /* use asyncron io */ + async_io=1; + if (*++pos) + my_default_record_cache_size=atoi(pos); + break; + case 'v': /* verbose */ + verbose=1; + break; + case 'm': /* records */ + recant=atoi(++pos); + break; + case 'f': + if ((first_key=atoi(++pos)) <0 || first_key >= NISAM_KEYS) + first_key=0; + break; + case 'k': + if ((keys=(uint) atoi(++pos)) < 1 || + keys > (uint) (NISAM_KEYS-first_key)) + keys=NISAM_KEYS-first_key; + break; + case 'P': + pack_type=0; /* Don't use DIFF_LENGTH */ + break; + case 'R': /* Length of record pointer */ + rec_pointer_size=atoi(++pos); + if (rec_pointer_size > 3) + rec_pointer_size=0; + break; + case 'S': + pack_fields=0; /* Static-length-records */ + break; + case 't': + testflag=atoi(++pos); /* testmod */ + break; + case '?': + case 'I': + case 'V': + printf("%s Ver 1.4 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); + puts("TCX Datakonsult AB, by Monty, for your professional use\n"); + printf("Usage: %s [-?ABIKLPRSVWltv] [-b#] [-k#] [-f#] [-m#] [-t#]\n",progname); + exit(0); + case '#': + DEBUGGER_ON; + DBUG_PUSH (++pos); + break; + default: + printf("Illegal option: '%c'\n",*pos); + break; + } + } + return; +} /* get options */ + + /* Ge ett randomv{rde inom ett intervall 0 <=x <= n */ + +static uint rnd(max_value) +uint max_value; +{ + return (uint) ((rand() & 32767)/32767.0*max_value); +} /* rnd */ + + + /* G|r en record av skiftande length */ + +static void fix_length(rec,length) +byte *rec; +uint length; +{ + bmove(rec+STANDAR_LENGTH, + "0123456789012345678901234567890123456789012345678901234567890", + length-STANDAR_LENGTH); + strfill(rec+length,STANDAR_LENGTH+60-length,' '); +} /* fix_length */ + + + /* Put maybe a blob in record */ + +static void put_blob_in_record(blob_pos,blob_buffer) +char *blob_pos,**blob_buffer; +{ + ulong i,length; + if (use_blob) + { + if (rnd(10) == 0) + { + if (! *blob_buffer && + !(*blob_buffer=my_malloc((uint) use_blob,MYF(MY_WME)))) + { + use_blob=0; + return; + } + length=rnd(use_blob); + for (i=0 ; i < length ; i++) + (*blob_buffer)[i]=(char) (length+i); + longstore(blob_pos,length); + bmove(blob_pos+4,(char*) blob_buffer,sizeof(char*)); + } + else + { + longstore(blob_pos,0); + } + } + return; +} + + +static void copy_key(info,inx,rec,key_buff) +N_INFO *info; +uint inx; +uchar *rec,*key_buff; +{ + N_KEYSEG *keyseg; + + for (keyseg=info->s->keyinfo[inx].seg ; keyseg->base.type ; keyseg++) + { + memcpy(key_buff,rec+keyseg->base.start,(size_t) keyseg->base.length); + key_buff+=keyseg->base.length; + } + return; +} diff --git a/isam/test3.c b/isam/test3.c new file mode 100644 index 00000000000..935c194106d --- /dev/null +++ b/isam/test3.c @@ -0,0 +1,479 @@ +/* Copyright (C) 2000 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 */ + +/* Test av locking */ + +#include "nisam.h" +#include <sys/types.h> +#ifdef HAVE_SYS_WAIT_H +# include <sys/wait.h> +#endif +#ifndef WEXITSTATUS +# define WEXITSTATUS(stat_val) ((unsigned)(stat_val) >> 8) +#endif +#ifndef WIFEXITED +# define WIFEXITED(stat_val) (((stat_val) & 255) == 0) +#endif + + +#if defined(HAVE_LRAND48) +#define rnd(X) (lrand48() % X) +#define rnd_init(X) srand48(X) +#else +#define rnd(X) (random() % X) +#define rnd_init(X) srandom(X) +#endif + + +const char *filename= "test3.ISM"; +uint tests=10,forks=10,key_cacheing=0,use_log=0; + +static void get_options(int argc, char *argv[]); +void start_test(int id); +int test_read(N_INFO *,int),test_write(N_INFO *,int,int), + test_update(N_INFO *,int,int),test_rrnd(N_INFO *,int); + +struct record { + char id[8]; + uint32 nr; + char text[10]; +} record; + + +int main(int argc,char **argv) +{ + int status,wait_ret; + uint i; + N_KEYDEF keyinfo[10]; + N_RECINFO recinfo[10]; + MY_INIT(argv[0]); + + get_options(argc,argv); + + keyinfo[0].seg[0].base.start=0; + keyinfo[0].seg[0].base.length=8; + keyinfo[0].seg[0].base.type=HA_KEYTYPE_TEXT; + keyinfo[0].seg[0].base.flag=HA_SPACE_PACK; + keyinfo[0].seg[1].base.type=0; + keyinfo[0].base.flag = (uint8) HA_PACK_KEY; + keyinfo[1].seg[0].base.start=8; + keyinfo[1].seg[0].base.length=sizeof(uint32); + keyinfo[1].seg[0].base.type=HA_KEYTYPE_LONG_INT; + keyinfo[1].seg[0].base.flag=0; + keyinfo[1].seg[1].base.type=0; + keyinfo[1].base.flag =HA_NOSAME; + + recinfo[0].base.type=0; + recinfo[0].base.length=sizeof(record.id); + recinfo[1].base.type=0; + recinfo[1].base.length=sizeof(record.nr); + recinfo[2].base.type=0; + recinfo[2].base.length=sizeof(record.text); + recinfo[3].base.type=FIELD_LAST; + + puts("- Creating isam-file"); + my_delete(filename,MYF(0)); /* Remove old locks under gdb */ + if (nisam_create(filename,2,&keyinfo[0],&recinfo[0],10000,0,0,0,0L)) + exit(1); + + rnd_init(0); + printf("- Starting %d processes\n",forks); fflush(stdout); + for (i=0 ; i < forks; i++) + { + if (!fork()) + { + start_test(i+1); + sleep(1); + return 0; + } + VOID(rnd(1)); + } + + for (i=0 ; i < forks ; i++) + while ((wait_ret=wait(&status)) && wait_ret == -1); + return 0; +} + + +static void get_options(argc,argv) +int argc; +char *argv[]; +{ + char *pos,*progname; + DEBUGGER_OFF; + + progname= argv[0]; + + while (--argc >0 && *(pos = *(++argv)) == '-' ) { + switch(*++pos) { + case 'l': + use_log=1; + break; + case 'f': + forks=atoi(++pos); + break; + case 't': + tests=atoi(++pos); + break; + case 'K': /* Use key cacheing */ + key_cacheing=1; + break; + case 'A': /* All flags */ + use_log=key_cacheing=1; + break; + case '?': + case 'I': + case 'V': + printf("%s Ver 1.0 for %s at %s\n",progname,SYSTEM_TYPE,MACHINE_TYPE); + puts("TCX Datakonsult AB, by Monty, for your professional use\n"); + puts("Test av locking with threads\n"); + printf("Usage: %s [-?lKA] [-f#] [-t#]\n",progname); + exit(0); + case '#': + DEBUGGER_ON; + DBUG_PUSH (++pos); + break; + default: + printf("Illegal option: '%c'\n",*pos); + break; + } + } + return; +} + + +void start_test(int id) +{ + uint i; + int error,lock_type; + N_ISAMINFO isam_info; + N_INFO *file,*file1,*file2,*lock; + + if (use_log) + nisam_log(1); + if (!(file1=nisam_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED)) || + !(file2=nisam_open(filename,O_RDWR,HA_OPEN_WAIT_IF_LOCKED))) + { + fprintf(stderr,"Can't open isam-file: %s\n",filename); + exit(1); + } + if (key_cacheing && rnd(2) == 0) + init_key_cache(65536L,(uint) IO_SIZE*4*10); + printf("Process %d, pid: %d\n",id,getpid()); fflush(stdout); + + for (error=i=0 ; i < tests && !error; i++) + { + file= (rnd(2) == 1) ? file1 : file2; + lock=0 ; lock_type=0; + if (rnd(10) == 0) + { + if (nisam_lock_database(lock=(rnd(2) ? file1 : file2), + lock_type=(rnd(2) == 0 ? F_RDLCK : F_WRLCK))) + { + fprintf(stderr,"%2d: start: Can't lock table %d\n",id,my_errno); + error=1; + break; + } + } + switch (rnd(4)) { + case 0: error=test_read(file,id); break; + case 1: error=test_rrnd(file,id); break; + case 2: error=test_write(file,id,lock_type); break; + case 3: error=test_update(file,id,lock_type); break; + } + if (lock) + nisam_lock_database(lock,F_UNLCK); + } + if (!error) + { + nisam_info(file1,&isam_info,0); + printf("%2d: End of test. Records: %ld Deleted: %ld\n", + id,isam_info.records,isam_info.deleted); + fflush(stdout); + } + + nisam_close(file1); + nisam_close(file2); + if (use_log) + nisam_log(0); + if (error) + { + printf("%2d: Aborted\n",id); fflush(stdout); + exit(1); + } +} + + +int test_read(N_INFO *file,int id) +{ + uint i,lock,found,next,prev; + ulong find; + + lock=0; + if (rnd(2) == 0) + { + lock=1; + if (nisam_lock_database(file,F_RDLCK)) + { + fprintf(stderr,"%2d: Can't lock table %d\n",id,my_errno); + return 1; + } + } + + found=next=prev=0; + for (i=0 ; i < 100 ; i++) + { + find=rnd(100000); + if (!nisam_rkey(file,record.id,1,(byte*) &find, + sizeof(find),HA_READ_KEY_EXACT)) + found++; + else + { + if (my_errno != HA_ERR_KEY_NOT_FOUND) + { + fprintf(stderr,"%2d: Got error %d from read in read\n",id,my_errno); + return 1; + } + else if (!nisam_rnext(file,record.id,1)) + next++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in read\n",id,my_errno); + return 1; + } + else if (!nisam_rprev(file,record.id,1)) + prev++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in read\n", + id,my_errno); + return 1; + } + } + } + } + } + if (lock) + { + if (nisam_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + return 1; + } + } + printf("%2d: read: found: %5d next: %5d prev: %5d\n", + id,found,next,prev); + fflush(stdout); + return 0; +} + + +int test_rrnd(N_INFO *file,int id) +{ + uint count,lock; + + lock=0; + if (rnd(2) == 0) + { + lock=1; + if (nisam_lock_database(file,F_RDLCK)) + { + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + nisam_close(file); + return 1; + } + if (rnd(2) == 0) + nisam_extra(file,HA_EXTRA_CACHE); + } + + count=0; + if (nisam_rrnd(file,record.id,0L)) + { + if (my_errno == HA_ERR_END_OF_FILE) + goto end; + fprintf(stderr,"%2d: Can't read first record (%d)\n",id,my_errno); + return 1; + } + for (count=1 ; !nisam_rrnd(file,record.id,NI_POS_ERROR) ;count++) ; + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rrnd\n",id,my_errno); + return 1; + } + +end: + if (lock) + { + nisam_extra(file,HA_EXTRA_NO_CACHE); + if (nisam_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + exit(0); + } + } + printf("%2d: rrnd: %5d\n",id,count); fflush(stdout); + return 0; +} + + +int test_write(N_INFO *file,int id,int lock_type) +{ + uint i,tries,count,lock; + + lock=0; + if (rnd(2) == 0 || lock_type == F_RDLCK) + { + lock=1; + if (nisam_lock_database(file,F_WRLCK)) + { + if (lock_type == F_RDLCK && my_errno == EDEADLK) + { + printf("%2d: write: deadlock\n",id); fflush(stdout); + return 0; + } + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + nisam_close(file); + return 1; + } + if (rnd(2) == 0) + nisam_extra(file,HA_EXTRA_WRITE_CACHE); + } + + sprintf(record.id,"%7d",getpid()); + strmov(record.text,"Testing..."); + + tries=(uint) rnd(100)+10; + for (i=count=0 ; i < tries ; i++) + { + record.nr=rnd(80000)+20000; + if (!nisam_write(file,record.id)) + count++; + else + { + if (my_errno != HA_ERR_FOUND_DUPP_KEY) + { + fprintf(stderr,"%2d: Got error %d (errno %d) from write\n",id,my_errno, + errno); + return 1; + } + } + } + if (lock) + { + nisam_extra(file,HA_EXTRA_NO_CACHE); + if (nisam_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"%2d: Can't unlock table\n",id); + exit(0); + } + } + printf("%2d: write: %5d\n",id,count); fflush(stdout); + return 0; +} + + +int test_update(N_INFO *file,int id,int lock_type) +{ + uint i,lock,found,next,prev,update; + ulong find; + struct record new_record; + + lock=0; + if (rnd(2) == 0 || lock_type == F_RDLCK) + { + lock=1; + if (nisam_lock_database(file,F_WRLCK)) + { + if (lock_type == F_RDLCK && my_errno == EDEADLK) + { + printf("%2d: write: deadlock\n",id); fflush(stdout); + return 0; + } + fprintf(stderr,"%2d: Can't lock table (%d)\n",id,my_errno); + return 1; + } + } + bzero((char*) &new_record,sizeof(new_record)); + strmov(new_record.text,"Updated"); + + found=next=prev=update=0; + for (i=0 ; i < 100 ; i++) + { + find=rnd(100000); + if (!nisam_rkey(file,record.id,1,(byte*) &find, + sizeof(find),HA_READ_KEY_EXACT)) + found++; + else + { + if (my_errno != HA_ERR_KEY_NOT_FOUND) + { + fprintf(stderr,"%2d: Got error %d from read in update\n",id,my_errno); + return 1; + } + else if (!nisam_rnext(file,record.id,1)) + next++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in update\n", + id,my_errno); + return 1; + } + else if (!nisam_rprev(file,record.id,1)) + prev++; + else + { + if (my_errno != HA_ERR_END_OF_FILE) + { + fprintf(stderr,"%2d: Got error %d from rnext in update\n", + id,my_errno); + return 1; + } + continue; + } + } + } + memcpy_fixed(new_record.id,record.id,sizeof(record.id)); + new_record.nr=rnd(20000)+40000; + if (!nisam_update(file,record.id,new_record.id)) + update++; + else + { + if (my_errno != HA_ERR_RECORD_CHANGED && + my_errno != HA_ERR_RECORD_DELETED && + my_errno != HA_ERR_FOUND_DUPP_KEY) + { + fprintf(stderr,"%2d: Got error %d from update\n",id,my_errno); + return 1; + } + } + } + if (lock) + { + if (nisam_lock_database(file,F_UNLCK)) + { + fprintf(stderr,"Can't unlock table,id, error%d\n",my_errno); + return 1; + } + } + printf("%2d: update: %5d\n",id,update); fflush(stdout); + return 0; +} diff --git a/isam/test_all b/isam/test_all new file mode 100755 index 00000000000..5de37e44585 --- /dev/null +++ b/isam/test_all @@ -0,0 +1,30 @@ +echo "test2 -L -K -W -P" +test2 -L -K -W -P +echo "test2 -L -K -W -P -A" +test2 -L -K -W -P -A +echo "test2 -L -K -W -P -S -R1 -m500" +test2 -L -K -W -P -S -R1 -m500 +echo "test2 -L -K -R1 -m2000 ; Should give error 135" +test2 -L -K -R1 -m2000 +echo "test2 -L -K -P -S -R3 -m50 -b1000000" +test2 -L -K -P -S -R3 -m50 -b1000000 +echo "test2 -L -B" +test2 -L -B +echo "test2 -L -K -W -P -m50 -l" +test2 -L -K -W -P -m50 -l +isamlog +echo "test2 -L -K -W -P -m50 -l -b100" +test2 -L -K -W -P -m50 -l -b100 +isamlog +echo "time test2" +time test2 +echo "time test2 -K" +time test2 -K +echo "time test2 -L" +time test2 -L +echo "time test2 -L -K" +time test2 -L -K +echo "time test2 -L -K -W" +time test2 -L -K -W +echo "time test2 -L -K -W -S" +time test2 -L -K -W -S diff --git a/isam/test_all.res b/isam/test_all.res new file mode 100644 index 00000000000..756a05f869c --- /dev/null +++ b/isam/test_all.res @@ -0,0 +1,356 @@ +test2 -L -K -W -P +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +Write cacheing used +Locking used +test2 -L -K -W -P -A +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +Write cacheing used +Asyncron io with locking used +test2 -L -K -W -P -S -R1 -m500 +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 484 +Update records: 48 +Same-key-read: 3 +Delete records: 484 +Record pointer size: 1 +Key cacheing used +Write cacheing used +Locking used +test2 -L -K -R1 -m2000 ; Should give error 135 +- Creating isam-file +- Writing key:s +Error: 135 in write at record: 1122 +got error: 135 when using NISAM-database +test2 -L -K -P -S -R3 -m50 -b1000000 +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 50 +Update records: 5 +Same-key-read: 2 +Delete records: 50 +Record pointer size: 3 +Key cacheing used +Locking used +test2 -L -B +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 912 +Update records: 81 +Same-key-read: 5 +Delete records: 912 +Locking used +blobs used +test2 -L -K -W -P -m50 -l +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 50 +Update records: 5 +Same-key-read: 2 +Delete records: 50 +Key cacheing used +Write cacheing used +Locking used +Commands Used count Errors Recover errors +open 3 0 0 +write 150 0 0 +update 15 0 0 +delete 150 0 0 +close 3 0 0 +extra 18 0 0 +Total 339 0 0 +test2 -L -K -W -P -m50 -l -b100 +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 50 +Update records: 5 +Same-key-read: 2 +Delete records: 50 +Key cacheing used +Write cacheing used +Locking used +Commands Used count Errors Recover errors +open 4 0 0 +write 200 0 0 +update 20 0 0 +delete 200 0 0 +close 4 0 0 +extra 24 0 0 +Total 452 0 0 +time test2 +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +4.77user 6.81system 0:15.07elapsed 76%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps +time test2 -K +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +6.09user 4.33system 0:11.66elapsed 89%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps +time test2 -L +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Locking used +5.01user 5.20system 0:10.86elapsed 94%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps +time test2 -L -K +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +Locking used +5.63user 0.97system 0:07.85elapsed 84%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps +time test2 -L -K -W +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +Write cacheing used +Locking used +5.28user 1.32system 0:08.86elapsed 74%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps +time test2 -L -K -W -S +- Creating isam-file +- Writing key:s +- Delete +- Update +- Same key: first - next -> last - prev -> first +- All keys: first - next -> last - prev -> first +- Test if: Read first - next - prev - prev - next == first +- Test if: Read last - prev - next - next - prev == last +- Test read key-part +- Read key (first) - next - delete - next -> last +- Read last of key - prev - delete - prev -> first +- Test if: Read rrnd - same +- Test ni_records_in_range +- ni_info +- ni_extra(CACHE) + ni_rrnd.... + ni_extra(NO_CACHE) +- Removing keys + +Following test have been made: +Write records: 915 +Update records: 82 +Same-key-read: 6 +Delete records: 915 +Key cacheing used +Write cacheing used +Locking used +5.32user 0.62system 0:06.13elapsed 96%CPU (0avgtext+0avgdata 0maxresident)k +0inputs+0outputs (0major+0minor)pagefaults 0swaps diff --git a/isam/update.c b/isam/update.c new file mode 100644 index 00000000000..ffcea740986 --- /dev/null +++ b/isam/update.c @@ -0,0 +1,119 @@ +/* Copyright (C) 2000 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 */ + +/* Uppdaterare nuvarande record i en pisam-databas */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <errno.h> +#endif + + /* Updaterar senaste l{sta record i databasen */ + +int nisam_update(register N_INFO *info, const byte *oldrec, const byte *newrec) +{ + int flag,key_changed,save_errno; + reg3 ulong pos; + uint i,length; + uchar old_key[N_MAX_KEY_BUFF],*new_key; + DBUG_ENTER("nisam_update"); + + if (!(info->update & HA_STATE_AKTIV)) + { + my_errno=HA_ERR_KEY_NOT_FOUND; + DBUG_RETURN(-1); + } + if (info->s->base.options & HA_OPTION_READ_ONLY_DATA) + { + my_errno=EACCES; + DBUG_RETURN(-1); + } + pos=info->lastpos; +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_WRLCK,1)) DBUG_RETURN(-1); +#endif + if ((*info->s->compare_record)(info,oldrec)) + { + save_errno=my_errno; + goto err_end; /* Record has changed */ + } + if (info->s->state.key_file_length >= + info->s->base.max_key_file_length - + info->s->blocksize* INDEX_BLOCK_MARGIN *info->s->state.keys) + { + my_errno=HA_ERR_INDEX_FILE_FULL; + goto err_end; + } + + /* Flyttar de element i isamfilen som m}ste flyttas */ + + new_key=info->lastkey+info->s->base.max_key_length; + key_changed=HA_STATE_KEY_CHANGED; /* We changed current database */ + /* Remove key that didn't change */ + for (i=0 ; i < info->s->state.keys ; i++) + { + length=_nisam_make_key(info,i,new_key,newrec,pos); + if (length != _nisam_make_key(info,i,old_key,oldrec,pos) || + memcmp((byte*) old_key,(byte*) new_key,length)) + { + if ((int) i == info->lastinx) + key_changed|=HA_STATE_WRITTEN; /* Mark that keyfile changed */ + if (_nisam_ck_delete(info,i,old_key)) goto err; + if (_nisam_ck_write(info,i,new_key)) goto err; + } + } + + if ((*info->s->update_record)(info,pos,newrec)) + goto err; + + info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | + key_changed); + nisam_log_record(LOG_UPDATE,info,newrec,info->lastpos,0); + VOID(_nisam_writeinfo(info,test(key_changed))); + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(0); + +err: + DBUG_PRINT("error",("key: %d errno: %d",i,my_errno)); + save_errno=my_errno; + if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL) + { + info->errkey= (int) i; + flag=0; + do + { + length=_nisam_make_key(info,i,new_key,newrec,pos); + if (length != _nisam_make_key(info,i,old_key,oldrec,pos) || + memcmp((byte*) old_key,(byte*) new_key,length)) + { + if ((flag++ && _nisam_ck_delete(info,i,new_key)) || + _nisam_ck_write(info,i,old_key)) + break; + } + } while (i-- != 0); + } + info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_AKTIV | + key_changed); + err_end: + nisam_log_record(LOG_UPDATE,info,newrec,info->lastpos,my_errno); + VOID(_nisam_writeinfo(info,1)); + allow_break(); /* Allow SIGHUP & SIGINT */ + if (save_errno == HA_ERR_KEY_NOT_FOUND) + my_errno=HA_ERR_CRASHED; + else + my_errno=save_errno; + DBUG_RETURN(-1); +} /* nisam_update */ diff --git a/isam/write.c b/isam/write.c new file mode 100644 index 00000000000..110dc70fe53 --- /dev/null +++ b/isam/write.c @@ -0,0 +1,840 @@ +/* Copyright (C) 2000 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 */ + +/* Skriver ett record till en isam-databas */ + +#include "isamdef.h" +#ifdef __WIN__ +#include <errno.h> +#endif + + /* Functions declared in this file */ + +static int w_search(N_INFO *info,N_KEYDEF *keyinfo,uchar *key, + ulong pos, uchar *father_buff, uchar *father_keypos, + ulong father_page); +static int _nisam_balance_page(N_INFO *info,N_KEYDEF *keyinfo,uchar *key, + uchar *curr_buff,uchar *father_buff, + uchar *father_keypos,ulong father_page); + + + /* Write new record to database */ + +int nisam_write(N_INFO *info, const byte *record) +{ + uint i; + ulong filepos; + uchar *buff; + DBUG_ENTER("nisam_write"); + DBUG_PRINT("enter",("isam: %d data: %d",info->s->kfile,info->dfile)); + + if (info->s->base.options & HA_OPTION_READ_ONLY_DATA) + { + my_errno=EACCES; + DBUG_RETURN(-1); + } +#ifndef NO_LOCKING + if (_nisam_readinfo(info,F_WRLCK,1)) DBUG_RETURN(-1); +#endif + dont_break(); /* Dont allow SIGHUP or SIGINT */ +#if !defined(NO_LOCKING) && defined(USE_RECORD_LOCK) + if (!info->locked && my_lock(info->dfile,F_WRLCK,0L,F_TO_EOF, + MYF(MY_SEEK_NOT_DONE) | info->lock_wait)) + goto err; +#endif + filepos= ((info->s->state.dellink != NI_POS_ERROR) ? + info->s->state.dellink : + info->s->state.data_file_length); + + if (info->s->base.reloc == 1L && info->s->base.records == 1L && + info->s->state.records == 1L) + { /* System file */ + my_errno=HA_ERR_RECORD_FILE_FULL; + goto err2; + } + if (info->s->state.key_file_length >= + info->s->base.max_key_file_length - + info->s->blocksize* INDEX_BLOCK_MARGIN *info->s->state.keys) + { + my_errno=HA_ERR_INDEX_FILE_FULL; + goto err2; + } + + /* Write all keys to indextree */ + buff=info->lastkey+info->s->base.max_key_length; + for (i=0 ; i < info->s->state.keys ; i++) + { + VOID(_nisam_make_key(info,i,buff,record,filepos)); + if (_nisam_ck_write(info,i,buff)) goto err; + } + + if ((*info->s->write_record)(info,record)) + goto err; + + info->update= (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED |HA_STATE_AKTIV | + HA_STATE_WRITTEN); + info->s->state.records++; + info->lastpos=filepos; + nisam_log_record(LOG_WRITE,info,record,filepos,0); + VOID(_nisam_writeinfo(info,1)); + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(0); + +err: + if (my_errno == HA_ERR_FOUND_DUPP_KEY || my_errno == HA_ERR_RECORD_FILE_FULL) + { + info->errkey= (int) i; + while ( i-- > 0) + { + VOID(_nisam_make_key(info,i,buff,record,filepos)); + if (_nisam_ck_delete(info,i,buff)) + break; + } + } + info->update=(HA_STATE_CHANGED | HA_STATE_ROW_CHANGED | HA_STATE_WRITTEN); +err2: + nisam_log_record(LOG_WRITE,info,record,filepos,my_errno); + VOID(_nisam_writeinfo(info,1)); + allow_break(); /* Allow SIGHUP & SIGINT */ + DBUG_RETURN(-1); +} /* nisam_write */ + + + /* Write one key to btree */ + +int _nisam_ck_write(register N_INFO *info, uint keynr, uchar *key) +{ + int error; + DBUG_ENTER("_nisam_ck_write"); + + if ((error=w_search(info,info->s->keyinfo+keynr,key, + info->s->state.key_root[keynr], (uchar *) 0, (uchar*) 0, + 0L)) > 0) + error=_nisam_enlarge_root(info,keynr,key); + DBUG_RETURN(error); +} /* _nisam_ck_write */ + + + /* Make a new root with key as only pointer */ + +int _nisam_enlarge_root(register N_INFO *info, uint keynr, uchar *key) +{ + uint t_length,nod_flag; + reg2 N_KEYDEF *keyinfo; + S_PARAM s_temp; + ISAM_SHARE *share=info->s; + DBUG_ENTER("_nisam_enlarge_root"); + + info->page_changed=1; + nod_flag= (share->state.key_root[keynr] != NI_POS_ERROR) ? + share->base.key_reflength : 0; + _nisam_kpointer(info,info->buff+2,share->state.key_root[keynr]); /* if nod */ + keyinfo=share->keyinfo+keynr; + t_length=_nisam_get_pack_key_length(keyinfo,nod_flag,(uchar*) 0,(uchar*) 0, + key,&s_temp); + putint(info->buff,t_length+2+nod_flag,nod_flag); + _nisam_store_key(keyinfo,info->buff+2+nod_flag,&s_temp); + if ((share->state.key_root[keynr]= _nisam_new(info,keyinfo)) == + NI_POS_ERROR || + _nisam_write_keypage(info,keyinfo,share->state.key_root[keynr],info->buff)) + DBUG_RETURN(-1); + DBUG_RETURN(0); +} /* _nisam_enlarge_root */ + + + /* S|ker reda p} vart nyckeln skall s{ttas och placerar den dit */ + /* Returnerar -1 om fel ; 0 om ok. 1 om nyckel propagerar upp}t */ + +static int w_search(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *key, ulong page, uchar *father_buff, + uchar *father_keypos, ulong father_page) +{ + int error,flag; + uint comp_flag,nod_flag; + uchar *temp_buff,*keypos; + uchar keybuff[N_MAX_KEY_BUFF]; + DBUG_ENTER("w_search"); + DBUG_PRINT("enter",("page: %ld",page)); + + if (page == NI_POS_ERROR) + DBUG_RETURN(1); /* No key, make new */ + + if (keyinfo->base.flag & HA_SORT_ALLOWS_SAME) + comp_flag=SEARCH_BIGGER; /* Put after same key */ + else if (keyinfo->base.flag & HA_NOSAME) + comp_flag=SEARCH_FIND; /* No dupplicates */ + else + comp_flag=SEARCH_SAME; /* Keys in rec-pos order */ + + if (!(temp_buff= (uchar*) my_alloca((uint) keyinfo->base.block_length+ + N_MAX_KEY_BUFF))) + DBUG_RETURN(-1); + if (!_nisam_fetch_keypage(info,keyinfo,page,temp_buff,0)) + goto err; + + flag=(*keyinfo->bin_search)(info,keyinfo,temp_buff,key,0,comp_flag,&keypos, + keybuff); + nod_flag=test_if_nod(temp_buff); + if (flag == 0) + { + my_errno=HA_ERR_FOUND_DUPP_KEY; + /* get position to record with dupplicated key */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&keypos,keybuff)); + info->dupp_key_pos=_nisam_dpos(info,test_if_nod(temp_buff),keypos); + my_afree((byte*) temp_buff); + DBUG_RETURN(-1); + } + if ((error=w_search(info,keyinfo,key,_nisam_kpos(nod_flag,keypos), + temp_buff,keypos,page)) >0) + { + error=_nisam_insert(info,keyinfo,key,temp_buff,keypos,keybuff,father_buff, + father_keypos,father_page); + if (_nisam_write_keypage(info,keyinfo,page,temp_buff)) + goto err; + } + my_afree((byte*) temp_buff); + DBUG_RETURN(error); +err: + my_afree((byte*) temp_buff); + DBUG_PRINT("exit",("Error: %d",my_errno)); + DBUG_RETURN (-1); +} /* w_search */ + + + /* Insert new key at right of key_pos */ + /* Returns 2 if key contains key to upper level */ + +int _nisam_insert(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *key, uchar *anc_buff, uchar *key_pos, uchar *key_buff, + uchar *father_buff, uchar *father_key_pos, ulong father_page) +{ + uint a_length,t_length,nod_flag; + uchar *endpos; + int key_offset; + S_PARAM s_temp; + DBUG_ENTER("_nisam_insert"); + DBUG_PRINT("enter",("key_pos: %lx",key_pos)); + DBUG_EXECUTE("key",_nisam_print_key(DBUG_FILE,keyinfo->seg,key);); + + nod_flag=test_if_nod(anc_buff); + a_length=getint(anc_buff); + endpos= anc_buff+ a_length; + t_length=_nisam_get_pack_key_length(keyinfo,nod_flag, + (key_pos == endpos ? (uchar*) 0 : key_pos), + (key_pos == anc_buff+2+nod_flag ? + (uchar*) 0 : key_buff),key,&s_temp); +#ifndef DBUG_OFF + if (key_pos != anc_buff+2+nod_flag) + DBUG_DUMP("prev_key",(byte*) key_buff,_nisam_keylength(keyinfo,key_buff)); + if (keyinfo->base.flag & HA_PACK_KEY) + { + DBUG_PRINT("test",("t_length: %d ref_len: %d", + t_length,s_temp.ref_length)); + DBUG_PRINT("test",("n_ref_len: %d n_length: %d key: %lx", + s_temp.n_ref_length,s_temp.n_length,s_temp.key)); + } +#endif + key_offset = (endpos-key_pos); + if((int) t_length < 0) + key_offset += (int) t_length; + if (key_offset < 0) + { + DBUG_PRINT("error",("Found a bug: negative key_offset %d\n", key_offset)); + DBUG_RETURN(-1); + } + if ((int) t_length >= 0) /* t_length is almost always > 0 */ + bmove_upp((byte*) endpos+t_length,(byte*) endpos,(uint)key_offset ); + else + { + /* This may happen if a key was deleted and the next key could be + compressed better than before */ + DBUG_DUMP("anc_buff",(byte*) anc_buff,a_length); + + bmove(key_pos,key_pos - (int) t_length,(uint)key_offset); + } + _nisam_store_key(keyinfo,key_pos,&s_temp); + a_length+=t_length; + putint(anc_buff,a_length,nod_flag); + if (a_length <= keyinfo->base.block_length) + DBUG_RETURN(0); /* There is room on page */ + + /* Page is full */ + + if (!(keyinfo->base.flag & (HA_PACK_KEY | HA_SPACE_PACK_USED)) && + father_buff) + DBUG_RETURN(_nisam_balance_page(info,keyinfo,key,anc_buff,father_buff, + father_key_pos,father_page)); + DBUG_RETURN(_nisam_splitt_page(info,keyinfo,key,anc_buff,key_buff)); +} /* _nisam_insert */ + + + /* splitt a full page in two and assign emerging item to key */ + +int _nisam_splitt_page(register N_INFO *info, register N_KEYDEF *keyinfo, + uchar *key, uchar *buff, uchar *key_buff) +{ + uint length,a_length,key_ref_length,t_length,nod_flag; + uchar *key_pos,*pos; + ulong new_pos; + S_PARAM s_temp; + DBUG_ENTER("ni_splitt_page"); + DBUG_DUMP("buff",(byte*) buff,getint(buff)); + + nod_flag=test_if_nod(buff); + key_ref_length=2+nod_flag; + key_pos=_nisam_find_half_pos(info,keyinfo,buff,key_buff); + length=(uint) (key_pos-buff); + a_length=getint(buff); + putint(buff,length,nod_flag); + info->page_changed=1; + + /* Correct new page pointer */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&key_pos,key_buff)); + if (nod_flag) + { + DBUG_PRINT("test",("Splitting nod")); + pos=key_pos-nod_flag; + memcpy((byte*) info->buff+2,(byte*) pos,(size_t) nod_flag); + } + + /* Move midle item to key and pointer to new page */ + if ((new_pos=_nisam_new(info,keyinfo)) == NI_POS_ERROR) + DBUG_RETURN(-1); + _nisam_kpointer(info,_nisam_move_key(keyinfo,key,key_buff),new_pos); + + /* Store new page */ + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&key_pos,key_buff)); + t_length=_nisam_get_pack_key_length(keyinfo,nod_flag,(uchar *) 0, (uchar*) 0, + key_buff, &s_temp); + s_temp.n_length= *key_pos; /* Needed by ni_store_key */ + length=(uint) ((buff+a_length)-key_pos); + memcpy((byte*) info->buff+key_ref_length+t_length,(byte*) key_pos, + (size_t) length); + _nisam_store_key(keyinfo,info->buff+key_ref_length,&s_temp); + putint(info->buff,length+t_length+key_ref_length,nod_flag); + + if (_nisam_write_keypage(info,keyinfo,new_pos,info->buff)) + DBUG_RETURN(-1); + DBUG_DUMP("key",(byte*) key,_nisam_keylength(keyinfo,key)); + DBUG_RETURN(2); /* Middle key up */ +} /* _nisam_splitt_page */ + + + /* find out how much more room a key will take */ + +#ifdef QQ +uint _nisam_get_pack_key_length(N_KEYDEF *keyinfo, uint nod_flag, uchar *key_pos, uchar *key_buff, uchar *key, S_PARAM *s_temp) + + /* If nod: Length of nod-pointer */ + /* Position to pos after key in buff */ + /* Last key before current key */ + /* Current key */ + /* How next key will be packed */ +{ + reg1 N_KEYSEG *keyseg; + int length; + uint key_length,ref_length,n_length,diff_flag,same_length; + uchar *start,*end,*key_end; + + s_temp->key=key; + if (!(keyinfo->base.flag & HA_PACK_KEY)) + return (s_temp->totlength=_nisam_keylength(keyinfo,key)+nod_flag); + s_temp->ref_length=s_temp->n_ref_length=s_temp->n_length=0; + s_temp->prev_length=0; + + same_length=0; keyseg=keyinfo->seg; + key_length=_nisam_keylength(keyinfo,key)+nod_flag; + + if (keyseg->base.flag & HA_SPACE_PACK) + { + diff_flag=1; + end=key_end= key+ *key+1; + if (key_buff) + { + if (*key == *key_buff && *key) + same_length=1; /* Don't use key-pack if length == 0 */ + else if (*key > *key_buff) + end=key+ *key_buff+1; + key_buff++; + } + key++; + } + else + { + diff_flag=0; + key_end=end= key+keyseg->base.length; + } + + start=key; + if (key_buff) + while (key < end && *key == *key_buff) + { + key++; key_buff++; + } + + s_temp->key=key; s_temp->key_length= (uint) (key_end-key); + + if (same_length && key == key_end) + { + s_temp->ref_length=128; + length=(int) key_length-(int)(key_end-start); /* Same as prev key */ + if (key_pos) + { + s_temp->n_length= *key_pos; + key_pos=0; /* Can't combine with next */ + } + } + else + { + if (start != key) + { /* Starts as prev key */ + s_temp->ref_length= (uint) (key-start)+128; + length=(int) (1+key_length-(uint) (key-start)); + } + else + length=(int) (key_length+ (1-diff_flag)); /* Not packed key */ + } + s_temp->totlength=(uint) length; + + DBUG_PRINT("test",("tot_length: %d length: %d uniq_key_length: %d", + key_length,length,s_temp->key_length)); + + /* If something after that is not 0 length test if we can combine */ + + if (key_pos && (n_length= *key_pos)) + { + key_pos++; + ref_length=0; + if (n_length & 128) + { + if ((ref_length=n_length & 127)) + if (diff_flag) + n_length= *key_pos++; /* Length of key-part */ + else + n_length=keyseg->base.length - ref_length; + } + else + if (*start == *key_pos && diff_flag && start != key_end) + length++; /* One new pos for ref.len */ + + DBUG_PRINT("test",("length: %d key_pos: %lx",length,key_pos)); + if (n_length != 128) + { /* Not same key after */ + key=start+ref_length; + while (n_length > 0 && key < key_end && *key == *key_pos) + { + key++; key_pos++; + ref_length++; + n_length--; + length--; /* We gained one char */ + } + + if (n_length == 0 && diff_flag) + { + n_length=128; /* Same as prev key */ + length--; /* We don't need key-length */ + } + else if (ref_length) + s_temp->n_ref_length=ref_length | 128; + } + s_temp->n_length=n_length; + } + return (uint) length; +} /* _nisam_get_pack_key_length */ + +#else + +uint +_nisam_get_pack_key_length(N_KEYDEF *keyinfo, + uint nod_flag, /* If nod: Length of nod-pointer */ + uchar *key_pos, /* Position to pos after key in buff */ + uchar *key_buff,/* Last key before current key */ + uchar *key, /* Current key */ + S_PARAM *s_temp/* How next key will be packed */ + ) +{ + reg1 N_KEYSEG *keyseg; + int length; + uint key_length,ref_length,n_length,diff_flag,same_length,org_key_length=0; + uchar *start,*end,*key_end; + + s_temp->key=key; + if (!(keyinfo->base.flag & HA_PACK_KEY)) + return (s_temp->totlength=_nisam_keylength(keyinfo,key)+nod_flag); + s_temp->ref_length=s_temp->n_ref_length=s_temp->n_length=0; + + same_length=0; keyseg=keyinfo->seg; + key_length=_nisam_keylength(keyinfo,key)+nod_flag; + s_temp->prev_key=key_buff; + + if (keyseg->base.flag & HA_SPACE_PACK) + { + diff_flag=1; + end=key_end= key+ *key+1; + if (key_buff) + { + org_key_length= (uint) *key_buff; + if (*key == *key_buff && *key) + same_length=1; /* Don't use key-pack if length == 0 */ + else if (*key > *key_buff) + end=key+ org_key_length+1; + key_buff++; + } + key++; + } + else + { + diff_flag=0; + key_end=end= key+(org_key_length=keyseg->base.length); + } + + start=key; + if (key_buff) + while (key < end && *key == *key_buff) + { + key++; key_buff++; + } + + s_temp->key=key; s_temp->key_length= (uint) (key_end-key); + + if (same_length && key == key_end) + { + s_temp->ref_length=128; + length=(int) key_length-(int)(key_end-start); /* Same as prev key */ + if (key_pos) + { /* Can't combine with next */ + s_temp->n_length= *key_pos; /* Needed by _nisam_store_key */ + key_pos=0; + } + } + else + { + if (start != key) + { /* Starts as prev key */ + s_temp->ref_length= (uint) (key-start)+128; + length=(int) (1+key_length-(uint) (key-start)); + } + else + length=(int) (key_length+ (1-diff_flag)); /* Not packed key */ + } + s_temp->totlength=(uint) length; + s_temp->prev_length=0; + DBUG_PRINT("test",("tot_length: %d length: %d uniq_key_length: %d", + key_length,length,s_temp->key_length)); + + /* If something after that is not 0 length test if we can combine */ + + if (key_pos && (n_length= *key_pos++)) + { + if (n_length == 128) + { + /* + We put a different key between two identical keys + Extend next key to have same prefix as this key + */ + if (s_temp->ref_length) + { /* make next key longer */ + s_temp->part_of_prev_key= s_temp->ref_length; + s_temp->prev_length= org_key_length - (s_temp->ref_length-128); + s_temp->n_length= s_temp->prev_length; + s_temp->prev_key+= diff_flag + (s_temp->ref_length - 128); + length+= s_temp->prev_length+diff_flag; + } + else + { /* Can't use prev key */ + s_temp->part_of_prev_key=0; + s_temp->prev_length= org_key_length; + s_temp->n_length= org_key_length; + s_temp->prev_key+= diff_flag; /* To start of key */ + length+= org_key_length; + } + return (uint) length; + } + + if (n_length & 128) + { + ref_length=n_length & 127; + if (diff_flag) /* If SPACE_PACK */ + n_length= *key_pos++; /* Length of key-part */ + else + n_length=keyseg->base.length - ref_length; + + /* Test if new keys has fewer characters that match the previous key */ + if (!s_temp->ref_length) + { /* Can't use prev key */ + s_temp->part_of_prev_key= 0; + s_temp->prev_length= ref_length; + s_temp->n_length= n_length+ref_length; + s_temp->prev_key+= diff_flag; /* To start of key */ + return (uint) length+ref_length-diff_flag; + } + if (ref_length+128 > s_temp->ref_length) + { + /* We must copy characters from the original key to the next key */ + s_temp->part_of_prev_key= s_temp->ref_length; + s_temp->prev_length= ref_length+128 - s_temp->ref_length; + s_temp->n_length= n_length + s_temp->prev_length; + s_temp->prev_key+= diff_flag + s_temp->ref_length -128; + return (uint) length + s_temp->prev_length; + } + } + else + { + ref_length=0; + if (*start == *key_pos && diff_flag && start != key_end) + length++; /* One new pos for ref.len */ + } + DBUG_PRINT("test",("length: %d key_pos: %lx",length,key_pos)); + + key=start+ref_length; + while (n_length > 0 && key < key_end && *key == *key_pos) + { + key++; key_pos++; + ref_length++; + n_length--; + length--; /* We gained one char */ + } + + if (n_length == 0 && diff_flag) + { + n_length=128; /* Same as prev key */ + length--; /* We don't need key-length */ + } + else if (ref_length) + s_temp->n_ref_length=ref_length | 128; + s_temp->n_length=n_length; + } + return (uint) length; +} /* _nisam_get_pack_key_length */ +#endif + + + /* store a key in page-buffert */ + +void _nisam_store_key(N_KEYDEF *keyinfo, register uchar *key_pos, + register S_PARAM *s_temp) +{ + uint length; + uchar *start; + + if (! (keyinfo->base.flag & HA_PACK_KEY)) + { + memcpy((byte*) key_pos,(byte*) s_temp->key,(size_t) s_temp->totlength); + return; + } + start=key_pos; + if ((*key_pos=(uchar) s_temp->ref_length)) + key_pos++; + if (s_temp->ref_length == 0 || + (s_temp->ref_length > 128 && + (keyinfo->seg[0].base.flag & HA_SPACE_PACK))) + *key_pos++= (uchar) s_temp->key_length; + bmove((byte*) key_pos,(byte*) s_temp->key, + (length=s_temp->totlength-(uint) (key_pos-start))); + key_pos+=length; + + if (s_temp->prev_length) + { + /* Extend next key because new key didn't have same prefix as prev key */ + if (s_temp->part_of_prev_key) + *key_pos++ = s_temp->part_of_prev_key; + if (keyinfo->seg[0].base.flag & HA_SPACE_PACK) + *key_pos++= s_temp->n_length; + memcpy(key_pos, s_temp->prev_key, s_temp->prev_length); + return; + } + + if ((*key_pos = (uchar) s_temp->n_ref_length)) + { + if (! (keyinfo->seg[0].base.flag & HA_SPACE_PACK)) + return; /* Don't save keylength */ + key_pos++; /* Store ref for next key */ + } + *key_pos = (uchar) s_temp->n_length; + return; +} /* _nisam_store_key */ + + + /* Calculate how to much to move to split a page in two */ + /* Returns pointer and key for get_key() to get mid key */ + /* There is at last 2 keys after pointer in buff */ + +uchar *_nisam_find_half_pos(N_INFO *info, N_KEYDEF *keyinfo, uchar *page, uchar *key) +{ + uint keys,length,key_ref_length,nod_flag; + uchar *end,*lastpos; + DBUG_ENTER("_nisam_find_half_pos"); + + nod_flag=test_if_nod(page); + key_ref_length=2+nod_flag; + length=getint(page)-key_ref_length; + page+=key_ref_length; + if (!(keyinfo->base.flag & (HA_PACK_KEY | HA_SPACE_PACK_USED))) + { + keys=(length/(keyinfo->base.keylength+nod_flag))/2; + DBUG_RETURN(page+keys*(keyinfo->base.keylength+nod_flag)); + } + + end=page+length/2-key_ref_length; /* This is aprox. half */ + *key='\0'; + do + { + lastpos=page; + VOID((*keyinfo->get_key)(keyinfo,nod_flag,&page,key)); + } while (page < end); + + DBUG_PRINT("exit",("returns: %lx page: %lx half: %lx",lastpos,page,end)); + DBUG_RETURN(lastpos); +} /* _nisam_find_half_pos */ + + + /* Balance page with not packed keys with page on right/left */ + /* returns 0 if balance was done */ + +static int _nisam_balance_page(register N_INFO *info, N_KEYDEF *keyinfo, + uchar *key, uchar *curr_buff, uchar *father_buff, + uchar *father_key_pos, ulong father_page) +{ + my_bool right; + uint k_length,father_length,father_keylength,nod_flag,curr_keylength, + right_length,left_length,new_right_length,new_left_length,extra_length, + length,keys; + uchar *pos,*buff,*extra_buff; + ulong next_page,new_pos; + byte tmp_part_key[N_MAX_KEY_BUFF]; + DBUG_ENTER("_nisam_balance_page"); + + k_length=keyinfo->base.keylength; + father_length=getint(father_buff); + father_keylength=k_length+info->s->base.key_reflength; + nod_flag=test_if_nod(curr_buff); + curr_keylength=k_length+nod_flag; + info->page_changed=1; + + if ((father_key_pos != father_buff+father_length && (info->s->rnd++ & 1)) || + father_key_pos == father_buff+2+info->s->base.key_reflength) + { + right=1; + next_page= _nisam_kpos(info->s->base.key_reflength, + father_key_pos+father_keylength); + buff=info->buff; + DBUG_PRINT("test",("use right page: %lu",next_page)); + } + else + { + right=0; + father_key_pos-=father_keylength; + next_page= _nisam_kpos(info->s->base.key_reflength,father_key_pos); + /* Fix that curr_buff is to left */ + buff=curr_buff; curr_buff=info->buff; + DBUG_PRINT("test",("use left page: %lu",next_page)); + } /* father_key_pos ptr to parting key */ + + if (!_nisam_fetch_keypage(info,keyinfo,next_page,info->buff,0)) + goto err; + DBUG_DUMP("next",(byte*) info->buff,getint(info->buff)); + + /* Test if there is room to share keys */ + + left_length=getint(curr_buff); + right_length=getint(buff); + keys=(left_length+right_length-4-nod_flag*2)/curr_keylength; + + if ((right ? right_length : left_length) + curr_keylength <= + keyinfo->base.block_length) + { /* Merge buffs */ + new_left_length=2+nod_flag+(keys/2)*curr_keylength; + new_right_length=2+nod_flag+((keys+1)/2)*curr_keylength; + putint(curr_buff,new_left_length,nod_flag); + putint(buff,new_right_length,nod_flag); + + if (left_length < new_left_length) + { /* Move keys buff -> leaf */ + pos=curr_buff+left_length; + memcpy((byte*) pos,(byte*) father_key_pos, (size_t) k_length); + memcpy((byte*) pos+k_length, (byte*) buff+2, + (size_t) (length=new_left_length - left_length - k_length)); + pos=buff+2+length; + memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); + bmove((byte*) buff+2,(byte*) pos+k_length,new_right_length); + } + else + { /* Move keys -> buff */ + + bmove_upp((byte*) buff+new_right_length,(byte*) buff+right_length, + right_length-2); + length=new_right_length-right_length-k_length; + memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + pos=curr_buff+new_left_length; + memcpy((byte*) father_key_pos,(byte*) pos,(size_t) k_length); + memcpy((byte*) buff+2,(byte*) pos+k_length,(size_t) length); + } + + if (_nisam_write_keypage(info,keyinfo,next_page,info->buff) || + _nisam_write_keypage(info,keyinfo,father_page,father_buff)) + goto err; + DBUG_RETURN(0); + } + + /* curr_buff[] and buff[] are full, lets splitt and make new nod */ + + extra_buff=info->buff+info->s->base.max_block; + new_left_length=new_right_length=2+nod_flag+(keys+1)/3*curr_keylength; + if (keys == 5) /* Too few keys to balance */ + new_left_length-=curr_keylength; + extra_length=nod_flag+left_length+right_length-new_left_length-new_right_length-curr_keylength; + DBUG_PRINT("info",("left_length: %d right_length: %d new_left_length: %d new_right_length: %d extra_length: %d", + left_length, right_length, + new_left_length, new_right_length, + extra_length)); + putint(curr_buff,new_left_length,nod_flag); + putint(buff,new_right_length,nod_flag); + putint(extra_buff,extra_length+2,nod_flag); + + /* move first largest keys to new page */ + pos=buff+right_length-extra_length; + memcpy((byte*) extra_buff+2,pos,(size_t) extra_length); + + /* Save new parting key */ + memcpy(tmp_part_key, pos-k_length,k_length); + + /* Make place for new keys */ + bmove_upp((byte*) buff+new_right_length,(byte*) pos-k_length, + right_length-extra_length-k_length-2); + /* Copy keys from left page */ + pos= curr_buff+new_left_length; + memcpy((byte*) buff+2,(byte*) pos+k_length, + (size_t) (length=left_length-new_left_length-k_length)); + /* Copy old parting key */ + memcpy((byte*) buff+2+length,father_key_pos,(size_t) k_length); + + /* Move new parting keys up */ + memcpy((byte*) (right ? key : father_key_pos),pos, (size_t) k_length); + memcpy((byte*) (right ? father_key_pos : key), tmp_part_key, k_length); + + if ((new_pos=_nisam_new(info,keyinfo)) == NI_POS_ERROR) + goto err; + _nisam_kpointer(info,key+k_length,new_pos); + if (_nisam_write_keypage(info,keyinfo,(right ? new_pos : next_page), + info->buff) || + _nisam_write_keypage(info,keyinfo,(right ? next_page : new_pos),extra_buff)) + goto err; + + DBUG_RETURN(1); /* Middle key up */ + +err: + DBUG_RETURN(-1); +} /* _nisam_balance_page */ |