HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
lifutils.c
Go to the documentation of this file.
1 
88 #ifdef LIF_STAND_ALONE
89 #include <unistd.h>
90 #include "user_config.h"
91 #include "lifsup.h"
92 #include "lifutils.h"
93 #include "td02lif.h"
94 
95 #else
96 #include "hardware/user_config.h"
97 #include "vector.h"
98 #include "drives_sup.h"
99 #include "lifsup.h"
100 #include "lifutils.h"
101 #endif
102 
103 extern int debuglevel;
104 extern hpdir_t hpdir;
105 
110 MEMSPACE
111 void lif_help(int full)
112 {
113  printf( "lif help\n" );
114 
115  if(full)
116  {
117  printf(
118  "lif add lifimage lifname from_ascii_file\n"
119  "lif addbin lifimage lifname from_lif_file\n"
120  "lif create lifimage label directory_sectors sectors\n"
121  "lif createdisk lifimage label model\n"
122  "lif del lifimage name\n"
123  "lif dir lifimage\n"
124  "lif extract lifimage lifname to_ascii_file\n"
125  "lif extractbin lifimage lifname to_lif_file\n"
126  " extracts a file into a sigle file LIF image\n"
127  "lif rename lifimage oldlifname newlifname\n"
128  "lif renamevol lifimage name\n"
129  "Use -d after 'lif' keyword to enable LIF filesystem debugging\n"
130  "\n"
131  );
132  }
133 }
134 
135 
138 MEMSPACE
139 int lif_tests(int argc, char *argv[])
140 {
141 
142  int ind=0;
143  char *ptr;
144 
145 // display arguments for debugging
146 #ifdef TELEDISK
147  int i;
148 #endif
149 #if 0
150  for(i=0;i<argc;++i)
151  printf("%d:%s\n", i, argv[i]);
152  printf("\n");
153 #endif
154 
155 // Nothing to do ?
156  if(argc < 2)
157  return (0);
158 
159 // Argument 1
160  ind = 1;
161  ptr = argv[ind++];
162 
163 // Argument 1 missing ?
164 // Nothing to do
165 // argc should really get this
166  if(!ptr || !*ptr)
167  return(1);
168 
169 
170  if(MATCHI_LEN(argv[0],"lif"))
171  {
172  if(MATCHI(ptr,"help") || MATCHI(ptr,"-help") || MATCHI(ptr,"-?") )
173  {
174  lif_help(1);
175  return(1);
176  }
177  }
178 
179 // Turn one debugging
180 // in the future we can add tests for specific messages
181  debuglevel &= ~0x400;
182  if (MATCHARGS(ptr,"-d", (ind + 0) ,argc))
183  {
184  debuglevel |= 0x400;
185  ptr = argv[ind++];
186  }
187 
188  if (MATCHARGS(ptr,"addbin", (ind + 3) ,argc))
189  {
190  lif_add_lif_file(argv[ind],argv[ind+1],argv[ind+2]);
191 
192  return(1);
193  }
194 
195  if (MATCHARGS(ptr,"add", (ind + 3) ,argc))
196  {
197  lif_add_ascii_file_as_e010(argv[ind],argv[ind+1],argv[ind+2]);
198  return(1);
199  }
200  if (MATCHARGS(ptr,"createdisk", (ind + 3) ,argc))
201  {
203  long dir,sectors;
204  char *name = argv[ind];
205  char *label = argv[ind+1];
206  char *model = argv[ind+2];
207  if( MATCHI_LEN(model,"hp"))
208  model +=2;
209  if(hpdir_find_drive(model,0, 0))
210  {
211  dir = lif_dir_count(hpdir.BLOCKS);
212  sectors = hpdir.BLOCKS;
213  // NOTE: we could grab the directory size for non 0 entries in the hpdir.ini file - I use a computed value which is also fine
214  lif_create_image(name, label, dir, sectors);
215  return(1);
216  }
217  printf("Disk: %s not found in hpdir.ini\n", model);
218  return(1);
219  }
220  if (MATCHARGS(ptr,"create", (ind + 4) ,argc))
221  {
223  lif_create_image(argv[ind],argv[ind+1], atol(argv[ind+2]), atol(argv[ind+3]) );
224  return(1);
225  }
226  if (MATCHARGS(ptr,"del", (ind + 2) ,argc))
227  {
228  lif_del_file(argv[ind],argv[ind+1]);
229  return(1);
230  }
231  if (MATCHARGS(ptr,"dir", (ind + 1) ,argc))
232  {
233  lif_dir(argv[ind]);
234  return(1);
235  }
236  if (MATCHARGS(ptr,"extractbin", (ind + 3) ,argc))
237  {
238 
239  lif_extract_lif_as_lif(argv[ind],argv[ind+1],argv[ind+2]);
240  return(1);
241  }
242  if (MATCHARGS(ptr,"extract", (ind + 3) ,argc))
243  {
244 
245  lif_extract_e010_as_ascii(argv[ind],argv[ind+1],argv[ind+2]);
246  return(1);
247  }
248  if (MATCHARGS(ptr,"rename", (ind + 3) ,argc))
249  {
250  lif_rename_file(argv[ind],argv[ind+1],argv[ind+2]);
251  return(1);
252  }
253 
254  if (MATCHARGS(ptr,"renamevol", (ind + 2) ,argc))
255  {
256  lif_rename_volume(argv[ind],argv[ind+1]);
257  return(1);
258  }
259 
260  if(MATCHI_LEN(argv[0],"td02lif"))
261  {
262  if(MATCHI(ptr,"help") || MATCHI(ptr,"-help") || MATCHI(ptr,"-?") )
263  {
264 #ifdef TELEDISK
265  td0_help(1);
266  return(1);
267 #else
268  printf("td02lif support not enabled\n");
269  return(1);
270 #endif
271  }
272 #ifdef TELEDISK
273 // shift the arguments down by 1
274  for(i=1;i<argc;++i)
275  {
276  argv[i-1] = argv[i];
277  }
278  argv[argc--] = NULL;
279 
280  td02lif(argc,argv);
281  return(1);
282 #endif
283  }
284  return(0);
285 }
286 
287 
292 MEMSPACE
293 void *lif_calloc(long size)
294 {
295  uint8_t *p = safecalloc(size,1);
296  if(!p)
297  printf("lif_calloc:[%ld] not enough free memory\n", size);
298 
299  return(p);
300 }
301 
302 
307 MEMSPACE
308 void lif_free(void *p)
309 {
310  if(!p)
311  printf("lif_free: NULL pointer\n");
312  else
313  safefree(p);
314 }
315 
316 
321 MEMSPACE
322 char *lif_stralloc(char *str)
323 {
324  int len = strlen(str);
325  char *p = (char *)lif_calloc(len+4);
326  if(!p)
327  return(NULL);
328  strcpy(p,str);
329  return(p);
330 }
331 
332 
338 MEMSPACE
339 FILE *lif_open(char *name, char *mode)
340 {
341  FILE *fp = fopen(name, mode);
342  if( fp == NULL)
343  {
344  printf("lif_open: Can't open:[%s] mode:[%s]\n", name, mode);
345  return(NULL);
346  }
347  return(fp);
348 }
349 
350 
356 MEMSPACE
357 stat_t *lif_stat(char *name, stat_t *p)
358 {
359  if(stat(name, p) < 0)
360  {
361  printf("lif_stat: Can't stat:%s\n", name);
362  return(NULL);
363  }
364  return(p);
365 }
366 
367 
373 MEMSPACE
374 int lif_seek_msg(FILE *fp, long offset, char *msg)
375 {
376  if(ftell(fp) != offset)
377  {
378  if(fseek(fp, offset, SEEK_SET) < 0)
379  {
380  printf("lif_read_msg: %s Seek error %ld\n", msg, offset);
381  return(0);
382  }
383  }
384  return(1);
385 }
386 
387 
395 MEMSPACE
396 long lif_read(lif_t *LIF, void *buf, long offset, int bytes)
397 {
398  long len;
399 
400  if(!lif_seek_msg(LIF->fp,offset,LIF->name))
401  return(0);
402 
404  len = fread(buf, 1, bytes, LIF->fp);
405  if( len != bytes)
406  {
407 
408  if(debuglevel & LIF_DEBUG)
409  printf("lif_read: read:[%s] offset:[%ld] write:[%ld] expected:[%d]\n",
410  LIF->name, (long)offset, (long)len, (int)bytes);
411  }
412  return(len);
413 }
414 
415 
423 MEMSPACE
424 int lif_write(lif_t *LIF, void *buf, long offset, int bytes)
425 {
426  int len;
427 
428 // Seek to write position
429  if(!lif_seek_msg(LIF->fp, offset,LIF->name))
430  return(0);
431 
433  len = fwrite(buf, 1, bytes, LIF->fp);
434  if( len != bytes)
435  {
436  if(debuglevel & LIF_DEBUG)
437  printf("lif_write: Write:[%s] offset:[%ld] write:[%d] expected:[%d]\n",
438  LIF->name, offset, len, bytes);
439  }
440  return(len);
441 }
442 
443 
448 MEMSPACE
449 int lif_chars(int c, int index __attribute__((unused)))
450 {
451 // Series 80 names allow most characters even though the LIF spec does not
452  if(c < 32 || c > 127)
453  return(0);
454  if(c == '.' || c == ':' || c == '"' || c == '\'' )
455  return(0);
456 // Linux and Windows Path separators
457  if(c == '/' || c == '\\')
458  return(0);
459  return(c);
460 #if 0
461  if(c == ' ')
462  return(c);
463  if(c >= 'a' && c <= 'z')
464  return(c-0x20);
465  if(c >= 'A' && c <= 'Z')
466  return(c);
467  if(index > 0)
468  {
469  if(c >= '0' && c <= '9')
470  return(c);
471  if(c == '$' || c == '_' || c == '-')
472  return(c);
473  }
474  return(0);
475 #endif
476 }
477 
478 
484 MEMSPACE
485 int lif_B2S(uint8_t *B, uint8_t *name, int size)
486 {
487  int i;
488  int status = 1;
489  for(i=0;i<size;++i)
490  {
491  if( !lif_chars(B[i],i))
492  status = 0;
493  }
494  for(i=0;i<size;++i)
495  name[i] = B[i];
496  name[i] = 0;
497 // remove space padding
498  trim_tail((char *)name);
499  return(status);
500 }
501 
502 
506 MEMSPACE
507 int lif_checkname(char *name)
508 {
509  int i;
510  int status = 1;
511  for(i=0;name[i];++i)
512  {
513  if(!lif_chars(name[i],i))
514  status = 0;
515  }
516  return(status);
517 }
518 
519 
525 MEMSPACE
526 void lif_S2B(uint8_t *B, uint8_t *name, int size)
527 {
528  int i;
529  for(i=0;name[i] && i<size;++i)
530  {
531  B[i] = name[i];
532  }
533  for(;i<size;++i)
534  B[i] = ' ';
535 }
536 
537 
548 MEMSPACE
549 int lif_fixname(uint8_t *B, char *name, int size)
550 {
551  uint8_t c,ret;
552  int i,index;
553  char *ptr;
554  uint8_t *save = B;
555 
556  index = 0;
557 // remove any "/"
558  ptr = basename(name);
559 
560  for(i=0; ptr[i] && index < size;++i)
561  {
562  c = ptr[i];
563 // trim off extensions
564  if(c == '.')
565  break;
566  if( (ret = lif_chars(c,i)) )
567  *B++ = ret;
568  else
569  *B++ = ' ';
570  }
571  while(i < size)
572  {
573  *B++ = ' ';
574  ++i;
575  };
576  *B = 0;
577  return(strlen((char *)save));
578 }
579 
580 
585 MEMSPACE
586 void lif_vol2str(lif_t *LIF, uint8_t *B)
587 {
588  V2B_MSB(B,0,2,LIF->VOL.LIFid);
589  lif_S2B(B+2,LIF->VOL.Label,6);
590  V2B_MSB(B,8,4,LIF->VOL.DirStartSector);
591  V2B_MSB(B,12,2,LIF->VOL.System3000LIFid);
592  V2B_MSB(B,14,2,0);
593  V2B_MSB(B,16,4,LIF->VOL.DirSectors);
594  V2B_MSB(B,20,2,LIF->VOL.LIFVersion);
595  V2B_MSB(B,22,2,0);
596  V2B_MSB(B,24,4,LIF->VOL.tracks_per_side);
597  V2B_MSB(B,28,4,LIF->VOL.sides);
598  V2B_MSB(B,32,4,LIF->VOL.sectors_per_track);
599  memcpy((void *) (B+36),LIF->VOL.date,6);
600 }
601 
602 
607 MEMSPACE
608 void lif_str2vol(uint8_t *B, lif_t *LIF)
609 {
610 
611  LIF->VOL.LIFid = B2V_MSB(B,0,2);
612  lif_B2S(B+2,LIF->VOL.Label,6);
613  LIF->VOL.DirStartSector = B2V_MSB(B,8,4);
614  LIF->VOL.System3000LIFid = B2V_MSB(B,12,2);
615  LIF->VOL.zero1 = B2V_MSB(B,14,2);
616  LIF->VOL.DirSectors = B2V_MSB(B,16,4);
617  LIF->VOL.LIFVersion = B2V_MSB(B,20,2);
618  LIF->VOL.zero2 = B2V_MSB(B,22,2);
619  LIF->VOL.tracks_per_side = B2V_MSB(B,24,4);
620  LIF->VOL.sides = B2V_MSB(B,28,4);
621  LIF->VOL.sectors_per_track = B2V_MSB(B,32,4);
622  memcpy((void *) LIF->VOL.date, (B+36),6);
623 }
624 
625 
630 MEMSPACE
631 void lif_dir2str(lif_t *LIF, uint8_t *B)
632 {
633  lif_S2B(B,LIF->DIR.filename,10); // 0
634  V2B_MSB(B,10,2,LIF->DIR.FileType); // 10
635  V2B_MSB(B,12,4,LIF->DIR.FileStartSector); // 12
636  V2B_MSB(B,16,4,LIF->DIR.FileSectors); // 16
637  memcpy(B+20,LIF->DIR.date,6); // 20
638  V2B_MSB(B,26,2,LIF->DIR.VolNumber); // 26
639  V2B_LSB(B,28,2,LIF->DIR.FileBytes); // 28
640  V2B_LSB(B,30,2,LIF->DIR.SectorSize); // 30
641 }
642 
643 
648 MEMSPACE
649 void lif_str2dir(uint8_t *B, lif_t *LIF)
650 {
651  lif_B2S(B,LIF->DIR.filename,10);
652  LIF->DIR.FileType = B2V_MSB(B, 10, 2);
653  LIF->DIR.FileStartSector = B2V_MSB(B, 12, 4);
654  LIF->DIR.FileSectors = B2V_MSB(B, 16, 4);
655  memcpy(LIF->DIR.date,B+20,6);
656  LIF->DIR.VolNumber = B2V_MSB(B, 26, 2);
657  LIF->DIR.FileBytes = B2V_LSB(B, 28, 2);
658  LIF->DIR.SectorSize= B2V_LSB(B, 30, 2);
659 }
660 
661 
667 MEMSPACE
668 uint8_t lif_BIN2BCD(uint8_t data)
669 {
670  return( ( (data/10U) << 4 ) | (data%10U) );
671 }
672 
673 
677 MEMSPACE
678 int lif_BCD2BIN(uint8_t bin)
679 {
680  return( ((bin>>4)*10U)+(bin & 0x0f) );
681 }
682 
683 
693 MEMSPACE
694 void lif_time2lifbcd(time_t t, uint8_t *bcd)
695 {
696  tm_t tm;
697  int i;
698 
699  if(t == 0)
700  {
701  for(i=0;i<6;++i)
702  bcd[i] = 0;
703  return;
704  }
705 
706  gmtime_r((time_t *) &t, (tm_t *)&tm);
707  bcd[0] = lif_BIN2BCD(tm.tm_year % 100);
708  bcd[1] = lif_BIN2BCD(tm.tm_mon+1);
709  bcd[2] = lif_BIN2BCD(tm.tm_mday);
710  bcd[3] = lif_BIN2BCD(tm.tm_hour);
711  bcd[4] = lif_BIN2BCD(tm.tm_min);
712  bcd[5] = lif_BIN2BCD(tm.tm_sec);
713 }
714 
715 
723 MEMSPACE
724 time_t lif_lifbcd2time(uint8_t *bcd)
725 {
726  int year,mon;
727  tm_t tm;
728  time_t t;
729  int i;
730  int zerof = 1;
731 
732  for(i=0;i<6;++i)
733  {
734  if(lif_BCD2BIN(bcd[i]))
735  zerof = 0;
736  }
737  if(!zerof)
738  {
739  year = lif_BCD2BIN(bcd[0]);
740  mon = lif_BCD2BIN(bcd[1])-1;
741  if(year < 70)
742  year += 100;
743  tm.tm_year = year;
744  tm.tm_mon = mon;
745  tm.tm_mday = lif_BCD2BIN(bcd[2]);
746  tm.tm_hour = lif_BCD2BIN(bcd[3]);
747  tm.tm_min= lif_BCD2BIN(bcd[4]);
748  tm.tm_sec= lif_BCD2BIN(bcd[5]);
749 
750  t = timegm( (tm_t *) &tm);
751  return(t);
752  }
753  return(0);
754 }
755 
756 
765 MEMSPACE
767 {
768  tm_t tm;
769  static char _lif_ctime_buf[32];
770  char *ptr;
771  memset(_lif_ctime_buf,0,sizeof(_lif_ctime_buf));
772  ptr = asctime_r( gmtime_r(tp,&tm), _lif_ctime_buf);
773  trim_tail(ptr);
774  return(ptr);
775 }
776 
777 
785 MEMSPACE
786 char *lif_lifbcd2timestr(uint8_t *bcd)
787 {
788  static char _timestr[32];
789  memset(_timestr,0,sizeof(_timestr));
790  time_t t = lif_lifbcd2time(bcd);
791 
792  if(t)
793  strcpy(_timestr,lif_ctime_gmt((time_t *)&t));
794  else
795  strcpy(_timestr,"<EMPTY>");
796  return(_timestr);
797 }
798 
799 
801 
805 MEMSPACE
807 {
808  memset((void *) LIF,0,sizeof(lif_t));
809 }
810 
811 
815 MEMSPACE
817 {
818  memset((void *) &LIF->DIR,0,sizeof(lifdir_t));
819 }
820 
821 
825 MEMSPACE
827 {
828  memset((void *) &LIF->VOL,0,sizeof(lifvol_t));
829 }
830 
831 
835 MEMSPACE
836 void lif_dump_vol(lif_t *LIF,char *msg)
837 {
838  printf("\n%s\n",msg);
839  printf("LIF name: %s\n", LIF->name);
840  printf("LIF sectors: %8lXh\n", (long)LIF->sectors);
841  printf("LIF bytes: %8lXh\n", (long)LIF->imagebytes);
842  printf("LIF filestart: %8lXh\n", (long)LIF->filestart);
843  printf("LIF file sectors: %8lXh\n", (long)LIF->filesectors);
844  printf("LIF used: %8lXh\n", (long)LIF->usedsectors);
845  printf("LIF free: %8lXh\n", (long)LIF->freesectors);
846  printf("LIF files: %8lXh\n",(long)LIF->files);
847  printf("LIF purged: %8lXh\n",(long)LIF->purged);
848 
849  printf("VOL Label: %s\n", LIF->VOL.Label);
850  printf("VOL LIFid: %8Xh\n",(int)LIF->VOL.LIFid);
851  printf("VOL Dir start: %8lXh\n",(long)LIF->VOL.DirStartSector);
852  printf("VOL Dir sectors: %8lXh\n",(long)LIF->VOL.DirSectors);
853  printf("VOL 3000LIFid: %8Xh\n",(unsigned int)LIF->VOL.System3000LIFid);
854  printf("VOL LIFVersion: %8Xh\n",(unsigned int)LIF->VOL.LIFVersion);
855  printf("VOL Date: %s\n", lif_lifbcd2timestr(LIF->VOL.date));
856  printf("DIR File Name: %s\n", LIF->DIR.filename);
857  printf("DIR File Type: %8Xh\n", (int)LIF->DIR.FileType);
858  printf("DIR File Volume#: %8Xh\n", (int)LIF->DIR.VolNumber);
859  printf("DIR File start: %8lXh\n", (long)LIF->DIR.FileStartSector);
860  printf("DIR File sectors: %8lXh\n", (long)LIF->DIR.FileSectors);
861  printf("DIR File bytes: %8lXh\n", (long)LIF->DIR.FileBytes);
862  printf("DIR File sector size: %8Xh\n", (int)LIF->DIR.SectorSize);
863  printf("DIR File Date: %s\n", lif_lifbcd2timestr(LIF->DIR.date));
864  printf("\n");
865 }
866 
867 
871 MEMSPACE
873 {
874  int status = 1;
875  uint32_t filestart;
876 
877  if( !lif_checkname((char *)LIF->VOL.Label) )
878  {
879  if(debuglevel & LIF_DEBUG)
880  printf("LIF Volume invalid Volume Name");
881  status = 0;
882  }
883 
884  if(LIF->VOL.System3000LIFid != 0x1000)
885  {
886  if(debuglevel & LIF_DEBUG)
887  printf("LIF Volume invalid System3000 ID (%04XH) expected 1000H\n", LIF->VOL.System3000LIFid);
888  status = 0;
889  }
890 
891  if(LIF->VOL.LIFVersion > 1)
892  {
893  if(debuglevel & LIF_DEBUG)
894  printf("LIF Version: %04XH > 1\n", LIF->VOL.LIFVersion);
895  status = 0;
896  }
897 
898  if(LIF->VOL.zero1 != 0)
899  {
900  if(debuglevel & LIF_DEBUG)
901  printf("LIF Volume invalid bytes at offset 14&15 should be zero\n");
902  status = 0;
903  }
904 
905  if(LIF->VOL.zero2 != 0)
906  {
907  if(debuglevel & LIF_DEBUG)
908  printf("LIF Volume invalid bytes at offset 22&23 should be zero\n");
909  status = 0;
910  }
911 
912  if(LIF->VOL.DirStartSector < 1)
913  {
914  if(debuglevel & LIF_DEBUG)
915  printf("LIF Volume invalid start sector:%ld\n", (long)LIF->VOL.DirStartSector);
916  status = 0;
917  }
918 
919  if(LIF->VOL.DirSectors < 1)
920  {
921  if(debuglevel & LIF_DEBUG)
922  printf("LIF Volume invalid Directory Sector Count < 1\n");
923  status = 0;
924  }
925 
926 // File area start and size
927  filestart = LIF->VOL.DirStartSector + LIF->VOL.DirSectors;
928  if(filestart > LIF->sectors)
929  {
930  if(debuglevel & LIF_DEBUG)
931  printf("LIF Volume invalid file start > image size\n");
932  status = 0;
933  }
934 
935  return(status);
936 }
937 
938 
945 MEMSPACE
947 {
948  int status = 1;
949 
950 // We do not check purged or end of DIRECTORY ercordss
951  if(LIF->DIR.FileType == 0)
952  {
953  return(1);
954  }
955 
956  if(LIF->DIR.FileType == 0xffff)
957  {
958  return(1);
959  }
960 
961  if( !lif_checkname((char *)LIF->DIR.filename) )
962  {
963  status = 0;
964  if(debuglevel & LIF_DEBUG)
965  printf("LIF Directory:[%s] invalid Name\n",LIF->DIR.filename);
966  }
967 
968  if(LIF->filestart)
969  {
970  if(LIF->DIR.FileStartSector < LIF->filestart)
971  {
972  status = 0;
973  if(debuglevel & LIF_DEBUG)
974  printf("LIF Directory:[%s] invalid start sector:%lXh < fie area start:%lXh\n",
975  LIF->DIR.filename,
976  (long)LIF->DIR.FileStartSector,
977  (long)LIF->filestart);
978  }
979  }
980 
981  if(LIF->sectors)
982  {
983  if( (LIF->DIR.FileStartSector + LIF->DIR.FileSectors) > (LIF->sectors) )
984  {
985  status = 0;
986  if(debuglevel & LIF_DEBUG)
987  {
988  printf("LIF Directory:[%s] invalid end sector:%lXh > total sectors:%lXh\n",
989  LIF->DIR.filename,
990  (long)LIF->DIR.FileStartSector + LIF->DIR.FileSectors,
991  (long)LIF->sectors);
992  }
993  }
994  }
995 
996  if(LIF->DIR.VolNumber != 0x8001)
997  {
998  status = 0;
999  if(debuglevel & LIF_DEBUG)
1000  printf("LIF Directory:[%s] invalid Volume Number:%Xh\n", LIF->DIR.filename, (int)LIF->DIR.VolNumber);
1001  }
1002 
1003 // Only inforce file type checks for types we know
1004 // 0xE010 .. 0xE013 are the file types that can use LIF->DIR.FileBytes
1005  if((LIF->DIR.FileType & 0xFFFC) == 0xE010)
1006  {
1007  if(LIF->DIR.FileBytes)
1008  {
1009 // Error test
1010  if( lif_bytes2sectors(LIF->DIR.FileBytes) > LIF->DIR.FileSectors )
1011  {
1012  status = 0;
1013  if(debuglevel & LIF_DEBUG)
1014  printf("LIF Directory:[%s] invalid FileBytes:%ld as sectors:%ld > FileSectors:%ld\n",
1015  LIF->DIR.filename,
1016  (long) LIF->DIR.FileBytes,
1017  (long) lif_bytes2sectors(LIF->DIR.FileBytes),
1018  (long) LIF->DIR.FileSectors);
1019  }
1020 
1021 // Warning test only
1022 // Does this LIF entry have more FileSectors then FileBytes when converted to sectors?
1023  if( lif_bytes2sectors(LIF->DIR.FileBytes) < LIF->DIR.FileSectors )
1024  {
1025  if(debuglevel & LIF_DEBUG)
1026  printf("LIF Directory:[%s] warning FileBytes:%ld as sectors:%ld < FileSectors:%ld\n",
1027  LIF->DIR.filename,
1028  (long) LIF->DIR.FileBytes,
1029  (long) lif_bytes2sectors(LIF->DIR.FileBytes),
1030  (long) LIF->DIR.FileSectors);
1031  }
1032  if(debuglevel & LIF_DEBUG && LIF->DIR.FileBytes == 0)
1033  {
1034  status = 0;
1035  printf("LIF Directory:[%s] invalid FileBytes == 0\n",
1036  LIF->DIR.filename);
1037  }
1038  }
1039  }
1040 
1041 //FIXME check file types?!
1042  if(LIF->DIR.SectorSize != LIF_SECTOR_SIZE)
1043  {
1044  // MG do we fail this ?
1045  // status = 0;
1046  if(debuglevel & LIF_DEBUG)
1047  printf("LIF Directory:[%s] warning sector size :%ld != %d\n", LIF->name,
1048  (long)LIF->DIR.SectorSize, (int) LIF_SECTOR_SIZE);
1049  }
1050 
1051  return(status);
1052 }
1053 
1054 
1061 MEMSPACE
1062 lif_t *lif_create_volume(char *imagename, char *liflabel, long dirstart, long dirsectors, long filesectors)
1063 {
1064  long size;
1065  long i;
1066  long offset;
1067  long count;
1068  uint8_t buffer[LIF_SECTOR_SIZE];
1069 
1070  time_t t = time(NULL);
1071 
1072  lif_t *LIF = lif_calloc(sizeof(lif_t)+4);
1073  if(LIF == NULL)
1074  return(NULL);
1075 
1076  printf("Creating:%s, Label:[%s], Directory Start %ld, Directory Size: %ld, File Sectors:%ld\n",
1077  imagename, liflabel, dirstart, dirsectors, filesectors );
1078 
1079  if(debuglevel & LIF_DEBUG)
1080  lif_dump_vol(LIF,"lif_create_volume");
1081 
1082  lif_image_clear(LIF);
1083 
1084 // Initialize volume header
1085  LIF->VOL.LIFid = 0x8000;
1086  lif_fixname(LIF->VOL.Label, liflabel, 6);
1087  LIF->VOL.DirStartSector = dirstart;
1088  LIF->VOL.DirSectors = dirsectors;
1089  LIF->VOL.System3000LIFid = 0x1000;
1090  LIF->VOL.tracks_per_side = 0;
1091  LIF->VOL.sides = 0;
1092  LIF->VOL.sectors_per_track = 0;
1094 
1095  lif_time2lifbcd(t, LIF->VOL.date);
1096 
1097 // update LIF headers
1098  LIF->name = lif_stralloc(imagename);
1099  if(LIF->name == NULL)
1100  {
1101  lif_close_volume(LIF);
1102  return(NULL);
1103  }
1104 
1105 // Initilize all LIF headers
1106  LIF->filesectors = filesectors;
1107  LIF->filestart = dirstart + dirsectors;
1108  LIF->sectors = (LIF->filestart+LIF->filesectors);
1109  LIF->imagebytes = LIF->sectors * (long)LIF_SECTOR_SIZE;
1110  LIF->freesectors = LIF->filesectors;
1111  LIF->usedsectors = 0;
1112  LIF->files = 0;
1113  LIF->purged = 0;
1114  LIF->dirindex = 0;
1115  LIF->EOFindex = 0;
1116 
1117  memset(buffer,0,LIF_SECTOR_SIZE);
1118 
1119  lif_vol2str(LIF,buffer);
1120 
1121 // Write Volume header
1122  LIF->fp = lif_open(LIF->name,"wb+");
1123  if(LIF->fp == NULL)
1124  {
1125  lif_close_volume(LIF);
1126  return(NULL);
1127  }
1128 
1129  offset = 0;
1130  count = 0;
1131 
1132  size = lif_write(LIF, buffer, offset, LIF_SECTOR_SIZE);
1133 
1134  if(size < LIF_SECTOR_SIZE)
1135  {
1136  lif_close_volume(LIF);
1137  return(NULL);
1138  }
1139  offset += size;
1140  ++count;
1141 
1142  memset(buffer,0,LIF_SECTOR_SIZE);
1143 
1144 // Space BETWEEN Volume header and Directory area
1145  for(i=1;i<dirstart;++i)
1146  {
1147  size = lif_write(LIF, buffer, offset, LIF_SECTOR_SIZE);
1148  if(size < LIF_SECTOR_SIZE)
1149  {
1150  lif_close_volume(LIF);
1151  return(NULL);
1152  }
1153  offset += size;
1154  printf("\tWrote: %ld\r", count);
1155  ++count;
1156  }
1157 
1158 // Write Directory sectors
1159  lif_dir_clear(LIF);
1160  LIF->DIR.FileType = 0xffff;
1161 
1162  for(i=0;i<LIF_SECTOR_SIZE;i+=LIF_DIR_SIZE)
1163  lif_dir2str(LIF,buffer+i);
1164 
1165  for(i=0;i<dirsectors;++i)
1166  {
1167  size = lif_write(LIF, buffer, offset, LIF_SECTOR_SIZE);
1168  if(size < LIF_SECTOR_SIZE)
1169  {
1170  lif_close_volume(LIF);
1171  return(NULL);
1172  }
1173  offset += size;
1174  if((count % 100) == 0)
1175  printf("\tWrote: %ld\r", count);
1176  ++count;
1177  }
1178 
1179 // File area sectors
1180  memset(buffer,0,LIF_SECTOR_SIZE);
1181  for(i=0;i<filesectors;++i)
1182  {
1183  size = lif_write(LIF, buffer, offset, LIF_SECTOR_SIZE);
1184  if(size < LIF_SECTOR_SIZE)
1185  {
1186  lif_close_volume(LIF);
1187  return(NULL);
1188  }
1189  offset += size;
1190  if((count % 100) == 0)
1191  printf("\tWrote: %ld\r", count);
1192  ++count;
1193  }
1194  printf("\tWrote: %ld\n", count);
1195 
1196  lif_rewinddir(LIF);
1197 
1198 // As a sanity check verify basic values
1199  if( !lif_check_volume(LIF) )
1200  {
1201  lif_closedir(LIF);
1202  return(NULL);
1203  }
1204 
1205 // Scan directory and verify values
1206  if( !lif_updatefree(LIF))
1207  {
1208  lif_closedir(LIF);
1209  return(NULL);
1210  }
1211 
1212  return(LIF);
1213 }
1214 
1215 
1219 MEMSPACE
1221 {
1222  if(LIF)
1223  {
1224  if(LIF->fp)
1225  {
1226  fseek(LIF->fp, 0, SEEK_END);
1227  fclose(LIF->fp);
1228  LIF->fp = NULL;
1229  sync();
1230  }
1231 
1232  if(LIF->name)
1233  lif_free(LIF->name);
1234 
1235  lif_vol_clear(LIF);
1236 
1237  lif_free(LIF);
1238  }
1239 }
1240 
1241 
1246 MEMSPACE
1247 uint32_t lif_bytes2sectors(uint32_t bytes)
1248 {
1249  uint32_t sectors = (bytes/(long)LIF_SECTOR_SIZE);
1250  if(bytes % (long)LIF_SECTOR_SIZE)
1251  ++sectors;
1252  return(sectors);
1253 }
1254 
1255 
1260 MEMSPACE
1262 {
1263 // Directory index
1264  LIF->dirindex = -1;
1265 }
1266 
1267 
1272 MEMSPACE
1274 {
1275  return( lif_close_volume(LIF) );
1276 }
1277 
1278 
1283 MEMSPACE
1284 int lif_checkdirindex(lif_t * LIF, int index)
1285 {
1286  if(index < 0 || lif_bytes2sectors((long) index * LIF_DIR_SIZE) > LIF->VOL.DirSectors)
1287  {
1288  printf("lif_checkdirindex:[%s] direcory index:[%d] out of bounds\n",LIF->name, index);
1289  if(debuglevel & LIF_DEBUG)
1290  lif_dump_vol(LIF,"lif_chckdirindex");
1291  return(0);
1292  }
1293  return(1);
1294 }
1295 
1296 
1301 MEMSPACE
1302 int lif_readdirindex(lif_t *LIF, int index)
1303 {
1304  uint32_t offset;
1305  uint32_t size;
1306  uint8_t dir[LIF_DIR_SIZE];
1307 
1308 // Verify that the records is withing directory limits
1309  if( !lif_checkdirindex(LIF, index) )
1310  {
1311  return(0);
1312  }
1313 
1314 // Compute offset
1315  offset = ((long)index * LIF_DIR_SIZE) + (LIF->VOL.DirStartSector * (long)LIF_SECTOR_SIZE);
1316 
1317 // read raw data
1318  size = lif_read(LIF, dir, offset, sizeof(dir));
1319  if(size < (long)sizeof(dir) )
1320  {
1321  return(0);
1322  }
1323 
1324 // Convert into directory structure
1325  lif_str2dir(dir, LIF);
1326 
1327 // Update EOF index
1328  if( LIF->DIR.FileType == 0xffffUL )
1329  LIF->EOFindex = index;
1330 
1331  if( !lif_check_dir(LIF))
1332  {
1333  printf("lif_check_dir: error, index:%d\n", (int)index);
1334  if(debuglevel & LIF_DEBUG)
1335  {
1336  lif_dump_vol(LIF,"lif_readdirindex");
1337  }
1338  return(0);
1339  }
1340  return(1);
1341 }
1342 
1343 
1348 MEMSPACE
1349 int lif_writedirindex(lif_t *LIF, int index)
1350 {
1351  long offset;
1352  uint8_t dir[LIF_DIR_SIZE];
1353 
1354 // Validate the record
1355  if(!lif_check_dir(LIF))
1356  {
1357  if(debuglevel & LIF_DEBUG)
1358  lif_dump_vol(LIF,"lif_writedirindex");
1359  return(0);
1360  }
1361 
1362 // check for out of bounds
1363  if( !lif_checkdirindex(LIF, index))
1364  return(0);
1365 
1366 // Update EOF index
1367  if( LIF->DIR.FileType == 0xffffUL )
1368  LIF->EOFindex = index;
1369 
1370  offset = ((long)index * LIF_DIR_SIZE) + (LIF->VOL.DirStartSector * (long)LIF_SECTOR_SIZE);
1371 
1372 // store LIF->DIR settings into dir
1373  lif_dir2str(LIF, dir);
1374 
1375  if( lif_write(LIF, dir, offset, sizeof(dir)) < (int ) sizeof(dir) )
1376  return(0);
1377 
1378  return(1);
1379 }
1380 
1381 
1386 MEMSPACE
1387 int lif_writedirEOF(lif_t *LIF, int index)
1388 {
1389 // Create a director EOF
1390  lif_dir_clear(LIF);
1391  LIF->DIR.FileType = 0xffff;
1392  LIF->EOFindex = index;
1393  return( lif_writedirindex(LIF,index));
1394 }
1395 
1396 
1402 MEMSPACE
1404 {
1405  while(1)
1406  {
1407 // Advance index first
1408 // We start initialized at -1 by lif_open_volume() and lif_rewinddir()
1409  LIF->dirindex++;
1410 
1411  if( !lif_readdirindex(LIF, LIF->dirindex) )
1412  break;
1413 
1414  if( LIF->DIR.FileType == 0xffffUL )
1415  break;
1416 
1417  if(LIF->DIR.FileType)
1418  return( (lifdir_t *) &LIF->DIR );
1419 
1420 // Skip purged records
1421  }
1422  return( NULL );
1423 }
1424 
1425 
1429 MEMSPACE
1431 {
1432  int index = 0;
1433  int purgeindex = -1;
1434 
1435 // Start of free space
1436  uint32_t start = LIF->filestart;
1437 // Free sectors
1438  LIF->freesectors = LIF->filesectors;
1439 // Used sectors
1440  LIF->usedsectors = 0;
1441 // Purged files
1442  LIF->purged= 0;
1443 // Files
1444  LIF->files = 0;
1445 // Director pointer
1446  LIF->dirindex = 0;
1447 // Directory EOF record
1448  LIF->EOFindex = 0;
1449 
1451  while(1)
1452  {
1453  if( !lif_readdirindex(LIF,index) )
1454  {
1455  return(NULL);
1456  }
1457 
1458  if(LIF->DIR.FileType == 0xffff)
1459  {
1460  LIF->EOFindex = index;
1461  if(purgeindex != -1)
1462  {
1463  LIF->EOFindex = purgeindex;
1464 
1465 // Adjust purged file count
1466  LIF->purged -= (index - purgeindex);
1467 // update EOF
1468  if(!lif_writedirEOF(LIF,purgeindex))
1469  {
1470  return(NULL);
1471  }
1472  }
1473  break;
1474  }
1475  if(LIF->DIR.FileType == 0)
1476  {
1477  if(purgeindex == -1)
1478  purgeindex = index;
1479  LIF->purged++;
1480  ++index;
1481  continue;
1482  }
1483  purgeindex = -1;
1484  if(start > LIF->DIR.FileStartSector)
1485  {
1486  if(debuglevel & LIF_DEBUG)
1487  printf("lif_update_free:[%s] error previous record file area overlaps this one:[%s]\n", LIF->name, LIF->DIR.filename);
1488 
1489  }
1490  LIF->usedsectors += LIF->DIR.FileSectors;
1491  LIF->freesectors -= LIF->DIR.FileSectors;
1492  LIF->files++;
1493  ++index;
1494  start = LIF->DIR.FileStartSector + LIF->DIR.FileSectors;
1495  }
1496 // rewind
1497  lif_rewinddir(LIF);
1498  return(LIF);
1499 }
1500 
1501 
1506 MEMSPACE
1507 int lif_newdir(lif_t *LIF, long sectors)
1508 {
1509  int index;
1510  long start;
1511 
1512  int freestate, freeindex;
1513  long freestart;
1514 
1515 // Directory index
1516  index = 0;
1517 
1518 // Start of free space
1519  start = LIF->filestart;
1520 
1521 // Update all file information
1522  if(lif_updatefree(LIF) == NULL)
1523  {
1524  printf("lif_newdir: not enough free space:[%ld] for size:[%ld]\n", (long)LIF->freesectors, (long) sectors);
1525  return(-1);
1526  }
1527 
1528 // Not enough room ?
1529  if(sectors > (long)LIF->freesectors)
1530  {
1531  printf("lif_newdir: not enough free space:[%ld]\n", (long)LIF->freesectors);
1532  return(-1);
1533  }
1534 
1535 // 0 = reading valid records
1536 // 1 = found purged records
1537 // 2 = found pured records with enough space to reuse
1538  freestate = 0;
1539 
1540 // Update free space and EOF pointers
1541  while(1)
1542  {
1543 // Write new EOF after current one
1544  if( !lif_readdirindex(LIF,index) )
1545  {
1546 #if 0
1547  printf("lif_newdir: index:[%d] failed read at:[%ld] of [%ld] sectors, free:[%ld]\n",
1548  (int) index, (long) start, (long)sectors, (long)LIF->freesectors);
1549 #endif
1550  break;
1551  }
1552 
1553 // We hit the EOF record
1554  if(LIF->DIR.FileType == 0xffff)
1555  {
1556 
1557 // Was enough free space found in purged area ?
1558 // Do NOT need to update EOF!
1559  if(freestate == 2)
1560  {
1561 // Update free pace for new file
1562  lif_dir_clear(LIF);
1563  LIF->DIR.FileStartSector = freestart;
1564  LIF->DIR.FileSectors = sectors;
1565  LIF->usedsectors += sectors;
1566  LIF->freesectors -= sectors;
1567  LIF->files++;
1568  LIF->purged--;
1569  LIF->dirindex = freeindex;
1570 // Write new record (FileType is still EOF until data is updated by user)
1571  if( !lif_writedirindex(LIF,freeindex))
1572  {
1573  break;
1574  }
1575  return(freeindex);
1576  }
1577 
1578  if(debuglevel & LIF_DEBUG)
1579  printf("lif_newdir: index:[%d] adding at:[%ld]to purged space:[%ld] sectors, free:[%ld]\n",
1580  (int) index,(long)start,(long) sectors, (long)LIF->freesectors);
1581 
1582 // Write new EOF after current one
1583  if( !lif_writedirEOF(LIF,index+1) )
1584  {
1585  break;
1586  }
1587 
1588  lif_dir_clear(LIF);
1589  LIF->DIR.FileStartSector = start;
1590  LIF->DIR.FileSectors = sectors;
1591  LIF->usedsectors += sectors;
1592  LIF->freesectors -= sectors;
1593  LIF->files++;
1594  LIF->dirindex = index;
1595 // Write new record (FileType is still EOF until data is updated by user )
1596  if( !lif_writedirindex(LIF,index))
1597  {
1598  break;
1599  }
1600  return(index);
1601  }
1602 
1603 // PURGED records
1604  if(LIF->DIR.FileType == 0)
1605  {
1606  if(freestate == 0)
1607  {
1608  freeindex = index;
1609  freestart = start;
1610  freestate = 1;
1611  }
1612  ++index;
1613  continue;
1614  }
1615 
1616 // VALID record
1617  if(freestate == 1)
1618  {
1619  long freesectors;
1620 // Compute sectors of purged space between valid records
1621 // Note: LIF specs prohibit using old purged file start and sectors!
1622  freesectors = LIF->DIR.FileStartSector - start;
1623  if(freesectors >= sectors )
1624  freestate = 2; // Found free space in purged record
1625  else
1626  freestate = 0; // Try again
1627  }
1628 
1629 // Computed start of next record
1630 // Note: MUST be <= actual start of next VALID record or we have overlapping records!
1631  start = LIF->DIR.FileStartSector + LIF->DIR.FileSectors;
1632  ++index;
1633  }
1634 // ERROR
1635  return(-1);
1636 }
1637 
1638 
1643 MEMSPACE
1644 lif_t *lif_open_volume(char *name, char *mode)
1645 {
1646  lif_t *LIF;
1647  stat_t sb, *sp;
1648  uint8_t buffer[LIF_SECTOR_SIZE];
1649 
1650  sp = lif_stat(name, (stat_t *)&sb);
1651  if(sp == NULL)
1652  return(NULL);
1653 
1654 // To read LIF volume we must have at minimum two sectors
1655 // volume header a directory entry
1656  if(sp->st_size < (long)LIF_SECTOR_SIZE*2)
1657  {
1658  if(debuglevel & LIF_DEBUG)
1659  printf("lif_open_volume:[%s] error volume header area too small:[%ld]\n", name, (long)sp->st_size);
1660  return(NULL);
1661  }
1662 
1663 // Allocate LIF structure
1664  LIF = lif_calloc(sizeof(lif_t)+4);
1665  if(!LIF)
1666  return(NULL);
1667 
1668  LIF->name = lif_stralloc(name);
1669  if(!LIF->name)
1670  {
1671  lif_closedir(LIF);
1672  return(NULL);
1673  }
1674 
1675  LIF->imagebytes = sp->st_size;
1676  LIF->sectors = lif_bytes2sectors(sp->st_size);
1677 
1678  LIF->fp = lif_open(LIF->name,mode);
1679  if(!LIF->fp)
1680  {
1681  lif_closedir(LIF);
1682  return(NULL);
1683  }
1684 
1685 // Volume header must be it least one sector
1686  if( lif_read(LIF, buffer, 0, LIF_SECTOR_SIZE) < LIF_SECTOR_SIZE)
1687  {
1688  if(debuglevel & LIF_DEBUG)
1689  printf("lif_open_volume:[%s] error read volume header failed\n", name);
1690  lif_closedir(LIF);
1691  return(NULL);
1692  }
1693 
1694 // Unpack Volumes has the Directory start sector
1695  lif_str2vol(buffer, LIF);
1696 
1697 // Validate basic Volume headers
1698  if( !lif_check_volume(LIF) )
1699  {
1700  if(debuglevel & LIF_DEBUG)
1701  printf("lif_open_volume:[%s] error volume validate failed\n", LIF->name);
1702  lif_closedir(LIF);
1703  return(NULL);
1704  }
1705 
1706 // Initialize remaining LIF headers
1707  LIF->filestart = LIF->VOL.DirStartSector + LIF->VOL.DirSectors;
1708  LIF->filesectors = LIF->sectors - LIF->filestart;
1709  LIF->freesectors = LIF->filesectors;
1710  LIF->usedsectors = 0;
1711  LIF->purged = 0;
1712  LIF->files = 0;
1713  LIF->dirindex = 0;
1714  LIF->EOFindex = 0;
1715 
1716  if( lif_updatefree(LIF) == NULL)
1717  {
1718  if(debuglevel & LIF_DEBUG)
1719  printf("lif_open_volume:[%s] error directory check failed\n", LIF->name);
1720  lif_closedir(LIF);
1721  return(NULL);
1722  }
1723 
1724  if(debuglevel &LIF_DEBUG)
1725  lif_dump_vol(LIF, "Volume Listing");
1726  return( LIF );
1727 }
1728 
1729 
1733 MEMSPACE
1734 void lif_dir(char *lifimagename)
1735 {
1736  long bytes;
1737  lif_t *LIF;
1738  int index = 0;
1739  char *vol;
1740 
1741  int warn = ' ';
1742 
1743  LIF = lif_open_volume(lifimagename,"rb+");
1744  if(LIF == NULL)
1745  return;
1746 
1747  vol = (char *)LIF->VOL.Label;
1748  if(!vol[0])
1749  vol = "<EMPTY>";
1750 
1751  printf("Volume:[%s] Date:[%s]\n", vol, lif_lifbcd2timestr(LIF->VOL.date));
1752 
1753  printf("NAME TYPE START SECTOR SIZE RECSIZE DATE\n");
1754  while(1)
1755  {
1756 
1757  if(!lif_readdirindex(LIF,index))
1758  break;
1759 
1760  if(LIF->DIR.FileType == 0xffff)
1761  break;
1762 
1763  bytes = (LIF->DIR.FileSectors * (long)LIF_SECTOR_SIZE);
1764 
1765  if((LIF->DIR.FileType & 0xFFFC) == 0xE010)
1766  {
1767  if(LIF->DIR.FileBytes && lif_bytes2sectors(LIF->DIR.FileBytes) == LIF->DIR.FileSectors)
1768  {
1769  bytes = LIF->DIR.FileBytes;
1770  }
1771  else
1772  {
1773  warn = '!';
1774  if(debuglevel & LIF_DEBUG)
1775  {
1776  printf("LIF Directory:[%s] warning FileBytes:%ld as sectors:%ld != FileSectors:%ld\n",
1777  LIF->DIR.filename,
1778  (long) LIF->DIR.FileBytes,
1779  (long) lif_bytes2sectors(LIF->DIR.FileBytes),
1780  (long) LIF->DIR.FileSectors);
1781  }
1782  }
1783  }
1784 
1785 // name type start size
1786  printf("%-10s %04Xh %8lXh %9ld%c %4d %s\n",
1787  (LIF->DIR.FileType ? (char *)LIF->DIR.filename : "<PURGED>"),
1788  (int)LIF->DIR.FileType,
1789  (long)LIF->DIR.FileStartSector,
1790  (long)bytes,
1791  warn,
1792  (int)LIF->DIR.SectorSize,
1793  lif_lifbcd2timestr(LIF->DIR.date));
1794 
1795  ++index;
1796  }
1797 
1798  printf("\n");
1799  printf("%8ld Files\n", (long)LIF->files);
1800  printf("%8ld Purged\n", (long)LIF->purged);
1801  printf("%8ld Dir start\n", (long)LIF->VOL.DirStartSector);
1802  printf("%8ld Dir sectors\n", (long)LIF->VOL.DirSectors);
1803  printf("%8ld Used sectors\n", (long)LIF->usedsectors);
1804  printf("%8ld Free sectors\n", (long)LIF->freesectors);
1805 
1806  lif_closedir(LIF);
1807 }
1808 
1809 
1814 MEMSPACE
1815 int lif_find_file(lif_t *LIF, char *liflabel)
1816 {
1817  int index;
1818 
1819  if( !lif_checkname(liflabel) )
1820  {
1821  if(debuglevel & LIF_DEBUG)
1822  printf("lif_find_file:[%s] invalid characters\n", liflabel);
1823  return(-1);
1824  }
1825  if(strlen(liflabel) > 10)
1826  {
1827  if(debuglevel & LIF_DEBUG)
1828  printf("lif_find_file:[%s] liflabel too big\n", liflabel);
1829  return(-1);
1830  }
1831 
1832  if(LIF == NULL)
1833  return(-1);
1834 
1835  index = 0;
1836  while(1)
1837  {
1838  if(!lif_readdirindex(LIF,index))
1839  return(-1);
1840 
1841  if(LIF->DIR.FileType == 0xffff)
1842  return(-1);
1843 
1844  if( LIF->DIR.FileType && (strcasecmp((char *)LIF->DIR.filename,liflabel) == 0) )
1845  break;
1846  ++index;
1847  }
1848  return(index);
1849 }
1850 
1851 
1871 MEMSPACE
1877 int lif_e010_pad_sector(long offset, uint8_t *wbuf)
1878 {
1879  int ind;
1880  int pos,rem;
1881 
1882 // Compute the current offset in this sector
1883  pos = (offset % (long)LIF_SECTOR_SIZE);
1884  if(!pos)
1885  return(0);
1886 
1887 // Number of bytes free in this sector
1888  rem = (long)LIF_SECTOR_SIZE - (long)pos;
1889 
1890 // Bytes written to wbuf
1891  ind = 0;
1892 // PAD
1893  wbuf[ind++] = 0xEF;
1894  while(ind<rem)
1895  wbuf[ind++] = 0xff;
1896 
1897  pos = (offset + (long) ind) % (long) LIF_SECTOR_SIZE;
1898 // NEW SECTOR
1899 // Debugging make sure we are at sector boundry
1900  if(pos)
1901  {
1902  if(debuglevel & LIF_DEBUG)
1903  printf("lif_e010_dap_sector: expected sector boundry: offset:%d\n", (int) pos);
1904  return(-1);
1905  }
1906  return(ind);
1907 }
1908 
1909 
1916 MEMSPACE
1917 int lif_ascii_string_to_e010(char *str, long offset, uint8_t *wbuf)
1918 {
1919  int ind;
1920  int len;
1921  int pos,rem;
1922 
1923 // String size
1924  len = strlen(str);
1925 
1926 // Output buffer index
1927  ind = 0;
1928 
1929 // Compute the current offset in this sector
1930  pos = (offset % (long) LIF_SECTOR_SIZE);
1931 // Number of bytes free in this sector
1932  rem = (long) LIF_SECTOR_SIZE - (long) pos;
1933 
1938  if(rem < 6)
1939  {
1940  ind = lif_e010_pad_sector(offset, wbuf);
1941  if(ind < 0)
1942  return(ind);
1943 
1944 // Compute the current offset in this sector
1945  pos = ((offset + ind) % (long) LIF_SECTOR_SIZE);
1946 // Number of bytes free in this sector
1947  rem = (long) LIF_SECTOR_SIZE - pos;
1948  }
1949 
1950 // Note: IMPORTANT we have >= 6 bytes!!!
1951 
1952 // Do not have to split, there is enough room
1953  if(rem >= (3 + len))
1954  {
1955 
1956 // Write string in new sector
1957 // The full string + header will fit
1958  wbuf[ind++] = 0xDF;
1959  wbuf[ind++] = len & 0xff;
1960  wbuf[ind++] = (len >> 8) & 0xff;
1961 // Write string
1962  while(*str)
1963  wbuf[ind++] = *str++;
1964  }
1965  else /* No enough room split string */
1966  {
1967 // Split strings need at least 6 header bytes
1968 // We KNOW that there are at least 6 bytes in this sector
1969 
1970 // CURRENT SECTOR
1971 // String spans a sector , so split the string
1972 
1973 // 1st Split string header
1974  wbuf[ind++] = 0xCF;
1975  wbuf[ind++] = len & 0xff;
1976  wbuf[ind++] = (len >>8) & 0xff;
1977 // Write as much of the string as we can in this sector
1978  while(*str && ind<rem)
1979  wbuf[ind++] = *str++;
1980 
1981 // NEW SECTOR
1982 // Debugging make sure we are at sector boundry
1983  if(((offset + (long) ind) % (long) LIF_SECTOR_SIZE))
1984  {
1985  if(debuglevel & LIF_DEBUG)
1986  printf("Expected sector boundry, offset:%d\n", (int) ((offset + ind) % LIF_SECTOR_SIZE) );
1987  return(-1);
1988  }
1989 
1990 // Update remining string length
1991  len = strlen(str);
1992 // 2nd Split string header
1993  wbuf[ind++] = 0x6F;
1994  wbuf[ind++] = (len & 0xff);
1995  wbuf[ind++] = (len>>8) & 0xff;
1996 // Write string
1997  while(*str)
1998  wbuf[ind++] = *str++;
1999  }
2000 
2001  return(ind);
2002 }
2003 
2004 
2011 MEMSPACE
2012 long lif_add_ascii_file_as_e010_wrapper(lif_t *LIF, uint32_t offset, char *username)
2013 {
2014  long bytes;
2015  int count;
2016  int size;
2017  int len;
2018  FILE *fi;
2019 
2020 // strings are limited to less then this
2021  char str[LIF_SECTOR_SIZE+1];
2022 // output buffer must be larger then a single sectors because of either headers or padding
2023  uint8_t obuf[LIF_SECTOR_SIZE*2];
2024 
2025  fi = lif_open(username, "rb");
2026  if(fi == NULL)
2027  return(-1);
2028 
2029  bytes = 0;
2030  count = 0;
2031 
2032 // Read user file and write LIF records
2033 // reserve 3 + LIF header bytes + 1 (EOS)
2034  while( fgets((char *)str,(int)sizeof(str) - 4, fi) != NULL )
2035  {
2036  trim_tail((char *)str);
2037 
2038  strcat((char *)str,"\r"); // HP85 lines end with "\r"
2039 
2040  size = lif_ascii_string_to_e010(str, offset, obuf);
2041 // Write string
2042 // Now Write string
2043  if(LIF)
2044  {
2045  len = lif_write(LIF, obuf, offset, size);
2046  if(len < size)
2047  {
2048  fclose(fi);
2049  return(-1);
2050  }
2051  }
2052 
2053  offset += size;
2054  bytes += size;
2055  count += size;
2056 
2057  if(count > 256)
2058  {
2059  count = 0;
2060  if(LIF)
2061  printf("\tWrote: %8ld\r", (long)bytes);
2062  }
2063  }
2064 
2065  fclose(fi);
2066 
2067 // Write EOF
2068  str[0] = 0;
2069 // We only want to return the count of bytes in the file NOT the padding at the end
2070 // Write EOF string with padding
2071  size = lif_ascii_string_to_e010(str, offset,obuf);
2072 
2073  if(LIF)
2074  {
2075  printf("\tWrote: %8ld\r", (long)bytes);
2076  len = lif_write(LIF, obuf, offset, size);
2077  if(len < size)
2078  return(-1);
2079 
2080  }
2081 
2082  offset += size;
2083  bytes += size;
2084 
2085 // PAD
2086  size = lif_e010_pad_sector(offset, obuf);
2087  if(LIF)
2088  {
2089  len = lif_write(LIF, obuf, offset, size);
2090  if(len < size)
2091  return(-1);
2092  }
2093 
2094  if(LIF)
2095  printf("\tWrote: %8ld\r",(long)bytes);
2096 
2097  return(bytes);
2098 }
2099 
2100 
2108 MEMSPACE
2109 long lif_add_ascii_file_as_e010(char *lifimagename, char *lifname, char *userfile)
2110 {
2111  long bytes;
2112  long sectors;
2113  long offset;
2114  int index;
2115  lif_t *LIF;
2116  stat_t st, *sp;
2117 
2118  if(!*lifimagename)
2119  {
2120  printf("lif_add_ascii_file_as_e010: lifimagename is empty\n");
2121  return(-1);
2122  }
2123  if(!*lifname)
2124  {
2125  printf("lif_add_ascii_file_as_e010: lifname is empty\n");
2126  return(-1);
2127  }
2128  if(!*userfile)
2129  {
2130  printf("lif_add_ascii_file_as_e010: userfile is empty\n");
2131  return(-1);
2132  }
2133 
2134 //Get size and date info
2135  sp = lif_stat(userfile, (stat_t *)&st);
2136  if(!sp)
2137  {
2138  printf("lif_add_ascii_file_as_e010: userfile not found\n");
2139  return(-1);
2140  }
2141 
2142  if(debuglevel & LIF_DEBUG)
2143  printf("LIF image:[%s], LIF name:[%s], user file:[%s]\n",
2144  lifimagename, lifname, userfile);
2145 
2146 // Find out how big converted file will be
2147  bytes = lif_add_ascii_file_as_e010_wrapper(NULL,0,userfile);
2148  sectors = lif_bytes2sectors(bytes);
2149 
2150  LIF = lif_open_volume(lifimagename,"r+");
2151  if(LIF == NULL)
2152  return(-1);
2153 
2154 // Now find free record
2155  index = lif_newdir(LIF, sectors);
2156  if(index == -1)
2157  {
2158  printf("LIF image:[%s], not enough free space for:[%s]\n",
2159  lifimagename, userfile);
2160  lif_closedir(LIF);
2161  return(-1);
2162  }
2163 
2164 // Initialize the free directory entry
2165  lif_fixname(LIF->DIR.filename, lifname,10);
2166  LIF->DIR.FileType = 0xe010; // 10
2167  lif_time2lifbcd(sp->st_mtime, LIF->DIR.date);
2168 
2169  LIF->DIR.VolNumber = 0x8001; // 26
2170  LIF->DIR.FileBytes = bytes; // 28
2171  LIF->DIR.SectorSize = 0x100; // 30
2172  offset = LIF->DIR.FileStartSector * (long) LIF_SECTOR_SIZE;
2173 
2174  if(debuglevel & LIF_DEBUG)
2175  lif_dump_vol(LIF,"lif_after lif_newdir");
2176 
2177 // Write converted file into free space first
2178  bytes = lif_add_ascii_file_as_e010_wrapper(LIF,offset,userfile);
2179 
2180  if(debuglevel & LIF_DEBUG)
2181  {
2182  printf("New Directory Information AFTER write\n");
2183  printf("Name: %s\n", LIF->DIR.filename);
2184  printf("Index: %4d\n", (int)index);
2185  printf("First Sector: %4lxH\n", (long) LIF->DIR.FileStartSector);
2186  printf("File Sectors: %4lxH\n", (long)LIF->DIR.FileSectors);
2187  }
2188 
2189 // Write directory record
2190 // Note: lif_newdir alrwady did the new EOF
2191  if( !lif_writedirindex(LIF,index))
2192  {
2193  lif_closedir(LIF);
2194  return(-1);
2195  }
2196 
2197  lif_closedir(LIF);
2198 
2199  printf("\tWrote: %8ld\n", bytes);
2200 
2201 // Return file size
2202  return(bytes);
2203 }
2204 
2205 
2212 MEMSPACE
2213 int lif_extract_e010_as_ascii(char *lifimagename, char *lifname, char *username)
2214 {
2215  lif_t *LIF;
2216  uint32_t start, end; // sectors
2217  long offset, bytes; // bytes
2218  int index;
2219  int i, len,size;
2220  int status = 1;
2221  int done = 0;
2222 
2223  time_t t;
2224 
2225  int ind,wind;
2226  FILE *fo;
2227 
2228 // read buffer
2229  uint8_t buf[LIF_SECTOR_SIZE+4];
2230 // Write buffer, FYI: will ALWAYS be smaller then the read data buffer
2231  uint8_t wbuf[LIF_SECTOR_SIZE+4];
2232 
2233  LIF = lif_open_volume(lifimagename,"r");
2234  if(LIF == NULL)
2235  {
2236  printf("LIF image not found:%s\n", lifimagename);
2237  return(0);
2238  }
2239 
2240  index = lif_find_file(LIF, lifname);
2241  if(index == -1)
2242  {
2243  printf("LIF File not found:%s\n", lifname);
2244  lif_closedir(LIF);
2245  return(0);
2246  }
2247 
2248  if((LIF->DIR.FileType & 0xFFFC) != 0xE010)
2249  {
2250  printf("File %s has wrong type:[%04XH] expected 0xE010..0xE013\n", username, (int) LIF->DIR.FileType);
2251  lif_closedir(LIF);
2252  return(0);
2253  }
2254 
2255  start = LIF->DIR.FileStartSector;
2256  end = start + LIF->DIR.FileSectors;
2257 
2258  t = lif_lifbcd2time(LIF->DIR.date);
2259 
2260  offset = start * (long) LIF_SECTOR_SIZE;
2261 
2262  fo = lif_open(username,"wb");
2263  if(fo == NULL)
2264  {
2265  lif_closedir(LIF);
2266  return(0);
2267  }
2268 
2269  printf("Extracting: %s\n", username);
2270 
2271  bytes = 0;
2272  wind = 0;
2273  ind = 0;
2274 
2275  while(lif_bytes2sectors(offset) <= end)
2276  {
2277 // LIF images are always multiples of LIF_SECTOR_SIZE
2278  size = lif_read(LIF, buf, offset, LIF_SECTOR_SIZE);
2279  if(size < LIF_SECTOR_SIZE)
2280  {
2281  status = 0;
2282  break;
2283  }
2284 
2285  ind = 0;
2286  while(ind < LIF_SECTOR_SIZE && !done)
2287  {
2288  if(buf[ind] == 0xDF || buf[ind] == 0xCF || buf[ind] == 0x6F)
2289  {
2290  ++ind;
2291  len = buf[ind++] & 0xff;
2292  len |= ((buf[ind++] & 0xff) <<8);
2293 // EOF ?
2294  if(len == 0)
2295  {
2296  done = 1;
2297  break;
2298  }
2299  if(len >= LIF_SECTOR_SIZE)
2300  {
2301  printf("lif_extract_e010_as_ascii: string too big size = %d\n", (int)len);
2302  status = 0;
2303  done = 1;
2304  break;
2305  }
2306  }
2307  else if(buf[ind] == 0xEF)
2308  {
2309 // skip remaining bytes in sector
2310  ind = 0;
2311  break;
2312  }
2313  else
2314  {
2315  printf("lif_extract_e010_as_ascii: unexpected control byte:[%02XH] @ offset: %8lx, ind:%02XH\n", (int)buf[ind], offset, (int)ind);
2316  status = 0;
2317  done = 1;
2318  break;
2319  }
2320 // write string
2321  for(i=0;i <len && ind < LIF_SECTOR_SIZE;++i)
2322  {
2323  if(buf[ind] == '\r' && i == len-1)
2324  {
2325  wbuf[wind++] = '\n';
2326  ++ind;
2327  break;
2328  }
2329  else
2330  {
2331  wbuf[wind++] = buf[ind++];
2332  }
2333 
2334  if(wind >= LIF_SECTOR_SIZE)
2335  {
2336  size = fwrite(wbuf,1,wind,fo);
2337  if(size < wind)
2338  {
2339  printf("lif_extract_e010_as_ascii: write error\n");
2340  status = 0;
2341  done = 1;
2342  break;
2343  }
2344  bytes += size;
2345  printf("\tWrote: %8ld\r", bytes);
2346  wind = 0;
2347  }
2348 
2349  } // for(i=0;i <len && ind < LIF_SECTOR_SIZE;++i)
2350 
2351  } // while(ind < LIF_SECTOR_SIZE && status)
2352 
2353  offset += (long) LIF_SECTOR_SIZE;
2354 
2355  } // while(offset <= end)
2356 
2357  lif_closedir(LIF);
2358 // Flush any remaining bytes
2359  if(wind)
2360  {
2361  size = fwrite(wbuf,1,wind,fo);
2362  if(size < wind)
2363  {
2364  printf("lif_extract_e010_as_ascii: write error\n");
2365  status = 0;
2366  }
2367  bytes += size;
2368  }
2369  fclose(fo);
2370  if(t)
2371  {
2372  struct utimbuf times;
2373  times.modtime = t;
2374  times.actime = t;
2375  utime(username, (struct utimbuf *) &times);
2376  }
2377  sync();
2378  printf("\tWrote: %8ld\n", bytes);
2379  return(status);
2380 }
2381 
2382 
2389 MEMSPACE
2390 int lif_extract_lif_as_lif(char *lifimagename, char *lifname, char *username)
2391 {
2392 // Master image lif_t structure
2393  lif_t *LIF;
2394  lif_t *ULIF;
2395 
2396  long offset, uoffset, bytes;
2397  int index;
2398  int i, size;
2399  int sectors;
2400 
2401  uint8_t buf[LIF_SECTOR_SIZE+4];
2402 
2403  LIF = lif_open_volume(lifimagename,"r");
2404  if(LIF == NULL)
2405  {
2406  printf("LIF image not found:%s\n", lifimagename);
2407  return(0);
2408  }
2409 
2410  index = lif_find_file(LIF, lifname);
2411  if(index == -1)
2412  {
2413  printf("File not found:%s\n", lifname);
2414  lif_closedir(LIF);
2415  return(0);
2416  }
2417 
2418  sectors = LIF->DIR.FileSectors;
2419 
2420 //Initialize the user file lif_t structure
2421  ULIF = lif_create_volume(username, "HFSLIF",1,1,sectors);
2422  if(ULIF == NULL)
2423  {
2424  lif_closedir(LIF);
2425  return(0);
2426  }
2427 
2428 // Only the start sector changes
2429 
2430 // Copy directory record
2431  ULIF->DIR = LIF->DIR;
2432 
2433  ULIF->DIR.FileStartSector = 2;
2434  ULIF->filesectors = LIF->DIR.FileSectors;
2435 
2436  if( !lif_writedirindex(ULIF,0))
2437  {
2438  lif_closedir(LIF);
2439  lif_closedir(ULIF);
2440  return(0);
2441  }
2442  if( !lif_writedirEOF(ULIF,1) )
2443  {
2444  lif_closedir(LIF);
2445  lif_closedir(ULIF);
2446  return(0);
2447  }
2448 
2449  uoffset = ULIF->filestart * (long) LIF_SECTOR_SIZE;
2450 
2451  offset = LIF->DIR.FileStartSector * (long) LIF_SECTOR_SIZE;
2452 
2453  bytes = uoffset;
2454 
2455  for(i=0;i<(int)LIF->DIR.FileSectors;++i)
2456  {
2457  size = lif_read(LIF, buf, offset,LIF_SECTOR_SIZE);
2458  if(size < LIF_SECTOR_SIZE)
2459  {
2460  lif_closedir(LIF);
2461  lif_closedir(ULIF);
2462  return(0);
2463  }
2464 
2465  lif_write(ULIF,buf,uoffset,LIF_SECTOR_SIZE);
2466  if(size < LIF_SECTOR_SIZE)
2467  {
2468  lif_closedir(LIF);
2469  lif_closedir(ULIF);
2470  return(0);
2471  }
2472  bytes += size;
2473  offset += size;
2474  uoffset += size;
2475  printf("\tWrote: %8ld\r", bytes);
2476  }
2477  lif_closedir(LIF);
2478  lif_closedir(ULIF);
2479  printf("\tWrote: %8ld\n", bytes);
2480  return(1);
2481 }
2482 
2483 
2490 MEMSPACE
2491 long lif_add_lif_file(char *lifimagename, char *lifname, char *userfile)
2492 {
2493 // Master image lif_t structure
2494  lif_t *LIF;
2495  lif_t *ULIF;
2496  int index = 0;
2497  long offset, uoffset, start, bytes;
2498  int i, size;
2499 
2500  uint8_t buf[LIF_SECTOR_SIZE+4];
2501 
2502  if(!*lifimagename)
2503  {
2504  printf("lif_add: lifimagename is empty\n");
2505  return(-1);
2506  }
2507  if(!*lifname)
2508  {
2509  printf("lif_add: lifname is empty\n");
2510  return(-1);
2511  }
2512  if(!*userfile)
2513  {
2514  printf("lif_add: userfile is empty\n");
2515  return(-1);
2516  }
2517 
2518  if(debuglevel & LIF_DEBUG)
2519  printf("LIF image:[%s], LIF name:[%s], user file:[%s]\n",
2520  lifimagename, lifname, userfile);
2521 
2522 // open userfile as LIF image
2523  ULIF = lif_open_volume(userfile,"rb+");
2524  if(ULIF == NULL)
2525  return(-1);
2526 
2527 // find lif file in user image
2528  index = lif_find_file(ULIF, lifname);
2529  if(index == -1)
2530  {
2531  printf("File not found:%s\n", lifname);
2532  lif_closedir(ULIF);
2533  return(0);
2534  }
2535 
2536  LIF = lif_open_volume(lifimagename,"rb+");
2537  if(LIF == NULL)
2538  return(-1);
2539 
2540 // Now find a new free record that is big enough
2541  index = lif_newdir(LIF, ULIF->DIR.FileSectors);
2542  if(index == -1)
2543  {
2544  printf("LIF image:[%s], not enough free space for:[%s]\n",
2545  lifimagename, userfile);
2546  lif_closedir(LIF);
2547  lif_closedir(ULIF);
2548  return(-1);
2549  }
2550 
2551 // Save start sector
2552  start = LIF->DIR.FileStartSector;
2553 
2554 // Copy user image directory record to master image directory record
2555  LIF->DIR = ULIF->DIR;
2556 
2557 // Restore FileStartSector
2558  LIF->DIR.FileStartSector = start;
2559 
2560 // Master lif image file start in bytes
2561  offset = LIF->DIR.FileStartSector * (long) LIF_SECTOR_SIZE;
2562 // User lif image file start in bytes
2563  uoffset = ULIF->DIR.FileStartSector * (long) LIF_SECTOR_SIZE;
2564  bytes = 0;
2565 // Copy file data
2566  for(i=0;i<(int)LIF->DIR.FileSectors;++i)
2567  {
2568 // Read
2569  size = lif_read(ULIF, buf, uoffset, LIF_SECTOR_SIZE);
2570  if(size < LIF_SECTOR_SIZE)
2571  {
2572  lif_closedir(LIF);
2573  lif_closedir(ULIF);
2574  return(-1);
2575  }
2576 
2577 // Write
2578  size = lif_write(LIF, buf, offset, LIF_SECTOR_SIZE);
2579  if(size < LIF_SECTOR_SIZE)
2580  {
2581  lif_closedir(LIF);
2582  lif_closedir(ULIF);
2583  return(-1);
2584  }
2585  offset += (long) LIF_SECTOR_SIZE;
2586  uoffset += (long) LIF_SECTOR_SIZE;
2587  bytes += (long) LIF_SECTOR_SIZE;
2588  printf("\tWrote: %8ld\r", bytes);
2589  }
2590  lif_closedir(ULIF);
2591 
2592 // Write directory record
2593  if( !lif_writedirindex(LIF,index))
2594  {
2595  lif_closedir(LIF);
2596  return(-1);
2597  }
2598  lif_closedir(LIF);
2599  printf("\tWrote: %8ld\n", bytes);
2600  return(bytes);
2601 }
2602 
2603 
2608 MEMSPACE
2609 int lif_del_file(char *lifimagename, char *lifname)
2610 {
2611  lif_t *LIF;
2612  int index;
2613 
2614  if(!*lifimagename)
2615  {
2616  printf("lif_del_file: lifimagename is empty\n");
2617  return(-1);
2618  }
2619  if(!*lifname)
2620  {
2621  printf("lif_del_file: lifname is empty\n");
2622  return(-1);
2623  }
2624  if(debuglevel & LIF_DEBUG)
2625  printf("LIF image:[%s], LIF name:[%s]\n",
2626  lifimagename, lifname);
2627 
2628  LIF = lif_open_volume(lifimagename,"rb+");
2629  if(LIF == NULL)
2630  return(-1);
2631 
2632 // Now find file record
2633  index = lif_find_file(LIF, lifname);
2634  if(index == -1)
2635  {
2636  lif_closedir(LIF);
2637  printf("LIF image:[%s] lif name:[%s] not found\n", lifimagename, lifname);
2638  return(0);
2639  }
2640 
2641 // IF the next record is EOF then update EOF
2642  if(index >= LIF->EOFindex-1)
2643  LIF->DIR.FileType = 0xffff;
2644  else
2645  LIF->DIR.FileType = 0;
2646 
2647 // re-Write directory record
2648  if( !lif_writedirindex(LIF,index) )
2649  {
2650  lif_closedir(LIF);
2651  return(-1);
2652  }
2653 
2654  lif_updatefree(LIF);
2655 
2656  lif_closedir(LIF);
2657  printf("Deleted: %10s\n", lifname);
2658 
2659  return(1);
2660 }
2661 
2662 
2668 MEMSPACE
2669 int lif_rename_file(char *lifimagename, char *oldlifname, char *newlifname)
2670 {
2671  int index;
2672  lif_t *LIF;
2673 
2674  if(!*lifimagename)
2675  {
2676  printf("lif_rename_file: lifimagename is empty\n");
2677  return(-1);
2678  }
2679  if(!*oldlifname)
2680  {
2681  printf("lif_rename_file: old lifname is empty\n");
2682  return(-1);
2683  }
2684  if(!*newlifname)
2685  {
2686  printf("lif_rename_file: new lifname is empty\n");
2687  return(-1);
2688  }
2689 
2690  if(!lif_checkname(newlifname))
2691  {
2692  printf("lif_rename_file: new lifname contains bad characters\n");
2693  return(-1);
2694 
2695  }
2696 
2697  LIF = lif_open_volume(lifimagename,"rb+");
2698  if(LIF == NULL)
2699  return(-1);
2700 
2701 // Now find file record
2702  index = lif_find_file(LIF, oldlifname);
2703  if(index == -1)
2704  {
2705  printf("lif_rename:[%s] lif name:[%s] not found\n", lifimagename, oldlifname);
2706  lif_closedir(LIF);
2707  return(0);
2708  }
2709  lif_fixname(LIF->DIR.filename, newlifname, 10);
2710 
2711 // re-Write directory record
2712  if( !lif_writedirindex(LIF,index))
2713  {
2714  lif_closedir(LIF);
2715  return(-1);
2716  }
2717  printf("renamed: %10s to %10s\n", oldlifname,newlifname);
2718 
2719  lif_closedir(LIF);
2720 
2721  return(1);
2722 }
2723 
2724 
2729 MEMSPACE
2730 int lif_rename_volume(char *lifimagename, char *volname)
2731 {
2732  lif_t *LIF;
2733  uint8_t buffer[LIF_SECTOR_SIZE+1];
2734 
2735 
2736  if(!*lifimagename)
2737  {
2738  printf("lif_rename_file: lifimagename is empty\n");
2739  return(-1);
2740  }
2741 
2742  if(!*volname)
2743  {
2744  printf("lif_rename_file: new Volume name empty\n");
2745  return(-1);
2746  }
2747 
2748  if(!lif_checkname(volname))
2749  {
2750  printf("lif_rename_file: new lifname contains bad characters\n");
2751  return(-1);
2752 
2753  }
2754 
2755  LIF = lif_open_volume(lifimagename,"rb+");
2756  if(LIF == NULL)
2757  return(-1);
2758 
2759  // Update volume name
2760  lif_fixname(LIF->VOL.Label, volname, 6);
2761 
2762 // Validate basic Volume headers
2763  if( !lif_check_volume(LIF) )
2764  {
2765  if(debuglevel & LIF_DEBUG)
2766  printf("lif_rename_volume:[%s] error volume validate failed\n", LIF->name);
2767  lif_closedir(LIF);
2768  return(-1);
2769  }
2770 
2771  // clear sector buffer
2772  memset(buffer,0,LIF_SECTOR_SIZE);
2773  // created update volume header sector from LIF structure
2774  lif_vol2str(LIF,buffer);
2775 
2776 // Volume header must be it least one sector
2777  if( lif_write(LIF, buffer, 0, LIF_SECTOR_SIZE) < LIF_SECTOR_SIZE)
2778  {
2779  if(debuglevel & LIF_DEBUG)
2780  printf("lif_rename_volume:[%s] error write volume header failed\n", LIF->name);
2781  lif_closedir(LIF);
2782  return(-1);
2783  }
2784 
2785  lif_close_volume(LIF);
2786 
2787  return(1);
2788 }
2789 
2797 MEMSPACE
2798 long lif_create_image(char *lifimagename, char *liflabel, uint32_t dirsectors, uint32_t sectors)
2799 {
2800  uint32_t dirstart,filestart,filesectors,end;
2801  lif_t *LIF;
2802 
2803  if(!*lifimagename)
2804  {
2805  printf("lif_create_image: lifimagename is empty\n");
2806  return(-1);
2807  }
2808  if(!*liflabel)
2809  {
2810  printf("lif_create_image: liflabel is empty\n");
2811  return(-1);
2812  }
2813  if(!dirsectors)
2814  {
2815  printf("lif_create_image: dirsectors is 0\n");
2816  return(-1);
2817  }
2818  if(!sectors)
2819  {
2820  printf("lif_create_image: sectors is 0\n");
2821  return(-1);
2822  }
2823 
2824  dirstart = 2;
2825  filestart = dirstart + dirsectors;
2826  filesectors = sectors - filestart;
2827  end = filestart + filesectors;
2828 
2829  LIF = lif_create_volume(lifimagename, liflabel, dirstart, dirsectors, filesectors);
2830  if(LIF == NULL)
2831  return(-1);
2832  lif_close_volume(LIF);
2833 
2834  printf("\tFormatting: wrote %ld sectors\n", (long)end);
2835  return(end);
2836 }
hpdir
hpdir_t hpdir
hpdir.ini file processing
Definition: drives.c:48
stat::st_size
off_t st_size
Definition: posix.h:114
fopen
MEMSPACE FILE * fopen(const char *path, const char *mode)
POSIX Open a file with path name and ascii file mode string.
Definition: posix.c:801
lif_lifbcd2time
MEMSPACE time_t lif_lifbcd2time(uint8_t *bcd)
convert BCD date into time_t value The BCD year is only the lower 2 digits of the year So We assume t...
Definition: lifutils.c:724
lif_read
MEMSPACE long lif_read(lif_t *LIF, void *buf, long offset, int bytes)
Read data from a LIF image Displays message on errors.
Definition: lifutils.c:396
stat
POSIX stat structure.
Definition: posix.h:105
lif_stat
MEMSPACE stat_t * lif_stat(char *name, stat_t *p)
Stat a file Displays message on errors.
Definition: lifutils.c:357
stat::st_mtime
time_t st_mtime
Definition: posix.h:118
lif_closedir
MEMSPACE void lif_closedir(lif_t *LIF)
Close LIF directory clear and free lif_t structure.
Definition: lifutils.c:1273
printf
MEMSPACE int printf(const char *format,...)
lif_checkname
MEMSPACE int lif_checkname(char *name)
Check volume LIF name or directory name is valid.
Definition: lifutils.c:507
lif_vol_clear
MEMSPACE void lif_vol_clear(lif_t *LIF)
Clear VOL part of LIF structure.
Definition: lifutils.c:826
lif_t::filestart
uint32_t filestart
Definition: lifutils.h:120
lifvol_t::System3000LIFid
uint16_t System3000LIFid
Definition: lifutils.h:63
tm::tm_min
int tm_min
Definition: time.h:43
lif_rename_file
MEMSPACE int lif_rename_file(char *lifimagename, char *oldlifname, char *newlifname)
Rename LIF file in LIF image.
Definition: lifutils.c:2669
lifvol_t::LIFid
uint16_t LIFid
Definition: lifutils.h:60
lif_help
MEMSPACE void lif_help(int full)
Help Menu for User invoked GPIB functions and tasks See: int gpib_tests(char *str)
Definition: lifutils.c:111
atol
MEMSPACE long atol(const char *str)
Convert ASCII string to number in base 10.
Definition: mathio.c:299
lifdir_t::FileStartSector
uint32_t FileStartSector
Definition: lifutils.h:101
td02lif.h
MATCHI
MEMSPACE int MATCHI(char *str, char *pat)
Compare two strings without case.
Definition: parsing.c:183
lif_extract_e010_as_ascii
MEMSPACE int lif_extract_e010_as_ascii(char *lifimagename, char *lifname, char *username)
Extract E010 type file from LIF image and save as user ASCII file.
Definition: lifutils.c:2213
sync
MEMSPACE void sync(void)
POSIX Sync all pending file changes and metadata on ALL files.
Definition: posix.c:1091
lif_t::DIR
lifdir_t DIR
Definition: lifutils.h:129
lif_t
Master LIF data structure Contains image file name Volume Structure Current Directory Entry read/writ...
Definition: lifutils.h:114
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
lifdir_t::FileSectors
uint32_t FileSectors
Definition: lifutils.h:102
timegm
MEMSPACE time_t timegm(tm_t *t)
Convert tm_t structure as GMT time into GMT seconds since 1900. All calculactions are in GMT regardle...
Definition: time.c:348
lif_extract_lif_as_lif
MEMSPACE int lif_extract_lif_as_lif(char *lifimagename, char *lifname, char *username)
Extract a file from LIF image entry as standalone LIF image.
Definition: lifutils.c:2390
lif_updatefree
MEMSPACE lif_t * lif_updatefree(lif_t *LIF)
Update free space @parameter[in] *LIF: LIF structure.
Definition: lifutils.c:1430
LIF_SECTOR_SIZE
#define LIF_SECTOR_SIZE
Depends on how much free ram we have.
Definition: lifutils.h:26
strcat
MEMSPACE WEAK_ATR char * strcat(char *dest, const char *src)
Append string.
Definition: stringsup.c:199
safecalloc
void * safecalloc(int size, int elements)
Safe Alloc - Display Error message if Calloc fails.
Definition: ram.c:122
lifvol_t::zero1
uint16_t zero1
Definition: lifutils.h:64
lif_add_lif_file
MEMSPACE long lif_add_lif_file(char *lifimagename, char *lifname, char *userfile)
Add LIF file from another LIF image.
Definition: lifutils.c:2491
fgets
MEMSPACE char * fgets(char *str, int size, FILE *stream)
get a string from stdin See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:432
lif_str2dir
MEMSPACE void lif_str2dir(uint8_t *B, lif_t *LIF)
Convert byte vector into byte vector.
Definition: lifutils.c:649
B2V_LSB
uint32_t B2V_LSB(uint8_t *B, int index, int size)
Convert a byte array into a value bytes are LSB ... MSB order.
Definition: vector.c:78
strcasecmp
MEMSPACE int WEAK_ATR strcasecmp(const char *str, const char *pat)
Compare two strings without case.
Definition: stringsup.c:328
lif_stralloc
MEMSPACE char * lif_stralloc(char *str)
Allocate and copy a string Displays message on errors.
Definition: lifutils.c:322
asctime_r
MEMSPACE char * asctime_r(tm_t *t, char *buf)
Convert tm_t *t structure into POSIX asctime() ASCII string *buf.
Definition: time.c:368
lif_writedirindex
MEMSPACE int lif_writedirindex(lif_t *LIF, int index)
Write LIF drectory record number N.
Definition: lifutils.c:1349
tm::tm_sec
int tm_sec
Definition: time.h:42
lifvol_t::sides
uint32_t sides
Definition: lifutils.h:69
fwrite
MEMSPACE size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
POSIX write nmemb elements from buf, size bytes each, to the stream fd.
Definition: posix.c:890
safefree
void safefree(void *p)
Safe free - Only free a pointer if it is in malloc memory range.
Definition: ram.c:158
lif_find_file
MEMSPACE int lif_find_file(lif_t *LIF, char *liflabel)
Find a LIF image file by name.
Definition: lifutils.c:1815
tm::tm_mon
int tm_mon
Definition: time.h:46
lif_t::filesectors
uint32_t filesectors
Definition: lifutils.h:121
tm::tm_mday
int tm_mday
Definition: time.h:45
lif_t::files
int files
Definition: lifutils.h:124
trim_tail
MEMSPACE void trim_tail(char *str)
Trim White space and control characters from end of string.
Definition: parsing.c:49
lif_newdir
MEMSPACE int lif_newdir(lif_t *LIF, long sectors)
Allocate index of free directory record.
Definition: lifutils.c:1507
lif_checkdirindex
MEMSPACE int lif_checkdirindex(lif_t *LIF, int index)
Check directory index limits.
Definition: lifutils.c:1284
lif_str2vol
MEMSPACE void lif_str2vol(uint8_t *B, lif_t *LIF)
Convert byte vector into LIF volume records.
Definition: lifutils.c:608
tm::tm_hour
int tm_hour
Definition: time.h:44
lif_free
MEMSPACE void lif_free(void *p)
Free allocated memory Displays message on errors.
Definition: lifutils.c:308
lif_check_volume
MEMSPACE int lif_check_volume(lif_t *LIF)
Check Volume Table for values in range.
Definition: lifutils.c:872
lif_t::name
char * name
Definition: lifutils.h:116
td0_help
void td0_help(int full)
Definition: td02lif.c:1556
lif_open_volume
MEMSPACE lif_t * lif_open_volume(char *name, char *mode)
Open LIF directory for reading.
Definition: lifutils.c:1644
lif_t::fp
FILE * fp
Definition: lifutils.h:117
stat
MEMSPACE int stat(char *name, struct stat *buf)
POSIX stat - get file status of named file.
Definition: posix.c:1368
lif_ascii_string_to_e010
MEMSPACE int lif_ascii_string_to_e010(char *str, long offset, uint8_t *wbuf)
Convert an ASCII string into HP85 E010 format.
Definition: lifutils.c:1917
lif_dir
MEMSPACE void lif_dir(char *lifimagename)
Display a LIF image file directory.
Definition: lifutils.c:1734
lif_time2lifbcd
MEMSPACE void lif_time2lifbcd(time_t t, uint8_t *bcd)
UNIX time to LIF BCD time format The BCD year is only the lower 2 digits of the year So We assume tha...
Definition: lifutils.c:694
lif_t::freesectors
uint32_t freesectors
Definition: lifutils.h:123
lif_tests
MEMSPACE int lif_tests(int argc, char *argv[])
LIF user tests.
Definition: lifutils.c:139
utimbuf::actime
time_t actime
Definition: posix.h:126
lif_t::imagebytes
uint32_t imagebytes
Definition: lifutils.h:119
lif_open
MEMSPACE FILE * lif_open(char *name, char *mode)
Open a file that must exist Displays message on errors.
Definition: lifutils.c:339
lif_dir2str
MEMSPACE void lif_dir2str(lif_t *LIF, uint8_t *B)
Convert LIF directory records into byte vector.
Definition: lifutils.c:631
strcpy
MEMSPACE WEAK_ATR char * strcpy(char *dest, const char *src)
copy a string
Definition: stringsup.c:160
lif_rename_volume
MEMSPACE int lif_rename_volume(char *lifimagename, char *volname)
Rename LIF VOLUME NAME.
Definition: lifutils.c:2730
lifdir_t::filename
uint8_t filename[10+1]
Definition: lifutils.h:99
NULL
#define NULL
Definition: user_config.h:85
lif_writedirEOF
MEMSPACE int lif_writedirEOF(lif_t *LIF, int index)
Write LIF drectory EOF.
Definition: lifutils.c:1387
LIF_DEBUG
#define LIF_DEBUG
Definition: debug.h:14
__file
FILE type structure.
Definition: posix.h:158
lif_S2B
MEMSPACE void lif_S2B(uint8_t *B, uint8_t *name, int size)
Convert string to LIF directory record.
Definition: lifutils.c:526
strlen
MEMSPACE size_t WEAK_ATR strlen(const char *str)
String Length.
Definition: stringsup.c:144
lifvol_t::Label
uint8_t Label[6+1]
Definition: lifutils.h:61
lif_lifbcd2timestr
MEMSPACE char * lif_lifbcd2timestr(uint8_t *bcd)
Definition: lifutils.c:786
basename
MEMSPACE char * basename(char *str)
POSIX Basename of filename.
Definition: posix.c:1469
lifvol_t::DirStartSector
uint32_t DirStartSector
Definition: lifutils.h:62
lif_t::sectors
uint32_t sectors
Definition: lifutils.h:118
lifvol_t
Disk Layout.
Definition: lifutils.h:58
lif_B2S
MEMSPACE int lif_B2S(uint8_t *B, uint8_t *name, int size)
Convert LIF space padded string name into normal string.
Definition: lifutils.c:485
user_config.h
Master Include for FatFs, RTC, Timers AVR8 - Part of HP85 disk emulator.
lifdir_t::FileBytes
uint16_t FileBytes
Definition: lifutils.h:105
drives_sup.h
lifvol_t::date
uint8_t date[6]
Definition: lifutils.h:71
LIF_DIR_SIZE
#define LIF_DIR_SIZE
LIF directory entry size.
Definition: lifutils.h:29
lif_rewinddir
MEMSPACE void lif_rewinddir(lif_t *LIF)
Rewind LIF directory Note readdir pre-increments the directory pointer index so we start at -1.
Definition: lifutils.c:1261
lif_bytes2sectors
MEMSPACE uint32_t lif_bytes2sectors(uint32_t bytes)
Convert bytes into used sectors.
Definition: lifutils.c:1247
lif_t::purged
int purged
Definition: lifutils.h:125
fclose
MEMSPACE int fclose(FILE *stream)
POSIX close a file stream.
Definition: posix.c:1261
lif_create_volume
MEMSPACE lif_t * lif_create_volume(char *imagename, char *liflabel, long dirstart, long dirsectors, long filesectors)
Create LIF image with Volume, Directory and optional empty filespace.
Definition: lifutils.c:1062
ftell
MEMSPACE long ftell(FILE *stream)
POSIX file position of open stream.
Definition: posix.c:600
SEEK_END
#define SEEK_END
Definition: posix.h:257
lif_fixname
MEMSPACE int lif_fixname(uint8_t *B, char *name, int size)
Convert name into a valid LIF name Only use the basename() part of the string and remove any file nam...
Definition: lifutils.c:549
lif_t::VOL
lifvol_t VOL
Definition: lifutils.h:128
lif_t::EOFindex
int EOFindex
Definition: lifutils.h:127
lif_BCD2BIN
MEMSPACE int lif_BCD2BIN(uint8_t bin)
Convert BCD in the range 0 and <= 99 to BIN BCD format: each hex nibble has a digit 0 ....
Definition: lifutils.c:678
lif_del_file
MEMSPACE int lif_del_file(char *lifimagename, char *lifname)
Delete LIF file in LIF image.
Definition: lifutils.c:2609
MATCHARGS
MEMSPACE int MATCHARGS(char *str, char *pat, int min, int argc)
Match two strings and compare argument index Display message if the number of arguments is too few.
Definition: parsing.c:161
lifutils.h
LIF file utilities.
fseek
MEMSPACE int fseek(FILE *stream, long offset, int whence)
POSIX seek to file possition.
Definition: posix.c:558
time
MEMSPACE time_t time(time_t *t)
Return second from epoch - POSIX function.
Definition: time.c:843
lifvol_t::tracks_per_side
uint32_t tracks_per_side
Definition: lifutils.h:68
SEEK_SET
#define SEEK_SET
Seek offset macros.
Definition: posix.h:255
lif_check_dir
MEMSPACE int lif_check_dir(lif_t *LIF)
Validate Directory record values We only do basic out of bounds tests for this record Purged or EOF d...
Definition: lifutils.c:946
lif_add_ascii_file_as_e010
MEMSPACE long lif_add_ascii_file_as_e010(char *lifimagename, char *lifname, char *userfile)
Convert and add ASCII file to the LIF image as type E010 format The basename of the lifname,...
Definition: lifutils.c:2109
vector.h
utimbuf::modtime
time_t modtime
Definition: posix.h:127
MATCHI_LEN
MEMSPACE int MATCHI_LEN(char *str, char *pat)
Compare two strings without case limted to length of pattern.
Definition: parsing.c:230
lif_ctime_gmt
MEMSPACE char * lif_ctime_gmt(time_t *tp)
GMT version of POSIX ctime().
Definition: lifutils.c:766
lifsup.h
LIF file utilities - utilities extracted from hp85disk project for stand alone use.
lifdir_t::VolNumber
uint16_t VolNumber
Definition: lifutils.h:104
lifvol_t::DirSectors
uint32_t DirSectors
Definition: lifutils.h:65
lif_e010_pad_sector
MEMSPACE int lif_e010_pad_sector(long offset, uint8_t *wbuf)
HP85 E010 ASCII LIF records ef [ff]* = no more data in this sector df size [ASCII] = data must fit in...
Definition: lifutils.c:1877
lifvol_t::zero2
uint16_t zero2
Definition: lifutils.h:67
debuglevel
int debuglevel
Debug flag - used to log GPIB and emulator messages.
Definition: gpib_task.c:33
lifvol_t::sectors_per_track
uint32_t sectors_per_track
Definition: lifutils.h:70
lifdir_t
Directory layout.
Definition: lifutils.h:97
tm
POSIX struct tm.
Definition: time.h:40
lif_t::dirindex
int dirindex
Definition: lifutils.h:126
tm::tm_year
int tm_year
Definition: time.h:47
lif_chars
MEMSPACE int lif_chars(int c, int index __attribute__((unused)))
Check if characters in a LIF volume or LIF file name are valid.
Definition: lifutils.c:449
B2V_MSB
uint32_t B2V_MSB(uint8_t *B, int index, int size)
Convert a byte array into a value bytes are MSB ... LSB order.
Definition: vector.c:58
time_t
uint32_t time_t
type of EPOCH result.
Definition: time.h:34
lifdir_t::FileType
uint16_t FileType
Definition: lifutils.h:100
lif_dump_vol
MEMSPACE void lif_dump_vol(lif_t *LIF, char *msg)
Dump LIF struture data for debugging.
Definition: lifutils.c:836
lif_t::usedsectors
uint32_t usedsectors
Definition: lifutils.h:122
lif_create_image
MEMSPACE long lif_create_image(char *lifimagename, char *liflabel, uint32_t dirsectors, uint32_t sectors)
Create/Format a LIF new disk image This can take a while to run, about 1 min for 10,...
Definition: lifutils.c:2798
lifdir_t::SectorSize
uint16_t SectorSize
Definition: lifutils.h:106
lif_calloc
MEMSPACE void * lif_calloc(long size)
Allocate and clear memory Displays message on errors.
Definition: lifutils.c:293
utime
MEMSPACE int utime(const char *filename, const struct utimbuf *times)
Set Modification and Access time of a file.
Definition: posix.c:1432
fread
MEMSPACE size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream)
POSIX read nmemb elements from buf, size bytes each, to the stream fd.
Definition: posix.c:823
hpdir_t::BLOCKS
long BLOCKS
Definition: drives_sup.h:50
lif_readdirindex
MEMSPACE int lif_readdirindex(lif_t *LIF, int index)
Read LIF directory record number N.
Definition: lifutils.c:1302
lif_readdir
MEMSPACE lifdir_t * lif_readdir(lif_t *LIF)
Read a directory records from LIF image advancind directory index.
Definition: lifutils.c:1403
lifvol_t::LIFVersion
uint16_t LIFVersion
Definition: lifutils.h:66
utimbuf
POSIX utimbuf structure.
Definition: posix.h:124
V2B_MSB
void V2B_MSB(uint8_t *B, int index, int size, uint32_t val)
Convert Value into byte array bytes are MSB ... LSB order.
Definition: vector.c:22
lif_close_volume
MEMSPACE void lif_close_volume(lif_t *LIF)
Free LIF structure and close any files.
Definition: lifutils.c:1220
lif_vol2str
MEMSPACE void lif_vol2str(lif_t *LIF, uint8_t *B)
Convert LIF volume records into byte vector.
Definition: lifutils.c:586
td02lif
int td02lif(int argc, char *argv[])
Convert a Teledisk LIF formatted disk image into a pure LIF image.
Definition: td02lif.c:1650
lif_image_clear
MEMSPACE void lif_image_clear(lif_t *LIF)
File seek with error message.
Definition: lifutils.c:806
V2B_LSB
void V2B_LSB(uint8_t *B, int index, int size, uint32_t val)
Convert Value into byte array bytes are LSB ... MSB order.
Definition: vector.c:41
lif_dir_clear
MEMSPACE void lif_dir_clear(lif_t *LIF)
Clear DIR part of LIF structure.
Definition: lifutils.c:816
hpdir_t
Definition: drives_sup.h:33
lif_write
MEMSPACE int lif_write(lif_t *LIF, void *buf, long offset, int bytes)
Write data to an LIF image Displays message on errors.
Definition: lifutils.c:424
lif_add_ascii_file_as_e010_wrapper
MEMSPACE long lif_add_ascii_file_as_e010_wrapper(lif_t *LIF, uint32_t offset, char *username)
Add ASCII file as E010 data to LIF image - or compute converted data size To find size of formatted r...
Definition: lifutils.c:2012
hpdir_find_drive
int hpdir_find_drive(char *model, int list, int verbose)
Find drive parameters in hpdir.ini file.
Definition: drives_sup.c:79
gmtime_r
MEMSPACE tm_t * gmtime_r(time_t *t, tm_t *result)
Convert epoch GMT time_t *tp into POSIX tm_t *result.
Definition: time.c:456
lifdir_t::date
uint8_t date[6]
Definition: lifutils.h:103
lif_BIN2BCD
MEMSPACE uint8_t lif_BIN2BCD(uint8_t data)
Convert number >= 0 and <= 99 to BCD.
Definition: lifutils.c:668
lif_dir_count
long lif_dir_count(long blocks)
LIF Directory blocks ~= sqrt(blocks);.
Definition: drives_sup.c:57
lif_seek_msg
MEMSPACE int lif_seek_msg(FILE *fp, long offset, char *msg)
Definition: lifutils.c:374