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

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