hp_write.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:7k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
  2.    
  3.    This program is free software; you can redistribute it and/or modify
  4.    it under the terms of the GNU General Public License as published by
  5.    the Free Software Foundation; either version 2 of the License, or
  6.    (at your option) any later version.
  7.    
  8.    This program is distributed in the hope that it will be useful,
  9.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  11.    GNU General Public License for more details.
  12.    
  13.    You should have received a copy of the GNU General Public License
  14.    along with this program; if not, write to the Free Software
  15.    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
  16. /* Write a record to heap-databas */
  17. #include "heapdef.h"
  18. #ifdef __WIN__
  19. #include <fcntl.h>
  20. #endif
  21. #define LOWFIND 1
  22. #define LOWUSED 2
  23. #define HIGHFIND 4
  24. #define HIGHUSED 8
  25. static byte *next_free_record_pos(HP_SHARE *info);
  26. static HASH_INFO *_hp_find_free_hash(HP_SHARE *info, HP_BLOCK *block,
  27.      ulong records);
  28. int heap_write(HP_INFO *info, const byte *record)
  29. {
  30.   uint key;
  31.   byte *pos;
  32.   HP_SHARE *share=info->s;
  33.   DBUG_ENTER("heap_write");
  34. #ifndef DBUG_OFF
  35.   if (info->mode && O_RDONLY)
  36.   {
  37.     DBUG_RETURN(my_errno=EACCES);
  38.   }
  39. #endif
  40.   if (!(pos=next_free_record_pos(share)))
  41.     DBUG_RETURN(my_errno);
  42.   share->changed=1;
  43.   for (key=0 ; key < share->keys ; key++)
  44.   {
  45.     if (_hp_write_key(share,share->keydef+key,record,pos))
  46.       goto err;
  47.   }
  48.   memcpy(pos,record,(size_t) share->reclength);
  49.   pos[share->reclength]=1; /* Mark record as not deleted */
  50.   if (++share->records == share->blength)
  51.     share->blength+= share->blength;
  52.   info->current_ptr=pos;
  53.   info->current_hash_ptr=0;
  54.   info->update|=HA_STATE_AKTIV;
  55.   DBUG_RETURN(0);
  56. err:
  57.   info->errkey= key;
  58.   do
  59.   {
  60.     if (_hp_delete_key(info,share->keydef+key,record,pos,0))
  61.       break;
  62.   } while (key-- > 0);
  63.   share->deleted++;
  64.   *((byte**) pos)=share->del_link;
  65.   share->del_link=pos;
  66.   pos[share->reclength]=0; /* Record deleted */
  67.   DBUG_RETURN(my_errno);
  68. } /* heap_write */
  69. /* Find where to place new record */
  70. static byte *next_free_record_pos(HP_SHARE *info)
  71. {
  72.   int block_pos;
  73.   byte *pos;
  74.   ulong length;
  75.   DBUG_ENTER("next_free_record_pos");
  76.   if (info->del_link)
  77.   {
  78.     pos=info->del_link;
  79.     info->del_link= *((byte**) pos);
  80.     info->deleted--;
  81.     DBUG_PRINT("exit",("Used old position: %lx",pos));
  82.     DBUG_RETURN(pos);
  83.   }
  84.   if (!(block_pos=(info->records % info->block.records_in_block)))
  85.   {
  86.     if (info->records > info->max_records && info->max_records)
  87.     {
  88.       my_errno=HA_ERR_RECORD_FILE_FULL;
  89.       DBUG_RETURN(NULL);
  90.     }
  91.     if (_hp_get_new_block(&info->block,&length))
  92.       DBUG_RETURN(NULL);
  93.     info->data_length+=length;
  94.   }
  95.   DBUG_PRINT("exit",("Used new position: %lx",
  96.      (byte*) info->block.level_info[0].last_blocks+block_pos*
  97.      info->block.recbuffer));
  98.   DBUG_RETURN((byte*) info->block.level_info[0].last_blocks+
  99.       block_pos*info->block.recbuffer);
  100. }
  101. /* Write a hash-key to the hash-index */
  102. int _hp_write_key(register HP_SHARE *info, HP_KEYDEF *keyinfo,
  103.   const byte *record, byte *recpos)
  104. {
  105.   int flag;
  106.   ulong halfbuff,hashnr,first_index;
  107.   byte *ptr_to_rec,*ptr_to_rec2;
  108.   HASH_INFO *empty,*gpos,*gpos2,*pos;
  109.   DBUG_ENTER("hp_write_key");
  110.   LINT_INIT(gpos); LINT_INIT(gpos2);
  111.   LINT_INIT(ptr_to_rec); LINT_INIT(ptr_to_rec2);
  112.   flag=0;
  113.   if (!(empty= _hp_find_free_hash(info,&keyinfo->block,info->records)))
  114.     DBUG_RETURN(-1); /* No more memory */
  115.   halfbuff= (long) info->blength >> 1;
  116.   pos=  hp_find_hash(&keyinfo->block,(first_index=info->records-halfbuff));
  117.   if (pos != empty) /* If some records */
  118.   {
  119.     do
  120.     {
  121.       hashnr=_hp_rec_hashnr(keyinfo,pos->ptr_to_rec);
  122.       if (flag == 0) /* First loop; Check if ok */
  123. if (_hp_mask(hashnr,info->blength,info->records) != first_index)
  124.   break;
  125.       if (!(hashnr & halfbuff))
  126.       { /* Key will not move */
  127. if (!(flag & LOWFIND))
  128. {
  129.   if (flag & HIGHFIND)
  130.   {
  131.     flag=LOWFIND | HIGHFIND;
  132.     /* key shall be moved to the current empty position */
  133.     gpos=empty;
  134.     ptr_to_rec=pos->ptr_to_rec;
  135.     empty=pos; /* This place is now free */
  136.   }
  137.   else
  138.   {
  139.     flag=LOWFIND | LOWUSED; /* key isn't changed */
  140.     gpos=pos;
  141.     ptr_to_rec=pos->ptr_to_rec;
  142.   }
  143. }
  144. else
  145. {
  146.   if (!(flag & LOWUSED))
  147.   {
  148.     /* Change link of previous LOW-key */
  149.     gpos->ptr_to_rec=ptr_to_rec;
  150.     gpos->next_key=pos;
  151.     flag= (flag & HIGHFIND) | (LOWFIND | LOWUSED);
  152.   }
  153.   gpos=pos;
  154.   ptr_to_rec=pos->ptr_to_rec;
  155. }
  156.       }
  157.       else
  158.       { /* key will be moved */
  159. if (!(flag & HIGHFIND))
  160. {
  161.   flag= (flag & LOWFIND) | HIGHFIND;
  162.   /* key shall be moved to the last (empty) position */
  163.   gpos2 = empty; empty=pos;
  164.   ptr_to_rec2=pos->ptr_to_rec;
  165. }
  166. else
  167. {
  168.   if (!(flag & HIGHUSED))
  169.   {
  170.     /* Change link of previous hash-key and save */
  171.     gpos2->ptr_to_rec=ptr_to_rec2;
  172.     gpos2->next_key=pos;
  173.     flag= (flag & LOWFIND) | (HIGHFIND | HIGHUSED);
  174.   }
  175.   gpos2=pos;
  176.   ptr_to_rec2=pos->ptr_to_rec;
  177. }
  178.       }
  179.     }
  180.     while ((pos=pos->next_key));
  181.     if ((flag & (LOWFIND | LOWUSED)) == LOWFIND)
  182.     {
  183.       gpos->ptr_to_rec=ptr_to_rec;
  184.       gpos->next_key=0;
  185.     }
  186.     if ((flag & (HIGHFIND | HIGHUSED)) == HIGHFIND)
  187.     {
  188.       gpos2->ptr_to_rec=ptr_to_rec2;
  189.       gpos2->next_key=0;
  190.     }
  191.   }
  192.   /* Check if we are at the empty position */
  193.   pos=hp_find_hash(&keyinfo->block,_hp_mask(_hp_rec_hashnr(keyinfo,record),
  194.     info->blength,info->records+1));
  195.   if (pos == empty)
  196.   {
  197.     pos->ptr_to_rec=recpos;
  198.     pos->next_key=0;
  199.   }
  200.   else
  201.   {
  202.     /* Check if more records in same hash-nr family */
  203.     empty[0]=pos[0];
  204.     gpos=hp_find_hash(&keyinfo->block,
  205.       _hp_mask(_hp_rec_hashnr(keyinfo,pos->ptr_to_rec),
  206.        info->blength,info->records+1));
  207.     if (pos == gpos)
  208.     {
  209.       pos->ptr_to_rec=recpos;
  210.       pos->next_key=empty;
  211.     }
  212.     else
  213.     {
  214.       pos->ptr_to_rec=recpos;
  215.       pos->next_key=0;
  216.       _hp_movelink(pos,gpos,empty);
  217.     }
  218.     /* Check if dupplicated keys */
  219.     if ((keyinfo->flag & HA_NOSAME) && pos == gpos)
  220.     {
  221.       pos=empty;
  222.       do
  223.       {
  224. if (! _hp_rec_key_cmp(keyinfo,record,pos->ptr_to_rec))
  225. {
  226.   DBUG_RETURN(my_errno=HA_ERR_FOUND_DUPP_KEY);
  227. }
  228.       } while ((pos=pos->next_key));
  229.     }
  230.   }
  231.   DBUG_RETURN(0);
  232. }
  233. /* Returns ptr to block, and allocates block if neaded */
  234. static HASH_INFO *_hp_find_free_hash(HP_SHARE *info,
  235.      HP_BLOCK *block, ulong records)
  236. {
  237.   uint block_pos;
  238.   ulong length;
  239.   if (records < block->last_allocated)
  240.     return hp_find_hash(block,records);
  241.   if (!(block_pos=(records % block->records_in_block)))
  242.   {
  243.     if (_hp_get_new_block(block,&length))
  244.       return(NULL);
  245.     info->index_length+=length;
  246.   }
  247.   block->last_allocated=records+1;
  248.   return((HASH_INFO*) ((byte*) block->level_info[0].last_blocks+
  249.        block_pos*block->recbuffer));
  250. }