HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
td02lif.c
Go to the documentation of this file.
1 
41 #include <string.h>
42 #include <stdlib.h>
43 #include <stdio.h>
44 
45 #include <inttypes.h>
46 
47 #include <time.h>
48 
49 #include "user_config.h"
50 #include "lifsup.h"
51 #include "lifutils.h"
52 #include "td02lif.h"
53 
55 #include "td0_lzss.h"
56 
58 
61 
68 void td0_B2S(uint8_t *B, int index, uint8_t *name, int size)
69 {
70  int i;
71  for(i=0;i<size;++i)
72  name[i] = B[index+i];
73  name[i] = 0;
74 }
75 
76 
87 uint16_t td0_crc16(uint8_t *B, uint16_t crc, uint16_t poly, int size)
88 {
89  int i,bit;
90  for(i=0; i<size; ++i)
91  {
92  crc ^= (0xff00 & ((uint16_t)B[i] << 8));
93 // Loop for 8 bits per byte
94  for (bit = 0; bit < 8; bit++)
95  {
96  if ((crc & 0x8000) != 0)
97  crc = (uint16_t) ((crc << 1) ^ poly);
98  else
99  crc <<= 1;
100  }
101  }
102  return (crc);
103 }
104 
105 
110 void td0_hexdump(uint8_t *data, int size)
111 {
112  long addr;
113  int i,len;
114 
115  addr = 0;
116  while(size > 0)
117  {
118  printf("%08lx : ", addr);
119  len = size > 16 ? 16 : size;
120 
121  for(i=0;i<len;++i)
122  printf("%02x ",0xff & data[addr+i]);
123 
124  for(;i<16;++i)
125  printf(" ");
126 
127  printf(" : ");
128  for(i=0;i<len;++i)
129  {
130  if(data[addr+i] >= 0x20 && data[addr+i] <= 0x7e)
131  putchar(data[addr+i]);
132  else
133  putchar('.');
134  }
135  for(;i<16;++i)
136  putchar('.');
137 
138  printf("\n");
139 
140  addr += len;
141  size -= len;
142  }
143  printf("\n");
144 }
145 
146 
152 {
153  int i;
154  uint16_t crc;
155 
156  td0_B2S(B, 0, p->Header, 2);
157  p->VolNO = B2V_LSB(B,2,1);
158  p->ChkSig = B2V_LSB(B,3,1);
159  p->TDVersion = B2V_LSB(B,4,1);
160  p->Density = B2V_LSB(B,5,1);
161  p->DriveType = B2V_LSB(B,6,1);
162  p->TrackDensity = B2V_LSB(B,7,1);
163  p->DosMode = B2V_LSB(B,8,1);
164  p->Sides = B2V_LSB(B,9,1);
165  p->CRC = B2V_LSB(B,10,2);
166 
167  crc = td0_crc16(B,0, 0xA097, 10);
168  if(p->CRC != crc)
169  printf("TeleDisk error Header CRC16:%04Xh != %04Xh\n", (int)crc, (int)p->CRC);
170  return(TD_HEADER_SIZE);
171 }
172 
173 
179 {
180  int i;
181  uint16_t crc;
182 
183  p->CRC = B2V_LSB(B,0,2);
184  p->Size = B2V_LSB(B,2,2);
185  p->Year = B2V_LSB(B,4,1);
186  p->Month = B2V_LSB(B,5,1);
187  p->Day = B2V_LSB(B,6,1);
188  p->Hour = B2V_LSB(B,7,1);
189  p->Minute = B2V_LSB(B,8,1);
190  p->Second = B2V_LSB(B,9,1);
191 
192  crc = td0_crc16(B+2,0,0xA097, 8);
193  return(crc);
194 }
195 
196 
202 {
203  int i;
204  uint16_t crc;
205 
206  p->PSectors = B2V_LSB(B,0,1);
207  p->PCyl = B2V_LSB(B,1,1);
208  p->PSide = B2V_LSB(B,2,1);
209  p->CRC = B2V_LSB(B,3,1);
210  crc = td0_crc16(B,0,0xA097, 3);
211 
212  crc &= 0xff;
213 
214  if(p->CRC != crc)
215  {
216 // EOF ?
217  if(p->PSectors != 255)
218  {
219  printf("TeleDisk error Track CRC16:%04Xh != %04Xh\n", (int)crc, (int)p->CRC);
220  printf("\tCyl:%02d, Side:%02d, Sectors:%d\n",
221  (int) p->PCyl, (int) p->PSide, (int) p->PSectors);
222  }
223  }
224  return(crc);
225 }
226 
227 
233 {
234  int i;
235  uint16_t crc;
236 
237  p->Cyl = B2V_LSB(B,0,1);
238  p->Side = B2V_LSB(B,1,1);
239  p->Sector = B2V_LSB(B,2,1);
240  p->SizeExp = B2V_LSB(B,3,1);
241  p->Flags = B2V_LSB(B,4,1);
242  p->CRC = B2V_LSB(B,5,1);
243  return(crc);
244 }
245 
246 
250 void td0_compressed(int flag)
251 {
252  disk.compressed = flag;
253  if(flag)
254  init_decompress();
255 }
256 
257 
264 int td0_read(void *p, int osize, int size, FILE *fp)
265 {
266  int c;
267  long count;
268  long ind = 0;
269 
270  if(disk.compressed)
271  {
272  uint8_t *ptr = p;
273  count = osize * size;
274  while(count--)
275  {
276  c = lzss_getbyte(fp);
277  if(c == EOF)
278  {
279  if(ind != (osize * size))
280  return(0);
281  break;
282  }
283  *ptr++ = (c & 0xff);
284  ++ind;
285  }
286  }
287  else
288  {
289  ind = fread(p,osize,size,fp);
290  if(ind != size)
291  return(0);
292  }
293  return(1);
294 }
295 
296 
307 int td0_rle(uint8_t *dst, uint8_t *src, int max)
308 {
309  int len;
310  int result; // expanded result size, -expanded result size
311 
312  uint8_t type; // Encoding type, we support 0,1,2
313 
314  int error = 0;
315  uint8_t size;
316  uint8_t repeat;
317 
318  result = 0; // expanded size
319 
320  len = B2V_LSB(src,0,2) -1;
321  type = B2V_LSB(src,2,1);
322  src += 3;
323 
324 // printf("td0_rle: len:%3d, type:%d\n", (int) len, (int)type);
325 
326  switch ( type )
327  {
328  case 0:
329 
330  if(len > max)
331  {
332  printf("td0_rle: type 0 len:%d > max:%d\n", len, max);
333  memcpy(dst,src,max);
334  result = -max;
335  error = 1;
336  }
337  else
338  {
339  memcpy(dst,src,len);
340  result = len;
341  }
342  break;
343 
344  case 1:
345  result = 0;
347  len = B2V_LSB(src,0,2) << 1;
348  if(len > max)
349  {
350  printf("td0_rle: type 1 len:%d > max:%d\n", len, max);
351  error = 1;
352  len = max;
353  }
354  while (len >= 2)
355  {
356  *dst++ = src[2];
357  *dst++ = src[3];
358  result += 2;
359  len -= 2;
360  }
361  break;
362  case 2:
363  result = 0;
364  do
365  {
366  if( !src[0] )
367  {
368  size = src[1];
369  src += 2;
370  len -= 2;
371 
372  if((result+size) > max)
373  {
374  printf("td0_rle: type 1 len:%d > max:%d\n",
375  result+size, max);
376  error = 1;
377  break;
378  }
379 
380  memcpy(dst,src,size);
381 
382  result += size;
383  dst += size;
384  src += size;
385  len -= size;
386  }
387  else
388  {
390  size = (1 << src[0]);
391 
393  repeat = src[1];
394 
395  src += 2;
396  len -= 2;
397 
398  while(repeat--)
399  {
400 
401  if((result+size) > max)
402  {
403  printf("td0_rle: type 1 len:%d > max:%d\n",
404  result+size, max);
405  error = 1;
406  break;
407  }
408 
409  memcpy(dst, src, size);
410 
411  result += size;
412  dst += size;
413  }
414 
415  src += size;
416  len -= size;
417  }
418 
419  }
420  while(len > 0);
421  if(len)
422  {
423  printf("td0_rle: type 1 len:%d < 0\n", len);
424  error = 1;
425  }
426  break;
427 
428  default:
429  printf("td0_rle error unsupported type:%02Xh\n", type);
430  result = -1;
431  break;
432  }
433  if(error)
434  return(-result);
435  return (result);
436 }
437 
438 
443 void td0_trackinfo(disk_t *disk, int trackind, int index)
444 {
445  printf("cylinder:%02d, head:%02d, sector:%02d, size:%d, index:%d\n",
446  (int) disk->track[trackind].sectors[index].cylinder,
447  (int) disk->track[trackind].sectors[index].side,
448  (int) disk->track[trackind].sectors[index].sector,
449  (int) disk->track[trackind].sectors[index].size,
450  (int) index);
451 
452 }
453 
454 
459 {
460  printf("cyl: %02d, side: %02d, sector: %02d, size: %03d\n",
461  (int) P->Cyl,
462  (int) P->Side,
463  (int) P->Sector,
464  (int) 128 << P->SizeExp);
465 }
466 
467 
472 long td0_density2bitrate(uint8_t density)
473 {
474  long bitrate = 250000;
475  switch(density)
476  {
477  case 0:
478  bitrate = 250000;
479  break;
480  case 1:
481  bitrate = 300000;
482  break;
483  case 2:
484  bitrate = 500000;
485  break;
486  default:
487  break;
488  }
489  return(bitrate);
490 }
491 
492 
506 int td0_open(disk_t *disk, char *name)
507 {
508  uint16_t crc;
509  tm_t tm;
510  char timebuf[32];
511  uint8_t headerbuf[TD_HEADER_SIZE+1];
512  uint8_t commentbuf[TD_COMMENT_SIZE+1];
513 
514 // default file time if not in comment
515  disk->t = time(0);
516 
517  memset((td_header_t *) &disk->td_header,0,sizeof(disk->td_header));
518  memset((td_comment_t *) &disk->td_comment,0,sizeof(disk->td_comment));
519 
520 // Default is NO comment
521  disk->comment = "";
522  disk->compressed = 0;
523 
524 // TeleDisk Image name
525  disk->td0_name = lif_stralloc(name);
526 
527  disk->fi = fopen(disk->td0_name,"rb");
528  if(!disk->fi)
529  {
530  printf("Error: Can't open TeleDisk file: %s\n",disk->td0_name);
531  return(0);
532  }
533 
535  if( fread( headerbuf, 1, TD_HEADER_SIZE, disk->fi ) != TD_HEADER_SIZE)
536  {
537  printf("Error: Can't read TeleDisk header: %s\n", disk->td0_name);
538  return(0);
539  }
541 
543  if(MATCH(disk->td_header.Header,"TD"))
544  {
545  td0_compressed(0);
546  }
547 
548  else if(MATCH(disk->td_header.Header,"td"))
549  {
550  td0_compressed(1);
551  }
552  else
553  {
554  printf("Error: bad TeleDisk header: %s\n", disk->td_header.Header);
555  return(0);
556  }
557 
559  printf("TeleDisk file: %s\n",disk->td0_name);
560  printf("\tVersion: %02d\n",disk->td_header.TDVersion);
561  if(disk->compressed)
562  printf("\tAdvanced Compression\n");
563  else
564  printf("\tNot Compressed\n");
565  printf("\tDensity: %02Xh\n",disk->td_header.Density);
566  printf("\tDriveType: %02Xh\n",disk->td_header.DriveType);
567  printf("\tTrackDensity: %02Xh\n",disk->td_header.TrackDensity & 0x7f);
568  printf("\tDosMode: %02Xh\n",disk->td_header.DosMode);
569  printf("\tSides: %02d\n",disk->td_header.Sides);
570 
571  if((disk->td_header.TDVersion>21) || (disk->td_header.TDVersion<10))
572  {
573  printf("Error: only TeleDisk versions 10 to 21 supported\n");
574  return(0);
575  }
576 
577 // Do we have a Comment Block ?
578 // Note: We have a Comment if bit 7 of TrackDensity is set
579  if(disk->td_header.TrackDensity & 0x80)
580  {
581 
582  if( !td0_read( commentbuf, 1, TD_COMMENT_SIZE, disk->fi ) )
583  {
584  printf("Error: reading commment\n");
585  return(0);
586  }
587 
588  crc = td0_unpack_comment_header(commentbuf, (td_comment_t *)&disk->td_comment);
589 
591  if(!disk->comment)
592  {
593  printf("Can't allocate comment buffer of %d bytes\n", disk->td_comment.Size);
594  return(0);
595  }
597 
598  if( !td0_read( disk->comment, 1, disk->td_comment.Size,disk->fi) )
599  {
600  printf("Error: reading commment\n");
601  return(0);
602  }
603 
604  crc = td0_crc16(disk->comment,crc,0xA097,disk->td_comment.Size);
605  if(disk->td_comment.CRC != crc)
606  {
607  printf("Warning: Comment CRC16:%04Xh != %04Xh\n", (int)crc, (int)disk->td_comment.CRC);
608  }
609 
610 // FIXME convert to LIF date
612  tm.tm_year = (int)disk->td_comment.Year;
613  tm.tm_mon = (int)disk->td_comment.Month;
614  tm.tm_mday = (int)disk->td_comment.Day;
615  tm.tm_hour = (int)disk->td_comment.Hour;
616  tm.tm_min = (int)disk->td_comment.Minute;
617  tm.tm_sec = (int)disk->td_comment.Second;
618  tm.tm_wday = 0;
619  tm.tm_yday = 0;
620 
621 // DATE
622  printf("\tComment Size: %d\n", disk->td_comment.Size);
623  printf("\tComment Date: %s\n", asctime_r((tm_t *) &tm, timebuf));
624 
625  disk->t = timegm((tm_t *) &tm);
626 
627 // COMMENT
628  printf("%s\n\n", disk->comment);
629  } // if(disk->td_header.TrackDensity & 0x80)
630  else
631  {
632  disk->td_comment.Size = 0;
633  printf("\tNo Comment Block\n");
634 // Empty Comment
635  }
636  return(1);
637 }
638 
639 
644 {
645  int i,j;
646  int cyl,head,sector,size;
647  int t,s;
648  int index, tracksectors, found, result;
649  int status;
650  long count = 0;
651  td_track_t td_track;
652  td_sector_t td_sector;
653  uint8_t trackbuf[TD_TRACK_SIZE];
654  uint8_t sectorbuf[TD_SECTOR_SIZE];
655 
656  uint8_t buffer[8*1024];
657 
658 // ==============================================================
659 // FIXME
660 // Reasons - eliminates the overhead of closing out on errors
661 // Open file in main
662 // Init data structures in main
663 // process TD0 to LIF data from main
664 // Close file in main
665 // Free data in main
666 // Move all disk processing into a function
667 // Move all sector processing into a fuction
668 // Move all track processing into a function
669 // ==============================================================
670 
672  for(t = 0; t < MAXTRACKS; ++t)
673  {
674 // PROCESS TRACK
675 
676 // TRACK HEADER
677  if( !td0_read(trackbuf,1, TD_TRACK_SIZE, disk->fi) )
678  {
679  printf("Error: reading track header\n");
680  return (0);
681  }
682  td0_unpack_track_header(trackbuf, (td_track_t *)&td_track);
683 
684  if(td_track.PCyl >= MAXCYL )
685  {
686  printf("Error: cylinder number %02d >= %02d\n", td_track.PCyl, MAXCYL);
687  return (0);
688 
689  }
690 
691  if(td_track.PSide >= MAXSIDES)
692  {
693  printf("Error: side number %02d >= %02d\n", td_track.PSide, MAXSIDES);
694  return (0);
695 
696  }
697 
698  disk->track[t].Cyl = td_track.PCyl;
699  disk->track[t].Side = td_track.PSide;
700 
701 // ====================================================
703  if(td_track.PSectors == 0xff)
704  break;
705 // ====================================================
706 
707  if(debuglevel & LIF_DEBUG)
708  printf("Track: Cyl: %02d, Side: %02d, Sectors: %02d\n",
709  (int)td_track.PCyl, (int)td_track.PSide, (int)td_track.PSectors);
710 
711 // Initialize track data
712 
714  for ( s=0; s < td_track.PSectors; s++ )
715  {
717  if( !td0_read(sectorbuf,1, TD_SECTOR_SIZE,disk->fi) )
718  {
719  printf("Error: reading sector header\n");
720  printf("\t Track: Cyl: %02d, Side: %02d, Sectors: %02d\n",
721  (int)td_track.PCyl, (int)td_track.PSide, (int)td_track.PSectors);
722  return (0);
723  }
724 
725  td0_unpack_sector_header(sectorbuf, (td_sector_t *)&td_sector);
726 
727 //FIXME add flag override for this test
728  if(td_track.PCyl != td_sector.Cyl )
729  {
730  printf("Warning: track Cyl:%d != sector Cyl:%d skipping\n",
731  (int)td_track.PCyl, (int) td_sector.Cyl);
732  printf("\t");
733  td0_sectorinfo((td_sector_t *) &td_sector);
734  }
735 
736 //FIXME add flag override for this test
737  if(td_track.PSide != td_sector.Side)
738  {
739  printf("Warning: track side:%d != sector side:%d skipping\n",
740  (int)td_track.PSide, (int) td_sector.Side);
741  printf("\t");
742  td0_sectorinfo((td_sector_t *) &td_sector);
743  }
744 
745 // We insert sectors using their sector ID as the index offset
746 // This "sorts" them in order
747 
748  index = td_sector.Sector;
749 
750 // FYI as long as MAXSECTORS is 256 this will NEVER happen
751 // This test is just here in case someone changes MAXSECTORS
752  if(index >= MAXSECTORS)
753  {
754  printf("Error: sector index: %d >= MAXSECTORS\n",
755  (int)index );
756  printf("\t");
757  td0_sectorinfo((td_sector_t *) &td_sector);
758  return (0);
759  }
760 
761 //FIXME we can not deal with duplicate sectors so skip them
762  if( disk->track[t].sectors[index].sector != -1)
763  {
764  printf("Warning: skipping duplicate sector: %02d, skipping\n", index);
765  printf("\t");
766  td0_sectorinfo((td_sector_t *) &td_sector);
767  continue;
768  }
769 
770  disk->track[t].sectors[index].sector = index;
771 
772  disk->track[t].sectors[index].cylinder = td_sector.Cyl;
773  disk->track[t].sectors[index].side = td_sector.Side;
774  disk->track[t].sectors[index].size = 128<<td_sector.SizeExp;
775  disk->track[t].sectors[index].data =
776  calloc(disk->track[t].sectors[index].size,1);
777 
778  if(td_sector.Flags & 0x02)
779  {
780  printf("Warning: alternate CRC flag not implemented\n");
781  printf("\t");
782  td0_trackinfo(disk,t,index);
783  }
784  if(td_sector.Flags & 0x04)
785  {
786  printf("Warning: alternate data mark not implemented\n");
787  printf("\t");
788  td0_trackinfo(disk,t,index);
789  }
790  if(td_sector.Flags & 0x20)
791  {
792  printf("Warning: missing address mark not implemented\n");
793  printf("\t");
794  td0_trackinfo(disk,t, index);
795  }
796 
799  if ( !(td_sector.SizeExp & 0xf8) && !(td_sector.Flags & 0x30))
800  {
801  int len;
802  uint16_t crc;
803 
805  memset(buffer,0,sizeof(buffer));
806 // Read Sector Data
807  if( !td0_read(buffer,1, sizeof(uint16_t),disk->fi) )
808  {
809  printf("Error: reading sector header\n");
810  printf("\t");
811  td0_trackinfo(disk,t,index);
812  return(0);
813  }
814  len = B2V_LSB(buffer,0,2);
815 
816  if( !td0_read(buffer+2,1, len,disk->fi) )
817  {
818  printf("Error: reading sector data\n");
819  printf("\t");
820  td0_trackinfo(disk, t,index);
821  return (0);
822  }
823 
827  result = td0_rle(disk->track[t].sectors[index].data, buffer, disk->track[t].sectors[index].size);
828 
829  if(result != disk->track[t].sectors[index].size)
830  {
831 // Very unlikey we can recover from this!
832  printf("Error: RLE result:%d != sector size:%d\n",
833  (int) result , (int) disk->track[t].sectors[index].size);
834  printf("\t");
835  td0_trackinfo(disk,t,index);
836  return (0);
837  }
838 
839  crc = td0_crc16(disk->track[t].sectors[index].data,0,0xA097,result);
840  crc &= 0xff;
841  if(td_sector.CRC != crc)
842  {
843  printf("Warning: Sector CRC16:%04Xh != %04Xh\n",
844  (int)crc, (int)td_sector.CRC);
845  printf("\t");
846  td0_trackinfo(disk,t,index);
847  }
848 
849  } // if ( !(td_sector.SizeExp & 0xf8) && !(td_sector.Flags & 0x30))
850  else
851  {
852  printf("Warning: unsupported sector flags:%02Xh\n", (int)td_sector.Flags);
853  printf("\t");
854  td0_trackinfo(disk,t,index);
855  }
856 
857  } // for ( i=0;i < td_track.PSectors;i++ )
858 
859 // Done processing first sector
861 
862  } // for(t = 0; t < MAXTRACKS; ++t)
863 
864  return (1);
865 }
866 
867 
897 {
898  int s,t;
899  int i,j;
900  int flag;
901  int sector,size;
902  int maxtracks;
903  int maxsectors[2];
904  int reject_side_two = 0;
905 
906 // SIDES
908 
909  if( disk->td_header.Sides == 2)
910  {
911  if( liftel.u.sides != -1)
912  {
913  if(liftel.u.sides == 1)
914  {
915  printf("Warning: Override number of sides from 2 to 1\n");
916  reject_side_two = 1;
917  liftel.Sides = 1;
918  }
919  else if( liftel.u.sides != 2)
920  {
921  printf("Error: NO sectors detected\n");
922  printf("Giving up!\n");
923  liftel.error = 1;
925  return(0);
926  }
927  }
928  }
929  else if( disk->td_header.Sides == 1)
930  {
931  if(liftel.u.sides != -1 && liftel.u.sides == 2)
932  {
933  printf("Error: NO sectors detected\n");
934  printf("Giving up!\n");
935  liftel.error = 1;
937  return(0);
938  }
939  }
940 
941 // TRACKS
942  liftel.Tracks = 0;
943 
944 // SIZE
945  liftel.Size = 0;
946 
947  for(i=0;i<2;++i)
948  {
949  liftel.s.first[i] = MAXSECTORS-1;
950  liftel.s.last[i] = 0;
951  liftel.s.size[i] = 0;
952  liftel.s.sectors[i] = 0;
953  }
954 
955 // ===================================================================
956 // Analize sector data - see notes at the start of this function
957 
958 // Find parameters of smallest possible format 35 tracks less a few
959 // Single sided 35 tracks
960  maxtracks = 30;
961  if(disk->td_header.Sides > 1)
962  maxtracks *= 2;
963 
964 // find FIRST sector and its SIZE in the search area
965  for( t = 0; t < maxtracks; ++ t)
966  {
967 // Scan side one only
968  for ( s=0; s < MAXSECTORS ; s++ )
969  {
970  sector = disk->track[t].sectors[s].sector;
971  if(sector == -1)
972  continue;
973 
974  size = disk->track[t].sectors[s].size;
975 
976 // We only test even tracks to be really safe
977  if(sector < liftel.s.first[t & 1] )
978  {
979  liftel.s.first[t & 1] = sector;
980  liftel.s.size[t & 1] = size;
981  }
982  } // for ( s=0; s < MAXSECTORS ; s++ )
983  } // for( t = 0; t < maxtracks; ++ t)
984 
985 // If size of first sector was NOT found
986  if(liftel.s.size[0] == 0)
987  {
988  printf("Error: NO sectors detected\n");
989  printf("Giving up!\n");
990  liftel.error = 1;
992  return(0);
993  }
994 
995 // Find LAST sectors matching SIZE Within the search area
996 // Find maximum NUMBER of sectors on each side matching size
997  for( t = 0; t < maxtracks; ++t)
998  {
999  maxsectors[t & 1] = 0;
1000 // Scan side one only
1001  for ( s = 0; s < MAXSECTORS ; s++ )
1002  {
1003  sector = disk->track[t].sectors[s].sector;
1004  if(sector == -1)
1005  continue;
1006 
1007  size = disk->track[t].sectors[s].size;
1008  if(liftel.s.size[t & 1] != size )
1009  continue;
1010 
1011 // Do NOT include replacement sectors in LAST test
1012  if( sector < 100 && sector > liftel.s.last[t & 1] )
1013  {
1014  liftel.s.last[t & 1] = sector;
1015  }
1016 // Count ALL sectors remaining regardless of type
1017  ++maxsectors[t & 1];
1018  } // for ( s=0; s < MAXSECTORS ; s++ )
1019 
1020 // Find the largest sector count
1021  if(maxsectors[t & 1] > liftel.s.sectors[t & 1])
1022  liftel.s.sectors[t & 1] = maxsectors[t & 1];
1023  } // for( t = 0; t < maxtracks; ++ t)
1024 
1025 // ===================================================================
1026 
1027  liftel.Size = liftel.s.size[0];
1028 
1029 // ===================================================================
1030 // See if SIDE two matches side one formatting
1031 
1032 // Note: We test the number of sides LATER on AFTER these tests
1033 // Normally first and second side will match
1034 // But if they do NOT match then we only want FIRST side
1035 
1036 // Sector SIZE of side 1 mismatch ?
1037  if(liftel.s.size[0] != liftel.s.size[1])
1038  {
1039 // Reject side two
1040  reject_side_two = 1;
1041  }
1042 
1043 // Sector COUNT mismatch ?
1044  if(liftel.s.sectors[0] != liftel.s.sectors[1])
1045  {
1046 // Reject side two
1047  reject_side_two = 1;
1048  }
1049 
1050  if(liftel.s.first[0] != liftel.s.first[1])
1051  {
1052 // If the FIRST sector differs on each side
1053 // Then the sector numbering MUST be a continuation
1054  if( ( liftel.s.last[0] + 1) != liftel.s.first[1] )
1055  {
1056 // Reject side two
1057  reject_side_two = 1;
1058  }
1059  }
1060 
1061 // If TeleDisk sides == 1 AND reject_side_two is set something is VERY VERY wrong
1062 
1063  if(disk->td_header.Sides == 1 && reject_side_two)
1064  {
1065  printf("Error: Damaged Disk - this should never happen\n");
1066  printf("Giving up!\n");
1067  liftel.error = 1;
1068  liftel.state = TD0_DONE;
1069  return(0);
1070  }
1071 
1072  if(reject_side_two)
1073  {
1074 // Update sides
1075  liftel.Sides = 1;
1076  }
1077 // ===================================================================
1078 
1079 // ===================================================================
1080 // DELETE rejected sectors and sides
1081 // ZERO fill missing sectors
1082 // REMAP "extra" sectors if they exist
1083 
1084 // Default size from fisrt sector of first side
1085  if(liftel.u.size != -1)
1086  {
1087  for(i=0; i< liftel.Sides; ++i)
1088  {
1089  if(liftel.s.size[i] && liftel.u.size != liftel.s.size[i])
1090  {
1091  printf("Warning: Side %d user size:[%03d] NOT same as detected:[%03d]\n",
1092  i, liftel.u.size, liftel.s.size[i]);
1093  }
1094  }
1095  liftel.Size = liftel.u.size;
1096  }
1097 
1098 // We just use the sector count from first side as second side count must match
1099 // (We previously tested sector count mismatch)
1100 
1101  liftel.Sectors = liftel.s.sectors[0];
1102  liftel.Tracks = 0;
1103 
1104  for( t = 0; t < MAXTRACKS; ++ t)
1105  {
1106  disk->track[t].Sectors = 0;
1107  disk->track[t].First = 0;
1108  disk->track[t].Last = 0;
1109  disk->track[t].Size = 0;
1110 
1111  for ( s=0; s < MAXSECTORS ; s++ )
1112  {
1113  sector = disk->track[t].sectors[s].sector;
1114  if(sector == -1)
1115  continue;
1116 
1117  flag = 0;
1118 
1119 // Delete SIZE mismatch
1120  size = disk->track[t].sectors[s].size;
1121  if(size != liftel.Size)
1122  flag = 1;
1123 
1124  if(disk->td_header.Sides == 2 && (t & 1) == 1)
1125  {
1126 // Delete unused SIDE two
1127  if(liftel.u.sides == 1)
1128  flag = 1;
1129 
1130 // Delete SIDE two - previously rejected
1131  if( reject_side_two)
1132  flag = 1;
1133  }
1134 
1135 // Delete and FREE rejected sectors
1136  if(flag )
1137  {
1138  disk->track[t].sectors[s].sector = -1;
1139  disk->track[t].sectors[s].size = 0;
1140  if(disk->track[t].sectors[s].data != NULL)
1141  {
1142  free(disk->track[t].sectors[s].data);
1143  disk->track[t].sectors[s].data = NULL;
1144  }
1145  }
1146  else
1147  {
1148 // Count valid sectors
1149  disk->track[t].Sectors++;
1150  }
1151  } // for ( s=0; s < MAXSECTORS ; s++ )
1152 
1153 // We have sectors on this side
1154  if( disk->track[t].Sectors )
1155  {
1156 #if 0
1157  printf("track: [%d]\n", t);
1158  printf("sector count :[%d]\n", maxsectors[t & 1]);
1159  printf("Sectors:%d\n", disk->track[t].Sectors);
1160  printf("flag:%d\n", flag);
1161  printf("liftel.Size:%d\n", liftel.Size);
1162 #endif
1163  disk->track[t].Sectors = liftel.s.sectors[t & 1];
1164  disk->track[t].First = liftel.s.first[t & 1];
1165  disk->track[t].Last = liftel.s.last[t & 1];
1166  disk->track[t].Size = liftel.Size;
1167  liftel.Tracks++;
1168  }
1169  }
1170 
1171 // Done deleting sectors
1172 // Track 0, Side 0 MUST have sectors at this point or FAIL
1173  if(disk->track[0].Sectors == 0)
1174  {
1175  printf("Warning: track:0 has zero sectors\n");
1176  liftel.error = 1;
1177  liftel.state = TD0_DONE;
1178  return(0);
1179  }
1180 
1181  for( t = 0; t < MAXTRACKS; ++ t)
1182  {
1183 // We have sectors on this side
1184  if( disk->track[t].Sectors == 0)
1185  continue;
1186 
1187 // Test if we have an missing sectors in a track
1188 // If we do have missing sectors on a track then
1189 // 1) Then look for sectors numbered >= 100 as replacements
1190 // 2) If none is found then ZERO fill the missing sector
1191  for (s = disk->track[t].First; s <= disk->track[t].Last; s++ )
1192  {
1193 // Sector is NOT missing
1194  if( disk->track[t].sectors[s].sector != -1)
1195  continue;
1196 
1197 // First look for a replacement >= 100
1198  for(j=100;j<MAXSECTORS;++j)
1199  {
1200 // Look for a replacement
1201  if(disk->track[t].sectors[j].sector == -1)
1202  continue;
1203 
1204 // We also must match the default sector size
1205  if(disk->track[t].sectors[j].size != liftel.Size)
1206  continue;
1207 
1208 // REMAP this on to the missing one
1209  disk->track[t].sectors[s].cylinder = disk->track[t].sectors[j].cylinder;
1210  disk->track[t].sectors[s].side = disk->track[t].sectors[j].side;
1211  disk->track[t].sectors[s].sector = s;
1212  disk->track[t].sectors[s].size = disk->track[t].sectors[j].size;
1213  disk->track[t].sectors[s].data = disk->track[t].sectors[j].data;
1214 
1215 // Delete the REMAP sector so it will not get reused !!
1216  disk->track[t].sectors[j].cylinder = 0;
1217  disk->track[t].sectors[j].side = 0;
1218  disk->track[t].sectors[j].sector = -1;
1219  disk->track[t].sectors[j].size = 0;
1220  disk->track[t].sectors[j].data = NULL;
1221 
1222  printf("Warning: Sector:%02d missing - found alternate sector:%02d\n",
1223  s, j);
1224  printf("\t Location: ");
1225 
1226  td0_trackinfo(disk,t,s);
1227  break;
1228 
1229  } // for(j=100;j<MAXSECTOR;++j)
1230 
1231 // Did we find a replacement ?
1232  if(disk->track[t].sectors[s].sector != -1)
1233  continue;
1234 
1235 // NO replacement
1236 
1237 // ZERO fill a new one with default size
1238  disk->track[t].sectors[s].data = calloc(liftel.Size,1);
1239  disk->track[t].sectors[s].size = liftel.Size;
1240 // mark sector as in use
1241  disk->track[t].sectors[s].sector = s;
1242 
1243  printf("Warning: Sector:%02d missing - zero filling\n", s);
1244  printf("\t Location: ");
1245  printf("Track: Cyl: %02d, Side: %02d\n",
1246  (int)disk->track[t].Cyl, (int)disk->track[t].Side);
1247 
1248  } // for (s = disk->track[t].First; s <= disk->track[t].Last; s++ )
1249 
1250  } // for( t = 0; t < MAXTRACKS; ++ t)
1251 // ============================================
1252 
1253 // ============================================
1254 
1256 
1257  printf("\n");
1258  printf("Disk Layout\n");
1259 
1260  printf("\t Sides: %2d\n", liftel.Sides);
1261  printf("\t Tracks: %2d\n", liftel.Tracks);
1262  printf("\t Sectors Per Track: %2d\n", liftel.Sectors);
1263  printf("\t Sector Size: %3d\n", liftel.Size);
1264 
1265 // Always display all sides as discovered
1266  for(i=0; i< disk->td_header.Sides; ++i)
1267  {
1268  printf("Side: %d\n", i);
1269  printf("\t Sector numbering: %2d to %2d\n",
1270  liftel.s.first[i], (liftel.s.first[i] + liftel.s.sectors[i] - 1) );
1271  printf("\t Sector size: %3d\n", liftel.s.size[i]);
1272  printf("\t Sector count: %2d\n", liftel.s.sectors[i]);
1273  }
1274  printf("\n");
1275 
1276 // If we rejected sides or sectors display a summary
1277  if(reject_side_two)
1278  {
1279  printf("Warning: rejecting side two\n");
1280  if(liftel.s.size[1] && liftel.s.size[0] != liftel.s.size[1])
1281  printf("\t Warning: Sector size: NOT the same on both sides\n");
1282 
1283  if(liftel.s.sectors[1] && liftel.s.sectors[0] != liftel.s.sectors[1])
1284  printf("\t Warning: Sector count: NOT the same on both sides\n");
1285 
1286  if(liftel.s.first[1] && liftel.s.first[0] != liftel.s.first[1])
1287  {
1288 // If the FIRST sector differs then it MUST be a continuation - or - FAIL
1289  if( ( liftel.s.last[0] + 1) != liftel.s.first[1] )
1290  printf("\t Warning: Sector range: NOT a continuation of side 0\n");
1291  }
1292  printf("\n");
1293  }
1294 
1295  return(1);
1296 }
1297 
1298 
1304 {
1305 
1306  int t,s;
1307  uint8_t *ptr;
1308  long sectors = 0;
1309 
1310  for(t=0; t<MAXTRACKS; ++t)
1311  {
1312  if(liftel.Sectors == disk->track[t].Sectors)
1313  {
1314  for (s = disk->track[t].First; s <= disk->track[t].Last; s++ )
1315  {
1316  ptr = disk->track[t].sectors[s].data;
1317  if(!ptr || disk->track[t].sectors[s].sector == -1)
1318  {
1319  printf("ERROR: track:%02d, sector:%02d == NULL\n", t, s);
1320  printf("\t Program error - should never happen\n");
1321  printf("\tGiving up!\n");
1322  liftel.error = 1;
1323  liftel.state = TD0_DONE;
1324  return(0);
1325  }
1326  ++sectors;
1327  }
1328  }
1329  }
1330 
1331  if(!sectors)
1332  {
1333  printf("LIF - no sectors left after processing\n");
1334  return(0);
1335  }
1336 
1337  sectors = 0;
1338  for(t=0; t<MAXTRACKS; ++t)
1339  {
1340  if(liftel.Sectors == disk->track[t].Sectors)
1341  {
1342  for (s = disk->track[t].First; s <= disk->track[t].Last; s++ )
1343  {
1344  ptr = disk->track[t].sectors[s].data;
1345 // PROCESS LIF SECTORS
1346  if( !td0_save_lif_sector(disk, ptr, liftel.Size, LIF) )
1347  {
1348  printf("LIF sectors processed: %ld\n", sectors);
1349  return (0);
1350  }
1351  ++sectors;
1352  }
1353  }
1354  }
1355 
1356  printf("LIF sectors processed: %ld\n", sectors);
1357  return(1);
1358 }
1359 
1360 
1366 int td0_save_lif_sector(disk_t *disk, uint8_t *data, int size, lif_t *LIF)
1367 {
1368  int i;
1369  int dir;
1370  int tmp;
1371  int count = 0;
1372 
1373  if(liftel.error)
1374  {
1375  printf("Error: exit\n");
1376  liftel.state = TD0_DONE;
1377  return(0);
1378  }
1379 
1380  if(liftel.state == TD0_DONE)
1381  return(1);
1382 
1383 // =======================================
1384  switch(liftel.state)
1385  {
1386 
1387  case TD0_START:
1388 // td0_hexdump(data,size);
1389  lif_str2vol(data, LIF);
1390  LIF->filestart = LIF->VOL.DirStartSector + LIF->VOL.DirSectors;
1391  LIF->sectors = LIF->filestart;
1392 
1393  if(lif_check_volume(LIF) == 0)
1394  {
1395  printf("LIF: [%s] position:%ld\n",
1396  LIF->name,(long)liftel.writeindex);
1397  printf("\t Error: Not a LIF image!\n");
1398 
1399  td0_hexdump(data, size);
1400  lif_dump_vol(LIF,"debug");
1401 
1402  liftel.error = 1;
1403  liftel.state = TD0_DONE;
1404  return(0);
1405  }
1406 
1407  if(debuglevel & LIF_DEBUG)
1408  {
1409  printf("LIF VOL Label: [%10s]\n", LIF->VOL.Label);
1410  printf("LIF VOL Date: %s\n", lif_lifbcd2timestr(LIF->VOL.date));
1411  printf("LIF DIR start: [%04lXh]\n", (long)LIF->VOL.DirStartSector);
1412  printf("LIF DIR sectors: [%04lXh]\n", (long)LIF->VOL.DirSectors);
1413  printf("LIF file start: [%04lXh]\n", (long)LIF->filestart);
1414  }
1415 
1416  liftel.sectorindex = 0;
1417  liftel.writeindex = 0;
1418 
1420  break;
1421 
1422 // Wait for director sectors
1423  case TD0_WAIT_DIRECTORY:
1424  if( liftel.sectorindex < LIF->VOL.DirStartSector )
1425  break;
1426 
1427 // Fall through
1429 
1430  case TD0_DIRECTORY:
1431  if( liftel.sectorindex < LIF->filestart)
1432  {
1433 // Process Directory entries in this sector
1434  for(dir=0; dir < size; dir+=32)
1435  {
1436  lif_str2dir(data+dir, LIF);
1437 
1438 // Directory EOF
1439  if(LIF->DIR.FileType == 0xffff)
1440  {
1441  if(debuglevel & LIF_DEBUG)
1442  {
1443  printf("LIF file sectors: [%04lXh]\n", (long) LIF->filesectors);
1444  printf("LIF image sectors: [%04lXh]\n", (long) LIF->sectors);
1445  }
1447  break;
1448  }
1449 
1450 // The offical LIF specification says we must not
1451 // trust file size or start if purged!
1452  if(LIF->DIR.FileType == 0)
1453  continue;
1454 
1455 // Adjust total sectors and used sectors
1456  if( (LIF->DIR.FileStartSector+LIF->DIR.FileSectors)
1457  > LIF->sectors )
1458  {
1459  LIF->sectors = (LIF->DIR.FileStartSector + LIF->DIR.FileSectors);
1460  }
1461  else
1462  {
1463  printf("LIF: [%s] position:%ld\n",
1464  LIF->name,(long)liftel.writeindex);
1465  printf("\t Warning: directory entry is out of order\n");
1466  printf("\t Treating as DirectoryEOF\n");
1467  td0_hexdump(data, size);
1468 
1469 // Update sector data
1470  LIF->DIR.FileType = 0xffff;
1471  lif_dir2str(LIF,data+dir);
1472 
1474  break;
1475  }
1476 
1477  LIF->filesectors = LIF->sectors - LIF->filestart;
1478  LIF->usedsectors = LIF->filesectors;
1479 
1480  if(!lif_check_dir(LIF))
1481  {
1482  printf("LIF: [%s] position:%ld\n",
1483  LIF->name,(long)liftel.writeindex);
1484  printf("\t Warning: bad director entry\n");
1485  printf("\t Treating as DirectoryEOF\n");
1486  td0_hexdump(data, size);
1487 
1488 // Update sector data
1489  LIF->DIR.FileType = 0xffff;
1490  lif_dir2str(LIF,data+dir);
1491 
1493  break;
1494  }
1495 
1496  if(debuglevel & LIF_DEBUG)
1497  {
1498  printf("LIF: [%s] start:%6lXh size:%6lXh %s\n",
1499  LIF->DIR.filename,
1500  (long)LIF->DIR.FileStartSector,
1501  (long)LIF->DIR.FileSectors,
1502  lif_lifbcd2timestr(LIF->DIR.date) );
1503  }
1504 
1505  } // for(dir=0; dir < size; dir+=32)
1506  break;
1507  }
1508 
1509 // Fall through
1511 
1512  case TD0_WAIT_FILE:
1513 //td0_hexdump(data,size);
1514  if( liftel.sectorindex < LIF->filestart)
1515  break;
1516 
1517 // Fall through
1518  LIF->filesectors = LIF->sectors - LIF->filestart;
1519  liftel.state = TD0_FILE;
1520 
1521  case TD0_FILE:
1522 //td0_hexdump(data,size);
1523  if( liftel.sectorindex < LIF->sectors)
1524  break;
1525 
1526 // Fall through
1527  liftel.state = TD0_DONE;
1528 
1529  case TD0_DONE:
1530  return(1);
1531  } // switch(liftel.state)
1532 
1537 
1538  tmp = fwrite(data, 1, size, LIF->fp);
1539  if(size != tmp)
1540  {
1541  printf("LIF: [%s] position:%ld\n",
1542  LIF->name,(long)liftel.writeindex);
1543  printf("\t Write error\n");
1544 
1545  liftel.state = TD0_DONE;
1546  liftel.error = 1;
1547  return(0);
1548  }
1549 
1550  liftel.sectorindex++;
1551  liftel.writeindex++;
1552  return(1);
1553 }
1554 
1555 
1556 void td0_help(int full)
1557 {
1558  printf("td02lif help\n");
1559  if(full)
1560  {
1561  printf(
1562  "Usage: td02lif [options] file.td0 file.lif\n"
1563  " td02lif help\n"
1564  "tdo2lif options:\n"
1565  "Notes: for any option that is NOT specified it is automatically detected\n"
1566  "\t -s256|512 | -s 256|512 - force sector size\n"
1567  "\t -h1|2 | -h 1|2 - force heads/surfaces\n"
1568  "\t -tNN | -t NN - force tracks\n"
1569  "\n"
1570  );
1571  }
1572 }
1573 
1574 
1578 {
1579  int i;
1580 // State Machine
1581  liftel.error = 0;
1582  liftel.state = TD0_INIT;
1583  liftel.sectorindex = 0;
1584  liftel.writeindex = 0;
1585 
1586 // Initialize liftel data
1587  liftel.Size = 0;
1588  liftel.Sides = 0;
1589  liftel.Sectors = 0;
1590  liftel.Tracks = 0;
1591  liftel.Cylinders = 0;
1592 
1593  for(i=0;i<2;++i)
1594  {
1595  liftel.s.first[i] = 0;
1596  liftel.s.size[i] = 0;
1597  liftel.s.last[i] = 0;
1598  liftel.s.sectors[i] = 0;
1599  }
1600 
1601 // User overrride
1602  liftel.u.size = -1;
1603  liftel.u.sides = -1;
1604  liftel.u.tracks = -1;
1605 }
1606 
1607 
1615 {
1616  int t,s;
1617 
1618  disk->fi = NULL;
1619  disk->td0_name = NULL;
1620  disk->compressed = 0;
1621  disk->t = 0;
1622 
1623  memset((td_header_t *) & disk->td_header, 0, sizeof(td_header_t));
1624  memset((td_comment_t *) & disk->td_comment,0,sizeof(td_comment_t));
1625 
1626  for(t = 0; t<MAXTRACKS; ++t)
1627  {
1628  for(s=0;s<MAXSECTORS;++s)
1629  {
1630  disk->track[t].Cyl = 0;
1631  disk->track[t].Side = 0;
1632  disk->track[t].Sectors = 0;
1633  disk->track[t].First = MAXSECTORS-1;
1634  disk->track[t].Last = 0;
1635 
1636  disk->track[t].sectors[s].cylinder = 0;
1637  disk->track[t].sectors[s].side = 0;
1638  disk->track[t].sectors[s].sector = -1;
1639  disk->track[t].sectors[s].size = 0;
1640  disk->track[t].sectors[s].data = NULL;
1641  }
1642  }
1643 }
1644 
1645 
1650 int td02lif(int argc, char *argv[])
1651 {
1652  FILE *fo;
1653  int status;
1654  int tmp;
1655  time_t t;
1656  char *ptr;
1657  char *telediskname = NULL;
1658  char *lifname = 0;
1659  int i;
1660 
1661 // Analisis Data
1662  td0_init_liftel();
1663 
1664 // Sectir data
1665  td0_init_sectors( (disk_t *) &disk );
1666 
1667  ptr = argv[0];
1668  if(!ptr)
1669  return(0);
1670 
1671  if(argc <= 1 || MATCHI(argv[1],"help") )
1672  {
1673  td0_help(1);
1674  return(1);
1675  }
1676 
1677  for(i=1;i<argc;++i)
1678  {
1679  ptr = argv[i];
1680  if(!ptr)
1681  break;
1682 
1683  if(!*ptr)
1684  continue;
1685 
1686  if(*ptr == '-')
1687  {
1688  ++ptr;
1689 // Sector size
1690  if(*ptr == 's')
1691  {
1692  ++ptr;
1693  if(*ptr || (ptr = argv[++i]) )
1694  {
1695  tmp = atoi(ptr);
1696  if(tmp != 256 && tmp != 512)
1697  {
1698  printf("ERROR: size:[%d] != 256 or 512\n", tmp);
1699  td0_help(1);
1700  return(1);
1701  }
1702  liftel.u.size = tmp;
1703  }
1704  continue;
1705  }
1706 
1707 // Heads
1708  if(*ptr == 'h')
1709  {
1710  ++ptr;
1711  if(*ptr || (ptr = argv[++i]) )
1712  {
1713  tmp = atoi(ptr);
1714  if(tmp != 1 && tmp != 2)
1715  {
1716  printf("ERROR: sides:[%d] != 1 or 2\n", tmp);
1717  td0_help(1);
1718  return(1);
1719 
1720  }
1721  liftel.u.sides = tmp;
1722  }
1723  continue;
1724  }
1725 
1726 // Tracks
1727  if(*ptr == 't')
1728  {
1729  ++ptr;
1730  if(*ptr || (ptr = argv[++i]) )
1731  {
1732  tmp = atoi(ptr);
1733  if(tmp >= MAXTRACKS)
1734  {
1735  printf("ERROR: tracks:[%d] >= %d\n", tmp, MAXTRACKS);
1736  td0_help(1);
1737  return(1);
1738 
1739  }
1740  liftel.u.tracks = atoi(ptr);
1741  }
1742  continue;
1743  }
1744 
1745  if(*ptr == '?' || MATCH(ptr,"help") )
1746  {
1747  td0_help(1);
1748  return(0);
1749  continue;
1750  }
1751  printf("ERROR: bad options:[%s]\n", ptr);
1752  td0_help(1);
1753  return(0);
1754  continue;
1755  }
1756  else if(telediskname == NULL)
1757  {
1758  telediskname = ptr;
1759  }
1760  else if(lifname == NULL)
1761  {
1762  lifname = ptr;
1763  }
1764  else if(MATCH(ptr,"help") )
1765  {
1766  td0_help(1);
1767  return(1);
1768  }
1769  else
1770  {
1771  printf("ERROR: bad options:[%s]\n", ptr);
1772  return(0);
1773  }
1774  }
1775 
1776  if( liftel.u.size != -1)
1777  printf("\tUser Override: sector size = %d\n", liftel.u.size);
1778 
1779  if( liftel.u.sides != -1)
1780  printf("\tUser Override: sides = %d\n", liftel.u.sides);
1781 
1782  if( liftel.u.tracks != -1)
1783  printf("\tUser Override: tracks = %d\n", liftel.u.tracks);
1784 
1785  if(!lifname|| !strlen(lifname))
1786  {
1787  printf("Expected LIF filename\n");
1788  return(0);
1789  }
1790 
1791  if(!telediskname|| !strlen(telediskname))
1792  {
1793  printf("Expected TeleDisk filename\n");
1794  return(0);
1795  }
1796 
1797  if(strcasecmp(telediskname,lifname) == 0)
1798  {
1799  printf("TeleDIsk and LIF names can not be the same\n");
1800  return(0);
1801  }
1802 
1803  lif_t *LIF = lif_calloc(sizeof(lif_t)+4);
1804  if(LIF == NULL)
1805  return(0);
1806 
1807  lif_image_clear(LIF);
1808 
1809 // LIF file name
1810  LIF->name = lif_stralloc(lifname);
1811  if(LIF->name == NULL)
1812  {
1813  lif_close_volume(LIF);
1814  return(0);
1815  }
1817  LIF->fp=fopen(LIF->name,"wb");
1818  if(LIF->fp==NULL)
1819  {
1820  printf("TeleDisk Can't open: %s\n",LIF->name);
1821  lif_close_volume(LIF);
1822  return (0);
1823  }
1824 
1825  if( !td0_open((disk_t *) &disk, telediskname) )
1826  {
1827  if(disk.fi)
1828  fclose(disk.fi);
1829  lif_close_volume(LIF);
1830  return(0);
1831  }
1832 
1833  if( !td0_read_disk( (disk_t *) &disk ) )
1834  {
1835  if(disk.fi)
1836  fclose(disk.fi);
1837  lif_close_volume(LIF);
1838  return(0);
1839  }
1840 
1841  if( !td0_analize_format((disk_t *) &disk) )
1842  {
1843  if(disk.fi)
1844  fclose(disk.fi);
1845  lif_close_volume(LIF);
1846  return(0);
1847  }
1848 
1849  if( !td0_save_lif((disk_t *) &disk, LIF) )
1850  {
1851  if(disk.fi)
1852  fclose(disk.fi);
1853  lif_close_volume(LIF);
1854  return(0);
1855  }
1856 
1858  printf("\n");
1859  printf("Done LIF image: [%s] wrote: [%04lXh] sectors\n\n",
1860  LIF->name, (long)liftel.writeindex);
1861 
1863  lif_close_volume(LIF);
1864 
1865  lif_dir(lifname);
1866  return(1);
1867 }
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
liftel_t::state
int state
Definition: td02lif.h:232
td_track_t::CRC
uint8_t CRC
Definition: td02lif.h:145
track_t::First
int First
Definition: td02lif.h:207
printf
MEMSPACE int printf(const char *format,...)
disk_t::compressed
int compressed
Definition: td02lif.h:217
sector_t::cylinder
int cylinder
Definition: td02lif.h:195
lif_t::filestart
uint32_t filestart
Definition: lifutils.h:120
tm::tm_min
int tm_min
Definition: time.h:43
debuglevel
int debuglevel
Debug flag - used to log GPIB and emulator messages.
Definition: gpib_task.c:33
liftel_t::last
int last[2]
Definition: td02lif.h:251
MATCH
MEMSPACE int MATCH(char *str, char *pat)
Compare two strings.
Definition: parsing.c:143
td_comment_t
Comment header Only used when bit 7 of TrackDensity in TeleDisk Header is set.
Definition: td02lif.h:121
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
disk_t::comment
uint8_t * comment
Definition: td02lif.h:221
lif_t::DIR
lifdir_t DIR
Definition: lifutils.h:129
disk_t::td_comment
td_comment_t td_comment
Definition: td02lif.h:220
lif_t
Master LIF data structure Contains image file name Volume Structure Current Directory Entry read/writ...
Definition: lifutils.h:114
td_track_t::PCyl
uint8_t PCyl
Definition: td02lif.h:140
liftel_t::Size
int Size
Definition: td02lif.h:238
td0_sectorinfo
void td0_sectorinfo(td_sector_t *P)
Display TeleDisk sector data.
Definition: td02lif.c:458
free
#define free(p)
Definition: user_config.h:126
TD0_DONE
@ TD0_DONE
Definition: td02lif.h:85
lifdir_t::FileSectors
uint32_t FileSectors
Definition: lifutils.h:102
MAXTRACKS
#define MAXTRACKS
Definition: td02lif.h:64
sector_t::side
int side
Definition: td02lif.h:196
TD_TRACK_SIZE
#define TD_TRACK_SIZE
Size of TeleDisk track header size.
Definition: td02lif.h:49
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
tm::tm_wday
int tm_wday
Definition: time.h:48
user_config.h
td_header_t::CRC
uint16_t CRC
Definition: td02lif.h:116
lif_str2dir
MEMSPACE void lif_str2dir(uint8_t *B, lif_t *LIF)
Convert byte vector into byte vector.
Definition: lifutils.c:649
td0_trackinfo
void td0_trackinfo(disk_t *disk, int trackind, int index)
Display TeleDisk track sector values.
Definition: td02lif.c:443
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
putchar
int putchar(int c)
put a character to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:363
td_comment_t::Minute
uint8_t Minute
Definition: td02lif.h:129
track_t::sectors
sector_t sectors[MAXSECTORS]
Definition: td02lif.h:210
td0_unpack_disk_header
int td0_unpack_disk_header(uint8_t *B, td_header_t *p)
Extract TeleDisk image header data in architecture nutral way.
Definition: td02lif.c:151
lif_stralloc
MEMSPACE char * lif_stralloc(char *str)
Allocate and copy a string Displays message on errors.
Definition: lifutils.c:322
dst
dst_t dst
DST start and stop in GMT epoch.
Definition: time.c:52
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
td_header_t::DosMode
uint8_t DosMode
Definition: td02lif.h:112
tm::tm_sec
int tm_sec
Definition: time.h:42
liftel
liftel_t liftel
Teledisk liftel analysis and user overrides.
Definition: td02lif.c:60
track_t::Size
int Size
Definition: td02lif.h:209
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
td0_crc16
uint16_t td0_crc16(uint8_t *B, uint16_t crc, uint16_t poly, int size)
Compute CRC16 of 8bit data.
Definition: td02lif.c:87
TD0_DIRECTORY
@ TD0_DIRECTORY
Definition: td02lif.h:82
TD0_WAIT_DIRECTORY
@ TD0_WAIT_DIRECTORY
Definition: td02lif.h:81
td_sector_t::Cyl
uint8_t Cyl
Definition: td02lif.h:153
td_comment_t::Year
uint8_t Year
Definition: td02lif.h:125
tm::tm_mon
int tm_mon
Definition: time.h:46
lif_t::filesectors
uint32_t filesectors
Definition: lifutils.h:121
td_track_t::PSide
uint8_t PSide
Definition: td02lif.h:141
td_header_t::DriveType
uint8_t DriveType
Definition: td02lif.h:102
td_comment_t::CRC
uint16_t CRC
Definition: td02lif.h:123
tm::tm_mday
int tm_mday
Definition: time.h:45
TD_SECTOR_SIZE
#define TD_SECTOR_SIZE
Size of TeleDisk sector header size.
Definition: td02lif.h:51
td_sector_t::SizeExp
uint8_t SizeExp
Definition: td02lif.h:156
liftel_t::sectors
int sectors[2]
Definition: td02lif.h:252
lif_str2vol
MEMSPACE void lif_str2vol(uint8_t *B, lif_t *LIF)
Convert byte vector into LIF volume records.
Definition: lifutils.c:608
disk_t::t
time_t t
Definition: td02lif.h:218
tm::tm_hour
int tm_hour
Definition: time.h:44
liftel_t::size
int size[2]
Definition: td02lif.h:250
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
liftel_t::first
int first[2]
Definition: td02lif.h:249
td0_help
void td0_help(int full)
Definition: td02lif.c:1556
td0_init_sectors
void td0_init_sectors(disk_t *disk)
Initialize track sector information Optionally decompress the data if decompression is enabled.
Definition: td02lif.c:1614
MAXSECTORS
#define MAXSECTORS
Maximum number of sectors per track Used for sector data and sorting tables.
Definition: td02lif.h:55
disk
disk_t disk
Dave Dunfiled LZSS expander.
Definition: td02lif.c:57
disk_t::track
track_t track[MAXTRACKS]
Definition: td02lif.h:222
lif_t::fp
FILE * fp
Definition: lifutils.h:117
td_sector_t::Side
uint8_t Side
Definition: td02lif.h:154
td_header_t::TrackDensity
uint8_t TrackDensity
Definition: td02lif.h:107
track_t::Cyl
int Cyl
Definition: td02lif.h:204
td_header_t
TeleDisk Image Header.
Definition: td02lif.h:89
lif_dir
MEMSPACE void lif_dir(char *lifimagename)
Display a LIF image file directory.
Definition: lifutils.c:1734
sector_t::sector
int sector
Definition: td02lif.h:197
liftel_t
Master TeleDisk Format Analisis structure We look for the specifications of LIF image stored inside t...
Definition: td02lif.h:228
lif_dir2str
MEMSPACE void lif_dir2str(lif_t *LIF, uint8_t *B)
Convert LIF directory records into byte vector.
Definition: lifutils.c:631
td0_rle
int td0_rle(uint8_t *dst, uint8_t *src, int max)
Expand a Run Length encoded TeleDisk sector Notes: the source length is encoded in the source data Cr...
Definition: td02lif.c:307
sector_t::size
int size
Definition: td02lif.h:198
liftel_t::s
struct liftel_t::@8 s
lifdir_t::filename
uint8_t filename[10+1]
Definition: lifutils.h:99
NULL
#define NULL
Definition: user_config.h:85
td0_unpack_track_header
int td0_unpack_track_header(uint8_t *B, td_track_t *p)
Extract TeleDisk track header data in architecture nutral way.
Definition: td02lif.c:201
td0_hexdump
void td0_hexdump(uint8_t *data, int size)
hex listing of data
Definition: td02lif.c:110
LIF_DEBUG
#define LIF_DEBUG
Definition: debug.h:14
td0_unpack_sector_header
int td0_unpack_sector_header(uint8_t *B, td_sector_t *p)
Extract TeleDisk sector header data in architecture nutral way.
Definition: td02lif.c:232
__file
FILE type structure.
Definition: posix.h:158
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
td_comment_t::Month
uint8_t Month
Definition: td02lif.h:126
td0_unpack_comment_header
int td0_unpack_comment_header(uint8_t *B, td_comment_t *p)
Extract TeleDisk comment header data in architecture nutral way.
Definition: td02lif.c:178
time.h
Common Linux/POSIX time functions.
lifvol_t::DirStartSector
uint32_t DirStartSector
Definition: lifutils.h:62
lif_t::sectors
uint32_t sectors
Definition: lifutils.h:118
TD_COMMENT_SIZE
#define TD_COMMENT_SIZE
Size of TeleDisk comment header.
Definition: td02lif.h:47
disk_t::fi
FILE * fi
Definition: td02lif.h:215
liftel_t::Sectors
int Sectors
Parameters of LIf image inside TeleDisk image - AFTER analysis.
Definition: td02lif.h:237
disk_t::td_header
td_header_t td_header
Definition: td02lif.h:219
track_t::Sectors
int Sectors
Definition: td02lif.h:206
lifvol_t::date
uint8_t date[6]
Definition: lifutils.h:71
disk_t::td0_name
char * td0_name
Definition: td02lif.h:216
TD0_WAIT_FILE
@ TD0_WAIT_FILE
Definition: td02lif.h:83
TD0_START
@ TD0_START
Definition: td02lif.h:80
fclose
MEMSPACE int fclose(FILE *stream)
POSIX close a file stream.
Definition: posix.c:1261
liftel_t::tracks
int tracks
Definition: td02lif.h:286
td_header_t::Header
uint8_t Header[2+1]
Definition: td02lif.h:91
lif_t::VOL
lifvol_t VOL
Definition: lifutils.h:128
atoi
MEMSPACE int atoi(const char *str)
Convert ASCII string to number in base 10.
Definition: mathio.c:284
td_comment_t::Second
uint8_t Second
Definition: td02lif.h:130
td_header_t::Sides
uint8_t Sides
Definition: td02lif.h:115
MAXCYL
#define MAXCYL
Maximum number of tracks per disk.
Definition: td02lif.h:62
lifutils.h
LIF file utilities.
td_comment_t::Size
uint16_t Size
Definition: td02lif.h:124
time
MEMSPACE time_t time(time_t *t)
Return second from epoch - POSIX function.
Definition: time.c:843
liftel_t::Sides
int Sides
Definition: td02lif.h:240
TD0_FILE
@ TD0_FILE
Definition: td02lif.h:84
td0_open
int td0_open(disk_t *disk, char *name)
Opne TeleDisk image file process header and optional comment block.
Definition: td02lif.c:506
MAXSIDES
#define MAXSIDES
Maximum number of sides per cyinder TeleDisk only trargetted floppies with 2 sides max!
Definition: td02lif.h:59
td0_analize_format
int td0_analize_format(disk_t *disk)
Analize TeleDisk disk sector data Problem: We may have a disk that was reformatted multiple times Eac...
Definition: td02lif.c:896
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
td_sector_t::Sector
uint8_t Sector
Definition: td02lif.h:155
td_header_t::ChkSig
uint8_t ChkSig
Definition: td02lif.h:93
td_comment_t::Day
uint8_t Day
Definition: td02lif.h:127
track_t::Side
int Side
Definition: td02lif.h:205
td0_B2S
void td0_B2S(uint8_t *B, int index, uint8_t *name, int size)
Create a string from data that has no EOS but known size.
Definition: td02lif.c:68
td0_compressed
void td0_compressed(int flag)
Enable TeleDisk image compression mode.
Definition: td02lif.c:250
td_comment_t::Hour
uint8_t Hour
Definition: td02lif.h:128
lifsup.h
LIF file utilities - utilities extracted from hp85disk project for stand alone use.
lifvol_t::DirSectors
uint32_t DirSectors
Definition: lifutils.h:65
liftel_t::sides
int sides
Definition: td02lif.h:285
calloc
#define calloc(n, s)
Definition: user_config.h:129
track_t::Last
int Last
Definition: td02lif.h:208
td0_density2bitrate
long td0_density2bitrate(uint8_t density)
Convert Denisity to bitrate @We don't care about the density except as a possible filter.
Definition: td02lif.c:472
tm
POSIX struct tm.
Definition: time.h:40
tm::tm_year
int tm_year
Definition: time.h:47
td_track_t
Track Header Contains the Physical Sectors, Cylinder, Side number From the original physical disk use...
Definition: td02lif.h:136
td0_read
int td0_read(void *p, int osize, int size, FILE *fp)
Read TeleDisk image data block Optionally decompress the data if decompression is enabled.
Definition: td02lif.c:264
time_t
uint32_t time_t
type of EPOCH result.
Definition: time.h:34
liftel_t::Cylinders
int Cylinders
Definition: td02lif.h:241
lifdir_t::FileType
uint16_t FileType
Definition: lifutils.h:100
td_sector_t::Flags
uint8_t Flags
Definition: td02lif.h:158
tm::tm_yday
int tm_yday
Definition: time.h:49
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
disk_t
Definition: td02lif.h:213
liftel_t::sectorindex
long sectorindex
Definition: td02lif.h:233
td_header_t::VolNO
uint8_t VolNO
Definition: td02lif.h:92
lif_calloc
MEMSPACE void * lif_calloc(long size)
Allocate and clear memory Displays message on errors.
Definition: lifutils.c:293
EOF
#define EOF
Definition: ff.h:349
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
td0_read_disk
int td0_read_disk(disk_t *disk)
Read TeleDisk image file and save all sector data.
Definition: td02lif.c:643
TD_HEADER_SIZE
#define TD_HEADER_SIZE
Size of TeleDisk image header.
Definition: td02lif.h:45
liftel_t::error
int error
Definition: td02lif.h:231
td0_save_lif
int td0_save_lif(disk_t *disk, lif_t *LIF)
save remaining sectors as LIF data
Definition: td02lif.c:1303
sector_t::data
uint8_t * data
Definition: td02lif.h:199
td0_init_liftel
void td0_init_liftel()
TeleDisk image Analisis structure Find attributes of LIF image stored in TeleDisk image.
Definition: td02lif.c:1577
td_track_t::PSectors
uint8_t PSectors
Definition: td02lif.h:138
liftel_t::u
struct liftel_t::@9 u
TD0_INIT
@ TD0_INIT
Definition: td02lif.h:79
td0_save_lif_sector
int td0_save_lif_sector(disk_t *disk, uint8_t *data, int size, lif_t *LIF)
Process all sectors on a track from TeleDisk image.
Definition: td02lif.c:1366
lif_close_volume
MEMSPACE void lif_close_volume(lif_t *LIF)
Free LIF structure and close any files.
Definition: lifutils.c:1220
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
td_header_t::TDVersion
uint8_t TDVersion
Definition: td02lif.h:96
td_sector_t::CRC
uint8_t CRC
Definition: td02lif.h:167
liftel_t::writeindex
long writeindex
Definition: td02lif.h:234
td_sector_t
Sector Header Contents of the Sector ID fields which preceeded the sector data From the original phys...
Definition: td02lif.h:151
lifdir_t::date
uint8_t date[6]
Definition: lifutils.h:103
liftel_t::Tracks
int Tracks
Definition: td02lif.h:239
td_header_t::Density
uint8_t Density
Definition: td02lif.h:97