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

MySQL数据库

开发平台:

Visual C++

  1. /* Copyright (C) 2000,2004 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. /* open a isam-database */
  14. #include "fulltext.h"
  15. #include "sp_defs.h"
  16. #include "rt_index.h"
  17. #include <m_ctype.h>
  18. #if defined(MSDOS) || defined(__WIN__)
  19. #ifdef __WIN__
  20. #include <fcntl.h>
  21. #else
  22. #include <process.h> /* Prototype for getpid */
  23. #endif
  24. #endif
  25. #ifdef VMS
  26. #include "static.c"
  27. #endif
  28. static void setup_key_functions(MI_KEYDEF *keyinfo);
  29. #define get_next_element(to,pos,size) { memcpy((char*) to,pos,(size_t) size); 
  30. pos+=size;}
  31. #define disk_pos_assert(pos, end_pos) 
  32. if (pos > end_pos)             
  33. {                              
  34.   my_errno=HA_ERR_CRASHED;     
  35.   goto err;                    
  36. }
  37. /******************************************************************************
  38. ** Return the shared struct if the table is already open.
  39. ** In MySQL the server will handle version issues.
  40. ******************************************************************************/
  41. MI_INFO *test_if_reopen(char *filename)
  42. {
  43.   LIST *pos;
  44.   for (pos=myisam_open_list ; pos ; pos=pos->next)
  45.   {
  46.     MI_INFO *info=(MI_INFO*) pos->data;
  47.     MYISAM_SHARE *share=info->s;
  48.     if (!strcmp(share->unique_file_name,filename) && share->last_version)
  49.       return info;
  50.   }
  51.   return 0;
  52. }
  53. /******************************************************************************
  54.   open a MyISAM database.
  55.   See my_base.h for the handle_locking argument
  56.   if handle_locking and HA_OPEN_ABORT_IF_CRASHED then abort if the table
  57.   is marked crashed or if we are not using locking and the table doesn't
  58.   have an open count of 0.
  59. ******************************************************************************/
  60. MI_INFO *mi_open(const char *name, int mode, uint open_flags)
  61. {
  62.   int lock_error,kfile,open_mode,save_errno,have_rtree=0;
  63.   uint i,j,len,errpos,head_length,base_pos,offset,info_length,keys,
  64.     key_parts,unique_key_parts,fulltext_keys,uniques;
  65.   char name_buff[FN_REFLEN], org_name[FN_REFLEN], index_name[FN_REFLEN],
  66.        data_name[FN_REFLEN];
  67.   char *disk_cache, *disk_pos, *end_pos;
  68.   MI_INFO info,*m_info,*old_info;
  69.   MYISAM_SHARE share_buff,*share;
  70.   ulong rec_per_key_part[MI_MAX_POSSIBLE_KEY*MI_MAX_KEY_SEG];
  71.   my_off_t key_root[MI_MAX_POSSIBLE_KEY],key_del[MI_MAX_KEY_BLOCK_SIZE];
  72.   ulonglong max_key_file_length, max_data_file_length;
  73.   DBUG_ENTER("mi_open");
  74.   LINT_INIT(m_info);
  75.   kfile= -1;
  76.   lock_error=1;
  77.   errpos=0;
  78.   head_length=sizeof(share_buff.state.header);
  79.   bzero((byte*) &info,sizeof(info));
  80.   my_realpath(name_buff, fn_format(org_name,name,"",MI_NAME_IEXT,4),MYF(0));
  81.   pthread_mutex_lock(&THR_LOCK_myisam);
  82.   if (!(old_info=test_if_reopen(name_buff)))
  83.   {
  84.     share= &share_buff;
  85.     bzero((gptr) &share_buff,sizeof(share_buff));
  86.     share_buff.state.rec_per_key_part=rec_per_key_part;
  87.     share_buff.state.key_root=key_root;
  88.     share_buff.state.key_del=key_del;
  89.     share_buff.key_cache= multi_key_cache_search(name_buff, strlen(name_buff));
  90.     if ((kfile=my_open(name_buff,(open_mode=O_RDWR) | O_SHARE,MYF(0))) < 0)
  91.     {
  92.       if ((errno != EROFS && errno != EACCES) ||
  93.   mode != O_RDONLY ||
  94.   (kfile=my_open(name_buff,(open_mode=O_RDONLY) | O_SHARE,MYF(0))) < 0)
  95. goto err;
  96.     }
  97.     share->mode=open_mode;
  98.     errpos=1;
  99.     if (my_read(kfile,(char*) share->state.header.file_version,head_length,
  100. MYF(MY_NABP)))
  101.     {
  102.       my_errno= HA_ERR_NOT_A_TABLE;
  103.       goto err;
  104.     }
  105.     if (memcmp((byte*) share->state.header.file_version,
  106.        (byte*) myisam_file_magic, 4))
  107.     {
  108.       DBUG_PRINT("error",("Wrong header in %s",name_buff));
  109.       DBUG_DUMP("error_dump",(char*) share->state.header.file_version,
  110. head_length);
  111.       my_errno=HA_ERR_NOT_A_TABLE;
  112.       goto err;
  113.     }
  114.     share->options= mi_uint2korr(share->state.header.options);
  115.     if (share->options &
  116. ~(HA_OPTION_PACK_RECORD | HA_OPTION_PACK_KEYS |
  117.   HA_OPTION_COMPRESS_RECORD | HA_OPTION_READ_ONLY_DATA |
  118.   HA_OPTION_TEMP_COMPRESS_RECORD | HA_OPTION_CHECKSUM |
  119.   HA_OPTION_TMP_TABLE | HA_OPTION_DELAY_KEY_WRITE))
  120.     {
  121.       DBUG_PRINT("error",("wrong options: 0x%lx", share->options));
  122.       my_errno=HA_ERR_OLD_FILE;
  123.       goto err;
  124.     }
  125.     /* Don't call realpath() if the name can't be a link */
  126.     if (!strcmp(name_buff, org_name) ||
  127.         my_readlink(index_name, org_name, MYF(0)) == -1)
  128.       (void) strmov(index_name, org_name);
  129.     (void) fn_format(data_name,org_name,"",MI_NAME_DEXT,2+4+16);
  130.     info_length=mi_uint2korr(share->state.header.header_length);
  131.     base_pos=mi_uint2korr(share->state.header.base_pos);
  132.     if (!(disk_cache=(char*) my_alloca(info_length+128)))
  133.     {
  134.       my_errno=ENOMEM;
  135.       goto err;
  136.     }
  137.     end_pos=disk_cache+info_length;
  138.     errpos=2;
  139.     VOID(my_seek(kfile,0L,MY_SEEK_SET,MYF(0)));
  140.     if (!(open_flags & HA_OPEN_TMP_TABLE))
  141.     {
  142.       if ((lock_error=my_lock(kfile,F_RDLCK,0L,F_TO_EOF,
  143.       MYF(open_flags & HA_OPEN_WAIT_IF_LOCKED ?
  144.   0 : MY_DONT_WAIT))) &&
  145.   !(open_flags & HA_OPEN_IGNORE_IF_LOCKED))
  146. goto err;
  147.     }
  148.     errpos=3;
  149.     if (my_read(kfile,disk_cache,info_length,MYF(MY_NABP)))
  150.     {
  151.       my_errno=HA_ERR_CRASHED;
  152.       goto err;
  153.     }
  154.     len=mi_uint2korr(share->state.header.state_info_length);
  155.     keys=    (uint) share->state.header.keys;
  156.     uniques= (uint) share->state.header.uniques;
  157.     fulltext_keys= (uint) share->state.header.fulltext_keys;
  158.     key_parts= mi_uint2korr(share->state.header.key_parts);
  159.     unique_key_parts= mi_uint2korr(share->state.header.unique_key_parts);
  160.     if (len != MI_STATE_INFO_SIZE)
  161.     {
  162.       DBUG_PRINT("warning",
  163.  ("saved_state_info_length: %d  state_info_length: %d",
  164.   len,MI_STATE_INFO_SIZE));
  165.     }
  166.     share->state_diff_length=len-MI_STATE_INFO_SIZE;
  167.     mi_state_info_read(disk_cache, &share->state);
  168.     len= mi_uint2korr(share->state.header.base_info_length);
  169.     if (len != MI_BASE_INFO_SIZE)
  170.     {
  171.       DBUG_PRINT("warning",("saved_base_info_length: %d  base_info_length: %d",
  172.     len,MI_BASE_INFO_SIZE))
  173.     }
  174.     disk_pos=my_n_base_info_read(disk_cache+base_pos, &share->base);
  175.     share->state.state_length=base_pos;
  176.     if (!(open_flags & HA_OPEN_FOR_REPAIR) &&
  177. ((share->state.changed & STATE_CRASHED) ||
  178.  ((open_flags & HA_OPEN_ABORT_IF_CRASHED) &&
  179.   (my_disable_locking && share->state.open_count))))
  180.     {
  181.       DBUG_PRINT("error",("Table is marked as crashed"));
  182.       my_errno=((share->state.changed & STATE_CRASHED_ON_REPAIR) ?
  183. HA_ERR_CRASHED_ON_REPAIR : HA_ERR_CRASHED_ON_USAGE);
  184.       goto err;
  185.     }
  186.     /* sanity check */
  187.     if (share->base.keystart > 65535 || share->base.rec_reflength > 8)
  188.     {
  189.       my_errno=HA_ERR_CRASHED;
  190.       goto err;
  191.     }
  192.     key_parts+=fulltext_keys*FT_SEGS;
  193.     if (share->base.max_key_length > MI_MAX_KEY_BUFF || keys > MI_MAX_KEY ||
  194. key_parts >= MI_MAX_KEY * MI_MAX_KEY_SEG)
  195.     {
  196.       DBUG_PRINT("error",("Wrong key info:  Max_key_length: %d  keys: %d  key_parts: %d", share->base.max_key_length, keys, key_parts));
  197.       my_errno=HA_ERR_UNSUPPORTED;
  198.       goto err;
  199.     }
  200.     /* Correct max_file_length based on length of sizeof(off_t) */
  201.     max_data_file_length=
  202.       (share->options & (HA_OPTION_PACK_RECORD | HA_OPTION_COMPRESS_RECORD)) ?
  203.       (((ulonglong) 1 << (share->base.rec_reflength*8))-1) :
  204.       (mi_safe_mul(share->base.pack_reclength,
  205.    (ulonglong) 1 << (share->base.rec_reflength*8))-1);
  206.     max_key_file_length=
  207.       mi_safe_mul(MI_MIN_KEY_BLOCK_LENGTH,
  208.   ((ulonglong) 1 << (share->base.key_reflength*8))-1);
  209. #if SIZEOF_OFF_T == 4
  210.     set_if_smaller(max_data_file_length, INT_MAX32);
  211.     set_if_smaller(max_key_file_length, INT_MAX32);
  212. #endif
  213. #if USE_RAID && SYSTEM_SIZEOF_OFF_T == 4
  214.     set_if_smaller(max_key_file_length, INT_MAX32);
  215.     if (!share->base.raid_type)
  216.     {
  217.       set_if_smaller(max_data_file_length, INT_MAX32);
  218.     }
  219.     else
  220.     {
  221.       set_if_smaller(max_data_file_length,
  222.      (ulonglong) share->base.raid_chunks << 31);
  223.     }
  224. #elif !defined(USE_RAID)
  225.     if (share->base.raid_type)
  226.     {
  227.       DBUG_PRINT("error",("Table uses RAID but we don't have RAID support"));
  228.       my_errno=HA_ERR_UNSUPPORTED;
  229.       goto err;
  230.     }
  231. #endif
  232.     share->base.max_data_file_length=(my_off_t) max_data_file_length;
  233.     share->base.max_key_file_length=(my_off_t) max_key_file_length;
  234.     if (share->options & HA_OPTION_COMPRESS_RECORD)
  235.       share->base.max_key_length+=2; /* For safety */
  236.     if (!my_multi_malloc(MY_WME,
  237.  &share,sizeof(*share),
  238.  &share->state.rec_per_key_part,sizeof(long)*key_parts,
  239.  &share->keyinfo,keys*sizeof(MI_KEYDEF),
  240.  &share->uniqueinfo,uniques*sizeof(MI_UNIQUEDEF),
  241.  &share->keyparts,
  242.  (key_parts+unique_key_parts+keys+uniques) *
  243.  sizeof(HA_KEYSEG),
  244.  &share->rec,
  245.  (share->base.fields+1)*sizeof(MI_COLUMNDEF),
  246.  &share->blobs,sizeof(MI_BLOB)*share->base.blobs,
  247.  &share->unique_file_name,strlen(name_buff)+1,
  248.  &share->index_file_name,strlen(index_name)+1,
  249.  &share->data_file_name,strlen(data_name)+1,
  250.  &share->state.key_root,keys*sizeof(my_off_t),
  251.  &share->state.key_del,
  252.  (share->state.header.max_block_size*sizeof(my_off_t)),
  253. #ifdef THREAD
  254.  &share->key_root_lock,sizeof(rw_lock_t)*keys,
  255. #endif
  256.  NullS))
  257.       goto err;
  258.     errpos=4;
  259.     *share=share_buff;
  260.     memcpy((char*) share->state.rec_per_key_part,
  261.    (char*) rec_per_key_part, sizeof(long)*key_parts);
  262.     memcpy((char*) share->state.key_root,
  263.    (char*) key_root, sizeof(my_off_t)*keys);
  264.     memcpy((char*) share->state.key_del,
  265.    (char*) key_del, (sizeof(my_off_t) *
  266.      share->state.header.max_block_size));
  267.     strmov(share->unique_file_name, name_buff);
  268.     share->unique_name_length= strlen(name_buff);
  269.     strmov(share->index_file_name,  index_name);
  270.     strmov(share->data_file_name,   data_name);
  271.     share->blocksize=min(IO_SIZE,myisam_block_size);
  272.     {
  273.       HA_KEYSEG *pos=share->keyparts;
  274.       for (i=0 ; i < keys ; i++)
  275.       {
  276. disk_pos=mi_keydef_read(disk_pos, &share->keyinfo[i]);
  277.         disk_pos_assert(disk_pos + share->keyinfo[i].keysegs * HA_KEYSEG_SIZE,
  278.   end_pos);
  279.         if (share->keyinfo[i].key_alg == HA_KEY_ALG_RTREE)
  280.           have_rtree=1;
  281. set_if_smaller(share->blocksize,share->keyinfo[i].block_length);
  282. share->keyinfo[i].seg=pos;
  283. for (j=0 ; j < share->keyinfo[i].keysegs; j++,pos++)
  284. {
  285.   disk_pos=mi_keyseg_read(disk_pos, pos);
  286.   if (pos->type == HA_KEYTYPE_TEXT || pos->type == HA_KEYTYPE_VARTEXT)
  287.   {
  288.     if (!pos->language)
  289.       pos->charset=default_charset_info;
  290.     else if (!(pos->charset= get_charset(pos->language, MYF(MY_WME))))
  291.     {
  292.       my_errno=HA_ERR_UNKNOWN_CHARSET;
  293.       goto err;
  294.     }
  295.   }
  296. }
  297. if (share->keyinfo[i].flag & HA_SPATIAL)
  298. {
  299. #ifdef HAVE_SPATIAL
  300.   uint sp_segs=SPDIMS*2;
  301.   share->keyinfo[i].seg=pos-sp_segs;
  302.   share->keyinfo[i].keysegs--;
  303. #else
  304.   my_errno=HA_ERR_UNSUPPORTED;
  305.   goto err;
  306. #endif
  307. }
  308.         else if (share->keyinfo[i].flag & HA_FULLTEXT)
  309. {
  310.           if (!fulltext_keys)
  311.           { /* 4.0 compatibility code, to be removed in 5.0 */
  312.             share->keyinfo[i].seg=pos-FT_SEGS;
  313.             share->keyinfo[i].keysegs-=FT_SEGS;
  314.           }
  315.           else
  316.           {
  317.             uint j;
  318.             share->keyinfo[i].seg=pos;
  319.             for (j=0; j < FT_SEGS; j++)
  320.             {
  321.               *pos=ft_keysegs[j];
  322.               pos[0].language= pos[-1].language;
  323.               if (!(pos[0].charset= pos[-1].charset))
  324.               {
  325.                 my_errno=HA_ERR_CRASHED;
  326.                 goto err;
  327.               }
  328.               pos++;
  329.             }
  330.           }
  331.           if (!share->ft2_keyinfo.seg)
  332.           {
  333.             memcpy(& share->ft2_keyinfo, & share->keyinfo[i], sizeof(MI_KEYDEF));
  334.             share->ft2_keyinfo.keysegs=1;
  335.             share->ft2_keyinfo.flag=0;
  336.             share->ft2_keyinfo.keylength=
  337.             share->ft2_keyinfo.minlength=
  338.             share->ft2_keyinfo.maxlength=HA_FT_WLEN+share->base.rec_reflength;
  339.             share->ft2_keyinfo.seg=pos-1;
  340.             share->ft2_keyinfo.end=pos;
  341.             setup_key_functions(& share->ft2_keyinfo);
  342.           }
  343. }
  344.         setup_key_functions(share->keyinfo+i);
  345. share->keyinfo[i].end=pos;
  346. pos->type=HA_KEYTYPE_END; /* End */
  347. pos->length=share->base.rec_reflength;
  348. pos->null_bit=0;
  349. pos->flag=0; /* For purify */
  350. pos++;
  351.       }
  352.       for (i=0 ; i < uniques ; i++)
  353.       {
  354. disk_pos=mi_uniquedef_read(disk_pos, &share->uniqueinfo[i]);
  355.         disk_pos_assert(disk_pos + share->uniqueinfo[i].keysegs *
  356. HA_KEYSEG_SIZE, end_pos);
  357. share->uniqueinfo[i].seg=pos;
  358. for (j=0 ; j < share->uniqueinfo[i].keysegs; j++,pos++)
  359. {
  360.   disk_pos=mi_keyseg_read(disk_pos, pos);
  361.   if (pos->type == HA_KEYTYPE_TEXT || pos->type == HA_KEYTYPE_VARTEXT)
  362.   {
  363.     if (!pos->language)
  364.       pos->charset=default_charset_info;
  365.     else if (!(pos->charset= get_charset(pos->language, MYF(MY_WME))))
  366.     {
  367.       my_errno=HA_ERR_UNKNOWN_CHARSET;
  368.       goto err;
  369.     }
  370.   }
  371. }
  372. share->uniqueinfo[i].end=pos;
  373. pos->type=HA_KEYTYPE_END; /* End */
  374. pos->null_bit=0;
  375. pos->flag=0;
  376. pos++;
  377.       }
  378.     }
  379.     disk_pos_assert(disk_pos + share->base.fields *MI_COLUMNDEF_SIZE, end_pos);
  380.     for (i=j=offset=0 ; i < share->base.fields ; i++)
  381.     {
  382.       disk_pos=mi_recinfo_read(disk_pos,&share->rec[i]);
  383.       share->rec[i].pack_type=0;
  384.       share->rec[i].huff_tree=0;
  385.       share->rec[i].offset=offset;
  386.       if (share->rec[i].type == (int) FIELD_BLOB)
  387.       {
  388. share->blobs[j].pack_length=
  389.   share->rec[i].length-mi_portable_sizeof_char_ptr;;
  390. share->blobs[j].offset=offset;
  391. j++;
  392.       }
  393.       offset+=share->rec[i].length;
  394.     }
  395.     share->rec[i].type=(int) FIELD_LAST; /* End marker */
  396.     if (! lock_error)
  397.     {
  398.       VOID(my_lock(kfile,F_UNLCK,0L,F_TO_EOF,MYF(MY_SEEK_NOT_DONE)));
  399.       lock_error=1; /* Database unlocked */
  400.     }
  401.     if (mi_open_datafile(&info, share, -1))
  402.       goto err;
  403.     errpos=5;
  404.     share->kfile=kfile;
  405.     share->this_process=(ulong) getpid();
  406.     share->last_process= share->state.process;
  407.     share->base.key_parts=key_parts;
  408.     share->base.all_key_parts=key_parts+unique_key_parts;
  409.     if (!(share->last_version=share->state.version))
  410.       share->last_version=1; /* Safety */
  411.     share->rec_reflength=share->base.rec_reflength; /* May be changed */
  412.     share->base.margin_key_file_length=(share->base.max_key_file_length -
  413. (keys ? MI_INDEX_BLOCK_MARGIN *
  414.  share->blocksize * keys : 0));
  415.     share->blocksize=min(IO_SIZE,myisam_block_size);
  416.     share->data_file_type=STATIC_RECORD;
  417.     if (share->options & HA_OPTION_COMPRESS_RECORD)
  418.     {
  419.       share->data_file_type = COMPRESSED_RECORD;
  420.       share->options|= HA_OPTION_READ_ONLY_DATA;
  421.       info.s=share;
  422.       if (_mi_read_pack_info(&info,
  423.      (pbool)
  424.      test(!(share->options &
  425.     (HA_OPTION_PACK_RECORD |
  426.      HA_OPTION_TEMP_COMPRESS_RECORD)))))
  427. goto err;
  428.     }
  429.     else if (share->options & HA_OPTION_PACK_RECORD)
  430.       share->data_file_type = DYNAMIC_RECORD;
  431.     my_afree((gptr) disk_cache);
  432.     mi_setup_functions(share);
  433. #ifdef THREAD
  434.     thr_lock_init(&share->lock);
  435.     VOID(pthread_mutex_init(&share->intern_lock,MY_MUTEX_INIT_FAST));
  436.     for (i=0; i<keys; i++)
  437.       VOID(my_rwlock_init(&share->key_root_lock[i], NULL));
  438.     if (!thr_lock_inited)
  439.     {
  440.       /* Probably a single threaded program; Don't use concurrent inserts */
  441.       myisam_concurrent_insert=0;
  442.     }
  443.     else if (myisam_concurrent_insert)
  444.     {
  445.       share->concurrent_insert=
  446. ((share->options & (HA_OPTION_READ_ONLY_DATA | HA_OPTION_TMP_TABLE |
  447.    HA_OPTION_COMPRESS_RECORD |
  448.    HA_OPTION_TEMP_COMPRESS_RECORD)) ||
  449.  (open_flags & HA_OPEN_TMP_TABLE) ||
  450.  have_rtree) ? 0 : 1;
  451.       if (share->concurrent_insert)
  452.       {
  453. share->lock.get_status=mi_get_status;
  454. share->lock.copy_status=mi_copy_status;
  455. share->lock.update_status=mi_update_status;
  456. share->lock.check_status=mi_check_status;
  457.       }
  458.     }
  459. #endif
  460.   }
  461.   else
  462.   {
  463.     share= old_info->s;
  464.     if (mode == O_RDWR && share->mode == O_RDONLY)
  465.     {
  466.       my_errno=EACCES; /* Can't open in write mode */
  467.       goto err;
  468.     }
  469.     if (mi_open_datafile(&info, share, old_info->dfile))
  470.       goto err;
  471.     errpos=5;
  472.     have_rtree= old_info->rtree_recursion_state != NULL;
  473.   }
  474.   /* alloc and set up private structure parts */
  475.   if (!my_multi_malloc(MY_WME,
  476.        &m_info,sizeof(MI_INFO),
  477.        &info.blobs,sizeof(MI_BLOB)*share->base.blobs,
  478.        &info.buff,(share->base.max_key_block_length*2+
  479.    share->base.max_key_length),
  480.        &info.lastkey,share->base.max_key_length*3+1,
  481.        &info.first_mbr_key, share->base.max_key_length,
  482.        &info.filename,strlen(name)+1,
  483.        &info.rtree_recursion_state,have_rtree ? 1024 : 0,
  484.        NullS))
  485.     goto err;
  486.   errpos=6;
  487.   if (!have_rtree)
  488.     info.rtree_recursion_state= NULL;
  489.   strmov(info.filename,name);
  490.   memcpy(info.blobs,share->blobs,sizeof(MI_BLOB)*share->base.blobs);
  491.   info.lastkey2=info.lastkey+share->base.max_key_length;
  492.   info.s=share;
  493.   info.lastpos= HA_OFFSET_ERROR;
  494.   info.update= (short) (HA_STATE_NEXT_FOUND+HA_STATE_PREV_FOUND);
  495.   info.opt_flag=READ_CHECK_USED;
  496.   info.this_unique= (ulong) info.dfile; /* Uniq number in process */
  497.   if (share->data_file_type == COMPRESSED_RECORD)
  498.     info.this_unique= share->state.unique;
  499.   info.this_loop=0; /* Update counter */
  500.   info.last_unique= share->state.unique;
  501.   info.last_loop=   share->state.update_count;
  502.   if (mode == O_RDONLY)
  503.     share->options|=HA_OPTION_READ_ONLY_DATA;
  504.   info.lock_type=F_UNLCK;
  505.   info.quick_mode=0;
  506.   info.bulk_insert=0;
  507.   info.ft1_to_ft2=0;
  508.   info.errkey= -1;
  509.   info.page_changed=1;
  510.   pthread_mutex_lock(&share->intern_lock);
  511.   info.read_record=share->read_record;
  512.   share->reopen++;
  513.   share->write_flag=MYF(MY_NABP | MY_WAIT_IF_FULL);
  514.   if (share->options & HA_OPTION_READ_ONLY_DATA)
  515.   {
  516.     info.lock_type=F_RDLCK;
  517.     share->r_locks++;
  518.     share->tot_locks++;
  519.   }
  520.   if ((open_flags & HA_OPEN_TMP_TABLE) ||
  521.       (share->options & HA_OPTION_TMP_TABLE))
  522.   {
  523.     share->temporary=share->delay_key_write=1;
  524.     share->write_flag=MYF(MY_NABP);
  525.     share->w_locks++; /* We don't have to update status */
  526.     share->tot_locks++;
  527.     info.lock_type=F_WRLCK;
  528.   }
  529.   if (((open_flags & HA_OPEN_DELAY_KEY_WRITE) ||
  530.       (share->options & HA_OPTION_DELAY_KEY_WRITE)) &&
  531.       myisam_delay_key_write)
  532.     share->delay_key_write=1;
  533.   info.state= &share->state.state; /* Change global values by default */
  534.   pthread_mutex_unlock(&share->intern_lock);
  535.   /* Allocate buffer for one record */
  536.   /* prerequisites: bzero(info) && info->s=share; are met. */
  537.   if (!mi_alloc_rec_buff(&info, -1, &info.rec_buff))
  538.     goto err;
  539.   bzero(info.rec_buff, mi_get_rec_buff_len(&info, info.rec_buff));
  540.   *m_info=info;
  541. #ifdef THREAD
  542.   thr_lock_data_init(&share->lock,&m_info->lock,(void*) m_info);
  543. #endif
  544.   m_info->open_list.data=(void*) m_info;
  545.   myisam_open_list=list_add(myisam_open_list,&m_info->open_list);
  546.   pthread_mutex_unlock(&THR_LOCK_myisam);
  547.   if (myisam_log_file >= 0)
  548.   {
  549.     intern_filename(name_buff,share->index_file_name);
  550.     _myisam_log(MI_LOG_OPEN,m_info,name_buff,(uint) strlen(name_buff));
  551.   }
  552.   DBUG_RETURN(m_info);
  553. err:
  554.   save_errno=my_errno ? my_errno : HA_ERR_END_OF_FILE;
  555.   switch (errpos) {
  556.   case 6:
  557.     my_free((gptr) m_info,MYF(0));
  558.     /* fall through */
  559.   case 5:
  560.     VOID(my_close(info.dfile,MYF(0)));
  561.     if (old_info)
  562.       break; /* Don't remove open table */
  563.     /* fall through */
  564.   case 4:
  565.     my_free((gptr) share,MYF(0));
  566.     /* fall through */
  567.   case 3:
  568.     if (! lock_error)
  569.       VOID(my_lock(kfile, F_UNLCK, 0L, F_TO_EOF, MYF(MY_SEEK_NOT_DONE)));
  570.     /* fall through */
  571.   case 2:
  572.     my_afree((gptr) disk_cache);
  573.     /* fall through */
  574.   case 1:
  575.     VOID(my_close(kfile,MYF(0)));
  576.     /* fall through */
  577.   case 0:
  578.   default:
  579.     break;
  580.   }
  581.   pthread_mutex_unlock(&THR_LOCK_myisam);
  582.   my_errno=save_errno;
  583.   DBUG_RETURN (NULL);
  584. } /* mi_open */
  585. byte *mi_alloc_rec_buff(MI_INFO *info, ulong length, byte **buf)
  586. {
  587.   uint extra;
  588.   uint32 old_length;
  589.   LINT_INIT(old_length);
  590.   if (! *buf || length > (old_length=mi_get_rec_buff_len(info, *buf)))
  591.   {
  592.     byte *newptr = *buf;
  593.     /* to simplify initial init of info->rec_buf in mi_open and mi_extra */
  594.     if (length == (ulong) -1)
  595.     {
  596.       length= max(info->s->base.pack_reclength,
  597.                   info->s->base.max_key_length);
  598.       /* Avoid unnecessary realloc */
  599.       if (newptr && length == old_length)
  600. return newptr;
  601.     }
  602.     extra= ((info->s->options & HA_OPTION_PACK_RECORD) ?
  603.     ALIGN_SIZE(MI_MAX_DYN_BLOCK_HEADER)+MI_SPLIT_LENGTH+
  604.     MI_REC_BUFF_OFFSET : 0);
  605.     if (extra && newptr)
  606.       newptr-= MI_REC_BUFF_OFFSET;
  607.     if (!(newptr=(byte*) my_realloc((gptr)newptr, length+extra+8,
  608.                                      MYF(MY_ALLOW_ZERO_PTR))))
  609.       return newptr;
  610.     *((uint32 *) newptr)= (uint32) length;
  611.     *buf= newptr+(extra ?  MI_REC_BUFF_OFFSET : 0);
  612.   }
  613.   return *buf;
  614. }
  615. ulonglong mi_safe_mul(ulonglong a, ulonglong b)
  616. {
  617.   ulonglong max_val= ~ (ulonglong) 0; /* my_off_t is unsigned */
  618.   if (!a || max_val / a < b)
  619.     return max_val;
  620.   return a*b;
  621. }
  622. /* Set up functions in structs */
  623. void mi_setup_functions(register MYISAM_SHARE *share)
  624. {
  625.   if (share->options & HA_OPTION_COMPRESS_RECORD)
  626.   {
  627.     share->read_record=_mi_read_pack_record;
  628.     share->read_rnd=_mi_read_rnd_pack_record;
  629.     if (!(share->options & HA_OPTION_TEMP_COMPRESS_RECORD))
  630.       share->calc_checksum=0; /* No checksum */
  631.     else if (share->options & HA_OPTION_PACK_RECORD)
  632.       share->calc_checksum= mi_checksum;
  633.     else
  634.       share->calc_checksum= mi_static_checksum;
  635.   }
  636.   else if (share->options & HA_OPTION_PACK_RECORD)
  637.   {
  638.     share->read_record=_mi_read_dynamic_record;
  639.     share->read_rnd=_mi_read_rnd_dynamic_record;
  640.     share->delete_record=_mi_delete_dynamic_record;
  641.     share->compare_record=_mi_cmp_dynamic_record;
  642.     share->compare_unique=_mi_cmp_dynamic_unique;
  643.     share->calc_checksum= mi_checksum;
  644.     /* add bits used to pack data to pack_reclength for faster allocation */
  645.     share->base.pack_reclength+= share->base.pack_bits;
  646.     if (share->base.blobs)
  647.     {
  648.       share->update_record=_mi_update_blob_record;
  649.       share->write_record=_mi_write_blob_record;
  650.     }
  651.     else
  652.     {
  653.       share->write_record=_mi_write_dynamic_record;
  654.       share->update_record=_mi_update_dynamic_record;
  655.     }
  656.   }
  657.   else
  658.   {
  659.     share->read_record=_mi_read_static_record;
  660.     share->read_rnd=_mi_read_rnd_static_record;
  661.     share->delete_record=_mi_delete_static_record;
  662.     share->compare_record=_mi_cmp_static_record;
  663.     share->update_record=_mi_update_static_record;
  664.     share->write_record=_mi_write_static_record;
  665.     share->compare_unique=_mi_cmp_static_unique;
  666.     share->calc_checksum= mi_static_checksum;
  667.   }
  668.   if (!(share->options & HA_OPTION_CHECKSUM))
  669.     share->calc_checksum=0;
  670.   return;
  671. }
  672. static void setup_key_functions(register MI_KEYDEF *keyinfo)
  673. {
  674.   if (keyinfo->key_alg == HA_KEY_ALG_RTREE)
  675.   {
  676. #ifdef HAVE_RTREE_KEYS
  677.     keyinfo->ck_insert = rtree_insert;
  678.     keyinfo->ck_delete = rtree_delete;
  679. #else
  680.     DBUG_ASSERT(0); /* mi_open should check it never happens */
  681. #endif
  682.   }
  683.   else
  684.   {
  685.     keyinfo->ck_insert = _mi_ck_write;
  686.     keyinfo->ck_delete = _mi_ck_delete;
  687.   }
  688.   if (keyinfo->flag & HA_BINARY_PACK_KEY)
  689.   { /* Simple prefix compression */
  690.     keyinfo->bin_search=_mi_seq_search;
  691.     keyinfo->get_key=_mi_get_binary_pack_key;
  692.     keyinfo->pack_key=_mi_calc_bin_pack_key_length;
  693.     keyinfo->store_key=_mi_store_bin_pack_key;
  694.   }
  695.   else if (keyinfo->flag & HA_VAR_LENGTH_KEY)
  696.   {
  697.     keyinfo->get_key= _mi_get_pack_key;
  698.     if (keyinfo->seg[0].flag & HA_PACK_KEY)
  699.     { /* Prefix compression */
  700.       if (!keyinfo->seg->charset || use_strnxfrm(keyinfo->seg->charset) ||
  701.           (keyinfo->seg->flag & HA_NULL_PART))
  702.         keyinfo->bin_search=_mi_seq_search;
  703.       else
  704.         keyinfo->bin_search=_mi_prefix_search;
  705.       keyinfo->pack_key=_mi_calc_var_pack_key_length;
  706.       keyinfo->store_key=_mi_store_var_pack_key;
  707.     }
  708.     else
  709.     {
  710.       keyinfo->bin_search=_mi_seq_search;
  711.       keyinfo->pack_key=_mi_calc_var_key_length; /* Variable length key */
  712.       keyinfo->store_key=_mi_store_static_key;
  713.     }
  714.   }
  715.   else
  716.   {
  717.     keyinfo->bin_search=_mi_bin_search;
  718.     keyinfo->get_key=_mi_get_static_key;
  719.     keyinfo->pack_key=_mi_calc_static_key_length;
  720.     keyinfo->store_key=_mi_store_static_key;
  721.   }
  722.   return;
  723. }
  724. /*
  725.    Function to save and store the header in the index file (.MYI)
  726. */
  727. uint mi_state_info_write(File file, MI_STATE_INFO *state, uint pWrite)
  728. {
  729.   uchar  buff[MI_STATE_INFO_SIZE + MI_STATE_EXTRA_SIZE];
  730.   uchar *ptr=buff;
  731.   uint i, keys= (uint) state->header.keys,
  732. key_blocks=state->header.max_block_size;
  733.   DBUG_ENTER("mi_state_info_write");
  734.   memcpy_fixed(ptr,&state->header,sizeof(state->header));
  735.   ptr+=sizeof(state->header);
  736.   /* open_count must be first because of _mi_mark_file_changed ! */
  737.   mi_int2store(ptr,state->open_count); ptr +=2;
  738.   *ptr++= (uchar)state->changed; *ptr++= state->sortkey;
  739.   mi_rowstore(ptr,state->state.records); ptr +=8;
  740.   mi_rowstore(ptr,state->state.del); ptr +=8;
  741.   mi_rowstore(ptr,state->split); ptr +=8;
  742.   mi_sizestore(ptr,state->dellink); ptr +=8;
  743.   mi_sizestore(ptr,state->state.key_file_length); ptr +=8;
  744.   mi_sizestore(ptr,state->state.data_file_length); ptr +=8;
  745.   mi_sizestore(ptr,state->state.empty); ptr +=8;
  746.   mi_sizestore(ptr,state->state.key_empty); ptr +=8;
  747.   mi_int8store(ptr,state->auto_increment); ptr +=8;
  748.   mi_int8store(ptr,(ulonglong) state->checksum);ptr +=8;
  749.   mi_int4store(ptr,state->process); ptr +=4;
  750.   mi_int4store(ptr,state->unique); ptr +=4;
  751.   mi_int4store(ptr,state->status); ptr +=4;
  752.   mi_int4store(ptr,state->update_count); ptr +=4;
  753.   ptr+=state->state_diff_length;
  754.   for (i=0; i < keys; i++)
  755.   {
  756.     mi_sizestore(ptr,state->key_root[i]); ptr +=8;
  757.   }
  758.   for (i=0; i < key_blocks; i++)
  759.   {
  760.     mi_sizestore(ptr,state->key_del[i]); ptr +=8;
  761.   }
  762.   if (pWrite & 2) /* From isamchk */
  763.   {
  764.     uint key_parts= mi_uint2korr(state->header.key_parts);
  765.     mi_int4store(ptr,state->sec_index_changed); ptr +=4;
  766.     mi_int4store(ptr,state->sec_index_used); ptr +=4;
  767.     mi_int4store(ptr,state->version); ptr +=4;
  768.     mi_int8store(ptr,state->key_map); ptr +=8;
  769.     mi_int8store(ptr,(ulonglong) state->create_time); ptr +=8;
  770.     mi_int8store(ptr,(ulonglong) state->recover_time); ptr +=8;
  771.     mi_int8store(ptr,(ulonglong) state->check_time); ptr +=8;
  772.     mi_sizestore(ptr,state->rec_per_key_rows); ptr+=8;
  773.     for (i=0 ; i < key_parts ; i++)
  774.     {
  775.       mi_int4store(ptr,state->rec_per_key_part[i]);  ptr+=4;
  776.     }
  777.   }
  778.   if (pWrite & 1)
  779.     DBUG_RETURN(my_pwrite(file,(char*) buff, (uint) (ptr-buff), 0L,
  780.   MYF(MY_NABP | MY_THREADSAFE)));
  781.   DBUG_RETURN(my_write(file,  (char*) buff, (uint) (ptr-buff),
  782.        MYF(MY_NABP)));
  783. }
  784. char *mi_state_info_read(char *ptr, MI_STATE_INFO *state)
  785. {
  786.   uint i,keys,key_parts,key_blocks;
  787.   memcpy_fixed(&state->header,ptr, sizeof(state->header));
  788.   ptr +=sizeof(state->header);
  789.   keys=(uint) state->header.keys;
  790.   key_parts=mi_uint2korr(state->header.key_parts);
  791.   key_blocks=state->header.max_block_size;
  792.   state->open_count = mi_uint2korr(ptr); ptr +=2;
  793.   state->changed= (bool) *ptr++;
  794.   state->sortkey = (uint) *ptr++;
  795.   state->state.records= mi_rowkorr(ptr); ptr +=8;
  796.   state->state.del = mi_rowkorr(ptr); ptr +=8;
  797.   state->split = mi_rowkorr(ptr); ptr +=8;
  798.   state->dellink= mi_sizekorr(ptr); ptr +=8;
  799.   state->state.key_file_length = mi_sizekorr(ptr); ptr +=8;
  800.   state->state.data_file_length= mi_sizekorr(ptr); ptr +=8;
  801.   state->state.empty = mi_sizekorr(ptr); ptr +=8;
  802.   state->state.key_empty= mi_sizekorr(ptr); ptr +=8;
  803.   state->auto_increment=mi_uint8korr(ptr); ptr +=8;
  804.   state->checksum=(ha_checksum) mi_uint8korr(ptr); ptr +=8;
  805.   state->process= mi_uint4korr(ptr); ptr +=4;
  806.   state->unique = mi_uint4korr(ptr); ptr +=4;
  807.   state->status = mi_uint4korr(ptr); ptr +=4;
  808.   state->update_count=mi_uint4korr(ptr); ptr +=4;
  809.   ptr+= state->state_diff_length;
  810.   for (i=0; i < keys; i++)
  811.   {
  812.     state->key_root[i]= mi_sizekorr(ptr); ptr +=8;
  813.   }
  814.   for (i=0; i < key_blocks; i++)
  815.   {
  816.     state->key_del[i] = mi_sizekorr(ptr); ptr +=8;
  817.   }
  818.   state->sec_index_changed = mi_uint4korr(ptr); ptr +=4;
  819.   state->sec_index_used =    mi_uint4korr(ptr); ptr +=4;
  820.   state->version     = mi_uint4korr(ptr); ptr +=4;
  821.   state->key_map     = mi_uint8korr(ptr); ptr +=8;
  822.   state->create_time = (time_t) mi_sizekorr(ptr); ptr +=8;
  823.   state->recover_time =(time_t) mi_sizekorr(ptr); ptr +=8;
  824.   state->check_time =  (time_t) mi_sizekorr(ptr); ptr +=8;
  825.   state->rec_per_key_rows=mi_sizekorr(ptr); ptr +=8;
  826.   for (i=0 ; i < key_parts ; i++)
  827.   {
  828.     state->rec_per_key_part[i]= mi_uint4korr(ptr); ptr+=4;
  829.   }
  830.   return ptr;
  831. }
  832. uint mi_state_info_read_dsk(File file, MI_STATE_INFO *state, my_bool pRead)
  833. {
  834.   char buff[MI_STATE_INFO_SIZE + MI_STATE_EXTRA_SIZE];
  835.   if (!myisam_single_user)
  836.   {
  837.     if (pRead)
  838.     {
  839.       if (my_pread(file, buff, state->state_length,0L, MYF(MY_NABP)))
  840. return (MY_FILE_ERROR);
  841.     }
  842.     else if (my_read(file, buff, state->state_length,MYF(MY_NABP)))
  843.       return (MY_FILE_ERROR);
  844.     mi_state_info_read(buff, state);
  845.   }
  846.   return 0;
  847. }
  848. /****************************************************************************
  849. **  store and read of MI_BASE_INFO
  850. ****************************************************************************/
  851. uint mi_base_info_write(File file, MI_BASE_INFO *base)
  852. {
  853.   uchar buff[MI_BASE_INFO_SIZE], *ptr=buff;
  854.   mi_sizestore(ptr,base->keystart); ptr +=8;
  855.   mi_sizestore(ptr,base->max_data_file_length); ptr +=8;
  856.   mi_sizestore(ptr,base->max_key_file_length); ptr +=8;
  857.   mi_rowstore(ptr,base->records); ptr +=8;
  858.   mi_rowstore(ptr,base->reloc); ptr +=8;
  859.   mi_int4store(ptr,base->mean_row_length); ptr +=4;
  860.   mi_int4store(ptr,base->reclength); ptr +=4;
  861.   mi_int4store(ptr,base->pack_reclength); ptr +=4;
  862.   mi_int4store(ptr,base->min_pack_length); ptr +=4;
  863.   mi_int4store(ptr,base->max_pack_length); ptr +=4;
  864.   mi_int4store(ptr,base->min_block_length); ptr +=4;
  865.   mi_int4store(ptr,base->fields); ptr +=4;
  866.   mi_int4store(ptr,base->pack_fields); ptr +=4;
  867.   *ptr++=base->rec_reflength;
  868.   *ptr++=base->key_reflength;
  869.   *ptr++=base->keys;
  870.   *ptr++=base->auto_key;
  871.   mi_int2store(ptr,base->pack_bits); ptr +=2;
  872.   mi_int2store(ptr,base->blobs); ptr +=2;
  873.   mi_int2store(ptr,base->max_key_block_length); ptr +=2;
  874.   mi_int2store(ptr,base->max_key_length); ptr +=2;
  875.   mi_int2store(ptr,base->extra_alloc_bytes); ptr +=2;
  876.   *ptr++= base->extra_alloc_procent;
  877.   *ptr++= base->raid_type;
  878.   mi_int2store(ptr,base->raid_chunks); ptr +=2;
  879.   mi_int4store(ptr,base->raid_chunksize); ptr +=4;
  880.   bzero(ptr,6); ptr +=6; /* extra */
  881.   return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP));
  882. }
  883. char *my_n_base_info_read(char *ptr, MI_BASE_INFO *base)
  884. {
  885.   base->keystart = mi_sizekorr(ptr); ptr +=8;
  886.   base->max_data_file_length = mi_sizekorr(ptr); ptr +=8;
  887.   base->max_key_file_length = mi_sizekorr(ptr); ptr +=8;
  888.   base->records =  (ha_rows) mi_sizekorr(ptr); ptr +=8;
  889.   base->reloc = (ha_rows) mi_sizekorr(ptr); ptr +=8;
  890.   base->mean_row_length = mi_uint4korr(ptr); ptr +=4;
  891.   base->reclength = mi_uint4korr(ptr); ptr +=4;
  892.   base->pack_reclength = mi_uint4korr(ptr); ptr +=4;
  893.   base->min_pack_length = mi_uint4korr(ptr); ptr +=4;
  894.   base->max_pack_length = mi_uint4korr(ptr); ptr +=4;
  895.   base->min_block_length = mi_uint4korr(ptr); ptr +=4;
  896.   base->fields = mi_uint4korr(ptr); ptr +=4;
  897.   base->pack_fields = mi_uint4korr(ptr); ptr +=4;
  898.   base->rec_reflength = *ptr++;
  899.   base->key_reflength = *ptr++;
  900.   base->keys= *ptr++;
  901.   base->auto_key= *ptr++;
  902.   base->pack_bits = mi_uint2korr(ptr); ptr +=2;
  903.   base->blobs = mi_uint2korr(ptr); ptr +=2;
  904.   base->max_key_block_length= mi_uint2korr(ptr); ptr +=2;
  905.   base->max_key_length = mi_uint2korr(ptr); ptr +=2;
  906.   base->extra_alloc_bytes = mi_uint2korr(ptr); ptr +=2;
  907.   base->extra_alloc_procent = *ptr++;
  908.   base->raid_type= *ptr++;
  909.   base->raid_chunks= mi_uint2korr(ptr); ptr +=2;
  910.   base->raid_chunksize= mi_uint4korr(ptr); ptr +=4;
  911.   /* TO BE REMOVED: Fix for old RAID files */
  912.   if (base->raid_type == 0)
  913.   {
  914.     base->raid_chunks=0;
  915.     base->raid_chunksize=0;
  916.   }
  917.   ptr+=6;
  918.   return ptr;
  919. }
  920. /*--------------------------------------------------------------------------
  921.   mi_keydef
  922. ---------------------------------------------------------------------------*/
  923. uint mi_keydef_write(File file, MI_KEYDEF *keydef)
  924. {
  925.   uchar buff[MI_KEYDEF_SIZE];
  926.   uchar *ptr=buff;
  927.   *ptr++ = (uchar) keydef->keysegs;
  928.   *ptr++ = keydef->key_alg; /* Rtree or Btree */
  929.   mi_int2store(ptr,keydef->flag); ptr +=2;
  930.   mi_int2store(ptr,keydef->block_length); ptr +=2;
  931.   mi_int2store(ptr,keydef->keylength); ptr +=2;
  932.   mi_int2store(ptr,keydef->minlength); ptr +=2;
  933.   mi_int2store(ptr,keydef->maxlength); ptr +=2;
  934.   return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP));
  935. }
  936. char *mi_keydef_read(char *ptr, MI_KEYDEF *keydef)
  937. {
  938.    keydef->keysegs = (uint) *ptr++;
  939.    keydef->key_alg = *ptr++; /* Rtree or Btree */
  940.    keydef->flag = mi_uint2korr(ptr); ptr +=2;
  941.    keydef->block_length = mi_uint2korr(ptr); ptr +=2;
  942.    keydef->keylength = mi_uint2korr(ptr); ptr +=2;
  943.    keydef->minlength = mi_uint2korr(ptr); ptr +=2;
  944.    keydef->maxlength = mi_uint2korr(ptr); ptr +=2;
  945.    keydef->block_size = keydef->block_length/MI_MIN_KEY_BLOCK_LENGTH-1;
  946.    keydef->underflow_block_length=keydef->block_length/3;
  947.    keydef->version = 0; /* Not saved */
  948.    return ptr;
  949. }
  950. /***************************************************************************
  951. **  mi_keyseg
  952. ***************************************************************************/
  953. int mi_keyseg_write(File file, const HA_KEYSEG *keyseg)
  954. {
  955.   uchar buff[HA_KEYSEG_SIZE];
  956.   uchar *ptr=buff;
  957.   *ptr++ =keyseg->type;
  958.   *ptr++ =keyseg->language;
  959.   *ptr++ =keyseg->null_bit;
  960.   *ptr++ =keyseg->bit_start;
  961.   *ptr++ =keyseg->bit_end;
  962.   *ptr++ =0; /* Not used */
  963.   mi_int2store(ptr,keyseg->flag); ptr+=2;
  964.   mi_int2store(ptr,keyseg->length); ptr+=2;
  965.   mi_int4store(ptr,keyseg->start); ptr+=4;
  966.   mi_int4store(ptr,keyseg->null_pos); ptr+=4;
  967.   return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP));
  968. }
  969. char *mi_keyseg_read(char *ptr, HA_KEYSEG *keyseg)
  970. {
  971.    keyseg->type = *ptr++;
  972.    keyseg->language = *ptr++;
  973.    keyseg->null_bit = *ptr++;
  974.    keyseg->bit_start = *ptr++;
  975.    keyseg->bit_end = *ptr++;
  976.    ptr++;
  977.    keyseg->flag = mi_uint2korr(ptr);  ptr +=2;
  978.    keyseg->length = mi_uint2korr(ptr);  ptr +=2;
  979.    keyseg->start = mi_uint4korr(ptr);  ptr +=4;
  980.    keyseg->null_pos = mi_uint4korr(ptr);  ptr +=4;
  981.    keyseg->charset=0; /* Will be filled in later */
  982.    return ptr;
  983. }
  984. /*--------------------------------------------------------------------------
  985.   mi_uniquedef
  986. ---------------------------------------------------------------------------*/
  987. uint mi_uniquedef_write(File file, MI_UNIQUEDEF *def)
  988. {
  989.   uchar buff[MI_UNIQUEDEF_SIZE];
  990.   uchar *ptr=buff;
  991.   mi_int2store(ptr,def->keysegs); ptr+=2;
  992.   *ptr++=  (uchar) def->key;
  993.   *ptr++ = (uchar) def->null_are_equal;
  994.   return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP));
  995. }
  996. char *mi_uniquedef_read(char *ptr, MI_UNIQUEDEF *def)
  997. {
  998.    def->keysegs = mi_uint2korr(ptr);
  999.    def->key = ptr[2];
  1000.    def->null_are_equal=ptr[3];
  1001.    return ptr+4; /* 1 extra byte */
  1002. }
  1003. /***************************************************************************
  1004. **  MI_COLUMNDEF
  1005. ***************************************************************************/
  1006. uint mi_recinfo_write(File file, MI_COLUMNDEF *recinfo)
  1007. {
  1008.   uchar buff[MI_COLUMNDEF_SIZE];
  1009.   uchar *ptr=buff;
  1010.   mi_int2store(ptr,recinfo->type); ptr +=2;
  1011.   mi_int2store(ptr,recinfo->length); ptr +=2;
  1012.   *ptr++ = recinfo->null_bit;
  1013.   mi_int2store(ptr,recinfo->null_pos); ptr+= 2;
  1014.   return my_write(file,(char*) buff, (uint) (ptr-buff), MYF(MY_NABP));
  1015. }
  1016. char *mi_recinfo_read(char *ptr, MI_COLUMNDEF *recinfo)
  1017. {
  1018.    recinfo->type=  mi_sint2korr(ptr); ptr +=2;
  1019.    recinfo->length=mi_uint2korr(ptr); ptr +=2;
  1020.    recinfo->null_bit= (uint8) *ptr++;
  1021.    recinfo->null_pos=mi_uint2korr(ptr); ptr +=2;
  1022.    return ptr;
  1023. }
  1024. /**************************************************************************
  1025. Open data file with or without RAID
  1026. We can't use dup() here as the data file descriptors need to have different
  1027. active seek-positions.
  1028. The argument file_to_dup is here for the future if there would on some OS
  1029. exist a dup()-like call that would give us two different file descriptors.
  1030. *************************************************************************/
  1031. int mi_open_datafile(MI_INFO *info, MYISAM_SHARE *share, File file_to_dup __attribute__((unused)))
  1032. {
  1033. #ifdef USE_RAID
  1034.   if (share->base.raid_type)
  1035.   {
  1036.     info->dfile=my_raid_open(share->data_file_name,
  1037.      share->mode | O_SHARE,
  1038.      share->base.raid_type,
  1039.      share->base.raid_chunks,
  1040.      share->base.raid_chunksize,
  1041.      MYF(MY_WME | MY_RAID));
  1042.   }
  1043.   else
  1044. #endif
  1045.     info->dfile=my_open(share->data_file_name, share->mode | O_SHARE,
  1046. MYF(MY_WME));
  1047.   return info->dfile >= 0 ? 0 : 1;
  1048. }
  1049. int mi_open_keyfile(MYISAM_SHARE *share)
  1050. {
  1051.   if ((share->kfile=my_open(share->unique_file_name, share->mode | O_SHARE,
  1052.     MYF(MY_WME))) < 0)
  1053.     return 1;
  1054.   return 0;
  1055. }
  1056. /*
  1057.   Disable all indexes.
  1058.   SYNOPSIS
  1059.     mi_disable_indexes()
  1060.     info        A pointer to the MyISAM storage engine MI_INFO struct.
  1061.   DESCRIPTION
  1062.     Disable all indexes.
  1063.   RETURN
  1064.     0  ok
  1065. */
  1066. int mi_disable_indexes(MI_INFO *info)
  1067. {
  1068.   MYISAM_SHARE *share= info->s;
  1069.   share->state.key_map= 0;
  1070.   return 0;
  1071. }
  1072. /*
  1073.   Enable all indexes
  1074.   SYNOPSIS
  1075.     mi_enable_indexes()
  1076.     info        A pointer to the MyISAM storage engine MI_INFO struct.
  1077.   DESCRIPTION
  1078.     Enable all indexes. The indexes might have been disabled
  1079.     by mi_disable_index() before.
  1080.     The function works only if both data and indexes are empty,
  1081.     otherwise a repair is required.
  1082.     To be sure, call handler::delete_all_rows() before.
  1083.   RETURN
  1084.     0  ok
  1085.     HA_ERR_CRASHED data or index is non-empty.
  1086. */
  1087. int mi_enable_indexes(MI_INFO *info)
  1088. {
  1089.   int error= 0;
  1090.   MYISAM_SHARE *share= info->s;
  1091.   if (share->state.state.data_file_length ||
  1092.       (share->state.state.key_file_length != share->base.keystart))
  1093.     error= HA_ERR_CRASHED;
  1094.   else
  1095.     share->state.key_map= ((ulonglong) 1L << share->base.keys) - 1;
  1096.   return error;
  1097. }
  1098. /*
  1099.   Test if indexes are disabled.
  1100.   SYNOPSIS
  1101.     mi_indexes_are_disabled()
  1102.     info        A pointer to the MyISAM storage engine MI_INFO struct.
  1103.   DESCRIPTION
  1104.     Test if indexes are disabled.
  1105.   RETURN
  1106.     0  indexes are not disabled
  1107.     1  all indexes are disabled
  1108.    [2  non-unique indexes are disabled - NOT YET IMPLEMENTED]
  1109. */
  1110. int mi_indexes_are_disabled(MI_INFO *info)
  1111. {
  1112.   MYISAM_SHARE *share= info->s;
  1113.   return (! share->state.key_map && share->base.keys);
  1114. }