HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
amigo.c
Go to the documentation of this file.
1 
21 #include "user_config.h"
22 
23 #include "defines.h"
24 #include "gpib.h"
25 #include "gpib_hal.h"
26 #include "gpib_task.h"
27 #include "amigo.h"
28 #include "debug.h"
29 
30 #ifdef AMIGO
31 
47 
88 
154 
155 extern uint8_t talking;
156 extern uint8_t listening;
157 
158 enum AMIGO_states
159 {
160  AMIGO_IDLE = 0,
161  AMIGO_REQUEST_STATUS,
162  AMIGO_REQUEST_STATUS_UNBUFFERED,
163  AMIGO_REQUEST_STATUS_BUFFERED,
164  AMIGO_REQUEST_LOGICAL_ADDRESS,
165  AMIGO_COLD_LOAD_READ,
166  AMIGO_READ_UNBUFFERED,
167  AMIGO_READ_BUFFERED,
168  AMIGO_WRITE_UNBUFFERED,
169  AMIGO_WRITE_BUFFERED,
170  AMIGO_INITIALIZE,
171 };
172 
175 
176 void amigo_init()
177 {
178  int i;
179  for(i=MAX_DEVICES-1;i>=0;--i)
180  {
181  if(Devices[i].TYPE == AMIGO_TYPE)
182  {
184  AMIGOs->state = AMIGO_IDLE;
185 
186  memset(AMIGOs->status,0,sizeof(AMIGOs->status));
187  memset(AMIGOs->logical_address,0,sizeof(AMIGOs->logical_address));
188 
190  AMIGOs->unitNO = 0;
191 
192  AMIGOs->sector = 0;
193  AMIGOs->head = 0;
194  AMIGOs->cyl = 0;
195 
196  AMIGOs->dsj = 2;
197  AMIGOs->Errors = 0;
198 
200  gpib_disable_PPR(AMIGOp->HEADER.PPR);
201  }
202  }
203 }
204 
205 
211 
213 {
214 
215  AMIGOs->logical_address[0] = 0xff & (AMIGOs->cyl >> 8);
216 //LSB
217  AMIGOs->logical_address[1] = 0xff & (AMIGOs->cyl);
218  AMIGOs->logical_address[2] = 0xff & (AMIGOs->head);
219  AMIGOs->logical_address[3] = 0xff & (AMIGOs->sector);
220  return(0);
221 }
222 
223 
229 
231 {
233 #if SDEBUG
235  printf("[AMIGO request status]\n");
236 #endif
237  AMIGOs->status[0] = 0x00; // Status 1
239  AMIGOs->status[1] = AMIGOs->unitNO; // Unit
240  AMIGOs->status[2] = 0x0d; // Status 2 (0110 = hp format) << 1, 1=HP9121
241  AMIGOs->status[3] = 0x00; //
242 
243  if(mmc_wp_status())
244  {
245  AMIGOs->status[3] |= 0x40; // Write protect 0x40, reserved = 0x20
246  AMIGOs->status[3] |= 0x20; // reserved = 0x20 ???
247  }
248 
249  if(AMIGOs->dsj == 2)
250  {
251  AMIGOs->status[0] = 0b00010011; // S1 error power on
252  AMIGOs->status[3] |= 0x08; // F bit, power up
253  }
254  else if(AMIGOs->Errors || AMIGOs->dsj == 1)
255  {
256 //FIXME added invalid unit error
257  if(AMIGOs->Errors & ERR_UNIT)
258  AMIGOs->status[0] = 0b00010011; // Unit Error
259  else if(AMIGOs->Errors & ERR_GPIB)
260  AMIGOs->status[0] = 0b00001010; // S1 error I/O error
261  else if(AMIGOs->Errors & ERR_DISK)
262  AMIGOs->status[3] |= 0x03; // Do disk in drive
263  else if(AMIGOs->Errors & ERR_WRITE)
264  AMIGOs->status[0] = 0b00010011; // S1 error write error
265  else if(AMIGOs->Errors & ERR_SEEK)
266  AMIGOs->status[3] |= 0x04; // Seek
267 
268  AMIGOs->status[3] |= 0x10; // E bit hardware failure
269  AMIGOs->status[2] |= 0x80; // Bit 15
270  }
271 
272  gpib_enable_PPR(AMIGOp->HEADER.PPR);
273  return(0);
274 }
275 
276 
284 
286 {
287  uint16_t status;
288  UINT len;
289 
290 #if SDEBUG
292  printf("[AMIGO send logical address]\n");
293 #endif
294  status = EOI_FLAG;
295  len = gpib_write_str(AMIGOs->logical_address,4,&status);
296 
297  if(status & ERROR_MASK || len != 4)
298  {
299  AMIGOs->Errors |= ERR_GPIB;
300  AMIGOs->dsj = 1;
301  if(debuglevel & GPIB_ERR)
302  printf("[AMIGO GPIB write error]\n");
303  }
304  gpib_enable_PPR(AMIGOp->HEADER.PPR);
305  return(status & ERROR_MASK);
306 }
307 
308 
316 
317 int amigo_send_status()
318 {
319  uint16_t status;
320  UINT len;
321 
322 #if SDEBUG
324  printf("[AMIGO send status]\n");
325 #endif
326  status = EOI_FLAG;
327  len = gpib_write_str(AMIGOs->status,4,&status);
328  if(status & ERROR_MASK || len != 4)
329  {
330  AMIGOs->Errors |= ERR_GPIB;
331  AMIGOs->dsj = 1;
332  if(debuglevel & GPIB_ERR)
333  printf("[AMIGO GPIB write error]\n");
334  }
335  else
336  {
337  AMIGOs->Errors = 0;
338  AMIGOs->dsj = 0;
339  }
340  gpib_enable_PPR(AMIGOp->HEADER.PPR);
341  return(status & ERROR_MASK);
342 }
343 
344 
351 
352 static DWORD amigo_chs_to_logical(AMIGOStateType *p, char *msg)
353 {
354  DWORD pos;
355  pos = (long) ( AMIGOp->GEOMETRY.SECTORS_PER_TRACK * p->head);
356  pos += (long) ( (AMIGOp->GEOMETRY.SECTORS_PER_TRACK * AMIGOp->GEOMETRY.HEADS) * p->cyl);
357  pos += (long) p->sector;
358  pos *= (long) AMIGOp->GEOMETRY.BYTES_PER_SECTOR;
359 
360 #if SDEBUG
362  printf("[AMIGO %s, P:%08lxH, U:%d C:%d H:%d S:%d]\n",
363  msg, pos, AMIGOs->unitNO, p->cyl, p->head, p->sector);
364 #endif
365  return(pos);
366 }
367 
368 
374 
375 static int amigo_overflow_check(AMIGOStateType *p, char *msg)
376 {
377  int stat = 0;
378  while(p->sector >= AMIGOp->GEOMETRY.SECTORS_PER_TRACK)
379  {
380  p->sector = 0;
381  p->head++;
382  while (p->head >= AMIGOp->GEOMETRY.HEADS)
383  {
384  p->head = 0;
385  p->cyl++;
386  if (p->cyl >= AMIGOp->GEOMETRY.CYLINDERS)
387  {
388  stat = 1;
389  if(debuglevel & GPIB_ERR && msg != NULL)
390  printf("[AMIGO %s pos OVERFLOW]\n", msg);
391  }
392  }
393  }
394  return(stat);
395 }
396 
397 
404 
405 int amigo_increment(char *msg)
406 {
407  int stat = 0;
408  AMIGOStateType tmp = *AMIGOs;
409 
410  ++tmp.sector;
411 
412  stat = amigo_overflow_check((AMIGOStateType *) &tmp, msg);
413  if(!stat)
414  {
415  *AMIGOs = tmp;
416  }
417  return(stat);
418 }
419 
420 
431 
432 int amigo_seek( AMIGOStateType *p)
433 {
434  int stat = 0;
435 
436  stat = amigo_overflow_check(p, "Seek");
437  if(stat)
438  {
439  AMIGOs->dsj = 1;
440  AMIGOs->Errors |= ERR_SEEK;
441  }
442  else
443  {
444  AMIGOs->sector = p->sector;
445  AMIGOs->head = p->head;
446  AMIGOs->cyl = p->cyl;
447  }
448 
449  gpib_enable_PPR(AMIGOp->HEADER.PPR);
450  return(stat);
451 }
452 
453 
459 
460 int amigo_verify(uint16_t sectors)
461 {
462  int len;
463  int stat = 0;
464  DWORD pos;
465 
466  pos = amigo_chs_to_logical(AMIGOs, "Verify Start");
467 
468 #if SDEBUG
470  printf("[AMIGO verify P:%08lXH, sectors:%04XH]\n", pos, sectors);
471 #endif
472 
473  while(sectors--)
474  {
475  pos = amigo_chs_to_logical(AMIGOs, "Verfify");
476 
477 #if SDEBUG
480 #endif
481 
482  // dbf_open_read sets error conditions like ERROR_READ, etc
483  len = dbf_open_read(AMIGOp->HEADER.NAME, pos, gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &AMIGOs->Errors);
484 
485 #if SDEBUG
487  gpib_timer_elapsed_end("disk READ ");
488 #endif
489  if(AMIGOs->Errors || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR )
490  {
491  AMIGOs->dsj = 1;
492  stat = 1;
493  break;
494  }
495  if(amigo_increment("Verify")) // address overflow
496  {
497  stat = 1;
498  break;
499  }
500  }
501  gpib_enable_PPR(AMIGOp->HEADER.PPR);
502  return(stat);
503 }
504 
505 
514 int amigo_format(uint8_t db)
515 {
516 
517  DWORD pos;
518  int len;
519  int stat = 0;
520 
521  AMIGOs->sector = 0;
522  AMIGOs->head = 0;
523  AMIGOs->cyl = 0;
524 
525  memset((void *) gpib_iobuff, db, AMIGOp->GEOMETRY.BYTES_PER_SECTOR);
526 
527 #if SDEBUG
529  printf("[AMIGO format]\n");
532 #endif
533  while( 1 )
534  {
536  pos = amigo_chs_to_logical(AMIGOs, "Format");
537 
538  len = dbf_open_write(AMIGOp->HEADER.NAME,
539  pos, gpib_iobuff,AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &AMIGOs->Errors);
540 
541  if(AMIGOs->Errors || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR)
542  {
543  AMIGOs->Errors |= ERR_WRITE;
544  AMIGOs->dsj = 1;
545  stat = 1;
546  break;
547  }
548 
550  if( amigo_increment(NULL) )
551  {
552 // reset sector,head,cyl
553  AMIGOs->sector = 0;
554  AMIGOs->head = 0;
555  AMIGOs->cyl = 0;
556  AMIGOs->dsj = 0;
557  stat = 0;
558  break;
559  }
560 
561  }
562 #if SDEBUG
564  gpib_timer_elapsed_end("Format");
566  printf("[AMIGO Format Done]\n");
567 #endif
568  gpib_enable_PPR(AMIGOp->HEADER.PPR);
569  return(stat);
570 }
571 
572 
573 #if 0
574 
581 {
582  uint16_t status;
583  int len;
584  DWORD pos;
585 
586 //READ SECTOR
587  pos = amigo_chs_to_logical(AMIGOs, "Buffered Read");
588 #if SDEBUG
590  printf("AMIGOs->state:%d\n", AMIGOs->state);
593 #endif
594 
595  len = dbf_open_read(AMIGOp->HEADER.NAME, pos, gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &AMIGOs->Errors);
596 
597 #if SDEBUG
599  gpib_timer_elapsed_end("disk READ ");
600 #endif
601  if(AMIGOs->Errors)
602  {
603  // Any AMIGOs->Errors errors were set in the dbf_pen_read() call
604  AMIGOs->dsj = 1;
605  if(debuglevel & GPIB_ERR)
606  printf("[AMIGO GPIB disk read error]\n");
607  gpib_enable_PPR(AMIGOp->HEADER.PPR);
608  return(0);
609  }
610 
611 #if SDEBUG
614 #endif
615 
616 // WRITE DATA
617  status = EOI_FLAG;
618  len = gpib_write_str(gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &status);
619 #if SDEBUG
621  gpib_timer_elapsed_end("GPIB write");
622 #endif
623  if(status & ERROR_MASK || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR)
624  {
625  AMIGOs->dsj = 1;
626  AMIGOs->Errors |= ERR_GPIB;
627  if(debuglevel & GPIB_ERR)
628  printf("[AMIGO GPIB bus write error]\n");
629  gpib_enable_PPR(AMIGOp->HEADER.PPR);
630  return(status & ERROR_MASK);
631  }
632 
635  if( amigo_increment("Buffered Read") ) //overflow
636  {
637  AMIGOs->dsj = 1;
638  AMIGOs->Errors |= ERR_SEEK;
639  }
640 
641  gpib_enable_PPR(AMIGOp->HEADER.PPR);
642  return(0);
643 }
644 #endif
645 
646 // ====================================================
653 
655 {
656  int len;
657  DWORD pos;
658 
659 //READ SECTOR
660  pos = amigo_chs_to_logical(AMIGOs, "Buffered Read");
661 #if SDEBUG
663  printf("AMIGOs->state:%d\n", AMIGOs->state);
666 #endif
667 
668  len = dbf_open_read(AMIGOp->HEADER.NAME, pos, gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &AMIGOs->Errors);
669 
670 #if SDEBUG
672  gpib_timer_elapsed_end("disk READ ");
673 #endif
674 
675  if(AMIGOs->Errors || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR)
676  {
677  AMIGOs->dsj = 1;
678  gpib_enable_PPR(AMIGOp->HEADER.PPR);
679  return(0);
680  }
681 
684  if( amigo_increment("Buffered Read") ) //overflow
685  {
686  AMIGOs->dsj = 1;
687  AMIGOs->Errors |= ERR_SEEK;
688  }
689 
690  gpib_enable_PPR(AMIGOp->HEADER.PPR);
691  return(0);
692 }
693 
699 
701 {
702  uint16_t status;
703  int len;
704 
705 #if SDEBUG
708 #endif
709 
710 // WRITE DATA
711  status = EOI_FLAG;
712  len = gpib_write_str(gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &status);
713 
714 #if SDEBUG
716  gpib_timer_elapsed_end("GPIB write");
717 #endif
718 
719  if(status & ERROR_MASK || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR)
720  {
721  AMIGOs->dsj = 1;
722  AMIGOs->Errors |= ERR_GPIB;
723  if(debuglevel & GPIB_ERR)
724  printf("[AMIGO GPIB write error]\n");
725  }
726 
727  gpib_enable_PPR(AMIGOp->HEADER.PPR);
728  return(status);
729 }
730 
731 
732 
737 
739 {
740  uint16_t status;
741  int len;
742 
743  DWORD pos;
744 
745  pos = amigo_chs_to_logical(AMIGOs, "Buffered Write");
746 
747 #if SDEBUG
750 #endif
751  status = 0;
752  len = gpib_read_str(gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &status);
753 
754 
755 #if SDEBUG
757  printf("AMIGOs->state:%d\n", AMIGOs->state);
759  gpib_timer_elapsed_end("GPIB read");
760 #endif
761 
762  if(status & ERROR_MASK || len != AMIGOp->GEOMETRY.BYTES_PER_SECTOR)
763  {
764  AMIGOs->dsj = 1;
765  AMIGOs->Errors |= ERR_GPIB;
766  if(debuglevel & GPIB_ERR)
767  printf("[AMIGO Write GPIB read error]\n");
768  gpib_enable_PPR(AMIGOp->HEADER.PPR);
769  return(status & ERROR_MASK);
770  }
771 
772 #if SDEBUG
775 #endif
776 
777  len = dbf_open_write(AMIGOp->HEADER.NAME, pos, gpib_iobuff, AMIGOp->GEOMETRY.BYTES_PER_SECTOR, &AMIGOs->Errors);
778 
779 #if SDEBUG
781  gpib_timer_elapsed_end("disk WRITE");
782 #endif
783 
784  if(AMIGOs->Errors)
785  {
786  AMIGOs->dsj = 1;
787  if(debuglevel & GPIB_ERR)
788  printf("[AMIGO Write disk write error]\n");
789  gpib_enable_PPR(AMIGOp->HEADER.PPR);
790  return(0);
791  }
792 
793  if( amigo_increment("Buffered Write") ) //overflow
794  {
795  AMIGOs->dsj = 1;
796  AMIGOs->Errors |= ERR_SEEK;
797  }
798 
799  gpib_enable_PPR(AMIGOp->HEADER.PPR);
800 
801  return(status & ERROR_MASK);
802 }
803 
804 
811 
812 int amigo_cmd_dsj()
813 {
814  uint8_t tmp[1];
815  uint16_t status;
816  int len;
817 
818  tmp[0] = AMIGOs->dsj;
819 
820  status = EOI_FLAG;
821  len = gpib_write_str(tmp, sizeof(tmp), &status);
822  if(status & ERROR_MASK || len != sizeof(tmp) )
823  {
824  AMIGOs->dsj = 1;
825  AMIGOs->Errors |= ERR_GPIB;
826  if(debuglevel & GPIB_ERR)
827  printf("[AIMGO: DSJ send failed]\n");
828  return(status & ERROR_MASK);
829  }
830  else
831  {
832 #if SDEBUG
834  printf("[DSJ %02XH]\n", AMIGOs->dsj);
835 #endif
836  }
837  AMIGOs->dsj = 0;
838  AMIGOs->Errors = 0;
839  return ( 0 );
840 }
841 
842 
849 
850 int amigo_cmd_wakeup()
851 {
852  uint8_t tmp[1];
853  uint16_t status;
854  UINT len;
855 
856 #if SDEBUG
858  printf("[AMIGO Wakeup]\n");
859 #endif
860  tmp[0] = AMIGOs->dsj;
861  len = gpib_write_str(tmp, 1, &status);
862  if(status & ERROR_MASK || len != 1)
863  {
864  AMIGOs->dsj = 1;
865  AMIGOs->Errors |= ERR_GPIB;
866  if(debuglevel & GPIB_ERR)
867  printf("[AMIGO GPIB write error]\n");
868  }
870  gpib_enable_PPR(AMIGOp->HEADER.PPR);
871  AMIGOs->dsj = 0;
872  return(status & ERROR_MASK);
873 }
874 
875 
881 
882 int amigo_cmd_clear()
883 {
884 #if SDEBUG
886  printf("[AMIGO Clear]\n");
887 #endif
888  AMIGOs->sector = 0;
889  AMIGOs->head = 0;
890  AMIGOs->cyl = 0;
895  AMIGOs->dsj = 0;
896  AMIGOs->Errors =0;
897 
898  gpib_enable_PPR(AMIGOp->HEADER.PPR);
899  return(0);
900 }
901 
902 
909 
910 int amigo_todo_op(uint8_t secondary, uint8_t opcode, int len)
911 {
913  printf("[L Amigo TODO secondary: %02XH, state:%02XH, opcode:%02XH, len:%3d, listening:%02XH, talking:%02XH]\n",
914  secondary, AMIGOs->state, opcode, len, listening, talking);
915  else if(AMIGO_is_MTA(talking))
916  printf("[T Amigo TODO secondary: %02XH, state:%02XH, opcode:%02XH, len:%3d, listening:%02XH, talking:%02XH]\n",
917  secondary, AMIGOs->state, opcode, len, listening, talking);
918  else if(talking == UNT)
919  printf("[UNT Amigo TODO secondary: %02XH, state:%02XH, opcode:%02XH, len:%3d, listening:%02XH, talking:%02XH]\n",
920  secondary, AMIGOs->state, opcode, len, listening, talking);
921  else
922  printf("[U Amigo TODO secondary: %02XH, state:%02XH, opcode:%02XH, len:%3d, listening:%02XH, talking:%02XH]\n",
923  secondary, AMIGOs->state, opcode, len, listening, talking);
924  gpib_enable_PPR(AMIGOp->HEADER.PPR);
925  return(0);
926 }
927 
928 
934 
935 int amigo_todo(uint8_t secondary)
936 {
938  printf("[L Amigo TODO secondary: %02XH, state:%02XH, listening:%02XH, talking:%02XH]\n",
939  secondary,AMIGOs->state,listening,talking);
940  else if(AMIGO_is_MTA(talking))
941  printf("[T Amigo TODO secondary: %02XH, state:%02XH, listening:%02XH, talking:%02XH]\n",
942  secondary,AMIGOs->state,listening,talking);
943  else if(talking == UNT)
944  printf("[UNT Amigo TODO secondary: %02XH, state:%02XH, listening:%02XH, talking:%02XH]\n",
945  secondary,AMIGOs->state,listening,talking);
946  else
947  printf("[E Amigo ERROR secondary: %02XH, state:%02XH, listening:%02XH, talking:%02XH]\n",
948  secondary,AMIGOs->state,listening,talking);
949  gpib_enable_PPR(AMIGOp->HEADER.PPR);
950  return(0);
951 }
952 
953 
957 void amigo_check_unit(uint8_t unit)
958 {
959  if(unit != 15)
960  AMIGOs->unitNO = unit;
961  if(AMIGOs->unitNO != 0)
962  AMIGOs->Errors |= ERR_UNIT;
963 }
964 
965 
985 
986 int Amigo_Command( int secondary )
987 {
988  uint8_t op; // Current OP Code
989  uint8_t *ptr;
990  uint16_t status; // Current status
991  UINT len; // Size of Data/Op Codes/Parameters read in bytes
992 
993 #if SDEBUG
995  printf("[AMIGO Command(%02XH): listen:%02XH, talk:%02XH]\n",
997 #endif
998 
1000  if (secondary == 0x7e && AMIGO_is_MTA(talking))
1001  {
1002  gpib_disable_PPR(AMIGOp->HEADER.PPR);
1003  status = EOI_FLAG;
1004  len = gpib_write_str(gpib_iobuff, GPIB_IOBUFF_LEN, &status);
1005  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1006  if(status & ERROR_MASK)
1007  {
1008  AMIGOs->dsj = 1;
1009  AMIGOs->Errors |= ERR_GPIB;
1010  if(debuglevel & GPIB_ERR)
1011  printf("[AMIGO_Command:GPIB write error]\n");
1012  }
1013  return(status & ERROR_MASK);
1014  }
1015 
1017  if (secondary == 0x7f && AMIGO_is_MLA(listening))
1018  {
1019  gpib_disable_PPR(AMIGOp->HEADER.PPR);
1020 #if SDEBUG
1023 #endif
1024  status = EOI_FLAG;
1025  len = gpib_read_str(gpib_iobuff, GPIB_IOBUFF_LEN, &status);
1026 #if SDEBUG
1028  gpib_timer_elapsed_end("GPIB read");
1029 #endif
1030  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1031  if(status & ERROR_MASK)
1032  {
1033  AMIGOs->dsj = 1;
1034  AMIGOs->Errors |= ERR_GPIB;
1035  if(debuglevel & GPIB_ERR)
1036  printf("[AMIGO Command:GPIB read error]\n");
1037  }
1038  return(status & ERROR_MASK);
1039  }
1040 
1042  return(0);
1043 
1044  gpib_disable_PPR(AMIGOp->HEADER.PPR);
1045 
1048 
1049  status = EOI_FLAG;
1050  len = gpib_read_str(gpib_iobuff, GPIB_IOBUFF_LEN, &status);
1051  if(status & ERROR_MASK)
1052  {
1053  AMIGOs->dsj = 1;
1054  AMIGOs->Errors |= ERR_GPIB;
1055  if(debuglevel & GPIB_ERR)
1056  printf("[AMIGO Command:GPIB read error]\n");
1057  return(status & ERROR_MASK);
1058  }
1059 #if SDEBUG
1061  printf("[AMIGO Command(%02XH): GPIB read bytes:%02XH]\n",
1062  secondary, len);
1063 #endif
1064  if(!len)
1065  {
1066  AMIGOs->dsj = 1;
1067  AMIGOs->Errors |= ERR_GPIB;
1068  return(status & ERROR_MASK);
1069  }
1070 
1071  ptr = gpib_iobuff;
1072  op = *ptr++;
1073 
1074  if (secondary == 0x68)
1075  {
1076  if(op == 0x00 && len == 2)
1077  {
1079 
1080  AMIGOStateType tmp;
1081 #if SDEBUG
1083  printf("[AMIGO Cold Load Read Command]\n");
1084 #endif
1085  AMIGOs->unitNO = 0;
1087  AMIGOs->dsj = 0;
1088  AMIGOs->Errors = 0;
1090  tmp.cyl = 0;
1091  tmp.head = ( (0xff & *ptr) >> 6) & 0x03;
1092  tmp.sector = 0x3f & *ptr;
1093  ++ptr;
1094 //update to real address on sucess
1095  amigo_seek((AMIGOStateType *) &tmp);
1096  AMIGOs->state = AMIGO_COLD_LOAD_READ;
1097  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1098  return(status & ERROR_MASK);
1099  }
1100  else if(op == 0x02 && len == 5)
1101  {
1105 
1106  AMIGOStateType tmp;
1107 #if SDEBUG
1109  printf("[AMIGO Seek len=5]\n");
1110 #endif
1111  amigo_check_unit(0xff & *ptr++);
1114 
1116  tmp.cyl = 0xff & *ptr++;
1117  tmp.head = 0xff & *ptr++;
1118  tmp.sector = 0xff & *ptr++;
1119 //update to real address on sucess
1120  amigo_seek((AMIGOStateType *)&tmp);
1121  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1122  return(status & ERROR_MASK);
1123  }
1124  else if(op == 0x02 && len == 6)
1125  {
1129 
1130  AMIGOStateType tmp;
1131 #if SDEBUG
1133  printf("[AMIGO Seek len=6]\n");
1134 #endif
1135  amigo_check_unit(0xff & *ptr++);
1139  tmp.cyl = (0xff & *ptr++) << 8; // MSB
1140  tmp.cyl |= (0xff & *ptr++); // LSB
1141  tmp.head = 0xff & *ptr++;
1142  tmp.sector = 0xff & *ptr++;
1143 //update to real address on sucess
1144  amigo_seek((AMIGOStateType *)&tmp);
1145  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1146  return(status & ERROR_MASK);
1147  }
1148  else if(op == 0x03 && len == 2)
1149  {
1151 #if SDEBUG
1153  printf("[AMIGO Request Status Buffered Command]\n");
1154 #endif
1155  amigo_check_unit(0xff & *ptr++);
1159  AMIGOs->state = AMIGO_REQUEST_STATUS_BUFFERED;
1160  return(status & ERROR_MASK);
1161  }
1162  else if(op == 0x05 && len == 2)
1163  {
1165 #if SDEBUG
1167  printf("[AMIGO Read Unbuffered Command]\n");
1168 #endif
1169  amigo_check_unit(0xff & *ptr++);
1172  AMIGOs->state = AMIGO_READ_UNBUFFERED;
1173 
1174 #if 0
1175  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1176  return(status & ERROR_MASK);
1177 #else
1178  return( amigo_buffered_read_command() );
1179 #endif
1180  }
1181  else if(op == 0x07 && len == 4)
1182  {
1183  uint16_t sectors;
1184 #if SDEBUG
1186  printf("[AMIGO Verify]\n");
1187 #endif
1188  amigo_check_unit(0xff & *ptr++);
1191  sectors = (0xff & *ptr++) << 8;
1192  sectors |= (0xff & *ptr++);
1193  return ( amigo_verify( sectors) );
1194  }
1195  else if(op == 0x08 && len == 2)
1196  {
1197 #if SDEBUG
1199  printf("[AMIGO Write Unbuffered Command]\n");
1200 #endif
1201  amigo_check_unit(0xff & *ptr++);
1204  AMIGOs->state = AMIGO_WRITE_UNBUFFERED;
1205  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1206  return(status & ERROR_MASK);
1207  }
1208  else if((op == 0x0B || op == 0x2b) && len == 2)
1209  {
1210 #if SDEBUG
1212  printf("[AMIGO Initialize Command]\n");
1213 #endif
1214  amigo_check_unit(0xff & *ptr++);
1217  AMIGOs->state = AMIGO_INITIALIZE;
1218  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1219  return(status & ERROR_MASK);
1220  }
1221  else if(op == 0x14 && len == 2)
1222  {
1223 #if SDEBUG
1225  printf("[AMIGO Request Logical Address Command]\n");
1226 #endif
1228  AMIGOs->state = AMIGO_REQUEST_LOGICAL_ADDRESS;
1229  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1230  return(status & ERROR_MASK);
1231  }
1232  }
1233  else if (secondary == 0x69)
1234  {
1235  if(op == 0x08 && len == 2)
1236  {
1237 #if SDEBUG
1239  printf("[AMIGO Write Buffered Command]\n");
1240 #endif
1241  amigo_check_unit(0xff & *ptr++);
1244  AMIGOs->state = AMIGO_WRITE_BUFFERED;
1245  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1246  return(status & ERROR_MASK);
1247  }
1248  }
1249  else if (secondary == 0x6A)
1250  {
1251  if(op == 0x03 && len == 2)
1252  {
1253 #if SDEBUG
1255  printf("[AMIGO Request Status Unbuffered Command]\n");
1256 #endif
1257  amigo_check_unit(0xff & *ptr++);
1260  AMIGOs->state = AMIGO_REQUEST_STATUS_UNBUFFERED;
1262  return(status & ERROR_MASK);
1263  }
1264  if(op == 0x05 && len == 2)
1265  {
1266 #if SDEBUG
1268  printf("[AMIGO Read Buffered Command]\n");
1269 #endif
1270  amigo_check_unit(0xff & *ptr++);
1273  AMIGOs->state = AMIGO_READ_BUFFERED;
1274 
1275 #if 0
1276  gpib_enable_PPR(AMIGOp->HEADER.PPR);
1277  return(status & ERROR_MASK);
1278 #else
1279  return( amigo_buffered_read_command() );
1280 #endif
1281  }
1282  }
1283  else if (secondary == 0x6C)
1284  {
1285  if(op == 0x18 && len == 5)
1286  {
1288 
1289  uint8_t db;
1290 #if SDEBUG
1292  printf("[AMIGO Format]\n");
1293 #endif
1294  amigo_check_unit(0xff & *ptr++);
1297  ++ptr; // override not used
1298  ++ptr; // interleave not used
1299  db = 0xff & *ptr++;
1300  amigo_format(db);
1301  return(status & ERROR_MASK);
1302  }
1303  }
1304  else if (secondary == 0x70) // HP-300 Clear
1305  {
1307  ++ptr; // Dummy byte
1308  return(status & ERROR_MASK);
1309  }
1310  return ( amigo_todo_op(secondary, op, len) );
1311 }
1312 
1313 
1321 
1322 int Amigo_Execute( int secondary )
1323 {
1324 #if SDEBUG
1326  printf("[AMIGO Execute(%02XH): listen:%02XH, talk:%02XH]\n",
1328 #endif
1329 
1330  if(talking == UNT)
1331  return(0);
1332 
1334  return(0);
1335 
1336  if(secondary != 0x60 && secondary != 0x68)
1337  return(0);
1338 
1339  gpib_disable_PPR(AMIGOp->HEADER.PPR);
1340 
1341  if(secondary == 0x60)
1342  {
1343  switch(AMIGOs->state)
1344  {
1345  case AMIGO_IDLE:
1346  return(0);
1347  case AMIGO_COLD_LOAD_READ:
1348 #if SDEBUG
1350  printf("[AMIGO Execute Cold Load Read]\n");
1351 #endif
1352  return ( amigo_buffered_read_execute() );
1353  case AMIGO_READ_UNBUFFERED:
1354 #if SDEBUG
1356  printf("[AMIGO Execute Read Unbuffered]\n");
1357 #endif
1358  return ( amigo_buffered_read_execute() );
1359  case AMIGO_READ_BUFFERED:
1360 #if SDEBUG
1362  printf("[AMIGO Execute Read Buffered]\n");
1363 #endif
1364  return ( amigo_buffered_read_execute() );
1365  case AMIGO_WRITE_UNBUFFERED:
1366 #if SDEBUG
1368  printf("[AMIGO Execute Write Unbuffered]\n");
1369 #endif
1370  return ( amigo_buffered_write() );
1371  case AMIGO_INITIALIZE:
1372 #if SDEBUG
1374  printf("[AMIGO Execute Initialize]\n");
1375 #endif
1376  return ( amigo_buffered_write() );
1377  case AMIGO_WRITE_BUFFERED:
1378 #if SDEBUG
1380  printf("[AMIGO Execute Write Buffered]\n");
1381 #endif
1382  return ( amigo_buffered_write() );
1383  default:
1384  return ( amigo_todo(secondary) );
1385  }
1386  AMIGOs->state = AMIGO_IDLE;
1387  }
1388  if(secondary == 0x68)
1389  {
1390  switch(AMIGOs->state)
1391  {
1392  case AMIGO_IDLE:
1393  return(0);
1394  case AMIGO_REQUEST_STATUS_BUFFERED:
1395 #if SDEBUG
1397  printf("[AMIGO Execute Request Status Buffered]\n");
1398 #endif
1399  return ( amigo_send_status() );
1400  case AMIGO_REQUEST_STATUS_UNBUFFERED:
1401 #if SDEBUG
1403  printf("[AMIGO Exicute Request Status Unbuffered]\n");
1404 #endif
1405  return ( amigo_send_status() );
1406  case AMIGO_REQUEST_LOGICAL_ADDRESS:
1407 #if SDEBUG
1409  printf("[AMIGO Execute Request Logical Address]\n");
1410 #endif
1411  return ( amigo_send_logical_address() );
1412  default:
1413  return ( amigo_todo(secondary) );
1414  }
1415  AMIGOs->state = AMIGO_IDLE;
1416  }
1417  return(0);
1418 }
1419 
1420 
1433 
1434 int AMIGO_COMMANDS(uint8_t ch)
1435 {
1436 
1438  {
1439 
1440  if(talking == UNT && AMIGO_is_MLA(listening))
1441  {
1442 // printf("AMIGO COMMANDS %02XH NO TALK ADDRESS!\n", ch);
1443  }
1444  if(listening == 0 && AMIGO_is_MTA(talking))
1445  {
1446 // printf("AMIGO COMMANDS %02XH NO LISTEN ADDRESS!\n", ch);
1447  }
1448 
1449  if(ch == 0x60 && (AMIGO_is_MTA(talking) || AMIGO_is_MLA(listening)) )
1450  {
1451  return (Amigo_Execute(ch) );
1452  }
1453 
1454  if(ch == 0x68 && AMIGO_is_MTA(talking) )
1455  {
1456  return (Amigo_Execute(ch) );
1457  }
1458 
1459  if(ch == 0x68 && AMIGO_is_MLA(listening) )// Single byte command
1460  {
1461  return (Amigo_Command(ch) );
1462  }
1463  if(ch == 0x69 && AMIGO_is_MLA(listening) )// Single byte command
1464  {
1465  return (Amigo_Command(ch) );
1466  }
1467  if(ch == 0x6a && AMIGO_is_MLA(listening) )// Single byte command
1468  {
1469  return (Amigo_Command(ch) );
1470  }
1471  if(ch == 0x6c && AMIGO_is_MLA(listening) )// Single byte command
1472  {
1473  return (Amigo_Command(ch) );
1474  }
1475  if(ch == 0x70 && AMIGO_is_MTA(talking))
1476  {
1477  gpib_disable_PPR(AMIGOp->HEADER.PPR);
1478  return( amigo_cmd_dsj() );
1479  }
1480  if(ch == 0x7e && ch == 0x7f)
1481  {
1482  return (Amigo_Command(ch) );
1483  }
1484  if(ch == 0x70 && AMIGO_is_MLA(listening))
1485  {
1486 // NOP
1487  }
1488  }
1489  return(0);
1490 }
1491 #endif //ifdef AMIGO
GPIB_RW_STR_TIMING
#define GPIB_RW_STR_TIMING
Definition: debug.h:11
Amigo_Command
int Amigo_Command(int secondary)
stat
POSIX stat structure.
Definition: posix.h:105
amigo_todo_op
int amigo_todo_op(uint8_t secondary, uint8_t opcode, int len)
ERR_WRITE
#define ERR_WRITE
Definition: defines.h:81
printf
MEMSPACE int printf(const char *format,...)
AMIGOStateType::dsj
uint8_t dsj
Definition: drives.h:67
amigo_buffered_read_command
int amigo_buffered_read_command(void)
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
listening
uint8_t listening
gpib listen address
Definition: gpib.c:90
AMIGO_COMMANDS
int AMIGO_COMMANDS(uint8_t ch)
amigo_verify
int amigo_verify(uint16_t sectors)
gpib_disable_PPR
void gpib_disable_PPR(int bit)
Disable PPR (Parallel Poll Response) for a device.
Definition: gpib.c:436
amigo_seek
int amigo_seek(AMIGOStateType *p)
AMIGO_is_MLA
int AMIGO_is_MLA(int address)
amigo.h
AMIGO disk emulator for HP85 disk emulator project for AVR.
MAX_DEVICES
#define MAX_DEVICES
Maximum number of emulated devices.
Definition: drives.h:37
amigo_increment
int amigo_increment(char *msg)
Devices
DeviceType Devices[MAX_DEVICES]
Definition: drives.c:32
UINT
unsigned int UINT
Definition: ff.h:53
amigo_format
int amigo_format(uint8_t db)
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
ERR_DISK
#define ERR_DISK
Definition: defines.h:84
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
amigo_cmd_clear
int amigo_cmd_clear(void)
defines.h
GPIB, AMIGO, SS80 and device defines.
ERR_SEEK
#define ERR_SEEK
Definition: defines.h:82
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
amigo_init
void amigo_init(void)
AMIGO_is_MTA
int AMIGO_is_MTA(int address)
NULL
#define NULL
Definition: user_config.h:85
AMIGO_TYPE
@ AMIGO_TYPE
Definition: drives.h:249
GPIB_IOBUFF_LEN
#define GPIB_IOBUFF_LEN
Definition: gpib.h:101
gpib_iobuff
uint8_t gpib_iobuff[GPIB_IOBUFF_LEN]
common IO buffer for gpib_read_str() and gpib_write_str()
Definition: gpib.c:77
amigo_cmd_wakeup
int amigo_cmd_wakeup(void)
GPIB_ERR
#define GPIB_ERR
Definition: debug.h:4
gpib_task.h
High level GPIB command handler for HP85 disk emulator project for AVR.
amigo_buffered_read_execute
int amigo_buffered_read_execute(void)
amigo_buffered_read
int amigo_buffered_read(void)
ERR_UNIT
#define ERR_UNIT
Definition: defines.h:86
amigo_request_status
int amigo_request_status(void)
GPIB_DEVICE_STATE_MESSAGES
#define GPIB_DEVICE_STATE_MESSAGES
Definition: debug.h:9
amigo_check_unit
void amigo_check_unit(uint8_t unit)
AMIGOStateType::head
uint8_t head
Definition: drives.h:65
EOI_FLAG
#define EOI_FLAG
bus flags
Definition: gpib.h:33
GPIB_DISK_IO_TIMING
#define GPIB_DISK_IO_TIMING
Definition: debug.h:10
amigo_send_status
int amigo_send_status(void)
amigo_todo
int amigo_todo(uint8_t secondary)
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
secondary
uint8_t secondary
gpib secondary
Definition: gpib.c:100
UNT
#define UNT
Definition: amigo.h:28
AMIGOStateType
AMIGO emulator state machine index.
Definition: drives.h:57
AMIGOStateType::cyl
uint8_t cyl
Definition: drives.h:64
ERR_GPIB
#define ERR_GPIB
Definition: defines.h:85
gpib_enable_PPR
void gpib_enable_PPR(int bit)
Definition: gpib.c:416
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
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
talking
uint8_t talking
gpib talk address
Definition: gpib.c:85
gpib.h
GPIB emulator for HP85 disk emulator project for AVR.
AMIGOStateType::sector
uint8_t sector
Definition: drives.h:66
amigo_buffered_write
int amigo_buffered_write(void)
amigo_send_logical_address
int amigo_send_logical_address(void)
amigo_request_logical_address
int amigo_request_logical_address(void)
Amigo_Execute
int Amigo_Execute(int secondary)
DWORD
unsigned long DWORD
Definition: ff.h:56
gpib_hal.h
GPIB emulator hardwware layer for HP85 disk emulator project for AVR.
amigo_cmd_dsj
int amigo_cmd_dsj(void)
debug.h