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

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. /* remove current record in heap-database */
  17. #include "heapdef.h"
  18. int heap_delete(HP_INFO *info, const byte *record)
  19. {
  20.   uint key;
  21.   byte *pos;
  22.   HP_SHARE *share=info->s;
  23.   DBUG_ENTER("heap_delete");
  24.   DBUG_PRINT("enter",("info: %lx  record: %lx",info,record));
  25.   test_active(info);
  26.   if (info->opt_flag & READ_CHECK_USED && _hp_rectest(info,record))
  27.     DBUG_RETURN(my_errno); /* Record changed */
  28.   share->changed=1;
  29.   if ( --(share->records) < share->blength >> 1) share->blength>>=1;
  30.   pos=info->current_ptr;
  31.   for (key=0 ; key < share->keys ; key++)
  32.   {
  33.     if (_hp_delete_key(info,share->keydef+key,record,pos,
  34.        key == (uint) info->lastinx))
  35.       goto err;
  36.   }
  37.   info->update=HA_STATE_DELETED;
  38.   *((byte**) pos)=share->del_link;
  39.   share->del_link=pos;
  40.   pos[share->reclength]=0; /* Record deleted */
  41.   share->deleted++;
  42.   info->current_hash_ptr=0;
  43.   DBUG_RETURN(0);
  44.  err:
  45.   if( ++(share->records) == share->blength) share->blength+= share->blength;
  46.   DBUG_RETURN(my_errno);
  47. }
  48. /* Remove one key from hash-table */
  49. /* Flag is set if we want's to correct info->current_ptr */
  50. int _hp_delete_key(HP_INFO *info, register HP_KEYDEF *keyinfo,
  51.    const byte *record, byte *recpos, int flag)
  52. {
  53.   ulong blength,pos2,pos_hashnr,lastpos_hashnr;
  54.   HASH_INFO *lastpos,*gpos,*pos,*pos3,*empty,*last_ptr;
  55.   HP_SHARE *share=info->s;
  56.   DBUG_ENTER("_hp_delete_key");
  57.   blength=share->blength;
  58.   if (share->records+1 == blength) blength+= blength;
  59.   lastpos=hp_find_hash(&keyinfo->block,share->records);
  60.   last_ptr=0;
  61.   /* Search after record with key */
  62.   pos= hp_find_hash(&keyinfo->block,
  63.     _hp_mask(_hp_rec_hashnr(keyinfo,record),blength,
  64.      share->records+1));
  65.   gpos = pos3 = 0;
  66.   while (pos->ptr_to_rec != recpos)
  67.   {
  68.     if (flag && !_hp_rec_key_cmp(keyinfo,record,pos->ptr_to_rec))
  69.       last_ptr=pos; /* Previous same key */
  70.     gpos=pos;
  71.     if (!(pos=pos->next_key))
  72.     {
  73.       DBUG_RETURN(my_errno=HA_ERR_CRASHED); /* This shouldn't happend */
  74.     }
  75.   }
  76.   /* Remove link to record */
  77.   if (flag)
  78.   {
  79.     /* Save for heap_rnext/heap_rprev */
  80.     info->current_hash_ptr=last_ptr;
  81.     info->current_ptr = last_ptr ? last_ptr->ptr_to_rec : 0;
  82.     DBUG_PRINT("info",("Corrected current_ptr to point at: %lx",
  83.        info->current_ptr));
  84.   }
  85.   empty=pos;
  86.   if (gpos)
  87.     gpos->next_key=pos->next_key; /* unlink current ptr */
  88.   else if (pos->next_key)
  89.   {
  90.     empty=pos->next_key;
  91.     pos->ptr_to_rec=empty->ptr_to_rec;
  92.     pos->next_key=empty->next_key;
  93.   }
  94.   if (empty == lastpos) /* deleted last hash key */
  95.     DBUG_RETURN (0);
  96.   /* Move the last key (lastpos) */
  97.   lastpos_hashnr=_hp_rec_hashnr(keyinfo,lastpos->ptr_to_rec);
  98.   /* pos is where lastpos should be */
  99.   pos=hp_find_hash(&keyinfo->block,_hp_mask(lastpos_hashnr,share->blength,
  100.     share->records));
  101.   if (pos == empty) /* Move to empty position. */
  102.   {
  103.     empty[0]=lastpos[0];
  104.     DBUG_RETURN(0);
  105.   }
  106.   pos_hashnr=_hp_rec_hashnr(keyinfo,pos->ptr_to_rec);
  107.   /* pos3 is where the pos should be */
  108.   pos3= hp_find_hash(&keyinfo->block,
  109.      _hp_mask(pos_hashnr,share->blength,share->records));
  110.   if (pos != pos3)
  111.   { /* pos is on wrong posit */
  112.     empty[0]=pos[0]; /* Save it here */
  113.     pos[0]=lastpos[0]; /* This shold be here */
  114.     _hp_movelink(pos,pos3,empty); /* Fix link to pos */
  115.     DBUG_RETURN(0);
  116.   }
  117.   pos2= _hp_mask(lastpos_hashnr,blength,share->records+1);
  118.   if (pos2 == _hp_mask(pos_hashnr,blength,share->records+1))
  119.   { /* Identical key-positions */
  120.     if (pos2 != share->records)
  121.     {
  122.       empty[0]=lastpos[0];
  123.       _hp_movelink(lastpos,pos,empty);
  124.       DBUG_RETURN(0);
  125.     }
  126.     pos3= pos; /* Link pos->next after lastpos */
  127.   }
  128.   else pos3= 0; /* Different positions merge */
  129.   empty[0]=lastpos[0];
  130.   _hp_movelink(pos3,empty,pos->next_key);
  131.   pos->next_key=empty;
  132.   DBUG_RETURN(0);
  133. }