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

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. /*
  14.   Functions to handle space-packed-records and blobs
  15.  
  16.   A row may be stored in one or more linked blocks.
  17.   The block size is between MI_MIN_BLOCK_LENGTH and MI_MAX_BLOCK_LENGTH.
  18.   Each block is aligned on MI_DYN_ALIGN_SIZE.
  19.   The reson for the max block size is to not have too many different types
  20.   of blocks.  For the differnet block types, look at _mi_get_block_info()
  21. */
  22. #include "myisamdef.h"
  23. /* Enough for comparing if number is zero */
  24. static char zero_string[]={0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
  25. static int write_dynamic_record(MI_INFO *info,const byte *record,
  26. ulong reclength);
  27. static int _mi_find_writepos(MI_INFO *info,ulong reclength,my_off_t *filepos,
  28.      ulong *length);
  29. static int update_dynamic_record(MI_INFO *info,my_off_t filepos,byte *record,
  30.  ulong reclength);
  31. static int delete_dynamic_record(MI_INFO *info,my_off_t filepos,
  32.  uint second_read);
  33. static int _mi_cmp_buffer(File file, const byte *buff, my_off_t filepos,
  34.   uint length);
  35. #ifdef THREAD
  36. /* Play it safe; We have a small stack when using threads */
  37. #undef my_alloca
  38. #undef my_afree
  39. #define my_alloca(A) my_malloc((A),MYF(0))
  40. #define my_afree(A) my_free((A),MYF(0))
  41. #endif
  42. /* Interface function from MI_INFO */
  43. int _mi_write_dynamic_record(MI_INFO *info, const byte *record)
  44. {
  45.   ulong reclength=_mi_rec_pack(info,info->rec_buff,record);
  46.   return (write_dynamic_record(info,info->rec_buff,reclength));
  47. }
  48. int _mi_update_dynamic_record(MI_INFO *info, my_off_t pos, const byte *record)
  49. {
  50.   uint length=_mi_rec_pack(info,info->rec_buff,record);
  51.   return (update_dynamic_record(info,pos,info->rec_buff,length));
  52. }
  53. int _mi_write_blob_record(MI_INFO *info, const byte *record)
  54. {
  55.   byte *rec_buff;
  56.   int error;
  57.   ulong reclength,reclength2,extra;
  58.   extra= (ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+
  59.   MI_DYN_DELETE_BLOCK_HEADER+1);
  60.   reclength= (info->s->base.pack_reclength +
  61.       _my_calc_total_blob_length(info,record)+ extra);
  62. #ifdef NOT_USED /* We now support big rows */
  63.   if (reclength > MI_DYN_MAX_ROW_LENGTH)
  64.   {
  65.     my_errno=HA_ERR_TO_BIG_ROW;
  66.     return -1;
  67.   }
  68. #endif
  69.   if (!(rec_buff=(byte*) my_alloca(reclength)))
  70.   {
  71.     my_errno=ENOMEM;
  72.     return(-1);
  73.   }
  74.   reclength2= _mi_rec_pack(info,rec_buff+ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER),
  75.    record);
  76.   DBUG_PRINT("info",("reclength: %lu  reclength2: %lu",
  77.      reclength, reclength2));
  78.   DBUG_ASSERT(reclength2 <= reclength);
  79.   error=write_dynamic_record(info,rec_buff+ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER),
  80.      reclength2);
  81.   my_afree(rec_buff);
  82.   return(error);
  83. }
  84. int _mi_update_blob_record(MI_INFO *info, my_off_t pos, const byte *record)
  85. {
  86.   byte *rec_buff;
  87.   int error;
  88.   ulong reclength,extra;
  89.   extra= (ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+
  90.   MI_DYN_DELETE_BLOCK_HEADER);
  91.   reclength= (info->s->base.pack_reclength+
  92.       _my_calc_total_blob_length(info,record)+ extra);
  93. #ifdef NOT_USED /* We now support big rows */
  94.   if (reclength > MI_DYN_MAX_ROW_LENGTH)
  95.   {
  96.     my_errno=HA_ERR_TO_BIG_ROW;
  97.     return -1;
  98.   }
  99. #endif
  100.   if (!(rec_buff=(byte*) my_alloca(reclength)))
  101.   {
  102.     my_errno=ENOMEM;
  103.     return(-1);
  104.   }
  105.   reclength=_mi_rec_pack(info,rec_buff+ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER),
  106.  record);
  107.   error=update_dynamic_record(info,pos,
  108.       rec_buff+ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER),
  109.       reclength);
  110.   my_afree(rec_buff);
  111.   return(error);
  112. }
  113. int _mi_delete_dynamic_record(MI_INFO *info)
  114. {
  115.   return delete_dynamic_record(info,info->lastpos,0);
  116. }
  117. /* Write record to data-file */
  118. static int write_dynamic_record(MI_INFO *info, const byte *record,
  119. ulong reclength)
  120. {
  121.   int flag;
  122.   ulong length;
  123.   my_off_t filepos;
  124.   DBUG_ENTER("write_dynamic_record");
  125.   flag=0;
  126.   do
  127.   {
  128.     if (_mi_find_writepos(info,reclength,&filepos,&length))
  129.       goto err;
  130.     if (_mi_write_part_record(info,filepos,length,info->s->state.dellink,
  131.       (byte**) &record,&reclength,&flag))
  132.       goto err;
  133.   } while (reclength);
  134.   DBUG_RETURN(0);
  135. err:
  136.   DBUG_RETURN(1);
  137. }
  138. /* Get a block for data ; The given data-area must be used !! */
  139. static int _mi_find_writepos(MI_INFO *info,
  140.      ulong reclength, /* record length */
  141.      my_off_t *filepos, /* Return file pos */
  142.      ulong *length)   /* length of block at filepos */
  143. {
  144.   MI_BLOCK_INFO block_info;
  145.   ulong tmp;
  146.   DBUG_ENTER("_mi_find_writepos");
  147.   if (info->s->state.dellink != HA_OFFSET_ERROR)
  148.   {
  149.     /* Deleted blocks exists;  Get last used block */
  150.     *filepos=info->s->state.dellink;
  151.     block_info.second_read=0;
  152.     info->rec_cache.seek_not_done=1;
  153.     if (!(_mi_get_block_info(&block_info,info->dfile,info->s->state.dellink) &
  154.    BLOCK_DELETED))
  155.     {
  156.       DBUG_PRINT("error",("Delete link crashed"));
  157.       my_errno=HA_ERR_WRONG_IN_RECORD;
  158.       DBUG_RETURN(-1);
  159.     }
  160.     info->s->state.dellink=block_info.next_filepos;
  161.     info->state->del--;
  162.     info->state->empty-= block_info.block_len;
  163.     *length= block_info.block_len;
  164.   }
  165.   else
  166.   {
  167.     /* No deleted blocks;  Allocate a new block */
  168.     *filepos=info->state->data_file_length;
  169.     if ((tmp=reclength+3 + test(reclength >= (65520-3))) <
  170. info->s->base.min_block_length)
  171.       tmp= info->s->base.min_block_length;
  172.     else
  173.       tmp= ((tmp+MI_DYN_ALIGN_SIZE-1) &
  174.     (~ (ulong) (MI_DYN_ALIGN_SIZE-1)));
  175.     if (info->state->data_file_length >
  176. (info->s->base.max_data_file_length - tmp))
  177.     {
  178.       my_errno=HA_ERR_RECORD_FILE_FULL;
  179.       DBUG_RETURN(-1);
  180.     }
  181.     if (tmp > MI_MAX_BLOCK_LENGTH)
  182.       tmp=MI_MAX_BLOCK_LENGTH;
  183.     *length= tmp;
  184.     info->state->data_file_length+= tmp;
  185.     info->s->state.split++;
  186.     info->update|=HA_STATE_WRITE_AT_END;
  187.   }
  188.   DBUG_RETURN(0);
  189. } /* _mi_find_writepos */
  190. /*
  191.   Unlink a deleted block from the deleted list.
  192.   This block will be combined with the preceding or next block to form
  193.   a big block.
  194. */
  195. static bool unlink_deleted_block(MI_INFO *info, MI_BLOCK_INFO *block_info)
  196. {
  197.   DBUG_ENTER("unlink_deleted_block");
  198.   if (block_info->filepos == info->s->state.dellink)
  199.   {
  200.     /* First deleted block;  We can just use this ! */
  201.     info->s->state.dellink=block_info->next_filepos;
  202.   }
  203.   else
  204.   {
  205.     MI_BLOCK_INFO tmp;
  206.     tmp.second_read=0;
  207.     /* Unlink block from the previous block */
  208.     if (!(_mi_get_block_info(&tmp,info->dfile,block_info->prev_filepos)
  209.   & BLOCK_DELETED))
  210.       DBUG_RETURN(1); /* Something is wrong */
  211.     mi_sizestore(tmp.header+4,block_info->next_filepos);
  212.     if (my_pwrite(info->dfile,(char*) tmp.header+4,8,
  213.   block_info->prev_filepos+4, MYF(MY_NABP)))
  214.       DBUG_RETURN(1);
  215.     /* Unlink block from next block */
  216.     if (block_info->next_filepos != HA_OFFSET_ERROR)
  217.     {
  218.       if (!(_mi_get_block_info(&tmp,info->dfile,block_info->next_filepos)
  219.     & BLOCK_DELETED))
  220. DBUG_RETURN(1); /* Something is wrong */
  221.       mi_sizestore(tmp.header+12,block_info->prev_filepos);
  222.       if (my_pwrite(info->dfile,(char*) tmp.header+12,8,
  223.     block_info->next_filepos+12,
  224.     MYF(MY_NABP)))
  225. DBUG_RETURN(1);
  226.     }
  227.   }
  228.   /* We now have one less deleted block */
  229.   info->state->del--;
  230.   info->state->empty-= block_info->block_len;
  231.   info->s->state.split--;
  232.   /*
  233.     If this was a block that we where accessing through table scan
  234.     (mi_rrnd() or mi_scan(), then ensure that we skip over this block
  235.     when doing next mi_rrnd() or mi_scan().
  236.   */
  237.   if (info->nextpos == block_info->filepos)
  238.     info->nextpos+=block_info->block_len;
  239.   DBUG_RETURN(0);
  240. }
  241. /*
  242.   Add a backward link to delete block
  243.   SYNOPSIS
  244.     update_backward_delete_link()
  245.     info MyISAM handler
  246.     delete_block Position to delete block to update.
  247. If this is 'HA_OFFSET_ERROR', nothing will be done
  248.     filepos Position to block that 'delete_block' should point to
  249.   RETURN
  250.     0  ok
  251.     1  error.  In this case my_error is set.
  252. */
  253. static int update_backward_delete_link(MI_INFO *info, my_off_t delete_block,
  254.        my_off_t filepos)
  255. {
  256.   MI_BLOCK_INFO block_info;
  257.   DBUG_ENTER("update_backward_delete_link");
  258.   if (delete_block != HA_OFFSET_ERROR)
  259.   {
  260.     block_info.second_read=0;
  261.     if (_mi_get_block_info(&block_info,info->dfile,delete_block)
  262. & BLOCK_DELETED)
  263.     {
  264.       char buff[8];
  265.       mi_sizestore(buff,filepos);
  266.       if (my_pwrite(info->dfile,buff, 8, delete_block+12, MYF(MY_NABP)))
  267. DBUG_RETURN(1); /* Error on write */
  268.     }
  269.     else
  270.     {
  271.       my_errno=HA_ERR_WRONG_IN_RECORD;
  272.       DBUG_RETURN(1); /* Wrong delete link */
  273.     }
  274.   }
  275.   DBUG_RETURN(0);
  276. }
  277. /* Delete datarecord from database */
  278. /* info->rec_cache.seek_not_done is updated in cmp_record */
  279. static int delete_dynamic_record(MI_INFO *info, my_off_t filepos,
  280.  uint second_read)
  281. {
  282.   uint length,b_type;
  283.   MI_BLOCK_INFO block_info,del_block;
  284.   int error;
  285.   my_bool remove_next_block;
  286.   DBUG_ENTER("delete_dynamic_record");
  287.   /* First add a link from the last block to the new one */
  288.   error= update_backward_delete_link(info, info->s->state.dellink, filepos);
  289.   block_info.second_read=second_read;
  290.   do
  291.   {
  292.     /* Remove block at 'filepos' */
  293.     if ((b_type=_mi_get_block_info(&block_info,info->dfile,filepos))
  294. & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  295.    BLOCK_FATAL_ERROR) ||
  296. (length=(uint) (block_info.filepos-filepos) +block_info.block_len) <
  297. MI_MIN_BLOCK_LENGTH)
  298.     {
  299.       my_errno=HA_ERR_WRONG_IN_RECORD;
  300.       DBUG_RETURN(1);
  301.     }
  302.     /* Check if next block is a delete block */
  303.     del_block.second_read=0;
  304.     remove_next_block=0;
  305.     if (_mi_get_block_info(&del_block,info->dfile,filepos+length) &
  306. BLOCK_DELETED && del_block.block_len+length < MI_DYN_MAX_BLOCK_LENGTH)
  307.     {
  308.       /* We can't remove this yet as this block may be the head block */
  309.       remove_next_block=1;
  310.       length+=del_block.block_len;
  311.     }
  312.     block_info.header[0]=0;
  313.     mi_int3store(block_info.header+1,length);
  314.     mi_sizestore(block_info.header+4,info->s->state.dellink);
  315.     if (b_type & BLOCK_LAST)
  316.       bfill(block_info.header+12,8,255);
  317.     else
  318.       mi_sizestore(block_info.header+12,block_info.next_filepos);
  319.     if (my_pwrite(info->dfile,(byte*) block_info.header,20,filepos,
  320.   MYF(MY_NABP)))
  321.       DBUG_RETURN(1);
  322.     info->s->state.dellink = filepos;
  323.     info->state->del++;
  324.     info->state->empty+=length;
  325.     filepos=block_info.next_filepos;
  326.     /* Now it's safe to unlink the deleted block directly after this one */
  327.     if (remove_next_block && unlink_deleted_block(info,&del_block))
  328.       error=1;
  329.   } while (!(b_type & BLOCK_LAST));
  330.   DBUG_RETURN(error);
  331. }
  332. /* Write a block to datafile */
  333. int _mi_write_part_record(MI_INFO *info,
  334.   my_off_t filepos, /* points at empty block */
  335.   ulong length, /* length of block */
  336.   my_off_t next_filepos,/* Next empty block */
  337.   byte **record, /* pointer to record ptr */
  338.   ulong *reclength, /* length of *record */
  339.   int *flag) /* *flag == 0 if header */
  340. {
  341.   ulong head_length,res_length,extra_length,long_block,del_length;
  342.   byte *pos,*record_end;
  343.   my_off_t  next_delete_block;
  344.   uchar temp[MI_SPLIT_LENGTH+MI_DYN_DELETE_BLOCK_HEADER];
  345.   DBUG_ENTER("_mi_write_part_record");
  346.   next_delete_block=HA_OFFSET_ERROR;
  347.   res_length=extra_length=0;
  348.   if (length > *reclength + MI_SPLIT_LENGTH)
  349.   { /* Splitt big block */
  350.     res_length=MY_ALIGN(length- *reclength - MI_EXTEND_BLOCK_LENGTH,
  351. MI_DYN_ALIGN_SIZE);
  352.     length-= res_length; /* Use this for first part */
  353.   }
  354.   long_block= (length < 65520L && *reclength < 65520L) ? 0 : 1;
  355.   if (length == *reclength+ 3 + long_block)
  356.   {
  357.     /* Block is exactly of the right length */
  358.     temp[0]=(uchar) (1+ *flag)+(uchar) long_block; /* Flag is 0 or 6 */
  359.     if (long_block)
  360.     {
  361.       mi_int3store(temp+1,*reclength);
  362.       head_length=4;
  363.     }
  364.     else
  365.     {
  366.       mi_int2store(temp+1,*reclength);
  367.       head_length=3;
  368.     }
  369.   }
  370.   else if (length-long_block < *reclength+4)
  371.   { /* To short block */
  372.     if (next_filepos == HA_OFFSET_ERROR)
  373.       next_filepos=info->s->state.dellink != HA_OFFSET_ERROR ?
  374. info->s->state.dellink : info->state->data_file_length;
  375.     if (*flag == 0) /* First block */
  376.     {
  377.       if (*reclength > MI_MAX_BLOCK_LENGTH)
  378.       {
  379. head_length= 16;
  380. temp[0]=13;
  381. mi_int4store(temp+1,*reclength);
  382. mi_int3store(temp+5,length-head_length);
  383. mi_sizestore((byte*) temp+8,next_filepos);
  384.       }
  385.       else
  386.       {
  387. head_length=5+8+long_block*2;
  388. temp[0]=5+(uchar) long_block;
  389. if (long_block)
  390. {
  391.   mi_int3store(temp+1,*reclength);
  392.   mi_int3store(temp+4,length-head_length);
  393.   mi_sizestore((byte*) temp+7,next_filepos);
  394. }
  395. else
  396. {
  397.   mi_int2store(temp+1,*reclength);
  398.   mi_int2store(temp+3,length-head_length);
  399.   mi_sizestore((byte*) temp+5,next_filepos);
  400. }
  401.       }
  402.     }
  403.     else
  404.     {
  405.       head_length=3+8+long_block;
  406.       temp[0]=11+(uchar) long_block;
  407.       if (long_block)
  408.       {
  409. mi_int3store(temp+1,length-head_length);
  410. mi_sizestore((byte*) temp+4,next_filepos);
  411.       }
  412.       else
  413.       {
  414. mi_int2store(temp+1,length-head_length);
  415. mi_sizestore((byte*) temp+3,next_filepos);
  416.       }
  417.     }
  418.   }
  419.   else
  420.   { /* Block with empty info last */
  421.     head_length=4+long_block;
  422.     extra_length= length- *reclength-head_length;
  423.     temp[0]= (uchar) (3+ *flag)+(uchar) long_block; /* 3,4 or 9,10 */
  424.     if (long_block)
  425.     {
  426.       mi_int3store(temp+1,*reclength);
  427.       temp[4]= (uchar) (extra_length);
  428.     }
  429.     else
  430.     {
  431.       mi_int2store(temp+1,*reclength);
  432.       temp[3]= (uchar) (extra_length);
  433.     }
  434.     length=   *reclength+head_length; /* Write only what is needed */
  435.   }
  436.   DBUG_DUMP("header",(byte*) temp,head_length);
  437. /* Make a long block for one write */
  438.   record_end= *record+length-head_length;
  439.   del_length=(res_length ? MI_DYN_DELETE_BLOCK_HEADER : 0);
  440.   bmove((byte*) (*record-head_length),(byte*) temp,head_length);
  441.   memcpy(temp,record_end,(size_t) (extra_length+del_length));
  442.   bzero((byte*) record_end,extra_length);
  443.   if (res_length)
  444.   {
  445.     /* Check first if we can join this block with the next one */
  446.     MI_BLOCK_INFO del_block;
  447.     my_off_t next_block=filepos+length+extra_length+res_length;
  448.     del_block.second_read=0;
  449.     if (next_block < info->state->data_file_length &&
  450. info->s->state.dellink != HA_OFFSET_ERROR)
  451.     {
  452.       if ((_mi_get_block_info(&del_block,info->dfile,next_block)
  453.    & BLOCK_DELETED) &&
  454.   res_length + del_block.block_len < MI_DYN_MAX_BLOCK_LENGTH)
  455.       {
  456. if (unlink_deleted_block(info,&del_block))
  457.   goto err;
  458. res_length+=del_block.block_len;
  459.       }
  460.     }
  461.     /* Create a delete link of the last part of the block */
  462.     pos=record_end+extra_length;
  463.     pos[0]= '';
  464.     mi_int3store(pos+1,res_length);
  465.     mi_sizestore(pos+4,info->s->state.dellink);
  466.     bfill(pos+12,8,255); /* End link */
  467.     next_delete_block=info->s->state.dellink;
  468.     info->s->state.dellink= filepos+length+extra_length;
  469.     info->state->del++;
  470.     info->state->empty+=res_length;
  471.     info->s->state.split++;
  472.   }
  473.   if (info->opt_flag & WRITE_CACHE_USED &&
  474.       info->update & HA_STATE_WRITE_AT_END)
  475.   {
  476.     if (info->update & HA_STATE_EXTEND_BLOCK)
  477.     {
  478.       info->update&= ~HA_STATE_EXTEND_BLOCK;
  479.       if (my_block_write(&info->rec_cache,(byte*) *record-head_length,
  480.  length+extra_length+del_length,filepos))
  481.       goto err;
  482.     }
  483.     else if (my_b_write(&info->rec_cache,(byte*) *record-head_length,
  484. length+extra_length+del_length))
  485.       goto err;
  486.   }
  487.   else
  488.   {
  489.     info->rec_cache.seek_not_done=1;
  490.     if (my_pwrite(info->dfile,(byte*) *record-head_length,length+extra_length+
  491.   del_length,filepos,info->s->write_flag))
  492.       goto err;
  493.   }
  494.   memcpy(record_end,temp,(size_t) (extra_length+del_length));
  495.   *record=record_end;
  496.   *reclength-=(length-head_length);
  497.   *flag=6;
  498.   if (del_length)
  499.   {
  500.     /* link the next delete block to this */
  501.     if (update_backward_delete_link(info, next_delete_block,
  502.     info->s->state.dellink))
  503.       goto err;
  504.   }
  505.   DBUG_RETURN(0);
  506. err:
  507.   DBUG_PRINT("exit",("errno: %d",my_errno));
  508.   DBUG_RETURN(1);
  509. } /*_mi_write_part_record */
  510. /* update record from datafile */
  511. static int update_dynamic_record(MI_INFO *info, my_off_t filepos, byte *record,
  512.  ulong reclength)
  513. {
  514.   int flag;
  515.   uint error;
  516.   ulong length;
  517.   MI_BLOCK_INFO block_info;
  518.   DBUG_ENTER("update_dynamic_record");
  519.   flag=block_info.second_read=0;
  520.   while (reclength > 0)
  521.   {
  522.     if (filepos != info->s->state.dellink)
  523.     {
  524.       block_info.next_filepos= HA_OFFSET_ERROR;
  525.       if ((error=_mi_get_block_info(&block_info,info->dfile,filepos))
  526.   & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  527.      BLOCK_FATAL_ERROR))
  528.       {
  529. DBUG_PRINT("error",("Got wrong block info"));
  530. if (!(error & BLOCK_FATAL_ERROR))
  531.   my_errno=HA_ERR_WRONG_IN_RECORD;
  532. goto err;
  533.       }
  534.       length=(ulong) (block_info.filepos-filepos) + block_info.block_len;
  535.       if (length < reclength)
  536.       {
  537. uint tmp=MY_ALIGN(reclength - length + 3 +
  538.   test(reclength >= 65520L),MI_DYN_ALIGN_SIZE);
  539. /* Don't create a block bigger than MI_MAX_BLOCK_LENGTH */
  540. tmp= min(length+tmp, MI_MAX_BLOCK_LENGTH)-length;
  541. /* Check if we can extend this block */
  542. if (block_info.filepos + block_info.block_len ==
  543.     info->state->data_file_length &&
  544.     info->state->data_file_length <
  545.     info->s->base.max_data_file_length-tmp)
  546. {
  547.   /* extend file */
  548.   DBUG_PRINT("info",("Extending file with %d bytes",tmp));
  549.   if (info->nextpos == info->state->data_file_length)
  550.     info->nextpos+= tmp;
  551.   info->state->data_file_length+= tmp;
  552.   info->update|= HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK;
  553.   length+=tmp;
  554. }
  555. else if (length < MI_MAX_BLOCK_LENGTH - MI_MIN_BLOCK_LENGTH)
  556. {
  557.   /*
  558.     Check if next block is a deleted block
  559.     Above we have MI_MIN_BLOCK_LENGTH to avoid the problem where
  560.     the next block is so small it can't be splited which could
  561.     casue problems
  562.   */
  563.   MI_BLOCK_INFO del_block;
  564.   del_block.second_read=0;
  565.   if (_mi_get_block_info(&del_block,info->dfile,
  566.  block_info.filepos + block_info.block_len) &
  567.       BLOCK_DELETED)
  568.   {
  569.     /* Use; Unlink it and extend the current block */
  570.     DBUG_PRINT("info",("Extending current block"));
  571.     if (unlink_deleted_block(info,&del_block))
  572.       goto err;
  573.     if ((length+=del_block.block_len) > MI_MAX_BLOCK_LENGTH)
  574.     {
  575.       /*
  576. New block was too big, link overflow part back to
  577. delete list
  578.       */
  579.       my_off_t next_pos;
  580.       ulong rest_length= length-MI_MAX_BLOCK_LENGTH;
  581.       set_if_bigger(rest_length, MI_MIN_BLOCK_LENGTH);
  582.       next_pos= del_block.filepos+ del_block.block_len - rest_length;
  583.       if (update_backward_delete_link(info, info->s->state.dellink,
  584.       next_pos))
  585. DBUG_RETURN(1);
  586.       /* create delete link for data that didn't fit into the page */
  587.       del_block.header[0]=0;
  588.       mi_int3store(del_block.header+1, rest_length);
  589.       mi_sizestore(del_block.header+4,info->s->state.dellink);
  590.       bfill(del_block.header+12,8,255);
  591.       if (my_pwrite(info->dfile,(byte*) del_block.header,20, next_pos,
  592.     MYF(MY_NABP)))
  593. DBUG_RETURN(1);
  594.       info->s->state.dellink= next_pos;
  595.       info->s->state.split++;
  596.       info->state->del++;
  597.       info->state->empty+= rest_length;
  598.       length-= rest_length;
  599.     }
  600.   }
  601. }
  602.       }
  603.     }
  604.     else
  605.     {
  606.       if (_mi_find_writepos(info,reclength,&filepos,&length))
  607. goto err;
  608.     }
  609.     if (_mi_write_part_record(info,filepos,length,block_info.next_filepos,
  610.       &record,&reclength,&flag))
  611.       goto err;
  612.     if ((filepos=block_info.next_filepos) == HA_OFFSET_ERROR)
  613.     {
  614.       /* Start writing data on deleted blocks */
  615.       filepos=info->s->state.dellink;
  616.     }
  617.   }
  618.   if (block_info.next_filepos != HA_OFFSET_ERROR)
  619.     if (delete_dynamic_record(info,block_info.next_filepos,1))
  620.       goto err;
  621.   DBUG_RETURN(0);
  622. err:
  623.   DBUG_RETURN(1);
  624. }
  625. /* Pack a record. Return new reclength */
  626. uint _mi_rec_pack(MI_INFO *info, register byte *to, register const byte *from)
  627. {
  628.   uint length,new_length,flag,bit,i;
  629.   char *pos,*end,*startpos,*packpos;
  630.   enum en_fieldtype type;
  631.   reg3 MI_COLUMNDEF *rec;
  632.   MI_BLOB *blob;
  633.   DBUG_ENTER("_mi_rec_pack");
  634.   flag=0 ; bit=1;
  635.   startpos=packpos=to; to+= info->s->base.pack_bits; blob=info->blobs;
  636.   rec=info->s->rec;
  637.   for (i=info->s->base.fields ; i-- > 0; from+= length,rec++)
  638.   {
  639.     length=(uint) rec->length;
  640.     if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL)
  641.     {
  642.       if (type == FIELD_BLOB)
  643.       {
  644. if (!blob->length)
  645.   flag|=bit;
  646. else
  647. {
  648.   char *temp_pos;
  649.   size_t tmp_length=length-mi_portable_sizeof_char_ptr;
  650.   memcpy((byte*) to,from,tmp_length);
  651.   memcpy_fixed(&temp_pos,from+tmp_length,sizeof(char*));
  652.   memcpy(to+tmp_length,temp_pos,(size_t) blob->length);
  653.   to+=tmp_length+blob->length;
  654. }
  655. blob++;
  656.       }
  657.       else if (type == FIELD_SKIP_ZERO)
  658.       {
  659. if (memcmp((byte*) from,zero_string,length) == 0)
  660.   flag|=bit;
  661. else
  662. {
  663.   memcpy((byte*) to,from,(size_t) length); to+=length;
  664. }
  665.       }
  666.       else if (type == FIELD_SKIP_ENDSPACE ||
  667.        type == FIELD_SKIP_PRESPACE)
  668.       {
  669. pos= (byte*) from; end= (byte*) from + length;
  670. if (type == FIELD_SKIP_ENDSPACE)
  671. { /* Pack trailing spaces */
  672.   while (end > from && *(end-1) == ' ')
  673.     end--;
  674. }
  675. else
  676. { /* Pack pref-spaces */
  677.   while (pos < end && *pos == ' ')
  678.     pos++;
  679. }
  680. new_length=(uint) (end-pos);
  681. if (new_length +1 + test(rec->length > 255 && new_length > 127)
  682.     < length)
  683. {
  684.   if (rec->length > 255 && new_length > 127)
  685.   {
  686.     to[0]=(char) ((new_length & 127)+128);
  687.     to[1]=(char) (new_length >> 7);
  688.     to+=2;
  689.   }
  690.   else
  691.     *to++= (char) new_length;
  692.   memcpy((byte*) to,pos,(size_t) new_length); to+=new_length;
  693.   flag|=bit;
  694. }
  695. else
  696. {
  697.   memcpy(to,from,(size_t) length); to+=length;
  698. }
  699.       }
  700.       else if (type == FIELD_VARCHAR)
  701.       {
  702. uint tmp_length=uint2korr(from);
  703. store_key_length_inc(to,tmp_length);
  704. memcpy(to,from+2,tmp_length);
  705. to+=tmp_length;
  706. continue;
  707.       }
  708.       else
  709.       {
  710. memcpy(to,from,(size_t) length); to+=length;
  711. continue; /* Normal field */
  712.       }
  713.       if ((bit= bit << 1) >= 256)
  714.       {
  715. *packpos++ = (char) (uchar) flag;
  716. bit=1; flag=0;
  717.       }
  718.     }
  719.     else
  720.     {
  721.       memcpy(to,from,(size_t) length); to+=length;
  722.     }
  723.   }
  724.   if (bit != 1)
  725.     *packpos= (char) (uchar) flag;
  726.   if (info->s->calc_checksum)
  727.     *to++=(char) info->checksum;
  728.   DBUG_PRINT("exit",("packed length: %d",(int) (to-startpos)));
  729.   DBUG_RETURN((uint) (to-startpos));
  730. } /* _mi_rec_pack */
  731. /*
  732.   Check if a record was correctly packed. Used only by myisamchk
  733.   Returns 0 if record is ok.
  734. */
  735. my_bool _mi_rec_check(MI_INFO *info,const char *record, byte *rec_buff,
  736.                       ulong packed_length, my_bool with_checksum)
  737. {
  738.   uint length,new_length,flag,bit,i;
  739.   char *pos,*end,*packpos,*to;
  740.   enum en_fieldtype type;
  741.   reg3 MI_COLUMNDEF *rec;
  742.   DBUG_ENTER("_mi_rec_check");
  743.   packpos=rec_buff; to= rec_buff+info->s->base.pack_bits;
  744.   rec=info->s->rec;
  745.   flag= *packpos; bit=1;
  746.   for (i=info->s->base.fields ; i-- > 0; record+= length, rec++)
  747.   {
  748.     length=(uint) rec->length;
  749.     if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL)
  750.     {
  751.       if (type == FIELD_BLOB)
  752.       {
  753. uint blob_length=
  754.   _mi_calc_blob_length(length-mi_portable_sizeof_char_ptr,record);
  755. if (!blob_length && !(flag & bit))
  756.   goto err;
  757. if (blob_length)
  758.   to+=length - mi_portable_sizeof_char_ptr+ blob_length;
  759.       }
  760.       else if (type == FIELD_SKIP_ZERO)
  761.       {
  762. if (memcmp((byte*) record,zero_string,length) == 0)
  763. {
  764.   if (!(flag & bit))
  765.     goto err;
  766. }
  767. else
  768.   to+=length;
  769.       }
  770.       else if (type == FIELD_SKIP_ENDSPACE ||
  771.        type == FIELD_SKIP_PRESPACE)
  772.       {
  773. pos= (byte*) record; end= (byte*) record + length;
  774. if (type == FIELD_SKIP_ENDSPACE)
  775. { /* Pack trailing spaces */
  776.   while (end > record && *(end-1) == ' ')
  777.     end--;
  778. }
  779. else
  780. { /* Pack pre-spaces */
  781.   while (pos < end && *pos == ' ')
  782.     pos++;
  783. }
  784. new_length=(uint) (end-pos);
  785. if (new_length +1 + test(rec->length > 255 && new_length > 127)
  786.     < length)
  787. {
  788.   if (!(flag & bit))
  789.     goto err;
  790.   if (rec->length > 255 && new_length > 127)
  791.   {
  792.     if (to[0] != (char) ((new_length & 127)+128) ||
  793. to[1] != (char) (new_length >> 7))
  794.       goto err;
  795.     to+=2;
  796.   }
  797.   else if (*to++ != (char) new_length)
  798.     goto err;
  799.   to+=new_length;
  800. }
  801. else
  802.   to+=length;
  803.       }
  804.       else if (type == FIELD_VARCHAR)
  805.       {
  806. uint tmp_length=uint2korr(record);
  807. to+=get_pack_length(tmp_length)+tmp_length;
  808. continue;
  809.       }
  810.       else
  811.       {
  812. to+=length;
  813. continue; /* Normal field */
  814.       }
  815.       if ((bit= bit << 1) >= 256)
  816.       {
  817. flag= *++packpos;
  818. bit=1;
  819.       }
  820.     }
  821.     else
  822.     {
  823.       to+=length;
  824.     }
  825.   }
  826.   if (packed_length != (uint) (to - rec_buff) + test(info->s->calc_checksum) ||
  827.       (bit != 1 && (flag & ~(bit - 1))))
  828.     goto err;
  829.   if (with_checksum && ((uchar) info->checksum != (uchar) *to))
  830.   {
  831.     DBUG_PRINT("error",("wrong checksum for row"));
  832.     goto err;
  833.   }
  834.   DBUG_RETURN(0);
  835. err:
  836.   DBUG_RETURN(1);
  837. }
  838. /* Unpacks a record */
  839. /* Returns -1 and my_errno =HA_ERR_RECORD_DELETED if reclength isn't */
  840. /* right. Returns reclength (>0) if ok */
  841. ulong _mi_rec_unpack(register MI_INFO *info, register byte *to, byte *from,
  842.      ulong found_length)
  843. {
  844.   uint flag,bit,length,rec_length,min_pack_length;
  845.   enum en_fieldtype type;
  846.   byte *from_end,*to_end,*packpos;
  847.   reg3 MI_COLUMNDEF *rec,*end_field;
  848.   DBUG_ENTER("_mi_rec_unpack");
  849.   to_end=to + info->s->base.reclength;
  850.   from_end=from+found_length;
  851.   flag= (uchar) *from; bit=1; packpos=from;
  852.   if (found_length < info->s->base.min_pack_length)
  853.     goto err;
  854.   from+= info->s->base.pack_bits;
  855.   min_pack_length=info->s->base.min_pack_length - info->s->base.pack_bits;
  856.   for (rec=info->s->rec , end_field=rec+info->s->base.fields ;
  857.        rec < end_field ; to+= rec_length, rec++)
  858.   {
  859.     rec_length=rec->length;
  860.     if ((type = (enum en_fieldtype) rec->type) != FIELD_NORMAL &&
  861. (type != FIELD_CHECK))
  862.     {
  863.       if (type == FIELD_VARCHAR)
  864.       {
  865. get_key_length(length,from);
  866. if (length > rec_length-2)
  867.   goto err;
  868. int2store(to,length);
  869. memcpy(to+2,from,length);
  870. from+=length;
  871. continue;
  872.       }
  873.       if (flag & bit)
  874.       {
  875. if (type == FIELD_BLOB || type == FIELD_SKIP_ZERO)
  876.   bzero((byte*) to,rec_length);
  877. else if (type == FIELD_SKIP_ENDSPACE ||
  878.  type == FIELD_SKIP_PRESPACE)
  879. {
  880.   if (rec->length > 255 && *from & 128)
  881.   {
  882.     if (from + 1 >= from_end)
  883.       goto err;
  884.     length= (*from & 127)+ ((uint) (uchar) *(from+1) << 7); from+=2;
  885.   }
  886.   else
  887.   {
  888.     if (from == from_end)
  889.       goto err;
  890.     length= (uchar) *from++;
  891.   }
  892.   min_pack_length--;
  893.   if (length >= rec_length ||
  894.       min_pack_length + length > (uint) (from_end - from))
  895.     goto err;
  896.   if (type == FIELD_SKIP_ENDSPACE)
  897.   {
  898.     memcpy(to,(byte*) from,(size_t) length);
  899.     bfill((byte*) to+length,rec_length-length,' ');
  900.   }
  901.   else
  902.   {
  903.     bfill((byte*) to,rec_length-length,' ');
  904.     memcpy(to+rec_length-length,(byte*) from,(size_t) length);
  905.   }
  906.   from+=length;
  907. }
  908.       }
  909.       else if (type == FIELD_BLOB)
  910.       {
  911. uint size_length=rec_length- mi_portable_sizeof_char_ptr;
  912. ulong blob_length=_mi_calc_blob_length(size_length,from);
  913. if ((ulong) (from_end-from) - size_length < blob_length ||
  914.     min_pack_length > (uint) (from_end -(from+size_length+blob_length)))
  915.   goto err;
  916. memcpy((byte*) to,(byte*) from,(size_t) size_length);
  917. from+=size_length;
  918. memcpy_fixed((byte*) to+size_length,(byte*) &from,sizeof(char*));
  919. from+=blob_length;
  920.       }
  921.       else
  922.       {
  923. if (type == FIELD_SKIP_ENDSPACE || type == FIELD_SKIP_PRESPACE)
  924.   min_pack_length--;
  925. if (min_pack_length + rec_length > (uint) (from_end - from))
  926.   goto err;
  927. memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length;
  928.       }
  929.       if ((bit= bit << 1) >= 256)
  930.       {
  931. flag= (uchar) *++packpos; bit=1;
  932.       }
  933.     }
  934.     else
  935.     {
  936.       if (min_pack_length > (uint) (from_end - from))
  937. goto err;
  938.       min_pack_length-=rec_length;
  939.       memcpy(to,(byte*) from,(size_t) rec_length); from+=rec_length;
  940.     }
  941.   }
  942.   if (info->s->calc_checksum)
  943.     from++;
  944.   if (to == to_end && from == from_end && (bit == 1 || !(flag & ~(bit-1))))
  945.     DBUG_RETURN(found_length);
  946. err:
  947.   my_errno=HA_ERR_RECORD_DELETED;
  948.   DBUG_PRINT("error",("to_end: %lx -> %lx  from_end: %lx -> %lx",
  949.       to,to_end,from,from_end));
  950.   DBUG_DUMP("from",(byte*) info->rec_buff,info->s->base.min_pack_length);
  951.   DBUG_RETURN(MY_FILE_ERROR);
  952. } /* _mi_rec_unpack */
  953. /* Calc length of blob. Update info in blobs->length */
  954. ulong _my_calc_total_blob_length(MI_INFO *info, const byte *record)
  955. {
  956.   ulong length;
  957.   MI_BLOB *blob,*end;
  958.   for (length=0, blob= info->blobs, end=blob+info->s->base.blobs ;
  959.        blob != end;
  960.        blob++)
  961.   {
  962.     blob->length=_mi_calc_blob_length(blob->pack_length,record + blob->offset);
  963.     length+=blob->length;
  964.   }
  965.   return length;
  966. }
  967. ulong _mi_calc_blob_length(uint length, const byte *pos)
  968. {
  969.   switch (length) {
  970.   case 1:
  971.     return (uint) (uchar) *pos;
  972.   case 2:
  973.     return (uint) uint2korr(pos);
  974.   case 3:
  975.     return uint3korr(pos);
  976.   case 4:
  977.     return uint4korr(pos);
  978.   default:
  979.     break;
  980.   }
  981.   return 0; /* Impossible */
  982. }
  983. void _my_store_blob_length(byte *pos,uint pack_length,uint length)
  984. {
  985.   switch (pack_length) {
  986.   case 1:
  987.     *pos= (uchar) length;
  988.     break;
  989.   case 2:
  990.     int2store(pos,length);
  991.     break;
  992.   case 3:
  993.     int3store(pos,length);
  994.     break;
  995.   case 4:
  996.     int4store(pos,length);
  997.   default:
  998.     break;
  999.   }
  1000.   return;
  1001. }
  1002. /* Read record from datafile */
  1003. /* Returns 0 if ok, -1 if error */
  1004. int _mi_read_dynamic_record(MI_INFO *info, my_off_t filepos, byte *buf)
  1005. {
  1006.   int flag;
  1007.   uint b_type,left_length;
  1008.   byte *to;
  1009.   MI_BLOCK_INFO block_info;
  1010.   File file;
  1011.   DBUG_ENTER("mi_read_dynamic_record");
  1012.   if (filepos != HA_OFFSET_ERROR)
  1013.   {
  1014.     LINT_INIT(to);
  1015.     LINT_INIT(left_length);
  1016.     file=info->dfile;
  1017.     block_info.next_filepos=filepos; /* for easyer loop */
  1018.     flag=block_info.second_read=0;
  1019.     do
  1020.     {
  1021.       if (info->opt_flag & WRITE_CACHE_USED &&
  1022.   info->rec_cache.pos_in_file <= block_info.next_filepos &&
  1023.   flush_io_cache(&info->rec_cache))
  1024. goto err;
  1025.       info->rec_cache.seek_not_done=1;
  1026.       if ((b_type=_mi_get_block_info(&block_info,file,
  1027.      block_info.next_filepos))
  1028.   & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  1029.      BLOCK_FATAL_ERROR))
  1030.       {
  1031. if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED))
  1032.   my_errno=HA_ERR_RECORD_DELETED;
  1033. goto err;
  1034.       }
  1035.       if (flag == 0) /* First block */
  1036.       {
  1037. flag=1;
  1038. if (block_info.rec_len > (uint) info->s->base.max_pack_length)
  1039.   goto panic;
  1040. if (info->s->base.blobs)
  1041. {
  1042.   if (!(to=mi_alloc_rec_buff(info, block_info.rec_len,
  1043.      &info->rec_buff)))
  1044.     goto err;
  1045. }
  1046. else
  1047.   to= info->rec_buff;
  1048. left_length=block_info.rec_len;
  1049.       }
  1050.       if (left_length < block_info.data_len || ! block_info.data_len)
  1051. goto panic; /* Wrong linked record */
  1052.       if (my_pread(file,(byte*) to,block_info.data_len,block_info.filepos,
  1053.    MYF(MY_NABP)))
  1054. goto panic;
  1055.       left_length-=block_info.data_len;
  1056.       to+=block_info.data_len;
  1057.     } while (left_length);
  1058.     info->update|= HA_STATE_AKTIV; /* We have a aktive record */
  1059.     fast_mi_writeinfo(info);
  1060.     DBUG_RETURN(_mi_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) !=
  1061. MY_FILE_ERROR ? 0 : -1);
  1062.   }
  1063.   fast_mi_writeinfo(info);
  1064.   DBUG_RETURN(-1); /* Wrong data to read */
  1065. panic:
  1066.   my_errno=HA_ERR_WRONG_IN_RECORD;
  1067. err:
  1068.   VOID(_mi_writeinfo(info,0));
  1069.   DBUG_RETURN(-1);
  1070. }
  1071. /* compare unique constraint between stored rows */
  1072. int _mi_cmp_dynamic_unique(MI_INFO *info, MI_UNIQUEDEF *def,
  1073.    const byte *record, my_off_t pos)
  1074. {
  1075.   byte *rec_buff,*old_record;
  1076.   int error;
  1077.   DBUG_ENTER("_mi_cmp_dynamic_unique");
  1078.   if (!(old_record=my_alloca(info->s->base.reclength)))
  1079.     DBUG_RETURN(1);
  1080.   /* Don't let the compare destroy blobs that may be in use */
  1081.   rec_buff=info->rec_buff;
  1082.   if (info->s->base.blobs)
  1083.     info->rec_buff=0;
  1084.   error=_mi_read_dynamic_record(info,pos,old_record);
  1085.   if (!error)
  1086.     error=mi_unique_comp(def, record, old_record, def->null_are_equal);
  1087.   if (info->s->base.blobs)
  1088.   {
  1089.     my_free(mi_get_rec_buff_ptr(info, info->rec_buff), MYF(MY_ALLOW_ZERO_PTR));
  1090.     info->rec_buff=rec_buff;
  1091.   }
  1092.   my_afree(old_record);
  1093.   DBUG_RETURN(error);
  1094. }
  1095. /* Compare of record one disk with packed record in memory */
  1096. int _mi_cmp_dynamic_record(register MI_INFO *info, register const byte *record)
  1097. {
  1098.   uint flag,reclength,b_type;
  1099.   my_off_t filepos;
  1100.   byte *buffer;
  1101.   MI_BLOCK_INFO block_info;
  1102.   DBUG_ENTER("_mi_cmp_dynamic_record");
  1103. /* We are going to do changes; dont let anybody disturb */
  1104.   dont_break(); /* Dont allow SIGHUP or SIGINT */
  1105.   if (info->opt_flag & WRITE_CACHE_USED)
  1106.   {
  1107.     info->update&= ~(HA_STATE_WRITE_AT_END | HA_STATE_EXTEND_BLOCK);
  1108.     if (flush_io_cache(&info->rec_cache))
  1109.       DBUG_RETURN(-1);
  1110.   }
  1111.   info->rec_cache.seek_not_done=1;
  1112. /* If nobody have touched the database we don't have to test rec */
  1113.   buffer=info->rec_buff;
  1114.   if ((info->opt_flag & READ_CHECK_USED))
  1115.   { /* If check isn't disabled  */
  1116.     if (info->s->base.blobs)
  1117.     {
  1118.       if (!(buffer=(byte*) my_alloca(info->s->base.pack_reclength+
  1119.      _my_calc_total_blob_length(info,record))))
  1120. DBUG_RETURN(-1);
  1121.     }
  1122.     reclength=_mi_rec_pack(info,buffer,record);
  1123.     record= buffer;
  1124.     filepos=info->lastpos;
  1125.     flag=block_info.second_read=0;
  1126.     block_info.next_filepos=filepos;
  1127.     while (reclength > 0)
  1128.     {
  1129.       if ((b_type=_mi_get_block_info(&block_info,info->dfile,
  1130.     block_info.next_filepos))
  1131.   & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  1132.      BLOCK_FATAL_ERROR))
  1133.       {
  1134. if (b_type & (BLOCK_SYNC_ERROR | BLOCK_DELETED))
  1135.   my_errno=HA_ERR_RECORD_CHANGED;
  1136. goto err;
  1137.       }
  1138.       if (flag == 0) /* First block */
  1139.       {
  1140. flag=1;
  1141. if (reclength != block_info.rec_len)
  1142. {
  1143.   my_errno=HA_ERR_RECORD_CHANGED;
  1144.   goto err;
  1145. }
  1146.       } else if (reclength < block_info.data_len)
  1147.       {
  1148. my_errno=HA_ERR_WRONG_IN_RECORD;
  1149. goto err;
  1150.       }
  1151.       reclength-=block_info.data_len;
  1152.       if (_mi_cmp_buffer(info->dfile,record,block_info.filepos,
  1153.  block_info.data_len))
  1154.       {
  1155. my_errno=HA_ERR_RECORD_CHANGED;
  1156. goto err;
  1157.       }
  1158.       flag=1;
  1159.       record+=block_info.data_len;
  1160.     }
  1161.   }
  1162.   my_errno=0;
  1163. err:
  1164.   if (buffer != info->rec_buff)
  1165.     my_afree((gptr) buffer);
  1166.   DBUG_RETURN(my_errno);
  1167. }
  1168. /* Compare file to buffert */
  1169. static int _mi_cmp_buffer(File file, const byte *buff, my_off_t filepos,
  1170.   uint length)
  1171. {
  1172.   uint next_length;
  1173.   char temp_buff[IO_SIZE*2];
  1174.   DBUG_ENTER("_mi_cmp_buffer");
  1175.   next_length= IO_SIZE*2 - (uint) (filepos & (IO_SIZE-1));
  1176.   while (length > IO_SIZE*2)
  1177.   {
  1178.     if (my_pread(file,temp_buff,next_length,filepos, MYF(MY_NABP)) ||
  1179. memcmp((byte*) buff,temp_buff,next_length))
  1180.       goto err;
  1181.     filepos+=next_length;
  1182.     buff+=next_length;
  1183.     length-= next_length;
  1184.     next_length=IO_SIZE*2;
  1185.   }
  1186.   if (my_pread(file,temp_buff,length,filepos,MYF(MY_NABP)))
  1187.     goto err;
  1188.   DBUG_RETURN(memcmp((byte*) buff,temp_buff,length));
  1189. err:
  1190.   DBUG_RETURN(1);
  1191. }
  1192. int _mi_read_rnd_dynamic_record(MI_INFO *info, byte *buf,
  1193. register my_off_t filepos,
  1194. my_bool skip_deleted_blocks)
  1195. {
  1196.   int flag,info_read,save_errno;
  1197.   uint left_len,b_type;
  1198.   byte *to;
  1199.   MI_BLOCK_INFO block_info;
  1200.   MYISAM_SHARE *share=info->s;
  1201.   DBUG_ENTER("_mi_read_rnd_dynamic_record");
  1202.   info_read=0;
  1203.   LINT_INIT(to);
  1204.   if (info->lock_type == F_UNLCK)
  1205.   {
  1206. #ifndef UNSAFE_LOCKING
  1207.     if (share->tot_locks == 0)
  1208.     {
  1209.       if (my_lock(share->kfile,F_RDLCK,0L,F_TO_EOF,
  1210.   MYF(MY_SEEK_NOT_DONE) | info->lock_wait))
  1211. DBUG_RETURN(my_errno);
  1212.     }
  1213. #else
  1214.     info->tmp_lock_type=F_RDLCK;
  1215. #endif
  1216.   }
  1217.   else
  1218.     info_read=1; /* memory-keyinfoblock is ok */
  1219.   flag=block_info.second_read=0;
  1220.   left_len=1;
  1221.   do
  1222.   {
  1223.     if (filepos >= info->state->data_file_length)
  1224.     {
  1225.       if (!info_read)
  1226.       { /* Check if changed */
  1227. info_read=1;
  1228. info->rec_cache.seek_not_done=1;
  1229. if (mi_state_info_read_dsk(share->kfile,&share->state,1))
  1230.   goto panic;
  1231.       }
  1232.       if (filepos >= info->state->data_file_length)
  1233.       {
  1234. my_errno= HA_ERR_END_OF_FILE;
  1235. goto err;
  1236.       }
  1237.     }
  1238.     if (info->opt_flag & READ_CACHE_USED)
  1239.     {
  1240.       if (_mi_read_cache(&info->rec_cache,(byte*) block_info.header,filepos,
  1241.  sizeof(block_info.header),
  1242.  (!flag && skip_deleted_blocks ? READING_NEXT : 0) |
  1243.  READING_HEADER))
  1244. goto panic;
  1245.       b_type=_mi_get_block_info(&block_info,-1,filepos);
  1246.     }
  1247.     else
  1248.     {
  1249.       if (info->opt_flag & WRITE_CACHE_USED &&
  1250.   info->rec_cache.pos_in_file <= filepos &&
  1251.   flush_io_cache(&info->rec_cache))
  1252. DBUG_RETURN(my_errno);
  1253.       info->rec_cache.seek_not_done=1;
  1254.       b_type=_mi_get_block_info(&block_info,info->dfile,filepos);
  1255.     }
  1256.     if (b_type & (BLOCK_DELETED | BLOCK_ERROR | BLOCK_SYNC_ERROR |
  1257.   BLOCK_FATAL_ERROR))
  1258.     {
  1259.       if ((b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR))
  1260.   && skip_deleted_blocks)
  1261.       {
  1262. filepos=block_info.filepos+block_info.block_len;
  1263. block_info.second_read=0;
  1264. continue; /* Search after next_record */
  1265.       }
  1266.       if (b_type & (BLOCK_DELETED | BLOCK_SYNC_ERROR))
  1267.       {
  1268. my_errno=HA_ERR_RECORD_DELETED;
  1269. info->lastpos=block_info.filepos;
  1270. info->nextpos=block_info.filepos+block_info.block_len;
  1271.       }
  1272.       goto err;
  1273.     }
  1274.     if (flag == 0) /* First block */
  1275.     {
  1276.       if (block_info.rec_len > (uint) share->base.max_pack_length)
  1277. goto panic;
  1278.       info->lastpos=filepos;
  1279.       if (share->base.blobs)
  1280.       {
  1281. if (!(to= mi_alloc_rec_buff(info, block_info.rec_len,
  1282.     &info->rec_buff)))
  1283.   goto err;
  1284.       }
  1285.       else
  1286. to= info->rec_buff;
  1287.       left_len=block_info.rec_len;
  1288.     }
  1289.     if (left_len < block_info.data_len)
  1290.       goto panic; /* Wrong linked record */
  1291.     /* copy information that is already read */
  1292.     {
  1293.       uint offset=(uint) (block_info.filepos - filepos);
  1294.       uint tmp_length= (sizeof(block_info.header) - offset);
  1295.       filepos=block_info.filepos;
  1296.       if (tmp_length > block_info.data_len)
  1297. tmp_length= block_info.data_len;
  1298.       if (tmp_length)
  1299.       {
  1300. memcpy((byte*) to, block_info.header+offset,tmp_length);
  1301. block_info.data_len-=tmp_length;
  1302. left_len-=tmp_length;
  1303. to+=tmp_length;
  1304. filepos+=tmp_length;
  1305.      }
  1306.     }
  1307.     /* read rest of record from file */
  1308.     if (block_info.data_len)
  1309.     {
  1310.       if (info->opt_flag & READ_CACHE_USED)
  1311.       {
  1312. if (_mi_read_cache(&info->rec_cache,(byte*) to,filepos,
  1313.    block_info.data_len,
  1314.    (!flag && skip_deleted_blocks) ? READING_NEXT :0))
  1315.   goto panic;
  1316.       }
  1317.       else
  1318.       {
  1319. /* VOID(my_seek(info->dfile,filepos,MY_SEEK_SET,MYF(0))); */
  1320. if (my_read(info->dfile,(byte*) to,block_info.data_len,MYF(MY_NABP)))
  1321. {
  1322.   if (my_errno == -1)
  1323.     my_errno= HA_ERR_WRONG_IN_RECORD; /* Unexpected end of file */
  1324.   goto err;
  1325. }
  1326.       }
  1327.     }
  1328.     if (flag++ == 0)
  1329.     {
  1330.       info->nextpos=block_info.filepos+block_info.block_len;
  1331.       skip_deleted_blocks=0;
  1332.     }
  1333.     left_len-=block_info.data_len;
  1334.     to+=block_info.data_len;
  1335.     filepos=block_info.next_filepos;
  1336.   } while (left_len);
  1337.   info->update|= HA_STATE_AKTIV | HA_STATE_KEY_CHANGED;
  1338.   fast_mi_writeinfo(info);
  1339.   if (_mi_rec_unpack(info,buf,info->rec_buff,block_info.rec_len) !=
  1340.       MY_FILE_ERROR)
  1341.     DBUG_RETURN(0);
  1342.   DBUG_RETURN(my_errno); /* Wrong record */
  1343. panic:
  1344.   my_errno=HA_ERR_WRONG_IN_RECORD; /* Something is fatal wrong */
  1345. err:
  1346.   save_errno=my_errno;
  1347.   VOID(_mi_writeinfo(info,0));
  1348.   DBUG_RETURN(my_errno=save_errno);
  1349. }
  1350. /* Read and process header from a dynamic-record-file */
  1351. uint _mi_get_block_info(MI_BLOCK_INFO *info, File file, my_off_t filepos)
  1352. {
  1353.   uint return_val=0;
  1354.   uchar *header=info->header;
  1355.   if (file >= 0)
  1356.   {
  1357.     VOID(my_seek(file,filepos,MY_SEEK_SET,MYF(0)));
  1358.     if (my_read(file,(char*) header,sizeof(info->header),MYF(0)) !=
  1359. sizeof(info->header))
  1360.       goto err;
  1361.   }
  1362.   DBUG_DUMP("header",(byte*) header,MI_BLOCK_INFO_HEADER_LENGTH);
  1363.   if (info->second_read)
  1364.   {
  1365.     if (info->header[0] <= 6 || info->header[0] == 13)
  1366.       return_val=BLOCK_SYNC_ERROR;
  1367.   }
  1368.   else
  1369.   {
  1370.     if (info->header[0] > 6 && info->header[0] != 13)
  1371.       return_val=BLOCK_SYNC_ERROR;
  1372.   }
  1373.   info->next_filepos= HA_OFFSET_ERROR; /* Dummy if no next block */
  1374.   switch (info->header[0]) {
  1375.   case 0:
  1376.     if ((info->block_len=(uint) mi_uint3korr(header+1)) <
  1377. MI_MIN_BLOCK_LENGTH ||
  1378. (info->block_len & (MI_DYN_ALIGN_SIZE -1)))
  1379.       goto err;
  1380.     info->filepos=filepos;
  1381.     info->next_filepos=mi_sizekorr(header+4);
  1382.     info->prev_filepos=mi_sizekorr(header+12);
  1383. #if SIZEOF_OFF_T == 4
  1384.     if ((mi_uint4korr(header+4) != 0 &&
  1385.  (mi_uint4korr(header+4) != (ulong) ~0 ||
  1386.   info->next_filepos != (ulong) ~0)) ||
  1387. (mi_uint4korr(header+12) != 0 &&
  1388.  (mi_uint4korr(header+12) != (ulong) ~0 ||
  1389.   info->prev_filepos != (ulong) ~0)))
  1390.       goto err;
  1391. #endif
  1392.     return return_val | BLOCK_DELETED; /* Deleted block */
  1393.   case 1:
  1394.     info->rec_len=info->data_len=info->block_len=mi_uint2korr(header+1);
  1395.     info->filepos=filepos+3;
  1396.     return return_val | BLOCK_FIRST | BLOCK_LAST;
  1397.   case 2:
  1398.     info->rec_len=info->data_len=info->block_len=mi_uint3korr(header+1);
  1399.     info->filepos=filepos+4;
  1400.     return return_val | BLOCK_FIRST | BLOCK_LAST;
  1401.   case 13:
  1402.     info->rec_len=mi_uint4korr(header+1);
  1403.     info->block_len=info->data_len=mi_uint3korr(header+5);
  1404.     info->next_filepos=mi_sizekorr(header+8);
  1405.     info->second_read=1;
  1406.     info->filepos=filepos+16;
  1407.     return return_val | BLOCK_FIRST;
  1408.   case 3:
  1409.     info->rec_len=info->data_len=mi_uint2korr(header+1);
  1410.     info->block_len=info->rec_len+ (uint) header[3];
  1411.     info->filepos=filepos+4;
  1412.     return return_val | BLOCK_FIRST | BLOCK_LAST;
  1413.   case 4:
  1414.     info->rec_len=info->data_len=mi_uint3korr(header+1);
  1415.     info->block_len=info->rec_len+ (uint) header[4];
  1416.     info->filepos=filepos+5;
  1417.     return return_val | BLOCK_FIRST | BLOCK_LAST;
  1418.   case 5:
  1419.     info->rec_len=mi_uint2korr(header+1);
  1420.     info->block_len=info->data_len=mi_uint2korr(header+3);
  1421.     info->next_filepos=mi_sizekorr(header+5);
  1422.     info->second_read=1;
  1423.     info->filepos=filepos+13;
  1424.     return return_val | BLOCK_FIRST;
  1425.   case 6:
  1426.     info->rec_len=mi_uint3korr(header+1);
  1427.     info->block_len=info->data_len=mi_uint3korr(header+4);
  1428.     info->next_filepos=mi_sizekorr(header+7);
  1429.     info->second_read=1;
  1430.     info->filepos=filepos+15;
  1431.     return return_val | BLOCK_FIRST;
  1432.     /* The following blocks are identical to 1-6 without rec_len */
  1433.   case 7:
  1434.     info->data_len=info->block_len=mi_uint2korr(header+1);
  1435.     info->filepos=filepos+3;
  1436.     return return_val | BLOCK_LAST;
  1437.   case 8:
  1438.     info->data_len=info->block_len=mi_uint3korr(header+1);
  1439.     info->filepos=filepos+4;
  1440.     return return_val | BLOCK_LAST;
  1441.   case 9:
  1442.     info->data_len=mi_uint2korr(header+1);
  1443.     info->block_len=info->data_len+ (uint) header[3];
  1444.     info->filepos=filepos+4;
  1445.     return return_val | BLOCK_LAST;
  1446.   case 10:
  1447.     info->data_len=mi_uint3korr(header+1);
  1448.     info->block_len=info->data_len+ (uint) header[4];
  1449.     info->filepos=filepos+5;
  1450.     return return_val | BLOCK_LAST;
  1451.   case 11:
  1452.     info->data_len=info->block_len=mi_uint2korr(header+1);
  1453.     info->next_filepos=mi_sizekorr(header+3);
  1454.     info->second_read=1;
  1455.     info->filepos=filepos+11;
  1456.     return return_val;
  1457.   case 12:
  1458.     info->data_len=info->block_len=mi_uint3korr(header+1);
  1459.     info->next_filepos=mi_sizekorr(header+4);
  1460.     info->second_read=1;
  1461.     info->filepos=filepos+12;
  1462.     return return_val;
  1463.   }
  1464. err:
  1465.   my_errno=HA_ERR_WRONG_IN_RECORD;  /* Garbage */
  1466.   return BLOCK_ERROR;
  1467. }