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

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