mi_check.c
上传用户:romrleung
上传日期:2022-05-23
资源大小:18897k
文件大小:134k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
  2.    This program is free software; you can redistribute it and/or modify
  3.    it under the terms of the GNU General Public License as published by
  4.    the Free Software Foundation; either version 2 of the License, or
  5.    (at your option) any later version.
  6.    This program is distributed in the hope that it will be useful,
  7.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  8.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  9.    GNU General Public License for more details.
  10.    You should have received a copy of the GNU General Public License
  11.    along with this program; if not, write to the Free Software
  12.    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
  13. /* Describe, check and repair of MyISAM tables */
  14. #include "ftdefs.h"
  15. #include <m_ctype.h>
  16. #include <stdarg.h>
  17. #include <my_getopt.h>
  18. #ifdef HAVE_SYS_VADVISE_H
  19. #include <sys/vadvise.h>
  20. #endif
  21. #ifdef HAVE_SYS_MMAN_H
  22. #include <sys/mman.h>
  23. #endif
  24. #include "rt_index.h"
  25. #ifndef USE_RAID
  26. #define my_raid_create(A,B,C,D,E,F,G) my_create(A,B,C,G)
  27. #define my_raid_delete(A,B,C) my_delete(A,B)
  28. #endif
  29. /* Functions defined in this file */
  30. static int check_k_link(MI_CHECK *param, MI_INFO *info,uint nr);
  31. static int chk_index(MI_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo,
  32.      my_off_t page, uchar *buff, ha_rows *keys,
  33.      ha_checksum *key_checksum, uint level);
  34. static uint isam_key_length(MI_INFO *info,MI_KEYDEF *keyinfo);
  35. static ha_checksum calc_checksum(ha_rows count);
  36. static int writekeys(MI_CHECK *param, MI_INFO *info,byte *buff,
  37.      my_off_t filepos);
  38. static int sort_one_index(MI_CHECK *param, MI_INFO *info,MI_KEYDEF *keyinfo,
  39.   my_off_t pagepos, File new_file);
  40. static int sort_key_read(MI_SORT_PARAM *sort_param,void *key);
  41. static int sort_ft_key_read(MI_SORT_PARAM *sort_param,void *key);
  42. static int sort_get_next_record(MI_SORT_PARAM *sort_param);
  43. static int sort_key_cmp(MI_SORT_PARAM *sort_param, const void *a,const void *b);
  44. static int sort_ft_key_write(MI_SORT_PARAM *sort_param, const void *a);
  45. static int sort_key_write(MI_SORT_PARAM *sort_param, const void *a);
  46. static my_off_t get_record_for_key(MI_INFO *info,MI_KEYDEF *keyinfo,
  47. uchar *key);
  48. static int sort_insert_key(MI_SORT_PARAM  *sort_param,
  49.                            reg1 SORT_KEY_BLOCKS *key_block,
  50.    uchar *key, my_off_t prev_block);
  51. static int sort_delete_record(MI_SORT_PARAM *sort_param);
  52. /*static int flush_pending_blocks(MI_CHECK *param);*/
  53. static SORT_KEY_BLOCKS *alloc_key_blocks(MI_CHECK *param, uint blocks,
  54.   uint buffer_length);
  55. static ha_checksum mi_byte_checksum(const byte *buf, uint length);
  56. static void set_data_file_type(SORT_INFO *sort_info, MYISAM_SHARE *share);
  57. void myisamchk_init(MI_CHECK *param)
  58. {
  59.   bzero((gptr) param,sizeof(*param));
  60.   param->opt_follow_links=1;
  61.   param->keys_in_use= ~(ulonglong) 0;
  62.   param->search_after_block=HA_OFFSET_ERROR;
  63.   param->auto_increment_value= 0;
  64.   param->use_buffers=USE_BUFFER_INIT;
  65.   param->read_buffer_length=READ_BUFFER_INIT;
  66.   param->write_buffer_length=READ_BUFFER_INIT;
  67.   param->sort_buffer_length=SORT_BUFFER_INIT;
  68.   param->sort_key_blocks=BUFFERS_WHEN_SORTING;
  69.   param->tmpfile_createflag=O_RDWR | O_TRUNC | O_EXCL;
  70.   param->myf_rw=MYF(MY_NABP | MY_WME | MY_WAIT_IF_FULL);
  71.   param->start_check_pos=0;
  72.   param->max_record_length= LONGLONG_MAX;
  73.   param->key_cache_block_size= KEY_CACHE_BLOCK_SIZE;
  74.   param->stats_method= MI_STATS_METHOD_NULLS_NOT_EQUAL;
  75. }
  76. /* Check the status flags for the table */
  77. int chk_status(MI_CHECK *param, register MI_INFO *info)
  78. {
  79.   MYISAM_SHARE *share=info->s;
  80.   if (mi_is_crashed_on_repair(info))
  81.     mi_check_print_warning(param,
  82.    "Table is marked as crashed and last repair failed");
  83.   else if (mi_is_crashed(info))
  84.     mi_check_print_warning(param,
  85.    "Table is marked as crashed");
  86.   if (share->state.open_count != (uint) (info->s->global_changed ? 1 : 0))
  87.   {
  88.     /* Don't count this as a real warning, as check can correct this ! */
  89.     uint save=param->warning_printed;
  90.     mi_check_print_warning(param,
  91.    share->state.open_count==1 ? 
  92.    "%d client is using or hasn't closed the table properly" : 
  93.    "%d clients are using or haven't closed the table properly",
  94.    share->state.open_count);
  95.     /* If this will be fixed by the check, forget the warning */
  96.     if (param->testflag & T_UPDATE_STATE)
  97.       param->warning_printed=save;
  98.   }
  99.   return 0;
  100. }
  101. /* Check delete links */
  102. int chk_del(MI_CHECK *param, register MI_INFO *info, uint test_flag)
  103. {
  104.   reg2 ha_rows i;
  105.   uint delete_link_length;
  106.   my_off_t empty,next_link,old_link;
  107.   char buff[22],buff2[22];
  108.   DBUG_ENTER("chk_del");
  109.   LINT_INIT(old_link);
  110.   param->record_checksum=0;
  111.   delete_link_length=((info->s->options & HA_OPTION_PACK_RECORD) ? 20 :
  112.       info->s->rec_reflength+1);
  113.   if (!(test_flag & T_SILENT))
  114.     puts("- check record delete-chain");
  115.   next_link=info->s->state.dellink;
  116.   if (info->state->del == 0)
  117.   {
  118.     if (test_flag & T_VERBOSE)
  119.     {
  120.       puts("No recordlinks");
  121.     }
  122.   }
  123.   else
  124.   {
  125.     if (test_flag & T_VERBOSE)
  126.       printf("Recordlinks:    ");
  127.     empty=0;
  128.     for (i= info->state->del ; i > 0L && next_link != HA_OFFSET_ERROR ; i--)
  129.     {
  130.       if (*killed_ptr(param))
  131.         DBUG_RETURN(1);
  132.       if (test_flag & T_VERBOSE)
  133. printf(" %9s",llstr(next_link,buff));
  134.       if (next_link >= info->state->data_file_length)
  135. goto wrong;
  136.       if (my_pread(info->dfile,(char*) buff,delete_link_length,
  137.    next_link,MYF(MY_NABP)))
  138.       {
  139. if (test_flag & T_VERBOSE) puts("");
  140. mi_check_print_error(param,"Can't read delete-link at filepos: %s",
  141.     llstr(next_link,buff));
  142. DBUG_RETURN(1);
  143.       }
  144.       if (*buff != '')
  145.       {
  146. if (test_flag & T_VERBOSE) puts("");
  147. mi_check_print_error(param,"Record at pos: %s is not remove-marked",
  148.     llstr(next_link,buff));
  149. goto wrong;
  150.       }
  151.       if (info->s->options & HA_OPTION_PACK_RECORD)
  152.       {
  153. my_off_t prev_link=mi_sizekorr(buff+12);
  154. if (empty && prev_link != old_link)
  155. {
  156.   if (test_flag & T_VERBOSE) puts("");
  157.   mi_check_print_error(param,"Deleted block at %s doesn't point back at previous delete link",llstr(next_link,buff2));
  158.   goto wrong;
  159. }
  160. old_link=next_link;
  161. next_link=mi_sizekorr(buff+4);
  162. empty+=mi_uint3korr(buff+1);
  163.       }
  164.       else
  165.       {
  166. param->record_checksum+=(ha_checksum) next_link;
  167. next_link=_mi_rec_pos(info->s,(uchar*) buff+1);
  168. empty+=info->s->base.pack_reclength;
  169.       }
  170.     }
  171.     if (test_flag & T_VERBOSE)
  172.       puts("n");
  173.     if (empty != info->state->empty)
  174.     {
  175.       mi_check_print_warning(param,
  176.      "Found %s deleted space in delete link chain. Should be %s",
  177.      llstr(empty,buff2),
  178.      llstr(info->state->empty,buff));
  179.     }
  180.     if (next_link != HA_OFFSET_ERROR)
  181.     {
  182.       mi_check_print_error(param,
  183.    "Found more than the expected %s deleted rows in delete link chain",
  184.    llstr(info->state->del, buff));
  185.       goto wrong;
  186.     }
  187.     if (i != 0)
  188.     {
  189.       mi_check_print_error(param,
  190.    "Found %s deleted rows in delete link chain. Should be %s",
  191.    llstr(info->state->del - i, buff2),
  192.    llstr(info->state->del, buff));
  193.       goto wrong;
  194.     }
  195.   }
  196.   DBUG_RETURN(0);
  197. wrong:
  198.   param->testflag|=T_RETRY_WITHOUT_QUICK;
  199.   if (test_flag & T_VERBOSE) puts("");
  200.   mi_check_print_error(param,"record delete-link-chain corrupted");
  201.   DBUG_RETURN(1);
  202. } /* chk_del */
  203. /* Check delete links in index file */
  204. static int check_k_link(MI_CHECK *param, register MI_INFO *info, uint nr)
  205. {
  206.   my_off_t next_link;
  207.   uint block_size=(nr+1)*MI_MIN_KEY_BLOCK_LENGTH;
  208.   ha_rows records;
  209.   char llbuff[21],*buff;
  210.   DBUG_ENTER("check_k_link");
  211.   if (param->testflag & T_VERBOSE)
  212.     printf("block_size %4d:",block_size);
  213.   next_link=info->s->state.key_del[nr];
  214.   records= (ha_rows) (info->state->key_file_length / block_size);
  215.   while (next_link != HA_OFFSET_ERROR && records > 0)
  216.   {
  217.     if (*killed_ptr(param))
  218.       DBUG_RETURN(1);
  219.     if (param->testflag & T_VERBOSE)
  220.       printf("%16s",llstr(next_link,llbuff));
  221.     if (next_link > info->state->key_file_length ||
  222. next_link & (info->s->blocksize-1))
  223.       DBUG_RETURN(1);
  224.     if (!(buff=key_cache_read(info->s->key_cache,
  225.                               info->s->kfile, next_link, DFLT_INIT_HITS,
  226.                               (byte*) info->buff,
  227.       myisam_block_size, block_size, 1)))
  228.       DBUG_RETURN(1);
  229.     next_link=mi_sizekorr(buff);
  230.     records--;
  231.     param->key_file_blocks+=block_size;
  232.   }
  233.   if (param->testflag & T_VERBOSE)
  234.   {
  235.     if (next_link != HA_OFFSET_ERROR)
  236.       printf("%16sn",llstr(next_link,llbuff));
  237.     else
  238.       puts("");
  239.   }
  240.   DBUG_RETURN (next_link != HA_OFFSET_ERROR);
  241. } /* check_k_link */
  242. /* Check sizes of files */
  243. int chk_size(MI_CHECK *param, register MI_INFO *info)
  244. {
  245.   int error=0;
  246.   register my_off_t skr,size;
  247.   char buff[22],buff2[22];
  248.   DBUG_ENTER("chk_size");
  249.   if (!(param->testflag & T_SILENT)) puts("- check file-size");
  250.   /* The following is needed if called externally (not from myisamchk) */
  251.   flush_key_blocks(info->s->key_cache,
  252.    info->s->kfile, FLUSH_FORCE_WRITE);
  253.   size=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0));
  254.   if ((skr=(my_off_t) info->state->key_file_length) != size)
  255.   {
  256.     if (skr > size)
  257.     {
  258.       error=1;
  259.       mi_check_print_error(param,
  260.    "Size of indexfile is: %-8s        Should be: %s",
  261.    llstr(size,buff), llstr(skr,buff2));
  262.     }
  263.     else
  264.       mi_check_print_warning(param,
  265.      "Size of indexfile is: %-8s      Should be: %s",
  266.      llstr(size,buff), llstr(skr,buff2));
  267.   }
  268.   if (!(param->testflag & T_VERY_SILENT) &&
  269.       ! (info->s->options & HA_OPTION_COMPRESS_RECORD) &&
  270.       ulonglong2double(info->state->key_file_length) >
  271.       ulonglong2double(info->s->base.margin_key_file_length)*0.9)
  272.     mi_check_print_warning(param,"Keyfile is almost full, %10s of %10s used",
  273.    llstr(info->state->key_file_length,buff),
  274.    llstr(info->s->base.max_key_file_length-1,buff));
  275.   size=my_seek(info->dfile,0L,MY_SEEK_END,MYF(0));
  276.   skr=(my_off_t) info->state->data_file_length;
  277.   if (info->s->options & HA_OPTION_COMPRESS_RECORD)
  278.     skr+= MEMMAP_EXTRA_MARGIN;
  279. #ifdef USE_RELOC
  280.   if (info->data_file_type == STATIC_RECORD &&
  281.       skr < (my_off_t) info->s->base.reloc*info->s->base.min_pack_length)
  282.     skr=(my_off_t) info->s->base.reloc*info->s->base.min_pack_length;
  283. #endif
  284.   if (skr != size)
  285.   {
  286.     info->state->data_file_length=size; /* Skip other errors */
  287.     if (skr > size && skr != size + MEMMAP_EXTRA_MARGIN)
  288.     {
  289.       error=1;
  290.       mi_check_print_error(param,"Size of datafile is: %-9s         Should be: %s",
  291.     llstr(size,buff), llstr(skr,buff2));
  292.       param->testflag|=T_RETRY_WITHOUT_QUICK;
  293.     }
  294.     else
  295.     {
  296.       mi_check_print_warning(param,
  297.      "Size of datafile is: %-9s       Should be: %s",
  298.      llstr(size,buff), llstr(skr,buff2));
  299.     }
  300.   }
  301.   if (!(param->testflag & T_VERY_SILENT) &&
  302.       !(info->s->options & HA_OPTION_COMPRESS_RECORD) &&
  303.       ulonglong2double(info->state->data_file_length) >
  304.       (ulonglong2double(info->s->base.max_data_file_length)*0.9))
  305.     mi_check_print_warning(param, "Datafile is almost full, %10s of %10s used",
  306.    llstr(info->state->data_file_length,buff),
  307.    llstr(info->s->base.max_data_file_length-1,buff2));
  308.   DBUG_RETURN(error);
  309. } /* chk_size */
  310. /* Check keys */
  311. int chk_key(MI_CHECK *param, register MI_INFO *info)
  312. {
  313.   uint key,found_keys=0,full_text_keys=0,result=0;
  314.   ha_rows keys;
  315.   ha_checksum old_record_checksum,init_checksum;
  316.   my_off_t all_keydata,all_totaldata,key_totlength,length;
  317.   ulong   *rec_per_key_part;
  318.   MYISAM_SHARE *share=info->s;
  319.   MI_KEYDEF *keyinfo;
  320.   char buff[22],buff2[22];
  321.   DBUG_ENTER("chk_key");
  322.   if (!(param->testflag & T_SILENT))
  323.     puts("- check key delete-chain");
  324.   param->key_file_blocks=info->s->base.keystart;
  325.   for (key=0 ; key < info->s->state.header.max_block_size ; key++)
  326.     if (check_k_link(param,info,key))
  327.     {
  328.       if (param->testflag & T_VERBOSE) puts("");
  329.       mi_check_print_error(param,"key delete-link-chain corrupted");
  330.       DBUG_RETURN(-1);
  331.     }
  332.   if (!(param->testflag & T_SILENT)) puts("- check index reference");
  333.   all_keydata=all_totaldata=key_totlength=0;
  334.   old_record_checksum=0;
  335.   init_checksum=param->record_checksum;
  336.   if (!(share->options &
  337. (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)))
  338.     old_record_checksum=calc_checksum(info->state->records+info->state->del-1)*
  339.       share->base.pack_reclength;
  340.   rec_per_key_part= param->rec_per_key_part;
  341.   for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ;
  342.        rec_per_key_part+=keyinfo->keysegs, key++, keyinfo++)
  343.   {
  344.     param->key_crc[key]=0;
  345.     if (!(((ulonglong) 1 << key) & share->state.key_map))
  346.     {
  347.       /* Remember old statistics for key */
  348.       memcpy((char*) rec_per_key_part,
  349.      (char*) (share->state.rec_per_key_part +
  350.       (uint) (rec_per_key_part - param->rec_per_key_part)),
  351.      keyinfo->keysegs*sizeof(*rec_per_key_part));
  352.       continue;
  353.     }
  354.     found_keys++;
  355.     param->record_checksum=init_checksum;
  356.     
  357.     bzero((char*) &param->unique_count,sizeof(param->unique_count));
  358.     bzero((char*) &param->notnull_count,sizeof(param->notnull_count));
  359.     if ((!(param->testflag & T_SILENT)))
  360.       printf ("- check data record references index: %dn",key+1);
  361.     if (keyinfo->flag & HA_FULLTEXT)
  362.       full_text_keys++;
  363.     if (share->state.key_root[key] == HA_OFFSET_ERROR &&
  364. (info->state->records == 0 || keyinfo->flag & HA_FULLTEXT))
  365.       continue;
  366.     if (!_mi_fetch_keypage(info,keyinfo,share->state.key_root[key],
  367.                            DFLT_INIT_HITS,info->buff,0))
  368.     {
  369.       mi_check_print_error(param,"Can't read indexpage from filepos: %s",
  370.   llstr(share->state.key_root[key],buff));
  371.       if (!(param->testflag & T_INFO))
  372. DBUG_RETURN(-1);
  373.       result= -1;
  374.       continue;
  375.     }
  376.     param->key_file_blocks+=keyinfo->block_length;
  377.     keys=0;
  378.     param->keydata=param->totaldata=0;
  379.     param->key_blocks=0;
  380.     param->max_level=0;
  381.     if (chk_index(param,info,keyinfo,share->state.key_root[key],info->buff,
  382.   &keys, param->key_crc+key,1))
  383.       DBUG_RETURN(-1);
  384.     if(!(keyinfo->flag & (HA_FULLTEXT | HA_SPATIAL)))
  385.     {
  386.       if (keys != info->state->records)
  387.       {
  388. mi_check_print_error(param,"Found %s keys of %s",llstr(keys,buff),
  389.     llstr(info->state->records,buff2));
  390. if (!(param->testflag & T_INFO))
  391. DBUG_RETURN(-1);
  392. result= -1;
  393. continue;
  394.       }
  395.       if (found_keys - full_text_keys == 1 &&
  396.   ((share->options &
  397.     (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ||
  398.    (param->testflag & T_DONT_CHECK_CHECKSUM)))
  399. old_record_checksum=param->record_checksum;
  400.       else if (old_record_checksum != param->record_checksum)
  401.       {
  402. if (key)
  403.   mi_check_print_error(param,"Key %u doesn't point at same records that key 1",
  404.       key+1);
  405. else
  406.   mi_check_print_error(param,"Key 1 doesn't point at all records");
  407. if (!(param->testflag & T_INFO))
  408.   DBUG_RETURN(-1);
  409. result= -1;
  410. continue;
  411.       }
  412.     }
  413.     if ((uint) share->base.auto_key -1 == key)
  414.     {
  415.       /* Check that auto_increment key is bigger than max key value */
  416.       ulonglong save_auto_value=info->s->state.auto_increment;
  417.       info->s->state.auto_increment=0;
  418.       info->lastinx=key;
  419.       _mi_read_key_record(info, 0L, info->rec_buff);
  420.       update_auto_increment(info, info->rec_buff);
  421.       if (info->s->state.auto_increment > save_auto_value)
  422.       {
  423. mi_check_print_warning(param,
  424.        "Auto-increment value: %s is smaller than max used value: %s",
  425.        llstr(save_auto_value,buff2),
  426.        llstr(info->s->state.auto_increment, buff));
  427.       }
  428.       if (param->testflag & T_AUTO_INC)
  429.       {
  430. set_if_bigger(info->s->state.auto_increment,
  431.       param->auto_increment_value);
  432.       }
  433.       else
  434. info->s->state.auto_increment=save_auto_value;
  435.       /* Check that there isn't a row with auto_increment = 0 in the table */
  436.       mi_extra(info,HA_EXTRA_KEYREAD,0);
  437.       bzero(info->lastkey,keyinfo->seg->length);
  438.       if (!mi_rkey(info, info->rec_buff, key, (const byte*) info->lastkey,
  439.    keyinfo->seg->length, HA_READ_KEY_EXACT))
  440.       {
  441. /* Don't count this as a real warning, as myisamchk can't correct it */
  442. uint save=param->warning_printed;
  443. mi_check_print_warning(param,
  444.        "Found row where the auto_increment column has the value 0");
  445. param->warning_printed=save;
  446.       }
  447.       mi_extra(info,HA_EXTRA_NO_KEYREAD,0);
  448.     }
  449.     length=(my_off_t) isam_key_length(info,keyinfo)*keys + param->key_blocks*2;
  450.     if (param->testflag & T_INFO && param->totaldata != 0L && keys != 0L)
  451.       printf("Key: %2d:  Keyblocks used: %3d%%  Packed: %4d%%  Max levels: %2dn",
  452.      key+1,
  453.      (int) (my_off_t2double(param->keydata)*100.0/my_off_t2double(param->totaldata)),
  454.      (int) ((my_off_t2double(length) - my_off_t2double(param->keydata))*100.0/
  455.     my_off_t2double(length)),
  456.      param->max_level);
  457.     all_keydata+=param->keydata; all_totaldata+=param->totaldata; key_totlength+=length;
  458.     if (param->testflag & T_STATISTICS)
  459.       update_key_parts(keyinfo, rec_per_key_part, param->unique_count,
  460.                        param->stats_method == MI_STATS_METHOD_IGNORE_NULLS?
  461.                        param->notnull_count: NULL, 
  462.                        (ulonglong)info->state->records);
  463.   }
  464.   if (param->testflag & T_INFO)
  465.   {
  466.     if (all_totaldata != 0L && found_keys > 0)
  467.       printf("Total:    Keyblocks used: %3d%%  Packed: %4d%%nn",
  468.      (int) (my_off_t2double(all_keydata)*100.0/
  469.     my_off_t2double(all_totaldata)),
  470.      (int) ((my_off_t2double(key_totlength) -
  471.      my_off_t2double(all_keydata))*100.0/
  472.      my_off_t2double(key_totlength)));
  473.     else if (all_totaldata != 0L && share->state.key_map)
  474.       puts("");
  475.   }
  476.   if (param->key_file_blocks != info->state->key_file_length &&
  477.       param->keys_in_use != ~(ulonglong) 0)
  478.     mi_check_print_warning(param, "Some data are unreferenced in keyfile");
  479.   if (found_keys != full_text_keys)
  480.     param->record_checksum=old_record_checksum-init_checksum; /* Remove delete links */
  481.   else
  482.     param->record_checksum=0;
  483.   DBUG_RETURN(result);
  484. } /* chk_key */
  485. static int chk_index_down(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo,
  486.                      my_off_t page, uchar *buff, ha_rows *keys,
  487.                      ha_checksum *key_checksum, uint level)
  488. {
  489.   char llbuff[22],llbuff2[22];
  490.   if (page > info->state->key_file_length || (page & (info->s->blocksize -1)))
  491.   {
  492.     my_off_t max_length=my_seek(info->s->kfile,0L,MY_SEEK_END,MYF(0));
  493.     mi_check_print_error(param,"Wrong pagepointer: %s at page: %s",
  494.                 llstr(page,llbuff),llstr(page,llbuff2));
  495.     if (page+info->s->blocksize > max_length)
  496.       goto err;
  497.     info->state->key_file_length=(max_length &
  498.                                   ~ (my_off_t) (info->s->blocksize-1));
  499.   }
  500.   if (!_mi_fetch_keypage(info,keyinfo,page, DFLT_INIT_HITS,buff,0))
  501.   {
  502.     mi_check_print_error(param,"Can't read key from filepos: %s",
  503.         llstr(page,llbuff));
  504.     goto err;
  505.   }
  506.   param->key_file_blocks+=keyinfo->block_length;
  507.   if (chk_index(param,info,keyinfo,page,buff,keys,key_checksum,level))
  508.     goto err;
  509.   return 0;
  510. err:
  511.   return 1;
  512. }
  513. /*
  514.   "Ignore NULLs" statistics collection method: process first index tuple.
  515.   SYNOPSIS
  516.     mi_collect_stats_nonulls_first()
  517.       keyseg   IN     Array of key part descriptions
  518.       notnull  INOUT  Array, notnull[i] = (number of {keypart1...keypart_i}
  519.                                            tuples that don't contain NULLs)
  520.       key      IN     Key values tuple
  521.   DESCRIPTION
  522.     Process the first index tuple - find out which prefix tuples don't
  523.     contain NULLs, and update the array of notnull counters accordingly.
  524. */
  525. static
  526. void mi_collect_stats_nonulls_first(HA_KEYSEG *keyseg, ulonglong *notnull,
  527.                                     uchar *key)
  528. {
  529.   uint first_null, kp;
  530.   first_null= ha_find_null(keyseg, key) - keyseg;
  531.   /*
  532.     All prefix tuples that don't include keypart_{first_null} are not-null
  533.     tuples (and all others aren't), increment counters for them.
  534.   */
  535.   for (kp= 0; kp < first_null; kp++)
  536.     notnull[kp]++;
  537. }
  538. /*
  539.   "Ignore NULLs" statistics collection method: process next index tuple.
  540.   SYNOPSIS
  541.     mi_collect_stats_nonulls_next()
  542.       keyseg   IN     Array of key part descriptions
  543.       notnull  INOUT  Array, notnull[i] = (number of {keypart1...keypart_i}
  544.                                            tuples that don't contain NULLs)
  545.       prev_key IN     Previous key values tuple
  546.       last_key IN     Next key values tuple
  547.   DESCRIPTION
  548.     Process the next index tuple:
  549.     1. Find out which prefix tuples of last_key don't contain NULLs, and
  550.        update the array of notnull counters accordingly.
  551.     2. Find the first keypart number where the prev_key and last_key tuples
  552.        are different(A), or last_key has NULL value(B), and return it, so the 
  553.        caller can count number of unique tuples for each key prefix. We don't 
  554.        need (B) to be counted, and that is compensated back in 
  555.        update_key_parts().
  556.   RETURN
  557.     1 + number of first keypart where values differ or last_key tuple has NULL
  558. */
  559. static
  560. int mi_collect_stats_nonulls_next(HA_KEYSEG *keyseg, ulonglong *notnull,
  561.                                   uchar *prev_key, uchar *last_key)
  562. {
  563.   uint diffs[2];
  564.   uint first_null_seg, kp;
  565.   HA_KEYSEG *seg;
  566.   /* 
  567.      Find the first keypart where values are different or either of them is
  568.      NULL. We get results in diffs array:
  569.      diffs[0]= 1 + number of first different keypart
  570.      diffs[1]=offset: (last_key + diffs[1]) points to first value in
  571.                       last_key that is NULL or different from corresponding
  572.                       value in prev_key.
  573.   */
  574.   ha_key_cmp(keyseg, prev_key, last_key, USE_WHOLE_KEY, 
  575.              SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL, diffs);
  576.   seg= keyseg + diffs[0] - 1;
  577.   /* Find first NULL in last_key */
  578.   first_null_seg= ha_find_null(seg, last_key + diffs[1]) - keyseg;
  579.   for (kp= 0; kp < first_null_seg; kp++)
  580.     notnull[kp]++;
  581.   /* 
  582.     Return 1+ number of first key part where values differ. Don't care if
  583.     these were NULLs and not .... We compensate for that in
  584.     update_key_parts.
  585.   */
  586.   return diffs[0];
  587. }
  588. /* Check if index is ok */
  589. static int chk_index(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo,
  590.      my_off_t page, uchar *buff, ha_rows *keys,
  591.      ha_checksum *key_checksum, uint level)
  592. {
  593.   int flag;
  594.   uint used_length,comp_flag,nod_flag,key_length=0;
  595.   uchar key[MI_MAX_POSSIBLE_KEY_BUFF],*temp_buff,*keypos,*old_keypos,*endpos;
  596.   my_off_t next_page,record;
  597.   char llbuff[22];
  598.   uint diff_pos[2];
  599.   DBUG_ENTER("chk_index");
  600.   DBUG_DUMP("buff",(byte*) buff,mi_getint(buff));
  601.   /* TODO: implement appropriate check for RTree keys */
  602.   if (keyinfo->flag & HA_SPATIAL)
  603.     DBUG_RETURN(0);
  604.   if (!(temp_buff=(uchar*) my_alloca((uint) keyinfo->block_length)))
  605.   {
  606.     mi_check_print_error(param,"Not enough memory for keyblock");
  607.     DBUG_RETURN(-1);
  608.   }
  609.   if (keyinfo->flag & HA_NOSAME)
  610.     comp_flag=SEARCH_FIND | SEARCH_UPDATE; /* Not real duplicates */
  611.   else
  612.     comp_flag=SEARCH_SAME; /* Keys in positionorder */
  613.   nod_flag=mi_test_if_nod(buff);
  614.   used_length=mi_getint(buff);
  615.   keypos=buff+2+nod_flag;
  616.   endpos=buff+used_length;
  617.   param->keydata+=used_length; param->totaldata+=keyinfo->block_length; /* INFO */
  618.   param->key_blocks++;
  619.   if (level > param->max_level)
  620.     param->max_level=level;
  621.   if (used_length > keyinfo->block_length)
  622.   {
  623.     mi_check_print_error(param,"Wrong pageinfo at page: %s",
  624.  llstr(page,llbuff));
  625.     goto err;
  626.   }
  627.   for ( ;; )
  628.   {
  629.     if (*killed_ptr(param))
  630.       goto err;
  631.     memcpy((char*) info->lastkey,(char*) key,key_length);
  632.     info->lastkey_length=key_length;
  633.     if (nod_flag)
  634.     {
  635.       next_page=_mi_kpos(nod_flag,keypos);
  636.       if (chk_index_down(param,info,keyinfo,next_page,
  637.                          temp_buff,keys,key_checksum,level+1))
  638. goto err;
  639.     }
  640.     old_keypos=keypos;
  641.     if (keypos >= endpos ||
  642. (key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,key)) == 0)
  643.       break;
  644.     if (keypos > endpos)
  645.     {
  646.       mi_check_print_error(param,"Wrong key block length at page: %s",llstr(page,llbuff));
  647.       goto err;
  648.     }
  649.     if ((*keys)++ &&
  650. (flag=ha_key_cmp(keyinfo->seg,info->lastkey,key,key_length,
  651.  comp_flag, diff_pos)) >=0)
  652.     {
  653.       DBUG_DUMP("old",(byte*) info->lastkey, info->lastkey_length);
  654.       DBUG_DUMP("new",(byte*) key, key_length);
  655.       DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos));
  656.       if (comp_flag & SEARCH_FIND && flag == 0)
  657. mi_check_print_error(param,"Found duplicated key at page %s",llstr(page,llbuff));
  658.       else
  659. mi_check_print_error(param,"Key in wrong position at page %s",llstr(page,llbuff));
  660.       goto err;
  661.     }
  662.     if (param->testflag & T_STATISTICS)
  663.     {
  664.       if (*keys != 1L) /* not first_key */
  665.       {
  666.         if (param->stats_method == MI_STATS_METHOD_NULLS_NOT_EQUAL)
  667.           ha_key_cmp(keyinfo->seg,info->lastkey,key,USE_WHOLE_KEY,
  668.                      SEARCH_FIND | SEARCH_NULL_ARE_NOT_EQUAL,
  669.                      diff_pos);
  670.         else if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS)
  671.         {
  672.           diff_pos[0]= mi_collect_stats_nonulls_next(keyinfo->seg, 
  673.                                                   param->notnull_count,
  674.                                                   info->lastkey, key);
  675.         }
  676. param->unique_count[diff_pos[0]-1]++;
  677.       }
  678.       else
  679.       {  
  680.         if (param->stats_method == MI_STATS_METHOD_IGNORE_NULLS)
  681.           mi_collect_stats_nonulls_first(keyinfo->seg, param->notnull_count,
  682.                                          key);
  683.       }
  684.     }
  685.     (*key_checksum)+= mi_byte_checksum((byte*) key,
  686.        key_length- info->s->rec_reflength);
  687.     record= _mi_dpos(info,0,key+key_length);
  688.     if (keyinfo->flag & HA_FULLTEXT) /* special handling for ft2 */
  689.     {
  690.       uint off;
  691.       int  subkeys;
  692.       get_key_full_length_rdonly(off, key);
  693.       subkeys=ft_sintXkorr(key+off);
  694.       if (subkeys < 0)
  695.       {
  696.         ha_rows tmp_keys=0;
  697.         if (chk_index_down(param,info,&info->s->ft2_keyinfo,record,
  698.                            temp_buff,&tmp_keys,key_checksum,1))
  699.           goto err;
  700.         if (tmp_keys + subkeys)
  701.         {
  702.           mi_check_print_error(param,
  703.                                "Number of words in the 2nd level tree "
  704.                                "does not match the number in the header. "
  705.                                "Parent word in on the page %s, offset %u",
  706.                                llstr(page,llbuff), (uint) (old_keypos-buff));
  707.           goto err;
  708.         }
  709.         (*keys)+=tmp_keys-1;
  710.         continue;
  711.       }
  712.       /* fall through */
  713.     }
  714.     if (record >= info->state->data_file_length)
  715.     {
  716. #ifndef DBUG_OFF
  717.       char llbuff2[22], llbuff3[22];
  718. #endif
  719.       mi_check_print_error(param,"Found key at page %s that points to record outside datafile",llstr(page,llbuff));
  720.       DBUG_PRINT("test",("page: %s  record: %s  filelength: %s",
  721.  llstr(page,llbuff),llstr(record,llbuff2),
  722.  llstr(info->state->data_file_length,llbuff3)));
  723.       DBUG_DUMP("key",(byte*) key,key_length);
  724.       DBUG_DUMP("new_in_page",(char*) old_keypos,(uint) (keypos-old_keypos));
  725.       goto err;
  726.     }
  727.     param->record_checksum+=(ha_checksum) record;
  728.   }
  729.   if (keypos != endpos)
  730.   {
  731.     mi_check_print_error(param,"Keyblock size at page %s is not correct.  Block length: %d  key length: %d",
  732.                 llstr(page,llbuff), used_length, (keypos - buff));
  733.     goto err;
  734.   }
  735.   my_afree((byte*) temp_buff);
  736.   DBUG_RETURN(0);
  737.  err:
  738.   my_afree((byte*) temp_buff);
  739.   DBUG_RETURN(1);
  740. } /* chk_index */
  741. /* Calculate a checksum of 1+2+3+4...N = N*(N+1)/2 without overflow */
  742. static ha_checksum calc_checksum(ha_rows count)
  743. {
  744.   ulonglong sum,a,b;
  745.   DBUG_ENTER("calc_checksum");
  746.   sum=0;
  747.   a=count; b=count+1;
  748.   if (a & 1)
  749.     b>>=1;
  750.   else
  751.     a>>=1;
  752.   while (b)
  753.   {
  754.     if (b & 1)
  755.       sum+=a;
  756.     a<<=1; b>>=1;
  757.   }
  758.   DBUG_PRINT("exit",("sum: %lx",(ulong) sum));
  759.   DBUG_RETURN((ha_checksum) sum);
  760. } /* calc_checksum */
  761. /* Calc length of key in normal isam */
  762. static uint isam_key_length(MI_INFO *info, register MI_KEYDEF *keyinfo)
  763. {
  764.   uint length;
  765.   HA_KEYSEG *keyseg;
  766.   DBUG_ENTER("isam_key_length");
  767.   length= info->s->rec_reflength;
  768.   for (keyseg=keyinfo->seg ; keyseg->type ; keyseg++)
  769.     length+= keyseg->length;
  770.   DBUG_PRINT("exit",("length: %d",length));
  771.   DBUG_RETURN(length);
  772. } /* key_length */
  773. /* Check that record-link is ok */
  774. int chk_data_link(MI_CHECK *param, MI_INFO *info,int extend)
  775. {
  776.   int error,got_error,flag;
  777.   uint key,left_length,b_type,field;
  778.   ha_rows records,del_blocks;
  779.   my_off_t used,empty,pos,splits,start_recpos,
  780.    del_length,link_used,start_block;
  781.   byte *record,*to;
  782.   char llbuff[22],llbuff2[22],llbuff3[22];
  783.   ha_checksum intern_record_checksum;
  784.   ha_checksum key_checksum[MI_MAX_POSSIBLE_KEY];
  785.   my_bool static_row_size;
  786.   MI_KEYDEF *keyinfo;
  787.   MI_BLOCK_INFO block_info;
  788.   DBUG_ENTER("chk_data_link");
  789.   if (!(param->testflag & T_SILENT))
  790.   {
  791.     if (extend)
  792.       puts("- check records and index references");
  793.     else
  794.       puts("- check record links");
  795.   }
  796.   if (!(record= (byte*) my_malloc(info->s->base.pack_reclength,MYF(0))))
  797.   {
  798.     mi_check_print_error(param,"Not enough memory for record");
  799.     DBUG_RETURN(-1);
  800.   }
  801.   records=del_blocks=0;
  802.   used=link_used=splits=del_length=0;
  803.   intern_record_checksum=param->glob_crc=0;
  804.   LINT_INIT(left_length);  LINT_INIT(start_recpos);  LINT_INIT(to);
  805.   got_error=error=0;
  806.   empty=info->s->pack.header_length;
  807.   /* Check how to calculate checksum of rows */
  808.   static_row_size=1;
  809.   if (info->s->data_file_type == COMPRESSED_RECORD)
  810.   {
  811.     for (field=0 ; field < info->s->base.fields ; field++)
  812.     {
  813.       if (info->s->rec[field].base_type == FIELD_BLOB ||
  814.   info->s->rec[field].base_type == FIELD_VARCHAR)
  815.       {
  816. static_row_size=0;
  817. break;
  818.       }
  819.     }
  820.   }
  821.   pos=my_b_tell(&param->read_cache);
  822.   bzero((char*) key_checksum, info->s->base.keys * sizeof(key_checksum[0]));
  823.   while (pos < info->state->data_file_length)
  824.   {
  825.     if (*killed_ptr(param))
  826.       goto err2;
  827.     switch (info->s->data_file_type) {
  828.     case STATIC_RECORD:
  829.       if (my_b_read(&param->read_cache,(byte*) record,
  830.     info->s->base.pack_reclength))
  831. goto err;
  832.       start_recpos=pos;
  833.       pos+=info->s->base.pack_reclength;
  834.       splits++;
  835.       if (*record == '')
  836.       {
  837. del_blocks++;
  838. del_length+=info->s->base.pack_reclength;
  839. continue; /* Record removed */
  840.       }
  841.       param->glob_crc+= mi_static_checksum(info,record);
  842.       used+=info->s->base.pack_reclength;
  843.       break;
  844.     case DYNAMIC_RECORD:
  845.       flag=block_info.second_read=0;
  846.       block_info.next_filepos=pos;
  847.       do
  848.       {
  849. if (_mi_read_cache(&param->read_cache,(byte*) block_info.header,
  850.    (start_block=block_info.next_filepos),
  851.    sizeof(block_info.header),
  852.    (flag ? 0 : READING_NEXT) | READING_HEADER))
  853.   goto err;
  854. if (start_block & (MI_DYN_ALIGN_SIZE-1))
  855. {
  856.   mi_check_print_error(param,"Wrong aligned block at %s",
  857.        llstr(start_block,llbuff));
  858.   goto err2;
  859. }
  860. b_type=_mi_get_block_info(&block_info,-1,start_block);
  861. if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  862.       BLOCK_FATAL_ERROR))
  863. {
  864.   if (b_type & BLOCK_SYNC_ERROR)
  865.   {
  866.     if (flag)
  867.     {
  868.       mi_check_print_error(param,"Unexpected byte: %d at link: %s",
  869.   (int) block_info.header[0],
  870.   llstr(start_block,llbuff));
  871.       goto err2;
  872.     }
  873.     pos=block_info.filepos+block_info.block_len;
  874.     goto next;
  875.   }
  876.   if (b_type & BLOCK_DELETED)
  877.   {
  878.     if (block_info.block_len < info->s->base.min_block_length)
  879.     {
  880.       mi_check_print_error(param,
  881.    "Deleted block with impossible length %lu at %s",
  882.    block_info.block_len,llstr(pos,llbuff));
  883.       goto err2;
  884.     }
  885.     if ((block_info.next_filepos != HA_OFFSET_ERROR &&
  886.  block_info.next_filepos >= info->state->data_file_length) ||
  887. (block_info.prev_filepos != HA_OFFSET_ERROR &&
  888.  block_info.prev_filepos >= info->state->data_file_length))
  889.     {
  890.       mi_check_print_error(param,"Delete link points outside datafile at %s",
  891.   llstr(pos,llbuff));
  892.       goto err2;
  893.     }
  894.     del_blocks++;
  895.     del_length+=block_info.block_len;
  896.     pos=block_info.filepos+block_info.block_len;
  897.     splits++;
  898.     goto next;
  899.   }
  900.   mi_check_print_error(param,"Wrong bytesec: %d-%d-%d at linkstart: %s",
  901.        block_info.header[0],block_info.header[1],
  902.        block_info.header[2],
  903.        llstr(start_block,llbuff));
  904.   goto err2;
  905. }
  906. if (info->state->data_file_length < block_info.filepos+
  907.     block_info.block_len)
  908. {
  909.   mi_check_print_error(param,
  910.        "Recordlink that points outside datafile at %s",
  911.        llstr(pos,llbuff));
  912.   got_error=1;
  913.   break;
  914. }
  915. splits++;
  916. if (!flag++) /* First block */
  917. {
  918.   start_recpos=pos;
  919.   pos=block_info.filepos+block_info.block_len;
  920.   if (block_info.rec_len > (uint) info->s->base.max_pack_length)
  921.   {
  922.     mi_check_print_error(param,"Found too long record (%lu) at %s",
  923.  (ulong) block_info.rec_len,
  924.  llstr(start_recpos,llbuff));
  925.     got_error=1;
  926.     break;
  927.   }
  928.   if (info->s->base.blobs)
  929.   {
  930.     if (!(to= mi_alloc_rec_buff(info, block_info.rec_len,
  931. &info->rec_buff)))
  932.     {
  933.       mi_check_print_error(param,
  934.    "Not enough memory (%lu) for blob at %s",
  935.    (ulong) block_info.rec_len,
  936.    llstr(start_recpos,llbuff));
  937.       got_error=1;
  938.       break;
  939.     }
  940.   }
  941.   else
  942.     to= info->rec_buff;
  943.   left_length=block_info.rec_len;
  944. }
  945. if (left_length < block_info.data_len)
  946. {
  947.   mi_check_print_error(param,"Found too long record (%lu) at %s",
  948.        (ulong) block_info.data_len,
  949.        llstr(start_recpos,llbuff));
  950.   got_error=1;
  951.   break;
  952. }
  953. if (_mi_read_cache(&param->read_cache,(byte*) to,block_info.filepos,
  954.    (uint) block_info.data_len,
  955.    flag == 1 ? READING_NEXT : 0))
  956.   goto err;
  957. to+=block_info.data_len;
  958. link_used+= block_info.filepos-start_block;
  959. used+= block_info.filepos - start_block + block_info.data_len;
  960. empty+=block_info.block_len-block_info.data_len;
  961. left_length-=block_info.data_len;
  962. if (left_length)
  963. {
  964.   if (b_type & BLOCK_LAST)
  965.   {
  966.     mi_check_print_error(param,
  967.  "Wrong record length %s of %s at %s",
  968.  llstr(block_info.rec_len-left_length,llbuff),
  969.  llstr(block_info.rec_len, llbuff2),
  970.  llstr(start_recpos,llbuff3));
  971.     got_error=1;
  972.     break;
  973.   }
  974.   if (info->state->data_file_length < block_info.next_filepos)
  975.   {
  976.     mi_check_print_error(param,
  977.  "Found next-recordlink that points outside datafile at %s",
  978.  llstr(block_info.filepos,llbuff));
  979.     got_error=1;
  980.     break;
  981.   }
  982. }
  983.       } while (left_length);
  984.       if (! got_error)
  985.       {
  986. if (_mi_rec_unpack(info,record,info->rec_buff,block_info.rec_len) ==
  987.     MY_FILE_ERROR)
  988. {
  989.   mi_check_print_error(param,"Found wrong record at %s",
  990.        llstr(start_recpos,llbuff));
  991.   got_error=1;
  992. }
  993. else
  994. {
  995.   info->checksum=mi_checksum(info,record);
  996.   if (param->testflag & (T_EXTEND | T_MEDIUM | T_VERBOSE))
  997.   {
  998.     if (_mi_rec_check(info,record, info->rec_buff,block_info.rec_len,
  999.                               test(info->s->calc_checksum)))
  1000.     {
  1001.       mi_check_print_error(param,"Found wrong packed record at %s",
  1002.   llstr(start_recpos,llbuff));
  1003.       got_error=1;
  1004.     }
  1005.   }
  1006.   if (!got_error)
  1007.     param->glob_crc+= info->checksum;
  1008. }
  1009.       }
  1010.       else if (!flag)
  1011. pos=block_info.filepos+block_info.block_len;
  1012.       break;
  1013.     case COMPRESSED_RECORD:
  1014.       if (_mi_read_cache(&param->read_cache,(byte*) block_info.header, pos,
  1015.  info->s->pack.ref_length, READING_NEXT))
  1016. goto err;
  1017.       start_recpos=pos;
  1018.       splits++;
  1019.       VOID(_mi_pack_get_block_info(info,&block_info, -1, start_recpos));
  1020.       pos=block_info.filepos+block_info.rec_len;
  1021.       if (block_info.rec_len < (uint) info->s->min_pack_length ||
  1022.   block_info.rec_len > (uint) info->s->max_pack_length)
  1023.       {
  1024. mi_check_print_error(param,
  1025.      "Found block with wrong recordlength: %d at %s",
  1026.      block_info.rec_len, llstr(start_recpos,llbuff));
  1027. got_error=1;
  1028. break;
  1029.       }
  1030.       if (_mi_read_cache(&param->read_cache,(byte*) info->rec_buff,
  1031. block_info.filepos, block_info.rec_len, READING_NEXT))
  1032. goto err;
  1033.       if (_mi_pack_rec_unpack(info,record,info->rec_buff,block_info.rec_len))
  1034.       {
  1035. mi_check_print_error(param,"Found wrong record at %s",
  1036.      llstr(start_recpos,llbuff));
  1037. got_error=1;
  1038.       }
  1039.       if (static_row_size)
  1040. param->glob_crc+= mi_static_checksum(info,record);
  1041.       else
  1042. param->glob_crc+= mi_checksum(info,record);
  1043.       link_used+= (block_info.filepos - start_recpos);
  1044.       used+= (pos-start_recpos);
  1045.     } /* switch */
  1046.     if (! got_error)
  1047.     {
  1048.       intern_record_checksum+=(ha_checksum) start_recpos;
  1049.       records++;
  1050.       if (param->testflag & T_WRITE_LOOP && records % WRITE_COUNT == 0)
  1051.       {
  1052. printf("%sr", llstr(records,llbuff)); VOID(fflush(stdout));
  1053.       }
  1054.       /* Check if keys match the record */
  1055.       for (key=0,keyinfo= info->s->keyinfo; key < info->s->base.keys;
  1056.    key++,keyinfo++)
  1057.       {
  1058. if ((((ulonglong) 1 << key) & info->s->state.key_map))
  1059. {
  1060.           if(!(keyinfo->flag & HA_FULLTEXT))
  1061.   {
  1062.     uint key_length=_mi_make_key(info,key,info->lastkey,record,
  1063.  start_recpos);
  1064.     if (extend)
  1065.     {
  1066.       /* We don't need to lock the key tree here as we don't allow
  1067.  concurrent threads when running myisamchk
  1068.       */
  1069.               int search_result= (keyinfo->flag & HA_SPATIAL) ?
  1070.                 rtree_find_first(info, key, info->lastkey, key_length,
  1071.                                  SEARCH_SAME) : 
  1072.                 _mi_search(info,keyinfo,info->lastkey,key_length,
  1073.                            SEARCH_SAME, info->s->state.key_root[key]);
  1074.               if (search_result)
  1075.               {
  1076.                 mi_check_print_error(param,"Record at: %10s  Can't find key for index: %2d",
  1077.                                      llstr(start_recpos,llbuff),key+1);
  1078.                 if (error++ > MAXERR || !(param->testflag & T_VERBOSE))
  1079.                   goto err2;
  1080.               }
  1081.     }
  1082.     else
  1083.       key_checksum[key]+=mi_byte_checksum((byte*) info->lastkey,
  1084.   key_length);
  1085.   }
  1086. }
  1087.       }
  1088.     }
  1089.     else
  1090.     {
  1091.       got_error=0;
  1092.       if (error++ > MAXERR || !(param->testflag & T_VERBOSE))
  1093. goto err2;
  1094.     }
  1095.   next:; /* Next record */
  1096.   }
  1097.   if (param->testflag & T_WRITE_LOOP)
  1098.   {
  1099.     VOID(fputs("          r",stdout)); VOID(fflush(stdout));
  1100.   }
  1101.   if (records != info->state->records)
  1102.   {
  1103.     mi_check_print_error(param,"Record-count is not ok; is %-10s   Should be: %s",
  1104. llstr(records,llbuff), llstr(info->state->records,llbuff2));
  1105.     error=1;
  1106.   }
  1107.   else if (param->record_checksum &&
  1108.    param->record_checksum != intern_record_checksum)
  1109.   {
  1110.     mi_check_print_error(param,
  1111.  "Keypointers and record positions doesn't match");
  1112.     error=1;
  1113.   }
  1114.   else if (param->glob_crc != info->s->state.checksum &&
  1115.    (info->s->options &
  1116.     (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD)))
  1117.   {
  1118.     mi_check_print_warning(param,
  1119.    "Record checksum is not the same as checksum stored in the index filen");
  1120.     error=1;
  1121.   }
  1122.   else if (!extend)
  1123.   {
  1124.     for (key=0 ; key < info->s->base.keys;  key++)
  1125.     {
  1126.       if (key_checksum[key] != param->key_crc[key] &&
  1127.           !(info->s->keyinfo[key].flag & (HA_FULLTEXT | HA_SPATIAL)))
  1128.       {
  1129. mi_check_print_error(param,"Checksum for key: %2d doesn't match checksum for records",
  1130.     key+1);
  1131. error=1;
  1132.       }
  1133.     }
  1134.   }
  1135.   if (del_length != info->state->empty)
  1136.   {
  1137.     mi_check_print_warning(param,
  1138.    "Found %s deleted space.   Should be %s",
  1139.    llstr(del_length,llbuff2),
  1140.    llstr(info->state->empty,llbuff));
  1141.   }
  1142.   if (used+empty+del_length != info->state->data_file_length)
  1143.   {
  1144.     mi_check_print_warning(param,
  1145.    "Found %s record-data and %s unused data and %s deleted-data",
  1146.    llstr(used,llbuff),llstr(empty,llbuff2),
  1147.    llstr(del_length,llbuff3));
  1148.     mi_check_print_warning(param,
  1149.    "Total %s, Should be: %s",
  1150.    llstr((used+empty+del_length),llbuff),
  1151.    llstr(info->state->data_file_length,llbuff2));
  1152.   }
  1153.   if (del_blocks != info->state->del)
  1154.   {
  1155.     mi_check_print_warning(param,
  1156.    "Found %10s deleted blocks       Should be: %s",
  1157.    llstr(del_blocks,llbuff),
  1158.    llstr(info->state->del,llbuff2));
  1159.   }
  1160.   if (splits != info->s->state.split)
  1161.   {
  1162.     mi_check_print_warning(param,
  1163.    "Found %10s parts                Should be: %s parts",
  1164.    llstr(splits,llbuff),
  1165.    llstr(info->s->state.split,llbuff2));
  1166.   }
  1167.   if (param->testflag & T_INFO)
  1168.   {
  1169.     if (param->warning_printed || param->error_printed)
  1170.       puts("");
  1171.     if (used != 0 && ! param->error_printed)
  1172.     {
  1173.       printf("Records:%18s    M.recordlength:%9lu   Packed:%14.0f%%n",
  1174.      llstr(records,llbuff), (long)((used-link_used)/records),
  1175.      (info->s->base.blobs ? 0.0 :
  1176.       (ulonglong2double((ulonglong) info->s->base.reclength*records)-
  1177.        my_off_t2double(used))/
  1178.       ulonglong2double((ulonglong) info->s->base.reclength*records)*100.0));
  1179.       printf("Recordspace used:%9.0f%%   Empty space:%12d%%  Blocks/Record: %6.2fn",
  1180.      (ulonglong2double(used-link_used)/ulonglong2double(used-link_used+empty)*100.0),
  1181.      (!records ? 100 : (int) (ulonglong2double(del_length+empty)/
  1182.       my_off_t2double(used)*100.0)),
  1183.      ulonglong2double(splits - del_blocks) / records);
  1184.     }
  1185.     printf("Record blocks:%12s    Delete blocks:%10sn",
  1186.    llstr(splits-del_blocks,llbuff),llstr(del_blocks,llbuff2));
  1187.     printf("Record data:  %12s    Deleted data: %10sn",
  1188.    llstr(used-link_used,llbuff),llstr(del_length,llbuff2));
  1189.     printf("Lost space:   %12s    Linkdata:     %10sn",
  1190.    llstr(empty,llbuff),llstr(link_used,llbuff2));
  1191.   }
  1192.   my_free((gptr) record,MYF(0));
  1193.   DBUG_RETURN (error);
  1194.  err:
  1195.   mi_check_print_error(param,"got error: %d when reading datafile at record: %s",my_errno, llstr(records,llbuff));
  1196.  err2:
  1197.   my_free((gptr) record,MYF(0));
  1198.   param->testflag|=T_RETRY_WITHOUT_QUICK;
  1199.   DBUG_RETURN(1);
  1200. } /* chk_data_link */
  1201. /* Recover old table by reading each record and writing all keys */
  1202. /* Save new datafile-name in temp_filename */
  1203. int mi_repair(MI_CHECK *param, register MI_INFO *info,
  1204.       my_string name, int rep_quick)
  1205. {
  1206.   int error,got_error;
  1207.   uint i;
  1208.   ha_rows start_records,new_header_length;
  1209.   my_off_t del;
  1210.   File new_file;
  1211.   MYISAM_SHARE *share=info->s;
  1212.   char llbuff[22],llbuff2[22];
  1213.   SORT_INFO sort_info;
  1214.   MI_SORT_PARAM sort_param;
  1215.   DBUG_ENTER("mi_repair");
  1216.   bzero((char *)&sort_info, sizeof(sort_info));
  1217.   bzero((char *)&sort_param, sizeof(sort_param));
  1218.   start_records=info->state->records;
  1219.   new_header_length=(param->testflag & T_UNPACK) ? 0L :
  1220.     share->pack.header_length;
  1221.   got_error=1;
  1222.   new_file= -1;
  1223.   sort_param.sort_info=&sort_info;
  1224.   if (!(param->testflag & T_SILENT))
  1225.   {
  1226.     printf("- recovering (with keycache) MyISAM-table '%s'n",name);
  1227.     printf("Data records: %sn", llstr(info->state->records,llbuff));
  1228.   }
  1229.   param->testflag|=T_REP; /* for easy checking */
  1230.   if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD))
  1231.     param->testflag|=T_CALC_CHECKSUM;
  1232.   if (!param->using_global_keycache)
  1233.     VOID(init_key_cache(dflt_key_cache, param->key_cache_block_size,
  1234.                         param->use_buffers, 0, 0));
  1235.   if (init_io_cache(&param->read_cache,info->dfile,
  1236.     (uint) param->read_buffer_length,
  1237.     READ_CACHE,share->pack.header_length,1,MYF(MY_WME)))
  1238.   {
  1239.     bzero(&info->rec_cache,sizeof(info->rec_cache));
  1240.     goto err;
  1241.   }
  1242.   if (!rep_quick)
  1243.     if (init_io_cache(&info->rec_cache,-1,(uint) param->write_buffer_length,
  1244.       WRITE_CACHE, new_header_length, 1,
  1245.       MYF(MY_WME | MY_WAIT_IF_FULL)))
  1246.       goto err;
  1247.   info->opt_flag|=WRITE_CACHE_USED;
  1248.   if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength,
  1249.    MYF(0))) ||
  1250.       !mi_alloc_rec_buff(info, -1, &sort_param.rec_buff))
  1251.   {
  1252.     mi_check_print_error(param, "Not enough memory for extra record");
  1253.     goto err;
  1254.   }
  1255.   if (!rep_quick)
  1256.   {
  1257.     /* Get real path for data file */
  1258.     if ((new_file=my_raid_create(fn_format(param->temp_filename,
  1259.    share->data_file_name, "",
  1260.    DATA_TMP_EXT, 2+4),
  1261.  0,param->tmpfile_createflag,
  1262.  share->base.raid_type,
  1263.  share->base.raid_chunks,
  1264.  share->base.raid_chunksize,
  1265.  MYF(0))) < 0)
  1266.     {
  1267.       mi_check_print_error(param,"Can't create new tempfile: '%s'",
  1268.    param->temp_filename);
  1269.       goto err;
  1270.     }
  1271.     if (filecopy(param,new_file,info->dfile,0L,new_header_length,
  1272.  "datafile-header"))
  1273.       goto err;
  1274.     info->s->state.dellink= HA_OFFSET_ERROR;
  1275.     info->rec_cache.file=new_file;
  1276.     if (param->testflag & T_UNPACK)
  1277.     {
  1278.       share->options&= ~HA_OPTION_COMPRESS_RECORD;
  1279.       mi_int2store(share->state.header.options,share->options);
  1280.     }
  1281.   }
  1282.   sort_info.info=info;
  1283.   sort_info.param = param;
  1284.   sort_param.read_cache=param->read_cache;
  1285.   sort_param.pos=sort_param.max_pos=share->pack.header_length;
  1286.   sort_param.filepos=new_header_length;
  1287.   param->read_cache.end_of_file=sort_info.filelength=
  1288.     my_seek(info->dfile,0L,MY_SEEK_END,MYF(0));
  1289.   sort_info.dupp=0;
  1290.   sort_param.fix_datafile= (my_bool) (! rep_quick);
  1291.   sort_param.master=1;
  1292.   sort_info.max_records= ~(ha_rows) 0;
  1293.   set_data_file_type(&sort_info, share);
  1294.   del=info->state->del;
  1295.   info->state->records=info->state->del=share->state.split=0;
  1296.   info->state->empty=0;
  1297.   param->glob_crc=0;
  1298.   if (param->testflag & T_CALC_CHECKSUM)
  1299.     param->calc_checksum=1;
  1300.   info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED);
  1301.   for (i=0 ; i < info->s->base.keys ; i++)
  1302.     share->state.key_root[i]= HA_OFFSET_ERROR;
  1303.   for (i=0 ; i < share->state.header.max_block_size ; i++)
  1304.     share->state.key_del[i]=  HA_OFFSET_ERROR;
  1305.   /*
  1306.     I think mi_repair and mi_repair_by_sort should do the same
  1307.     (according, e.g. to ha_myisam::repair), but as mi_repair doesn't
  1308.     touch key_map it cannot be used to T_CREATE_MISSING_KEYS.
  1309.     That is what the next line is for
  1310.   */
  1311.   if (param->testflag & T_CREATE_MISSING_KEYS)
  1312.     share->state.key_map= ((((ulonglong) 1L << share->base.keys)-1) &
  1313.    param->keys_in_use);
  1314.   info->state->key_file_length=share->base.keystart;
  1315.   lock_memory(param); /* Everything is alloced */
  1316.   while (!(error=sort_get_next_record(&sort_param)))
  1317.   {
  1318.     if (writekeys(param,info,(byte*)sort_param.record,sort_param.filepos))
  1319.     {
  1320.       if (my_errno != HA_ERR_FOUND_DUPP_KEY)
  1321. goto err;
  1322.       DBUG_DUMP("record",(byte*) sort_param.record,share->base.pack_reclength);
  1323.       mi_check_print_info(param,"Duplicate key %2d for record at %10s against new record at %10s",
  1324.   info->errkey+1,
  1325.   llstr(sort_param.start_recpos,llbuff),
  1326.   llstr(info->dupp_key_pos,llbuff2));
  1327.       if (param->testflag & T_VERBOSE)
  1328.       {
  1329. VOID(_mi_make_key(info,(uint) info->errkey,info->lastkey,
  1330.   sort_param.record,0L));
  1331. _mi_print_key(stdout,share->keyinfo[info->errkey].seg,info->lastkey,
  1332.       USE_WHOLE_KEY);
  1333.       }
  1334.       sort_info.dupp++;
  1335.       if ((param->testflag & (T_FORCE_UNIQUENESS|T_QUICK)) == T_QUICK)
  1336.       {
  1337.         param->testflag|=T_RETRY_WITHOUT_QUICK;
  1338. param->error_printed=1;
  1339. goto err;
  1340.       }
  1341.       continue;
  1342.     }
  1343.     if (sort_write_record(&sort_param))
  1344.       goto err;
  1345.   }
  1346.   if (error > 0 || write_data_suffix(&sort_info, (my_bool)!rep_quick) ||
  1347.       flush_io_cache(&info->rec_cache) || param->read_cache.error < 0)
  1348.     goto err;
  1349.   if (param->testflag & T_WRITE_LOOP)
  1350.   {
  1351.     VOID(fputs("          r",stdout)); VOID(fflush(stdout));
  1352.   }
  1353.   if (my_chsize(share->kfile,info->state->key_file_length,0,MYF(0)))
  1354.   {
  1355.     mi_check_print_warning(param,
  1356.    "Can't change size of indexfile, error: %d",
  1357.    my_errno);
  1358.     goto err;
  1359.   }
  1360.   if (rep_quick && del+sort_info.dupp != info->state->del)
  1361.   {
  1362.     mi_check_print_error(param,"Couldn't fix table with quick recovery: Found wrong number of deleted records");
  1363.     mi_check_print_error(param,"Run recovery again without -q");
  1364.     got_error=1;
  1365.     param->retry_repair=1;
  1366.     param->testflag|=T_RETRY_WITHOUT_QUICK;
  1367.     goto err;
  1368.   }
  1369.   if (param->testflag & T_SAFE_REPAIR)
  1370.   {
  1371.     /* Don't repair if we loosed more than one row */
  1372.     if (info->state->records+1 < start_records)
  1373.     {
  1374.       info->state->records=start_records;
  1375.       got_error=1;
  1376.       goto err;
  1377.     }
  1378.   }
  1379.   if (!rep_quick)
  1380.   {
  1381.     my_close(info->dfile,MYF(0));
  1382.     info->dfile=new_file;
  1383.     info->state->data_file_length=sort_param.filepos;
  1384.     share->state.version=(ulong) time((time_t*) 0); /* Force reopen */
  1385.   }
  1386.   else
  1387.   {
  1388.     info->state->data_file_length=sort_param.max_pos;
  1389.   }
  1390.   if (param->testflag & T_CALC_CHECKSUM)
  1391.     share->state.checksum=param->glob_crc;
  1392.   if (!(param->testflag & T_SILENT))
  1393.   {
  1394.     if (start_records != info->state->records)
  1395.       printf("Data records: %sn", llstr(info->state->records,llbuff));
  1396.     if (sort_info.dupp)
  1397.       mi_check_print_warning(param,
  1398.      "%s records have been removed",
  1399.      llstr(sort_info.dupp,llbuff));
  1400.   }
  1401.   got_error=0;
  1402.   /* If invoked by external program that uses thr_lock */
  1403.   if (&share->state.state != info->state)
  1404.     memcpy( &share->state.state, info->state, sizeof(*info->state));
  1405. err:
  1406.   if (!got_error)
  1407.   {
  1408.     /* Replace the actual file with the temporary file */
  1409.     if (new_file >= 0)
  1410.     {
  1411.       my_close(new_file,MYF(0));
  1412.       info->dfile=new_file= -1;
  1413.       if (change_to_newfile(share->data_file_name,MI_NAME_DEXT,
  1414.     DATA_TMP_EXT, share->base.raid_chunks,
  1415.     (param->testflag & T_BACKUP_DATA ?
  1416.      MYF(MY_REDEL_MAKE_BACKUP): MYF(0))) ||
  1417.   mi_open_datafile(info,share,-1))
  1418. got_error=1;
  1419.     }
  1420.   }
  1421.   if (got_error)
  1422.   {
  1423.     if (! param->error_printed)
  1424.       mi_check_print_error(param,"%d for record at pos %s",my_errno,
  1425.   llstr(sort_param.start_recpos,llbuff));
  1426.     if (new_file >= 0)
  1427.     {
  1428.       VOID(my_close(new_file,MYF(0)));
  1429.       VOID(my_raid_delete(param->temp_filename,info->s->base.raid_chunks,
  1430.   MYF(MY_WME)));
  1431.       info->rec_cache.file=-1; /* don't flush data to new_file, it's closed */
  1432.     }
  1433.     mi_mark_crashed_on_repair(info);
  1434.   }
  1435.   my_free(mi_get_rec_buff_ptr(info, sort_param.rec_buff),
  1436.                             MYF(MY_ALLOW_ZERO_PTR));
  1437.   my_free(sort_param.record,MYF(MY_ALLOW_ZERO_PTR));
  1438.   my_free(sort_info.buff,MYF(MY_ALLOW_ZERO_PTR));
  1439.   VOID(end_io_cache(&param->read_cache));
  1440.   info->opt_flag&= ~(READ_CACHE_USED | WRITE_CACHE_USED);
  1441.   VOID(end_io_cache(&info->rec_cache));
  1442.   got_error|=flush_blocks(param, share->key_cache, share->kfile);
  1443.   if (!got_error && param->testflag & T_UNPACK)
  1444.   {
  1445.     share->state.header.options[0]&= (uchar) ~HA_OPTION_COMPRESS_RECORD;
  1446.     share->pack.header_length=0;
  1447.     share->data_file_type=sort_info.new_data_file_type;
  1448.   }
  1449.   share->state.changed|= (STATE_NOT_OPTIMIZED_KEYS | STATE_NOT_SORTED_PAGES |
  1450.   STATE_NOT_ANALYZED);
  1451.   DBUG_RETURN(got_error);
  1452. }
  1453. /* Uppate keyfile when doing repair */
  1454. static int writekeys(MI_CHECK *param, register MI_INFO *info, byte *buff,
  1455.      my_off_t filepos)
  1456. {
  1457.   register uint i;
  1458.   uchar *key;
  1459.   DBUG_ENTER("writekeys");
  1460.   key=info->lastkey+info->s->base.max_key_length;
  1461.   for (i=0 ; i < info->s->base.keys ; i++)
  1462.   {
  1463.     if (((ulonglong) 1 << i) & info->s->state.key_map)
  1464.     {
  1465.       if (info->s->keyinfo[i].flag & HA_FULLTEXT )
  1466.       {
  1467.         if (_mi_ft_add(info,i,(char*) key,buff,filepos))
  1468.   goto err;
  1469.       }
  1470. #ifdef HAVE_SPATIAL
  1471.       else if (info->s->keyinfo[i].flag & HA_SPATIAL)
  1472.       {
  1473. uint key_length=_mi_make_key(info,i,key,buff,filepos);
  1474. if (rtree_insert(info, i, key, key_length))
  1475.   goto err;
  1476.       }
  1477. #endif /*HAVE_SPATIAL*/
  1478.       else
  1479.       {
  1480. uint key_length=_mi_make_key(info,i,key,buff,filepos);
  1481. if (_mi_ck_write(info,i,key,key_length))
  1482.   goto err;
  1483.       }
  1484.     }
  1485.   }
  1486.   DBUG_RETURN(0);
  1487.  err:
  1488.   if (my_errno == HA_ERR_FOUND_DUPP_KEY)
  1489.   {
  1490.     info->errkey=(int) i; /* This key was found */
  1491.     while ( i-- > 0 )
  1492.     {
  1493.       if (((ulonglong) 1 << i) & info->s->state.key_map)
  1494.       {
  1495. if (info->s->keyinfo[i].flag & HA_FULLTEXT)
  1496.         {
  1497.           if (_mi_ft_del(info,i,(char*) key,buff,filepos))
  1498.     break;
  1499.         }
  1500.         else
  1501. {
  1502.   uint key_length=_mi_make_key(info,i,key,buff,filepos);
  1503.   if (_mi_ck_delete(info,i,key,key_length))
  1504.     break;
  1505. }
  1506.       }
  1507.     }
  1508.   }
  1509.   /* Remove checksum that was added to glob_crc in sort_get_next_record */
  1510.   if (param->calc_checksum)
  1511.     param->glob_crc-= info->checksum;
  1512.   DBUG_PRINT("error",("errno: %d",my_errno));
  1513.   DBUG_RETURN(-1);
  1514. } /* writekeys */
  1515. /* Change all key-pointers that points to a records */
  1516. int movepoint(register MI_INFO *info, byte *record, my_off_t oldpos,
  1517.       my_off_t newpos, uint prot_key)
  1518. {
  1519.   register uint i;
  1520.   uchar *key;
  1521.   uint key_length;
  1522.   DBUG_ENTER("movepoint");
  1523.   key=info->lastkey+info->s->base.max_key_length;
  1524.   for (i=0 ; i < info->s->base.keys; i++)
  1525.   {
  1526.     if (i != prot_key && (((ulonglong) 1 << i) & info->s->state.key_map))
  1527.     {
  1528.       key_length=_mi_make_key(info,i,key,record,oldpos);
  1529.       if (info->s->keyinfo[i].flag & HA_NOSAME)
  1530.       { /* Change pointer direct */
  1531. uint nod_flag;
  1532. MI_KEYDEF *keyinfo;
  1533. keyinfo=info->s->keyinfo+i;
  1534. if (_mi_search(info,keyinfo,key,USE_WHOLE_KEY,
  1535.        (uint) (SEARCH_SAME | SEARCH_SAVE_BUFF),
  1536.        info->s->state.key_root[i]))
  1537.   DBUG_RETURN(-1);
  1538. nod_flag=mi_test_if_nod(info->buff);
  1539. _mi_dpointer(info,info->int_keypos-nod_flag-
  1540.      info->s->rec_reflength,newpos);
  1541. if (_mi_write_keypage(info,keyinfo,info->last_keypage,
  1542.                               DFLT_INIT_HITS,info->buff))
  1543.   DBUG_RETURN(-1);
  1544.       }
  1545.       else
  1546.       { /* Change old key to new */
  1547. if (_mi_ck_delete(info,i,key,key_length))
  1548.   DBUG_RETURN(-1);
  1549. key_length=_mi_make_key(info,i,key,record,newpos);
  1550. if (_mi_ck_write(info,i,key,key_length))
  1551.   DBUG_RETURN(-1);
  1552.       }
  1553.     }
  1554.   }
  1555.   DBUG_RETURN(0);
  1556. } /* movepoint */
  1557. /* Tell system that we want all memory for our cache */
  1558. void lock_memory(MI_CHECK *param __attribute__((unused)))
  1559. {
  1560. #ifdef SUN_OS /* Key-cacheing thrases on sun 4.1 */
  1561.   if (param->opt_lock_memory)
  1562.   {
  1563.     int success = mlockall(MCL_CURRENT); /* or plock(DATLOCK); */
  1564.     if (geteuid() == 0 && success != 0)
  1565.       mi_check_print_warning(param,
  1566.      "Failed to lock memory. errno %d",my_errno);
  1567.   }
  1568. #endif
  1569. } /* lock_memory */
  1570. /* Flush all changed blocks to disk */
  1571. int flush_blocks(MI_CHECK *param, KEY_CACHE *key_cache, File file)
  1572. {
  1573.   if (flush_key_blocks(key_cache, file, FLUSH_RELEASE))
  1574.   {
  1575.     mi_check_print_error(param,"%d when trying to write bufferts",my_errno);
  1576.     return(1);
  1577.   }
  1578.   if (!param->using_global_keycache)
  1579.     end_key_cache(key_cache,1);
  1580.   return 0;
  1581. } /* flush_blocks */
  1582. /* Sort index for more efficent reads */
  1583. int mi_sort_index(MI_CHECK *param, register MI_INFO *info, my_string name)
  1584. {
  1585.   reg2 uint key;
  1586.   reg1 MI_KEYDEF *keyinfo;
  1587.   File new_file;
  1588.   my_off_t index_pos[MI_MAX_POSSIBLE_KEY];
  1589.   uint r_locks,w_locks;
  1590.   int old_lock;
  1591.   MYISAM_SHARE *share=info->s;
  1592.   MI_STATE_INFO old_state;
  1593.   DBUG_ENTER("mi_sort_index");
  1594.   if (!(param->testflag & T_SILENT))
  1595.     printf("- Sorting index for MyISAM-table '%s'n",name);
  1596.   /* Get real path for index file */
  1597.   fn_format(param->temp_filename,name,"", MI_NAME_IEXT,2+4+32);
  1598.   if ((new_file=my_create(fn_format(param->temp_filename,param->temp_filename,
  1599.     "", INDEX_TMP_EXT,2+4),
  1600.   0,param->tmpfile_createflag,MYF(0))) <= 0)
  1601.   {
  1602.     mi_check_print_error(param,"Can't create new tempfile: '%s'",
  1603.  param->temp_filename);
  1604.     DBUG_RETURN(-1);
  1605.   }
  1606.   if (filecopy(param, new_file,share->kfile,0L,
  1607.        (ulong) share->base.keystart, "headerblock"))
  1608.     goto err;
  1609.   param->new_file_pos=share->base.keystart;
  1610.   for (key= 0,keyinfo= &share->keyinfo[0]; key < share->base.keys ;
  1611.        key++,keyinfo++)
  1612.   {
  1613.     if (!(((ulonglong) 1 << key) & share->state.key_map))
  1614.       continue;
  1615.     if (share->state.key_root[key] != HA_OFFSET_ERROR)
  1616.     {
  1617.       index_pos[key]=param->new_file_pos; /* Write first block here */
  1618.       if (sort_one_index(param,info,keyinfo,share->state.key_root[key],
  1619.  new_file))
  1620. goto err;
  1621.     }
  1622.     else
  1623.       index_pos[key]= HA_OFFSET_ERROR; /* No blocks */
  1624.   }
  1625.   /* Flush key cache for this file if we are calling this outside myisamchk */
  1626.   flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED);
  1627.   share->state.version=(ulong) time((time_t*) 0);
  1628.   old_state= share->state; /* save state if not stored */
  1629.   r_locks=   share->r_locks;
  1630.   w_locks=   share->w_locks;
  1631.   old_lock=  info->lock_type;
  1632. /* Put same locks as old file */
  1633.   share->r_locks= share->w_locks= share->tot_locks= 0;
  1634.   (void) _mi_writeinfo(info,WRITEINFO_UPDATE_KEYFILE);
  1635.   VOID(my_close(share->kfile,MYF(MY_WME)));
  1636.   share->kfile = -1;
  1637.   VOID(my_close(new_file,MYF(MY_WME)));
  1638.   if (change_to_newfile(share->index_file_name,MI_NAME_IEXT,INDEX_TMP_EXT,0,
  1639. MYF(0)) ||
  1640.       mi_open_keyfile(share))
  1641.     goto err2;
  1642.   info->lock_type= F_UNLCK; /* Force mi_readinfo to lock */
  1643.   _mi_readinfo(info,F_WRLCK,0); /* Will lock the table */
  1644.   info->lock_type=  old_lock;
  1645.   share->r_locks=   r_locks;
  1646.   share->w_locks=   w_locks;
  1647.   share->tot_locks= r_locks+w_locks;
  1648.   share->state=     old_state; /* Restore old state */
  1649.   info->state->key_file_length=param->new_file_pos;
  1650.   info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED);
  1651.   for (key=0 ; key < info->s->base.keys ; key++)
  1652.     info->s->state.key_root[key]=index_pos[key];
  1653.   for (key=0 ; key < info->s->state.header.max_block_size ; key++)
  1654.     info->s->state.key_del[key]=  HA_OFFSET_ERROR;
  1655.   info->s->state.changed&= ~STATE_NOT_SORTED_PAGES;
  1656.   DBUG_RETURN(0);
  1657. err:
  1658.   VOID(my_close(new_file,MYF(MY_WME)));
  1659. err2:
  1660.   VOID(my_delete(param->temp_filename,MYF(MY_WME)));
  1661.   DBUG_RETURN(-1);
  1662. } /* mi_sort_index */
  1663.  /* Sort records recursive using one index */
  1664. static int sort_one_index(MI_CHECK *param, MI_INFO *info, MI_KEYDEF *keyinfo,
  1665.   my_off_t pagepos, File new_file)
  1666. {
  1667.   uint length,nod_flag,used_length, key_length;
  1668.   uchar *buff,*keypos,*endpos;
  1669.   uchar key[MI_MAX_POSSIBLE_KEY_BUFF];
  1670.   my_off_t new_page_pos,next_page;
  1671.   char llbuff[22];
  1672.   DBUG_ENTER("sort_one_index");
  1673.   new_page_pos=param->new_file_pos;
  1674.   param->new_file_pos+=keyinfo->block_length;
  1675.   if (!(buff=(uchar*) my_alloca((uint) keyinfo->block_length)))
  1676.   {
  1677.     mi_check_print_error(param,"Not enough memory for key block");
  1678.     DBUG_RETURN(-1);
  1679.   }
  1680.   if (!_mi_fetch_keypage(info,keyinfo,pagepos,DFLT_INIT_HITS,buff,0))
  1681.   {
  1682.     mi_check_print_error(param,"Can't read key block from filepos: %s",
  1683. llstr(pagepos,llbuff));
  1684.     goto err;
  1685.   }
  1686.   if ((nod_flag=mi_test_if_nod(buff)) || keyinfo->flag & HA_FULLTEXT)
  1687.   {
  1688.     used_length=mi_getint(buff);
  1689.     keypos=buff+2+nod_flag;
  1690.     endpos=buff+used_length;
  1691.     for ( ;; )
  1692.     {
  1693.       if (nod_flag)
  1694.       {
  1695. next_page=_mi_kpos(nod_flag,keypos);
  1696. _mi_kpointer(info,keypos-nod_flag,param->new_file_pos); /* Save new pos */
  1697. if (sort_one_index(param,info,keyinfo,next_page,new_file))
  1698. {
  1699.   DBUG_PRINT("error",
  1700.      ("From page: %ld, keyoffset: %lu  used_length: %d",
  1701.       (ulong) pagepos, (ulong) (keypos - buff),
  1702.       (int) used_length));
  1703.   DBUG_DUMP("buff",(byte*) buff,used_length);
  1704.   goto err;
  1705. }
  1706.       }
  1707.       if (keypos >= endpos ||
  1708.   (key_length=(*keyinfo->get_key)(keyinfo,nod_flag,&keypos,key)) == 0)
  1709. break;
  1710.       DBUG_ASSERT(keypos <= endpos);
  1711.       if (keyinfo->flag & HA_FULLTEXT)
  1712.       {
  1713.         uint off;
  1714.         int  subkeys;
  1715.         get_key_full_length_rdonly(off, key);
  1716.         subkeys=ft_sintXkorr(key+off);
  1717.         if (subkeys < 0)
  1718.         {
  1719.           next_page= _mi_dpos(info,0,key+key_length);
  1720.           _mi_dpointer(info,keypos-nod_flag-info->s->rec_reflength,
  1721.                        param->new_file_pos); /* Save new pos */
  1722.           if (sort_one_index(param,info,&info->s->ft2_keyinfo,
  1723.                              next_page,new_file))
  1724.             goto err;
  1725.         }
  1726.       }
  1727.     }
  1728.   }
  1729.   /* Fill block with zero and write it to the new index file */
  1730.   length=mi_getint(buff);
  1731.   bzero((byte*) buff+length,keyinfo->block_length-length);
  1732.   if (my_pwrite(new_file,(byte*) buff,(uint) keyinfo->block_length,
  1733. new_page_pos,MYF(MY_NABP | MY_WAIT_IF_FULL)))
  1734.   {
  1735.     mi_check_print_error(param,"Can't write indexblock, error: %d",my_errno);
  1736.     goto err;
  1737.   }
  1738.   my_afree((gptr) buff);
  1739.   DBUG_RETURN(0);
  1740. err:
  1741.   my_afree((gptr) buff);
  1742.   DBUG_RETURN(1);
  1743. } /* sort_one_index */
  1744. /*
  1745.   Let temporary file replace old file.
  1746.   This assumes that the new file was created in the same
  1747.   directory as given by realpath(filename).
  1748.   This will ensure that any symlinks that are used will still work.
  1749.   Copy stats from old file to new file, deletes orignal and
  1750.   changes new file name to old file name
  1751. */
  1752. int change_to_newfile(const char * filename, const char * old_ext,
  1753.       const char * new_ext,
  1754.       uint raid_chunks __attribute__((unused)),
  1755.       myf MyFlags)
  1756. {
  1757.   char old_filename[FN_REFLEN],new_filename[FN_REFLEN];
  1758. #ifdef USE_RAID
  1759.   if (raid_chunks)
  1760.     return my_raid_redel(fn_format(old_filename,filename,"",old_ext,2+4),
  1761.  fn_format(new_filename,filename,"",new_ext,2+4),
  1762.  raid_chunks,
  1763.  MYF(MY_WME | MY_LINK_WARNING | MyFlags));
  1764. #endif
  1765.   /* Get real path to filename */
  1766.   (void) fn_format(old_filename,filename,"",old_ext,2+4+32);
  1767.   return my_redel(old_filename,
  1768.   fn_format(new_filename,old_filename,"",new_ext,2+4),
  1769.   MYF(MY_WME | MY_LINK_WARNING | MyFlags));
  1770. } /* change_to_newfile */
  1771. /* Locks a whole file */
  1772. /* Gives an error-message if file can't be locked */
  1773. int lock_file(MI_CHECK *param, File file, my_off_t start, int lock_type,
  1774.       const char *filetype, const char *filename)
  1775. {
  1776.   if (my_lock(file,lock_type,start,F_TO_EOF,
  1777.       param->testflag & T_WAIT_FOREVER ? MYF(MY_SEEK_NOT_DONE) :
  1778.       MYF(MY_SEEK_NOT_DONE |  MY_DONT_WAIT)))
  1779.   {
  1780.     mi_check_print_error(param," %d when locking %s '%s'",my_errno,filetype,filename);
  1781.     param->error_printed=2; /* Don't give that data is crashed */
  1782.     return 1;
  1783.   }
  1784.   return 0;
  1785. } /* lock_file */
  1786. /* Copy a block between two files */
  1787. int filecopy(MI_CHECK *param, File to,File from,my_off_t start,
  1788.      my_off_t length, const char *type)
  1789. {
  1790.   char tmp_buff[IO_SIZE],*buff;
  1791.   ulong buff_length;
  1792.   DBUG_ENTER("filecopy");
  1793.   buff_length=(ulong) min(param->write_buffer_length,length);
  1794.   if (!(buff=my_malloc(buff_length,MYF(0))))
  1795.   {
  1796.     buff=tmp_buff; buff_length=IO_SIZE;
  1797.   }
  1798.   VOID(my_seek(from,start,MY_SEEK_SET,MYF(0)));
  1799.   while (length > buff_length)
  1800.   {
  1801.     if (my_read(from,(byte*) buff,buff_length,MYF(MY_NABP)) ||
  1802. my_write(to,(byte*) buff,buff_length,param->myf_rw))
  1803.       goto err;
  1804.     length-= buff_length;
  1805.   }
  1806.   if (my_read(from,(byte*) buff,(uint) length,MYF(MY_NABP)) ||
  1807.       my_write(to,(byte*) buff,(uint) length,param->myf_rw))
  1808.     goto err;
  1809.   if (buff != tmp_buff)
  1810.     my_free(buff,MYF(0));
  1811.   DBUG_RETURN(0);
  1812. err:
  1813.   if (buff != tmp_buff)
  1814.     my_free(buff,MYF(0));
  1815.   mi_check_print_error(param,"Can't copy %s to tempfile, error %d",
  1816.        type,my_errno);
  1817.   DBUG_RETURN(1);
  1818. }
  1819. /*
  1820.   Repair table or given index using sorting
  1821.   SYNOPSIS
  1822.     mi_repair_by_sort()
  1823.     param Repair parameters
  1824.     info MyISAM handler to repair
  1825.     name Name of table (for warnings)
  1826.     rep_quick set to <> 0 if we should not change data file
  1827.   RESULT
  1828.     0 ok
  1829.     <>0 Error
  1830. */
  1831. int mi_repair_by_sort(MI_CHECK *param, register MI_INFO *info,
  1832.       const char * name, int rep_quick)
  1833. {
  1834.   int got_error;
  1835.   uint i;
  1836.   ulong length;
  1837.   ha_rows start_records;
  1838.   my_off_t new_header_length,del;
  1839.   File new_file;
  1840.   MI_SORT_PARAM sort_param;
  1841.   MYISAM_SHARE *share=info->s;
  1842.   HA_KEYSEG *keyseg;
  1843.   ulong   *rec_per_key_part;
  1844.   char llbuff[22];
  1845.   SORT_INFO sort_info;
  1846.   ulonglong key_map=share->state.key_map;
  1847.   DBUG_ENTER("mi_repair_by_sort");
  1848.   start_records=info->state->records;
  1849.   got_error=1;
  1850.   new_file= -1;
  1851.   new_header_length=(param->testflag & T_UNPACK) ? 0 :
  1852.     share->pack.header_length;
  1853.   if (!(param->testflag & T_SILENT))
  1854.   {
  1855.     printf("- recovering (with sort) MyISAM-table '%s'n",name);
  1856.     printf("Data records: %sn", llstr(start_records,llbuff));
  1857.   }
  1858.   param->testflag|=T_REP; /* for easy checking */
  1859.   if (info->s->options & (HA_OPTION_CHECKSUM | HA_OPTION_COMPRESS_RECORD))
  1860.     param->testflag|=T_CALC_CHECKSUM;
  1861.   bzero((char*)&sort_info,sizeof(sort_info));
  1862.   bzero((char *)&sort_param, sizeof(sort_param));
  1863.   if (!(sort_info.key_block=
  1864. alloc_key_blocks(param,
  1865.  (uint) param->sort_key_blocks,
  1866.  share->base.max_key_block_length))
  1867.       || init_io_cache(&param->read_cache,info->dfile,
  1868.        (uint) param->read_buffer_length,
  1869.        READ_CACHE,share->pack.header_length,1,MYF(MY_WME)) ||
  1870.       (! rep_quick &&
  1871.        init_io_cache(&info->rec_cache,info->dfile,
  1872.      (uint) param->write_buffer_length,
  1873.      WRITE_CACHE,new_header_length,1,
  1874.      MYF(MY_WME | MY_WAIT_IF_FULL) & param->myf_rw)))
  1875.     goto err;
  1876.   sort_info.key_block_end=sort_info.key_block+param->sort_key_blocks;
  1877.   info->opt_flag|=WRITE_CACHE_USED;
  1878.   info->rec_cache.file=info->dfile; /* for sort_delete_record */
  1879.   if (!(sort_param.record=(byte*) my_malloc((uint) share->base.pack_reclength,
  1880.    MYF(0))) ||
  1881.       !mi_alloc_rec_buff(info, -1, &sort_param.rec_buff))
  1882.   {
  1883.     mi_check_print_error(param, "Not enough memory for extra record");
  1884.     goto err;
  1885.   }
  1886.   if (!rep_quick)
  1887.   {
  1888.     /* Get real path for data file */
  1889.     if ((new_file=my_raid_create(fn_format(param->temp_filename,
  1890.    share->data_file_name, "",
  1891.    DATA_TMP_EXT, 2+4),
  1892.  0,param->tmpfile_createflag,
  1893.  share->base.raid_type,
  1894.  share->base.raid_chunks,
  1895.  share->base.raid_chunksize,
  1896.  MYF(0))) < 0)
  1897.     {
  1898.       mi_check_print_error(param,"Can't create new tempfile: '%s'",
  1899.    param->temp_filename);
  1900.       goto err;
  1901.     }
  1902.     if (filecopy(param, new_file,info->dfile,0L,new_header_length,
  1903.  "datafile-header"))
  1904.       goto err;
  1905.     if (param->testflag & T_UNPACK)
  1906.     {
  1907.       share->options&= ~HA_OPTION_COMPRESS_RECORD;
  1908.       mi_int2store(share->state.header.options,share->options);
  1909.     }
  1910.     share->state.dellink= HA_OFFSET_ERROR;
  1911.     info->rec_cache.file=new_file;
  1912.   }
  1913.   info->update= (short) (HA_STATE_CHANGED | HA_STATE_ROW_CHANGED);
  1914.   if (!(param->testflag & T_CREATE_MISSING_KEYS))
  1915.   {
  1916.     /*
  1917.       Flush key cache for this file if we are calling this outside
  1918.       myisamchk
  1919.     */
  1920.     flush_key_blocks(share->key_cache,share->kfile, FLUSH_IGNORE_CHANGED);
  1921.     /* Clear the pointers to the given rows */
  1922.     for (i=0 ; i < share->base.keys ; i++)
  1923.       share->state.key_root[i]= HA_OFFSET_ERROR;
  1924.     for (i=0 ; i < share->state.header.max_block_size ; i++)
  1925.       share->state.key_del[i]=  HA_OFFSET_ERROR;
  1926.     info->state->key_file_length=share->base.keystart;
  1927.   }
  1928.   else
  1929.   {
  1930.     if (flush_key_blocks(share->key_cache,share->kfile, FLUSH_FORCE_WRITE))
  1931.       goto err;
  1932.     key_map= ~key_map; /* Create the missing keys */
  1933.   }
  1934.   sort_info.info=info;
  1935.   sort_info.param = param;
  1936.   set_data_file_type(&sort_info, share);
  1937.   sort_param.filepos=new_header_length;
  1938.   sort_info.dupp=0;
  1939.   sort_info.buff=0;
  1940.   param->read_cache.end_of_file=sort_info.filelength=
  1941.     my_seek(param->read_cache.file,0L,MY_SEEK_END,MYF(0));
  1942.   sort_param.wordlist=NULL;
  1943.   if (share->data_file_type == DYNAMIC_RECORD)
  1944.     length=max(share->base.min_pack_length+1,share->base.min_block_length);
  1945.   else if (share->data_file_type == COMPRESSED_RECORD)
  1946.     length=share->base.min_block_length;
  1947.   else
  1948.     length=share->base.pack_reclength;
  1949.   sort_info.max_records=
  1950.     ((param->testflag & T_CREATE_MISSING_KEYS) ? info->state->records :
  1951.      (ha_rows) (sort_info.filelength/length+1));
  1952.   sort_param.key_cmp=sort_key_cmp;
  1953.   sort_param.lock_in_memory=lock_memory;
  1954.   sort_param.tmpdir=param->tmpdir;
  1955.   sort_param.sort_info=&sort_info;
  1956.   sort_param.fix_datafile= (my_bool) (! rep_quick);
  1957.   sort_param.master =1;
  1958.   
  1959.   del=info->state->del;
  1960.   param->glob_crc=0;
  1961.   if (param->testflag & T_CALC_CHECKSUM)
  1962.     param->calc_checksum=1;
  1963.   rec_per_key_part= param->rec_per_key_part;
  1964.   for (sort_param.key=0 ; sort_param.key < share->base.keys ;
  1965.        rec_per_key_part+=sort_param.keyinfo->keysegs, sort_param.key++)
  1966.   {
  1967.     sort_param.read_cache=param->read_cache;
  1968.     sort_param.keyinfo=share->keyinfo+sort_param.key;
  1969.     sort_param.seg=sort_param.keyinfo->seg;
  1970.     if (!(((ulonglong) 1 << sort_param.key) & key_map))
  1971.     {
  1972.       /* Remember old statistics for key */
  1973.       memcpy((char*) rec_per_key_part,
  1974.      (char*) (share->state.rec_per_key_part +
  1975.       (uint) (rec_per_key_part - param->rec_per_key_part)),
  1976.      sort_param.keyinfo->keysegs*sizeof(*rec_per_key_part));
  1977.       continue;
  1978.     }