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

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 whats in isam.log */
  17. #ifndef USE_MY_FUNC
  18. #define USE_MY_FUNC
  19. #endif
  20. #include "isamdef.h"
  21. #include <my_tree.h>
  22. #include <stdarg.h>
  23. #ifdef HAVE_GETRUSAGE
  24. #include <sys/resource.h>
  25. #endif
  26. #define FILENAME(A) (A ? A->show_name : "Unknown")
  27. struct file_info {
  28.   long process;
  29.   int  filenr,id;
  30.   my_string name,show_name,record;
  31.   N_INFO *isam;
  32.   bool closed,used;
  33.   ulong accessed;
  34. };
  35. struct test_if_open_param {
  36.   my_string name;
  37.   int max_id;
  38. };
  39. struct st_access_param
  40. {
  41.   ulong min_accessed;
  42.   struct file_info *found;
  43. };
  44. #define NO_FILEPOS (ulong) ~0L
  45. extern int main(int argc,char * *argv);
  46. static void get_options(int *argc,char ***argv);
  47. static int examine_log(my_string file_name,char **table_names);
  48. static int read_string(IO_CACHE *file,gptr *to,uint length);
  49. static int file_info_compare(void *a,void *b);
  50. static int test_if_open(struct file_info *key,element_count count,
  51. struct test_if_open_param *param);
  52. static void fix_blob_pointers(N_INFO *isam,byte *record);
  53. static uint set_maximum_open_files(uint);
  54. static int test_when_accessed(struct file_info *key,element_count count,
  55.       struct st_access_param *access_param);
  56. static void file_info_free(struct file_info *info);
  57. static int close_some_file(TREE *tree);
  58. static int reopen_closed_file(TREE *tree,struct file_info *file_info);
  59. static int find_record_with_key(struct file_info *file_info,byte *record);
  60. static void printf_log(const char *str,...);
  61. static bool cmp_filename(struct file_info *file_info,my_string name);
  62. static uint verbose=0,update=0,test_info=0,max_files=0,re_open_count=0,
  63.   recover=0,prefix_remove=0;
  64. static my_string log_filename=0,filepath=0,write_filename=0,record_pos_file=0;
  65. static ulong com_count[10][3],number_of_commands=(ulong) ~0L,start_offset=0,
  66.      record_pos= NO_FILEPOS,isamlog_filepos,isamlog_process;
  67. static const char *command_name[]=
  68. {"open","write","update","delete","close","extra","lock","re-open",NullS};
  69. int main(argc,argv)
  70. int argc;
  71. char **argv;
  72. {
  73.   int error,i,first;
  74.   ulong total_count,total_error,total_recover;
  75.   MY_INIT(argv[0]);
  76.   log_filename=nisam_log_filename;
  77.   get_options(&argc,&argv);
  78.  /* Nr of isam-files */
  79.   max_files=(set_maximum_open_files(min(max_files,8))-6)/2;
  80.   if (update)
  81.     printf("Trying to %s isamfiles according to log '%s'n",
  82.    (recover ? "recover" : "update"),log_filename);
  83.   error= examine_log(log_filename,argv);
  84.   if (update && ! error)
  85.     puts("isamfile:s updated successfully");
  86.   total_count=total_error=total_recover=0;
  87.   for (i=first=0 ; command_name[i] ; i++)
  88.   {
  89.     if (com_count[i][0])
  90.     {
  91.       if (!first++)
  92.       {
  93. if (verbose || update)
  94.   puts("");
  95. puts("Commands   Used count    Errors   Recover errors");
  96.       }
  97.       printf("%-12s%9ld%10ld%17ldn",command_name[i],com_count[i][0],
  98.      com_count[i][1],com_count[i][2]);
  99.       total_count+=com_count[i][0];
  100.       total_error+=com_count[i][1];
  101.       total_recover+=com_count[i][2];
  102.     }
  103.   }
  104.   if (total_count)
  105.     printf("%-12s%9ld%10ld%17ldn","Total",total_count,total_error,
  106.    total_recover);
  107.   if (re_open_count)
  108.     printf("Had to do %d re-open because of too few possibly open filesn",
  109.    re_open_count);
  110.   VOID(nisam_panic(HA_PANIC_CLOSE));
  111.   my_end(test_info ? MY_CHECK_ERROR | MY_GIVE_INFO : MY_CHECK_ERROR);
  112.   exit(error);
  113.   return 0; /* No compiler warning */
  114. } /* main */
  115. static void get_options(argc,argv)
  116. register int *argc;
  117. register char ***argv;
  118. {
  119.   int help,version;
  120.   const char *usage;
  121.   char *pos, option;
  122.   help=0;
  123.   usage="Usage: %s [-?iruvIV] [-c #] [-f #] [-F filepath/] [-o #] [-R file recordpos] [-w write_file] [log-filename [table ...]] n";
  124.   pos= (char*) "";
  125.   while (--*argc > 0 && *(pos = *(++*argv)) == '-' ) {
  126.     while (*++pos)
  127.     {
  128.       version=0;
  129.       switch((option=*pos)) {
  130.       case '#':
  131. DBUG_PUSH (++pos);
  132. pos= (char*) " "; /* Skipp rest of arg */
  133. break;
  134.       case 'c':
  135. if (! *++pos)
  136. {
  137.   if (!--*argc)
  138.     goto err;
  139.   else
  140.     pos= *(++*argv);
  141. }
  142. number_of_commands=(ulong) atol(pos);
  143. pos= (char*) " ";
  144. break;
  145.       case 'u':
  146. update=1;
  147. break;
  148.       case 'f':
  149. if (! *++pos)
  150. {
  151.   if (!--*argc)
  152.     goto err;
  153.   else
  154.     pos= *(++*argv);
  155. }
  156. max_files=(uint) atoi(pos);
  157. pos= (char*) " ";
  158. break;
  159.       case 'i':
  160. test_info=1;
  161. break;
  162.       case 'o':
  163. if (! *++pos)
  164. {
  165.   if (!--*argc)
  166.     goto err;
  167.   else
  168.     pos= *(++*argv);
  169. }
  170. start_offset=(ulong) atol(pos);
  171. pos= (char*) " ";
  172. break;
  173.       case 'p':
  174. if (! *++pos)
  175. {
  176.   if (!--*argc)
  177.     goto err;
  178.   else
  179.     pos= *(++*argv);
  180. }
  181. prefix_remove=atoi(pos);
  182. break;
  183.       case 'r':
  184. update=1;
  185. recover++;
  186. break;
  187.       case 'R':
  188. if (! *++pos)
  189. {
  190.   if (!--*argc)
  191.     goto err;
  192.   else
  193.     pos= *(++*argv);
  194. }
  195. record_pos_file=pos;
  196. if (!--*argc)
  197.   goto err;
  198. record_pos=(ulong) atol(*(++*argv));
  199. pos= (char*) " ";
  200. break;
  201.       case 'v':
  202. verbose++;
  203. break;
  204.       case 'w':
  205. if (! *++pos)
  206. {
  207.   if (!--*argc)
  208.     goto err;
  209.   else
  210.     pos= *(++*argv);
  211. }
  212. write_filename=pos;
  213. pos= (char*) " ";
  214. break;
  215.       case 'F':
  216. if (! *++pos)
  217. {
  218.   if (!--*argc)
  219.     goto err;
  220.   else
  221.     pos= *(++*argv);
  222. }
  223. filepath=pos;
  224. pos= (char*) " ";
  225. break;
  226.       case 'V':
  227. version=1;
  228. /* Fall through */
  229.       case 'I':
  230.       case '?':
  231. printf("%s  Ver 3.2 for %s at %sn",my_progname,SYSTEM_TYPE,
  232.        MACHINE_TYPE);
  233. puts("TCX Datakonsult AB, by Monty, for your professional usen");
  234. if (version)
  235.   break;
  236. puts("Write info about whats in a nisam log file.");
  237. printf("If no file name is given %s is usedn",log_filename);
  238. puts("");
  239. printf(usage,my_progname);
  240. puts("");
  241. puts("Options: -? or -I "Info"     -V "version"   -c "do only # commands"");
  242. puts("         -f "max open files" -F "filepath"  -i "extra info"");
  243. puts("         -o "offset"         -p # "remove # components from path"");
  244. puts("         -r "recover"        -R "file recordposition"");
  245. puts("         -u "update"         -v "verbose"   -w "write file"");
  246. puts("nOne can give a second and a third '-v' for more verbose.");
  247. puts("Normaly one does a update (-u).");
  248. puts("If a recover is done all writes and all possibly updates and deletes is donenand errors are only counted.");
  249. puts("If one gives table names as arguments only these tables will be updatedn");
  250. help=1;
  251. break;
  252.       default:
  253. printf("illegal option: "-%c"n",*pos);
  254. break;
  255.       }
  256.     }
  257.   }
  258.   if (! *argc)
  259.   {
  260.     if (help)
  261.     exit(0);
  262.     (*argv)++;
  263.   }
  264.   if (*argc >= 1)
  265.   {
  266.     log_filename=pos;
  267.     (*argc)--;
  268.     (*argv)++;
  269.   }
  270.   return;
  271.  err:
  272.   VOID(fprintf(stderr,"option "%c" used without or with wrong argumentn",
  273.        option));
  274.   exit(1);
  275. }
  276. static int examine_log(my_string file_name, char **table_names)
  277. {
  278.   uint command,result,files_open;
  279.   ulong access_time,length;
  280.   uint32 filepos;
  281.   int lock_command,ni_result;
  282.   char isam_file_name[FN_REFLEN];
  283.   uchar head[20];
  284.   gptr buff;
  285.   struct test_if_open_param open_param;
  286.   IO_CACHE cache;
  287.   File file;
  288.   FILE *write_file;
  289.   enum ha_extra_function extra_command;
  290.   TREE tree;
  291.   struct file_info file_info,*curr_file_info;
  292.   DBUG_ENTER("examine_log");
  293.   if ((file=my_open(file_name,O_RDONLY,MYF(MY_WME))) < 0)
  294.     DBUG_RETURN(1);
  295.   write_file=0;
  296.   if (write_filename)
  297.   {
  298.     if (!(write_file=my_fopen(write_filename,O_WRONLY,MYF(MY_WME))))
  299.     {
  300.       my_close(file,MYF(0));
  301.       DBUG_RETURN(1);
  302.     }
  303.   }
  304.   init_io_cache(&cache,file,0,READ_CACHE,start_offset,0,MYF(0));
  305.   bzero((gptr) com_count,sizeof(com_count));
  306.   init_tree(&tree,0,sizeof(file_info),(qsort_cmp) file_info_compare,1,
  307.     (void(*)(void*)) file_info_free);
  308.   VOID(init_key_cache(KEY_CACHE_SIZE,(uint) (10*4*(IO_SIZE+MALLOC_OVERHEAD))));
  309.   files_open=0; access_time=0;
  310.   while (access_time++ != number_of_commands &&
  311.  !my_b_read(&cache,(byte*) head,9))
  312.   {
  313.     isamlog_filepos=my_b_tell(&cache)-9L;
  314.     file_info.filenr=uint2korr(head+1);
  315.     isamlog_process=file_info.process=(long) uint4korr(head+3);
  316.     result=uint2korr(head+7);
  317.     if ((curr_file_info=(struct file_info*) tree_search(&tree,&file_info)))
  318.     {
  319.       curr_file_info->accessed=access_time;
  320.       if (update && curr_file_info->used && curr_file_info->closed)
  321.       {
  322. if (reopen_closed_file(&tree,curr_file_info))
  323. {
  324.   command=sizeof(com_count)/sizeof(com_count[0][0])/3;
  325.   result=0;
  326.   goto com_err;
  327. }
  328.       }
  329.     }
  330.     command=(uint) head[0];
  331.     if (command < sizeof(com_count)/sizeof(com_count[0][0])/3 &&
  332. (!curr_file_info || curr_file_info->used))
  333.     {
  334.       com_count[command][0]++;
  335.       if (result)
  336. com_count[command][1]++;
  337.     }
  338.     switch ((enum nisam_log_commands) command) {
  339.     case LOG_OPEN:
  340.       com_count[command][0]--; /* Must be counted explicite */
  341.       if (result)
  342. com_count[command][1]--;
  343.       if (curr_file_info)
  344. printf("nWarning: %s is opened twice with same process and filenumbern",
  345.        curr_file_info->show_name);
  346.       if (my_b_read(&cache,(byte*) head,2))
  347. goto err;
  348.       file_info.name=0;
  349.       file_info.show_name=0;
  350.       file_info.record=0;
  351.       if (read_string(&cache,(gptr*) &file_info.name,(uint) uint2korr(head)))
  352. goto err;
  353.       {
  354. uint i;
  355. char *pos=file_info.name,*to;
  356. for (i=0 ; i < prefix_remove ; i++)
  357. {
  358.   char *next;
  359.   if (!(next=strchr(pos,FN_LIBCHAR)))
  360.     break;
  361.   pos=next+1;
  362. }
  363. to=isam_file_name;
  364. if (filepath)
  365. {
  366.   strmov(isam_file_name,filepath);
  367.   convert_dirname(isam_file_name);
  368.   to=strend(isam_file_name);
  369. }
  370. strmov(to,pos);
  371. fn_ext(isam_file_name)[0]=0; /* Remove extension */
  372.       }
  373.       open_param.name=file_info.name;
  374.       open_param.max_id=0;
  375.       VOID(tree_walk(&tree,(tree_walk_action) test_if_open,(void*) &open_param,
  376.      left_root_right));
  377.       file_info.id=open_param.max_id+1;
  378.       file_info.show_name=my_memdup(isam_file_name,
  379.     (uint) strlen(isam_file_name)+6,
  380.     MYF(MY_WME));
  381.       if (file_info.id > 1)
  382. sprintf(strend(file_info.show_name),"<%d>",file_info.id);
  383.       file_info.closed=1;
  384.       file_info.accessed=access_time;
  385.       file_info.used=1;
  386.       if (table_names[0])
  387.       {
  388. char **name;
  389. file_info.used=0;
  390. for (name=table_names ; *name ; name++)
  391. {
  392.   if (!strcmp(*name,isam_file_name))
  393.     file_info.used=1; /* Update/log only this */
  394. }
  395.       }
  396.       if (update && file_info.used)
  397.       {
  398. if (files_open >= max_files)
  399. {
  400.   if (close_some_file(&tree))
  401.     goto com_err;
  402.   files_open--;
  403. }
  404. if (!(file_info.isam= nisam_open(isam_file_name,O_RDWR,
  405.       HA_OPEN_WAIT_IF_LOCKED)))
  406.   goto com_err;
  407. if (!(file_info.record=my_malloc(file_info.isam->s->base.reclength,
  408.  MYF(MY_WME))))
  409.   goto end;
  410. files_open++;
  411. file_info.closed=0;
  412.       }
  413.       VOID(tree_insert(&tree,(gptr) &file_info,0));
  414.       if (file_info.used)
  415.       {
  416. if (verbose && !record_pos_file)
  417.   printf_log("%s: open",file_info.show_name);
  418. com_count[command][0]++;
  419. if (result)
  420.   com_count[command][1]++;
  421.       }
  422.       break;
  423.     case LOG_CLOSE:
  424.       if (verbose && !record_pos_file &&
  425.   (!table_names[0] || (curr_file_info && curr_file_info->used)))
  426. printf_log("%s: %s -> %d",FILENAME(curr_file_info),
  427.        command_name[command],result);
  428.       if (curr_file_info)
  429.       {
  430. if (!curr_file_info->closed)
  431.   files_open--;
  432. file_info_free(curr_file_info);
  433. VOID(tree_delete(&tree,(gptr) curr_file_info));
  434.       }
  435.       break;
  436.     case LOG_EXTRA:
  437.       if (my_b_read(&cache,(byte*) head,sizeof(extra_command)))
  438. goto err;
  439.       memcpy_fixed(&extra_command,head,sizeof(extra_command));
  440.       if (verbose && !record_pos_file &&
  441.   (!table_names[0] || (curr_file_info && curr_file_info->used)))
  442. printf_log("%s: %s(%d) -> %d",FILENAME(curr_file_info),
  443.    command_name[command], extra_command,result);
  444.       if (update && curr_file_info && !curr_file_info->closed)
  445.       {
  446. if (nisam_extra(curr_file_info->isam,extra_command) != (int) result)
  447. {
  448.   VOID(fprintf(stderr,
  449.        "Warning: error %d, expected %d on command %s at %lxn",
  450.        my_errno,result,command_name[command],isamlog_filepos));
  451. }
  452.       }
  453.       break;
  454.     case LOG_DELETE:
  455.       if (my_b_read(&cache,(byte*) head,sizeof(filepos)))
  456. goto err;
  457.       memcpy_fixed(&filepos,head,sizeof(filepos));
  458.       if (verbose && (!record_pos_file ||
  459.       ((record_pos == filepos || record_pos == NO_FILEPOS) &&
  460.        !cmp_filename(curr_file_info,record_pos_file))) &&
  461.   (!table_names[0] || (curr_file_info && curr_file_info->used)))
  462. printf_log("%s: %s at %ld -> %d",FILENAME(curr_file_info),
  463.    command_name[command],(long) filepos,result);
  464.       if (update && curr_file_info && !curr_file_info->closed)
  465.       {
  466. if (nisam_rrnd(curr_file_info->isam,curr_file_info->record,filepos))
  467. {
  468.   if (!recover)
  469.     goto com_err;
  470.   com_count[command][2]++; /* Mark error */
  471. }
  472. ni_result=nisam_delete(curr_file_info->isam,curr_file_info->record);
  473. if ((ni_result == 0 && result) ||
  474.     (ni_result && (uint) my_errno != result))
  475. {
  476.   if (!recover)
  477.     goto com_err;
  478.   if (ni_result)
  479.     com_count[command][2]++; /* Mark error */
  480. }
  481.       }
  482.       break;
  483.     case LOG_WRITE:
  484.     case LOG_UPDATE:
  485.       if (my_b_read(&cache,(byte*) head,8))
  486. goto err;
  487.       filepos=uint4korr(head);
  488.       length=uint4korr(head+4);
  489.       buff=0;
  490.       if (read_string(&cache,&buff,(uint) length))
  491. goto err;
  492.       if ((!record_pos_file ||
  493.   ((record_pos == filepos || record_pos == NO_FILEPOS) &&
  494.    !cmp_filename(curr_file_info,record_pos_file))) &&
  495.   (!table_names[0] || (curr_file_info && curr_file_info->used)))
  496.       {
  497. if (write_file &&
  498.     (my_fwrite(write_file,buff,length,MYF(MY_WAIT_IF_FULL | MY_NABP))))
  499.   goto end;
  500. if (verbose)
  501.   printf_log("%s: %s at %ld, length=%ld -> %d",
  502.      FILENAME(curr_file_info),
  503.      command_name[command], filepos,length,result);
  504.       }
  505.       if (update && curr_file_info && !curr_file_info->closed)
  506.       {
  507. if (curr_file_info->isam->s->base.blobs)
  508.   fix_blob_pointers(curr_file_info->isam,buff);
  509. if ((enum nisam_log_commands) command == LOG_UPDATE)
  510. {
  511.   if (nisam_rrnd(curr_file_info->isam,curr_file_info->record,filepos))
  512.   {
  513.     if (!recover)
  514.     {
  515.       result=0;
  516.       goto com_err;
  517.     }
  518.     if (recover == 1 || result ||
  519. find_record_with_key(curr_file_info,buff))
  520.     {
  521.       com_count[command][2]++; /* Mark error */
  522.       break;
  523.     }
  524.   }
  525.   ni_result=nisam_update(curr_file_info->isam,curr_file_info->record,
  526.       buff);
  527.   if ((ni_result == 0 && result) ||
  528.       (ni_result && (uint) my_errno != result))
  529.   {
  530.     if (!recover)
  531.       goto com_err;
  532.     if (ni_result)
  533.       com_count[command][2]++; /* Mark error */
  534.   }
  535. }
  536. else
  537. {
  538.   ni_result=nisam_write(curr_file_info->isam,buff);
  539.   if ((ni_result == 0 && result) ||
  540.       (ni_result && (uint) my_errno != result))
  541.   {
  542.     if (!recover)
  543.       goto com_err;
  544.     if (ni_result)
  545.       com_count[command][2]++; /* Mark error */
  546.   }
  547.   if (! recover && filepos != curr_file_info->isam->lastpos)
  548.   {
  549.     printf("Warning: Wrote at position: %ld, should have been %ld",
  550.    curr_file_info->isam->lastpos,(long) filepos);
  551.     goto com_err;
  552.   }
  553. }
  554.       }
  555.       my_free(buff,MYF(0));
  556.       break;
  557.     case LOG_LOCK:
  558.       if (my_b_read(&cache,(byte*) head,sizeof(lock_command)))
  559. goto err;
  560.       memcpy_fixed(&lock_command,head,sizeof(lock_command));
  561.       if (verbose && !record_pos_file &&
  562.   (!table_names[0] || (curr_file_info && curr_file_info->used)))
  563. printf_log("%s: %s(%d) -> %dn",FILENAME(curr_file_info),
  564.    command_name[command],lock_command,result);
  565.       if (update && curr_file_info && !curr_file_info->closed)
  566.       {
  567. if (nisam_lock_database(curr_file_info->isam,lock_command) !=
  568.     (int) result)
  569.   goto com_err;
  570.       }
  571.       break;
  572.     default:
  573.       VOID(fprintf(stderr,
  574.    "Error: found unknown command %d in logfile, abortedn",
  575.    command));
  576.       goto end;
  577.     }
  578.   }
  579.   end_key_cache();
  580.   delete_tree(&tree);
  581.   VOID(end_io_cache(&cache));
  582.   VOID(my_close(file,MYF(0)));
  583.   if (write_file && my_fclose(write_file,MYF(MY_WME)))
  584.     DBUG_RETURN(1);
  585.   DBUG_RETURN(0);
  586.  err:
  587.   VOID(fprintf(stderr,"Got error %d when reading from logfilen",my_errno));
  588.   goto end;
  589.  com_err:
  590.   VOID(fprintf(stderr,"Got error %d, expected %d on command %s at %lxn",
  591.        my_errno,result,command_name[command],isamlog_filepos));
  592.  end:
  593.   end_key_cache();
  594.   delete_tree(&tree);
  595.   VOID(end_io_cache(&cache));
  596.   VOID(my_close(file,MYF(0)));
  597.   if (write_file)
  598.     VOID(my_fclose(write_file,MYF(MY_WME)));
  599.   DBUG_RETURN(1);
  600. }
  601. static int read_string(file,to,length)
  602. IO_CACHE *file;
  603. reg1 gptr *to;
  604. reg2 uint length;
  605. {
  606.   DBUG_ENTER("read_string");
  607.   if (*to)
  608.     my_free((gptr) *to,MYF(0));
  609.   if (!(*to= (gptr) my_malloc(length+1,MYF(MY_WME))) ||
  610.       my_b_read(file,(byte*) *to,length))
  611.   {
  612.     if (*to)
  613.       my_free(*to,MYF(0));
  614.     *to= 0;
  615.     DBUG_RETURN(1);
  616.   }
  617.   *((char*) *to+length)= '';
  618.   DBUG_RETURN (0);
  619. } /* read_string */
  620. static int file_info_compare(a,b)
  621. void *a;
  622. void *b;
  623. {
  624.   long lint;
  625.   if ((lint=((struct file_info*) a)->process -
  626.        ((struct file_info*) b)->process))
  627.     return lint < 0L ? -1 : 1;
  628.   return ((struct file_info*) a)->filenr - ((struct file_info*) b)->filenr;
  629. }
  630. /* ARGSUSED */
  631. static int test_if_open (key,count,param)
  632. struct file_info *key;
  633. element_count count __attribute__((unused));
  634. struct test_if_open_param *param;
  635. {
  636.   if (!strcmp(key->name,param->name) && key->id > param->max_id)
  637.     param->max_id=key->id;
  638.   return 0;
  639. }
  640. static void fix_blob_pointers(info,record)
  641. N_INFO *info;
  642. byte *record;
  643. {
  644.   byte *pos;
  645.   N_BLOB *blob,*end;
  646.   pos=record+info->s->base.reclength;
  647.   for (end=info->blobs+info->s->base.blobs, blob= info->blobs;
  648.        blob != end ;
  649.        blob++)
  650.   {
  651.     bmove(record+blob->offset+blob->pack_length,&pos,sizeof(char*));
  652.     pos+=_calc_blob_length(blob->pack_length,record+blob->offset);
  653.   }
  654. }
  655. static uint set_maximum_open_files(maximum_files)
  656. uint maximum_files;
  657. {
  658. #if defined(HAVE_GETRUSAGE) && defined(RLIMIT_NOFILE)
  659.   struct rlimit rlimit;
  660.   int old_max;
  661.   if (maximum_files > MY_NFILE)
  662.     maximum_files=MY_NFILE; /* Don't crash my_open */
  663.   if (!getrlimit(RLIMIT_NOFILE,&rlimit))
  664.   {
  665.     old_max=rlimit.rlim_max;
  666.     if (maximum_files && (int) maximum_files > old_max)
  667.       rlimit.rlim_max=maximum_files;
  668.     rlimit.rlim_cur=rlimit.rlim_max;
  669.     if (setrlimit(RLIMIT_NOFILE,&rlimit))
  670.     {
  671.       if (old_max != (int) maximum_files)
  672.       { /* Set as much as we can */
  673. rlimit.rlim_max=rlimit.rlim_cur=old_max;
  674. setrlimit(RLIMIT_NOFILE,&rlimit);
  675.       }
  676.     }
  677.     getrlimit(RLIMIT_NOFILE,&rlimit); /* Read if broken setrlimit */
  678.     if (maximum_files && maximum_files < rlimit.rlim_cur)
  679.       VOID(fprintf(stderr,"Warning: Error from setrlimit: Max open files is %dn",old_max));
  680.     return rlimit.rlim_cur;
  681.   }
  682. #endif
  683.   return min(maximum_files,MY_NFILE);
  684. }
  685. /* close the file with hasn't been accessed for the longest time */
  686. /* ARGSUSED */
  687. static int test_when_accessed (key,count,access_param)
  688. struct file_info *key;
  689. element_count count __attribute__((unused));
  690. struct st_access_param *access_param;
  691. {
  692.   if (key->accessed < access_param->min_accessed && ! key->closed)
  693.   {
  694.     access_param->min_accessed=key->accessed;
  695.     access_param->found=key;
  696.   }
  697.   return 0;
  698. }
  699. static void file_info_free(fileinfo)
  700. struct file_info *fileinfo;
  701. {
  702.   if (update)
  703.   {
  704.     if (!fileinfo->closed)
  705.       VOID(nisam_close(fileinfo->isam));
  706.     if (fileinfo->record)
  707.       my_free(fileinfo->record,MYF(0));
  708.   }
  709.   my_free(fileinfo->name,MYF(0));
  710.   my_free(fileinfo->show_name,MYF(0));
  711. }
  712. static int close_some_file(tree)
  713. TREE *tree;
  714. {
  715.   struct st_access_param access_param;
  716.   access_param.min_accessed=LONG_MAX;
  717.   access_param.found=0;
  718.   VOID(tree_walk(tree,(tree_walk_action) test_when_accessed,
  719.  (void*) &access_param,left_root_right));
  720.   if (!access_param.found)
  721.     return 1; /* No open file that is possibly to close */
  722.   if (nisam_close(access_param.found->isam))
  723.     return 1;
  724.   access_param.found->closed=1;
  725.   return 0;
  726. }
  727. static int reopen_closed_file(tree,fileinfo)
  728. TREE *tree;
  729. struct file_info *fileinfo;
  730. {
  731.   char name[FN_REFLEN];
  732.   if (close_some_file(tree))
  733.     return 1; /* No file to close */
  734.   strmov(name,fileinfo->show_name);
  735.   if (fileinfo->id > 1)
  736.     *strrchr(name,'<')=''; /* Remove "<id>" */
  737.   if (!(fileinfo->isam= nisam_open(name,O_RDWR,HA_OPEN_WAIT_IF_LOCKED)))
  738.     return 1;
  739.   fileinfo->closed=0;
  740.   re_open_count++;
  741.   return 0;
  742. }
  743. /* Try to find record with uniq key */
  744. static int find_record_with_key(file_info,record)
  745. struct file_info *file_info;
  746. byte *record;
  747. {
  748.   uint key;
  749.   N_INFO *info=file_info->isam;
  750.   uchar tmp_key[N_MAX_KEY_BUFF];
  751.   for (key=0 ; key < info->s->state.keys ; key++)
  752.   {
  753.     if (info->s->keyinfo[key].base.flag & HA_NOSAME)
  754.     {
  755.       VOID(_nisam_make_key(info,key,tmp_key,record,0L));
  756.       return nisam_rkey(info,file_info->record,(int) key,(char*) tmp_key,0,
  757.      HA_READ_KEY_EXACT);
  758.     }
  759.   }
  760.   return 1;
  761. }
  762. static void printf_log(const char *format,...)
  763. {
  764.   va_list args;
  765.   va_start(args,format);
  766.   if (verbose > 2)
  767.     printf("%9ld:",isamlog_filepos);
  768.   if (verbose > 1)
  769.     printf("%5ld ",isamlog_process); /* Write process number */
  770.   (void) vprintf((char*) format,args);
  771.   putchar('n');
  772.   va_end(args);
  773. }
  774. static bool cmp_filename(file_info,name)
  775. struct file_info *file_info;
  776. my_string name;
  777. {
  778.   if (!file_info)
  779.     return 1;
  780.   return strcmp(file_info->name,name) ? 1 : 0;
  781. }