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