HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
drives.c
Go to the documentation of this file.
1 
11 #include "user_config.h"
12 
13 #include <stdint.h>
14 #include "defines.h"
15 #include "drives.h"
16 #include "gpib_hal.h"
17 #include "gpib.h"
18 #include "gpib_task.h"
19 #include "amigo.h"
20 #include "ss80.h"
21 #include <time.h>
22 #include "lifutils.h"
23 #include "debug.h"
24 
25 extern hpdir_t hpdir;
26 
28 #define MAX_STACK 5
29 static int stack_ind = 0;
30 static int stack_p[MAX_STACK];
31 
33 
36 
40 
41 #ifdef AMIGO
42 AMIGODiskType *AMIGOp = NULL;
44 AMIGOStateType *AMIGOs = NULL;
45 #endif
46 
49 
50 #if defined (SET_DEFAULTS)
51 // =============================================
52 PRINTERDeviceType PRINTERDeviceDefault =
53 {
54  {
55  2, // GPIB Address
56  0xff, // PPR unused
57  "/printer.txt"
58  }
59 };
60 
61 // =============================================
62 #if defined(HP9122D)
63 SS80DiskType SS80DiskDefault =
65 {
66  { // HEADER
67  0, // GPIB Address
68  0, // PPR
69  "/ss80.lif" // File name
70  },
71  { // CONFIG
72  0x222 // ID
73  },
74  { // CONTROLLER
75  0x8001, // Installed Units 1 (15 is always on)
76  100, // GPIB data transfer rate in kB/s on the bus
77  4, // 5 = SS/80 integrated single-unit controller
78  },
79  { // UNIT
80  0, // Generic Unit Type, 0 = fixed, 1 = floppy, 2 = tape
81  0x00091220, // BCD Device number XX XX XY, X=Unit, Y=option
82  0x100, // Bytes per block
83  1, // Buffered Blocks
84  0, // Burst size = 0 for SS80
85  2000, // Block time in microseconds
86  50, // Continous transfer time in kB/s
87  10000, // Retry time in 0.01 secods
88  10000, // Access time in 0.01 seconds
89  31, // Maximum interleave factor
90  1, // Fixed volume byte, one bit per volume
91  1 // Removable volume byte, one bit per volume
92  },
93  { // VOLUME
94  0, // Maximum Cylinder - not used
95  0, // Maximum Head - not used
96  0, // Maximum Sector - not used
97  2463, // Maximum Block Number
98  1 // Interleave
99  }
100 };
101 #endif // #if defined(HP9122D)
102 
103 #if defined(HP9134D)
104 SS80DiskType SS80DiskDefault =
106 {
107  { // HEADER
108  0, // GPIB Address
109  0, // PPR
110  "/ss80.lif" // FILE name
111  },
112  { // CONFIG
113  0x222 // ID
114  },
115  { // CONTROLLER
116  0x8001, // Installed Units 1 (15 is always on)
117  100, // GPIB data transfer rate in kB/s on the bus
118  4, // 4 = SS/80 integrated single-unit controller
119  },
120  { // UNIT
121  0, // Generic Unit Type, 0 = fixed
122  0x091340, // BCD Device number XX XX XY, X=Unit, Y=option
123  0x100, // Bytes per block
124  1, // Buffered Blocks
125  0, // Burst size = 0 for SS80
126  2000, // Block time in microseconds
127  50, // Continous transfer time in kB/s
128  10000, // Retry time in 0.01 secods
129  10000, // Access time in 0.01 seconds
130  31, // Maximum interleave factor
131  1, // Fixed volume byte, one bit per volume
132  1 // Removable volume byte, one bit per volume
133  },
134  { // VOLUME
135  0, // Maximum Cylinder - not used
136  0, // Maximum Head - not used
137  0, // Maximum Sector - not used
138  58175, // Maximum Block Number
139  1 // Interleave
140  }
141 };
142 #endif // #if defined(HP9134D)
143 
144 #ifdef AMIGO
145 
146 #if defined(HP9121)
147 AMIGODiskType AMIGODiskDefault =
149 {
150  { // HEADER
151  1, // GPIB Address
152  1, // PPR
153  "/amigo.lif" // FILE name
154  },
155  { // CONFIG
156  0x0104 // ID
157  },
158  { // GEOMETRY
159  256, // Bytes Per Sector
160  16, // Sectors Per Track
161  2, // Sides
162  35 // Cylinders
163  }
164 };
165 #endif // #if defined(HP9121)
166 #endif // ifdef AMIGO
167 #endif // SET_DEFAULTS
168 
170 {
171  {"ACCESS_TIME", TOK_ACCESS_TIME},
172  {"ADDRESS", TOK_ADDRESS},
173  {"AMIGO", TOK_AMIGO},
174  {"BLOCKS", TOK_BLOCKS},
175  {"BLOCK_TIME", TOK_BLOCK_TIME},
176  {"BUFFERED_BLOCKS", TOK_BUFFERED_BLOCKS},
177  {"BURST_SIZE", TOK_BURST_SIZE},
178  {"BYTES_PER_BLOCK", TOK_BYTES_PER_BLOCK},
179  {"BYTES_PER_SECTOR", TOK_BYTES_PER_SECTOR},
180  {"CONFIG", TOK_CONFIG},
181  {"CONTINUOUS_TRANSFER_RATE", TOK_CONTINUOUS_TRANSFER_RATE},
182  {"CONTINOUS_TRANSFER_RATE", TOK_CONTINUOUS_TRANSFER_RATE},
183  {"CONTINIOUS_TRANSFER_RATE", TOK_CONTINUOUS_TRANSFER_RATE},
184  {"CONTROLLER", TOK_CONTROLLER},
185  {"CS80", TOK_CS80},
186  {"CYLINDERS", TOK_CYLINDERS},
187  {"DEBUG", TOK_DEBUG},
188  {"DEVICE_NUMBER", TOK_DEVICE_NUMBER},
189  {"DRIVE", TOK_DRIVE},
190  {"END", TOK_END},
191  {"FILE", TOK_FILE},
192  {"FIXED_VOLUMES", TOK_FIXED_VOLUMES},
193  {"GEOMETRY", TOK_GEOMETRY},
194  {"HEADER", TOK_HEADER},
195  {"HEADS", TOK_HEADS},
196  {"ID", TOK_ID},
197  {"INTERLEAVE", TOK_INTERLEAVE},
198  {"MAX_BLOCK_NUMBER", TOK_MAX_BLOCK_NUMBER},
199  {"MAX_CYLINDER", TOK_MAX_CYLINDER},
200  {"MAX_HEAD", TOK_MAX_HEAD},
201  {"MAXIMUM_INTERLEAVE", TOK_MAXIMUM_INTERLEAVE},
202  {"MAX_SECTOR", TOK_MAX_SECTOR},
203  {"OPTIMAL_RETRY_TIME", TOK_OPTIMAL_RETRY_TIME},
204  {"PPR", TOK_PPR},
205  {"PRINTER", TOK_PRINTER},
206  {"REMOVABLE_VOLUMES", TOK_REMOVABLE_VOLUMES},
207  {"SECTORS_PER_TRACK", TOK_SECTORS_PER_TRACK},
208  {"SS80", TOK_SS80},
209  {"SS80_DEFAULT", TOK_SS80_DEFAULT},
210  {"TRANSFER_RATE", TOK_TRANSFER_RATE},
211  {"TYPE", TOK_TYPE},
212  {"UNIT", TOK_UNIT},
213  {"UNITS_INSTALLED", TOK_UNITS_INSTALLED},
214  {"UNIT_TYPE", TOK_UNIT_TYPE},
215  {"VOLUME", TOK_VOLUME},
216  {"", TOK_INVALID},
217 };
218 
224 void print_var_P(__memx const char *str, uint32_t val)
225 {
226  char tmp[128];
227  int i=0;
228  while( *str && i < (int) (sizeof(tmp) - 2) )
229  tmp[i++] = *str++;
230  tmp[i++] = 0;
231 
232  printf(" %-25s = %8lxH (%ld)\n", tmp, val, val);
233 }
234 
235 
241 void print_str_P(__memx const char *str, char *arg)
242 {
243  char tmp[64];
244  int i=0;
245  while( *str && i < 62)
246  tmp[i++] = *str++;
247  tmp[i++] = 0;
248  printf(" %-25s = \"%s\"\n", tmp, arg);
249 }
250 
251 
256 int8_t tok_index(char *str)
257 {
258  int8_t i;
259  for (i = 0; (tokens[i].tok != TOK_INVALID) ; ++i )
260  {
261  if( MATCHI(str,tokens[i].name) )
262  return(i);
263  }
264  return(-1);
265 }
266 
267 
272 char *tok_name(uint8_t tok)
273 {
274  int i;
275  for (i = 0; (tokens[i].tok != TOK_INVALID) ; ++i )
276  {
277  if(tok == tokens[i].tok)
278  return(tokens[i].name);
279  }
280  return("");
281 }
282 
283 
290 void print_tok_val(uint8_t tok, uint8_t spaces, uint32_t val)
291 {
292  char *ptr = tok_name(tok);
293  while(spaces--)
294  putchar(' ');
295  printf("%-25s = %8lxH (%ld)\n", ptr, val, val);
296 }
297 
298 
305 void print_tok_str(uint8_t tok, uint8_t spaces, char *str)
306 {
307  char *ptr = tok_name(tok);
308  while(spaces--)
309  putchar(' ');
310  printf("%-25s = %s\n", ptr, str);
311 }
312 
313 
321 void print_tok(uint8_t tok, uint8_t spaces)
322 {
323  char *ptr = tok_name(tok);
324  while(spaces--)
325  putchar(' ');
326  printf("%s\n",ptr);
327 }
328 
329 
336 int Read_Config(char *name)
337 {
338  FILE *cfg;
339  int state = START_STATE;
340  int errors = 0;
341 
346 
347 #ifdef AMIGO
348  AMIGODiskType *AMIGOp = NULL;
350 #endif
351 
352  char *ptr;
353  int8_t index = 0;
354  int8_t ind;
355  val_t val;
356  int tok = TOK_INVALID;
357  int lines = 0;
358  int8_t ppr;
359  int8_t address;
360 
361  char str[128];
362  char token[128];
363 
364  init_Devices();
365 
366  printf("Reading: %s\n", name);
367  cfg = fopen(name, "rb");
368  if(cfg == NULL)
369  {
370  ++errors;
371 //FIXME
372  perror("Read_Config - open");
373  printf("Read_Config: open(%s) failed\n", name);
375  return(errors);
376  }
377 
378  while( (ptr = fgets(str, sizeof(str)-2, cfg)) != NULL)
379  {
380  ++lines;
381 
382  ptr = get_token(str, token, sizeof(token)-2);
383 
384 // Skip comments
385  if(token[0] == 0 || token[0] == '#')
386  continue;
387 
388  val.l = 0;
389  tok = TOK_INVALID;
390 
391  if( ( ind = tok_index(token) ) == -1)
392  {
393  printf("Unexpected token: %s, at line:%d\n", token,lines);
394  ++errors;
395  continue;
396  }
397  tok = tokens[ind].tok;
398 
399 // get optional argument
400  ptr = get_token(ptr, token, sizeof(token)-2);
401  if(MATCH(token,"="))
402  ptr = get_token(ptr, token, sizeof(token)-2);
403  val.l = get_value(token);
404  val.w = (uint16_t) 0xFFFF & val.l;
405  val.b = (uint8_t) 0xFF & val.l;
406 
407 //FIXME check for state and last state
408  if(tok == TOK_END )
409  {
410  state = pop_state();
411  continue;
412  }
413 
414  switch(state)
415  {
416  case START_STATE:
417  switch(tok)
418  {
419  case TOK_SS80_DEFAULT:
420  push_state(state);
421  state = SS80_STATE;
423  if(index == -1)
424  state = START_STATE;
425  else
426  SS80p = (SS80DiskType *) Devices[index].dev;
427  break;
428  case TOK_SS80:
429  case TOK_CS80:
430  push_state(state);
431  state = SS80_STATE;
432  index = alloc_device(SS80_TYPE);
433  if(index == -1)
434  {
435  state = START_STATE;
436  }
437  else
438  {
439  SS80p = (SS80DiskType *) Devices[index].dev;
440  hpdir_set_parameters(index,token); // also SS80p->HEADER.model
441  }
442  break;
443 #ifdef AMIGO
444  case TOK_AMIGO:
445  push_state(state);
446  state = AMIGO_STATE;
447  index = alloc_device(AMIGO_TYPE);
448  if(index == -1)
449  {
450  state = START_STATE;
451  }
452  else
453  {
454  AMIGOp = (AMIGODiskType *) Devices[index].dev;
455  hpdir_set_parameters(index,token); // also sets AMIGOp->HEADER.model
456  }
457  break;
458 #endif
459  case TOK_PRINTER:
460  push_state(state);
461  state = PRINTER_STATE;
462  index = alloc_device(PRINTER_TYPE);
463  if(index == -1)
464  state = START_STATE;
465  else
466  PRINTERp = (PRINTERDeviceType *) Devices[index].dev;
467  break;
468  case TOK_DEBUG:
469  debuglevel = val.w;
470  break;
471  default:
472  printf("Unexpected token: %s, at line:%d\n", ptr,lines);
473  errors++;
474  break;
475  }
476  break;
477 
478  case PRINTER_STATE:
479  if(tok == TOK_CONFIG)
480  {
481  push_state(state);
482  state = PRINTER_CONFIG;
483  }
484  else
485  {
486  printf("Unexpected PRINTER token: %s, at line:%d\n", ptr,lines);
487  ++errors;
488  break;
489  }
490  break;
491 
492  case PRINTER_CONFIG:
493  if(tok == TOK_ADDRESS)
494  {
495  address = val.b;
496  Devices[index].ADDRESS = address;
497  PRINTERp->HEADER.ADDRESS = address;
498 // NO PPR
499  Devices[index].PPR = 0xff;
500  PRINTERp->HEADER.PPR = 0xff;
501  }
502  else
503  {
504  printf("Unexpected PRINTER CONFIG token: %s, at line:%d\n", ptr,lines);
505  ++errors;
506  }
507  break;
508 
509  case SS80_STATE:
510  switch(tok)
511  {
512  case TOK_HEADER:
513  push_state(state);
514  state = SS80_HEADER;
515  break;
516  case TOK_CONFIG:
517  push_state(state);
518  state = SS80_CONFIG;
519  break;
520  case TOK_CONTROLLER:
521  push_state(state);
522  state = SS80_CONTROLLER;
523  break;
524  case TOK_UNIT:
525  push_state(state);
526  state = SS80_UNIT;
527  break;
528  case TOK_VOLUME:
529  push_state(state);
530  state = SS80_VOLUME;
531  break;
532  default:
533  printf("Unexpected SS80 START token: %s, at line:%d\n", ptr,lines);
534  ++errors;
535  break;
536  }
537  break;
538 
539  case SS80_HEADER:
540  switch(tok)
541  {
542  case TOK_ADDRESS:
543  address = val.b;
544  Devices[index].ADDRESS = address;
545  SS80p->HEADER.ADDRESS = address;
546  break;
547  case TOK_PPR:
548  ppr = val.b;
549  Devices[index].PPR = ppr;
550  SS80p->HEADER.PPR = ppr;
551  break;
552  case TOK_FILE:
554  break;
555  default:
556  printf("Unexpected SS80 CONFIG token: %s, at line:%d\n", ptr,lines);
557  ++errors;
558  break;
559  }
560  break;
561 
562  case SS80_CONFIG:
563  if(tok == TOK_ID )
564  {
565  SS80p->CONFIG.ID = val.w;
566  }
567  else
568  {
569  printf("Unexpected SS80 CONFIG token: %s, at line:%d\n", ptr,lines);
570  ++errors;
571  }
572  break;
573 
574  case SS80_CONTROLLER:
575  switch(tok)
576  {
577  case TOK_UNITS_INSTALLED:
579  break;
580  case TOK_TRANSFER_RATE:
582  break;
583  case TOK_TYPE:
584  SS80p->CONTROLLER.TYPE = val.b;
585  break;
586  default:
587  printf("Unexpected SS80 CONTROLLER token: %s, at line:%d\n", ptr,lines);
588  ++errors;
589  break;
590  }
591  break;
592 
593  case SS80_UNIT:
594  switch(tok)
595  {
596  case TOK_UNIT_TYPE:
597  SS80p->UNIT.UNIT_TYPE = val.b;
598  break;
599  case TOK_DEVICE_NUMBER:
600  SS80p->UNIT.DEVICE_NUMBER = val.l;
601  break;
602  case TOK_BYTES_PER_BLOCK:
603  SS80p->UNIT.BYTES_PER_BLOCK = val.w;
604  break;
605  case TOK_BUFFERED_BLOCKS:
606  SS80p->UNIT.BUFFERED_BLOCKS = 1 & val.b;
607  break;
608  case TOK_BURST_SIZE:
609  SS80p->UNIT.BURST_SIZE = val.b;
610  break;
611  case TOK_BLOCK_TIME:
612  SS80p->UNIT.BLOCK_TIME = val.w;
613  break;
616  break;
619  break;
620  case TOK_ACCESS_TIME:
621  SS80p->UNIT.ACCESS_TIME = val.w;
622  break;
625  break;
626  case TOK_FIXED_VOLUMES:
627  SS80p->UNIT.FIXED_VOLUMES = val.b;
628  break;
630  SS80p->UNIT.REMOVABLE_VOLUMES = val.b;
631  break;
632  default:
633  printf("Unexpected SS80 UNIT token: %s, at line:%d\n", ptr,lines);
634  ++errors;
635  break;
636  }
637  break;
638 
639  case SS80_VOLUME:
640  switch(tok)
641  {
642  case TOK_MAX_CYLINDER:
643  SS80p->VOLUME.MAX_CYLINDER = val.l;
644  break;
645  case TOK_MAX_HEAD:
646  SS80p->VOLUME.MAX_HEAD = val.b;
647  break;
648  case TOK_MAX_SECTOR:
649  SS80p->VOLUME.MAX_SECTOR = val.w;
650  break;
653  break;
654  case TOK_INTERLEAVE:
655  SS80p->VOLUME.INTERLEAVE = val.b;
656  break;
657  default:
658  printf("Unexpected SS80 VOLUME token: %s, at line:%d\n", ptr,lines);
659  ++errors;
660  break;
661  }
662  break;
663 
664 #ifdef AMIGO
665  case AMIGO_STATE:
666  switch(tok)
667  {
668  case TOK_HEADER:
669  push_state(state);
670  state = AMIGO_HEADER;
671  break;
672  case TOK_CONFIG:
673  push_state(state);
674  state = AMIGO_CONFIG;
675  break;
676  case TOK_GEOMETRY:
677  push_state(state);
678  state = AMIGO_GEOMETRY;
679  break;
680  default:
681  printf("Unexpected AMIGO START token: %s, at line:%d\n", ptr,lines);
682  ++errors;
683  break;
684  }
685  break;
686 
687  case AMIGO_HEADER:
688  switch(tok)
689  {
690  case TOK_DRIVE:
691  break;
692  case TOK_ADDRESS:
693  address = val.b;
694  Devices[index].ADDRESS = address;
695  AMIGOp->HEADER.ADDRESS = address;
696  break;
697  case TOK_PPR:
698  ppr = val.b;
699  Devices[index].PPR = ppr;
700  AMIGOp->HEADER.PPR = ppr;
701  break;
702  case TOK_FILE:
703  AMIGOp->HEADER.NAME = stralloc(token);
704  break;
705  default:
706  printf("Unexpected HEADER CONFIG token: %s, at line:%d\n", ptr,lines);
707  ++errors;
708  break;
709  }
710  break;
711 
712  case AMIGO_CONFIG:
713  if(tok == TOK_ID )
714  {
715  AMIGOp->CONFIG.ID = val.w;
716  }
717  else
718  {
719  printf("Unexpected AMIGO CONFIG token: %s, at line:%d\n", ptr,lines);
720  ++errors;
721  }
722  break;
723 
724  case AMIGO_GEOMETRY:
725  switch(tok)
726  {
728  AMIGOp->GEOMETRY.BYTES_PER_SECTOR = val.w;
729  break;
731  AMIGOp->GEOMETRY.SECTORS_PER_TRACK = val.w;
732  break;
733  case TOK_HEADS:
734  AMIGOp->GEOMETRY.HEADS = val.w;
735  break;
736  case TOK_CYLINDERS:
737  AMIGOp->GEOMETRY.CYLINDERS = val.w;
738  break;
739  default:
740  printf("Unexpected AMIGO GEMETRY token: %s, at line:%d\n", ptr,lines);
741  ++errors;
742  break;
743  }
744  break;
745 #endif // #ifdef AMIGO
746  default:
747  printf("Unexpected STATE: %s, at line:%d\n", ptr,lines);
748  ++errors;
749  break;
750 
751  } // switch
752  } //while
753  if(state != START_STATE)
754  {
755  printf("Missing END statement at line:%d\n", lines);
756  ++errors;
757  }
758  printf("Read_Config: read(%d) lines\n", lines);
759  if(errors)
760  printf("Read_Config: ****** errors(%d) ******\n", errors);
761 
762  if(fclose(cfg) == EOF)
763  {
764  perror("Read_Config - close error");
765  ++errors;
766  }
767 
768 // Post process and fixup any devices
769  verify_devices();
770 
771  return(errors);
772 }
773 
774 
778 void display_Addresses( int verbose )
779 {
780  int i;
781 
782  printf("Device Addresses\n");
783  for(i=0;i<MAX_DEVICES;++i)
784  {
785  if(Devices[i].TYPE == NO_TYPE)
786  continue;
787 
788  if(Devices[i].TYPE == SS80_TYPE || Devices[i].TYPE == AMIGO_TYPE)
789  {
790  if(Devices[i].TYPE == SS80_TYPE)
791  {
792  SS80p= (SS80DiskType *)Devices[i].dev;
793  printf("SS80 %s\n", SS80p->HEADER.model);
795  }
796 #ifdef AMIGO
797  if(Devices[i].TYPE == AMIGO_TYPE )
798  {
799  AMIGOp= (AMIGODiskType *)Devices[i].dev;
800  printf("AMIGO %s\n", AMIGOp->HEADER.model);
801  print_tok_str(TOK_FILE, 4, AMIGOp->HEADER.NAME);
802  }
803 #endif
804  print_tok_val(TOK_ADDRESS, 4, (uint32_t) Devices[i].ADDRESS);
805  print_tok_val(TOK_PPR, 4, (uint32_t) Devices[i].PPR);
806  print_tok_val(TOK_BLOCKS, 4, (uint32_t) Devices[i].BLOCKS);
807  }
808  if(Devices[i].TYPE == PRINTER_TYPE )
809  {
810  printf("PRINTER\n");
811  print_tok_val(TOK_ADDRESS, 4, (uint32_t) Devices[i].ADDRESS);
812  }
813 
814  if(verbose)
815  {
816 #if 0
817  int address = Devices[i].ADDRESS;
818  if(Devices[i].TYPE == SS80_TYPE)
819  {
820  printf(" SS80_MLA = %02XH\n",BASE_MLA + address );
821  printf(" SS80_MTA = %02XH\n",BASE_MTA + address );
822  printf(" SS80_MSA = %02XH\n",BASE_MSA + address );
823  }
824 #ifdef AMIGO
825  if(Devices[i].TYPE == AMIGO_TYPE )
826  {
827  printf(" AMIGO_MLA = %02XH\n",BASE_MLA + address );
828  printf(" AMIGO_MTA = %02XH\n",BASE_MTA + address );
829  printf(" AMIGO_MSA = %02XH\n",BASE_MSA + address );
830  }
831 #endif
832  if(Devices[i].TYPE == PRINTER_TYPE )
833  {
834  printf(" PRINTER_MLA = %02XH\n",BASE_MLA + address );
835  printf(" PRINTER_MTA = %02XH\n",BASE_MTA + address );
836  printf(" PRINTER_MSA = %02XH\n",BASE_MSA + address );
837  }
838 #endif
839  }
840  printf("\n");
841 
842  }
843  printf("\n");
844 }
845 
849 void display_Config( int verbose)
850 {
851  int i;
852 
855 
858 
859 #ifdef AMIGO
860  AMIGODiskType *AMIGOp = NULL;
862 #endif
863 
864  printf("Current Configuration Settings\n");
865  for(i=0;i<MAX_DEVICES;++i)
866  {
867  if(Devices[i].TYPE == NO_TYPE)
868  continue;
869 
870  if(Devices[i].TYPE == SS80_TYPE)
871  {
872  SS80p= (SS80DiskType *)Devices[i].dev;
873 
874  printf("SS80 %s\n", SS80p->HEADER.model);
875  printf(" # HP85 BASIC ADDRESS :D7%d0\n", (int) SS80p->HEADER.ADDRESS);
876 
878  print_tok_val(TOK_ADDRESS, 8, (uint32_t) SS80p->HEADER.ADDRESS);
879  print_tok_val(TOK_PPR, 8, (uint32_t) SS80p->HEADER.PPR);
881  print_tok(TOK_END,4);
882 
884  print_tok_val(TOK_ID, 8, (uint32_t) SS80p->CONFIG.ID);
885  print_tok(TOK_END,4);
886 
887 // CONTROLLER
888  if(verbose)
889  {
893  print_tok_val(TOK_TYPE, 8, (uint32_t) SS80p->CONTROLLER.TYPE);
894  print_tok(TOK_END,4);
895  }
896 
897 // UNIT
898  print_tok(TOK_UNIT,4);
899  if(verbose)
900  {
902  }
904  if(verbose)
905  {
916  }
917  print_tok(TOK_END,4);
918 
919 // VOLUME
920  if(verbose)
921  {
928  print_tok(TOK_END,4);
929  }
930  printf(" # BLOCKS = %ld\n", (long)SS80p->VOLUME.MAX_BLOCK_NUMBER+1);
931 
932  print_tok(TOK_END,0);
933  } // SS80_TYPE
934 
935 #ifdef AMIGO
936  if(Devices[i].TYPE == AMIGO_TYPE )
937  {
938  AMIGOp= (AMIGODiskType *)Devices[i].dev;
939 
940  printf("AMIGO %s\n", AMIGOp->HEADER.model);
941  printf(" # HP85 BASIC ADDRESS :D7%d0\n", (int) AMIGOp->HEADER.ADDRESS);
942 
944  print_tok_val(TOK_ADDRESS, 8, (uint32_t) AMIGOp->HEADER.ADDRESS);
945  print_tok_val(TOK_PPR, 8, (uint32_t) AMIGOp->HEADER.PPR);
946  print_tok_str(TOK_FILE, 8, AMIGOp->HEADER.NAME);
947  print_tok(TOK_END,4);
948 
950  print_tok_val(TOK_ID, 8, (uint32_t) AMIGOp->CONFIG.ID);
951  print_tok(TOK_END,4);
952 
953  if(verbose)
954  {
958  print_tok_val(TOK_HEADS, 8, (uint32_t) AMIGOp->GEOMETRY.HEADS);
959  print_tok_val(TOK_CYLINDERS, 8, (uint32_t) AMIGOp->GEOMETRY.CYLINDERS);
960  print_tok(TOK_END,4);
961  }
962  printf(" # BLOCKS = %ld\n", (long) (AMIGOp->GEOMETRY.CYLINDERS * AMIGOp->GEOMETRY.SECTORS_PER_TRACK * AMIGOp->GEOMETRY.HEADS) );
963 
964  print_tok(TOK_END,0);
965  }
966 #endif // #ifdef AMIGO
967 
968  if(Devices[i].TYPE == PRINTER_TYPE )
969  {
971 
973 
976  print_tok(TOK_END,4);
977 
978  print_tok(TOK_END,0);
979  }
980  printf("\n");
981  }
982  printf("\n");
983 }
984 
985 
986 
987 
990 //@return Devices[] index fopr matching type
991 int8_t find_type(int type)
992 {
993  int i;
994  for(i=0;i<MAX_DEVICES;++i)
995  {
996  if( Devices[i].TYPE == type)
997  return(i);
998  }
999  return(-1);
1000 }
1001 
1002 
1005 int8_t count_drive_types(uint8_t type)
1006 {
1007  int i;
1008  int count = 0;
1009  for(i=0;i<MAX_DEVICES;++i)
1010  {
1011  if( Devices[i].TYPE == type )
1012  ++count;
1013  }
1014  return(count);
1015 }
1016 
1017 
1021 char *type_to_str(int type)
1022 {
1023  if(type == NO_TYPE)
1024  return("NO_TYPE");
1025  else if(type == AMIGO_TYPE)
1026  return("AMIGO_TYPE");
1027  else if(type == SS80_TYPE)
1028  return("SS80_TYPE");
1029  else if(type == PRINTER_TYPE)
1030  return("PRINTER_TYPE");
1031  return("INVALID TYPE");
1032 }
1033 
1034 
1038 char *base_to_str(int base)
1039 {
1040  if(base == BASE_MLA)
1041  return("MLA");
1042  else if(base == BASE_MTA)
1043  return("MTA");
1044  else if(base == BASE_MSA)
1045  return("MSA");
1046  return("*INVALID BASE*");
1047 }
1048 
1049 
1052 int8_t find_free()
1053 {
1054  return(find_type(NO_TYPE));
1055 }
1056 
1057 
1063 int8_t find_device(int type, int address, int base)
1064 {
1065  int i;
1066 
1068  if(address < BASE_MLA || address >(BASE_MSA+30))
1069  return(-1);
1070 
1072  if(address < base || address > (base+30))
1073  return(-1);
1074 
1076  address -= base;
1077 
1079  for(i=0;i<MAX_DEVICES;++i)
1080  {
1081  if(Devices[i].TYPE == type && Devices[i].ADDRESS == address)
1082  return(i);
1083  }
1084  return(-1);
1085 }
1086 
1087 
1093 int8_t set_active_device(int8_t index)
1094 {
1095  int type,address;
1096 
1099  if(index == -1)
1100  {
1101  return(0);
1102  }
1103 
1104  if(index < 0 || index >= MAX_DEVICES)
1105  {
1106  if(debuglevel & GPIB_ERR)
1107  printf("set_active_device:(%d) out of range\n", index);
1108  return(0);
1109  }
1110 
1111  type = Devices[index].TYPE;
1112  address = Devices[index].ADDRESS;
1113  if(address < 0 || address > 30)
1114  {
1115  if(debuglevel & GPIB_ERR)
1116  printf("set_active_device: index:%d address: %02XH out of range\n", index,address);
1117  return(0);
1118  }
1119 
1120  if(Devices[index].dev == NULL)
1121  {
1122  if(debuglevel & GPIB_ERR)
1123  printf("set_active_device: index:%d type:%d:%s, dev == NULL\n",
1124  index,type,type_to_str(type));
1125  return(0);
1126  }
1127 
1128  if(type == NO_TYPE)
1129  {
1130  if(debuglevel & GPIB_ERR)
1131  printf("set_active_device: index %d uninitalized type:%d:%s\n",
1132  index,type,type_to_str(type));
1133  return(0);
1134  }
1135 
1136  if(type == PRINTER_TYPE)
1137  {
1138  PRINTERp = (PRINTERDeviceType *) Devices[index].dev;
1139  return(1);
1140  }
1141 
1142  if(type == AMIGO_TYPE || type == SS80_TYPE)
1143  {
1144  if(Devices[index].state == NULL)
1145  {
1146  if(debuglevel & GPIB_ERR)
1147  printf("set_active_device: index: %d type:%d:%s, state == NULL\n",
1148  index,type,type_to_str(type));
1149  return(0);
1150  }
1151 #ifdef AMIGO
1152  if(type == AMIGO_TYPE)
1153  {
1154  AMIGOp = (AMIGODiskType *) Devices[index].dev;
1155  AMIGOs = (AMIGOStateType *) Devices[index].state;
1156  return(1);
1157  }
1158 #endif
1159  if(type == SS80_TYPE)
1160  {
1161  SS80p = (SS80DiskType *) Devices[index].dev;
1162  SS80s = (SS80StateType *) Devices[index].state;
1163  return(1);
1164  }
1165  }
1166  if(debuglevel & GPIB_ERR)
1167  printf("set_active_device:(%d) invalid type:%d:%s\n",
1168  index,type,type_to_str(type));
1169  return(0);
1170 }
1171 
1172 
1177 void SS80_Set_Defaults(int8_t index)
1178 {
1179  int8_t defindex = find_type(SS80_DEFAULT_TYPE);
1180  SS80DiskType *SS80p = (SS80DiskType *) Devices[index].dev;
1181  SS80DiskType *SS80DEFAULTp;
1182 
1183  if(defindex < 0 )
1184  return;
1185 
1186  SS80DEFAULTp = (SS80DiskType *) Devices[defindex].dev;
1187 
1188  SS80p->HEADER.ADDRESS = SS80DEFAULTp->HEADER.ADDRESS;
1189  SS80p->HEADER.PPR = SS80DEFAULTp->HEADER.PPR;
1190  SS80p->HEADER.NAME = stralloc(SS80DEFAULTp->HEADER.NAME);
1191 
1192  SS80p->CONFIG.ID = SS80DEFAULTp->CONFIG.ID;
1195  SS80p->CONTROLLER.TYPE = SS80DEFAULTp->CONTROLLER.TYPE;
1196 
1197  SS80p->UNIT.UNIT_TYPE = SS80DEFAULTp->UNIT.UNIT_TYPE;
1198  SS80p->UNIT.DEVICE_NUMBER = SS80DEFAULTp->UNIT.DEVICE_NUMBER;
1199  SS80p->UNIT.BYTES_PER_BLOCK = SS80DEFAULTp->UNIT.BYTES_PER_BLOCK;
1200  SS80p->UNIT.BUFFERED_BLOCKS = SS80DEFAULTp->UNIT.BUFFERED_BLOCKS;
1201  SS80p->UNIT.BURST_SIZE = SS80DEFAULTp->UNIT.BURST_SIZE;
1202  SS80p->UNIT.BLOCK_TIME = SS80DEFAULTp->UNIT.BLOCK_TIME;
1205  SS80p->UNIT.ACCESS_TIME = SS80DEFAULTp->UNIT.ACCESS_TIME;
1207  SS80p->UNIT.FIXED_VOLUMES = SS80DEFAULTp->UNIT.FIXED_VOLUMES;
1208 
1209  SS80p->VOLUME.MAX_CYLINDER = SS80DEFAULTp->VOLUME.MAX_CYLINDER;
1210  SS80p->VOLUME.MAX_HEAD = SS80DEFAULTp->VOLUME.MAX_HEAD;
1211  SS80p->VOLUME.MAX_SECTOR = SS80DEFAULTp->VOLUME.MAX_SECTOR;
1213  SS80p->VOLUME.INTERLEAVE = SS80DEFAULTp->VOLUME.INTERLEAVE;
1214 };
1215 
1216 
1220 void free_device(int8_t index)
1221 {
1222 
1223  extern void printer_close(void);
1224 
1225  if(index < 0 || index >= MAX_DEVICES)
1226  return;
1227 
1228  if(Devices[index].TYPE == SS80_TYPE)
1229  {
1230  SS80DiskType *SS80p = (SS80DiskType *) Devices[index].dev;
1231  // Device Model Name
1233  // File
1235  }
1236 
1237 #ifdef AMIGO
1238  if(Devices[index].TYPE == AMIGO_TYPE)
1239  {
1240  AMIGODiskType *AMIGOp = (AMIGODiskType *) Devices[index].dev;
1241  // Device Model Name
1242  safefree(AMIGOp->HEADER.NAME);
1243  // File
1244  safefree(AMIGOp->HEADER.model);
1245  }
1246 #endif
1247 
1248  // FIXME use printer structure to permit multiple printers
1249  if(Devices[index].TYPE == PRINTER_TYPE)
1250  {
1251  printer_close();
1252  }
1253 
1254  safefree(Devices[index].dev);
1255  safefree(Devices[index].state);
1256 
1257  Devices[index].TYPE = NO_TYPE;
1258  Devices[index].ADDRESS = 0;
1259  Devices[index].PPR = 0xff;
1260  Devices[index].BLOCKS = 0;
1261  Devices[index].dev = NULL;
1262  Devices[index].state = NULL;
1263 
1264 }
1265 
1269 int8_t alloc_device(int type)
1270 {
1271  int8_t ind;
1272  int8_t index = -1;
1273 
1274 // Find a free slot
1275  ind = find_free();
1276  if(ind == -1)
1277  {
1278  if(debuglevel & GPIB_ERR)
1279  printf("alloc_device: Device table is full\n", type);
1280  return(ind);
1281  }
1282 
1283  switch(type)
1284  {
1285 // Same as SS80 type but sets initial defaults for any remaining SS80 drives
1286  case SS80_DEFAULT_TYPE:
1287  Devices[ind].TYPE = type;
1288  Devices[ind].dev = safecalloc(sizeof(SS80DiskType)+7,1);
1289  Devices[ind].state = safecalloc(sizeof(SS80StateType)+7,1);
1290  index = ind;
1291  break;
1292  case SS80_TYPE:
1293  Devices[ind].TYPE = type;
1294  Devices[ind].dev = safecalloc(sizeof(SS80DiskType)+7,1);
1295  Devices[ind].state = safecalloc(sizeof(SS80StateType)+7,1);
1296  index = ind;
1297  SS80_Set_Defaults(index); // Set any defaults we may have
1298  break;
1299 #ifdef AMIGO
1300  case AMIGO_TYPE:
1301  Devices[ind].TYPE = type;
1302  Devices[ind].dev = safecalloc(sizeof(AMIGODiskType)+7,1);
1303  Devices[ind].state = safecalloc(sizeof(AMIGOStateType)+7,1);
1304  index = ind;
1305  break;
1306 #endif
1307  case PRINTER_TYPE:
1308  Devices[ind].TYPE = type;
1309  Devices[ind].dev = safecalloc(sizeof(PRINTERDeviceType)+7,1);
1310  Devices[ind].state = NULL;
1311  index = ind;
1312  break;
1313  default:
1314  if(debuglevel & GPIB_ERR)
1315  printf("alloc_device: invalid type:%d:%s\n", type,type_to_str(type));
1316  break;
1317  }
1318  return(index);
1319 }
1320 
1321 
1322 // =============================================
1326 {
1327  int i;
1328  stack_ind = 0;
1329  for(i=0;i<MAX_DEVICES;++i)
1330  {
1331  Devices[i].TYPE = NO_TYPE;
1332  Devices[i].ADDRESS = 0;
1333  Devices[i].PPR = 0xff;
1334  Devices[i].BLOCKS = 0;
1335  Devices[i].dev = NULL;
1336  Devices[i].state = NULL;
1337  }
1338 }
1339 
1340 
1345 int push_state(int state)
1346 {
1347  if(stack_ind < MAX_STACK)
1348  stack_p[stack_ind++] = state;
1349  else
1350  return(START_STATE);
1351  return(state);
1352 }
1353 
1354 
1359 {
1360  if(stack_ind > 0)
1361  return(stack_p[--stack_ind]);
1362  else
1363  return(START_STATE);
1364 }
1365 
1366 
1368 int lines = 0;
1369 
1380 bool assign_value(char *str, uint32_t minval, uint32_t maxval, uint32_t *val)
1381 {
1382  uint32_t tmp;
1383  int bad = 0;
1384  char *ptr;
1385 
1386 // Skip spaces before assignment
1387  ptr = skipspaces(str);
1388 // Skip optional '='
1389  if(*ptr == '=')
1390  {
1391  ++ptr;
1392 // skip spaces after assignment
1393  ptr = skipspaces(ptr);
1394  }
1395  if(!*ptr)
1396  {
1397  if(debuglevel & GPIB_ERR)
1398  printf("line:%d, missing value\n", lines);
1399  bad = 1;
1400  }
1401  if(!bad)
1402  {
1403 // FIXME detect bad numbers
1404  tmp = get_value(ptr);
1405  *val = tmp;
1406  if((minval && (tmp < minval)))
1407  {
1408  printf("line:%d, %s is below range %d\n", lines, ptr,(int)minval);
1409  bad = 1;
1410  }
1411  if((maxval != 0xffffffffUL) && (tmp > maxval))
1412  {
1413  printf("line:%d, %s is above range %d\n", lines, ptr,(int)maxval);
1414  bad = 1;
1415  }
1416  }
1417  if(bad)
1418  return(0);
1419  return(1);
1420 }
1421 
1422 
1428 {
1429 #if defined(SET_DEFAULTS)
1430  int8_t index;
1431 
1433  if(find_type(SS80_TYPE) == -1)
1434  {
1435 #if defined(HP9122D)
1436  printf("set_Config_Defaults: Using default SS/80 9122D\n");
1437 #endif
1438 #if defined(HP9134D)
1439  printf("set_Config_Defaults: Using default SS/80 9134L\n");
1440 #endif
1441  index = find_free();
1442  if(index != -1)
1443  {
1444  Devices[index].TYPE = SS80_TYPE;
1445  Devices[index].ADDRESS = SS80DiskDefault.HEADER.ADDRESS;
1446  Devices[index].PPR = SS80DiskDefault.HEADER.PPR;
1447  Devices[index].dev = (void *)&SS80DiskDefault;
1448  Devices[index].state = safecalloc(sizeof(SS80StateType)+7,1);
1449  }
1450  }
1451 #ifdef AMIGO
1452 // Make sure we have a AMIGO defined
1453  if(find_type(AMIGO_TYPE) == -1)
1454  {
1455 #if defined(HP9121)
1456  printf("set_Config_Defaults: Using default Amigo 9121\n");
1457 #endif
1458  index = find_free();
1459  if(index != -1)
1460  {
1461  Devices[index].TYPE = AMIGO_TYPE;
1462  Devices[index].ADDRESS = AMIGODiskDefault.HEADER.ADDRESS;
1463  Devices[index].PPR = AMIGODiskDefault.HEADER.PPR;
1464  Devices[index].dev = (void *) &AMIGODiskDefault;
1465  Devices[index].state = safecalloc(sizeof(AMIGOStateType)+7,1);
1466  }
1467  }
1468 #endif //#ifdef AMIGO
1469 
1470 // Make sure we have a PRINTER defined
1471  if(find_type(PRINTER_TYPE) == -1)
1472  {
1473  printf("set_Config_Defaults: Using default PRINTER settings\n");
1474  index = find_free();
1475  if(index != -1)
1476  {
1477  Devices[index].TYPE = PRINTER_TYPE;
1478  Devices[index].ADDRESS = PRINTERDeviceDefault.HEADER.ADDRESS;
1479  Devices[index].PPR = 0xff;
1480  Devices[index].dev = (void *) &PRINTERDeviceDefault;
1481  Devices[index].state = NULL;
1482  }
1483  }
1484 #endif // SET_DEFAULTS
1485 
1486 }
1487 
1488 
1495 int8_t hpdir_set_device(int8_t index)
1496 {
1497  if(Devices[index].TYPE == SS80_TYPE)
1498  {
1499  SS80DiskType *SS80p = (SS80DiskType *) Devices[index].dev;
1500  SS80p->CONFIG.ID = hpdir.ID;
1503 
1504 // CHS NOT used in this emulator!
1505  SS80p->VOLUME.MAX_CYLINDER = 0; // hpdir.CYLINDERS-1;
1506  SS80p->VOLUME.MAX_HEAD = 0; // hpdir.HEADS-1;
1507  SS80p->VOLUME.MAX_SECTOR = 0; // hpdir.SECTORS-1;
1508 
1510  Devices[index].BLOCKS = hpdir.BLOCKS;
1512  return(1);
1513  }
1514 
1515 #ifdef AMIGO
1516  else if(Devices[index].TYPE == AMIGO_TYPE)
1517  {
1518  AMIGODiskType *AMIGOp = (AMIGODiskType *) Devices[index].dev;
1519  AMIGOp->CONFIG.ID = hpdir.ID;
1522  AMIGOp->GEOMETRY.HEADS = hpdir.HEADS;
1523  AMIGOp->GEOMETRY.CYLINDERS = hpdir.CYLINDERS;
1524  Devices[index].BLOCKS = hpdir.BLOCKS;
1525  AMIGOp->HEADER.model = stralloc(hpdir.model);
1526  return(1);
1527  }
1528 #endif
1529  printf("hpdir invalid type - NOT AMIGO of SS80\n");
1530  return(0);
1531 }
1532 
1533 
1541 int8_t hpdir_set_parameters(int8_t index, char *model)
1542 {
1543  if ( hpdir_find_drive( model, 0 ,1) )
1544  return(hpdir_set_device(index) );
1545  printf("WARNING: model NOT found in hpdir.ini!\n");
1546  return(0);
1547 }
1548 
1552 int8_t verify_device(int8_t index)
1553 {
1554  long sectors;
1555  int8_t type;
1556  int address,ppr;
1557  int8_t ret = 1;
1558 
1559  uint8_t ppr_bits = 0;
1560  uint8_t ppr_mask;
1561  uint32_t addr_bits = 0;
1562  uint32_t addr_mask;
1563 
1565  SS80DiskType *SS80p = NULL;
1566 
1567 #ifdef AMIGO
1568  AMIGODiskType *AMIGOp = NULL;
1570 #endif
1571 
1572  if(Devices[index].TYPE == NO_TYPE)
1573  return(ret);
1574 
1575  address = Devices[index].ADDRESS;
1576  type = Devices[index].TYPE;
1577  ppr = Devices[index].PPR;
1578 
1579  if(address < 0 || address > 31)
1580  {
1581  printf("Address (%d) out of range\n", (int) address);
1582  ret = 0;
1583  }
1584 
1585  addr_mask = (1L << address);
1586  if(addr_bits & addr_bits)
1587  {
1588  printf("Address (%d) duplicated\n", (int) address);
1589  ret = 0;
1590  }
1591  addr_bits |= addr_mask;
1592 
1593 
1594  // Printers do not use PPR
1595  if(type == PRINTER_TYPE)
1596  {
1597  Devices[index].PPR = 0xff;
1598  return(1);
1599  }
1600  if(type == SS80_TYPE || AMIGO_TYPE)
1601  {
1602  if(ppr < 0 || ppr > 7)
1603  {
1604  ret = 0;
1605  }
1606  ppr_mask = (1 << ppr);
1607  if(ppr_bits & ppr_bits)
1608  {
1609  printf("PPR (%d) duplicated\n", (int) ppr);
1610  ret = 0;
1611  }
1612  ppr_bits |= ppr_mask;
1613  }
1614  if(type == SS80_TYPE)
1615  {
1616  SS80p= (SS80DiskType *)Devices[index].dev;
1617  if( SS80p->UNIT.BYTES_PER_BLOCK != 256)
1618  {
1619 // SS80p->UNIT.BYTES_PER_BLOCK = 256;
1620  printf("Warning: %s BYTES_PER_BLOCK != 256, Adjusting to 256\n", SS80p->HEADER.model);
1621  ret = 0;
1622  }
1623  sectors = SS80p->VOLUME.MAX_BLOCK_NUMBER+1;
1624  Devices[index].BLOCKS = sectors;
1625  } // SS80_TYPE
1626 
1627 #ifdef AMIGO
1628  if(type == AMIGO_TYPE )
1629  {
1630  AMIGOp = (AMIGODiskType *)Devices[index].dev;
1631  if( AMIGOp->GEOMETRY.BYTES_PER_SECTOR != 256)
1632  {
1633  AMIGOp->GEOMETRY.BYTES_PER_SECTOR = 256;
1634  printf("Warning: %s BYTES_PER_SECTOR != 256, Adjusting to 256\n", SS80p->HEADER.model);
1635  ret = 0;
1636  }
1637  sectors = AMIGOp->GEOMETRY.SECTORS_PER_TRACK
1638  * AMIGOp->GEOMETRY.HEADS
1639  * AMIGOp->GEOMETRY.CYLINDERS;
1640  Devices[index].BLOCKS = sectors;
1641  }
1642 #endif // #ifdef AMIGO
1643  if(!ret)
1644  {
1645  printf("Device errors - removing: ");
1646  display_mount(index);
1647  free_device(index);
1648  }
1649  return(ret);
1650 }
1651 
1656 {
1657  int8_t i;
1658 
1659  for(i=0;i<MAX_DEVICES;++i)
1660  verify_device(i);
1661 
1662 }
1663 
1664 
1665 
1670 {
1671  int i;
1672  struct stat st;
1673  long sectors;
1674  char label[32];
1675  int count =0;
1676  int ss80 = 0;
1677  int amigo = 0;
1678 
1680  SS80DiskType *SS80p = NULL;
1681 
1682 #ifdef AMIGO
1683  AMIGODiskType *AMIGOp = NULL;
1685 #endif
1686 
1687  for(i=0;i<MAX_DEVICES;++i)
1688  {
1689  if(Devices[i].TYPE == NO_TYPE)
1690  continue;
1691 
1692  if(Devices[i].TYPE == SS80_TYPE)
1693  {
1694  SS80p= (SS80DiskType *)Devices[i].dev;
1695 
1696  if(stat(SS80p->HEADER.NAME, &st) == -1)
1697  {
1698  if( SS80p->UNIT.BYTES_PER_BLOCK != 256)
1699  {
1700  printf("Can not use non 256 byte sectors\n");
1701  continue;
1702  }
1703 //SS80p->VOLUME.MAX_CYLINDER;
1704 //SS80p->VOLUME.MAX_HEAD;
1705 //SS80p->VOLUME.MAX_SECTOR;
1706  sectors = Devices[i].BLOCKS;
1707  printf("formating %s %ld sectors\n", SS80p->HEADER.NAME, (long) sectors);
1708  sprintf(label,"SS80-%d", ss80);
1709 #ifdef LIF_SUPPORT
1711  label,
1712  lif_dir_count(sectors),
1713  sectors);
1714 #else
1715  printf("please create a SS80 LIF image with %ld sectors and 128 directory sectors\n", sectors);
1716 #endif
1717  ++count;
1718  ++ss80;
1719 
1720  }
1721  } // SS80_TYPE
1722 
1723 #ifdef AMIGO
1724  if(Devices[i].TYPE == AMIGO_TYPE )
1725  {
1726  AMIGOp= (AMIGODiskType *)Devices[i].dev;
1727  if(stat(AMIGOp->HEADER.NAME, &st) == -1)
1728  {
1729  if( AMIGOp->GEOMETRY.BYTES_PER_SECTOR != 256)
1730  {
1731  printf("Can not use non 256 byte sectors\n");
1732  continue;
1733  }
1734  sectors = Devices[i].BLOCKS;
1735  printf("formating %s %ld sectors\n", AMIGOp->HEADER.NAME, (long) sectors);
1736  sprintf(label,"AMIGO%d", amigo);
1737 #ifdef LIF_SUPPORT
1738  lif_create_image(AMIGOp->HEADER.NAME,
1739  label,
1740  lif_dir_count(sectors),
1741  sectors);
1742 #else
1743  printf("please create a AMIGO LIF image with %ld sectors and 15 directory sectors\n", sectors);
1744 #endif
1745  ++count;
1746  ++amigo;
1747  }
1748  }
1749 #endif // #ifdef AMIGO
1750  }
1751  if(count)
1752  sep();
1753 }
1754 
1755 
1760 void drives_help(int8_t full)
1761 {
1762  if(!full)
1763  {
1764  printf("drives help\n");
1765  return;
1766  }
1767 
1768  printf(\
1769  "mount\n"
1770  " displays a list of mounted drives one per line\n"
1771  "mount AMIGO|SS80 model address file\n"
1772  " Example: mount 9121 6 /amigo6.lif\n"
1773  " Example: mount 9134D 2 /amigo2.lif\n"
1774  " Note: drive model must exist in hpdir.ini [driveinfo] section\n"
1775  "mount PRINTER address\n"
1776  " Example: mount PRINTER 5\n"
1777  "umount address\n"
1778  " Example: umount 6\n"
1779  "\n"
1780  "addresses\n"
1781  " Display all device GPIB bus addresses and PPR values\n"
1782  "config [-v]\n"
1783  " Display current drives configuration\n"
1784  " -v Verbose - show full detail\n"
1785  "\n"
1786  );
1787 }
1788 
1790 {
1791  printf("Usage: \n");
1792  drives_help(1);
1793 }
1794 
1798 int drives_tests(int argc, char * argv[])
1799 {
1800  char *ptr;
1801  int ind;
1802 
1803  ind = 0;
1804  ptr = argv[ind++];
1805 
1806  if(!ptr)
1807  return(0);
1808 
1809  if (MATCHI(ptr,"drives") && MATCHI(argv[ind], "help"))
1810  {
1811  drives_help(1);
1812  return(1);
1813  }
1814 
1815  if (MATCHI(ptr,"mount") )
1816  {
1817  if(!mount(argc, argv))
1818  return(-1);
1819 #ifdef LCD_SUPPORT
1820  extern void update_drive_counts();
1821  update_drive_counts();
1822 #endif
1823 
1824  return(1);
1825  }
1826  if (MATCHI(ptr,"umount") )
1827  {
1828  if(!umount(argc, argv))
1829  return(-1);
1830 #ifdef LCD_SUPPORT
1831  extern void update_drive_counts();
1832  update_drive_counts();
1833 #endif
1834  return(1);
1835  }
1836 
1837  if (MATCHI(ptr,"addresses") )
1838  {
1839  ptr = argv[ind];
1840  if(ptr && *ptr && MATCH(ptr,"-v"))
1841  display_Addresses(1);
1842  else
1843  display_Addresses(0);
1844  return(1);
1845  }
1846 
1847  if (MATCHI(ptr,"config") )
1848  {
1849  ptr = argv[ind];
1850  if(ptr && *ptr && MATCH(ptr,"-v"))
1851  display_Config(1);
1852  else
1853  display_Config(0);
1854  return(1);
1855  }
1856  return(0);
1857 }
1858 
1859 
1862 int8_t index_address(int8_t address)
1863 {
1864  int8_t i;
1865  for(i=0;i<MAX_DEVICES;++i)
1866  {
1867  if(Devices[i].TYPE != NO_TYPE && Devices[i].ADDRESS == address)
1868  return(i);
1869  }
1870  return(-1);
1871 }
1872 
1875 int8_t test_address(int8_t address)
1876 {
1877  if(address < 0 || address > 31)
1878  {
1879  printf("WARNING Address (%d) out of range\n", (int)address);
1880  return(0);
1881  }
1882 
1883  if(index_address(address) == -1)
1884  return(1);
1885  printf("WARNING Address (%d) already in use\n", (int)address);
1886  return(0);
1887 }
1888 
1889 
1892 int8_t index_ppr(int8_t ppr)
1893 {
1894  int8_t i;
1895  for(i=0;i<MAX_DEVICES;++i)
1896  {
1897  if(Devices[i].TYPE != NO_TYPE && Devices[i].PPR == ppr)
1898  return(i);
1899  }
1900  return(-1);
1901 }
1902 
1905 int8_t test_ppr(int8_t ppr)
1906 {
1907  if(ppr < 0 || ppr > 7)
1908  {
1909  printf("WARNING PPR (%d) out of range\n", (int)ppr);
1910  return(0);
1911  }
1912 
1913  if(index_ppr(ppr) == -1)
1914  return(1);
1915  printf("WARNING PPR (%d) already in use\n", (int)ppr);
1916  return(0);
1917 }
1918 
1922 int8_t umount(int argc, char *argv[])
1923 {
1924 
1925  int8_t address;
1926  int8_t index;
1927 
1928  if(argc != 2)
1929  {
1930  printf("Usage:\n");
1931  printf(" umount address\n");
1932  printf(" - address is the device address\n");
1933  }
1934  address = atoi(argv[1]);
1935  index = index_address(address);
1936  if(index == -1)
1937  {
1938  printf("umount address:[%d] NOT found\n", address);
1939  return(-1);
1940  }
1941  free_device(index);
1942  return(index);
1943 }
1944 
1948 int8_t mount(int argc, char *argv[])
1949 {
1950  int8_t index = -1;
1951 
1955  SS80DiskType *SS80p = NULL;
1956 #ifdef AMIGO
1957  AMIGODiskType *AMIGOp = NULL;
1959 #endif
1960 #if 0
1961  int8_t i;
1962  for(i =1;i< argc;++i)
1963  {
1964  printf("argv[%d] = %s\n", (int) i, argv[i]);
1965 
1966  }
1967 #endif
1968  if(argc == 1)
1969  {
1970  display_mounts();
1971  return(1);
1972  }
1973  else if(argc == 3)
1974  {
1975  if(MATCHI(argv[1], "PRINTER"))
1976  {
1977  // FIXME - do we want to have separtate address and ppr ?
1978  int8_t address = atoi(argv[2]) & 0xff;
1979  index = alloc_device(PRINTER_TYPE);
1980  if(index < 0)
1981  {
1982  printf("Could not allocate PRINTER structure\n");
1983  return(0);
1984  }
1985  PRINTERp = (PRINTERDeviceType *) Devices[index].dev;
1986  PRINTERp->HEADER.ADDRESS = address;
1987  Devices[index].ADDRESS = address;
1988  Devices[index].PPR = 0xff;
1989  return( verify_device(index) );
1990  }
1991  }
1992  else if(argc == 4)
1993  {
1994  /*
1995  argv[1] = 9121
1996  argv[2] = 2
1997  argv[3] = amigo2.lif
1998  */
1999  if(!hpdir_find_drive(argv[1],0,0) )
2000  {
2001  printf("WARNING: model NOT found in hpdir.ini!\n");
2002  return(-1);
2003  }
2004  if(MATCHI(hpdir.TYPE, "SS80") || MATCHI(hpdir.TYPE,"CS80") )
2005  {
2006  // FIXME - do we want to have separtate address and ppr ?
2007  int8_t address = atoi(argv[2]) & 0xff;
2008  int8_t ppr = address;
2009  index = alloc_device(SS80_TYPE);
2010  if(index < 0)
2011  {
2012  printf("Could not allocate SS80 structure for %s\n",argv[2]);
2013  return(0);
2014  }
2015  SS80p = (SS80DiskType *) Devices[index].dev;
2016  if( !hpdir_set_parameters(index, argv[1] ) )
2017  return(-1);
2018  SS80p->HEADER.NAME = stralloc(argv[3]);
2019  SS80p->HEADER.ADDRESS = address;
2020  SS80p->HEADER.PPR = ppr;
2021  Devices[index].ADDRESS = address;
2022  Devices[index].PPR = ppr;
2023  return( verify_device(index) );
2024  }
2025 #ifdef AMIGO
2026  else if(MATCH(hpdir.TYPE, "AMIGO"))
2027  {
2028  // FIXME - do we want to have separtate address and ppr ?
2029  int8_t address = atoi(argv[2]) & 0xff;
2030  int8_t ppr = address;
2031  index = alloc_device(AMIGO_TYPE);
2032  if(index < 0)
2033  {
2034  printf("Could not allocate AMIGO structure for %s\n",argv[2]);
2035  return(0);
2036  }
2037  AMIGOp = (AMIGODiskType *) Devices[index].dev;
2038  if( !hpdir_set_parameters(index, argv[1] ) )
2039  return(-1);
2040  AMIGOp->HEADER.NAME = stralloc(argv[3]);
2041  AMIGOp->HEADER.ADDRESS = address;
2042  AMIGOp->HEADER.PPR = ppr;
2043  Devices[index].ADDRESS = address;
2044  Devices[index].PPR = ppr;
2045  return( verify_device(index) );
2046  }
2047 #endif
2048  else
2049  {
2050  printf("Expected AMIGO or SS80 or CS80 for [%s]\n",argv[1]);
2051  mount_usage();
2052  return(0);
2053  }
2054  }
2055  else
2056  {
2057  mount_usage();
2058  return(0);
2059  }
2060  return(1);
2061 }
2062 
2063 
2064 void display_mount(int8_t index )
2065 {
2066 
2070  SS80DiskType *SS80p = NULL;
2071 
2072 #ifdef AMIGO
2073  AMIGODiskType *AMIGOp = NULL;
2075 #endif
2076 
2077  if(Devices[index].TYPE == NO_TYPE)
2078  return;
2079 
2080  if(Devices[index].TYPE == SS80_TYPE)
2081  {
2082  SS80p= (SS80DiskType *)Devices[index].dev;
2083 
2084  printf("SS80 %-8s %2d %s\n", SS80p->HEADER.model, (int) SS80p->HEADER.ADDRESS, SS80p->HEADER.NAME);
2085  }
2086 
2087 #ifdef AMIGO
2088  if(Devices[index].TYPE == AMIGO_TYPE )
2089  {
2090  AMIGOp = (AMIGODiskType *)Devices[index].dev;
2091  printf("AMIGO %-8s %2d %s\n", AMIGOp->HEADER.model, (int) AMIGOp->HEADER.ADDRESS, AMIGOp->HEADER.NAME);
2092  }
2093 #endif // #ifdef AMIGO
2094 
2095  if(Devices[index].TYPE == PRINTER_TYPE )
2096  {
2097  PRINTERp= (PRINTERDeviceType *)Devices[index].dev;
2098  printf("PRINTER %-8s %2d\n", " ", (int) PRINTERp->HEADER.ADDRESS);
2099  }
2100 }
2101 
2103 {
2104  int8_t i;
2105  printf("Mounted drives\n");
2106  for(i=0;i<MAX_DEVICES;++i)
2107  display_mount(i);
2108  printf("\n");
2109 }
SS80VolumeType::MAX_CYLINDER
uint32_t MAX_CYLINDER
Definition: drives.h:186
get_value
MEMSPACE int32_t get_value(char *str)
get a number
Definition: parsing.c:432
TOK_INTERLEAVE
@ TOK_INTERLEAVE
Definition: drives.h:320
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
AMIGO_CONFIG
@ AMIGO_CONFIG
Definition: drives.h:238
BASE_MTA
#define BASE_MTA
Definition: defines.h:76
TOK_SECTORS_PER_TRACK
@ TOK_SECTORS_PER_TRACK
Definition: drives.h:330
stat
POSIX stat structure.
Definition: posix.h:105
TOK_REMOVABLE_VOLUMES
@ TOK_REMOVABLE_VOLUMES
Definition: drives.h:329
sprintf
#define sprintf(s, format, args...)
Definition: user_config.h:73
find_free
int8_t find_free()
Find first free Devices[] slot.
Definition: drives.c:1052
TOK_ACCESS_TIME
@ TOK_ACCESS_TIME
Definition: drives.h:296
SS80s
SS80StateType * SS80s
Definition: drives.c:39
MAX_STACK
#define MAX_STACK
Config Parser Stack.
Definition: drives.c:28
PRINTERp
PRINTERDeviceType * PRINTERp
Active Printer Device.
Definition: drives.c:35
SS80_Set_Defaults
void SS80_Set_Defaults(int8_t index)
Set Default Values for a new SS80 Device IF defaults have been defined Most values in the CONTROLER a...
Definition: drives.c:1177
printf
MEMSPACE int printf(const char *format,...)
SS80ControllerType::UNITS_INSTALLED
uint16_t UNITS_INSTALLED
Definition: drives.h:131
token_t
Definition: drives.h:288
AMIGO_GEOMETRY
@ AMIGO_GEOMETRY
Definition: drives.h:239
mount
int8_t mount(int argc, char *argv[])
mount disks - testing
Definition: drives.c:1948
TOK_UNITS_INSTALLED
@ TOK_UNITS_INSTALLED
Definition: drives.h:336
debuglevel
int debuglevel
Debug flag - used to log GPIB and emulator messages.
Definition: gpib_task.c:33
HeaderType::PPR
uint8_t PPR
Definition: drives.h:43
get_token
MEMSPACE char * get_token(char *str, char *token, int max)
return next token
Definition: parsing.c:323
SS80VolumeType::MAX_HEAD
uint8_t MAX_HEAD
Definition: drives.h:187
MATCH
MEMSPACE int MATCH(char *str, char *pat)
Compare two strings.
Definition: parsing.c:143
BASE_MSA
#define BASE_MSA
Definition: defines.h:77
MATCHI
MEMSPACE int MATCHI(char *str, char *pat)
Compare two strings without case.
Definition: parsing.c:183
PRINTER_TYPE
@ PRINTER_TYPE
Definition: drives.h:252
TOK_MAX_SECTOR
@ TOK_MAX_SECTOR
Definition: drives.h:325
token_t::tok
int tok
Definition: drives.h:291
TOK_END
@ TOK_END
Definition: drives.h:313
AMIGODiskType::GEOMETRY
AMIGOGeometryType GEOMETRY
Definition: drives.h:108
TOK_VOLUME
@ TOK_VOLUME
Definition: drives.h:338
drives_tests
int drives_tests(int argc, char *argv[])
GPIB user tests User invoked GPIB functions and tasks.
Definition: drives.c:1798
drives_help
void drives_help(int8_t full)
Help Menu for drives and configuration help See: int drives_tests(char *str)
Definition: drives.c:1760
SS80DiskType::VOLUME
SS80VolumeType VOLUME
Definition: drives.h:200
DeviceType::BLOCKS
uint32_t BLOCKS
Definition: drives.h:261
AMIGODiskType::HEADER
HeaderType HEADER
Definition: drives.h:106
DeviceType::TYPE
uint8_t TYPE
Definition: drives.h:258
display_mount
void display_mount(int8_t index)
Definition: drives.c:2064
safecalloc
void * safecalloc(int size, int elements)
Safe Alloc - Display Error message if Calloc fails.
Definition: ram.c:122
print_str_P
void print_str_P(__memx const char *str, char *arg)
Display Configuration File string.
Definition: drives.c:241
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
ss80.h
SS80 disk emulator for HP85 disk emulator project for AVR.
tok_name
char * tok_name(uint8_t tok)
return string of matching token
Definition: drives.c:272
putchar
int putchar(int c)
put a character to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:363
type_to_str
char * type_to_str(int type)
Convert a disk type into a string.
Definition: drives.c:1021
SS80StateType
SS80 Emulated disk state information.
Definition: drives.h:205
SS80ControllerType::TYPE
uint8_t TYPE
Definition: drives.h:133
amigo.h
AMIGO disk emulator for HP85 disk emulator project for AVR.
TOK_TRANSFER_RATE
@ TOK_TRANSFER_RATE
Definition: drives.h:333
SS80UnitType::MAXIMUM_INTERLEAVE
uint8_t MAXIMUM_INTERLEAVE
Definition: drives.h:166
MAX_DEVICES
#define MAX_DEVICES
Maximum number of emulated devices.
Definition: drives.h:37
AMIGODiskType
AMIGO Disk structure - ID bytes and layout.
Definition: drives.h:104
AMIGO_HEADER
@ AMIGO_HEADER
Definition: drives.h:237
stralloc
MEMSPACE char * stralloc(char *str)
Allocate space for string.
Definition: stringsup.c:406
val_t::l
uint32_t l
Definition: drives.h:285
TOK_OPTIMAL_RETRY_TIME
@ TOK_OPTIMAL_RETRY_TIME
Definition: drives.h:326
hpdir_t::TYPE
char TYPE[32]
Definition: drives_sup.h:37
hpdir_t::HEADS
long HEADS
Definition: drives_sup.h:44
pop_state
int pop_state()
Pop Parser State.
Definition: drives.c:1358
TOK_ID
@ TOK_ID
Definition: drives.h:319
safefree
void safefree(void *p)
Safe free - Only free a pointer if it is in malloc memory range.
Definition: ram.c:158
TOK_FIXED_VOLUMES
@ TOK_FIXED_VOLUMES
Definition: drives.h:315
test_ppr
int8_t test_ppr(int8_t ppr)
test if PPR is in use
Definition: drives.c:1905
Devices
DeviceType Devices[MAX_DEVICES]
Definition: drives.c:32
TOK_CS80
@ TOK_CS80
Definition: drives.h:308
hpdir_t::CYLINDERS
long CYLINDERS
Definition: drives_sup.h:43
TOK_AMIGO
@ TOK_AMIGO
Definition: drives.h:298
TOK_MAX_CYLINDER
@ TOK_MAX_CYLINDER
Definition: drives.h:322
AMIGOGeometryType::SECTORS_PER_TRACK
int16_t SECTORS_PER_TRACK
Definition: drives.h:98
printer_close
void printer_close()
Close current plot file and reset states.
Definition: printer.c:118
SS80DiskType::HEADER
HeaderType HEADER
Definition: drives.h:196
stat
MEMSPACE int stat(char *name, struct stat *buf)
POSIX stat - get file status of named file.
Definition: posix.c:1368
START_STATE
@ START_STATE
Definition: drives.h:226
AMIGODiskType::CONFIG
ConfigType CONFIG
Definition: drives.h:107
TOK_DEBUG
@ TOK_DEBUG
Definition: drives.h:310
hpdir_set_parameters
int8_t hpdir_set_parameters(int8_t index, char *model)
Lookup model in and set drive parameters if found.
Definition: drives.c:1541
defines.h
GPIB, AMIGO, SS80 and device defines.
assign_value
bool assign_value(char *str, uint32_t minval, uint32_t maxval, uint32_t *val)
assigned a value
Definition: drives.c:1380
SS80UnitType::BYTES_PER_BLOCK
uint16_t BYTES_PER_BLOCK
Definition: drives.h:159
TOK_GEOMETRY
@ TOK_GEOMETRY
Definition: drives.h:316
set_active_device
int8_t set_active_device(int8_t index)
Set the Active disk or device pointers Since we can be called multiple times per single GPIB state we...
Definition: drives.c:1093
hpdir_t::model
char model[MODEL_SIZE]
Definition: drives_sup.h:35
verify_devices
void verify_devices()
Post process and Verify all devices.
Definition: drives.c:1655
SS80_CONTROLLER
@ SS80_CONTROLLER
Definition: drives.h:230
AMIGOGeometryType::CYLINDERS
int16_t CYLINDERS
Definition: drives.h:100
print_tok_str
void print_tok_str(uint8_t tok, uint8_t spaces, char *str)
Display Configuration File variable.
Definition: drives.c:305
DeviceType::state
void * state
Definition: drives.h:263
SS80UnitType::OPTIMAL_RETRY_TIME
uint16_t OPTIMAL_RETRY_TIME
Definition: drives.h:164
hpdir_t::BYTES_PER_SECTOR
long BYTES_PER_SECTOR
Definition: drives_sup.h:46
SS80DiskType::CONTROLLER
SS80ControllerType CONTROLLER
Definition: drives.h:198
NULL
#define NULL
Definition: user_config.h:85
AMIGO_TYPE
@ AMIGO_TYPE
Definition: drives.h:249
TOK_MAX_BLOCK_NUMBER
@ TOK_MAX_BLOCK_NUMBER
Definition: drives.h:321
GPIB_ERR
#define GPIB_ERR
Definition: debug.h:4
__file
FILE type structure.
Definition: posix.h:158
HeaderType::NAME
char * NAME
Maximun lengh of device file name.
Definition: drives.h:45
TOK_BYTES_PER_BLOCK
@ TOK_BYTES_PER_BLOCK
Definition: drives.h:303
gpib_task.h
High level GPIB command handler for HP85 disk emulator project for AVR.
SS80UnitType::REMOVABLE_VOLUMES
uint8_t REMOVABLE_VOLUMES
Definition: drives.h:168
display_Config
void display_Config(int verbose)
Display current Configuration File values.
Definition: drives.c:849
time.h
Common Linux/POSIX time functions.
SS80_CONFIG
@ SS80_CONFIG
Definition: drives.h:229
SS80_VOLUME
@ SS80_VOLUME
Definition: drives.h:232
TOK_CYLINDERS
@ TOK_CYLINDERS
Definition: drives.h:309
AMIGOGeometryType::HEADS
int16_t HEADS
Definition: drives.h:99
perror
MEMSPACE void perror(const char *s)
POSIX perror() - convert POSIX errno to text with user message.
Definition: posix.c:1850
test_address
int8_t test_address(int8_t address)
test if address is in use
Definition: drives.c:1875
SS80VolumeType::MAX_BLOCK_NUMBER
uint32_t MAX_BLOCK_NUMBER
Definition: drives.h:189
hpdir_set_device
int8_t hpdir_set_device(int8_t index)
Set Device parameters from hpdir information.
Definition: drives.c:1495
TOK_BYTES_PER_SECTOR
@ TOK_BYTES_PER_SECTOR
Definition: drives.h:304
PRINTERDeviceType::HEADER
HeaderType HEADER
Definition: drives.h:79
TOK_FILE
@ TOK_FILE
Definition: drives.h:314
index_ppr
int8_t index_ppr(int8_t ppr)
return index matching ppr
Definition: drives.c:1892
SS80UnitType::BURST_SIZE
uint8_t BURST_SIZE
Definition: drives.h:161
mount_usage
void mount_usage()
Definition: drives.c:1789
TOK_DEVICE_NUMBER
@ TOK_DEVICE_NUMBER
Definition: drives.h:311
sep
MEMSPACE void sep()
print seperator
Definition: parsing.c:35
fclose
MEMSPACE int fclose(FILE *stream)
POSIX close a file stream.
Definition: posix.c:1261
TOK_DRIVE
@ TOK_DRIVE
Definition: drives.h:312
print_var_P
void print_var_P(__memx const char *str, uint32_t val)
Display Configuration File variable.
Definition: drives.c:224
SS80DiskType::UNIT
SS80UnitType UNIT
Definition: drives.h:199
atoi
MEMSPACE int atoi(const char *str)
Convert ASCII string to number in base 10.
Definition: mathio.c:284
DeviceType::ADDRESS
uint8_t ADDRESS
Definition: drives.h:259
alloc_device
int8_t alloc_device(int type)
Allocate a Device structure for a disk or printer.
Definition: drives.c:1269
SS80_HEADER
@ SS80_HEADER
Definition: drives.h:228
AMIGOGeometryType::BYTES_PER_SECTOR
int16_t BYTES_PER_SECTOR
Definition: drives.h:97
lines
int lines
Config file line number.
Definition: drives.c:1368
val_t::w
uint16_t w
Definition: drives.h:284
base_to_str
char * base_to_str(int base)
Convert base address into a string identifier.
Definition: drives.c:1038
display_mounts
void display_mounts()
Definition: drives.c:2102
print_tok_val
void print_tok_val(uint8_t tok, uint8_t spaces, uint32_t val)
Display Configuration File variable.
Definition: drives.c:290
SS80UnitType::BUFFERED_BLOCKS
uint8_t BUFFERED_BLOCKS
Definition: drives.h:160
TOK_PPR
@ TOK_PPR
Definition: drives.h:327
lifutils.h
LIF file utilities.
BASE_MLA
#define BASE_MLA
=========================================================
Definition: defines.h:75
TOK_MAXIMUM_INTERLEAVE
@ TOK_MAXIMUM_INTERLEAVE
Definition: drives.h:324
push_state
int push_state(int state)
Push Parser State.
Definition: drives.c:1345
DeviceType
Device Type.
Definition: drives.h:256
PRINTERDeviceType
Printer structure.
Definition: drives.h:77
umount
int8_t umount(int argc, char *argv[])
umount disks - testing
Definition: drives.c:1922
SS80UnitType::DEVICE_NUMBER
uint32_t DEVICE_NUMBER
Definition: drives.h:158
tokens
token_t tokens[]
Definition: drives.c:169
__memx
#define __memx
Definition: mathio.h:38
TOK_CONFIG
@ TOK_CONFIG
Definition: drives.h:305
TOK_CONTINUOUS_TRANSFER_RATE
@ TOK_CONTINUOUS_TRANSFER_RATE
Definition: drives.h:306
find_type
int8_t find_type(int type)
Seach Devices[] for ANY definitions of a disk type.
Definition: drives.c:991
SS80_STATE
@ SS80_STATE
Definition: drives.h:227
drives.h
SS80DiskType::CONFIG
ConfigType CONFIG
Definition: drives.h:197
DeviceType::PPR
uint8_t PPR
Definition: drives.h:260
SS80ControllerType::TRANSFER_RATE
uint16_t TRANSFER_RATE
Definition: drives.h:132
hpdir_t::ID
long ID
Definition: drives_sup.h:38
AMIGOStateType
AMIGO emulator state machine index.
Definition: drives.h:57
TOK_HEADS
@ TOK_HEADS
Definition: drives.h:318
stack_ind
static int stack_ind
Definition: drives.c:29
format_drives
void format_drives()
Format devices that have no image file.
Definition: drives.c:1669
verify_device
int8_t verify_device(int8_t index)
Verify a device and delete it is there are any errors.
Definition: drives.c:1552
TOK_ADDRESS
@ TOK_ADDRESS
Definition: drives.h:297
SS80UnitType::BLOCK_TIME
uint16_t BLOCK_TIME
Definition: drives.h:162
ConfigType::ID
uint16_t ID
Definition: drives.h:52
TOK_UNIT_TYPE
@ TOK_UNIT_TYPE
Definition: drives.h:337
index_address
int8_t index_address(int8_t address)
return index matching address
Definition: drives.c:1862
TOK_HEADER
@ TOK_HEADER
Definition: drives.h:317
SS80DiskType
Disk Information Structure.
Definition: drives.h:194
SS80VolumeType::INTERLEAVE
uint8_t INTERLEAVE
Definition: drives.h:190
val_t::b
uint8_t b
Definition: drives.h:283
SS80_TYPE
@ SS80_TYPE
Definition: drives.h:251
token
MEMSPACE int token(char *str, char *pat)
Search for token in a string matching user pattern.
Definition: parsing.c:392
display_Addresses
void display_Addresses(int verbose)
Display Configuration device address saummary.
Definition: drives.c:778
set_Config_Defaults
void set_Config_Defaults()
Set Defaults for any missing disk or printer devices These are only used if the Config file omits the...
Definition: drives.c:1427
TOK_CONTROLLER
@ TOK_CONTROLLER
Definition: drives.h:307
HeaderType::ADDRESS
uint8_t ADDRESS
Definition: drives.h:42
free_device
void free_device(int8_t index)
Free a Device structure for a disk or printer.
Definition: drives.c:1220
TOK_BLOCKS
@ TOK_BLOCKS
Definition: drives.h:299
Read_Config
int Read_Config(char *name)
Read and parse a config file using POSIX functions Set all drive parameters and debuglevel.
Definition: drives.c:336
gpib.h
GPIB emulator for HP85 disk emulator project for AVR.
find_device
int8_t find_device(int type, int address, int base)
Find a device with matching type AND address.
Definition: drives.c:1063
val_t
Definition: drives.h:281
SS80p
SS80DiskType * SS80p
Active SS80 Device.
Definition: drives.c:38
SS80UnitType::ACCESS_TIME
uint16_t ACCESS_TIME
Definition: drives.h:165
TOK_BUFFERED_BLOCKS
@ TOK_BUFFERED_BLOCKS
Definition: drives.h:301
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
TOK_BURST_SIZE
@ TOK_BURST_SIZE
Definition: drives.h:302
SS80VolumeType::MAX_SECTOR
uint16_t MAX_SECTOR
Definition: drives.h:188
SS80UnitType::FIXED_VOLUMES
uint8_t FIXED_VOLUMES
Definition: drives.h:167
EOF
#define EOF
Definition: ff.h:349
skipspaces
MEMSPACE char * skipspaces(char *ptr)
Skip white space in a string - tabs and spaces.
Definition: parsing.c:70
TOK_PRINTER
@ TOK_PRINTER
Definition: drives.h:328
hpdir_t::BLOCKS
long BLOCKS
Definition: drives_sup.h:50
tok_index
int8_t tok_index(char *str)
return the tokens index of the matching string
Definition: drives.c:256
TOK_TYPE
@ TOK_TYPE
Definition: drives.h:334
SS80_DEFAULT_TYPE
@ SS80_DEFAULT_TYPE
Definition: drives.h:250
TOK_UNIT
@ TOK_UNIT
Definition: drives.h:335
gpib_hal.h
GPIB emulator hardwware layer for HP85 disk emulator project for AVR.
count_drive_types
int8_t count_drive_types(uint8_t type)
Count number of devices of a sertain type.
Definition: drives.c:1005
print_tok
void print_tok(uint8_t tok, uint8_t spaces)
Display Configuration File variable.
Definition: drives.c:321
hpdir_t::SECTORS
long SECTORS
Definition: drives_sup.h:45
init_Devices
void init_Devices()
Init Config Parser Stack Called only durring power up so we do not have to free memory.
Definition: drives.c:1325
AMIGO_STATE
@ AMIGO_STATE
Definition: drives.h:236
hpdir_t
Definition: drives_sup.h:33
SS80UnitType::UNIT_TYPE
uint8_t UNIT_TYPE
Definition: drives.h:157
PRINTER_STATE
@ PRINTER_STATE
Definition: drives.h:240
PRINTER_CONFIG
@ PRINTER_CONFIG
Definition: drives.h:241
TOK_BLOCK_TIME
@ TOK_BLOCK_TIME
Definition: drives.h:300
TOK_INVALID
@ TOK_INVALID
Definition: drives.h:339
TOK_SS80_DEFAULT
@ TOK_SS80_DEFAULT
Definition: drives.h:332
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
debug.h
DeviceType::dev
void * dev
Definition: drives.h:262
stack_p
static int stack_p[MAX_STACK]
Definition: drives.c:30
hpdir_t::DEVICE_NUMBER
long DEVICE_NUMBER
Definition: drives_sup.h:41
hpdir
hpdir_t hpdir
hpdir.ini file processing
Definition: drives.c:48
lif_dir_count
long lif_dir_count(long blocks)
LIF Directory blocks ~= sqrt(blocks);.
Definition: drives_sup.c:57
TOK_SS80
@ TOK_SS80
Definition: drives.h:331
SS80_UNIT
@ SS80_UNIT
Definition: drives.h:231
HeaderType::model
char * model
Definition: drives.h:46
SS80UnitType::CONTINUOUS_TRANSFER_RATE
uint16_t CONTINUOUS_TRANSFER_RATE
Definition: drives.h:163
NO_TYPE
@ NO_TYPE
Definition: drives.h:248
TOK_MAX_HEAD
@ TOK_MAX_HEAD
Definition: drives.h:323