HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
ss80.c
Go to the documentation of this file.
1 
19 #include "user_config.h"
20 
21 #include "defines.h"
22 #include "gpib_hal.h"
23 #include "gpib.h"
24 #include "gpib_task.h"
25 #include "amigo.h"
26 #include "ss80.h"
27 #include "vector.h"
28 #include "debug.h"
29 
87 
89 extern uint8_t talking;
90 extern uint8_t listening;
91 extern uint8_t spoll;
92 
96 {
102 };
103 
106 int TD[] =
107 {
108  0x3F,
109  0x5f | ATN_FLAG,
110  0x20 | ATN_FLAG,
111  0x65 | ATN_FLAG,
112  0x20,
113  0x40,
114  0x34, // NOP
115  0x37, // Format
116  0x00,
117  0x0a,
118  -1
119 };
120 
123 void SS80_Test(void)
124 {
125 #if SDEBUG
127  printf("[SS80 Test]\n");
128 #endif
129 #if SDEBUG
131  {
132  printf("[SS80 Test Done]\n");
133  sep();
134  }
135 #endif
136 }
137 
138 
154 
159 void V2B_MSB_Index1(uint8_t *B, int index,int size, uint32_t val)
160 {
162  V2B_MSB(B, index-1,size, val);
163 }
164 
165 
169 uint8_t *SS80ControllerPack(int *size)
170 {
171  static uint8_t B[5];
172  *size = 5;
173 /*
174  uint8_t C1; //< MSB units installed bit field
175  //< one bit per unit, Unit 15 always set
176  uint8_t C2; //< LSB
177  uint8_t C3; //< MSB Instantaneous transfer rate in kb/s
178  uint8_t C4; //< LSB
179  uint8_t C5; //< Controller type
180  //< 0 = CS/80 integrated single unit controller.
181  //< 1 = CS/8O integrated multi-unit controller.
182  //< 2 = CS/8O integrated multi-port controller.
183  //< 4 = SS/8O integrated single unit controller.
184 //< 5 = SS/80 integrated multi-unit controller.
185 //< 6 = SS/80 integrated multi-port controller.
186 */
190 
191  return(B);
192 }
193 
194 
198 uint8_t *SS80UnitPack(int *size)
199 {
200  static uint8_t B[19];
201  *size = 19;
202 /*
203  uint8_t U1; //< Type 0-Fixed, 1-Flexible, 2-Tape
204  //< (+128-dumb, does not detect media change)
205  uint8_t U2; //< MSB Device number
206  uint8_t U3; //< (HP 9133 = 09 13 30)
207  uint8_t U4; //< LSB
208  uint8_t U5; //< MSB Bytes per block
209  uint8_t U6; //< LSB
210  uint8_t U7; //< Number of buffered blocks
211  uint8_t U8; //< Burst size (0 for SS/80)
212  uint8_t U9; //< MSB Block time in ms
213 uint8_t U10; //< LSB
214 uint8_t U11; //< MSB Continous average transfer rate
215 uint8_t U12; //< LSB
216 uint8_t U13; //< MSB Optimal retry in tens of ms
217 uint8_t U14; //< LSB
218 uint8_t U15; //< MSB Access time in tens of ms
219 uint8_t U16; //< LSB
220 uint8_t U17; //< Maximum interleave or 0
221 uint8_t U18; //< Fixed volume byte, one bit per volume,
222 //< ie 00000111 = 3 volumes
223 uint8_t U19; //< Removable volume byte, one bit per volume,
224 //< ie 00000111 = 3 volumes
225 */
238  return(B);
239 }
240 
241 
245 uint8_t *SS80VolumePack(int *size)
246 {
247  static uint8_t B[13];
248  *size = 13;
249 /*
250  uint8_t V1; //< MSB Max cylinder
251  uint8_t V2; //<
252  uint8_t V3; //< LSB
253  uint8_t V4; //< Maximum head 0 based
254  uint8_t V5; //< MSB Maximum sector 0 based
255  uint8_t V6; //< LSB
256  uint8_t V7; //< MSB Max number of blocks
257  uint8_t V8;
258  uint8_t V9;
259  uint8_t V10;
260 uint8_t V11;
261 uint8_t V12; //< LSB
262 uint8_t V13; //< Interleave
263 */
269  return(B);
270 }
271 
272 
275 void SS80_init(void)
276 {
277 
278  int i;
279  for(i=MAX_DEVICES-1;i>=0;--i)
280  {
281  if(Devices[i].TYPE == SS80_TYPE)
282  {
283  if(!set_active_device(i))
284  continue;
285  Clear_Common(15);
286 // Power On State
287  SS80s->qstat = 2;
288 #if SDEBUG
290  printf("[SS80 %02XH INIT]\n", Devices[i].ADDRESS);
291 #endif
294  }
295  }
296 }
297 
298 
307 {
308  int ret = 0;
309 
311  switch(SS80s->estate)
312  {
313  case EXEC_IDLE:
314  break;
316  ret = SS80_locate_and_read();
318  break;
320  ret = SS80_locate_and_write();
322  break;
323  case EXEC_SEND_STATUS:
324  ret = SS80_send_status();
326  break;
327  case EXEC_DESCRIBE:
328  ret = SS80_describe();
330  break;
331  default:
332  if(debuglevel & GPIB_ERR)
333  printf("[SS80 EXEC state:%d error]\n", SS80s->estate);
335  break;
336  }
338  return(ret);
339 }
340 
341 
344 uint32_t SS80_Blocks_to_Bytes(uint32_t block)
345 {
346  return(block * SS80p->UNIT.BYTES_PER_BLOCK);
347 
348 }
349 
350 
353 uint32_t SS80_Bytes_to_Blocks(uint32_t bytes)
354 {
355  return(bytes / SS80p->UNIT.BYTES_PER_BLOCK);
356 }
357 
358 
373 
375 {
376  DWORD total_bytes;
377  DWORD count;
378  int chunk;
379  int len;
380  uint16_t status;
381  uint32_t Address = SS80_Blocks_to_Bytes(SS80s->AddressBlocks);
382 
383  SS80s->qstat = 0;
384 
385  status = 0;
386 
387  if( GPIB_IO_RD(IFC) == 0)
388  return(IFC_FLAG);
389 
393 
394 #if SDEBUG
396  printf("[SS80 Locate and Read at %08lXH(%lXH)]\n",
397  (long) Address, (long) SS80s->Length);
398 #endif
399 
400  if( SS80_cmd_seek() )
401  {
402  return(SS80_error_return());
403  }
404 
405  count = SS80s->Length;
406  total_bytes = 0;
407  while(count > 0 )
408  {
409  if( GPIB_IO_RD(IFC) == 0)
410  {
411  return(IFC_FLAG);
412  }
413 
414  if(count > 256)
415  {
416  chunk = 256;
417  status = 0; // GPIB status
418  }
419  else
420  {
421  chunk = count;
422  status |= EOI_FLAG; // GPIB EOI on final charater
423  }
424 
425 #if SDEBUG
428 #endif
429 
430 // FIXME len != chunk
431  len = dbf_open_read(SS80p->HEADER.NAME, Address, gpib_iobuff, chunk, &SS80s->Errors);
432 
433 #if SDEBUG
435  gpib_timer_elapsed_end("disk READ ");
437  printf("[SS80 Disk Read %02XH bytes]\n", len);
438 #endif
439  if(len < 0)
440  {
441  SS80s->qstat = 1;
443  if(debuglevel & GPIB_ERR)
444  printf("[SS80 Disk Read Error]\n");
445  return( SS80_error_return() );
446  }
447 
448 #if SDEBUG
451 #endif
452  len = gpib_write_str(gpib_iobuff, chunk, &status);
453 #if SDEBUG
455  gpib_timer_elapsed_end("GPIB write");
456 #endif
457  if( len != chunk)
458  {
459  SS80s->qstat = 1;
460  if(debuglevel & GPIB_ERR)
461  printf("[SS80 GPIB Write Error]\n");
462  if(status & ERROR_MASK)
463  {
464  SS80s->Errors |= ERR_GPIB;
465  break;
466  }
467  }
468 
469  Address += chunk;
470  total_bytes = total_bytes + len;
471  count -= len;
472  }
474  if(count > 0)
475  {
476  if(debuglevel & GPIB_ERR)
477  printf("[SS80 Buffered Read DID NOT FINISH]\n");
478  }
479  else
480  {
481 #if SDEBUG
483  printf("[SS80 Buffered Read Total(%lXH) bytes]\n", (long) total_bytes);
484 #endif
485  }
486 
488  return (status & ERROR_MASK);
489 }
490 
491 
505 
507 {
508  DWORD total_bytes; // Keeps track of data actually written to disk
509  int chunk, count, len;
510  int io_skip;
511  uint16_t status;
512  uint32_t Address = SS80_Blocks_to_Bytes(SS80s->AddressBlocks);
513 
514  io_skip = 0;
515 
516 #if SDEBUG
518  printf("[SS80 Locate and Write at %08lXH(%lXH)]\n", (long)Address, (long)SS80s->Length);
519 #endif
520 
521  SS80s->qstat = 0;
522 
523  if( GPIB_IO_RD(IFC) == 0)
524  return(IFC_FLAG);
525 
526  if( SS80_cmd_seek() )
527  {
528  SS80s->Errors |= ERR_WRITE;
529  io_skip = 1;
530  }
531 
532  count = SS80s->Length;
533  total_bytes = 0;
534 
535  status = 0;
536 
537  while(count > 0) // Loop until we have received Length sectors or EOI
538  {
539  if( GPIB_IO_RD(IFC) == 0)
540  {
541  return(IFC_FLAG);
542  }
543 
544  if(count > 256)
545  chunk = 256;
546  else
547  chunk = count;
548 
550 
551 #if SDEBUG
554 #endif
555  len = gpib_read_str(gpib_iobuff, (UINT) chunk, &status);
556 
557 #if SDEBUG
559  gpib_timer_elapsed_end("GPIB read ");
560 #endif
561 
562  if( len != chunk)
563  {
564  if(status & ERROR_MASK)
565  {
566  if(debuglevel & GPIB_ERR)
567  printf("[GPIB Read Error]\n");
568  SS80s->Errors |= ERR_WRITE;
569  SS80s->qstat = 1;
570  break;
571  }
572  if(!len && (EOI_FLAG & status))
573  break;
574  }
575 
576  if(!io_skip)
577  {
578  if(len)
579  {
580  int len2;
581 #if SDEBUG
584 #endif
585  len2 = dbf_open_write(SS80p->HEADER.NAME, Address, gpib_iobuff, len, &SS80s->Errors);
586 #if SDEBUG
588  gpib_timer_elapsed_end("disk WRITE");
589 #endif
590  if(len2 != len)
591  {
592  SS80s->Errors |= ERR_WRITE;
593  if(mmc_wp_status())
594  SS80s->Errors |= ERR_WP;
595  SS80s->qstat = 1;
596  io_skip = 1; // Stop writing
597  if(debuglevel & GPIB_ERR)
598  printf("[Disk Write Error]\n");
599  }
600  else
601  {
602 #if SDEBUG
604  printf("[SS80 Locate and Write wrote(%02XH)]\n", len2);
605 #endif
606  Address += len;
607  }
608 
609  if(len2 >= len)
610  total_bytes += len2;
611  else
612  total_bytes += len;
613  } // len
614  } // io_skip
615 
616  count -= len;
617 
618  if(status & EOI_FLAG)
619  break;
620  }
621 
622  if(count > 0)
623  {
624  if(debuglevel & GPIB_ERR)
625  printf("[SS80 Locate and Write DID NOT FINISH]\n");
626  }
627  else
628  {
629 #if SDEBUG
631  printf("[SS80 Locate and Write Wrote Total(%lxH)]\n", total_bytes);
632 #endif
633  }
635  return ( status & ERROR_MASK );
636 }
637 
638 
643 {
644  { 2, "Channel Parity Error or the Loopback command failed." },
645  { 5, "Illegal Opcode or command not allowed on Unit 15" },
646  { 6, "illegal Volume or Unit number was specified for this device." },
647  { 7, "Address Bounds exceeded for this device." },
648  { 8, "Parameter Bounds error (other than unit, volume, or target address)"},
649  { 9, "Parameter field wrong length for the opcode preceding it." },
650  { 10,"Unexpected Secondary. Except for active reject,fault or access errors" },
651  { 12,"Message Length error" },
652  { 17,"Unit 15 Cross-unit error" },
653  { 19,"Controller hardware Fault" },
654  { 22,"Unit hardware Fault" },
655  { 24,"Diagnostic failed selftest" },
656  { 30,"Power Fail, or a new Tape or Flexible disc was loaded." },
657  { 31,"Re-try preceding transaction" },
658  { 33,"Uninitialized media" },
659  { 34,"No Spares Available" },
660  { 35,"Not Ready, medium has not loaded?" },
661  { 36,"Write attempt while write protected" },
662  { 37,"Tapes: read block accessed but not yet written. Disks: Validate key mismatch" },
663  { 40,"Unrecoverable Data" },
664  { 41,"Unrecoverable Data P1-P6 will contain the address of first bad block" },
665  { 43,"End of File End encountered on file structured device." },
666  { 44,"End of Volume attempted to access past end" },
667  { 51,"Flexible disc or tape medium is wearing out" },
668  { 52,"Latency induced bit because of retries" },
669  { 55,"Device has automatically spared a block." },
670  { 57,"Too many recoverable data error status messages queued since last request" },
671  { 59,"Recoverable Data latency was introduced because of retries" },
672  { -1,NULL }
673 };
674 
683 int SS80_test_extended_status(uint8_t *p, int bit)
684 {
685  uint8_t mask;
686  int byte = bit >> 3;
687  bit &= 7;
688  mask = (0x80 >> bit);
689  return((p[byte] & mask) ? 1 : 0);
690 }
691 
692 
700 void SS80_set_extended_status(uint8_t *p, int bit)
701 {
702  uint8_t mask;
703  int byte = bit >> 3;
704  bit &= 7;
705  mask = (0x80 >> bit);
706  p[byte] |= mask;
707 }
708 
709 
714 void SS80_display_extended_status(uint8_t *p, char *message)
715 {
716 
717  int i,bit;
718  int status = 0;
719 
720 // See if ANY bits are set
721  for(i=0;i<8;++i)
722  {
723  if(p[i])
724  status = 1;
725  }
726 // Display all of the messages
727  if(status)
728  {
729  printf("%s:\n",message);
730  for(i=0;faults[i].index != -1;++i)
731  {
732  bit = faults[i].index;
733  if(SS80_test_extended_status(p,bit))
734  {
735  printf(" %d:%s\n", faults[i].index, faults[i].msg);
736  }
737  }
738  }
739 }
740 
741 
853 
854 int SS80_send_status( void )
855 {
856  uint8_t tmp[20];
857  uint16_t status;
858 
859 #if SDEBUG
861  printf("[SS80 Send Status]\n");
862 #endif
863 
864  Mem_Clear(tmp);
865 
866  tmp[0] = ((SS80s->volNO& 0x0f) << 4) | (SS80s->unitNO & 0x0f);
867 
868  tmp[1] = 0xff;
869 
870 // Bit 6 Module addressing
871 // TODO unit support
872  if(SS80s->Errors & ERR_UNIT)
873  SS80_set_extended_status(tmp+2, 6);
874 
875 // Bit 7 Address Bounds
876  if(SS80s->Errors & ERR_SEEK)
877  SS80_set_extended_status(tmp+2, 7);
878 
879 // Bit 22 Unit fault
880  if(SS80s->Errors & ERR_READ)
881  SS80_set_extended_status(tmp+2, 22);
882 
883 // Bit 22 Unit fault
884  if(SS80s->Errors & ERR_WRITE)
885  SS80_set_extended_status(tmp+2, 22);
886 
888 
889 // Bit 36 Write Protect
890  if(SS80s->Errors & ERR_WP)
891  tmp[6] = 0b00001000;
892 
894 
897  if(!SS80s->Errors)
898  {
899 /* tmp[10] = SS80s->AddressBlocks.B[5] MSB unused */
900 /* tmp[11] = SS80s->AddressBlocks.B[4] unused */
901 /* index 0 offset */
902  V2B_MSB(tmp,10,6,SS80s->AddressBlocks);
903  }
904 
906  if(SS80s->Errors)
907  SS80s->qstat = 1;
908 
909  status = EOI_FLAG;
910  if(gpib_write_str(tmp, sizeof(tmp), &status) != sizeof(tmp))
911  {
912  if(debuglevel & GPIB_ERR)
913  printf("[SS80 Send Status FAILED]\n");
914  }
915 
916  return ( status & ERROR_MASK );
917 }
918 
919 
928 
929 int SS80_describe( void )
930 {
931  uint16_t status;
932 
933  uint8_t *B;
934  int size;
935 
936 #if SDEBUG
938  printf("[SS80 Describe]\n");
939 #endif
940 
941  status = 0;
942 
943  B = SS80ControllerPack(&size);
944  if(gpib_write_str(B,size, &status) != size)
945  {
946  if(debuglevel & GPIB_ERR)
947  printf("[SS80 Describe Controller FAILED]\n");
948  return(status & ERROR_MASK);
949  }
950 
951  status = 0;
952 
953  B = SS80UnitPack(&size);
954  if(gpib_write_str(B,size, &status) != size)
955  {
956  if(debuglevel & GPIB_ERR)
957  printf("[SS80 Describe Unit FAILED]\n");
958  return(status & ERROR_MASK);
959  }
960 
961  status = EOI_FLAG;
962  B = SS80VolumePack(&size);
963  if(gpib_write_str(B,size,&status) != size)
964  {
965  if(debuglevel & GPIB_ERR)
966  printf("[SS80 Describe Volume FAILED]\n");
967  return(status & ERROR_MASK);
968  }
969 
970  return(0);
971 }
972 
973 
977 void SS80_Check_Unit(uint8_t unit)
978 {
979  if(unit != 0 && unit != 15)
980  {
981  SS80s->Errors |= ERR_UNIT;
982  if(debuglevel & GPIB_ERR)
983  printf("[SS80 UNIT:%d invalid]\n", (int) unit);
984  }
985  else
986  {
987  SS80s->unitNO = unit;
988  }
989 }
990 
991 
995 void SS80_Check_Volume(uint8_t volume)
996 {
997  if(volume != 0)
998  {
999  SS80s->Errors |= ERR_UNIT;
1000  if(debuglevel & GPIB_ERR)
1001  printf("[SS80 Volume:%d invalid]\n", (int) volume);
1002  }
1003  else
1004  {
1005  SS80s->volNO = volume;
1006  }
1007 }
1008 
1009 
1046 
1048 {
1049  int ch; // Current OP Code
1050  uint16_t status; // Current status
1051  int len; // Size of Data/Op Codes/Parameters read in bytes
1052  int ind; // Buffer index
1053 
1055 
1056  status = EOI_FLAG;
1057  len = gpib_read_str(gpib_iobuff, GPIB_IOBUFF_LEN, &status);
1058  if(status & ERROR_MASK)
1059  {
1061  if(debuglevel & GPIB_ERR)
1062  printf("[SS80 Command State GPIB Read ERROR]\n");
1063  return(status & ERROR_MASK);
1064  }
1065 
1066  if(!len)
1067  return(0);
1068 
1069  if( !(status & EOI_FLAG) )
1070  {
1071  if(debuglevel & GPIB_ERR)
1072  printf("[GPIB buffer OVERFLOW!]\n");
1073  }
1074 
1075  ind = 0;
1076  while(ind < len)
1077  {
1078  ch = gpib_iobuff[ind++];
1079 
1082  if(ch >= 0x20 && ch <= 0x2f)
1083  {
1084  SS80_Check_Unit(ch - 0x20);
1085 // TODO unit support
1086 
1087 #if SDEBUG
1089  printf("[SS80 Set Unit:(%d)]\n", SS80s->unitNO);
1090 #endif
1091  continue;
1092  }
1093 
1096  if(ch >= 0x40 && ch <= 0x4f)
1097  {
1098  SS80_Check_Volume(ch - 0x40);
1099 #if SDEBUG
1101  printf("[SS80 Set Volume: (%d)]\n", SS80s->volNO);
1102 #endif
1103  continue;
1104  }
1105 
1106  if(ch == 0x00)
1107  {
1111 #if SDEBUG
1113  printf("[SS80 Locate and Read]\n");
1114 #endif
1115  break;
1116  }
1117 
1118  if(ch == 0x02)
1119  {
1123 #if SDEBUG
1125  printf("[SS80 Locate and Write]\n");
1126 #endif
1127  break;
1128  }
1129 
1140 
1144 // Set address
1145  if (ch == 0x10)
1146  {
1147 /* upper two MSB unused */
1148 /* gpib_iobuff[ind+0] MSB unused */
1149 /* gpib_iobuff[ind+1] MSB unused */
1151  ind += 6;
1152 #if SDEBUG
1154  printf("[SS80 Set Address:(%08lXH)]\n",
1156 #endif
1157  continue;
1158  }
1159 
1161 // Set Length
1162  if(ch == 0x18)
1163  {
1164 /* gpib_iobuff[ind+0] MSB */
1165  SS80s->Length = B2V_MSB(gpib_iobuff,ind,4);
1166  ind += 4;
1167 #if SDEBUG
1169  printf("[SS80 Set Length:(%08lXH)]\n", (long)SS80s->Length);
1170 #endif
1171  continue;
1172  }
1173 
1174 // SS80 $S80 4-47
1175 // NO OP
1176  if(ch == 0x34)
1177  {
1178 #if SDEBUG
1180  printf("[SS80 NO-OP]\n");
1181 #endif
1182  continue;
1183  }
1184 // SS80 $S80 4-79
1185 // Set RPS
1186  if (ch == 0x39)
1187  {
1188  ind += 2;
1189 #if SDEBUG
1191  printf("[SS80 Set RPS NO-OP]\n");
1192 #endif
1193  continue;
1194  }
1195 
1196 // SS80 $S80 4-75
1197 // Set Release
1198  if (ch == 0x3B)
1199  {
1200  ind++;
1201 #if SDEBUG
1203  printf("[SS80 Set Release NO-OP]\n");
1204 #endif
1205  continue;
1206  }
1207 
1213 
1214  if(ch == 0x48)
1215  {
1218  ind++;
1219 #if SDEBUG
1221  printf("[SS80 Set Return Addressing - TODO]\n");
1222 #endif
1223  continue;
1224  }
1225 
1229 
1230  if(ch == 0x04)
1231  {
1235 #if SDEBUG
1237  printf("[SS80 Locate and Verify - TODO]\n");
1238 #endif
1239  break;
1240  }
1241 
1242  if(ch == 0x0E)
1243  {
1244 #if SDEBUG
1246  printf("[SS80 Release NO-OP]\n");
1247 #endif
1248  break;
1251  }
1252  if(ch == 0x0F)
1253  {
1254 #if SDEBUG
1256  printf("[SS80 Release Denied NO-OP]\n");
1257 #endif
1258  break;
1261  }
1262 
1269 
1270  if(ch == 0x31)
1271  {
1274  ind += 2;
1275 #if SDEBUG
1277  printf("[SS80 Validate Key - TODO]\n");
1278 #endif
1279  break;
1280  }
1281 
1282  if(ch == 0x35)
1283  {
1285 #if SDEBUG
1287  printf("[SS80 Describe]\n");
1288 #endif
1289  break;
1290  }
1291 
1298 
1299  if(ch == 0x37)
1300  {
1301  ind += 2;
1304 #if SDEBUG
1306  printf("[SS80 Initialize Media TODO]\n");
1307 #endif
1308  break;
1309  }
1310 
1317 
1318  if(ch == 0x3E)
1319  {
1321 #if SDEBUG
1323  {
1324  printf("[SS80 Set Status Mask - TODO]\n");
1325  SS80_display_extended_status(gpib_iobuff+ind, "TODO Mask these Status Bits");
1326  }
1327 #endif
1328  ind += 8;
1329  break;
1330  }
1331 
1332  if (ch == 0x4C)
1333  {
1335 #if SDEBUG
1337  printf("[SS80 Door UnLock - TODO]\n");
1338 #endif
1339  break;
1340  }
1341 
1342  if (ch == 0x4D)
1343  {
1345 #if SDEBUG
1347  printf("[SS80 Door Lock - TODO]\n");
1348 #endif
1349  break;
1350  }
1351 
1352  if(ch == 0x0D)
1353  {
1355 #if SDEBUG
1357  printf("[SS80 Request Status]\n");
1358 #endif
1359  break;
1360  }
1361 
1368 
1369  if(ch == 0x33)
1370  {
1372  ind += 3;
1373 #if SDEBUG
1375  printf("[SS80 Initiate Diagnostic - TODO]\n");
1376 #endif
1377  break;
1378  }
1379 
1380  if(debuglevel & GPIB_ERR)
1381  printf("[SS80 Invalid OP Code (%02XH)]\n", ch & 0xff);
1382 
1383  break;
1384  }
1385 
1386  if( ind != len)
1387  {
1388  if(debuglevel & GPIB_ERR)
1389  printf("[SS80 Execute Command, Error at (%d) of (%d) OP Codes]\n",
1390  ind, len);
1391  }
1392 
1394 
1395  return(status & ERROR_MASK);
1396 }
1397 
1398 
1439 
1441 
1443 {
1445 
1446  int ch; // Current OP Code
1447  uint16_t status; // Current status
1448  int len; // Size of Data/Op Codes/Parameters read in bytes
1449  int ind; // Buffer index
1451 
1453 
1456 
1457  status = EOI_FLAG;
1458  len = gpib_read_str(gpib_iobuff, GPIB_IOBUFF_LEN, &status);
1459  if(status & ERROR_MASK)
1460  {
1462  if(debuglevel & GPIB_ERR)
1463  printf("[GPIB Read ERROR]\n");
1464  return(status & ERROR_MASK);
1465  }
1466 
1467  if(!len)
1468  return(0);
1469 
1470  if( !(status & EOI_FLAG) )
1471  {
1472  if(debuglevel & GPIB_ERR)
1473  printf("[GPIB buffer OVERFLOW!]\n");
1474  }
1475 
1476  ind = 0;
1477 
1478  while(ind < len)
1479  {
1480  ch = gpib_iobuff[ind++];
1482  if(ch >= 0x20 && ch <= 0x2f)
1483  {
1484  SS80_Check_Unit(ch - 0x20);
1485  continue;
1486  }
1487 
1489  if(ch == 0x01)
1490  {
1493  ind++;
1494 #if SDEBUG
1496  printf("[SS80 HP-IB Parity Checking - TODO]\n");
1497 #endif
1499  break;
1500  }
1504 
1506  if(ch == 0x02)
1507  {
1508  ind += 4;
1511 #if SDEBUG
1513  printf("[SS80 Read Loopback - TODO]\n");
1514 #endif
1515  break;
1516  }
1520 
1522  if(ch == 0x03)
1523  {
1524  ind += 4;
1527 #if SDEBUG
1529  printf("[SS80 Write Loopback - TODO]\n");
1530 #endif
1531  break;
1532  }
1533 
1537 
1541  if(ch == 0x08) // 0x08 OP Code
1542  {
1543 #if SDEBUG
1545  printf("[SS80 Channel Independent Clear (%d)]\n", SS80s->unitNO);
1546 #endif
1548  }
1549 
1553 
1556  if(ch == 0x09) // 0x09 OP Code
1557  {
1558 #if SDEBUG
1560  printf("[SS80 Cancel (%d)]\n", SS80s->unitNO);
1561 #endif
1562  return(SS80_Cancel( ) );
1563 
1564  break;
1565  }
1566 
1567  if(debuglevel & GPIB_ERR)
1568  printf("[SS80 Invalid OP Code (%02XH)]\n", ch & 0xff);
1569  break;
1570  }
1571 
1572  if( ind != len)
1573  {
1574  if(debuglevel & GPIB_ERR)
1575  printf("[SS80 Transparent Command, Error at (%d) of (%d) OP Codes]\n",
1576  ind,len);
1577  }
1578 
1579  return(status & ERROR_MASK);
1580 }
1581 
1582 
1592 
1593 int SS80_cmd_seek( void )
1594 {
1599  {
1600  SS80s->qstat = 1;
1601  SS80s->Errors |= ERR_SEEK;
1602 
1603  if(debuglevel & GPIB_ERR)
1604  printf("[SS80 Seek OVERFLOW at %08lXH]\n",
1606  return(1);
1607  }
1608 
1609 #if SDEBUG
1611  printf("[SS80 Seek:%08lXH]\n", (long)SS80_Blocks_to_Bytes(SS80s->AddressBlocks));
1612 #endif
1613  return (0);
1614 }
1615 
1616 
1640 
1641 int SS80_Report( void )
1642 {
1643  uint8_t tmp[1];
1644  uint16_t status;
1645 
1646  tmp[0] = SS80s->qstat;
1647 
1648  status = EOI_FLAG;
1649  if( gpib_write_str(tmp, sizeof(tmp), &status) != sizeof(tmp))
1650  {
1651  if(debuglevel & GPIB_ERR)
1652  printf("[SS80 qstat send FAILED]\n");
1653  return(status & ERROR_MASK);
1654  }
1655  else
1656  {
1657 #if SDEBUG
1659  printf("[SS80 qstat %02X]\n", SS80s->qstat);
1660 #endif
1661  }
1662  SS80s->qstat = 0;
1663  return ( 0 );
1664 }
1665 
1666 
1725 
1728 
1731 
1734 void Clear_Common(int u)
1735 {
1736  if(u != SS80s->unitNO && u != 15)
1737  return;
1738 
1739  if(u == 15)
1740  SS80s->unitNO = 0;
1741  SS80s->volNO = 0;
1742  SS80s->AddressBlocks = 0;
1743  SS80s->Length = 0;
1744  SS80s->estate = EXEC_IDLE;
1745 
1754 
1759  SS80s->Errors = 0; // Status Mask
1760  SS80s->qstat = 0; // Clear qstat
1761 }
1762 
1763 
1774 
1776 {
1777  Clear_Common( u );
1779  return(0);
1780 }
1781 
1782 
1792 
1794 {
1795  Clear_Common(15);
1798  return(0);
1799 }
1800 
1801 
1810 
1812 {
1813 #if SDEBUG
1815  printf("[SS80 SDC]\n");
1816 #endif
1817  Clear_Common( u );
1819  return(0);
1820 }
1821 
1822 
1830 
1831 int SS80_Amigo_Clear( void )
1832 {
1833  uint8_t parity[1];
1834  uint16_t status;
1835 
1836  status = 0;
1837  if( gpib_read_str(parity, sizeof(parity), &status) != sizeof(parity))
1838  {
1839  if(debuglevel & GPIB_ERR)
1840  printf("[GPIB Read Error]\n");
1841  return(status & ERROR_MASK);
1842  }
1843  else
1844  {
1845 #if SDEBUG
1847  printf("[Amigo Clear]\n");
1848 #endif
1849  }
1850  Clear_Common(15);
1852  return(0);
1853 }
1854 
1855 
1864 
1866 {
1868 
1869  SS80s->estate = EXEC_IDLE;
1871  return(0);
1872 }
1873 
1874 
1879 
1880 int SS80_increment( void )
1881 {
1882  SS80s->AddressBlocks += 1L;
1883 #if SDEBUG
1885  printf("[SS80 Increment to (%lXH)]\n",
1887 #endif
1888  return(0);
1889 }
1890 
1891 
1897 
1899 {
1900  uint8_t tmp[1];
1901  uint16_t status = EOI_FLAG;
1902 
1903  SS80s->qstat = 1;
1904  tmp[0] = SS80s->qstat;
1905 
1906  status = EOI_FLAG;
1907  if( gpib_write_str(tmp,sizeof(tmp), &status) != sizeof(tmp))
1908  {
1909  if(debuglevel & GPIB_ERR)
1910  printf("[GPIB Error Return - Write ERROR]\n");
1911  return(status & ERROR_MASK);
1912  }
1913  SS80s->qstat = 0;
1914  return(0);
1915 }
1916 
1917 
1929 int SS80_COMMANDS(uint8_t ch)
1930 {
1932  {
1933  if(ch == 0x65 )
1934  {
1935  if(SS80_is_MLA(listening))
1936  {
1937 #if SDEBUG
1939  printf("[SS80 Command State]\n");
1940 #endif
1941  return ( SS80_Command_State() );
1942  }
1943  return(0);
1944  }
1945 
1946  if(ch == 0x6e )
1947  {
1949  {
1951  printf("[SS00 Execute State]\n");
1952  return ( SS80_Execute_State() );
1953 
1954  }
1955  return(0);
1956  }
1957 
1958  if(ch == 0x70 )
1959  {
1960  if(SS80_is_MTA(talking) )
1961  {
1962 #if SDEBUG
1964  printf("[SS80 Report State]\n");
1965 #endif
1966  return( SS80_Report() );
1967  }
1968 
1969  if(SS80_is_MLA(listening))
1970  {
1971 #if SDEBUG
1973  printf("[Amigo Clear]\n");
1974 #endif
1976  return( SS80_Amigo_Clear() );
1977  }
1978  return (0);
1979  }
1980  if(ch == 0x72 )
1981  {
1982  if(SS80_is_MLA(listening) )
1983  {
1984 #if SDEBUG
1986  printf("[SS80 Transparent]\n");
1987 #endif
1988  return( SS80_Transparent_State() );
1989  }
1990  }
1991  if(debuglevel & GPIB_ERR)
1992  printf("[SS80 SC Unknown: %02XH, listen:%02XH, talk:%02XH]\n",
1993  0xff & ch, 0xff & listening, 0xff & talking);
1994  return(0);
1995  }
1996 
1997  if(debuglevel & GPIB_ERR)
1998  printf("[SS80 Unknown SC: %02XH, listen:%02XH, talk:%02XH]\n",
1999  0xff & ch, 0xff & listening, 0xff & talking);
2000  return(0);
2001 }
SS80StateType::Errors
int Errors
Errors.
Definition: drives.h:212
SS80VolumeType::MAX_CYLINDER
uint32_t MAX_CYLINDER
Definition: drives.h:186
GPIB_RW_STR_TIMING
#define GPIB_RW_STR_TIMING
Definition: debug.h:11
SS80_Test
void SS80_Test(void)
SS80 GPIB function.
Definition: ss80.c:123
SS80s
SS80StateType * SS80s
Definition: drives.c:39
EXEC_LOCATE_AND_WRITE
@ EXEC_LOCATE_AND_WRITE
Definition: ss80.c:99
ERR_WRITE
#define ERR_WRITE
Definition: defines.h:81
printf
MEMSPACE int printf(const char *format,...)
SS80ControllerType::UNITS_INSTALLED
uint16_t UNITS_INSTALLED
Definition: drives.h:131
TD
int TD[]
SS80 GPIB test string.
Definition: ss80.c:106
dbf_open_write
int dbf_open_write(char *name, uint32_t pos, void *buff, int size, int *errors)
Open, Seek, Write data and Close FatFs functions.
Definition: gpib_hal.c:379
gpib_timer_elapsed_begin
void gpib_timer_elapsed_begin(void)
Start measuring time - used with hpib_timer_elapsed_end()
Definition: gpib.c:133
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
SS80VolumeType::MAX_HEAD
uint8_t MAX_HEAD
Definition: drives.h:187
SS80_Channel_Independent_Clear
int SS80_Channel_Independent_Clear(int u)
Channel Independent Clear.
Definition: ss80.c:1775
EXEC_SEND_STATUS
@ EXEC_SEND_STATUS
Definition: ss80.c:100
EXEC_LOCATE_AND_READ
@ EXEC_LOCATE_AND_READ
Definition: ss80.c:98
SS80_test_extended_status
int SS80_test_extended_status(uint8_t *p, int bit)
SS80 test extendend status bits and display them.
Definition: ss80.c:683
fault_t
Fault bit and Message type.
Definition: defines.h:91
gpib_disable_PPR
void gpib_disable_PPR(int bit)
Disable PPR (Parallel Poll Response) for a device.
Definition: gpib.c:436
SS80_Bytes_to_Blocks
uint32_t SS80_Bytes_to_Blocks(uint32_t bytes)
SS80 Return current block addresss from bytes.
Definition: ss80.c:353
SS80UnitPack
uint8_t * SS80UnitPack(int *size)
Pack Unit data into bytes.
Definition: ss80.c:198
Clear_Common
void Clear_Common(int u)
Universal and Decvice clear code.
Definition: ss80.c:1734
V2B_MSB_Index1
void V2B_MSB_Index1(uint8_t *B, int index, int size, uint32_t val)
Initialize SS80 state and bus.
Definition: ss80.c:159
SS80DiskType::VOLUME
SS80VolumeType VOLUME
Definition: drives.h:200
ss80.h
SS80 disk emulator for HP85 disk emulator project for AVR.
SS80StateType::volNO
BYTE volNO
SS80 Volume.
Definition: drives.h:216
ss80exec
ss80exec
Definition: ss80.c:95
GPIB_IO_RD
#define GPIB_IO_RD(a)
changes pin mode to read, then read
Definition: gpib_hal.h:115
SS80ControllerType::TYPE
uint8_t TYPE
Definition: drives.h:133
amigo.h
AMIGO disk emulator for HP85 disk emulator project for AVR.
SS80_Check_Unit
void SS80_Check_Unit(uint8_t unit)
Check unit number and assign.
Definition: ss80.c:977
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
faults
fault_t faults[]
fault messages TODO move these to __memx
Definition: ss80.c:642
SS80_Report
int SS80_Report(void)
send QSTAT
Definition: ss80.c:1641
fault_t::index
int index
Definition: defines.h:93
SS80_Amigo_Clear
int SS80_Amigo_Clear(void)
Amigo Clear.
Definition: ss80.c:1831
Devices
DeviceType Devices[MAX_DEVICES]
Definition: drives.c:32
IFC_FLAG
#define IFC_FLAG
Definition: gpib.h:37
SS80_display_extended_status
void SS80_display_extended_status(uint8_t *p, char *message)
SS80 Display extended 8 byte status messages for debuging.
Definition: ss80.c:714
SS80_describe
int SS80_describe(void)
Send Controler,Unit and Volume description.
Definition: ss80.c:929
SS80_locate_and_read
int SS80_locate_and_read(void)
SS80 Locate and Read COmmend.
Definition: ss80.c:374
UINT
unsigned int UINT
Definition: ff.h:53
SS80DiskType::HEADER
HeaderType HEADER
Definition: drives.h:196
dbf_open_read
int dbf_open_read(char *name, uint32_t pos, void *buff, int size, int *errors)
Open, Seek, Read data and Close FatFs functions.
Definition: gpib_hal.c:314
ERROR_MASK
#define ERROR_MASK
Definition: gpib.h:49
mmc_wp_status
MEMSPACE int mmc_wp_status()
MMC Card Write Protect status.
Definition: mmc_hal.c:344
defines.h
GPIB, AMIGO, SS80 and device defines.
ERR_SEEK
#define ERR_SEEK
Definition: defines.h:82
SS80UnitType::BYTES_PER_BLOCK
uint16_t BYTES_PER_BLOCK
Definition: drives.h:159
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
SS80_Blocks_to_Bytes
uint32_t SS80_Blocks_to_Bytes(uint32_t block)
SS80 Return current address in bytes.
Definition: ss80.c:344
SS80UnitType::OPTIMAL_RETRY_TIME
uint16_t OPTIMAL_RETRY_TIME
Definition: drives.h:164
SS80DiskType::CONTROLLER
SS80ControllerType CONTROLLER
Definition: drives.h:198
SS80_send_status
int SS80_send_status(void)
SS80 send detailed status.
Definition: ss80.c:854
SS80_Execute_State
int SS80_Execute_State(void)
SS80 Execute state process.
Definition: ss80.c:306
NULL
#define NULL
Definition: user_config.h:85
SS80StateType::unitNO
BYTE unitNO
SS80 Unit.
Definition: drives.h:214
GPIB_IOBUFF_LEN
#define GPIB_IOBUFF_LEN
Definition: gpib.h:101
SS80_Cancel
int SS80_Cancel()
Cancel transaction.
Definition: ss80.c:1865
gpib_iobuff
uint8_t gpib_iobuff[GPIB_IOBUFF_LEN]
common IO buffer for gpib_read_str() and gpib_write_str()
Definition: gpib.c:77
GPIB_ERR
#define GPIB_ERR
Definition: debug.h:4
GPIB_TODO
#define GPIB_TODO
Definition: debug.h:8
HeaderType::NAME
char * NAME
Maximun lengh of device file name.
Definition: drives.h:45
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
SS80_init
void SS80_init(void)
SS80 nitialize all devices Initialize ALL SS80 devives.
Definition: ss80.c:275
SS80_set_extended_status
void SS80_set_extended_status(uint8_t *p, int bit)
SS80 set extendend status bits.
Definition: ss80.c:700
ERR_UNIT
#define ERR_UNIT
Definition: defines.h:86
GPIB_DEVICE_STATE_MESSAGES
#define GPIB_DEVICE_STATE_MESSAGES
Definition: debug.h:9
SS80_increment
int SS80_increment(void)
Increment position Addressy by one block.
Definition: ss80.c:1880
SS80VolumeType::MAX_BLOCK_NUMBER
uint32_t MAX_BLOCK_NUMBER
Definition: drives.h:189
SS80_cmd_seek
int SS80_cmd_seek(void)
Limit tests for disk seek.
Definition: ss80.c:1593
SS80UnitType::BURST_SIZE
uint8_t BURST_SIZE
Definition: drives.h:161
SS80_is_MTA
int SS80_is_MTA(int address)
Check if SS80 talking address.
Definition: gpib_task.c:87
SS80_error_return
int SS80_error_return(void)
Common SS80 Error Return function. Send qstat = 1.
Definition: ss80.c:1898
sep
MEMSPACE void sep()
print seperator
Definition: parsing.c:35
SS80DiskType::UNIT
SS80UnitType UNIT
Definition: drives.h:199
EOI_FLAG
#define EOI_FLAG
bus flags
Definition: gpib.h:33
SS80UnitType::BUFFERED_BLOCKS
uint8_t BUFFERED_BLOCKS
Definition: drives.h:160
SS80ControllerPack
uint8_t * SS80ControllerPack(int *size)
Pack Controller data into bytes.
Definition: ss80.c:169
listening
uint8_t listening
gpib listen address
Definition: gpib.c:90
GPIB_DISK_IO_TIMING
#define GPIB_DISK_IO_TIMING
Definition: debug.h:10
EXEC_DESCRIBE
@ EXEC_DESCRIBE
Definition: ss80.c:101
SS80UnitType::DEVICE_NUMBER
uint32_t DEVICE_NUMBER
Definition: drives.h:158
SS80_Selected_Device_Clear
int SS80_Selected_Device_Clear(int u)
Selected Device Clear.
Definition: ss80.c:1811
gpib_write_str
int gpib_write_str(uint8_t *buf, int size, uint16_t *status)
Send string to GPIB BUS - controlled by status flags.
Definition: gpib.c:1396
spoll
uint8_t spoll
gpib serial poll status
Definition: gpib.c:95
SS80ControllerType::TRANSFER_RATE
uint16_t TRANSFER_RATE
Definition: drives.h:132
ERR_WP
#define ERR_WP
Definition: defines.h:83
ERR_GPIB
#define ERR_GPIB
Definition: defines.h:85
gpib_enable_PPR
void gpib_enable_PPR(int bit)
Definition: gpib.c:416
vector.h
ATN_FLAG
#define ATN_FLAG
Definition: gpib.h:35
SS80UnitType::BLOCK_TIME
uint16_t BLOCK_TIME
Definition: drives.h:162
gpib_timer_elapsed_end
void gpib_timer_elapsed_end(char *msg)
Display user message and time delta since gpib_timer_elapsed_begin() call.
Definition: gpib.c:164
SS80StateType::AddressBlocks
uint32_t AddressBlocks
Address in Blocks.
Definition: drives.h:218
gpib_read_str
int gpib_read_str(uint8_t *buf, int size, uint16_t *status)
Read string from GPIB BUS - controlled by status flags.
Definition: gpib.c:1307
SS80VolumeType::INTERLEAVE
uint8_t INTERLEAVE
Definition: drives.h:190
SS80_TYPE
@ SS80_TYPE
Definition: drives.h:251
SS80_Transparent_State
int SS80_Transparent_State(void)
(0x70 or 0x72) OP Code processing.
Definition: ss80.c:1442
SS80_is_MLA
int SS80_is_MLA(int address)
Check if SS80 listening address.
Definition: gpib_task.c:74
talking
uint8_t talking
GPIB BUS states.
Definition: gpib.c:85
SS80_Universal_Device_Clear
int SS80_Universal_Device_Clear(void)
Universal Device CLear.
Definition: ss80.c:1793
EXEC_IDLE
@ EXEC_IDLE
Definition: ss80.c:97
Mem_Clear
#define Mem_Clear(a)
Definition: user_config.h:96
gpib.h
GPIB emulator for HP85 disk emulator project for AVR.
ERR_READ
#define ERR_READ
Definition: defines.h:80
B2V_MSB
uint32_t B2V_MSB(uint8_t *B, int index, int size)
Convert a byte array into a value bytes are MSB ... LSB order.
Definition: vector.c:58
SS80_Check_Volume
void SS80_Check_Volume(uint8_t volume)
Check volume number and assign.
Definition: ss80.c:995
SS80p
SS80DiskType * SS80p
Active SS80 Device.
Definition: drives.c:38
SS80UnitType::ACCESS_TIME
uint16_t ACCESS_TIME
Definition: drives.h:165
SS80StateType::estate
int estate
Execute state index.
Definition: drives.h:208
SS80VolumeType::MAX_SECTOR
uint16_t MAX_SECTOR
Definition: drives.h:188
SS80UnitType::FIXED_VOLUMES
uint8_t FIXED_VOLUMES
Definition: drives.h:167
SS80_locate_and_write
int SS80_locate_and_write(void)
SS80 Locate and Write Commend.
Definition: ss80.c:506
SS80VolumePack
uint8_t * SS80VolumePack(int *size)
Pack Voulme data into bytes.
Definition: ss80.c:245
DWORD
unsigned long DWORD
Definition: ff.h:56
SS80_COMMANDS
int SS80_COMMANDS(uint8_t ch)
SS80 COMMANDS States.
Definition: ss80.c:1929
SS80StateType::qstat
uint8_t qstat
Qstat variable.
Definition: drives.h:210
gpib_hal.h
GPIB emulator hardwware layer for HP85 disk emulator project for AVR.
V2B_MSB
void V2B_MSB(uint8_t *B, int index, int size, uint32_t val)
Convert Value into byte array bytes are MSB ... LSB order.
Definition: vector.c:22
SS80StateType::Length
uint32_t Length
Length in Bytes.
Definition: drives.h:220
SS80UnitType::UNIT_TYPE
uint8_t UNIT_TYPE
Definition: drives.h:157
IFC
#define IFC
Definition: gpib_hal.h:64
debug.h
SS80_Command_State
int SS80_Command_State(void)
Process OP Codes following 0x65 Command State.
Definition: ss80.c:1047
SS80UnitType::CONTINUOUS_TRANSFER_RATE
uint16_t CONTINUOUS_TRANSFER_RATE
Definition: drives.h:163