3 FlashPoint.c -- FlashPoint SCCB Manager for Linux
5 This file contains the FlashPoint SCCB Manager from BusLogic's FlashPoint
6 Driver Developer's Kit, with minor modifications by Leonard N. Zubkoff for
7 Linux compatibility. It was provided by BusLogic in the form of 16 separate
8 source files, which would have unnecessarily cluttered the scsi directory, so
9 the individual files have been combined into this single file.
11 Copyright 1995-1996 by Mylex Corporation. All Rights Reserved
13 This file is available under both the GNU General Public License
14 and a BSD-style copyright; see LICENSE.FlashPoint for details.
19 #ifdef CONFIG_SCSI_FLASHPOINT
24 #define CRCMASK 0xA001
26 #define FAILURE 0xFFFFFFFFL
29 typedef void (*CALL_BK_FN) (struct sccb *);
31 struct sccb_mgr_info {
33 unsigned char si_present;
34 unsigned char si_intvect;
38 u16 si_per_targ_init_sync;
39 u16 si_per_targ_fast_nego;
40 u16 si_per_targ_ultra_nego;
41 u16 si_per_targ_no_disc;
42 u16 si_per_targ_wide_nego;
44 unsigned char si_card_family;
45 unsigned char si_bustype;
46 unsigned char si_card_model[3];
47 unsigned char si_relative_cardnum;
48 unsigned char si_reserved[4];
50 unsigned char si_XlatInfo[4];
52 u32 si_secondary_range;
55 #define SCSI_PARITY_ENA 0x0001
56 #define LOW_BYTE_TERM 0x0010
57 #define HIGH_BYTE_TERM 0x0020
58 #define BUSTYPE_PCI 0x3
60 #define SUPPORT_16TAR_32LUN 0x0002
61 #define SOFT_RESET 0x0004
62 #define EXTENDED_TRANSLATION 0x0008
63 #define POST_ALL_UNDERRRUNS 0x0040
64 #define FLAG_SCAM_ENABLED 0x0080
65 #define FLAG_SCAM_LEVEL2 0x0100
67 #define HARPOON_FAMILY 0x02
69 /* SCCB struct used for both SCCB and UCB manager compiles!
70 * The UCB Manager treats the SCCB as it's 'native hardware structure'
75 unsigned char OperationCode;
76 unsigned char ControlByte;
77 unsigned char CdbLength;
78 unsigned char RequestSenseLength;
81 unsigned char CcbRes[2];
82 unsigned char HostStatus;
83 unsigned char TargetStatus;
86 unsigned char Cdb[12];
87 unsigned char CcbRes1;
88 unsigned char Reserved1;
92 CALL_BK_FN SccbCallback; /* VOID (*SccbCallback)(); */
93 u32 SccbIOPort; /* Identifies board base port */
94 unsigned char SccbStatus;
95 unsigned char SCCBRes2;
98 u32 Sccb_XferCnt; /* actual transfer count */
100 u32 SccbVirtDataPtr; /* virtual addr for OS/2 */
104 unsigned char Sccb_scsimsg; /* identify msg for selection */
105 unsigned char Sccb_tag;
106 unsigned char Sccb_scsistat;
107 unsigned char Sccb_idmsg; /* image of last msg in */
108 struct sccb *Sccb_forwardlink;
109 struct sccb *Sccb_backlink;
111 unsigned char Save_Cdb[6];
112 unsigned char Save_CdbLen;
113 unsigned char Sccb_XferState;
119 #define SCATTER_GATHER_COMMAND 0x02
120 #define RESIDUAL_COMMAND 0x03
121 #define RESIDUAL_SG_COMMAND 0x04
122 #define RESET_COMMAND 0x81
124 #define F_USE_CMD_Q 0x20 /*Inidcates TAGGED command. */
125 #define TAG_TYPE_MASK 0xC0 /*Type of tag msg to send. */
126 #define SCCB_DATA_XFER_OUT 0x10 /* Write */
127 #define SCCB_DATA_XFER_IN 0x08 /* Read */
129 #define NO_AUTO_REQUEST_SENSE 0x01 /* No Request Sense Buffer */
131 #define BUS_FREE_ST 0
133 #define SELECT_BDR_ST 2 /* Select w\ Bus Device Reset */
134 #define SELECT_SN_ST 3 /* Select w\ Sync Nego */
135 #define SELECT_WN_ST 4 /* Select w\ Wide Data Nego */
136 #define SELECT_Q_ST 5 /* Select w\ Tagged Q'ing */
138 #define DATA_OUT_ST 7
140 #define DISCONNECT_ST 9
143 #define F_HOST_XFER_DIR 0x01
144 #define F_ALL_XFERRED 0x02
145 #define F_SG_XFER 0x04
146 #define F_AUTO_SENSE 0x08
147 #define F_ODD_BALL_CNT 0x10
148 #define F_NO_DATA_YET 0x80
150 #define F_STATUSLOADED 0x01
151 #define F_DEV_SELECTED 0x04
153 #define SCCB_COMPLETE 0x00 /* SCCB completed without error */
154 #define SCCB_DATA_UNDER_RUN 0x0C
155 #define SCCB_SELECTION_TIMEOUT 0x11 /* Set SCSI selection timed out */
156 #define SCCB_DATA_OVER_RUN 0x12
157 #define SCCB_PHASE_SEQUENCE_FAIL 0x14 /* Target bus phase sequence failure */
159 #define SCCB_GROSS_FW_ERR 0x27 /* Major problem! */
160 #define SCCB_BM_ERR 0x30 /* BusMaster error. */
161 #define SCCB_PARITY_ERR 0x34 /* SCSI parity error */
163 #define SCCB_IN_PROCESS 0x00
164 #define SCCB_SUCCESS 0x01
165 #define SCCB_ABORT 0x02
166 #define SCCB_ERROR 0x04
168 #define ORION_FW_REV 3110
170 #define QUEUE_DEPTH 254+1 /*1 for Normal disconnect 32 for Q'ing. */
172 #define MAX_MB_CARDS 4 /* Max. no of cards suppoerted on Mother Board */
174 #define MAX_SCSI_TAR 16
176 #define LUN_MASK 0x1f
178 #define SG_BUF_CNT 16 /*Number of prefetched elements. */
180 #define SG_ELEMENT_SIZE 8 /*Eight byte per element. */
182 #define RD_HARPOON(ioport) inb((u32)ioport)
183 #define RDW_HARPOON(ioport) inw((u32)ioport)
184 #define RD_HARP32(ioport,offset,data) (data = inl((u32)(ioport + offset)))
185 #define WR_HARPOON(ioport,val) outb((u8) val, (u32)ioport)
186 #define WRW_HARPOON(ioport,val) outw((u16)val, (u32)ioport)
187 #define WR_HARP32(ioport,offset,data) outl(data, (u32)(ioport + offset))
189 #define TAR_SYNC_MASK (BIT(7)+BIT(6))
190 #define SYNC_TRYING BIT(6)
191 #define SYNC_SUPPORTED (BIT(7)+BIT(6))
193 #define TAR_WIDE_MASK (BIT(5)+BIT(4))
194 #define WIDE_ENABLED BIT(4)
195 #define WIDE_NEGOCIATED BIT(5)
197 #define TAR_TAG_Q_MASK (BIT(3)+BIT(2))
198 #define TAG_Q_TRYING BIT(2)
199 #define TAG_Q_REJECT BIT(3)
201 #define TAR_ALLOW_DISC BIT(0)
203 #define EE_SYNC_MASK (BIT(0)+BIT(1))
204 #define EE_SYNC_5MB BIT(0)
205 #define EE_SYNC_10MB BIT(1)
206 #define EE_SYNC_20MB (BIT(0)+BIT(1))
208 #define EE_WIDE_SCSI BIT(7)
210 struct sccb_mgr_tar_info {
212 struct sccb *TarSelQ_Head;
213 struct sccb *TarSelQ_Tail;
214 unsigned char TarLUN_CA; /*Contingent Allgiance */
215 unsigned char TarTagQ_Cnt;
216 unsigned char TarSelQ_Cnt;
217 unsigned char TarStatus;
218 unsigned char TarEEValue;
219 unsigned char TarSyncCtrl;
220 unsigned char TarReserved[2]; /* for alignment */
221 unsigned char LunDiscQ_Idx[MAX_LUN];
222 unsigned char TarLUNBusy[MAX_LUN];
226 unsigned char niModel; /* Model No. of card */
227 unsigned char niCardNo; /* Card no. */
228 u32 niBaseAddr; /* Port Address of card */
229 unsigned char niSysConf; /* Adapter Configuration byte -
230 Byte 16 of eeprom map */
231 unsigned char niScsiConf; /* SCSI Configuration byte -
232 Byte 17 of eeprom map */
233 unsigned char niScamConf; /* SCAM Configuration byte -
234 Byte 20 of eeprom map */
235 unsigned char niAdapId; /* Host Adapter ID -
236 Byte 24 of eerpom map */
237 unsigned char niSyncTbl[MAX_SCSI_TAR / 2]; /* Sync/Wide byte
239 unsigned char niScamTbl[MAX_SCSI_TAR][4]; /* Compressed Scam name
249 struct sccb *currentSCCB;
250 struct sccb_mgr_info *cardInfo;
254 unsigned short cmdCounter;
255 unsigned char discQCount;
256 unsigned char tagQ_Lst;
257 unsigned char cardIndex;
258 unsigned char scanIndex;
259 unsigned char globalFlags;
261 struct nvram_info *pNvRamInfo;
262 struct sccb *discQ_Tbl[QUEUE_DEPTH];
266 #define F_TAG_STARTED 0x01
267 #define F_CONLUN_IO 0x02
268 #define F_DO_RENEGO 0x04
269 #define F_NO_FILTER 0x08
270 #define F_GREEN_PC 0x10
271 #define F_HOST_XFER_ACT 0x20
272 #define F_NEW_SCCB_CMD 0x40
273 #define F_UPDATE_EEPROM 0x80
275 #define ID_STRING_LENGTH 32
276 #define TYPE_CODE0 0x63 /*Level2 Mstr (bits 7-6), */
278 #define SLV_TYPE_CODE0 0xA3 /*Priority Bit set (bits 7-6), */
280 #define ASSIGN_ID 0x00
281 #define SET_P_FLAG 0x01
282 #define CFG_CMPLT 0x03
283 #define DOM_MSTR 0x0F
284 #define SYNC_PTRN 0x1F
288 #define MISC_CODE 0x14
289 #define CLR_P_FLAG 0x18
291 #define INIT_SELTD 0x01
292 #define LEVEL2_TAR 0x02
294 enum scam_id_st { ID0, ID1, ID2, ID3, ID4, ID5, ID6, ID7, ID8, ID9, ID10, ID11,
296 ID13, ID14, ID15, ID_UNUSED, ID_UNASSIGNED, ID_ASSIGNED, LEGACY,
297 CLR_PRIORITY, NO_ID_AVAIL
300 typedef struct SCCBscam_info {
302 unsigned char id_string[ID_STRING_LENGTH];
303 enum scam_id_st state;
309 #define DISC_PRIV 0x40
314 #define SIX_BYTE_CMD 0x06
315 #define TWELVE_BYTE_CMD 0x0C
318 #define MAX_OFFSET 0x0F /* Maxbyteoffset for Sync Xfers */
320 #define EEPROM_WD_CNT 256
322 #define EEPROM_CHECK_SUM 0
323 #define FW_SIGNATURE 2
324 #define MODEL_NUMB_0 4
325 #define MODEL_NUMB_2 6
326 #define MODEL_NUMB_4 8
327 #define SYSTEM_CONFIG 16
328 #define SCSI_CONFIG 17
329 #define BIOS_CONFIG 18
330 #define SCAM_CONFIG 20
331 #define ADAPTER_SCSI_ID 24
333 #define IGNORE_B_SCAN 32
334 #define SEND_START_ENA 34
335 #define DEVICE_ENABLE 36
337 #define SYNC_RATE_TBL 38
338 #define SYNC_RATE_TBL01 38
339 #define SYNC_RATE_TBL23 40
340 #define SYNC_RATE_TBL45 42
341 #define SYNC_RATE_TBL67 44
342 #define SYNC_RATE_TBL89 46
343 #define SYNC_RATE_TBLab 48
344 #define SYNC_RATE_TBLcd 50
345 #define SYNC_RATE_TBLef 52
347 #define EE_SCAMBASE 256
349 #define SCAM_ENABLED BIT(2)
350 #define SCAM_LEVEL2 BIT(3)
352 #define RENEGO_ENA BIT(10)
353 #define CONNIO_ENA BIT(11)
354 #define GREEN_PC_ENA BIT(12)
356 #define AUTO_RATE_00 00
357 #define AUTO_RATE_05 01
358 #define AUTO_RATE_10 02
359 #define AUTO_RATE_20 03
361 #define WIDE_NEGO_BIT BIT(7)
362 #define DISC_ENABLE_BIT BIT(6)
364 #define hp_vendor_id_0 0x00 /* LSB */
365 #define ORION_VEND_0 0x4B
367 #define hp_vendor_id_1 0x01 /* MSB */
368 #define ORION_VEND_1 0x10
370 #define hp_device_id_0 0x02 /* LSB */
371 #define ORION_DEV_0 0x30
373 #define hp_device_id_1 0x03 /* MSB */
374 #define ORION_DEV_1 0x81
376 /* Sub Vendor ID and Sub Device ID only available in
377 Harpoon Version 2 and higher */
379 #define hp_sub_device_id_0 0x06 /* LSB */
381 #define hp_semaphore 0x0C
382 #define SCCB_MGR_ACTIVE BIT(0)
383 #define TICKLE_ME BIT(1)
384 #define SCCB_MGR_PRESENT BIT(3)
385 #define BIOS_IN_USE BIT(4)
387 #define hp_sys_ctrl 0x0F
389 #define STOP_CLK BIT(0) /*Turn off BusMaster Clock */
390 #define DRVR_RST BIT(1) /*Firmware Reset to 80C15 chip */
391 #define HALT_MACH BIT(3) /*Halt State Machine */
392 #define HARD_ABORT BIT(4) /*Hard Abort */
394 #define hp_host_blk_cnt 0x13
396 #define XFER_BLK64 0x06 /* 1 1 0 64 byte per block */
398 #define BM_THRESHOLD 0x40 /* PCI mode can only xfer 16 bytes */
400 #define hp_int_mask 0x17
402 #define INT_CMD_COMPL BIT(0) /* DMA command complete */
403 #define INT_EXT_STATUS BIT(1) /* Extended Status Set */
405 #define hp_xfer_cnt_lo 0x18
406 #define hp_xfer_cnt_hi 0x1A
407 #define hp_xfer_cmd 0x1B
409 #define XFER_HOST_DMA 0x00 /* 0 0 0 Transfer Host -> DMA */
410 #define XFER_DMA_HOST 0x01 /* 0 0 1 Transfer DMA -> Host */
412 #define XFER_HOST_AUTO 0x00 /* 0 0 Auto Transfer Size */
414 #define XFER_DMA_8BIT 0x20 /* 0 1 8 BIT Transfer Size */
416 #define DISABLE_INT BIT(7) /*Do not interrupt at end of cmd. */
418 #define HOST_WRT_CMD ((DISABLE_INT + XFER_HOST_DMA + XFER_HOST_AUTO + XFER_DMA_8BIT))
419 #define HOST_RD_CMD ((DISABLE_INT + XFER_DMA_HOST + XFER_HOST_AUTO + XFER_DMA_8BIT))
421 #define hp_host_addr_lo 0x1C
422 #define hp_host_addr_hmi 0x1E
424 #define hp_ee_ctrl 0x22
426 #define EXT_ARB_ACK BIT(7)
427 #define SCSI_TERM_ENA_H BIT(6) /* SCSI high byte terminator */
428 #define SEE_MS BIT(5)
429 #define SEE_CS BIT(3)
430 #define SEE_CLK BIT(2)
431 #define SEE_DO BIT(1)
432 #define SEE_DI BIT(0)
435 #define EE_WRITE 0x05
437 #define EWEN_ADDR 0x03C0
439 #define EWDS_ADDR 0x0000
441 #define hp_bm_ctrl 0x26
443 #define SCSI_TERM_ENA_L BIT(0) /*Enable/Disable external terminators */
444 #define FLUSH_XFER_CNTR BIT(1) /*Flush transfer counter */
445 #define FORCE1_XFER BIT(5) /*Always xfer one byte in byte mode */
446 #define FAST_SINGLE BIT(6) /*?? */
448 #define BMCTRL_DEFAULT (FORCE1_XFER|FAST_SINGLE|SCSI_TERM_ENA_L)
450 #define hp_sg_addr 0x28
451 #define hp_page_ctrl 0x29
453 #define SCATTER_EN BIT(0)
454 #define SGRAM_ARAM BIT(1)
455 #define G_INT_DISABLE BIT(3) /* Enable/Disable all Interrupts */
456 #define NARROW_SCSI_CARD BIT(4) /* NARROW/WIDE SCSI config pin */
458 #define hp_pci_stat_cfg 0x2D
460 #define REC_MASTER_ABORT BIT(5) /*received Master abort */
462 #define hp_rev_num 0x33
464 #define hp_stack_data 0x34
465 #define hp_stack_addr 0x35
467 #define hp_ext_status 0x36
469 #define BM_FORCE_OFF BIT(0) /*Bus Master is forced to get off */
470 #define PCI_TGT_ABORT BIT(0) /*PCI bus master transaction aborted */
471 #define PCI_DEV_TMOUT BIT(1) /*PCI Device Time out */
472 #define CMD_ABORTED BIT(4) /*Command aborted */
473 #define BM_PARITY_ERR BIT(5) /*parity error on data received */
474 #define PIO_OVERRUN BIT(6) /*Slave data overrun */
475 #define BM_CMD_BUSY BIT(7) /*Bus master transfer command busy */
476 #define BAD_EXT_STATUS (BM_FORCE_OFF | PCI_DEV_TMOUT | CMD_ABORTED | \
477 BM_PARITY_ERR | PIO_OVERRUN)
479 #define hp_int_status 0x37
481 #define EXT_STATUS_ON BIT(1) /*Extended status is valid */
482 #define SCSI_INTERRUPT BIT(2) /*Global indication of a SCSI int. */
483 #define INT_ASSERTED BIT(5) /* */
485 #define hp_fifo_cnt 0x38
487 #define hp_intena 0x40
490 #define PROG_HLT BIT(6)
491 #define PARITY BIT(5)
494 #define SCAM_SEL BIT(2)
496 #define TIMEOUT BIT(0)
497 #define BUS_FREE BIT(15)
498 #define XFER_CNT_0 BIT(14)
499 #define PHASE BIT(13)
500 #define IUNKWN BIT(12)
501 #define ICMD_COMP BIT(11)
502 #define ITICKLE BIT(10)
503 #define IDO_STRT BIT(9)
504 #define ITAR_DISC BIT(8)
505 #define AUTO_INT (BIT(12)+BIT(11)+BIT(10)+BIT(9)+BIT(8))
506 #define CLR_ALL_INT 0xFFFF
507 #define CLR_ALL_INT_1 0xFF00
509 #define hp_intstat 0x42
511 #define hp_scsisig 0x44
513 #define SCSI_SEL BIT(7)
514 #define SCSI_BSY BIT(6)
515 #define SCSI_REQ BIT(5)
516 #define SCSI_ACK BIT(4)
517 #define SCSI_ATN BIT(3)
518 #define SCSI_CD BIT(2)
519 #define SCSI_MSG BIT(1)
520 #define SCSI_IOBIT BIT(0)
522 #define S_SCSI_PHZ (BIT(2)+BIT(1)+BIT(0))
523 #define S_MSGO_PH (BIT(2)+BIT(1) )
524 #define S_MSGI_PH (BIT(2)+BIT(1)+BIT(0))
525 #define S_DATAI_PH ( BIT(0))
526 #define S_DATAO_PH 0x00
527 #define S_ILL_PH ( BIT(1) )
529 #define hp_scsictrl_0 0x45
531 #define SEL_TAR BIT(6)
532 #define ENA_ATN BIT(4)
533 #define ENA_RESEL BIT(2)
534 #define SCSI_RST BIT(1)
535 #define ENA_SCAM_SEL BIT(0)
537 #define hp_portctrl_0 0x46
539 #define SCSI_PORT BIT(7)
540 #define SCSI_INBIT BIT(6)
541 #define DMA_PORT BIT(5)
542 #define DMA_RD BIT(4)
543 #define HOST_PORT BIT(3)
544 #define HOST_WRT BIT(2)
545 #define SCSI_BUS_EN BIT(1)
546 #define START_TO BIT(0)
548 #define hp_scsireset 0x47
550 #define SCSI_INI BIT(6)
551 #define SCAM_EN BIT(5)
552 #define DMA_RESET BIT(3)
553 #define HPSCSI_RESET BIT(2)
554 #define PROG_RESET BIT(1)
555 #define FIFO_CLR BIT(0)
557 #define hp_xfercnt_0 0x48
558 #define hp_xfercnt_2 0x4A
560 #define hp_fifodata_0 0x4C
561 #define hp_addstat 0x4E
563 #define SCAM_TIMER BIT(7)
564 #define SCSI_MODE8 BIT(3)
565 #define SCSI_PAR_ERR BIT(0)
567 #define hp_prgmcnt_0 0x4F
569 #define hp_selfid_0 0x50
570 #define hp_selfid_1 0x51
571 #define hp_arb_id 0x52
573 #define hp_select_id 0x53
575 #define hp_synctarg_base 0x54
576 #define hp_synctarg_12 0x54
577 #define hp_synctarg_13 0x55
578 #define hp_synctarg_14 0x56
579 #define hp_synctarg_15 0x57
581 #define hp_synctarg_8 0x58
582 #define hp_synctarg_9 0x59
583 #define hp_synctarg_10 0x5A
584 #define hp_synctarg_11 0x5B
586 #define hp_synctarg_4 0x5C
587 #define hp_synctarg_5 0x5D
588 #define hp_synctarg_6 0x5E
589 #define hp_synctarg_7 0x5F
591 #define hp_synctarg_0 0x60
592 #define hp_synctarg_1 0x61
593 #define hp_synctarg_2 0x62
594 #define hp_synctarg_3 0x63
596 #define NARROW_SCSI BIT(4)
597 #define DEFAULT_OFFSET 0x0F
599 #define hp_autostart_0 0x64
600 #define hp_autostart_1 0x65
601 #define hp_autostart_3 0x67
603 #define AUTO_IMMED BIT(5)
604 #define SELECT BIT(6)
605 #define END_DATA (BIT(7)+BIT(6))
607 #define hp_gp_reg_0 0x68
608 #define hp_gp_reg_1 0x69
609 #define hp_gp_reg_3 0x6B
611 #define hp_seltimeout 0x6C
613 #define TO_4ms 0x67 /* 3.9959ms */
615 #define TO_5ms 0x03 /* 4.9152ms */
616 #define TO_10ms 0x07 /* 11.xxxms */
617 #define TO_250ms 0x99 /* 250.68ms */
618 #define TO_290ms 0xB1 /* 289.99ms */
620 #define hp_clkctrl_0 0x6D
622 #define PWR_DWN BIT(6)
623 #define ACTdeassert BIT(4)
624 #define CLK_40MHZ (BIT(1) + BIT(0))
626 #define CLKCTRL_DEFAULT (ACTdeassert | CLK_40MHZ)
628 #define hp_fiforead 0x6E
629 #define hp_fifowrite 0x6F
631 #define hp_offsetctr 0x70
632 #define hp_xferstat 0x71
634 #define FIFO_EMPTY BIT(6)
636 #define hp_portctrl_1 0x72
638 #define CHK_SCSI_P BIT(3)
639 #define HOST_MODE8 BIT(0)
641 #define hp_xfer_pad 0x73
643 #define ID_UNLOCK BIT(3)
645 #define hp_scsidata_0 0x74
646 #define hp_scsidata_1 0x75
648 #define hp_aramBase 0x80
649 #define BIOS_DATA_OFFSET 0x60
650 #define BIOS_RELATIVE_CARD 0x64
652 #define AR3 (BIT(9) + BIT(8))
653 #define SDATA BIT(10)
655 #define CRD_OP BIT(11) /* Cmp Reg. w/ Data */
657 #define CRR_OP BIT(12) /* Cmp Reg. w. Reg. */
659 #define CPE_OP (BIT(14)+BIT(11)) /* Cmp SCSI phs & Branch EQ */
661 #define CPN_OP (BIT(14)+BIT(12)) /* Cmp SCSI phs & Branch NOT EQ */
663 #define ADATA_OUT 0x00
664 #define ADATA_IN BIT(8)
665 #define ACOMMAND BIT(10)
666 #define ASTATUS (BIT(10)+BIT(8))
667 #define AMSG_OUT (BIT(10)+BIT(9))
668 #define AMSG_IN (BIT(10)+BIT(9)+BIT(8))
670 #define BRH_OP BIT(13) /* Branch */
674 #define NOT_EQ BIT(9)
676 #define TCB_OP (BIT(13)+BIT(11)) /* Test condition & branch */
678 #define FIFO_0 BIT(10)
680 #define MPM_OP BIT(15) /* Match phase and move data */
682 #define MRR_OP BIT(14) /* Move DReg. to Reg. */
684 #define S_IDREG (BIT(2)+BIT(1)+BIT(0))
688 #define D_BUCKET (BIT(2) + BIT(1) + BIT(0))
690 #define RAT_OP (BIT(14)+BIT(13)+BIT(11))
692 #define SSI_OP (BIT(15)+BIT(11))
694 #define SSI_ITAR_DISC (ITAR_DISC >> 8)
695 #define SSI_IDO_STRT (IDO_STRT >> 8)
697 #define SSI_ICMD_COMP (ICMD_COMP >> 8)
698 #define SSI_ITICKLE (ITICKLE >> 8)
700 #define SSI_IUNKWN (IUNKWN >> 8)
701 #define SSI_INO_CC (IUNKWN >> 8)
702 #define SSI_IRFAIL (IUNKWN >> 8)
704 #define NP 0x10 /*Next Phase */
705 #define NTCMD 0x02 /*Non- Tagged Command start */
706 #define CMDPZ 0x04 /*Command phase */
707 #define DINT 0x12 /*Data Out/In interrupt */
708 #define DI 0x13 /*Data Out */
709 #define DC 0x19 /*Disconnect Message */
710 #define ST 0x1D /*Status Phase */
711 #define UNKNWN 0x24 /*Unknown bus action */
712 #define CC 0x25 /*Command Completion failure */
713 #define TICK 0x26 /*New target reselected us. */
714 #define SELCHK 0x28 /*Select & Check SCSI ID latch reg */
716 #define ID_MSG_STRT hp_aramBase + 0x00
717 #define NON_TAG_ID_MSG hp_aramBase + 0x06
718 #define CMD_STRT hp_aramBase + 0x08
719 #define SYNC_MSGS hp_aramBase + 0x08
721 #define TAG_STRT 0x00
722 #define DISCONNECT_START 0x10/2
723 #define END_DATA_START 0x14/2
724 #define CMD_ONLY_STRT CMDPZ/2
725 #define SELCHK_STRT SELCHK/2
727 #define GET_XFER_CNT(port, xfercnt) {RD_HARP32(port,hp_xfercnt_0,xfercnt); xfercnt &= 0xFFFFFF;}
728 /* #define GET_XFER_CNT(port, xfercnt) (xfercnt = RD_HARPOON(port+hp_xfercnt_2), \
730 xfercnt |= RDW_HARPOON((unsigned short)(port+hp_xfercnt_0)))
732 #define HP_SETUP_ADDR_CNT(port,addr,count) (WRW_HARPOON((port+hp_host_addr_lo), (unsigned short)(addr & 0x0000FFFFL)),\
734 WRW_HARPOON((port+hp_host_addr_hmi), (unsigned short)(addr & 0x0000FFFFL)),\
735 WR_HARP32(port,hp_xfercnt_0,count),\
736 WRW_HARPOON((port+hp_xfer_cnt_lo), (unsigned short)(count & 0x0000FFFFL)),\
738 WR_HARPOON(port+hp_xfer_cnt_hi, (count & 0xFF)))
740 #define ACCEPT_MSG(port) {while(RD_HARPOON(port+hp_scsisig) & SCSI_REQ){}\
741 WR_HARPOON(port+hp_scsisig, S_ILL_PH);}
743 #define ACCEPT_MSG_ATN(port) {while(RD_HARPOON(port+hp_scsisig) & SCSI_REQ){}\
744 WR_HARPOON(port+hp_scsisig, (S_ILL_PH|SCSI_ATN));}
746 #define DISABLE_AUTO(port) (WR_HARPOON(port+hp_scsireset, PROG_RESET),\
747 WR_HARPOON(port+hp_scsireset, 0x00))
749 #define ARAM_ACCESS(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
750 (RD_HARPOON(p_port+hp_page_ctrl) | SGRAM_ARAM)))
752 #define SGRAM_ACCESS(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
753 (RD_HARPOON(p_port+hp_page_ctrl) & ~SGRAM_ARAM)))
755 #define MDISABLE_INT(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
756 (RD_HARPOON(p_port+hp_page_ctrl) | G_INT_DISABLE)))
758 #define MENABLE_INT(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
759 (RD_HARPOON(p_port+hp_page_ctrl) & ~G_INT_DISABLE)))
761 static unsigned char FPT_sisyncn(u32 port, unsigned char p_card,
762 unsigned char syncFlag);
763 static void FPT_ssel(u32 port, unsigned char p_card);
764 static void FPT_sres(u32 port, unsigned char p_card,
765 struct sccb_card *pCurrCard);
766 static void FPT_shandem(u32 port, unsigned char p_card,
767 struct sccb *pCurrSCCB);
768 static void FPT_stsyncn(u32 port, unsigned char p_card);
769 static void FPT_sisyncr(u32 port, unsigned char sync_pulse,
770 unsigned char offset);
771 static void FPT_sssyncv(u32 p_port, unsigned char p_id,
772 unsigned char p_sync_value,
773 struct sccb_mgr_tar_info *currTar_Info);
774 static void FPT_sresb(u32 port, unsigned char p_card);
775 static void FPT_sxfrp(u32 p_port, unsigned char p_card);
776 static void FPT_schkdd(u32 port, unsigned char p_card);
777 static unsigned char FPT_RdStack(u32 port, unsigned char index);
778 static void FPT_WrStack(u32 portBase, unsigned char index,
780 static unsigned char FPT_ChkIfChipInitialized(u32 ioPort);
782 static void FPT_SendMsg(u32 port, unsigned char message);
783 static void FPT_queueFlushTargSccb(unsigned char p_card, unsigned char thisTarg,
784 unsigned char error_code);
786 static void FPT_sinits(struct sccb *p_sccb, unsigned char p_card);
787 static void FPT_RNVRamData(struct nvram_info *pNvRamInfo);
789 static unsigned char FPT_siwidn(u32 port, unsigned char p_card);
790 static void FPT_stwidn(u32 port, unsigned char p_card);
791 static void FPT_siwidr(u32 port, unsigned char width);
793 static void FPT_queueSelectFail(struct sccb_card *pCurrCard,
794 unsigned char p_card);
795 static void FPT_queueDisconnect(struct sccb *p_SCCB, unsigned char p_card);
796 static void FPT_queueCmdComplete(struct sccb_card *pCurrCard,
797 struct sccb *p_SCCB, unsigned char p_card);
798 static void FPT_queueSearchSelect(struct sccb_card *pCurrCard,
799 unsigned char p_card);
800 static void FPT_queueFlushSccb(unsigned char p_card, unsigned char error_code);
801 static void FPT_queueAddSccb(struct sccb *p_SCCB, unsigned char card);
802 static unsigned char FPT_queueFindSccb(struct sccb *p_SCCB,
803 unsigned char p_card);
804 static void FPT_utilUpdateResidual(struct sccb *p_SCCB);
805 static unsigned short FPT_CalcCrc16(unsigned char buffer[]);
806 static unsigned char FPT_CalcLrc(unsigned char buffer[]);
808 static void FPT_Wait1Second(u32 p_port);
809 static void FPT_Wait(u32 p_port, unsigned char p_delay);
810 static void FPT_utilEEWriteOnOff(u32 p_port, unsigned char p_mode);
811 static void FPT_utilEEWrite(u32 p_port, unsigned short ee_data,
812 unsigned short ee_addr);
813 static unsigned short FPT_utilEERead(u32 p_port,
814 unsigned short ee_addr);
815 static unsigned short FPT_utilEEReadOrg(u32 p_port,
816 unsigned short ee_addr);
817 static void FPT_utilEESendCmdAddr(u32 p_port, unsigned char ee_cmd,
818 unsigned short ee_addr);
820 static void FPT_phaseDataOut(u32 port, unsigned char p_card);
821 static void FPT_phaseDataIn(u32 port, unsigned char p_card);
822 static void FPT_phaseCommand(u32 port, unsigned char p_card);
823 static void FPT_phaseStatus(u32 port, unsigned char p_card);
824 static void FPT_phaseMsgOut(u32 port, unsigned char p_card);
825 static void FPT_phaseMsgIn(u32 port, unsigned char p_card);
826 static void FPT_phaseIllegal(u32 port, unsigned char p_card);
828 static void FPT_phaseDecode(u32 port, unsigned char p_card);
829 static void FPT_phaseChkFifo(u32 port, unsigned char p_card);
830 static void FPT_phaseBusFree(u32 p_port, unsigned char p_card);
832 static void FPT_XbowInit(u32 port, unsigned char scamFlg);
833 static void FPT_BusMasterInit(u32 p_port);
834 static void FPT_DiagEEPROM(u32 p_port);
836 static void FPT_dataXferProcessor(u32 port,
837 struct sccb_card *pCurrCard);
838 static void FPT_busMstrSGDataXferStart(u32 port,
839 struct sccb *pCurrSCCB);
840 static void FPT_busMstrDataXferStart(u32 port,
841 struct sccb *pCurrSCCB);
842 static void FPT_hostDataXferAbort(u32 port, unsigned char p_card,
843 struct sccb *pCurrSCCB);
844 static void FPT_hostDataXferRestart(struct sccb *currSCCB);
846 static unsigned char FPT_SccbMgr_bad_isr(u32 p_port,
847 unsigned char p_card,
848 struct sccb_card *pCurrCard,
849 unsigned short p_int);
851 static void FPT_SccbMgrTableInitAll(void);
852 static void FPT_SccbMgrTableInitCard(struct sccb_card *pCurrCard,
853 unsigned char p_card);
854 static void FPT_SccbMgrTableInitTarget(unsigned char p_card,
855 unsigned char target);
857 static void FPT_scini(unsigned char p_card, unsigned char p_our_id,
858 unsigned char p_power_up);
860 static int FPT_scarb(u32 p_port, unsigned char p_sel_type);
861 static void FPT_scbusf(u32 p_port);
862 static void FPT_scsel(u32 p_port);
863 static void FPT_scasid(unsigned char p_card, u32 p_port);
864 static unsigned char FPT_scxferc(u32 p_port, unsigned char p_data);
865 static unsigned char FPT_scsendi(u32 p_port,
866 unsigned char p_id_string[]);
867 static unsigned char FPT_sciso(u32 p_port,
868 unsigned char p_id_string[]);
869 static void FPT_scwirod(u32 p_port, unsigned char p_data_bit);
870 static void FPT_scwiros(u32 p_port, unsigned char p_data_bit);
871 static unsigned char FPT_scvalq(unsigned char p_quintet);
872 static unsigned char FPT_scsell(u32 p_port, unsigned char targ_id);
873 static void FPT_scwtsel(u32 p_port);
874 static void FPT_inisci(unsigned char p_card, u32 p_port,
875 unsigned char p_our_id);
876 static void FPT_scsavdi(unsigned char p_card, u32 p_port);
877 static unsigned char FPT_scmachid(unsigned char p_card,
878 unsigned char p_id_string[]);
880 static void FPT_autoCmdCmplt(u32 p_port, unsigned char p_card);
881 static void FPT_autoLoadDefaultMap(u32 p_port);
883 static struct sccb_mgr_tar_info FPT_sccbMgrTbl[MAX_CARDS][MAX_SCSI_TAR] =
885 static struct sccb_card FPT_BL_Card[MAX_CARDS] = { {0} };
886 static SCCBSCAM_INFO FPT_scamInfo[MAX_SCSI_TAR] = { {{0}} };
887 static struct nvram_info FPT_nvRamInfo[MAX_MB_CARDS] = { {0} };
889 static unsigned char FPT_mbCards = 0;
890 static unsigned char FPT_scamHAString[] =
891 { 0x63, 0x07, 'B', 'U', 'S', 'L', 'O', 'G', 'I', 'C',
892 ' ', 'B', 'T', '-', '9', '3', '0',
893 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
894 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20
897 static unsigned short FPT_default_intena = 0;
899 static void (*FPT_s_PhaseTbl[8]) (u32, unsigned char) = {
902 /*---------------------------------------------------------------------
904 * Function: FlashPoint_ProbeHostAdapter
906 * Description: Setup and/or Search for cards and return info to caller.
908 *---------------------------------------------------------------------*/
910 static int FlashPoint_ProbeHostAdapter(struct sccb_mgr_info *pCardInfo)
912 static unsigned char first_time = 1;
914 unsigned char i, j, id, ScamFlg;
915 unsigned short temp, temp2, temp3, temp4, temp5, temp6;
917 struct nvram_info *pCurrNvRam;
919 ioport = pCardInfo->si_baseaddr;
921 if (RD_HARPOON(ioport + hp_vendor_id_0) != ORION_VEND_0)
924 if ((RD_HARPOON(ioport + hp_vendor_id_1) != ORION_VEND_1))
927 if ((RD_HARPOON(ioport + hp_device_id_0) != ORION_DEV_0))
930 if ((RD_HARPOON(ioport + hp_device_id_1) != ORION_DEV_1))
933 if (RD_HARPOON(ioport + hp_rev_num) != 0x0f) {
935 /* For new Harpoon then check for sub_device ID LSB
936 the bits(0-3) must be all ZERO for compatible with
937 current version of SCCBMgr, else skip this Harpoon
940 if (RD_HARPOON(ioport + hp_sub_device_id_0) & 0x0f)
945 FPT_SccbMgrTableInitAll();
950 if (FPT_RdStack(ioport, 0) != 0x00) {
951 if (FPT_ChkIfChipInitialized(ioport) == 0) {
953 WR_HARPOON(ioport + hp_semaphore, 0x00);
954 FPT_XbowInit(ioport, 0); /*Must Init the SCSI before attempting */
955 FPT_DiagEEPROM(ioport);
957 if (FPT_mbCards < MAX_MB_CARDS) {
958 pCurrNvRam = &FPT_nvRamInfo[FPT_mbCards];
960 pCurrNvRam->niBaseAddr = ioport;
961 FPT_RNVRamData(pCurrNvRam);
968 WR_HARPOON(ioport + hp_clkctrl_0, CLKCTRL_DEFAULT);
969 WR_HARPOON(ioport + hp_sys_ctrl, 0x00);
972 pCardInfo->si_id = pCurrNvRam->niAdapId;
976 char)(FPT_utilEERead(ioport,
978 2)) & (unsigned char)0x0FF);
980 pCardInfo->si_lun = 0x00;
981 pCardInfo->si_fw_revision = ORION_FW_REV;
988 for (id = 0; id < (16 / 2); id++) {
991 temp = (unsigned short)pCurrNvRam->niSyncTbl[id];
992 temp = ((temp & 0x03) + ((temp << 4) & 0xc0)) +
993 (((temp << 4) & 0x0300) + ((temp << 8) & 0xc000));
996 FPT_utilEERead(ioport,
997 (unsigned short)((SYNC_RATE_TBL / 2)
1000 for (i = 0; i < 2; temp >>= 8, i++) {
1007 switch (temp & 0x3) {
1008 case AUTO_RATE_20: /* Synchronous, 20 mega-transfers/second */
1011 case AUTO_RATE_10: /* Synchronous, 10 mega-transfers/second */
1014 case AUTO_RATE_05: /* Synchronous, 5 mega-transfers/second */
1017 case AUTO_RATE_00: /* Asynchronous */
1021 if (temp & DISC_ENABLE_BIT)
1024 if (temp & WIDE_NEGO_BIT)
1030 pCardInfo->si_per_targ_init_sync = temp2;
1031 pCardInfo->si_per_targ_no_disc = temp3;
1032 pCardInfo->si_per_targ_wide_nego = temp4;
1033 pCardInfo->si_per_targ_fast_nego = temp5;
1034 pCardInfo->si_per_targ_ultra_nego = temp6;
1037 i = pCurrNvRam->niSysConf;
1040 char)(FPT_utilEERead(ioport, (SYSTEM_CONFIG / 2)));
1043 ScamFlg = pCurrNvRam->niScamConf;
1046 (unsigned char)FPT_utilEERead(ioport, SCAM_CONFIG / 2);
1048 pCardInfo->si_mflags = 0x0000;
1051 pCardInfo->si_mflags |= SCSI_PARITY_ENA;
1054 pCardInfo->si_mflags |= SOFT_RESET;
1057 pCardInfo->si_mflags |= EXTENDED_TRANSLATION;
1059 if (ScamFlg & SCAM_ENABLED)
1060 pCardInfo->si_mflags |= FLAG_SCAM_ENABLED;
1062 if (ScamFlg & SCAM_LEVEL2)
1063 pCardInfo->si_mflags |= FLAG_SCAM_LEVEL2;
1065 j = (RD_HARPOON(ioport + hp_bm_ctrl) & ~SCSI_TERM_ENA_L);
1067 j |= SCSI_TERM_ENA_L;
1069 WR_HARPOON(ioport + hp_bm_ctrl, j);
1071 j = (RD_HARPOON(ioport + hp_ee_ctrl) & ~SCSI_TERM_ENA_H);
1073 j |= SCSI_TERM_ENA_H;
1075 WR_HARPOON(ioport + hp_ee_ctrl, j);
1077 if (!(RD_HARPOON(ioport + hp_page_ctrl) & NARROW_SCSI_CARD))
1079 pCardInfo->si_mflags |= SUPPORT_16TAR_32LUN;
1081 pCardInfo->si_card_family = HARPOON_FAMILY;
1082 pCardInfo->si_bustype = BUSTYPE_PCI;
1085 pCardInfo->si_card_model[0] = '9';
1086 switch (pCurrNvRam->niModel & 0x0f) {
1088 pCardInfo->si_card_model[1] = '3';
1089 pCardInfo->si_card_model[2] = '0';
1092 pCardInfo->si_card_model[1] = '5';
1093 pCardInfo->si_card_model[2] = '0';
1096 pCardInfo->si_card_model[1] = '3';
1097 pCardInfo->si_card_model[2] = '2';
1100 pCardInfo->si_card_model[1] = '5';
1101 pCardInfo->si_card_model[2] = '2';
1105 temp = FPT_utilEERead(ioport, (MODEL_NUMB_0 / 2));
1106 pCardInfo->si_card_model[0] = (unsigned char)(temp >> 8);
1107 temp = FPT_utilEERead(ioport, (MODEL_NUMB_2 / 2));
1109 pCardInfo->si_card_model[1] = (unsigned char)(temp & 0x00FF);
1110 pCardInfo->si_card_model[2] = (unsigned char)(temp >> 8);
1113 if (pCardInfo->si_card_model[1] == '3') {
1114 if (RD_HARPOON(ioport + hp_ee_ctrl) & BIT(7))
1115 pCardInfo->si_mflags |= LOW_BYTE_TERM;
1116 } else if (pCardInfo->si_card_model[2] == '0') {
1117 temp = RD_HARPOON(ioport + hp_xfer_pad);
1118 WR_HARPOON(ioport + hp_xfer_pad, (temp & ~BIT(4)));
1119 if (RD_HARPOON(ioport + hp_ee_ctrl) & BIT(7))
1120 pCardInfo->si_mflags |= LOW_BYTE_TERM;
1121 WR_HARPOON(ioport + hp_xfer_pad, (temp | BIT(4)));
1122 if (RD_HARPOON(ioport + hp_ee_ctrl) & BIT(7))
1123 pCardInfo->si_mflags |= HIGH_BYTE_TERM;
1124 WR_HARPOON(ioport + hp_xfer_pad, temp);
1126 temp = RD_HARPOON(ioport + hp_ee_ctrl);
1127 temp2 = RD_HARPOON(ioport + hp_xfer_pad);
1128 WR_HARPOON(ioport + hp_ee_ctrl, (temp | SEE_CS));
1129 WR_HARPOON(ioport + hp_xfer_pad, (temp2 | BIT(4)));
1131 for (i = 0; i < 8; i++) {
1133 if (!(RD_HARPOON(ioport + hp_ee_ctrl) & BIT(7)))
1135 WR_HARPOON(ioport + hp_xfer_pad, (temp2 & ~BIT(4)));
1136 WR_HARPOON(ioport + hp_xfer_pad, (temp2 | BIT(4)));
1138 WR_HARPOON(ioport + hp_ee_ctrl, temp);
1139 WR_HARPOON(ioport + hp_xfer_pad, temp2);
1140 if (!(temp3 & BIT(7)))
1141 pCardInfo->si_mflags |= LOW_BYTE_TERM;
1142 if (!(temp3 & BIT(6)))
1143 pCardInfo->si_mflags |= HIGH_BYTE_TERM;
1146 ARAM_ACCESS(ioport);
1148 for (i = 0; i < 4; i++) {
1150 pCardInfo->si_XlatInfo[i] =
1151 RD_HARPOON(ioport + hp_aramBase + BIOS_DATA_OFFSET + i);
1154 /* return with -1 if no sort, else return with
1155 logical card number sorted by BIOS (zero-based) */
1157 pCardInfo->si_relative_cardnum =
1159 char)(RD_HARPOON(ioport + hp_aramBase + BIOS_RELATIVE_CARD) - 1);
1161 SGRAM_ACCESS(ioport);
1163 FPT_s_PhaseTbl[0] = FPT_phaseDataOut;
1164 FPT_s_PhaseTbl[1] = FPT_phaseDataIn;
1165 FPT_s_PhaseTbl[2] = FPT_phaseIllegal;
1166 FPT_s_PhaseTbl[3] = FPT_phaseIllegal;
1167 FPT_s_PhaseTbl[4] = FPT_phaseCommand;
1168 FPT_s_PhaseTbl[5] = FPT_phaseStatus;
1169 FPT_s_PhaseTbl[6] = FPT_phaseMsgOut;
1170 FPT_s_PhaseTbl[7] = FPT_phaseMsgIn;
1172 pCardInfo->si_present = 0x01;
1177 /*---------------------------------------------------------------------
1179 * Function: FlashPoint_HardwareResetHostAdapter
1181 * Description: Setup adapter for normal operation (hard reset).
1183 *---------------------------------------------------------------------*/
1185 static void *FlashPoint_HardwareResetHostAdapter(struct sccb_mgr_info
1188 struct sccb_card *CurrCard = NULL;
1189 struct nvram_info *pCurrNvRam;
1190 unsigned char i, j, thisCard, ScamFlg;
1191 unsigned short temp, sync_bit_map, id;
1194 ioport = pCardInfo->si_baseaddr;
1196 for (thisCard = 0; thisCard <= MAX_CARDS; thisCard++) {
1198 if (thisCard == MAX_CARDS)
1199 return (void *)FAILURE;
1201 if (FPT_BL_Card[thisCard].ioPort == ioport) {
1203 CurrCard = &FPT_BL_Card[thisCard];
1204 FPT_SccbMgrTableInitCard(CurrCard, thisCard);
1208 else if (FPT_BL_Card[thisCard].ioPort == 0x00) {
1210 FPT_BL_Card[thisCard].ioPort = ioport;
1211 CurrCard = &FPT_BL_Card[thisCard];
1214 for (i = 0; i < FPT_mbCards; i++) {
1215 if (CurrCard->ioPort ==
1216 FPT_nvRamInfo[i].niBaseAddr)
1217 CurrCard->pNvRamInfo =
1220 FPT_SccbMgrTableInitCard(CurrCard, thisCard);
1221 CurrCard->cardIndex = thisCard;
1222 CurrCard->cardInfo = pCardInfo;
1228 pCurrNvRam = CurrCard->pNvRamInfo;
1231 ScamFlg = pCurrNvRam->niScamConf;
1234 (unsigned char)FPT_utilEERead(ioport, SCAM_CONFIG / 2);
1237 FPT_BusMasterInit(ioport);
1238 FPT_XbowInit(ioport, ScamFlg);
1240 FPT_autoLoadDefaultMap(ioport);
1242 for (i = 0, id = 0x01; i != pCardInfo->si_id; i++, id <<= 1) {
1245 WR_HARPOON(ioport + hp_selfid_0, id);
1246 WR_HARPOON(ioport + hp_selfid_1, 0x00);
1247 WR_HARPOON(ioport + hp_arb_id, pCardInfo->si_id);
1248 CurrCard->ourId = pCardInfo->si_id;
1250 i = (unsigned char)pCardInfo->si_mflags;
1251 if (i & SCSI_PARITY_ENA)
1252 WR_HARPOON(ioport + hp_portctrl_1, (HOST_MODE8 | CHK_SCSI_P));
1254 j = (RD_HARPOON(ioport + hp_bm_ctrl) & ~SCSI_TERM_ENA_L);
1255 if (i & LOW_BYTE_TERM)
1256 j |= SCSI_TERM_ENA_L;
1257 WR_HARPOON(ioport + hp_bm_ctrl, j);
1259 j = (RD_HARPOON(ioport + hp_ee_ctrl) & ~SCSI_TERM_ENA_H);
1260 if (i & HIGH_BYTE_TERM)
1261 j |= SCSI_TERM_ENA_H;
1262 WR_HARPOON(ioport + hp_ee_ctrl, j);
1264 if (!(pCardInfo->si_mflags & SOFT_RESET)) {
1266 FPT_sresb(ioport, thisCard);
1268 FPT_scini(thisCard, pCardInfo->si_id, 0);
1271 if (pCardInfo->si_mflags & POST_ALL_UNDERRRUNS)
1272 CurrCard->globalFlags |= F_NO_FILTER;
1275 if (pCurrNvRam->niSysConf & 0x10)
1276 CurrCard->globalFlags |= F_GREEN_PC;
1278 if (FPT_utilEERead(ioport, (SYSTEM_CONFIG / 2)) & GREEN_PC_ENA)
1279 CurrCard->globalFlags |= F_GREEN_PC;
1282 /* Set global flag to indicate Re-Negotiation to be done on all
1285 if (pCurrNvRam->niScsiConf & 0x04)
1286 CurrCard->globalFlags |= F_DO_RENEGO;
1288 if (FPT_utilEERead(ioport, (SCSI_CONFIG / 2)) & RENEGO_ENA)
1289 CurrCard->globalFlags |= F_DO_RENEGO;
1293 if (pCurrNvRam->niScsiConf & 0x08)
1294 CurrCard->globalFlags |= F_CONLUN_IO;
1296 if (FPT_utilEERead(ioport, (SCSI_CONFIG / 2)) & CONNIO_ENA)
1297 CurrCard->globalFlags |= F_CONLUN_IO;
1300 temp = pCardInfo->si_per_targ_no_disc;
1302 for (i = 0, id = 1; i < MAX_SCSI_TAR; i++, id <<= 1) {
1305 FPT_sccbMgrTbl[thisCard][i].TarStatus |= TAR_ALLOW_DISC;
1308 sync_bit_map = 0x0001;
1310 for (id = 0; id < (MAX_SCSI_TAR / 2); id++) {
1313 temp = (unsigned short)pCurrNvRam->niSyncTbl[id];
1314 temp = ((temp & 0x03) + ((temp << 4) & 0xc0)) +
1315 (((temp << 4) & 0x0300) + ((temp << 8) & 0xc000));
1318 FPT_utilEERead(ioport,
1319 (unsigned short)((SYNC_RATE_TBL / 2)
1322 for (i = 0; i < 2; temp >>= 8, i++) {
1324 if (pCardInfo->si_per_targ_init_sync & sync_bit_map) {
1326 FPT_sccbMgrTbl[thisCard][id * 2 +
1328 (unsigned char)temp;
1332 FPT_sccbMgrTbl[thisCard][id * 2 +
1335 FPT_sccbMgrTbl[thisCard][id * 2 +
1337 (unsigned char)(temp & ~EE_SYNC_MASK);
1340 /* if ((pCardInfo->si_per_targ_wide_nego & sync_bit_map) ||
1343 if (pCardInfo->si_per_targ_wide_nego & sync_bit_map) {
1345 FPT_sccbMgrTbl[thisCard][id * 2 +
1351 else { /* NARROW SCSI */
1352 FPT_sccbMgrTbl[thisCard][id * 2 +
1362 WR_HARPOON((ioport + hp_semaphore),
1363 (unsigned char)(RD_HARPOON((ioport + hp_semaphore)) |
1366 return (void *)CurrCard;
1369 static void FlashPoint_ReleaseHostAdapter(void *pCurrCard)
1376 struct nvram_info *pCurrNvRam;
1378 pCurrNvRam = ((struct sccb_card *)pCurrCard)->pNvRamInfo;
1381 FPT_WrStack(pCurrNvRam->niBaseAddr, 0, pCurrNvRam->niModel);
1382 FPT_WrStack(pCurrNvRam->niBaseAddr, 1, pCurrNvRam->niSysConf);
1383 FPT_WrStack(pCurrNvRam->niBaseAddr, 2, pCurrNvRam->niScsiConf);
1384 FPT_WrStack(pCurrNvRam->niBaseAddr, 3, pCurrNvRam->niScamConf);
1385 FPT_WrStack(pCurrNvRam->niBaseAddr, 4, pCurrNvRam->niAdapId);
1387 for (i = 0; i < MAX_SCSI_TAR / 2; i++)
1388 FPT_WrStack(pCurrNvRam->niBaseAddr,
1389 (unsigned char)(i + 5),
1390 pCurrNvRam->niSyncTbl[i]);
1392 portBase = pCurrNvRam->niBaseAddr;
1394 for (i = 0; i < MAX_SCSI_TAR; i++) {
1395 regOffset = hp_aramBase + 64 + i * 4;
1396 pScamTbl = (u32 *)&pCurrNvRam->niScamTbl[i];
1397 scamData = *pScamTbl;
1398 WR_HARP32(portBase, regOffset, scamData);
1402 FPT_WrStack(((struct sccb_card *)pCurrCard)->ioPort, 0, 0);
1406 static void FPT_RNVRamData(struct nvram_info *pNvRamInfo)
1414 pNvRamInfo->niModel = FPT_RdStack(pNvRamInfo->niBaseAddr, 0);
1415 pNvRamInfo->niSysConf = FPT_RdStack(pNvRamInfo->niBaseAddr, 1);
1416 pNvRamInfo->niScsiConf = FPT_RdStack(pNvRamInfo->niBaseAddr, 2);
1417 pNvRamInfo->niScamConf = FPT_RdStack(pNvRamInfo->niBaseAddr, 3);
1418 pNvRamInfo->niAdapId = FPT_RdStack(pNvRamInfo->niBaseAddr, 4);
1420 for (i = 0; i < MAX_SCSI_TAR / 2; i++)
1421 pNvRamInfo->niSyncTbl[i] =
1422 FPT_RdStack(pNvRamInfo->niBaseAddr, (unsigned char)(i + 5));
1424 portBase = pNvRamInfo->niBaseAddr;
1426 for (i = 0; i < MAX_SCSI_TAR; i++) {
1427 regOffset = hp_aramBase + 64 + i * 4;
1428 RD_HARP32(portBase, regOffset, scamData);
1429 pScamTbl = (u32 *)&pNvRamInfo->niScamTbl[i];
1430 *pScamTbl = scamData;
1435 static unsigned char FPT_RdStack(u32 portBase, unsigned char index)
1437 WR_HARPOON(portBase + hp_stack_addr, index);
1438 return RD_HARPOON(portBase + hp_stack_data);
1441 static void FPT_WrStack(u32 portBase, unsigned char index, unsigned char data)
1443 WR_HARPOON(portBase + hp_stack_addr, index);
1444 WR_HARPOON(portBase + hp_stack_data, data);
1447 static unsigned char FPT_ChkIfChipInitialized(u32 ioPort)
1449 if ((RD_HARPOON(ioPort + hp_arb_id) & 0x0f) != FPT_RdStack(ioPort, 4))
1451 if ((RD_HARPOON(ioPort + hp_clkctrl_0) & CLKCTRL_DEFAULT)
1454 if ((RD_HARPOON(ioPort + hp_seltimeout) == TO_250ms) ||
1455 (RD_HARPOON(ioPort + hp_seltimeout) == TO_290ms))
1461 /*---------------------------------------------------------------------
1463 * Function: FlashPoint_StartCCB
1465 * Description: Start a command pointed to by p_Sccb. When the
1466 * command is completed it will be returned via the
1467 * callback function.
1469 *---------------------------------------------------------------------*/
1470 static void FlashPoint_StartCCB(void *curr_card, struct sccb *p_Sccb)
1473 unsigned char thisCard, lun;
1474 struct sccb *pSaveSccb;
1475 CALL_BK_FN callback;
1476 struct sccb_card *pCurrCard = curr_card;
1478 thisCard = pCurrCard->cardIndex;
1479 ioport = pCurrCard->ioPort;
1481 if ((p_Sccb->TargID >= MAX_SCSI_TAR) || (p_Sccb->Lun >= MAX_LUN)) {
1483 p_Sccb->HostStatus = SCCB_COMPLETE;
1484 p_Sccb->SccbStatus = SCCB_ERROR;
1485 callback = (CALL_BK_FN) p_Sccb->SccbCallback;
1492 FPT_sinits(p_Sccb, thisCard);
1494 if (!pCurrCard->cmdCounter) {
1495 WR_HARPOON(ioport + hp_semaphore,
1496 (RD_HARPOON(ioport + hp_semaphore)
1497 | SCCB_MGR_ACTIVE));
1499 if (pCurrCard->globalFlags & F_GREEN_PC) {
1500 WR_HARPOON(ioport + hp_clkctrl_0, CLKCTRL_DEFAULT);
1501 WR_HARPOON(ioport + hp_sys_ctrl, 0x00);
1505 pCurrCard->cmdCounter++;
1507 if (RD_HARPOON(ioport + hp_semaphore) & BIOS_IN_USE) {
1509 WR_HARPOON(ioport + hp_semaphore,
1510 (RD_HARPOON(ioport + hp_semaphore)
1512 if (p_Sccb->OperationCode == RESET_COMMAND) {
1514 pCurrCard->currentSCCB;
1515 pCurrCard->currentSCCB = p_Sccb;
1516 FPT_queueSelectFail(&FPT_BL_Card[thisCard], thisCard);
1517 pCurrCard->currentSCCB =
1520 FPT_queueAddSccb(p_Sccb, thisCard);
1524 else if ((RD_HARPOON(ioport + hp_page_ctrl) & G_INT_DISABLE)) {
1526 if (p_Sccb->OperationCode == RESET_COMMAND) {
1528 pCurrCard->currentSCCB;
1529 pCurrCard->currentSCCB = p_Sccb;
1530 FPT_queueSelectFail(&FPT_BL_Card[thisCard], thisCard);
1531 pCurrCard->currentSCCB =
1534 FPT_queueAddSccb(p_Sccb, thisCard);
1540 MDISABLE_INT(ioport);
1542 if ((pCurrCard->globalFlags & F_CONLUN_IO) &&
1543 ((FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].
1544 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
1548 if ((pCurrCard->currentSCCB == NULL) &&
1549 (FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].TarSelQ_Cnt == 0)
1550 && (FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].TarLUNBusy[lun]
1553 pCurrCard->currentSCCB = p_Sccb;
1554 FPT_ssel(p_Sccb->SccbIOPort, thisCard);
1559 if (p_Sccb->OperationCode == RESET_COMMAND) {
1560 pSaveSccb = pCurrCard->currentSCCB;
1561 pCurrCard->currentSCCB = p_Sccb;
1562 FPT_queueSelectFail(&FPT_BL_Card[thisCard],
1564 pCurrCard->currentSCCB = pSaveSccb;
1566 FPT_queueAddSccb(p_Sccb, thisCard);
1570 MENABLE_INT(ioport);
1575 /*---------------------------------------------------------------------
1577 * Function: FlashPoint_AbortCCB
1579 * Description: Abort the command pointed to by p_Sccb. When the
1580 * command is completed it will be returned via the
1581 * callback function.
1583 *---------------------------------------------------------------------*/
1584 static int FlashPoint_AbortCCB(void *pCurrCard, struct sccb *p_Sccb)
1588 unsigned char thisCard;
1589 CALL_BK_FN callback;
1590 struct sccb *pSaveSCCB;
1591 struct sccb_mgr_tar_info *currTar_Info;
1593 ioport = ((struct sccb_card *)pCurrCard)->ioPort;
1595 thisCard = ((struct sccb_card *)pCurrCard)->cardIndex;
1597 if (!(RD_HARPOON(ioport + hp_page_ctrl) & G_INT_DISABLE)) {
1599 if (FPT_queueFindSccb(p_Sccb, thisCard)) {
1601 ((struct sccb_card *)pCurrCard)->cmdCounter--;
1603 if (!((struct sccb_card *)pCurrCard)->cmdCounter)
1604 WR_HARPOON(ioport + hp_semaphore,
1605 (RD_HARPOON(ioport + hp_semaphore)
1607 char)(~(SCCB_MGR_ACTIVE |
1610 p_Sccb->SccbStatus = SCCB_ABORT;
1611 callback = p_Sccb->SccbCallback;
1618 if (((struct sccb_card *)pCurrCard)->currentSCCB ==
1620 p_Sccb->SccbStatus = SCCB_ABORT;
1626 if (p_Sccb->Sccb_tag) {
1627 MDISABLE_INT(ioport);
1628 if (((struct sccb_card *)pCurrCard)->
1629 discQ_Tbl[p_Sccb->Sccb_tag] ==
1631 p_Sccb->SccbStatus = SCCB_ABORT;
1632 p_Sccb->Sccb_scsistat =
1634 p_Sccb->Sccb_scsimsg =
1637 if (((struct sccb_card *)
1638 pCurrCard)->currentSCCB ==
1640 ((struct sccb_card *)
1642 currentSCCB = p_Sccb;
1650 ((struct sccb_card *)
1652 currentSCCB = p_Sccb;
1653 FPT_queueSelectFail((struct sccb_card *)pCurrCard, thisCard);
1654 ((struct sccb_card *)
1656 currentSCCB = pSaveSCCB;
1659 MENABLE_INT(ioport);
1663 &FPT_sccbMgrTbl[thisCard][p_Sccb->
1666 if (FPT_BL_Card[thisCard].
1667 discQ_Tbl[currTar_Info->
1668 LunDiscQ_Idx[p_Sccb->Lun]]
1670 p_Sccb->SccbStatus = SCCB_ABORT;
1680 /*---------------------------------------------------------------------
1682 * Function: FlashPoint_InterruptPending
1684 * Description: Do a quick check to determine if there is a pending
1685 * interrupt for this card and disable the IRQ Pin if so.
1687 *---------------------------------------------------------------------*/
1688 static unsigned char FlashPoint_InterruptPending(void *pCurrCard)
1692 ioport = ((struct sccb_card *)pCurrCard)->ioPort;
1694 if (RD_HARPOON(ioport + hp_int_status) & INT_ASSERTED) {
1703 /*---------------------------------------------------------------------
1705 * Function: FlashPoint_HandleInterrupt
1707 * Description: This is our entry point when an interrupt is generated
1708 * by the card and the upper level driver passes it on to
1711 *---------------------------------------------------------------------*/
1712 static int FlashPoint_HandleInterrupt(void *pcard)
1714 struct sccb *currSCCB;
1715 unsigned char thisCard, result, bm_status, bm_int_st;
1716 unsigned short hp_int;
1717 unsigned char i, target;
1718 struct sccb_card *pCurrCard = pcard;
1721 thisCard = pCurrCard->cardIndex;
1722 ioport = pCurrCard->ioPort;
1724 MDISABLE_INT(ioport);
1726 if ((bm_int_st = RD_HARPOON(ioport + hp_int_status)) & EXT_STATUS_ON)
1727 bm_status = RD_HARPOON(ioport + hp_ext_status) &
1728 (unsigned char)BAD_EXT_STATUS;
1732 WR_HARPOON(ioport + hp_int_mask, (INT_CMD_COMPL | SCSI_INTERRUPT));
1734 while ((hp_int = RDW_HARPOON((ioport + hp_intstat)) &
1735 FPT_default_intena) | bm_status) {
1737 currSCCB = pCurrCard->currentSCCB;
1739 if (hp_int & (FIFO | TIMEOUT | RESET | SCAM_SEL) || bm_status) {
1741 FPT_SccbMgr_bad_isr(ioport, thisCard, pCurrCard,
1743 WRW_HARPOON((ioport + hp_intstat),
1744 (FIFO | TIMEOUT | RESET | SCAM_SEL));
1749 MENABLE_INT(ioport);
1754 else if (hp_int & ICMD_COMP) {
1756 if (!(hp_int & BUS_FREE)) {
1757 /* Wait for the BusFree before starting a new command. We
1758 must also check for being reselected since the BusFree
1759 may not show up if another device reselects us in 1.5us or
1760 less. SRR Wednesday, 3/8/1995.
1763 (RDW_HARPOON((ioport + hp_intstat)) &
1764 (BUS_FREE | RSEL))) ;
1767 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1769 FPT_phaseChkFifo(ioport, thisCard);
1771 /* WRW_HARPOON((ioport+hp_intstat),
1772 (BUS_FREE | ICMD_COMP | ITAR_DISC | XFER_CNT_0));
1775 WRW_HARPOON((ioport + hp_intstat), CLR_ALL_INT_1);
1777 FPT_autoCmdCmplt(ioport, thisCard);
1781 else if (hp_int & ITAR_DISC) {
1783 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1784 FPT_phaseChkFifo(ioport, thisCard);
1786 if (RD_HARPOON(ioport + hp_gp_reg_1) ==
1789 WR_HARPOON(ioport + hp_gp_reg_1, 0x00);
1790 currSCCB->Sccb_XferState |= F_NO_DATA_YET;
1792 currSCCB->Sccb_savedATC = currSCCB->Sccb_ATC;
1795 currSCCB->Sccb_scsistat = DISCONNECT_ST;
1796 FPT_queueDisconnect(currSCCB, thisCard);
1798 /* Wait for the BusFree before starting a new command. We
1799 must also check for being reselected since the BusFree
1800 may not show up if another device reselects us in 1.5us or
1801 less. SRR Wednesday, 3/8/1995.
1804 (RDW_HARPOON((ioport + hp_intstat)) &
1806 && !((RDW_HARPOON((ioport + hp_intstat)) & PHASE)
1807 && RD_HARPOON((ioport + hp_scsisig)) ==
1808 (SCSI_BSY | SCSI_REQ | SCSI_CD | SCSI_MSG |
1812 The additional loop exit condition above detects a timing problem
1813 with the revision D/E harpoon chips. The caller should reset the
1814 host adapter to recover when 0xFE is returned.
1817 (RDW_HARPOON((ioport + hp_intstat)) &
1818 (BUS_FREE | RSEL))) {
1819 MENABLE_INT(ioport);
1823 WRW_HARPOON((ioport + hp_intstat),
1824 (BUS_FREE | ITAR_DISC));
1826 pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
1830 else if (hp_int & RSEL) {
1832 WRW_HARPOON((ioport + hp_intstat),
1833 (PROG_HLT | RSEL | PHASE | BUS_FREE));
1835 if (RDW_HARPOON((ioport + hp_intstat)) & ITAR_DISC) {
1836 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1837 FPT_phaseChkFifo(ioport, thisCard);
1839 if (RD_HARPOON(ioport + hp_gp_reg_1) ==
1841 WR_HARPOON(ioport + hp_gp_reg_1, 0x00);
1842 currSCCB->Sccb_XferState |=
1844 currSCCB->Sccb_savedATC =
1848 WRW_HARPOON((ioport + hp_intstat),
1849 (BUS_FREE | ITAR_DISC));
1850 currSCCB->Sccb_scsistat = DISCONNECT_ST;
1851 FPT_queueDisconnect(currSCCB, thisCard);
1854 FPT_sres(ioport, thisCard, pCurrCard);
1855 FPT_phaseDecode(ioport, thisCard);
1859 else if ((hp_int & IDO_STRT) && (!(hp_int & BUS_FREE))) {
1861 WRW_HARPOON((ioport + hp_intstat),
1862 (IDO_STRT | XFER_CNT_0));
1863 FPT_phaseDecode(ioport, thisCard);
1867 else if ((hp_int & IUNKWN) || (hp_int & PROG_HLT)) {
1868 WRW_HARPOON((ioport + hp_intstat),
1869 (PHASE | IUNKWN | PROG_HLT));
1870 if ((RD_HARPOON(ioport + hp_prgmcnt_0) & (unsigned char)
1871 0x3f) < (unsigned char)SELCHK) {
1872 FPT_phaseDecode(ioport, thisCard);
1874 /* Harpoon problem some SCSI target device respond to selection
1875 with short BUSY pulse (<400ns) this will make the Harpoon is not able
1876 to latch the correct Target ID into reg. x53.
1877 The work around require to correct this reg. But when write to this
1878 reg. (0x53) also increment the FIFO write addr reg (0x6f), thus we
1879 need to read this reg first then restore it later. After update to 0x53 */
1882 char)(RD_HARPOON(ioport + hp_fifowrite));
1885 char)(RD_HARPOON(ioport + hp_gp_reg_3));
1886 WR_HARPOON(ioport + hp_xfer_pad,
1887 (unsigned char)ID_UNLOCK);
1888 WR_HARPOON(ioport + hp_select_id,
1889 (unsigned char)(target | target <<
1891 WR_HARPOON(ioport + hp_xfer_pad,
1892 (unsigned char)0x00);
1893 WR_HARPOON(ioport + hp_fifowrite, i);
1894 WR_HARPOON(ioport + hp_autostart_3,
1895 (AUTO_IMMED + TAG_STRT));
1899 else if (hp_int & XFER_CNT_0) {
1901 WRW_HARPOON((ioport + hp_intstat), XFER_CNT_0);
1903 FPT_schkdd(ioport, thisCard);
1907 else if (hp_int & BUS_FREE) {
1909 WRW_HARPOON((ioport + hp_intstat), BUS_FREE);
1911 if (pCurrCard->globalFlags & F_HOST_XFER_ACT) {
1913 FPT_hostDataXferAbort(ioport, thisCard,
1917 FPT_phaseBusFree(ioport, thisCard);
1920 else if (hp_int & ITICKLE) {
1922 WRW_HARPOON((ioport + hp_intstat), ITICKLE);
1923 pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
1926 if (((struct sccb_card *)pCurrCard)->
1927 globalFlags & F_NEW_SCCB_CMD) {
1929 pCurrCard->globalFlags &= ~F_NEW_SCCB_CMD;
1931 if (pCurrCard->currentSCCB == NULL)
1932 FPT_queueSearchSelect(pCurrCard, thisCard);
1934 if (pCurrCard->currentSCCB != NULL) {
1935 pCurrCard->globalFlags &= ~F_NEW_SCCB_CMD;
1936 FPT_ssel(ioport, thisCard);
1945 MENABLE_INT(ioport);
1950 /*---------------------------------------------------------------------
1952 * Function: Sccb_bad_isr
1954 * Description: Some type of interrupt has occurred which is slightly
1955 * out of the ordinary. We will now decode it fully, in
1956 * this routine. This is broken up in an attempt to save
1959 *---------------------------------------------------------------------*/
1960 static unsigned char FPT_SccbMgr_bad_isr(u32 p_port, unsigned char p_card,
1961 struct sccb_card *pCurrCard,
1962 unsigned short p_int)
1964 unsigned char temp, ScamFlg;
1965 struct sccb_mgr_tar_info *currTar_Info;
1966 struct nvram_info *pCurrNvRam;
1968 if (RD_HARPOON(p_port + hp_ext_status) &
1969 (BM_FORCE_OFF | PCI_DEV_TMOUT | BM_PARITY_ERR | PIO_OVERRUN)) {
1971 if (pCurrCard->globalFlags & F_HOST_XFER_ACT) {
1973 FPT_hostDataXferAbort(p_port, p_card,
1974 pCurrCard->currentSCCB);
1977 if (RD_HARPOON(p_port + hp_pci_stat_cfg) & REC_MASTER_ABORT)
1979 WR_HARPOON(p_port + hp_pci_stat_cfg,
1980 (RD_HARPOON(p_port + hp_pci_stat_cfg) &
1981 ~REC_MASTER_ABORT));
1983 WR_HARPOON(p_port + hp_host_blk_cnt, 0x00);
1987 if (pCurrCard->currentSCCB != NULL) {
1989 if (!pCurrCard->currentSCCB->HostStatus)
1990 pCurrCard->currentSCCB->HostStatus =
1993 FPT_sxfrp(p_port, p_card);
1995 temp = (unsigned char)(RD_HARPOON(p_port + hp_ee_ctrl) &
1996 (EXT_ARB_ACK | SCSI_TERM_ENA_H));
1997 WR_HARPOON(p_port + hp_ee_ctrl,
1998 ((unsigned char)temp | SEE_MS | SEE_CS));
1999 WR_HARPOON(p_port + hp_ee_ctrl, temp);
2002 (RDW_HARPOON((p_port + hp_intstat)) &
2003 (BUS_FREE | RESET))) {
2004 FPT_phaseDecode(p_port, p_card);
2009 else if (p_int & RESET) {
2011 WR_HARPOON(p_port + hp_clkctrl_0, CLKCTRL_DEFAULT);
2012 WR_HARPOON(p_port + hp_sys_ctrl, 0x00);
2013 if (pCurrCard->currentSCCB != NULL) {
2015 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
2017 FPT_hostDataXferAbort(p_port, p_card,
2018 pCurrCard->currentSCCB);
2021 DISABLE_AUTO(p_port);
2023 FPT_sresb(p_port, p_card);
2025 while (RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST) {
2028 pCurrNvRam = pCurrCard->pNvRamInfo;
2030 ScamFlg = pCurrNvRam->niScamConf;
2033 (unsigned char)FPT_utilEERead(p_port,
2037 FPT_XbowInit(p_port, ScamFlg);
2039 FPT_scini(p_card, pCurrCard->ourId, 0);
2044 else if (p_int & FIFO) {
2046 WRW_HARPOON((p_port + hp_intstat), FIFO);
2048 if (pCurrCard->currentSCCB != NULL)
2049 FPT_sxfrp(p_port, p_card);
2052 else if (p_int & TIMEOUT) {
2054 DISABLE_AUTO(p_port);
2056 WRW_HARPOON((p_port + hp_intstat),
2057 (PROG_HLT | TIMEOUT | SEL | BUS_FREE | PHASE |
2060 pCurrCard->currentSCCB->HostStatus = SCCB_SELECTION_TIMEOUT;
2063 &FPT_sccbMgrTbl[p_card][pCurrCard->currentSCCB->TargID];
2064 if ((pCurrCard->globalFlags & F_CONLUN_IO)
2065 && ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
2067 currTar_Info->TarLUNBusy[pCurrCard->currentSCCB->Lun] =
2070 currTar_Info->TarLUNBusy[0] = 0;
2072 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
2073 currTar_Info->TarSyncCtrl = 0;
2074 currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2077 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
2078 currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
2081 FPT_sssyncv(p_port, pCurrCard->currentSCCB->TargID, NARROW_SCSI,
2084 FPT_queueCmdComplete(pCurrCard, pCurrCard->currentSCCB, p_card);
2088 else if (p_int & SCAM_SEL) {
2090 FPT_scarb(p_port, LEVEL2_TAR);
2092 FPT_scasid(p_card, p_port);
2096 WRW_HARPOON((p_port + hp_intstat), SCAM_SEL);
2102 /*---------------------------------------------------------------------
2104 * Function: SccbMgrTableInit
2106 * Description: Initialize all Sccb manager data structures.
2108 *---------------------------------------------------------------------*/
2110 static void FPT_SccbMgrTableInitAll(void)
2112 unsigned char thisCard;
2114 for (thisCard = 0; thisCard < MAX_CARDS; thisCard++) {
2115 FPT_SccbMgrTableInitCard(&FPT_BL_Card[thisCard], thisCard);
2117 FPT_BL_Card[thisCard].ioPort = 0x00;
2118 FPT_BL_Card[thisCard].cardInfo = NULL;
2119 FPT_BL_Card[thisCard].cardIndex = 0xFF;
2120 FPT_BL_Card[thisCard].ourId = 0x00;
2121 FPT_BL_Card[thisCard].pNvRamInfo = NULL;
2125 /*---------------------------------------------------------------------
2127 * Function: SccbMgrTableInit
2129 * Description: Initialize all Sccb manager data structures.
2131 *---------------------------------------------------------------------*/
2133 static void FPT_SccbMgrTableInitCard(struct sccb_card *pCurrCard,
2134 unsigned char p_card)
2136 unsigned char scsiID, qtag;
2138 for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
2139 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
2142 for (scsiID = 0; scsiID < MAX_SCSI_TAR; scsiID++) {
2143 FPT_sccbMgrTbl[p_card][scsiID].TarStatus = 0;
2144 FPT_sccbMgrTbl[p_card][scsiID].TarEEValue = 0;
2145 FPT_SccbMgrTableInitTarget(p_card, scsiID);
2148 pCurrCard->scanIndex = 0x00;
2149 pCurrCard->currentSCCB = NULL;
2150 pCurrCard->globalFlags = 0x00;
2151 pCurrCard->cmdCounter = 0x00;
2152 pCurrCard->tagQ_Lst = 0x01;
2153 pCurrCard->discQCount = 0;
2157 /*---------------------------------------------------------------------
2159 * Function: SccbMgrTableInit
2161 * Description: Initialize all Sccb manager data structures.
2163 *---------------------------------------------------------------------*/
2165 static void FPT_SccbMgrTableInitTarget(unsigned char p_card,
2166 unsigned char target)
2169 unsigned char lun, qtag;
2170 struct sccb_mgr_tar_info *currTar_Info;
2172 currTar_Info = &FPT_sccbMgrTbl[p_card][target];
2174 currTar_Info->TarSelQ_Cnt = 0;
2175 currTar_Info->TarSyncCtrl = 0;
2177 currTar_Info->TarSelQ_Head = NULL;
2178 currTar_Info->TarSelQ_Tail = NULL;
2179 currTar_Info->TarTagQ_Cnt = 0;
2180 currTar_Info->TarLUN_CA = 0;
2182 for (lun = 0; lun < MAX_LUN; lun++) {
2183 currTar_Info->TarLUNBusy[lun] = 0;
2184 currTar_Info->LunDiscQ_Idx[lun] = 0;
2187 for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
2188 if (FPT_BL_Card[p_card].discQ_Tbl[qtag] != NULL) {
2189 if (FPT_BL_Card[p_card].discQ_Tbl[qtag]->TargID ==
2191 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
2192 FPT_BL_Card[p_card].discQCount--;
2198 /*---------------------------------------------------------------------
2202 * Description: Read in a message byte from the SCSI bus, and check
2203 * for a parity error.
2205 *---------------------------------------------------------------------*/
2207 static unsigned char FPT_sfm(u32 port, struct sccb *pCurrSCCB)
2209 unsigned char message;
2210 unsigned short TimeOutLoop;
2213 while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2214 (TimeOutLoop++ < 20000)) {
2217 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
2219 message = RD_HARPOON(port + hp_scsidata_0);
2221 WR_HARPOON(port + hp_scsisig, SCSI_ACK + S_MSGI_PH);
2223 if (TimeOutLoop > 20000)
2224 message = 0x00; /* force message byte = 0 if Time Out on Req */
2226 if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
2227 (RD_HARPOON(port + hp_addstat) & SCSI_PAR_ERR)) {
2228 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2229 WR_HARPOON(port + hp_xferstat, 0);
2230 WR_HARPOON(port + hp_fiforead, 0);
2231 WR_HARPOON(port + hp_fifowrite, 0);
2232 if (pCurrSCCB != NULL) {
2233 pCurrSCCB->Sccb_scsimsg = MSG_PARITY_ERROR;
2237 ACCEPT_MSG_ATN(port);
2239 while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2240 (TimeOutLoop++ < 20000)) {
2242 if (TimeOutLoop > 20000) {
2243 WRW_HARPOON((port + hp_intstat), PARITY);
2246 if ((RD_HARPOON(port + hp_scsisig) & S_SCSI_PHZ) !=
2248 WRW_HARPOON((port + hp_intstat), PARITY);
2251 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
2253 RD_HARPOON(port + hp_scsidata_0);
2255 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2260 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2261 WR_HARPOON(port + hp_xferstat, 0);
2262 WR_HARPOON(port + hp_fiforead, 0);
2263 WR_HARPOON(port + hp_fifowrite, 0);
2267 /*---------------------------------------------------------------------
2269 * Function: FPT_ssel
2271 * Description: Load up automation and select target device.
2273 *---------------------------------------------------------------------*/
2275 static void FPT_ssel(u32 port, unsigned char p_card)
2278 unsigned char auto_loaded, i, target, *theCCB;
2281 struct sccb_card *CurrCard;
2282 struct sccb *currSCCB;
2283 struct sccb_mgr_tar_info *currTar_Info;
2284 unsigned char lastTag, lun;
2286 CurrCard = &FPT_BL_Card[p_card];
2287 currSCCB = CurrCard->currentSCCB;
2288 target = currSCCB->TargID;
2289 currTar_Info = &FPT_sccbMgrTbl[p_card][target];
2290 lastTag = CurrCard->tagQ_Lst;
2294 if ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) == TAG_Q_REJECT)
2295 currSCCB->ControlByte &= ~F_USE_CMD_Q;
2297 if (((CurrCard->globalFlags & F_CONLUN_IO) &&
2298 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING)))
2300 lun = currSCCB->Lun;
2304 if (CurrCard->globalFlags & F_TAG_STARTED) {
2305 if (!(currSCCB->ControlByte & F_USE_CMD_Q)) {
2306 if ((currTar_Info->TarLUN_CA == 0)
2307 && ((currTar_Info->TarStatus & TAR_TAG_Q_MASK)
2310 if (currTar_Info->TarTagQ_Cnt != 0) {
2311 currTar_Info->TarLUNBusy[lun] = 1;
2312 FPT_queueSelectFail(CurrCard, p_card);
2318 currTar_Info->TarLUNBusy[lun] = 1;
2324 currTar_Info->TarLUNBusy[lun] = 1;
2328 /*!Use cmd Q Tagged */
2330 if (currTar_Info->TarLUN_CA == 1) {
2331 FPT_queueSelectFail(CurrCard, p_card);
2336 currTar_Info->TarLUNBusy[lun] = 1;
2338 } /*else use cmd Q tagged */
2341 /*if glob tagged started */
2343 currTar_Info->TarLUNBusy[lun] = 1;
2346 if ((((CurrCard->globalFlags & F_CONLUN_IO) &&
2347 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
2348 || (!(currSCCB->ControlByte & F_USE_CMD_Q)))) {
2349 if (CurrCard->discQCount >= QUEUE_DEPTH) {
2350 currTar_Info->TarLUNBusy[lun] = 1;
2351 FPT_queueSelectFail(CurrCard, p_card);
2355 for (i = 1; i < QUEUE_DEPTH; i++) {
2356 if (++lastTag >= QUEUE_DEPTH)
2358 if (CurrCard->discQ_Tbl[lastTag] == NULL) {
2359 CurrCard->tagQ_Lst = lastTag;
2360 currTar_Info->LunDiscQ_Idx[lun] = lastTag;
2361 CurrCard->discQ_Tbl[lastTag] = currSCCB;
2362 CurrCard->discQCount++;
2366 if (i == QUEUE_DEPTH) {
2367 currTar_Info->TarLUNBusy[lun] = 1;
2368 FPT_queueSelectFail(CurrCard, p_card);
2376 WR_HARPOON(port + hp_select_id, target);
2377 WR_HARPOON(port + hp_gp_reg_3, target); /* Use by new automation logic */
2379 if (currSCCB->OperationCode == RESET_COMMAND) {
2380 WRW_HARPOON((port + ID_MSG_STRT), (MPM_OP + AMSG_OUT +
2382 Sccb_idmsg & ~DISC_PRIV)));
2384 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + NP);
2386 currSCCB->Sccb_scsimsg = TARGET_RESET;
2388 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
2390 currSCCB->Sccb_scsistat = SELECT_BDR_ST;
2392 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
2393 currTar_Info->TarSyncCtrl = 0;
2394 currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2397 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
2398 currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
2401 FPT_sssyncv(port, target, NARROW_SCSI, currTar_Info);
2402 FPT_SccbMgrTableInitTarget(p_card, target);
2406 else if (currSCCB->Sccb_scsistat == ABORT_ST) {
2407 WRW_HARPOON((port + ID_MSG_STRT), (MPM_OP + AMSG_OUT +
2409 Sccb_idmsg & ~DISC_PRIV)));
2411 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
2413 WRW_HARPOON((port + SYNC_MSGS + 0), (MPM_OP + AMSG_OUT +
2418 >> 6) | (unsigned char)
2420 WRW_HARPOON((port + SYNC_MSGS + 2),
2421 (MPM_OP + AMSG_OUT + currSCCB->Sccb_tag));
2422 WRW_HARPOON((port + SYNC_MSGS + 4), (BRH_OP + ALWAYS + NP));
2424 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
2429 else if (!(currTar_Info->TarStatus & WIDE_NEGOCIATED)) {
2430 auto_loaded = FPT_siwidn(port, p_card);
2431 currSCCB->Sccb_scsistat = SELECT_WN_ST;
2434 else if (!((currTar_Info->TarStatus & TAR_SYNC_MASK)
2435 == SYNC_SUPPORTED)) {
2436 auto_loaded = FPT_sisyncn(port, p_card, 0);
2437 currSCCB->Sccb_scsistat = SELECT_SN_ST;
2442 if (currSCCB->ControlByte & F_USE_CMD_Q) {
2444 CurrCard->globalFlags |= F_TAG_STARTED;
2446 if ((currTar_Info->TarStatus & TAR_TAG_Q_MASK)
2448 currSCCB->ControlByte &= ~F_USE_CMD_Q;
2450 /* Fix up the start instruction with a jump to
2451 Non-Tag-CMD handling */
2452 WRW_HARPOON((port + ID_MSG_STRT),
2453 BRH_OP + ALWAYS + NTCMD);
2455 WRW_HARPOON((port + NON_TAG_ID_MSG),
2456 (MPM_OP + AMSG_OUT +
2457 currSCCB->Sccb_idmsg));
2459 WR_HARPOON(port + hp_autostart_3,
2460 (SELECT + SELCHK_STRT));
2462 /* Setup our STATE so we know what happened when
2463 the wheels fall off. */
2464 currSCCB->Sccb_scsistat = SELECT_ST;
2466 currTar_Info->TarLUNBusy[lun] = 1;
2470 WRW_HARPOON((port + ID_MSG_STRT),
2471 (MPM_OP + AMSG_OUT +
2472 currSCCB->Sccb_idmsg));
2474 WRW_HARPOON((port + ID_MSG_STRT + 2),
2475 (MPM_OP + AMSG_OUT +
2476 (((unsigned char)(currSCCB->
2479 >> 6) | (unsigned char)0x20)));
2481 for (i = 1; i < QUEUE_DEPTH; i++) {
2482 if (++lastTag >= QUEUE_DEPTH)
2484 if (CurrCard->discQ_Tbl[lastTag] ==
2488 (MPM_OP + AMSG_OUT +
2490 CurrCard->tagQ_Lst = lastTag;
2491 currSCCB->Sccb_tag = lastTag;
2492 CurrCard->discQ_Tbl[lastTag] =
2494 CurrCard->discQCount++;
2499 if (i == QUEUE_DEPTH) {
2500 currTar_Info->TarLUNBusy[lun] = 1;
2501 FPT_queueSelectFail(CurrCard, p_card);
2506 currSCCB->Sccb_scsistat = SELECT_Q_ST;
2508 WR_HARPOON(port + hp_autostart_3,
2509 (SELECT + SELCHK_STRT));
2515 WRW_HARPOON((port + ID_MSG_STRT),
2516 BRH_OP + ALWAYS + NTCMD);
2518 WRW_HARPOON((port + NON_TAG_ID_MSG),
2519 (MPM_OP + AMSG_OUT + currSCCB->Sccb_idmsg));
2521 currSCCB->Sccb_scsistat = SELECT_ST;
2523 WR_HARPOON(port + hp_autostart_3,
2524 (SELECT + SELCHK_STRT));
2527 theCCB = (unsigned char *)&currSCCB->Cdb[0];
2529 cdb_reg = port + CMD_STRT;
2531 for (i = 0; i < currSCCB->CdbLength; i++) {
2532 WRW_HARPOON(cdb_reg, (MPM_OP + ACOMMAND + *theCCB));
2537 if (currSCCB->CdbLength != TWELVE_BYTE_CMD)
2538 WRW_HARPOON(cdb_reg, (BRH_OP + ALWAYS + NP));
2542 WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
2543 WR_HARPOON(port + hp_xferstat, 0x00);
2545 WRW_HARPOON((port + hp_intstat), (PROG_HLT | TIMEOUT | SEL | BUS_FREE));
2547 WR_HARPOON(port + hp_portctrl_0, (SCSI_PORT));
2549 if (!(currSCCB->Sccb_MGRFlags & F_DEV_SELECTED)) {
2550 WR_HARPOON(port + hp_scsictrl_0,
2551 (SEL_TAR | ENA_ATN | ENA_RESEL | ENA_SCAM_SEL));
2554 /* auto_loaded = (RD_HARPOON(port+hp_autostart_3) & (unsigned char)0x1F);
2555 auto_loaded |= AUTO_IMMED; */
2556 auto_loaded = AUTO_IMMED;
2560 WR_HARPOON(port + hp_autostart_3, auto_loaded);
2566 /*---------------------------------------------------------------------
2568 * Function: FPT_sres
2570 * Description: Hookup the correct CCB and handle the incoming messages.
2572 *---------------------------------------------------------------------*/
2574 static void FPT_sres(u32 port, unsigned char p_card,
2575 struct sccb_card *pCurrCard)
2578 unsigned char our_target, message, lun = 0, tag, msgRetryCount;
2580 struct sccb_mgr_tar_info *currTar_Info;
2581 struct sccb *currSCCB;
2583 if (pCurrCard->currentSCCB != NULL) {
2585 &FPT_sccbMgrTbl[p_card][pCurrCard->currentSCCB->TargID];
2588 WR_HARPOON((port + hp_scsictrl_0), (ENA_RESEL | ENA_SCAM_SEL));
2590 currSCCB = pCurrCard->currentSCCB;
2591 if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
2592 currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
2593 currSCCB->Sccb_scsistat = BUS_FREE_ST;
2595 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
2596 currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2597 currSCCB->Sccb_scsistat = BUS_FREE_ST;
2599 if (((pCurrCard->globalFlags & F_CONLUN_IO) &&
2600 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
2602 currTar_Info->TarLUNBusy[currSCCB->Lun] = 0;
2603 if (currSCCB->Sccb_scsistat != ABORT_ST) {
2604 pCurrCard->discQCount--;
2605 pCurrCard->discQ_Tbl[currTar_Info->
2606 LunDiscQ_Idx[currSCCB->
2611 currTar_Info->TarLUNBusy[0] = 0;
2612 if (currSCCB->Sccb_tag) {
2613 if (currSCCB->Sccb_scsistat != ABORT_ST) {
2614 pCurrCard->discQCount--;
2615 pCurrCard->discQ_Tbl[currSCCB->
2619 if (currSCCB->Sccb_scsistat != ABORT_ST) {
2620 pCurrCard->discQCount--;
2621 pCurrCard->discQ_Tbl[currTar_Info->
2628 FPT_queueSelectFail(&FPT_BL_Card[p_card], p_card);
2631 WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
2633 our_target = (unsigned char)(RD_HARPOON(port + hp_select_id) >> 4);
2634 currTar_Info = &FPT_sccbMgrTbl[p_card][our_target];
2639 currTar_Info = &FPT_sccbMgrTbl[p_card][our_target];
2642 while (!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) {
2643 if (!(RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) {
2645 WRW_HARPOON((port + hp_intstat), PHASE);
2650 WRW_HARPOON((port + hp_intstat), PHASE);
2651 if ((RD_HARPOON(port + hp_scsisig) & S_SCSI_PHZ) == S_MSGI_PH) {
2653 message = FPT_sfm(port, pCurrCard->currentSCCB);
2656 if (message <= (0x80 | LUN_MASK)) {
2657 lun = message & (unsigned char)LUN_MASK;
2660 TarStatus & TAR_TAG_Q_MASK) ==
2662 if (currTar_Info->TarTagQ_Cnt !=
2668 ACCEPT_MSG(port); /*Release the ACK for ID msg. */
2704 /*End Tag cmds supported! */
2706 /*End valid ID message. */
2709 ACCEPT_MSG_ATN(port);
2713 /* End good id message. */
2719 ACCEPT_MSG_ATN(port);
2722 (RDW_HARPOON((port + hp_intstat)) &
2724 && !(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)
2725 && (RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) ;
2732 if (msgRetryCount == 1) {
2733 FPT_SendMsg(port, MSG_PARITY_ERROR);
2735 FPT_SendMsg(port, TARGET_RESET);
2737 FPT_sssyncv(port, our_target, NARROW_SCSI,
2740 if (FPT_sccbMgrTbl[p_card][our_target].
2741 TarEEValue & EE_SYNC_MASK) {
2743 FPT_sccbMgrTbl[p_card][our_target].
2744 TarStatus &= ~TAR_SYNC_MASK;
2748 if (FPT_sccbMgrTbl[p_card][our_target].
2749 TarEEValue & EE_WIDE_SCSI) {
2751 FPT_sccbMgrTbl[p_card][our_target].
2752 TarStatus &= ~TAR_WIDE_MASK;
2755 FPT_queueFlushTargSccb(p_card, our_target,
2757 FPT_SccbMgrTableInitTarget(p_card, our_target);
2761 } while (message == 0);
2763 if (((pCurrCard->globalFlags & F_CONLUN_IO) &&
2764 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))) {
2765 currTar_Info->TarLUNBusy[lun] = 1;
2766 pCurrCard->currentSCCB =
2767 pCurrCard->discQ_Tbl[currTar_Info->LunDiscQ_Idx[lun]];
2768 if (pCurrCard->currentSCCB != NULL) {
2771 ACCEPT_MSG_ATN(port);
2774 currTar_Info->TarLUNBusy[0] = 1;
2777 if (pCurrCard->discQ_Tbl[tag] != NULL) {
2778 pCurrCard->currentSCCB =
2779 pCurrCard->discQ_Tbl[tag];
2780 currTar_Info->TarTagQ_Cnt--;
2783 ACCEPT_MSG_ATN(port);
2786 pCurrCard->currentSCCB =
2787 pCurrCard->discQ_Tbl[currTar_Info->LunDiscQ_Idx[0]];
2788 if (pCurrCard->currentSCCB != NULL) {
2791 ACCEPT_MSG_ATN(port);
2796 if (pCurrCard->currentSCCB != NULL) {
2797 if (pCurrCard->currentSCCB->Sccb_scsistat == ABORT_ST) {
2798 /* During Abort Tag command, the target could have got re-selected
2799 and completed the command. Check the select Q and remove the CCB
2800 if it is in the Select Q */
2801 FPT_queueFindSccb(pCurrCard->currentSCCB, p_card);
2805 while (!(RDW_HARPOON((port + hp_intstat)) & (PHASE | RESET)) &&
2806 !(RD_HARPOON(port + hp_scsisig) & SCSI_REQ) &&
2807 (RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) ;
2810 static void FPT_SendMsg(u32 port, unsigned char message)
2812 while (!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) {
2813 if (!(RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) {
2815 WRW_HARPOON((port + hp_intstat), PHASE);
2820 WRW_HARPOON((port + hp_intstat), PHASE);
2821 if ((RD_HARPOON(port + hp_scsisig) & S_SCSI_PHZ) == S_MSGO_PH) {
2822 WRW_HARPOON((port + hp_intstat),
2823 (BUS_FREE | PHASE | XFER_CNT_0));
2825 WR_HARPOON(port + hp_portctrl_0, SCSI_BUS_EN);
2827 WR_HARPOON(port + hp_scsidata_0, message);
2829 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2833 WR_HARPOON(port + hp_portctrl_0, 0x00);
2835 if ((message == ABORT_TASK_SET) || (message == TARGET_RESET) ||
2836 (message == ABORT_TASK)) {
2838 (RDW_HARPOON((port + hp_intstat)) &
2839 (BUS_FREE | PHASE))) {
2842 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
2843 WRW_HARPOON((port + hp_intstat), BUS_FREE);
2849 /*---------------------------------------------------------------------
2851 * Function: FPT_sdecm
2853 * Description: Determine the proper response to the message from the
2856 *---------------------------------------------------------------------*/
2857 static void FPT_sdecm(unsigned char message, u32 port, unsigned char p_card)
2859 struct sccb *currSCCB;
2860 struct sccb_card *CurrCard;
2861 struct sccb_mgr_tar_info *currTar_Info;
2863 CurrCard = &FPT_BL_Card[p_card];
2864 currSCCB = CurrCard->currentSCCB;
2866 currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
2868 if (message == RESTORE_POINTERS) {
2869 if (!(currSCCB->Sccb_XferState & F_NO_DATA_YET)) {
2870 currSCCB->Sccb_ATC = currSCCB->Sccb_savedATC;
2872 FPT_hostDataXferRestart(currSCCB);
2876 WR_HARPOON(port + hp_autostart_1,
2877 (AUTO_IMMED + DISCONNECT_START));
2880 else if (message == COMMAND_COMPLETE) {
2882 if (currSCCB->Sccb_scsistat == SELECT_Q_ST) {
2883 currTar_Info->TarStatus &=
2884 ~(unsigned char)TAR_TAG_Q_MASK;
2885 currTar_Info->TarStatus |= (unsigned char)TAG_Q_REJECT;
2892 else if ((message == NOP) || (message >= IDENTIFY_BASE) ||
2893 (message == INITIATE_RECOVERY) ||
2894 (message == RELEASE_RECOVERY)) {
2897 WR_HARPOON(port + hp_autostart_1,
2898 (AUTO_IMMED + DISCONNECT_START));
2901 else if (message == MESSAGE_REJECT) {
2903 if ((currSCCB->Sccb_scsistat == SELECT_SN_ST) ||
2904 (currSCCB->Sccb_scsistat == SELECT_WN_ST) ||
2905 ((currTar_Info->TarStatus & TAR_SYNC_MASK) == SYNC_TRYING)
2906 || ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) ==
2909 WRW_HARPOON((port + hp_intstat), BUS_FREE);
2913 while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2914 (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)))
2918 if (currSCCB->Lun == 0x00) {
2919 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
2921 currTar_Info->TarStatus |=
2922 (unsigned char)SYNC_SUPPORTED;
2924 currTar_Info->TarEEValue &=
2928 else if (currSCCB->Sccb_scsistat ==
2931 currTar_Info->TarStatus =
2933 TarStatus & ~WIDE_ENABLED) |
2936 currTar_Info->TarEEValue &=
2941 else if ((currTar_Info->
2942 TarStatus & TAR_TAG_Q_MASK) ==
2944 currTar_Info->TarStatus =
2946 TarStatus & ~(unsigned char)
2947 TAR_TAG_Q_MASK) | TAG_Q_REJECT;
2949 currSCCB->ControlByte &= ~F_USE_CMD_Q;
2950 CurrCard->discQCount--;
2951 CurrCard->discQ_Tbl[currSCCB->
2953 currSCCB->Sccb_tag = 0x00;
2958 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
2960 if (currSCCB->Lun == 0x00) {
2961 WRW_HARPOON((port + hp_intstat),
2963 CurrCard->globalFlags |= F_NEW_SCCB_CMD;
2969 if ((CurrCard->globalFlags & F_CONLUN_IO) &&
2971 TarStatus & TAR_TAG_Q_MASK) !=
2973 currTar_Info->TarLUNBusy[currSCCB->
2976 currTar_Info->TarLUNBusy[0] = 1;
2978 currSCCB->ControlByte &=
2979 ~(unsigned char)F_USE_CMD_Q;
2981 WR_HARPOON(port + hp_autostart_1,
2982 (AUTO_IMMED + DISCONNECT_START));
2990 while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2991 (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)))
2995 if (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)) {
2996 WR_HARPOON(port + hp_autostart_1,
2997 (AUTO_IMMED + DISCONNECT_START));
3002 else if (message == EXTENDED_MESSAGE) {
3005 FPT_shandem(port, p_card, currSCCB);
3008 else if (message == IGNORE_WIDE_RESIDUE) {
3010 ACCEPT_MSG(port); /* ACK the RESIDUE MSG */
3012 message = FPT_sfm(port, currSCCB);
3014 if (currSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
3016 WR_HARPOON(port + hp_autostart_1,
3017 (AUTO_IMMED + DISCONNECT_START));
3022 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
3023 currSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3025 ACCEPT_MSG_ATN(port);
3026 WR_HARPOON(port + hp_autostart_1,
3027 (AUTO_IMMED + DISCONNECT_START));
3031 /*---------------------------------------------------------------------
3033 * Function: FPT_shandem
3035 * Description: Decide what to do with the extended message.
3037 *---------------------------------------------------------------------*/
3038 static void FPT_shandem(u32 port, unsigned char p_card, struct sccb *pCurrSCCB)
3040 unsigned char length, message;
3042 length = FPT_sfm(port, pCurrSCCB);
3046 message = FPT_sfm(port, pCurrSCCB);
3049 if (message == EXTENDED_SDTR) {
3051 if (length == 0x03) {
3054 FPT_stsyncn(port, p_card);
3057 pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3058 ACCEPT_MSG_ATN(port);
3060 } else if (message == EXTENDED_WDTR) {
3062 if (length == 0x02) {
3065 FPT_stwidn(port, p_card);
3068 pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3069 ACCEPT_MSG_ATN(port);
3071 WR_HARPOON(port + hp_autostart_1,
3077 pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3078 ACCEPT_MSG_ATN(port);
3080 WR_HARPOON(port + hp_autostart_1,
3081 (AUTO_IMMED + DISCONNECT_START));
3084 if (pCurrSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
3086 WR_HARPOON(port + hp_autostart_1,
3087 (AUTO_IMMED + DISCONNECT_START));
3090 if (pCurrSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)
3091 WR_HARPOON(port + hp_autostart_1,
3092 (AUTO_IMMED + DISCONNECT_START));
3096 /*---------------------------------------------------------------------
3098 * Function: FPT_sisyncn
3100 * Description: Read in a message byte from the SCSI bus, and check
3101 * for a parity error.
3103 *---------------------------------------------------------------------*/
3105 static unsigned char FPT_sisyncn(u32 port, unsigned char p_card,
3106 unsigned char syncFlag)
3108 struct sccb *currSCCB;
3109 struct sccb_mgr_tar_info *currTar_Info;
3111 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3112 currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3114 if (!((currTar_Info->TarStatus & TAR_SYNC_MASK) == SYNC_TRYING)) {
3116 WRW_HARPOON((port + ID_MSG_STRT),
3117 (MPM_OP + AMSG_OUT +
3119 Sccb_idmsg & ~(unsigned char)DISC_PRIV)));
3121 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
3123 WRW_HARPOON((port + SYNC_MSGS + 0),
3124 (MPM_OP + AMSG_OUT + EXTENDED_MESSAGE));
3125 WRW_HARPOON((port + SYNC_MSGS + 2), (MPM_OP + AMSG_OUT + 0x03));
3126 WRW_HARPOON((port + SYNC_MSGS + 4),
3127 (MPM_OP + AMSG_OUT + EXTENDED_SDTR));
3129 if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_20MB)
3131 WRW_HARPOON((port + SYNC_MSGS + 6),
3132 (MPM_OP + AMSG_OUT + 12));
3134 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) ==
3137 WRW_HARPOON((port + SYNC_MSGS + 6),
3138 (MPM_OP + AMSG_OUT + 25));
3140 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) ==
3143 WRW_HARPOON((port + SYNC_MSGS + 6),
3144 (MPM_OP + AMSG_OUT + 50));
3147 WRW_HARPOON((port + SYNC_MSGS + 6),
3148 (MPM_OP + AMSG_OUT + 00));
3150 WRW_HARPOON((port + SYNC_MSGS + 8), (RAT_OP));
3151 WRW_HARPOON((port + SYNC_MSGS + 10),
3152 (MPM_OP + AMSG_OUT + DEFAULT_OFFSET));
3153 WRW_HARPOON((port + SYNC_MSGS + 12), (BRH_OP + ALWAYS + NP));
3155 if (syncFlag == 0) {
3156 WR_HARPOON(port + hp_autostart_3,
3157 (SELECT + SELCHK_STRT));
3158 currTar_Info->TarStatus =
3160 TarStatus & ~(unsigned char)TAR_SYNC_MASK) |
3161 (unsigned char)SYNC_TRYING);
3163 WR_HARPOON(port + hp_autostart_3,
3164 (AUTO_IMMED + CMD_ONLY_STRT));
3172 currTar_Info->TarStatus |= (unsigned char)SYNC_SUPPORTED;
3173 currTar_Info->TarEEValue &= ~EE_SYNC_MASK;
3178 /*---------------------------------------------------------------------
3180 * Function: FPT_stsyncn
3182 * Description: The has sent us a Sync Nego message so handle it as
3185 *---------------------------------------------------------------------*/
3186 static void FPT_stsyncn(u32 port, unsigned char p_card)
3188 unsigned char sync_msg, offset, sync_reg, our_sync_msg;
3189 struct sccb *currSCCB;
3190 struct sccb_mgr_tar_info *currTar_Info;
3192 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3193 currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3195 sync_msg = FPT_sfm(port, currSCCB);
3197 if ((sync_msg == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3198 WR_HARPOON(port + hp_autostart_1,
3199 (AUTO_IMMED + DISCONNECT_START));
3205 offset = FPT_sfm(port, currSCCB);
3207 if ((offset == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3208 WR_HARPOON(port + hp_autostart_1,
3209 (AUTO_IMMED + DISCONNECT_START));
3213 if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_20MB)
3215 our_sync_msg = 12; /* Setup our Message to 20mb/s */
3217 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_10MB)
3219 our_sync_msg = 25; /* Setup our Message to 10mb/s */
3221 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_5MB)
3223 our_sync_msg = 50; /* Setup our Message to 5mb/s */
3226 our_sync_msg = 0; /* Message = Async */
3228 if (sync_msg < our_sync_msg) {
3229 sync_msg = our_sync_msg; /*if faster, then set to max. */
3232 if (offset == ASYNC)
3235 if (offset > MAX_OFFSET)
3236 offset = MAX_OFFSET;
3242 sync_reg = 0x20; /* Use 10MB/s */
3246 sync_reg = 0x40; /* Use 6.6MB/s */
3250 sync_reg = 0x60; /* Use 5MB/s */
3254 sync_reg = 0x80; /* Use 4MB/s */
3258 sync_reg = 0xA0; /* Use 3.33MB/s */
3262 sync_reg = 0xC0; /* Use 2.85MB/s */
3266 sync_reg = 0xE0; /* Use 2.5MB/s */
3268 if (sync_msg > 100) {
3270 sync_reg = 0x00; /* Use ASYNC */
3274 if (currTar_Info->TarStatus & WIDE_ENABLED)
3280 sync_reg |= (offset | NARROW_SCSI);
3282 FPT_sssyncv(port, currSCCB->TargID, sync_reg, currTar_Info);
3284 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
3288 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3289 ~(unsigned char)TAR_SYNC_MASK) |
3290 (unsigned char)SYNC_SUPPORTED);
3292 WR_HARPOON(port + hp_autostart_1,
3293 (AUTO_IMMED + DISCONNECT_START));
3298 ACCEPT_MSG_ATN(port);
3300 FPT_sisyncr(port, sync_msg, offset);
3302 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3303 ~(unsigned char)TAR_SYNC_MASK) |
3304 (unsigned char)SYNC_SUPPORTED);
3308 /*---------------------------------------------------------------------
3310 * Function: FPT_sisyncr
3312 * Description: Answer the targets sync message.
3314 *---------------------------------------------------------------------*/
3315 static void FPT_sisyncr(u32 port, unsigned char sync_pulse,
3316 unsigned char offset)
3319 WRW_HARPOON((port + SYNC_MSGS + 0),
3320 (MPM_OP + AMSG_OUT + EXTENDED_MESSAGE));
3321 WRW_HARPOON((port + SYNC_MSGS + 2), (MPM_OP + AMSG_OUT + 0x03));
3322 WRW_HARPOON((port + SYNC_MSGS + 4),
3323 (MPM_OP + AMSG_OUT + EXTENDED_SDTR));
3324 WRW_HARPOON((port + SYNC_MSGS + 6), (MPM_OP + AMSG_OUT + sync_pulse));
3325 WRW_HARPOON((port + SYNC_MSGS + 8), (RAT_OP));
3326 WRW_HARPOON((port + SYNC_MSGS + 10), (MPM_OP + AMSG_OUT + offset));
3327 WRW_HARPOON((port + SYNC_MSGS + 12), (BRH_OP + ALWAYS + NP));
3330 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3331 WRW_HARPOON((port + hp_intstat), CLR_ALL_INT_1);
3333 WR_HARPOON(port + hp_autostart_3, (AUTO_IMMED + CMD_ONLY_STRT));
3335 while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | AUTO_INT))) {
3339 /*---------------------------------------------------------------------
3341 * Function: FPT_siwidn
3343 * Description: Read in a message byte from the SCSI bus, and check
3344 * for a parity error.
3346 *---------------------------------------------------------------------*/
3348 static unsigned char FPT_siwidn(u32 port, unsigned char p_card)
3350 struct sccb *currSCCB;
3351 struct sccb_mgr_tar_info *currTar_Info;
3353 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3354 currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3356 if (!((currTar_Info->TarStatus & TAR_WIDE_MASK) == WIDE_NEGOCIATED)) {
3358 WRW_HARPOON((port + ID_MSG_STRT),
3359 (MPM_OP + AMSG_OUT +
3361 Sccb_idmsg & ~(unsigned char)DISC_PRIV)));
3363 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
3365 WRW_HARPOON((port + SYNC_MSGS + 0),
3366 (MPM_OP + AMSG_OUT + EXTENDED_MESSAGE));
3367 WRW_HARPOON((port + SYNC_MSGS + 2), (MPM_OP + AMSG_OUT + 0x02));
3368 WRW_HARPOON((port + SYNC_MSGS + 4),
3369 (MPM_OP + AMSG_OUT + EXTENDED_WDTR));
3370 WRW_HARPOON((port + SYNC_MSGS + 6), (RAT_OP));
3371 WRW_HARPOON((port + SYNC_MSGS + 8),
3372 (MPM_OP + AMSG_OUT + SM16BIT));
3373 WRW_HARPOON((port + SYNC_MSGS + 10), (BRH_OP + ALWAYS + NP));
3375 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
3377 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3378 ~(unsigned char)TAR_WIDE_MASK) |
3379 (unsigned char)WIDE_ENABLED);
3386 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3387 ~(unsigned char)TAR_WIDE_MASK) |
3390 currTar_Info->TarEEValue &= ~EE_WIDE_SCSI;
3395 /*---------------------------------------------------------------------
3397 * Function: FPT_stwidn
3399 * Description: The has sent us a Wide Nego message so handle it as
3402 *---------------------------------------------------------------------*/
3403 static void FPT_stwidn(u32 port, unsigned char p_card)
3405 unsigned char width;
3406 struct sccb *currSCCB;
3407 struct sccb_mgr_tar_info *currTar_Info;
3409 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3410 currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3412 width = FPT_sfm(port, currSCCB);
3414 if ((width == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3415 WR_HARPOON(port + hp_autostart_1,
3416 (AUTO_IMMED + DISCONNECT_START));
3420 if (!(currTar_Info->TarEEValue & EE_WIDE_SCSI))
3424 currTar_Info->TarStatus |= WIDE_ENABLED;
3427 width = NARROW_SCSI;
3428 currTar_Info->TarStatus &= ~WIDE_ENABLED;
3431 FPT_sssyncv(port, currSCCB->TargID, width, currTar_Info);
3433 if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
3435 currTar_Info->TarStatus |= WIDE_NEGOCIATED;
3438 ((currTar_Info->TarStatus & TAR_SYNC_MASK) ==
3440 ACCEPT_MSG_ATN(port);
3442 FPT_sisyncn(port, p_card, 1);
3443 currSCCB->Sccb_scsistat = SELECT_SN_ST;
3447 WR_HARPOON(port + hp_autostart_1,
3448 (AUTO_IMMED + DISCONNECT_START));
3454 ACCEPT_MSG_ATN(port);
3456 if (currTar_Info->TarEEValue & EE_WIDE_SCSI)
3461 FPT_siwidr(port, width);
3463 currTar_Info->TarStatus |= (WIDE_NEGOCIATED | WIDE_ENABLED);
3467 /*---------------------------------------------------------------------
3469 * Function: FPT_siwidr
3471 * Description: Answer the targets Wide nego message.
3473 *---------------------------------------------------------------------*/
3474 static void FPT_siwidr(u32 port, unsigned char width)
3477 WRW_HARPOON((port + SYNC_MSGS + 0),
3478 (MPM_OP + AMSG_OUT + EXTENDED_MESSAGE));
3479 WRW_HARPOON((port + SYNC_MSGS + 2), (MPM_OP + AMSG_OUT + 0x02));
3480 WRW_HARPOON((port + SYNC_MSGS + 4),
3481 (MPM_OP + AMSG_OUT + EXTENDED_WDTR));
3482 WRW_HARPOON((port + SYNC_MSGS + 6), (RAT_OP));
3483 WRW_HARPOON((port + SYNC_MSGS + 8), (MPM_OP + AMSG_OUT + width));
3484 WRW_HARPOON((port + SYNC_MSGS + 10), (BRH_OP + ALWAYS + NP));
3487 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3488 WRW_HARPOON((port + hp_intstat), CLR_ALL_INT_1);
3490 WR_HARPOON(port + hp_autostart_3, (AUTO_IMMED + CMD_ONLY_STRT));
3492 while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | AUTO_INT))) {
3496 /*---------------------------------------------------------------------
3498 * Function: FPT_sssyncv
3500 * Description: Write the desired value to the Sync Register for the
3503 *---------------------------------------------------------------------*/
3504 static void FPT_sssyncv(u32 p_port, unsigned char p_id,
3505 unsigned char p_sync_value,
3506 struct sccb_mgr_tar_info *currTar_Info)
3508 unsigned char index;
3515 index = 12; /* hp_synctarg_0 */
3518 index = 13; /* hp_synctarg_1 */
3521 index = 14; /* hp_synctarg_2 */
3524 index = 15; /* hp_synctarg_3 */
3527 index = 8; /* hp_synctarg_4 */
3530 index = 9; /* hp_synctarg_5 */
3533 index = 10; /* hp_synctarg_6 */
3536 index = 11; /* hp_synctarg_7 */
3539 index = 4; /* hp_synctarg_8 */
3542 index = 5; /* hp_synctarg_9 */
3545 index = 6; /* hp_synctarg_10 */
3548 index = 7; /* hp_synctarg_11 */
3551 index = 0; /* hp_synctarg_12 */
3554 index = 1; /* hp_synctarg_13 */
3557 index = 2; /* hp_synctarg_14 */
3560 index = 3; /* hp_synctarg_15 */
3564 WR_HARPOON(p_port + hp_synctarg_base + index, p_sync_value);
3566 currTar_Info->TarSyncCtrl = p_sync_value;
3569 /*---------------------------------------------------------------------
3571 * Function: FPT_sresb
3573 * Description: Reset the desired card's SCSI bus.
3575 *---------------------------------------------------------------------*/
3576 static void FPT_sresb(u32 port, unsigned char p_card)
3578 unsigned char scsiID, i;
3580 struct sccb_mgr_tar_info *currTar_Info;
3582 WR_HARPOON(port + hp_page_ctrl,
3583 (RD_HARPOON(port + hp_page_ctrl) | G_INT_DISABLE));
3584 WRW_HARPOON((port + hp_intstat), CLR_ALL_INT);
3586 WR_HARPOON(port + hp_scsictrl_0, SCSI_RST);
3588 scsiID = RD_HARPOON(port + hp_seltimeout);
3589 WR_HARPOON(port + hp_seltimeout, TO_5ms);
3590 WRW_HARPOON((port + hp_intstat), TIMEOUT);
3592 WR_HARPOON(port + hp_portctrl_0, (SCSI_PORT | START_TO));
3594 while (!(RDW_HARPOON((port + hp_intstat)) & TIMEOUT)) {
3597 WR_HARPOON(port + hp_seltimeout, scsiID);
3599 WR_HARPOON(port + hp_scsictrl_0, ENA_SCAM_SEL);
3601 FPT_Wait(port, TO_5ms);
3603 WRW_HARPOON((port + hp_intstat), CLR_ALL_INT);
3605 WR_HARPOON(port + hp_int_mask, (RD_HARPOON(port + hp_int_mask) | 0x00));
3607 for (scsiID = 0; scsiID < MAX_SCSI_TAR; scsiID++) {
3608 currTar_Info = &FPT_sccbMgrTbl[p_card][scsiID];
3610 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
3611 currTar_Info->TarSyncCtrl = 0;
3612 currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
3615 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
3616 currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
3619 FPT_sssyncv(port, scsiID, NARROW_SCSI, currTar_Info);
3621 FPT_SccbMgrTableInitTarget(p_card, scsiID);
3624 FPT_BL_Card[p_card].scanIndex = 0x00;
3625 FPT_BL_Card[p_card].currentSCCB = NULL;
3626 FPT_BL_Card[p_card].globalFlags &= ~(F_TAG_STARTED | F_HOST_XFER_ACT
3628 FPT_BL_Card[p_card].cmdCounter = 0x00;
3629 FPT_BL_Card[p_card].discQCount = 0x00;
3630 FPT_BL_Card[p_card].tagQ_Lst = 0x01;
3632 for (i = 0; i < QUEUE_DEPTH; i++)
3633 FPT_BL_Card[p_card].discQ_Tbl[i] = NULL;
3635 WR_HARPOON(port + hp_page_ctrl,
3636 (RD_HARPOON(port + hp_page_ctrl) & ~G_INT_DISABLE));
3640 /*---------------------------------------------------------------------
3642 * Function: FPT_ssenss
3644 * Description: Setup for the Auto Sense command.
3646 *---------------------------------------------------------------------*/
3647 static void FPT_ssenss(struct sccb_card *pCurrCard)
3650 struct sccb *currSCCB;
3652 currSCCB = pCurrCard->currentSCCB;
3654 currSCCB->Save_CdbLen = currSCCB->CdbLength;
3656 for (i = 0; i < 6; i++) {
3658 currSCCB->Save_Cdb[i] = currSCCB->Cdb[i];
3661 currSCCB->CdbLength = SIX_BYTE_CMD;
3662 currSCCB->Cdb[0] = REQUEST_SENSE;
3663 currSCCB->Cdb[1] = currSCCB->Cdb[1] & (unsigned char)0xE0; /*Keep LUN. */
3664 currSCCB->Cdb[2] = 0x00;
3665 currSCCB->Cdb[3] = 0x00;
3666 currSCCB->Cdb[4] = currSCCB->RequestSenseLength;
3667 currSCCB->Cdb[5] = 0x00;
3669 currSCCB->Sccb_XferCnt = (u32)currSCCB->RequestSenseLength;
3671 currSCCB->Sccb_ATC = 0x00;
3673 currSCCB->Sccb_XferState |= F_AUTO_SENSE;
3675 currSCCB->Sccb_XferState &= ~F_SG_XFER;
3677 currSCCB->Sccb_idmsg = currSCCB->Sccb_idmsg & ~(unsigned char)DISC_PRIV;
3679 currSCCB->ControlByte = 0x00;
3681 currSCCB->Sccb_MGRFlags &= F_STATUSLOADED;
3684 /*---------------------------------------------------------------------
3686 * Function: FPT_sxfrp
3688 * Description: Transfer data into the bit bucket until the device
3689 * decides to switch phase.
3691 *---------------------------------------------------------------------*/
3693 static void FPT_sxfrp(u32 p_port, unsigned char p_card)
3695 unsigned char curr_phz;
3697 DISABLE_AUTO(p_port);
3699 if (FPT_BL_Card[p_card].globalFlags & F_HOST_XFER_ACT) {
3701 FPT_hostDataXferAbort(p_port, p_card,
3702 FPT_BL_Card[p_card].currentSCCB);
3706 /* If the Automation handled the end of the transfer then do not
3707 match the phase or we will get out of sync with the ISR. */
3709 if (RDW_HARPOON((p_port + hp_intstat)) &
3710 (BUS_FREE | XFER_CNT_0 | AUTO_INT))
3713 WR_HARPOON(p_port + hp_xfercnt_0, 0x00);
3715 curr_phz = RD_HARPOON(p_port + hp_scsisig) & (unsigned char)S_SCSI_PHZ;
3717 WRW_HARPOON((p_port + hp_intstat), XFER_CNT_0);
3719 WR_HARPOON(p_port + hp_scsisig, curr_phz);
3721 while (!(RDW_HARPOON((p_port + hp_intstat)) & (BUS_FREE | RESET)) &&
3723 (RD_HARPOON(p_port + hp_scsisig) & (unsigned char)S_SCSI_PHZ)))
3725 if (curr_phz & (unsigned char)SCSI_IOBIT) {
3726 WR_HARPOON(p_port + hp_portctrl_0,
3727 (SCSI_PORT | HOST_PORT | SCSI_INBIT));
3729 if (!(RD_HARPOON(p_port + hp_xferstat) & FIFO_EMPTY)) {
3730 RD_HARPOON(p_port + hp_fifodata_0);
3733 WR_HARPOON(p_port + hp_portctrl_0,
3734 (SCSI_PORT | HOST_PORT | HOST_WRT));
3735 if (RD_HARPOON(p_port + hp_xferstat) & FIFO_EMPTY) {
3736 WR_HARPOON(p_port + hp_fifodata_0, 0xFA);
3739 } /* End of While loop for padding data I/O phase */
3741 while (!(RDW_HARPOON((p_port + hp_intstat)) & (BUS_FREE | RESET))) {
3742 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_REQ)
3746 WR_HARPOON(p_port + hp_portctrl_0,
3747 (SCSI_PORT | HOST_PORT | SCSI_INBIT));
3748 while (!(RD_HARPOON(p_port + hp_xferstat) & FIFO_EMPTY)) {
3749 RD_HARPOON(p_port + hp_fifodata_0);
3752 if (!(RDW_HARPOON((p_port + hp_intstat)) & (BUS_FREE | RESET))) {
3753 WR_HARPOON(p_port + hp_autostart_0,
3754 (AUTO_IMMED + DISCONNECT_START));
3755 while (!(RDW_HARPOON((p_port + hp_intstat)) & AUTO_INT)) {
3758 if (RDW_HARPOON((p_port + hp_intstat)) &
3759 (ICMD_COMP | ITAR_DISC))
3761 (RDW_HARPOON((p_port + hp_intstat)) &
3762 (BUS_FREE | RSEL))) ;
3766 /*---------------------------------------------------------------------
3768 * Function: FPT_schkdd
3770 * Description: Make sure data has been flushed from both FIFOs and abort
3771 * the operations if necessary.
3773 *---------------------------------------------------------------------*/
3775 static void FPT_schkdd(u32 port, unsigned char p_card)
3777 unsigned short TimeOutLoop;
3778 unsigned char sPhase;
3780 struct sccb *currSCCB;
3782 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3784 if ((currSCCB->Sccb_scsistat != DATA_OUT_ST) &&
3785 (currSCCB->Sccb_scsistat != DATA_IN_ST)) {
3789 if (currSCCB->Sccb_XferState & F_ODD_BALL_CNT) {
3791 currSCCB->Sccb_ATC += (currSCCB->Sccb_XferCnt - 1);
3793 currSCCB->Sccb_XferCnt = 1;
3795 currSCCB->Sccb_XferState &= ~F_ODD_BALL_CNT;
3796 WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
3797 WR_HARPOON(port + hp_xferstat, 0x00);
3802 currSCCB->Sccb_ATC += currSCCB->Sccb_XferCnt;
3804 currSCCB->Sccb_XferCnt = 0;
3807 if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
3808 (currSCCB->HostStatus == SCCB_COMPLETE)) {
3810 currSCCB->HostStatus = SCCB_PARITY_ERR;
3811 WRW_HARPOON((port + hp_intstat), PARITY);
3814 FPT_hostDataXferAbort(port, p_card, currSCCB);
3816 while (RD_HARPOON(port + hp_scsisig) & SCSI_ACK) {
3821 while (RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY) {
3822 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
3825 if (RD_HARPOON(port + hp_offsetctr) & (unsigned char)0x1F) {
3828 if (RDW_HARPOON((port + hp_intstat)) & RESET) {
3831 if ((RD_HARPOON(port + hp_scsisig) & SCSI_REQ)
3832 || (TimeOutLoop++ > 0x3000))
3836 sPhase = RD_HARPOON(port + hp_scsisig) & (SCSI_BSY | S_SCSI_PHZ);
3837 if ((!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY)) ||
3838 (RD_HARPOON(port + hp_offsetctr) & (unsigned char)0x1F) ||
3839 (sPhase == (SCSI_BSY | S_DATAO_PH)) ||
3840 (sPhase == (SCSI_BSY | S_DATAI_PH))) {
3842 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3844 if (!(currSCCB->Sccb_XferState & F_ALL_XFERRED)) {
3845 if (currSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
3846 FPT_phaseDataIn(port, p_card);
3850 FPT_phaseDataOut(port, p_card);
3853 FPT_sxfrp(port, p_card);
3854 if (!(RDW_HARPOON((port + hp_intstat)) &
3855 (BUS_FREE | ICMD_COMP | ITAR_DISC | RESET))) {
3856 WRW_HARPOON((port + hp_intstat), AUTO_INT);
3857 FPT_phaseDecode(port, p_card);
3864 WR_HARPOON(port + hp_portctrl_0, 0x00);
3868 /*---------------------------------------------------------------------
3870 * Function: FPT_sinits
3872 * Description: Setup SCCB manager fields in this SCCB.
3874 *---------------------------------------------------------------------*/
3876 static void FPT_sinits(struct sccb *p_sccb, unsigned char p_card)
3878 struct sccb_mgr_tar_info *currTar_Info;
3880 if ((p_sccb->TargID >= MAX_SCSI_TAR) || (p_sccb->Lun >= MAX_LUN)) {
3883 currTar_Info = &FPT_sccbMgrTbl[p_card][p_sccb->TargID];
3885 p_sccb->Sccb_XferState = 0x00;
3886 p_sccb->Sccb_XferCnt = p_sccb->DataLength;
3888 if ((p_sccb->OperationCode == SCATTER_GATHER_COMMAND) ||
3889 (p_sccb->OperationCode == RESIDUAL_SG_COMMAND)) {
3891 p_sccb->Sccb_SGoffset = 0;
3892 p_sccb->Sccb_XferState = F_SG_XFER;
3893 p_sccb->Sccb_XferCnt = 0x00;
3896 if (p_sccb->DataLength == 0x00)
3898 p_sccb->Sccb_XferState |= F_ALL_XFERRED;
3900 if (p_sccb->ControlByte & F_USE_CMD_Q) {
3901 if ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) == TAG_Q_REJECT)
3902 p_sccb->ControlByte &= ~F_USE_CMD_Q;
3905 currTar_Info->TarStatus |= TAG_Q_TRYING;
3908 /* For !single SCSI device in system & device allow Disconnect
3909 or command is tag_q type then send Cmd with Disconnect Enable
3910 else send Cmd with Disconnect Disable */
3913 if (((!(FPT_BL_Card[p_card].globalFlags & F_SINGLE_DEVICE)) &&
3914 (currTar_Info->TarStatus & TAR_ALLOW_DISC)) ||
3915 (currTar_Info->TarStatus & TAG_Q_TRYING)) {
3917 if ((currTar_Info->TarStatus & TAR_ALLOW_DISC) ||
3918 (currTar_Info->TarStatus & TAG_Q_TRYING)) {
3919 p_sccb->Sccb_idmsg = IDENTIFY(true, p_sccb->Lun);
3921 p_sccb->Sccb_idmsg = IDENTIFY(false, p_sccb->Lun);
3924 p_sccb->HostStatus = 0x00;
3925 p_sccb->TargetStatus = 0x00;
3926 p_sccb->Sccb_tag = 0x00;
3927 p_sccb->Sccb_MGRFlags = 0x00;
3928 p_sccb->Sccb_sgseg = 0x00;
3929 p_sccb->Sccb_ATC = 0x00;
3930 p_sccb->Sccb_savedATC = 0x00;
3932 p_sccb->SccbVirtDataPtr = 0x00;
3933 p_sccb->Sccb_forwardlink = NULL;
3934 p_sccb->Sccb_backlink = NULL;
3936 p_sccb->Sccb_scsistat = BUS_FREE_ST;
3937 p_sccb->SccbStatus = SCCB_IN_PROCESS;
3938 p_sccb->Sccb_scsimsg = NOP;
3942 /*---------------------------------------------------------------------
3944 * Function: Phase Decode
3946 * Description: Determine the phase and call the appropriate function.
3948 *---------------------------------------------------------------------*/
3950 static void FPT_phaseDecode(u32 p_port, unsigned char p_card)
3952 unsigned char phase_ref;
3953 void (*phase) (u32, unsigned char);
3955 DISABLE_AUTO(p_port);
3958 (unsigned char)(RD_HARPOON(p_port + hp_scsisig) & S_SCSI_PHZ);
3960 phase = FPT_s_PhaseTbl[phase_ref];
3962 (*phase) (p_port, p_card); /* Call the correct phase func */
3965 /*---------------------------------------------------------------------
3967 * Function: Data Out Phase
3969 * Description: Start up both the BusMaster and Xbow.
3971 *---------------------------------------------------------------------*/
3973 static void FPT_phaseDataOut(u32 port, unsigned char p_card)
3976 struct sccb *currSCCB;
3978 currSCCB = FPT_BL_Card[p_card].currentSCCB;
3979 if (currSCCB == NULL) {
3980 return; /* Exit if No SCCB record */
3983 currSCCB->Sccb_scsistat = DATA_OUT_ST;
3984 currSCCB->Sccb_XferState &= ~(F_HOST_XFER_DIR | F_NO_DATA_YET);
3986 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3988 WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
3990 WR_HARPOON(port + hp_autostart_0, (END_DATA + END_DATA_START));
3992 FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
3994 if (currSCCB->Sccb_XferCnt == 0) {
3996 if ((currSCCB->ControlByte & SCCB_DATA_XFER_OUT) &&
3997 (currSCCB->HostStatus == SCCB_COMPLETE))
3998 currSCCB->HostStatus = SCCB_DATA_OVER_RUN;
4000 FPT_sxfrp(port, p_card);
4001 if (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | RESET)))
4002 FPT_phaseDecode(port, p_card);
4006 /*---------------------------------------------------------------------
4008 * Function: Data In Phase
4010 * Description: Startup the BusMaster and the XBOW.
4012 *---------------------------------------------------------------------*/
4014 static void FPT_phaseDataIn(u32 port, unsigned char p_card)
4017 struct sccb *currSCCB;
4019 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4021 if (currSCCB == NULL) {
4022 return; /* Exit if No SCCB record */
4025 currSCCB->Sccb_scsistat = DATA_IN_ST;
4026 currSCCB->Sccb_XferState |= F_HOST_XFER_DIR;
4027 currSCCB->Sccb_XferState &= ~F_NO_DATA_YET;
4029 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
4031 WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
4033 WR_HARPOON(port + hp_autostart_0, (END_DATA + END_DATA_START));
4035 FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
4037 if (currSCCB->Sccb_XferCnt == 0) {
4039 if ((currSCCB->ControlByte & SCCB_DATA_XFER_IN) &&
4040 (currSCCB->HostStatus == SCCB_COMPLETE))
4041 currSCCB->HostStatus = SCCB_DATA_OVER_RUN;
4043 FPT_sxfrp(port, p_card);
4044 if (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | RESET)))
4045 FPT_phaseDecode(port, p_card);
4050 /*---------------------------------------------------------------------
4052 * Function: Command Phase
4054 * Description: Load the CDB into the automation and start it up.
4056 *---------------------------------------------------------------------*/
4058 static void FPT_phaseCommand(u32 p_port, unsigned char p_card)
4060 struct sccb *currSCCB;
4064 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4066 if (currSCCB->OperationCode == RESET_COMMAND) {
4068 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4069 currSCCB->CdbLength = SIX_BYTE_CMD;
4072 WR_HARPOON(p_port + hp_scsisig, 0x00);
4074 ARAM_ACCESS(p_port);
4076 cdb_reg = p_port + CMD_STRT;
4078 for (i = 0; i < currSCCB->CdbLength; i++) {
4080 if (currSCCB->OperationCode == RESET_COMMAND)
4082 WRW_HARPOON(cdb_reg, (MPM_OP + ACOMMAND + 0x00));
4085 WRW_HARPOON(cdb_reg,
4086 (MPM_OP + ACOMMAND + currSCCB->Cdb[i]));
4090 if (currSCCB->CdbLength != TWELVE_BYTE_CMD)
4091 WRW_HARPOON(cdb_reg, (BRH_OP + ALWAYS + NP));
4093 WR_HARPOON(p_port + hp_portctrl_0, (SCSI_PORT));
4095 currSCCB->Sccb_scsistat = COMMAND_ST;
4097 WR_HARPOON(p_port + hp_autostart_3, (AUTO_IMMED | CMD_ONLY_STRT));
4098 SGRAM_ACCESS(p_port);
4101 /*---------------------------------------------------------------------
4103 * Function: Status phase
4105 * Description: Bring in the status and command complete message bytes
4107 *---------------------------------------------------------------------*/
4109 static void FPT_phaseStatus(u32 port, unsigned char p_card)
4111 /* Start-up the automation to finish off this command and let the
4112 isr handle the interrupt for command complete when it comes in.
4113 We could wait here for the interrupt to be generated?
4116 WR_HARPOON(port + hp_scsisig, 0x00);
4118 WR_HARPOON(port + hp_autostart_0, (AUTO_IMMED + END_DATA_START));
4121 /*---------------------------------------------------------------------
4123 * Function: Phase Message Out
4125 * Description: Send out our message (if we have one) and handle whatever
4128 *---------------------------------------------------------------------*/
4130 static void FPT_phaseMsgOut(u32 port, unsigned char p_card)
4132 unsigned char message, scsiID;
4133 struct sccb *currSCCB;
4134 struct sccb_mgr_tar_info *currTar_Info;
4136 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4138 if (currSCCB != NULL) {
4140 message = currSCCB->Sccb_scsimsg;
4141 scsiID = currSCCB->TargID;
4143 if (message == TARGET_RESET) {
4145 currTar_Info = &FPT_sccbMgrTbl[p_card][scsiID];
4146 currTar_Info->TarSyncCtrl = 0;
4147 FPT_sssyncv(port, scsiID, NARROW_SCSI, currTar_Info);
4149 if (FPT_sccbMgrTbl[p_card][scsiID].
4150 TarEEValue & EE_SYNC_MASK) {
4152 FPT_sccbMgrTbl[p_card][scsiID].TarStatus &=
4157 if (FPT_sccbMgrTbl[p_card][scsiID].
4158 TarEEValue & EE_WIDE_SCSI) {
4160 FPT_sccbMgrTbl[p_card][scsiID].TarStatus &=
4164 FPT_queueFlushSccb(p_card, SCCB_COMPLETE);
4165 FPT_SccbMgrTableInitTarget(p_card, scsiID);
4166 } else if (currSCCB->Sccb_scsistat == ABORT_ST) {
4167 currSCCB->HostStatus = SCCB_COMPLETE;
4168 if (FPT_BL_Card[p_card].discQ_Tbl[currSCCB->Sccb_tag] !=
4170 FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4172 FPT_sccbMgrTbl[p_card][scsiID].TarTagQ_Cnt--;
4177 else if (currSCCB->Sccb_scsistat < COMMAND_ST) {
4179 if (message == NOP) {
4180 currSCCB->Sccb_MGRFlags |= F_DEV_SELECTED;
4182 FPT_ssel(port, p_card);
4187 if (message == ABORT_TASK_SET)
4189 FPT_queueFlushSccb(p_card, SCCB_COMPLETE);
4193 message = ABORT_TASK_SET;
4196 WRW_HARPOON((port + hp_intstat), (BUS_FREE | PHASE | XFER_CNT_0));
4198 WR_HARPOON(port + hp_portctrl_0, SCSI_BUS_EN);
4200 WR_HARPOON(port + hp_scsidata_0, message);
4202 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
4206 WR_HARPOON(port + hp_portctrl_0, 0x00);
4208 if ((message == ABORT_TASK_SET) || (message == TARGET_RESET) ||
4209 (message == ABORT_TASK)) {
4211 while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | PHASE))) {
4214 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
4215 WRW_HARPOON((port + hp_intstat), BUS_FREE);
4217 if (currSCCB != NULL) {
4219 if ((FPT_BL_Card[p_card].
4220 globalFlags & F_CONLUN_IO)
4222 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4223 TarStatus & TAR_TAG_Q_MASK) !=
4225 FPT_sccbMgrTbl[p_card][currSCCB->
4227 TarLUNBusy[currSCCB->Lun] = 0;
4229 FPT_sccbMgrTbl[p_card][currSCCB->
4233 FPT_queueCmdComplete(&FPT_BL_Card[p_card],
4238 FPT_BL_Card[p_card].globalFlags |=
4245 FPT_sxfrp(port, p_card);
4251 if (message == MSG_PARITY_ERROR) {
4252 currSCCB->Sccb_scsimsg = NOP;
4253 WR_HARPOON(port + hp_autostart_1,
4254 (AUTO_IMMED + DISCONNECT_START));
4256 FPT_sxfrp(port, p_card);
4261 /*---------------------------------------------------------------------
4263 * Function: Message In phase
4265 * Description: Bring in the message and determine what to do with it.
4267 *---------------------------------------------------------------------*/
4269 static void FPT_phaseMsgIn(u32 port, unsigned char p_card)
4271 unsigned char message;
4272 struct sccb *currSCCB;
4274 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4276 if (FPT_BL_Card[p_card].globalFlags & F_HOST_XFER_ACT) {
4278 FPT_phaseChkFifo(port, p_card);
4281 message = RD_HARPOON(port + hp_scsidata_0);
4282 if ((message == DISCONNECT) || (message == SAVE_POINTERS)) {
4284 WR_HARPOON(port + hp_autostart_1,
4285 (AUTO_IMMED + END_DATA_START));
4291 message = FPT_sfm(port, currSCCB);
4294 FPT_sdecm(message, port, p_card);
4297 if (currSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
4299 WR_HARPOON(port + hp_autostart_1,
4300 (AUTO_IMMED + DISCONNECT_START));
4306 /*---------------------------------------------------------------------
4308 * Function: Illegal phase
4310 * Description: Target switched to some illegal phase, so all we can do
4311 * is report an error back to the host (if that is possible)
4312 * and send an ABORT message to the misbehaving target.
4314 *---------------------------------------------------------------------*/
4316 static void FPT_phaseIllegal(u32 port, unsigned char p_card)
4318 struct sccb *currSCCB;
4320 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4322 WR_HARPOON(port + hp_scsisig, RD_HARPOON(port + hp_scsisig));
4323 if (currSCCB != NULL) {
4325 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4326 currSCCB->Sccb_scsistat = ABORT_ST;
4327 currSCCB->Sccb_scsimsg = ABORT_TASK_SET;
4330 ACCEPT_MSG_ATN(port);
4333 /*---------------------------------------------------------------------
4335 * Function: Phase Check FIFO
4337 * Description: Make sure data has been flushed from both FIFOs and abort
4338 * the operations if necessary.
4340 *---------------------------------------------------------------------*/
4342 static void FPT_phaseChkFifo(u32 port, unsigned char p_card)
4345 struct sccb *currSCCB;
4347 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4349 if (currSCCB->Sccb_scsistat == DATA_IN_ST) {
4351 while ((!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY)) &&
4352 (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)) {
4355 if (!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY)) {
4356 currSCCB->Sccb_ATC += currSCCB->Sccb_XferCnt;
4358 currSCCB->Sccb_XferCnt = 0;
4360 if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
4361 (currSCCB->HostStatus == SCCB_COMPLETE)) {
4362 currSCCB->HostStatus = SCCB_PARITY_ERR;
4363 WRW_HARPOON((port + hp_intstat), PARITY);
4366 FPT_hostDataXferAbort(port, p_card, currSCCB);
4368 FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
4370 while ((!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY))
4371 && (RD_HARPOON(port + hp_ext_status) &
4378 /*End Data In specific code. */
4379 GET_XFER_CNT(port, xfercnt);
4381 WR_HARPOON(port + hp_xfercnt_0, 0x00);
4383 WR_HARPOON(port + hp_portctrl_0, 0x00);
4385 currSCCB->Sccb_ATC += (currSCCB->Sccb_XferCnt - xfercnt);
4387 currSCCB->Sccb_XferCnt = xfercnt;
4389 if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
4390 (currSCCB->HostStatus == SCCB_COMPLETE)) {
4392 currSCCB->HostStatus = SCCB_PARITY_ERR;
4393 WRW_HARPOON((port + hp_intstat), PARITY);
4396 FPT_hostDataXferAbort(port, p_card, currSCCB);
4398 WR_HARPOON(port + hp_fifowrite, 0x00);
4399 WR_HARPOON(port + hp_fiforead, 0x00);
4400 WR_HARPOON(port + hp_xferstat, 0x00);
4402 WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
4405 /*---------------------------------------------------------------------
4407 * Function: Phase Bus Free
4409 * Description: We just went bus free so figure out if it was
4410 * because of command complete or from a disconnect.
4412 *---------------------------------------------------------------------*/
4413 static void FPT_phaseBusFree(u32 port, unsigned char p_card)
4415 struct sccb *currSCCB;
4417 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4419 if (currSCCB != NULL) {
4423 if (currSCCB->OperationCode == RESET_COMMAND) {
4425 if ((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4426 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4427 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
4428 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4429 TarLUNBusy[currSCCB->Lun] = 0;
4431 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4434 FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB,
4437 FPT_queueSearchSelect(&FPT_BL_Card[p_card], p_card);
4441 else if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
4442 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus |=
4443 (unsigned char)SYNC_SUPPORTED;
4444 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4448 else if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
4449 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus =
4450 (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4451 TarStatus & ~WIDE_ENABLED) | WIDE_NEGOCIATED;
4453 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4457 else if (currSCCB->Sccb_scsistat == SELECT_Q_ST) {
4458 /* Make sure this is not a phony BUS_FREE. If we were
4459 reselected or if BUSY is NOT on then this is a
4460 valid BUS FREE. SRR Wednesday, 5/10/1995. */
4462 if ((!(RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) ||
4463 (RDW_HARPOON((port + hp_intstat)) & RSEL)) {
4464 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4465 TarStatus &= ~TAR_TAG_Q_MASK;
4466 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4467 TarStatus |= TAG_Q_REJECT;
4477 currSCCB->Sccb_scsistat = BUS_FREE_ST;
4479 if (!currSCCB->HostStatus) {
4480 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4483 if ((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4484 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4485 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
4486 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4487 TarLUNBusy[currSCCB->Lun] = 0;
4489 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4492 FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB,
4497 FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4499 } /*end if !=null */
4502 /*---------------------------------------------------------------------
4504 * Function: Auto Load Default Map
4506 * Description: Load the Automation RAM with the default map values.
4508 *---------------------------------------------------------------------*/
4509 static void FPT_autoLoadDefaultMap(u32 p_port)
4513 ARAM_ACCESS(p_port);
4514 map_addr = p_port + hp_aramBase;
4516 WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0xC0)); /*ID MESSAGE */
4518 WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0x20)); /*SIMPLE TAG QUEUEING MSG */
4520 WRW_HARPOON(map_addr, RAT_OP); /*RESET ATTENTION */
4522 WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0x00)); /*TAG ID MSG */
4524 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 0 */
4526 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 1 */
4528 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 2 */
4530 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 3 */
4532 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 4 */
4534 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 5 */
4536 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 6 */
4538 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 7 */
4540 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 8 */
4542 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 9 */
4544 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 10 */
4546 WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00)); /*CDB BYTE 11 */
4548 WRW_HARPOON(map_addr, (CPE_OP + ADATA_OUT + DINT)); /*JUMP IF DATA OUT */
4550 WRW_HARPOON(map_addr, (TCB_OP + FIFO_0 + DI)); /*JUMP IF NO DATA IN FIFO */
4551 map_addr += 2; /*This means AYNC DATA IN */
4552 WRW_HARPOON(map_addr, (SSI_OP + SSI_IDO_STRT)); /*STOP AND INTERRUPT */
4554 WRW_HARPOON(map_addr, (CPE_OP + ADATA_IN + DINT)); /*JUMP IF NOT DATA IN PHZ */
4556 WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + ST)); /*IF NOT MSG IN CHECK 4 DATA IN */
4558 WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x02)); /*SAVE DATA PTR MSG? */
4560 WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + DC)); /*GO CHECK FOR DISCONNECT MSG */
4562 WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_AR1)); /*SAVE DATA PTRS MSG */
4564 WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + ST)); /*IF NOT MSG IN CHECK DATA IN */
4566 WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x04)); /*DISCONNECT MSG? */
4568 WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + UNKNWN)); /*UKNKNOWN MSG */
4570 WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_BUCKET)); /*XFER DISCONNECT MSG */
4572 WRW_HARPOON(map_addr, (SSI_OP + SSI_ITAR_DISC)); /*STOP AND INTERRUPT */
4574 WRW_HARPOON(map_addr, (CPN_OP + ASTATUS + UNKNWN)); /*JUMP IF NOT STATUS PHZ. */
4576 WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_AR0)); /*GET STATUS BYTE */
4578 WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + CC)); /*ERROR IF NOT MSG IN PHZ */
4580 WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x00)); /*CHECK FOR CMD COMPLETE MSG. */
4582 WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + CC)); /*ERROR IF NOT CMD COMPLETE MSG. */
4584 WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_BUCKET)); /*GET CMD COMPLETE MSG */
4586 WRW_HARPOON(map_addr, (SSI_OP + SSI_ICMD_COMP)); /*END OF COMMAND */
4589 WRW_HARPOON(map_addr, (SSI_OP + SSI_IUNKWN)); /*RECEIVED UNKNOWN MSG BYTE */
4591 WRW_HARPOON(map_addr, (SSI_OP + SSI_INO_CC)); /*NO COMMAND COMPLETE AFTER STATUS */
4593 WRW_HARPOON(map_addr, (SSI_OP + SSI_ITICKLE)); /*BIOS Tickled the Mgr */
4595 WRW_HARPOON(map_addr, (SSI_OP + SSI_IRFAIL)); /*EXPECTED ID/TAG MESSAGES AND */
4596 map_addr += 2; /* DIDN'T GET ONE */
4597 WRW_HARPOON(map_addr, (CRR_OP + AR3 + S_IDREG)); /* comp SCSI SEL ID & AR3 */
4599 WRW_HARPOON(map_addr, (BRH_OP + EQUAL + 0x00)); /*SEL ID OK then Conti. */
4601 WRW_HARPOON(map_addr, (SSI_OP + SSI_INO_CC)); /*NO COMMAND COMPLETE AFTER STATUS */
4603 SGRAM_ACCESS(p_port);
4606 /*---------------------------------------------------------------------
4608 * Function: Auto Command Complete
4610 * Description: Post command back to host and find another command
4613 *---------------------------------------------------------------------*/
4615 static void FPT_autoCmdCmplt(u32 p_port, unsigned char p_card)
4617 struct sccb *currSCCB;
4618 unsigned char status_byte;
4620 currSCCB = FPT_BL_Card[p_card].currentSCCB;
4622 status_byte = RD_HARPOON(p_port + hp_gp_reg_0);
4624 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarLUN_CA = 0;
4626 if (status_byte != SAM_STAT_GOOD) {
4628 if (status_byte == SAM_STAT_TASK_SET_FULL) {
4630 if (((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4631 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4632 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))) {
4633 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4634 TarLUNBusy[currSCCB->Lun] = 1;
4635 if (FPT_BL_Card[p_card].discQCount != 0)
4636 FPT_BL_Card[p_card].discQCount--;
4637 FPT_BL_Card[p_card].
4638 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4640 LunDiscQ_Idx[currSCCB->Lun]] =
4643 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4645 if (currSCCB->Sccb_tag) {
4646 if (FPT_BL_Card[p_card].discQCount != 0)
4647 FPT_BL_Card[p_card].
4649 FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4653 if (FPT_BL_Card[p_card].discQCount != 0)
4654 FPT_BL_Card[p_card].
4656 FPT_BL_Card[p_card].
4657 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4659 LunDiscQ_Idx[0]] = NULL;
4663 currSCCB->Sccb_MGRFlags |= F_STATUSLOADED;
4665 FPT_queueSelectFail(&FPT_BL_Card[p_card], p_card);
4670 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
4671 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus |=
4672 (unsigned char)SYNC_SUPPORTED;
4674 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4676 FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4678 if (((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4679 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4680 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))) {
4681 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4682 TarLUNBusy[currSCCB->Lun] = 1;
4683 if (FPT_BL_Card[p_card].discQCount != 0)
4684 FPT_BL_Card[p_card].discQCount--;
4685 FPT_BL_Card[p_card].
4686 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4688 LunDiscQ_Idx[currSCCB->Lun]] =
4691 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4693 if (currSCCB->Sccb_tag) {
4694 if (FPT_BL_Card[p_card].discQCount != 0)
4695 FPT_BL_Card[p_card].
4697 FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4701 if (FPT_BL_Card[p_card].discQCount != 0)
4702 FPT_BL_Card[p_card].
4704 FPT_BL_Card[p_card].
4705 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4707 LunDiscQ_Idx[0]] = NULL;
4714 if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
4716 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus =
4717 (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4718 TarStatus & ~WIDE_ENABLED) | WIDE_NEGOCIATED;
4720 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4722 FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4724 if (((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4725 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4726 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))) {
4727 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4728 TarLUNBusy[currSCCB->Lun] = 1;
4729 if (FPT_BL_Card[p_card].discQCount != 0)
4730 FPT_BL_Card[p_card].discQCount--;
4731 FPT_BL_Card[p_card].
4732 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4734 LunDiscQ_Idx[currSCCB->Lun]] =
4737 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4739 if (currSCCB->Sccb_tag) {
4740 if (FPT_BL_Card[p_card].discQCount != 0)
4741 FPT_BL_Card[p_card].
4743 FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4747 if (FPT_BL_Card[p_card].discQCount != 0)
4748 FPT_BL_Card[p_card].
4750 FPT_BL_Card[p_card].
4751 discQ_Tbl[FPT_sccbMgrTbl[p_card]
4753 LunDiscQ_Idx[0]] = NULL;
4760 if (status_byte == SAM_STAT_CHECK_CONDITION) {
4761 if (FPT_BL_Card[p_card].globalFlags & F_DO_RENEGO) {
4762 if (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4763 TarEEValue & EE_SYNC_MASK) {
4764 FPT_sccbMgrTbl[p_card][currSCCB->
4766 TarStatus &= ~TAR_SYNC_MASK;
4768 if (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4769 TarEEValue & EE_WIDE_SCSI) {
4770 FPT_sccbMgrTbl[p_card][currSCCB->
4772 TarStatus &= ~TAR_WIDE_MASK;
4777 if (!(currSCCB->Sccb_XferState & F_AUTO_SENSE)) {
4779 currSCCB->SccbStatus = SCCB_ERROR;
4780 currSCCB->TargetStatus = status_byte;
4782 if (status_byte == SAM_STAT_CHECK_CONDITION) {
4784 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4787 if (currSCCB->RequestSenseLength !=
4788 NO_AUTO_REQUEST_SENSE) {
4790 if (currSCCB->RequestSenseLength == 0)
4791 currSCCB->RequestSenseLength =
4794 FPT_ssenss(&FPT_BL_Card[p_card]);
4795 FPT_BL_Card[p_card].globalFlags |=
4798 if (((FPT_BL_Card[p_card].
4799 globalFlags & F_CONLUN_IO)
4801 ((FPT_sccbMgrTbl[p_card]
4803 TarStatus & TAR_TAG_Q_MASK) !=
4805 FPT_sccbMgrTbl[p_card]
4807 TarLUNBusy[currSCCB->Lun] =
4809 if (FPT_BL_Card[p_card].
4811 FPT_BL_Card[p_card].
4813 FPT_BL_Card[p_card].
4814 discQ_Tbl[FPT_sccbMgrTbl
4822 FPT_sccbMgrTbl[p_card]
4825 if (currSCCB->Sccb_tag) {
4826 if (FPT_BL_Card[p_card].
4831 FPT_BL_Card[p_card].
4832 discQ_Tbl[currSCCB->
4836 if (FPT_BL_Card[p_card].
4841 FPT_BL_Card[p_card].
4856 if ((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
4857 ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4858 TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
4859 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarLUNBusy[currSCCB->
4862 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarLUNBusy[0] = 0;
4864 FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB, p_card);
4867 #define SHORT_WAIT 0x0000000F
4868 #define LONG_WAIT 0x0000FFFFL
4870 /*---------------------------------------------------------------------
4872 * Function: Data Transfer Processor
4874 * Description: This routine performs two tasks.
4875 * (1) Start data transfer by calling HOST_DATA_XFER_START
4876 * function. Once data transfer is started, (2) Depends
4877 * on the type of data transfer mode Scatter/Gather mode
4878 * or NON Scatter/Gather mode. In NON Scatter/Gather mode,
4879 * this routine checks Sccb_MGRFlag (F_HOST_XFER_ACT bit) for
4880 * data transfer done. In Scatter/Gather mode, this routine
4881 * checks bus master command complete and dual rank busy
4882 * bit to keep chaining SC transfer command. Similarly,
4883 * in Scatter/Gather mode, it checks Sccb_MGRFlag
4884 * (F_HOST_XFER_ACT bit) for data transfer done.
4886 *---------------------------------------------------------------------*/
4888 static void FPT_dataXferProcessor(u32 port, struct sccb_card *pCurrCard)
4890 struct sccb *currSCCB;
4892 currSCCB = pCurrCard->currentSCCB;
4894 if (currSCCB->Sccb_XferState & F_SG_XFER) {
4895 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
4897 currSCCB->Sccb_sgseg += (unsigned char)SG_BUF_CNT;
4898 currSCCB->Sccb_SGoffset = 0x00;
4900 pCurrCard->globalFlags |= F_HOST_XFER_ACT;
4902 FPT_busMstrSGDataXferStart(port, currSCCB);
4906 if (!(pCurrCard->globalFlags & F_HOST_XFER_ACT)) {
4907 pCurrCard->globalFlags |= F_HOST_XFER_ACT;
4909 FPT_busMstrDataXferStart(port, currSCCB);
4914 /*---------------------------------------------------------------------
4916 * Function: BusMaster Scatter Gather Data Transfer Start
4920 *---------------------------------------------------------------------*/
4921 static void FPT_busMstrSGDataXferStart(u32 p_port, struct sccb *pcurrSCCB)
4923 u32 count, addr, tmpSGCnt;
4924 unsigned int sg_index;
4925 unsigned char sg_count, i;
4927 struct blogic_sg_seg *segp;
4929 if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR)
4930 count = ((u32)HOST_RD_CMD) << 24;
4932 count = ((u32)HOST_WRT_CMD) << 24;
4936 sg_index = pcurrSCCB->Sccb_sgseg;
4937 reg_offset = hp_aramBase;
4939 i = (unsigned char)(RD_HARPOON(p_port + hp_page_ctrl) &
4940 ~(SGRAM_ARAM | SCATTER_EN));
4942 WR_HARPOON(p_port + hp_page_ctrl, i);
4944 while ((sg_count < (unsigned char)SG_BUF_CNT) &&
4945 ((sg_index * (unsigned int)SG_ELEMENT_SIZE) <
4946 pcurrSCCB->DataLength)) {
4948 segp = (struct blogic_sg_seg *)(pcurrSCCB->DataPointer) +
4950 tmpSGCnt += segp->segbytes;
4951 count |= segp->segbytes;
4952 addr = segp->segdata;
4954 if ((!sg_count) && (pcurrSCCB->Sccb_SGoffset)) {
4956 ((count & 0x00FFFFFFL) - pcurrSCCB->Sccb_SGoffset);
4958 (count & 0xFF000000L) | pcurrSCCB->Sccb_SGoffset;
4959 tmpSGCnt = count & 0x00FFFFFFL;
4962 WR_HARP32(p_port, reg_offset, addr);
4965 WR_HARP32(p_port, reg_offset, count);
4968 count &= 0xFF000000L;
4974 pcurrSCCB->Sccb_XferCnt = tmpSGCnt;
4976 WR_HARPOON(p_port + hp_sg_addr, (sg_count << 4));
4978 if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
4980 WR_HARP32(p_port, hp_xfercnt_0, tmpSGCnt);
4982 WR_HARPOON(p_port + hp_portctrl_0,
4983 (DMA_PORT | SCSI_PORT | SCSI_INBIT));
4984 WR_HARPOON(p_port + hp_scsisig, S_DATAI_PH);
4989 if ((!(RD_HARPOON(p_port + hp_synctarg_0) & NARROW_SCSI)) &&
4990 (tmpSGCnt & 0x000000001)) {
4992 pcurrSCCB->Sccb_XferState |= F_ODD_BALL_CNT;
4996 WR_HARP32(p_port, hp_xfercnt_0, tmpSGCnt);
4998 WR_HARPOON(p_port + hp_portctrl_0,
4999 (SCSI_PORT | DMA_PORT | DMA_RD));
5000 WR_HARPOON(p_port + hp_scsisig, S_DATAO_PH);
5003 WR_HARPOON(p_port + hp_page_ctrl, (unsigned char)(i | SCATTER_EN));
5007 /*---------------------------------------------------------------------
5009 * Function: BusMaster Data Transfer Start
5013 *---------------------------------------------------------------------*/
5014 static void FPT_busMstrDataXferStart(u32 p_port, struct sccb *pcurrSCCB)
5018 if (!(pcurrSCCB->Sccb_XferState & F_AUTO_SENSE)) {
5020 count = pcurrSCCB->Sccb_XferCnt;
5022 addr = (u32)(unsigned long)pcurrSCCB->DataPointer + pcurrSCCB->Sccb_ATC;
5026 addr = pcurrSCCB->SensePointer;
5027 count = pcurrSCCB->RequestSenseLength;
5031 HP_SETUP_ADDR_CNT(p_port, addr, count);
5033 if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
5035 WR_HARPOON(p_port + hp_portctrl_0,
5036 (DMA_PORT | SCSI_PORT | SCSI_INBIT));
5037 WR_HARPOON(p_port + hp_scsisig, S_DATAI_PH);
5039 WR_HARPOON(p_port + hp_xfer_cmd,
5040 (XFER_DMA_HOST | XFER_HOST_AUTO | XFER_DMA_8BIT));
5045 WR_HARPOON(p_port + hp_portctrl_0,
5046 (SCSI_PORT | DMA_PORT | DMA_RD));
5047 WR_HARPOON(p_port + hp_scsisig, S_DATAO_PH);
5049 WR_HARPOON(p_port + hp_xfer_cmd,
5050 (XFER_HOST_DMA | XFER_HOST_AUTO | XFER_DMA_8BIT));
5055 /*---------------------------------------------------------------------
5057 * Function: BusMaster Timeout Handler
5059 * Description: This function is called after a bus master command busy time
5060 * out is detected. This routines issue halt state machine
5061 * with a software time out for command busy. If command busy
5062 * is still asserted at the end of the time out, it issues
5063 * hard abort with another software time out. It hard abort
5064 * command busy is also time out, it'll just give up.
5066 *---------------------------------------------------------------------*/
5067 static unsigned char FPT_busMstrTimeOut(u32 p_port)
5069 unsigned long timeout;
5071 timeout = LONG_WAIT;
5073 WR_HARPOON(p_port + hp_sys_ctrl, HALT_MACH);
5075 while ((!(RD_HARPOON(p_port + hp_ext_status) & CMD_ABORTED))
5079 if (RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY) {
5080 WR_HARPOON(p_port + hp_sys_ctrl, HARD_ABORT);
5082 timeout = LONG_WAIT;
5083 while ((RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY)
5088 RD_HARPOON(p_port + hp_int_status); /*Clear command complete */
5090 if (RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY) {
5099 /*---------------------------------------------------------------------
5101 * Function: Host Data Transfer Abort
5103 * Description: Abort any in progress transfer.
5105 *---------------------------------------------------------------------*/
5106 static void FPT_hostDataXferAbort(u32 port, unsigned char p_card,
5107 struct sccb *pCurrSCCB)
5110 unsigned long timeout;
5111 unsigned long remain_cnt;
5113 struct blogic_sg_seg *segp;
5115 FPT_BL_Card[p_card].globalFlags &= ~F_HOST_XFER_ACT;
5117 if (pCurrSCCB->Sccb_XferState & F_AUTO_SENSE) {
5119 if (!(RD_HARPOON(port + hp_int_status) & INT_CMD_COMPL)) {
5121 WR_HARPOON(port + hp_bm_ctrl,
5122 (RD_HARPOON(port + hp_bm_ctrl) |
5124 timeout = LONG_WAIT;
5126 while ((RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)
5130 WR_HARPOON(port + hp_bm_ctrl,
5131 (RD_HARPOON(port + hp_bm_ctrl) &
5134 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5136 if (FPT_busMstrTimeOut(port)) {
5138 if (pCurrSCCB->HostStatus == 0x00)
5140 pCurrSCCB->HostStatus =
5145 if (RD_HARPOON(port + hp_int_status) &
5148 if (RD_HARPOON(port + hp_ext_status) &
5151 if (pCurrSCCB->HostStatus ==
5154 pCurrSCCB->HostStatus =
5161 else if (pCurrSCCB->Sccb_XferCnt) {
5163 if (pCurrSCCB->Sccb_XferState & F_SG_XFER) {
5165 WR_HARPOON(port + hp_page_ctrl,
5166 (RD_HARPOON(port + hp_page_ctrl) &
5169 WR_HARPOON(port + hp_sg_addr, 0x00);
5171 sg_ptr = pCurrSCCB->Sccb_sgseg + SG_BUF_CNT;
5174 (unsigned int)(pCurrSCCB->DataLength /
5177 sg_ptr = (u32)(pCurrSCCB->DataLength /
5181 remain_cnt = pCurrSCCB->Sccb_XferCnt;
5183 while (remain_cnt < 0x01000000L) {
5186 segp = (struct blogic_sg_seg *)(pCurrSCCB->
5187 DataPointer) + (sg_ptr * 2);
5188 if (remain_cnt > (unsigned long)segp->segbytes)
5190 (unsigned long)segp->segbytes;
5195 if (remain_cnt < 0x01000000L) {
5197 pCurrSCCB->Sccb_SGoffset = remain_cnt;
5199 pCurrSCCB->Sccb_sgseg = (unsigned short)sg_ptr;
5201 if ((unsigned long)(sg_ptr * SG_ELEMENT_SIZE) ==
5202 pCurrSCCB->DataLength && (remain_cnt == 0))
5204 pCurrSCCB->Sccb_XferState |=
5210 if (pCurrSCCB->HostStatus == 0x00) {
5212 pCurrSCCB->HostStatus =
5218 if (!(pCurrSCCB->Sccb_XferState & F_HOST_XFER_DIR)) {
5220 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5222 FPT_busMstrTimeOut(port);
5227 if (RD_HARPOON(port + hp_int_status) &
5230 if (RD_HARPOON(port + hp_ext_status) &
5233 if (pCurrSCCB->HostStatus ==
5236 pCurrSCCB->HostStatus =
5247 if ((RD_HARPOON(port + hp_fifo_cnt)) >= BM_THRESHOLD) {
5249 timeout = SHORT_WAIT;
5251 while ((RD_HARPOON(port + hp_ext_status) &
5253 && ((RD_HARPOON(port + hp_fifo_cnt)) >=
5254 BM_THRESHOLD) && timeout--) {
5258 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5260 WR_HARPOON(port + hp_bm_ctrl,
5261 (RD_HARPOON(port + hp_bm_ctrl) |
5264 timeout = LONG_WAIT;
5266 while ((RD_HARPOON(port + hp_ext_status) &
5267 BM_CMD_BUSY) && timeout--) {
5270 WR_HARPOON(port + hp_bm_ctrl,
5271 (RD_HARPOON(port + hp_bm_ctrl) &
5274 if (RD_HARPOON(port + hp_ext_status) &
5277 if (pCurrSCCB->HostStatus == 0x00) {
5279 pCurrSCCB->HostStatus =
5283 FPT_busMstrTimeOut(port);
5287 if (RD_HARPOON(port + hp_int_status) & INT_EXT_STATUS) {
5289 if (RD_HARPOON(port + hp_ext_status) &
5292 if (pCurrSCCB->HostStatus == 0x00) {
5294 pCurrSCCB->HostStatus =
5305 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5307 timeout = LONG_WAIT;
5309 while ((RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)
5313 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5315 if (pCurrSCCB->HostStatus == 0x00) {
5317 pCurrSCCB->HostStatus = SCCB_BM_ERR;
5320 FPT_busMstrTimeOut(port);
5324 if (RD_HARPOON(port + hp_int_status) & INT_EXT_STATUS) {
5326 if (RD_HARPOON(port + hp_ext_status) & BAD_EXT_STATUS) {
5328 if (pCurrSCCB->HostStatus == 0x00) {
5330 pCurrSCCB->HostStatus = SCCB_BM_ERR;
5336 if (pCurrSCCB->Sccb_XferState & F_SG_XFER) {
5338 WR_HARPOON(port + hp_page_ctrl,
5339 (RD_HARPOON(port + hp_page_ctrl) &
5342 WR_HARPOON(port + hp_sg_addr, 0x00);
5344 pCurrSCCB->Sccb_sgseg += SG_BUF_CNT;
5346 pCurrSCCB->Sccb_SGoffset = 0x00;
5348 if ((u32)(pCurrSCCB->Sccb_sgseg * SG_ELEMENT_SIZE) >=
5349 pCurrSCCB->DataLength) {
5351 pCurrSCCB->Sccb_XferState |= F_ALL_XFERRED;
5352 pCurrSCCB->Sccb_sgseg =
5353 (unsigned short)(pCurrSCCB->DataLength /
5359 if (!(pCurrSCCB->Sccb_XferState & F_AUTO_SENSE))
5360 pCurrSCCB->Sccb_XferState |= F_ALL_XFERRED;
5364 WR_HARPOON(port + hp_int_mask, (INT_CMD_COMPL | SCSI_INTERRUPT));
5367 /*---------------------------------------------------------------------
5369 * Function: Host Data Transfer Restart
5371 * Description: Reset the available count due to a restore data
5374 *---------------------------------------------------------------------*/
5375 static void FPT_hostDataXferRestart(struct sccb *currSCCB)
5377 unsigned long data_count;
5378 unsigned int sg_index;
5379 struct blogic_sg_seg *segp;
5381 if (currSCCB->Sccb_XferState & F_SG_XFER) {
5383 currSCCB->Sccb_XferCnt = 0;
5385 sg_index = 0xffff; /*Index by long words into sg list. */
5386 data_count = 0; /*Running count of SG xfer counts. */
5389 while (data_count < currSCCB->Sccb_ATC) {
5392 segp = (struct blogic_sg_seg *)(currSCCB->DataPointer) +
5394 data_count += segp->segbytes;
5397 if (data_count == currSCCB->Sccb_ATC) {
5399 currSCCB->Sccb_SGoffset = 0;
5404 currSCCB->Sccb_SGoffset =
5405 data_count - currSCCB->Sccb_ATC;
5408 currSCCB->Sccb_sgseg = (unsigned short)sg_index;
5412 currSCCB->Sccb_XferCnt =
5413 currSCCB->DataLength - currSCCB->Sccb_ATC;
5417 /*---------------------------------------------------------------------
5419 * Function: FPT_scini
5421 * Description: Setup all data structures necessary for SCAM selection.
5423 *---------------------------------------------------------------------*/
5425 static void FPT_scini(unsigned char p_card, unsigned char p_our_id,
5426 unsigned char p_power_up)
5429 unsigned char loser, assigned_id;
5432 unsigned char i, k, ScamFlg;
5433 struct sccb_card *currCard;
5434 struct nvram_info *pCurrNvRam;
5436 currCard = &FPT_BL_Card[p_card];
5437 p_port = currCard->ioPort;
5438 pCurrNvRam = currCard->pNvRamInfo;
5441 ScamFlg = pCurrNvRam->niScamConf;
5442 i = pCurrNvRam->niSysConf;
5445 (unsigned char)FPT_utilEERead(p_port, SCAM_CONFIG / 2);
5447 char)(FPT_utilEERead(p_port, (SYSTEM_CONFIG / 2)));
5449 if (!(i & 0x02)) /* check if reset bus in AutoSCSI parameter set */
5452 FPT_inisci(p_card, p_port, p_our_id);
5454 /* Force to wait 1 sec after SCSI bus reset. Some SCAM device FW
5455 too slow to return to SCAM selection */
5458 FPT_Wait1Second(p_port);
5460 FPT_Wait(p_port, TO_250ms); */
5462 FPT_Wait1Second(p_port);
5464 if ((ScamFlg & SCAM_ENABLED) && (ScamFlg & SCAM_LEVEL2)) {
5465 while (!(FPT_scarb(p_port, INIT_SELTD))) {
5471 FPT_scxferc(p_port, SYNC_PTRN);
5472 FPT_scxferc(p_port, DOM_MSTR);
5475 &FPT_scamInfo[p_our_id].id_string[0]);
5476 } while (loser == 0xFF);
5480 if ((p_power_up) && (!loser)) {
5481 FPT_sresb(p_port, p_card);
5482 FPT_Wait(p_port, TO_250ms);
5484 while (!(FPT_scarb(p_port, INIT_SELTD))) {
5490 FPT_scxferc(p_port, SYNC_PTRN);
5491 FPT_scxferc(p_port, DOM_MSTR);
5494 &FPT_scamInfo[p_our_id].
5496 } while (loser == 0xFF);
5508 FPT_scamInfo[p_our_id].state = ID_ASSIGNED;
5510 if (ScamFlg & SCAM_ENABLED) {
5512 for (i = 0; i < MAX_SCSI_TAR; i++) {
5513 if ((FPT_scamInfo[i].state == ID_UNASSIGNED) ||
5514 (FPT_scamInfo[i].state == ID_UNUSED)) {
5515 if (FPT_scsell(p_port, i)) {
5516 FPT_scamInfo[i].state = LEGACY;
5517 if ((FPT_scamInfo[i].
5518 id_string[0] != 0xFF)
5519 || (FPT_scamInfo[i].
5520 id_string[1] != 0xFA)) {
5523 id_string[0] = 0xFF;
5525 id_string[1] = 0xFA;
5526 if (pCurrNvRam == NULL)
5536 FPT_sresb(p_port, p_card);
5537 FPT_Wait1Second(p_port);
5538 while (!(FPT_scarb(p_port, INIT_SELTD))) {
5541 FPT_scasid(p_card, p_port);
5546 else if ((loser) && (ScamFlg & SCAM_ENABLED)) {
5547 FPT_scamInfo[p_our_id].id_string[0] = SLV_TYPE_CODE0;
5549 FPT_scwtsel(p_port);
5552 while (FPT_scxferc(p_port, 0x00) != SYNC_PTRN) {
5555 i = FPT_scxferc(p_port, 0x00);
5556 if (i == ASSIGN_ID) {
5560 &FPT_scamInfo[p_our_id].id_string[0]))) {
5561 i = FPT_scxferc(p_port, 0x00);
5562 if (FPT_scvalq(i)) {
5563 k = FPT_scxferc(p_port, 0x00);
5565 if (FPT_scvalq(k)) {
5578 FPT_scamInfo[currCard->
5580 state = ID_ASSIGNED;
5581 FPT_scamInfo[currCard->
5591 else if (i == SET_P_FLAG) {
5592 if (!(FPT_scsendi(p_port,
5593 &FPT_scamInfo[p_our_id].
5595 FPT_scamInfo[p_our_id].id_string[0] |=
5598 } while (!assigned_id);
5600 while (FPT_scxferc(p_port, 0x00) != CFG_CMPLT) {
5604 if (ScamFlg & SCAM_ENABLED) {
5606 if (currCard->globalFlags & F_UPDATE_EEPROM) {
5607 FPT_scsavdi(p_card, p_port);
5608 currCard->globalFlags &= ~F_UPDATE_EEPROM;
5613 for (i=0,k=0; i < MAX_SCSI_TAR; i++)
5615 if ((FPT_scamInfo[i].state == ID_ASSIGNED) ||
5616 (FPT_scamInfo[i].state == LEGACY))
5621 currCard->globalFlags |= F_SINGLE_DEVICE;
5623 currCard->globalFlags &= ~F_SINGLE_DEVICE;
5627 /*---------------------------------------------------------------------
5629 * Function: FPT_scarb
5631 * Description: Gain control of the bus and wait SCAM select time (250ms)
5633 *---------------------------------------------------------------------*/
5635 static int FPT_scarb(u32 p_port, unsigned char p_sel_type)
5637 if (p_sel_type == INIT_SELTD) {
5639 while (RD_HARPOON(p_port + hp_scsisig) & (SCSI_SEL | SCSI_BSY)) {
5642 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_SEL)
5645 if (RD_HARPOON(p_port + hp_scsidata_0) != 00)
5648 WR_HARPOON(p_port + hp_scsisig,
5649 (RD_HARPOON(p_port + hp_scsisig) | SCSI_BSY));
5651 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_SEL) {
5653 WR_HARPOON(p_port + hp_scsisig,
5654 (RD_HARPOON(p_port + hp_scsisig) &
5659 WR_HARPOON(p_port + hp_scsisig,
5660 (RD_HARPOON(p_port + hp_scsisig) | SCSI_SEL));
5662 if (RD_HARPOON(p_port + hp_scsidata_0) != 00) {
5664 WR_HARPOON(p_port + hp_scsisig,
5665 (RD_HARPOON(p_port + hp_scsisig) &
5666 ~(SCSI_BSY | SCSI_SEL)));
5671 WR_HARPOON(p_port + hp_clkctrl_0, (RD_HARPOON(p_port + hp_clkctrl_0)
5673 WR_HARPOON(p_port + hp_scsireset, SCAM_EN);
5674 WR_HARPOON(p_port + hp_scsidata_0, 0x00);
5675 WR_HARPOON(p_port + hp_scsidata_1, 0x00);
5676 WR_HARPOON(p_port + hp_portctrl_0, SCSI_BUS_EN);
5678 WR_HARPOON(p_port + hp_scsisig,
5679 (RD_HARPOON(p_port + hp_scsisig) | SCSI_MSG));
5681 WR_HARPOON(p_port + hp_scsisig, (RD_HARPOON(p_port + hp_scsisig)
5684 FPT_Wait(p_port, TO_250ms);
5689 /*---------------------------------------------------------------------
5691 * Function: FPT_scbusf
5693 * Description: Release the SCSI bus and disable SCAM selection.
5695 *---------------------------------------------------------------------*/
5697 static void FPT_scbusf(u32 p_port)
5699 WR_HARPOON(p_port + hp_page_ctrl,
5700 (RD_HARPOON(p_port + hp_page_ctrl) | G_INT_DISABLE));
5702 WR_HARPOON(p_port + hp_scsidata_0, 0x00);
5704 WR_HARPOON(p_port + hp_portctrl_0, (RD_HARPOON(p_port + hp_portctrl_0)
5707 WR_HARPOON(p_port + hp_scsisig, 0x00);
5709 WR_HARPOON(p_port + hp_scsireset, (RD_HARPOON(p_port + hp_scsireset)
5712 WR_HARPOON(p_port + hp_clkctrl_0, (RD_HARPOON(p_port + hp_clkctrl_0)
5715 WRW_HARPOON((p_port + hp_intstat), (BUS_FREE | AUTO_INT | SCAM_SEL));
5717 WR_HARPOON(p_port + hp_page_ctrl,
5718 (RD_HARPOON(p_port + hp_page_ctrl) & ~G_INT_DISABLE));
5721 /*---------------------------------------------------------------------
5723 * Function: FPT_scasid
5725 * Description: Assign an ID to all the SCAM devices.
5727 *---------------------------------------------------------------------*/
5729 static void FPT_scasid(unsigned char p_card, u32 p_port)
5731 unsigned char temp_id_string[ID_STRING_LENGTH];
5733 unsigned char i, k, scam_id;
5734 unsigned char crcBytes[3];
5735 struct nvram_info *pCurrNvRam;
5736 unsigned short *pCrcBytes;
5738 pCurrNvRam = FPT_BL_Card[p_card].pNvRamInfo;
5744 for (k = 0; k < ID_STRING_LENGTH; k++) {
5745 temp_id_string[k] = (unsigned char)0x00;
5748 FPT_scxferc(p_port, SYNC_PTRN);
5749 FPT_scxferc(p_port, ASSIGN_ID);
5751 if (!(FPT_sciso(p_port, &temp_id_string[0]))) {
5753 pCrcBytes = (unsigned short *)&crcBytes[0];
5754 *pCrcBytes = FPT_CalcCrc16(&temp_id_string[0]);
5755 crcBytes[2] = FPT_CalcLrc(&temp_id_string[0]);
5756 temp_id_string[1] = crcBytes[2];
5757 temp_id_string[2] = crcBytes[0];
5758 temp_id_string[3] = crcBytes[1];
5759 for (k = 4; k < ID_STRING_LENGTH; k++)
5760 temp_id_string[k] = (unsigned char)0x00;
5762 i = FPT_scmachid(p_card, temp_id_string);
5764 if (i == CLR_PRIORITY) {
5765 FPT_scxferc(p_port, MISC_CODE);
5766 FPT_scxferc(p_port, CLR_P_FLAG);
5767 i = 0; /*Not the last ID yet. */
5770 else if (i != NO_ID_AVAIL) {
5772 FPT_scxferc(p_port, ID_0_7);
5774 FPT_scxferc(p_port, ID_8_F);
5776 scam_id = (i & (unsigned char)0x07);
5778 for (k = 1; k < 0x08; k <<= 1)
5780 scam_id += 0x08; /*Count number of zeros in DB0-3. */
5782 FPT_scxferc(p_port, scam_id);
5784 i = 0; /*Not the last ID yet. */
5794 FPT_scxferc(p_port, SYNC_PTRN);
5795 FPT_scxferc(p_port, CFG_CMPLT);
5798 /*---------------------------------------------------------------------
5800 * Function: FPT_scsel
5802 * Description: Select all the SCAM devices.
5804 *---------------------------------------------------------------------*/
5806 static void FPT_scsel(u32 p_port)
5809 WR_HARPOON(p_port + hp_scsisig, SCSI_SEL);
5810 FPT_scwiros(p_port, SCSI_MSG);
5812 WR_HARPOON(p_port + hp_scsisig, (SCSI_SEL | SCSI_BSY));
5814 WR_HARPOON(p_port + hp_scsisig,
5815 (SCSI_SEL | SCSI_BSY | SCSI_IOBIT | SCSI_CD));
5816 WR_HARPOON(p_port + hp_scsidata_0,
5817 (unsigned char)(RD_HARPOON(p_port + hp_scsidata_0) |
5818 (unsigned char)(BIT(7) + BIT(6))));
5820 WR_HARPOON(p_port + hp_scsisig, (SCSI_BSY | SCSI_IOBIT | SCSI_CD));
5821 FPT_scwiros(p_port, SCSI_SEL);
5823 WR_HARPOON(p_port + hp_scsidata_0,
5824 (unsigned char)(RD_HARPOON(p_port + hp_scsidata_0) &
5825 ~(unsigned char)BIT(6)));
5826 FPT_scwirod(p_port, BIT(6));
5828 WR_HARPOON(p_port + hp_scsisig,
5829 (SCSI_SEL | SCSI_BSY | SCSI_IOBIT | SCSI_CD));
5832 /*---------------------------------------------------------------------
5834 * Function: FPT_scxferc
5836 * Description: Handshake the p_data (DB4-0) across the bus.
5838 *---------------------------------------------------------------------*/
5840 static unsigned char FPT_scxferc(u32 p_port, unsigned char p_data)
5842 unsigned char curr_data, ret_data;
5844 curr_data = p_data | BIT(7) | BIT(5); /*Start with DB7 & DB5 asserted. */
5846 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5848 curr_data &= ~BIT(7);
5850 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5852 FPT_scwirod(p_port, BIT(7)); /*Wait for DB7 to be released. */
5853 while (!(RD_HARPOON(p_port + hp_scsidata_0) & BIT(5))) ;
5855 ret_data = (RD_HARPOON(p_port + hp_scsidata_0) & (unsigned char)0x1F);
5857 curr_data |= BIT(6);
5859 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5861 curr_data &= ~BIT(5);
5863 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5865 FPT_scwirod(p_port, BIT(5)); /*Wait for DB5 to be released. */
5867 curr_data &= ~(BIT(4) | BIT(3) | BIT(2) | BIT(1) | BIT(0)); /*Release data bits */
5868 curr_data |= BIT(7);
5870 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5872 curr_data &= ~BIT(6);
5874 WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5876 FPT_scwirod(p_port, BIT(6)); /*Wait for DB6 to be released. */
5881 /*---------------------------------------------------------------------
5883 * Function: FPT_scsendi
5885 * Description: Transfer our Identification string to determine if we
5886 * will be the dominant master.
5888 *---------------------------------------------------------------------*/
5890 static unsigned char FPT_scsendi(u32 p_port, unsigned char p_id_string[])
5892 unsigned char ret_data, byte_cnt, bit_cnt, defer;
5896 for (byte_cnt = 0; byte_cnt < ID_STRING_LENGTH; byte_cnt++) {
5898 for (bit_cnt = 0x80; bit_cnt != 0; bit_cnt >>= 1) {
5901 ret_data = FPT_scxferc(p_port, 00);
5903 else if (p_id_string[byte_cnt] & bit_cnt)
5905 ret_data = FPT_scxferc(p_port, 02);
5909 ret_data = FPT_scxferc(p_port, 01);
5914 if ((ret_data & 0x1C) == 0x10)
5915 return 0x00; /*End of isolation stage, we won! */
5917 if (ret_data & 0x1C)
5920 if ((defer) && (!(ret_data & 0x1F)))
5921 return 0x01; /*End of isolation stage, we lost. */
5928 return 0x01; /*We lost */
5930 return 0; /*We WON! Yeeessss! */
5933 /*---------------------------------------------------------------------
5935 * Function: FPT_sciso
5937 * Description: Transfer the Identification string.
5939 *---------------------------------------------------------------------*/
5941 static unsigned char FPT_sciso(u32 p_port, unsigned char p_id_string[])
5943 unsigned char ret_data, the_data, byte_cnt, bit_cnt;
5947 for (byte_cnt = 0; byte_cnt < ID_STRING_LENGTH; byte_cnt++) {
5949 for (bit_cnt = 0; bit_cnt < 8; bit_cnt++) {
5951 ret_data = FPT_scxferc(p_port, 0);
5953 if (ret_data & 0xFC)
5959 if (ret_data & BIT(1)) {
5964 if ((ret_data & 0x1F) == 0) {
5966 if(bit_cnt != 0 || bit_cnt != 8)
5970 FPT_scxferc(p_port, SYNC_PTRN);
5971 FPT_scxferc(p_port, ASSIGN_ID);
5983 p_id_string[byte_cnt] = the_data;
5990 /*---------------------------------------------------------------------
5992 * Function: FPT_scwirod
5994 * Description: Sample the SCSI data bus making sure the signal has been
5995 * deasserted for the correct number of consecutive samples.
5997 *---------------------------------------------------------------------*/
5999 static void FPT_scwirod(u32 p_port, unsigned char p_data_bit)
6004 while (i < MAX_SCSI_TAR) {
6006 if (RD_HARPOON(p_port + hp_scsidata_0) & p_data_bit)
6017 /*---------------------------------------------------------------------
6019 * Function: FPT_scwiros
6021 * Description: Sample the SCSI Signal lines making sure the signal has been
6022 * deasserted for the correct number of consecutive samples.
6024 *---------------------------------------------------------------------*/
6026 static void FPT_scwiros(u32 p_port, unsigned char p_data_bit)
6031 while (i < MAX_SCSI_TAR) {
6033 if (RD_HARPOON(p_port + hp_scsisig) & p_data_bit)
6044 /*---------------------------------------------------------------------
6046 * Function: FPT_scvalq
6048 * Description: Make sure we received a valid data byte.
6050 *---------------------------------------------------------------------*/
6052 static unsigned char FPT_scvalq(unsigned char p_quintet)
6054 unsigned char count;
6056 for (count = 1; count < 0x08; count <<= 1) {
6057 if (!(p_quintet & count))
6061 if (p_quintet & 0x18)
6068 /*---------------------------------------------------------------------
6070 * Function: FPT_scsell
6072 * Description: Select the specified device ID using a selection timeout
6073 * less than 4ms. If somebody responds then it is a legacy
6074 * drive and this ID must be marked as such.
6076 *---------------------------------------------------------------------*/
6078 static unsigned char FPT_scsell(u32 p_port, unsigned char targ_id)
6082 WR_HARPOON(p_port + hp_page_ctrl,
6083 (RD_HARPOON(p_port + hp_page_ctrl) | G_INT_DISABLE));
6085 ARAM_ACCESS(p_port);
6087 WR_HARPOON(p_port + hp_addstat,
6088 (RD_HARPOON(p_port + hp_addstat) | SCAM_TIMER));
6089 WR_HARPOON(p_port + hp_seltimeout, TO_4ms);
6091 for (i = p_port + CMD_STRT; i < p_port + CMD_STRT + 12; i += 2) {
6092 WRW_HARPOON(i, (MPM_OP + ACOMMAND));
6094 WRW_HARPOON(i, (BRH_OP + ALWAYS + NP));
6096 WRW_HARPOON((p_port + hp_intstat),
6097 (RESET | TIMEOUT | SEL | BUS_FREE | AUTO_INT));
6099 WR_HARPOON(p_port + hp_select_id, targ_id);
6101 WR_HARPOON(p_port + hp_portctrl_0, SCSI_PORT);
6102 WR_HARPOON(p_port + hp_autostart_3, (SELECT | CMD_ONLY_STRT));
6103 WR_HARPOON(p_port + hp_scsictrl_0, (SEL_TAR | ENA_RESEL));
6105 while (!(RDW_HARPOON((p_port + hp_intstat)) &
6106 (RESET | PROG_HLT | TIMEOUT | AUTO_INT))) {
6109 if (RDW_HARPOON((p_port + hp_intstat)) & RESET)
6110 FPT_Wait(p_port, TO_250ms);
6112 DISABLE_AUTO(p_port);
6114 WR_HARPOON(p_port + hp_addstat,
6115 (RD_HARPOON(p_port + hp_addstat) & ~SCAM_TIMER));
6116 WR_HARPOON(p_port + hp_seltimeout, TO_290ms);
6118 SGRAM_ACCESS(p_port);
6120 if (RDW_HARPOON((p_port + hp_intstat)) & (RESET | TIMEOUT)) {
6122 WRW_HARPOON((p_port + hp_intstat),
6123 (RESET | TIMEOUT | SEL | BUS_FREE | PHASE));
6125 WR_HARPOON(p_port + hp_page_ctrl,
6126 (RD_HARPOON(p_port + hp_page_ctrl) &
6129 return 0; /*No legacy device */
6134 while (!(RDW_HARPOON((p_port + hp_intstat)) & BUS_FREE)) {
6135 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_REQ) {
6136 WR_HARPOON(p_port + hp_scsisig,
6137 (SCSI_ACK + S_ILL_PH));
6142 WRW_HARPOON((p_port + hp_intstat), CLR_ALL_INT_1);
6144 WR_HARPOON(p_port + hp_page_ctrl,
6145 (RD_HARPOON(p_port + hp_page_ctrl) &
6148 return 1; /*Found one of them oldies! */
6152 /*---------------------------------------------------------------------
6154 * Function: FPT_scwtsel
6156 * Description: Wait to be selected by another SCAM initiator.
6158 *---------------------------------------------------------------------*/
6160 static void FPT_scwtsel(u32 p_port)
6162 while (!(RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL)) {
6166 /*---------------------------------------------------------------------
6168 * Function: FPT_inisci
6170 * Description: Setup the data Structure with the info from the EEPROM.
6172 *---------------------------------------------------------------------*/
6174 static void FPT_inisci(unsigned char p_card, u32 p_port, unsigned char p_our_id)
6176 unsigned char i, k, max_id;
6177 unsigned short ee_data;
6178 struct nvram_info *pCurrNvRam;
6180 pCurrNvRam = FPT_BL_Card[p_card].pNvRamInfo;
6182 if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6189 for (i = 0; i < max_id; i++) {
6191 for (k = 0; k < 4; k++)
6192 FPT_scamInfo[i].id_string[k] =
6193 pCurrNvRam->niScamTbl[i][k];
6194 for (k = 4; k < ID_STRING_LENGTH; k++)
6195 FPT_scamInfo[i].id_string[k] =
6196 (unsigned char)0x00;
6198 if (FPT_scamInfo[i].id_string[0] == 0x00)
6199 FPT_scamInfo[i].state = ID_UNUSED; /*Default to unused ID. */
6201 FPT_scamInfo[i].state = ID_UNASSIGNED; /*Default to unassigned ID. */
6205 for (i = 0; i < max_id; i++) {
6206 for (k = 0; k < ID_STRING_LENGTH; k += 2) {
6208 FPT_utilEERead(p_port,
6210 short)((EE_SCAMBASE / 2) +
6211 (unsigned short)(i *
6212 ((unsigned short)ID_STRING_LENGTH / 2)) + (unsigned short)(k / 2)));
6213 FPT_scamInfo[i].id_string[k] =
6214 (unsigned char)ee_data;
6216 FPT_scamInfo[i].id_string[k + 1] =
6217 (unsigned char)ee_data;
6220 if ((FPT_scamInfo[i].id_string[0] == 0x00) ||
6221 (FPT_scamInfo[i].id_string[0] == 0xFF))
6223 FPT_scamInfo[i].state = ID_UNUSED; /*Default to unused ID. */
6226 FPT_scamInfo[i].state = ID_UNASSIGNED; /*Default to unassigned ID. */
6230 for (k = 0; k < ID_STRING_LENGTH; k++)
6231 FPT_scamInfo[p_our_id].id_string[k] = FPT_scamHAString[k];
6235 /*---------------------------------------------------------------------
6237 * Function: FPT_scmachid
6239 * Description: Match the Device ID string with our values stored in
6242 *---------------------------------------------------------------------*/
6244 static unsigned char FPT_scmachid(unsigned char p_card,
6245 unsigned char p_id_string[])
6248 unsigned char i, k, match;
6250 for (i = 0; i < MAX_SCSI_TAR; i++) {
6254 for (k = 0; k < ID_STRING_LENGTH; k++) {
6255 if (p_id_string[k] != FPT_scamInfo[i].id_string[k])
6260 FPT_scamInfo[i].state = ID_ASSIGNED;
6266 if (p_id_string[0] & BIT(5))
6271 if (((p_id_string[0] & 0x06) == 0x02)
6272 || ((p_id_string[0] & 0x06) == 0x04))
6273 match = p_id_string[1] & (unsigned char)0x1F;
6280 if (FPT_scamInfo[match].state == ID_UNUSED) {
6281 for (k = 0; k < ID_STRING_LENGTH; k++) {
6282 FPT_scamInfo[match].id_string[k] =
6286 FPT_scamInfo[match].state = ID_ASSIGNED;
6288 if (FPT_BL_Card[p_card].pNvRamInfo == NULL)
6289 FPT_BL_Card[p_card].globalFlags |=
6297 if (match == 0xFF) {
6298 if (p_id_string[0] & BIT(5))
6301 match = MAX_SCSI_TAR - 1;
6305 if (p_id_string[0] & BIT(7)) {
6306 return CLR_PRIORITY;
6309 if (p_id_string[0] & BIT(5))
6314 if (((p_id_string[0] & 0x06) == 0x02)
6315 || ((p_id_string[0] & 0x06) == 0x04))
6316 match = p_id_string[1] & (unsigned char)0x1F;
6324 if (FPT_scamInfo[match].state == ID_UNASSIGNED) {
6325 for (k = 0; k < ID_STRING_LENGTH; k++) {
6326 FPT_scamInfo[match].id_string[k] =
6330 FPT_scamInfo[match].id_string[0] |= BIT(7);
6331 FPT_scamInfo[match].state = ID_ASSIGNED;
6332 if (FPT_BL_Card[p_card].pNvRamInfo == NULL)
6333 FPT_BL_Card[p_card].globalFlags |=
6341 if (match == 0xFF) {
6342 if (p_id_string[0] & BIT(5))
6345 match = MAX_SCSI_TAR - 1;
6352 /*---------------------------------------------------------------------
6354 * Function: FPT_scsavdi
6356 * Description: Save off the device SCAM ID strings.
6358 *---------------------------------------------------------------------*/
6360 static void FPT_scsavdi(unsigned char p_card, u32 p_port)
6362 unsigned char i, k, max_id;
6363 unsigned short ee_data, sum_data;
6367 for (i = 1; i < EE_SCAMBASE / 2; i++) {
6368 sum_data += FPT_utilEERead(p_port, i);
6371 FPT_utilEEWriteOnOff(p_port, 1); /* Enable write access to the EEPROM */
6373 if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6379 for (i = 0; i < max_id; i++) {
6381 for (k = 0; k < ID_STRING_LENGTH; k += 2) {
6382 ee_data = FPT_scamInfo[i].id_string[k + 1];
6384 ee_data |= FPT_scamInfo[i].id_string[k];
6385 sum_data += ee_data;
6386 FPT_utilEEWrite(p_port, ee_data,
6387 (unsigned short)((EE_SCAMBASE / 2) +
6388 (unsigned short)(i *
6389 ((unsigned short)ID_STRING_LENGTH / 2)) + (unsigned short)(k / 2)));
6393 FPT_utilEEWrite(p_port, sum_data, EEPROM_CHECK_SUM / 2);
6394 FPT_utilEEWriteOnOff(p_port, 0); /* Turn off write access */
6397 /*---------------------------------------------------------------------
6399 * Function: FPT_XbowInit
6401 * Description: Setup the Xbow for normal operation.
6403 *---------------------------------------------------------------------*/
6405 static void FPT_XbowInit(u32 port, unsigned char ScamFlg)
6409 i = RD_HARPOON(port + hp_page_ctrl);
6410 WR_HARPOON(port + hp_page_ctrl, (unsigned char)(i | G_INT_DISABLE));
6412 WR_HARPOON(port + hp_scsireset, 0x00);
6413 WR_HARPOON(port + hp_portctrl_1, HOST_MODE8);
6415 WR_HARPOON(port + hp_scsireset, (DMA_RESET | HPSCSI_RESET | PROG_RESET |
6418 WR_HARPOON(port + hp_scsireset, SCSI_INI);
6420 WR_HARPOON(port + hp_clkctrl_0, CLKCTRL_DEFAULT);
6422 WR_HARPOON(port + hp_scsisig, 0x00); /* Clear any signals we might */
6423 WR_HARPOON(port + hp_scsictrl_0, ENA_SCAM_SEL);
6425 WRW_HARPOON((port + hp_intstat), CLR_ALL_INT);
6427 FPT_default_intena = RESET | RSEL | PROG_HLT | TIMEOUT |
6428 BUS_FREE | XFER_CNT_0 | AUTO_INT;
6430 if ((ScamFlg & SCAM_ENABLED) && (ScamFlg & SCAM_LEVEL2))
6431 FPT_default_intena |= SCAM_SEL;
6433 WRW_HARPOON((port + hp_intena), FPT_default_intena);
6435 WR_HARPOON(port + hp_seltimeout, TO_290ms);
6437 /* Turn on SCSI_MODE8 for narrow cards to fix the
6438 strapping issue with the DUAL CHANNEL card */
6439 if (RD_HARPOON(port + hp_page_ctrl) & NARROW_SCSI_CARD)
6440 WR_HARPOON(port + hp_addstat, SCSI_MODE8);
6442 WR_HARPOON(port + hp_page_ctrl, i);
6446 /*---------------------------------------------------------------------
6448 * Function: FPT_BusMasterInit
6450 * Description: Initialize the BusMaster for normal operations.
6452 *---------------------------------------------------------------------*/
6454 static void FPT_BusMasterInit(u32 p_port)
6457 WR_HARPOON(p_port + hp_sys_ctrl, DRVR_RST);
6458 WR_HARPOON(p_port + hp_sys_ctrl, 0x00);
6460 WR_HARPOON(p_port + hp_host_blk_cnt, XFER_BLK64);
6462 WR_HARPOON(p_port + hp_bm_ctrl, (BMCTRL_DEFAULT));
6464 WR_HARPOON(p_port + hp_ee_ctrl, (SCSI_TERM_ENA_H));
6466 RD_HARPOON(p_port + hp_int_status); /*Clear interrupts. */
6467 WR_HARPOON(p_port + hp_int_mask, (INT_CMD_COMPL | SCSI_INTERRUPT));
6468 WR_HARPOON(p_port + hp_page_ctrl, (RD_HARPOON(p_port + hp_page_ctrl) &
6472 /*---------------------------------------------------------------------
6474 * Function: FPT_DiagEEPROM
6476 * Description: Verfiy checksum and 'Key' and initialize the EEPROM if
6479 *---------------------------------------------------------------------*/
6481 static void FPT_DiagEEPROM(u32 p_port)
6483 unsigned short index, temp, max_wd_cnt;
6485 if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6486 max_wd_cnt = EEPROM_WD_CNT;
6488 max_wd_cnt = EEPROM_WD_CNT * 2;
6490 temp = FPT_utilEERead(p_port, FW_SIGNATURE / 2);
6492 if (temp == 0x4641) {
6494 for (index = 2; index < max_wd_cnt; index++) {
6496 temp += FPT_utilEERead(p_port, index);
6500 if (temp == FPT_utilEERead(p_port, EEPROM_CHECK_SUM / 2)) {
6502 return; /*EEPROM is Okay so return now! */
6506 FPT_utilEEWriteOnOff(p_port, (unsigned char)1);
6508 for (index = 0; index < max_wd_cnt; index++) {
6510 FPT_utilEEWrite(p_port, 0x0000, index);
6515 FPT_utilEEWrite(p_port, 0x4641, FW_SIGNATURE / 2);
6517 FPT_utilEEWrite(p_port, 0x3920, MODEL_NUMB_0 / 2);
6519 FPT_utilEEWrite(p_port, 0x3033, MODEL_NUMB_2 / 2);
6521 FPT_utilEEWrite(p_port, 0x2020, MODEL_NUMB_4 / 2);
6523 FPT_utilEEWrite(p_port, 0x70D3, SYSTEM_CONFIG / 2);
6525 FPT_utilEEWrite(p_port, 0x0010, BIOS_CONFIG / 2);
6527 FPT_utilEEWrite(p_port, 0x0003, SCAM_CONFIG / 2);
6529 FPT_utilEEWrite(p_port, 0x0007, ADAPTER_SCSI_ID / 2);
6532 FPT_utilEEWrite(p_port, 0x0000, IGNORE_B_SCAN / 2);
6534 FPT_utilEEWrite(p_port, 0x0000, SEND_START_ENA / 2);
6536 FPT_utilEEWrite(p_port, 0x0000, DEVICE_ENABLE / 2);
6539 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL01 / 2);
6541 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL23 / 2);
6543 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL45 / 2);
6545 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL67 / 2);
6547 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL89 / 2);
6549 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLab / 2);
6551 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLcd / 2);
6553 FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLef / 2);
6556 FPT_utilEEWrite(p_port, 0x6C46, 64 / 2); /*PRODUCT ID */
6558 FPT_utilEEWrite(p_port, 0x7361, 66 / 2); /* FlashPoint LT */
6560 FPT_utilEEWrite(p_port, 0x5068, 68 / 2);
6562 FPT_utilEEWrite(p_port, 0x696F, 70 / 2);
6564 FPT_utilEEWrite(p_port, 0x746E, 72 / 2);
6566 FPT_utilEEWrite(p_port, 0x4C20, 74 / 2);
6568 FPT_utilEEWrite(p_port, 0x2054, 76 / 2);
6570 FPT_utilEEWrite(p_port, 0x2020, 78 / 2);
6573 index = ((EE_SCAMBASE / 2) + (7 * 16));
6574 FPT_utilEEWrite(p_port, (0x0700 + TYPE_CODE0), index);
6575 temp += (0x0700 + TYPE_CODE0);
6577 FPT_utilEEWrite(p_port, 0x5542, index); /*Vendor ID code */
6578 temp += 0x5542; /* BUSLOGIC */
6580 FPT_utilEEWrite(p_port, 0x4C53, index);
6583 FPT_utilEEWrite(p_port, 0x474F, index);
6586 FPT_utilEEWrite(p_port, 0x4349, index);
6589 FPT_utilEEWrite(p_port, 0x5442, index); /*Vendor unique code */
6590 temp += 0x5442; /* BT- 930 */
6592 FPT_utilEEWrite(p_port, 0x202D, index);
6595 FPT_utilEEWrite(p_port, 0x3339, index);
6597 index++; /*Serial # */
6598 FPT_utilEEWrite(p_port, 0x2030, index); /* 01234567 */
6601 FPT_utilEEWrite(p_port, 0x5453, index);
6604 FPT_utilEEWrite(p_port, 0x5645, index);
6607 FPT_utilEEWrite(p_port, 0x2045, index);
6610 FPT_utilEEWrite(p_port, 0x202F, index);
6613 FPT_utilEEWrite(p_port, 0x4F4A, index);
6616 FPT_utilEEWrite(p_port, 0x204E, index);
6619 FPT_utilEEWrite(p_port, 0x3539, index);
6622 FPT_utilEEWrite(p_port, temp, EEPROM_CHECK_SUM / 2);
6624 FPT_utilEEWriteOnOff(p_port, (unsigned char)0);
6628 /*---------------------------------------------------------------------
6630 * Function: Queue Search Select
6632 * Description: Try to find a new command to execute.
6634 *---------------------------------------------------------------------*/
6636 static void FPT_queueSearchSelect(struct sccb_card *pCurrCard,
6637 unsigned char p_card)
6639 unsigned char scan_ptr, lun;
6640 struct sccb_mgr_tar_info *currTar_Info;
6641 struct sccb *pOldSccb;
6643 scan_ptr = pCurrCard->scanIndex;
6645 currTar_Info = &FPT_sccbMgrTbl[p_card][scan_ptr];
6646 if ((pCurrCard->globalFlags & F_CONLUN_IO) &&
6647 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
6649 if (currTar_Info->TarSelQ_Cnt != 0) {
6652 if (scan_ptr == MAX_SCSI_TAR)
6655 for (lun = 0; lun < MAX_LUN; lun++) {
6656 if (currTar_Info->TarLUNBusy[lun] == 0) {
6658 pCurrCard->currentSCCB =
6659 currTar_Info->TarSelQ_Head;
6663 currentSCCB != NULL)
6666 currentSCCB->Lun)) {
6670 pCurrCard->currentSCCB =
6676 if (pCurrCard->currentSCCB ==
6679 if (pOldSccb != NULL) {
6722 pCurrCard->scanIndex = scan_ptr;
6724 pCurrCard->globalFlags |=
6734 if (scan_ptr == MAX_SCSI_TAR) {
6740 if ((currTar_Info->TarSelQ_Cnt != 0) &&
6741 (currTar_Info->TarLUNBusy[0] == 0)) {
6743 pCurrCard->currentSCCB =
6744 currTar_Info->TarSelQ_Head;
6746 currTar_Info->TarSelQ_Head =
6747 (struct sccb *)(pCurrCard->currentSCCB)->
6750 if (currTar_Info->TarSelQ_Head == NULL) {
6751 currTar_Info->TarSelQ_Tail = NULL;
6752 currTar_Info->TarSelQ_Cnt = 0;
6754 currTar_Info->TarSelQ_Cnt--;
6755 currTar_Info->TarSelQ_Head->
6756 Sccb_backlink = (struct sccb *)NULL;
6760 if (scan_ptr == MAX_SCSI_TAR)
6763 pCurrCard->scanIndex = scan_ptr;
6765 pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
6772 if (scan_ptr == MAX_SCSI_TAR) {
6777 } while (scan_ptr != pCurrCard->scanIndex);
6780 /*---------------------------------------------------------------------
6782 * Function: Queue Select Fail
6784 * Description: Add the current SCCB to the head of the Queue.
6786 *---------------------------------------------------------------------*/
6788 static void FPT_queueSelectFail(struct sccb_card *pCurrCard,
6789 unsigned char p_card)
6791 unsigned char thisTarg;
6792 struct sccb_mgr_tar_info *currTar_Info;
6794 if (pCurrCard->currentSCCB != NULL) {
6796 (unsigned char)(((struct sccb *)(pCurrCard->currentSCCB))->
6798 currTar_Info = &FPT_sccbMgrTbl[p_card][thisTarg];
6800 pCurrCard->currentSCCB->Sccb_backlink = (struct sccb *)NULL;
6802 pCurrCard->currentSCCB->Sccb_forwardlink =
6803 currTar_Info->TarSelQ_Head;
6805 if (currTar_Info->TarSelQ_Cnt == 0) {
6806 currTar_Info->TarSelQ_Tail = pCurrCard->currentSCCB;
6810 currTar_Info->TarSelQ_Head->Sccb_backlink =
6811 pCurrCard->currentSCCB;
6814 currTar_Info->TarSelQ_Head = pCurrCard->currentSCCB;
6816 pCurrCard->currentSCCB = NULL;
6817 currTar_Info->TarSelQ_Cnt++;
6821 /*---------------------------------------------------------------------
6823 * Function: Queue Command Complete
6825 * Description: Call the callback function with the current SCCB.
6827 *---------------------------------------------------------------------*/
6829 static void FPT_queueCmdComplete(struct sccb_card *pCurrCard,
6830 struct sccb *p_sccb, unsigned char p_card)
6833 unsigned char i, SCSIcmd;
6834 CALL_BK_FN callback;
6835 struct sccb_mgr_tar_info *currTar_Info;
6837 SCSIcmd = p_sccb->Cdb[0];
6839 if (!(p_sccb->Sccb_XferState & F_ALL_XFERRED)) {
6842 ControlByte & (SCCB_DATA_XFER_OUT | SCCB_DATA_XFER_IN))
6843 && (p_sccb->HostStatus == SCCB_COMPLETE)
6844 && (p_sccb->TargetStatus != SAM_STAT_CHECK_CONDITION))
6846 if ((SCSIcmd == READ_6) ||
6847 (SCSIcmd == WRITE_6) ||
6848 (SCSIcmd == READ_10) ||
6849 (SCSIcmd == WRITE_10) ||
6850 (SCSIcmd == WRITE_VERIFY) ||
6851 (SCSIcmd == START_STOP) ||
6852 (pCurrCard->globalFlags & F_NO_FILTER)
6854 p_sccb->HostStatus = SCCB_DATA_UNDER_RUN;
6857 if (p_sccb->SccbStatus == SCCB_IN_PROCESS) {
6858 if (p_sccb->HostStatus || p_sccb->TargetStatus)
6859 p_sccb->SccbStatus = SCCB_ERROR;
6861 p_sccb->SccbStatus = SCCB_SUCCESS;
6864 if (p_sccb->Sccb_XferState & F_AUTO_SENSE) {
6866 p_sccb->CdbLength = p_sccb->Save_CdbLen;
6867 for (i = 0; i < 6; i++) {
6868 p_sccb->Cdb[i] = p_sccb->Save_Cdb[i];
6872 if ((p_sccb->OperationCode == RESIDUAL_SG_COMMAND) ||
6873 (p_sccb->OperationCode == RESIDUAL_COMMAND)) {
6875 FPT_utilUpdateResidual(p_sccb);
6878 pCurrCard->cmdCounter--;
6879 if (!pCurrCard->cmdCounter) {
6881 if (pCurrCard->globalFlags & F_GREEN_PC) {
6882 WR_HARPOON(pCurrCard->ioPort + hp_clkctrl_0,
6883 (PWR_DWN | CLKCTRL_DEFAULT));
6884 WR_HARPOON(pCurrCard->ioPort + hp_sys_ctrl, STOP_CLK);
6887 WR_HARPOON(pCurrCard->ioPort + hp_semaphore,
6888 (RD_HARPOON(pCurrCard->ioPort + hp_semaphore) &
6893 if (pCurrCard->discQCount != 0) {
6894 currTar_Info = &FPT_sccbMgrTbl[p_card][p_sccb->TargID];
6895 if (((pCurrCard->globalFlags & F_CONLUN_IO) &&
6896 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
6898 pCurrCard->discQCount--;
6899 pCurrCard->discQ_Tbl[currTar_Info->
6900 LunDiscQ_Idx[p_sccb->Lun]] = NULL;
6902 if (p_sccb->Sccb_tag) {
6903 pCurrCard->discQCount--;
6904 pCurrCard->discQ_Tbl[p_sccb->Sccb_tag] = NULL;
6906 pCurrCard->discQCount--;
6907 pCurrCard->discQ_Tbl[currTar_Info->
6908 LunDiscQ_Idx[0]] = NULL;
6914 callback = (CALL_BK_FN) p_sccb->SccbCallback;
6916 pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
6917 pCurrCard->currentSCCB = NULL;
6920 /*---------------------------------------------------------------------
6922 * Function: Queue Disconnect
6924 * Description: Add SCCB to our disconnect array.
6926 *---------------------------------------------------------------------*/
6927 static void FPT_queueDisconnect(struct sccb *p_sccb, unsigned char p_card)
6929 struct sccb_mgr_tar_info *currTar_Info;
6931 currTar_Info = &FPT_sccbMgrTbl[p_card][p_sccb->TargID];
6933 if (((FPT_BL_Card[p_card].globalFlags & F_CONLUN_IO) &&
6934 ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))) {
6935 FPT_BL_Card[p_card].discQ_Tbl[currTar_Info->
6936 LunDiscQ_Idx[p_sccb->Lun]] =
6939 if (p_sccb->Sccb_tag) {
6940 FPT_BL_Card[p_card].discQ_Tbl[p_sccb->Sccb_tag] =
6942 FPT_sccbMgrTbl[p_card][p_sccb->TargID].TarLUNBusy[0] =
6944 FPT_sccbMgrTbl[p_card][p_sccb->TargID].TarTagQ_Cnt++;
6946 FPT_BL_Card[p_card].discQ_Tbl[currTar_Info->
6947 LunDiscQ_Idx[0]] = p_sccb;
6950 FPT_BL_Card[p_card].currentSCCB = NULL;
6953 /*---------------------------------------------------------------------
6955 * Function: Queue Flush SCCB
6957 * Description: Flush all SCCB's back to the host driver for this target.
6959 *---------------------------------------------------------------------*/
6961 static void FPT_queueFlushSccb(unsigned char p_card, unsigned char error_code)
6963 unsigned char qtag, thisTarg;
6964 struct sccb *currSCCB;
6965 struct sccb_mgr_tar_info *currTar_Info;
6967 currSCCB = FPT_BL_Card[p_card].currentSCCB;
6968 if (currSCCB != NULL) {
6969 thisTarg = (unsigned char)currSCCB->TargID;
6970 currTar_Info = &FPT_sccbMgrTbl[p_card][thisTarg];
6972 for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
6974 if (FPT_BL_Card[p_card].discQ_Tbl[qtag] &&
6975 (FPT_BL_Card[p_card].discQ_Tbl[qtag]->TargID ==
6978 FPT_BL_Card[p_card].discQ_Tbl[qtag]->
6979 HostStatus = (unsigned char)error_code;
6981 FPT_queueCmdComplete(&FPT_BL_Card[p_card],
6982 FPT_BL_Card[p_card].
6983 discQ_Tbl[qtag], p_card);
6985 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
6986 currTar_Info->TarTagQ_Cnt--;
6994 /*---------------------------------------------------------------------
6996 * Function: Queue Flush Target SCCB
6998 * Description: Flush all SCCB's back to the host driver for this target.
7000 *---------------------------------------------------------------------*/
7002 static void FPT_queueFlushTargSccb(unsigned char p_card, unsigned char thisTarg,
7003 unsigned char error_code)
7006 struct sccb_mgr_tar_info *currTar_Info;
7008 currTar_Info = &FPT_sccbMgrTbl[p_card][thisTarg];
7010 for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
7012 if (FPT_BL_Card[p_card].discQ_Tbl[qtag] &&
7013 (FPT_BL_Card[p_card].discQ_Tbl[qtag]->TargID == thisTarg)) {
7015 FPT_BL_Card[p_card].discQ_Tbl[qtag]->HostStatus =
7016 (unsigned char)error_code;
7018 FPT_queueCmdComplete(&FPT_BL_Card[p_card],
7019 FPT_BL_Card[p_card].
7020 discQ_Tbl[qtag], p_card);
7022 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
7023 currTar_Info->TarTagQ_Cnt--;
7030 static void FPT_queueAddSccb(struct sccb *p_SCCB, unsigned char p_card)
7032 struct sccb_mgr_tar_info *currTar_Info;
7033 currTar_Info = &FPT_sccbMgrTbl[p_card][p_SCCB->TargID];
7035 p_SCCB->Sccb_forwardlink = NULL;
7037 p_SCCB->Sccb_backlink = currTar_Info->TarSelQ_Tail;
7039 if (currTar_Info->TarSelQ_Cnt == 0) {
7041 currTar_Info->TarSelQ_Head = p_SCCB;
7046 currTar_Info->TarSelQ_Tail->Sccb_forwardlink = p_SCCB;
7049 currTar_Info->TarSelQ_Tail = p_SCCB;
7050 currTar_Info->TarSelQ_Cnt++;
7053 /*---------------------------------------------------------------------
7055 * Function: Queue Find SCCB
7057 * Description: Search the target select Queue for this SCCB, and
7058 * remove it if found.
7060 *---------------------------------------------------------------------*/
7062 static unsigned char FPT_queueFindSccb(struct sccb *p_SCCB,
7063 unsigned char p_card)
7066 struct sccb_mgr_tar_info *currTar_Info;
7068 currTar_Info = &FPT_sccbMgrTbl[p_card][p_SCCB->TargID];
7070 q_ptr = currTar_Info->TarSelQ_Head;
7072 while (q_ptr != NULL) {
7074 if (q_ptr == p_SCCB) {
7076 if (currTar_Info->TarSelQ_Head == q_ptr) {
7078 currTar_Info->TarSelQ_Head =
7079 q_ptr->Sccb_forwardlink;
7082 if (currTar_Info->TarSelQ_Tail == q_ptr) {
7084 currTar_Info->TarSelQ_Tail =
7085 q_ptr->Sccb_backlink;
7088 if (q_ptr->Sccb_forwardlink != NULL) {
7089 q_ptr->Sccb_forwardlink->Sccb_backlink =
7090 q_ptr->Sccb_backlink;
7093 if (q_ptr->Sccb_backlink != NULL) {
7094 q_ptr->Sccb_backlink->Sccb_forwardlink =
7095 q_ptr->Sccb_forwardlink;
7098 currTar_Info->TarSelQ_Cnt--;
7104 q_ptr = q_ptr->Sccb_forwardlink;
7112 /*---------------------------------------------------------------------
7114 * Function: Utility Update Residual Count
7116 * Description: Update the XferCnt to the remaining byte count.
7117 * If we transferred all the data then just write zero.
7118 * If Non-SG transfer then report Total Cnt - Actual Transfer
7119 * Cnt. For SG transfers add the count fields of all
7120 * remaining SG elements, as well as any partial remaining
7123 *---------------------------------------------------------------------*/
7125 static void FPT_utilUpdateResidual(struct sccb *p_SCCB)
7127 unsigned long partial_cnt;
7128 unsigned int sg_index;
7129 struct blogic_sg_seg *segp;
7131 if (p_SCCB->Sccb_XferState & F_ALL_XFERRED) {
7133 p_SCCB->DataLength = 0x0000;
7136 else if (p_SCCB->Sccb_XferState & F_SG_XFER) {
7138 partial_cnt = 0x0000;
7140 sg_index = p_SCCB->Sccb_sgseg;
7143 if (p_SCCB->Sccb_SGoffset) {
7145 partial_cnt = p_SCCB->Sccb_SGoffset;
7149 while (((unsigned long)sg_index *
7150 (unsigned long)SG_ELEMENT_SIZE) < p_SCCB->DataLength) {
7151 segp = (struct blogic_sg_seg *)(p_SCCB->DataPointer) +
7153 partial_cnt += segp->segbytes;
7157 p_SCCB->DataLength = partial_cnt;
7162 p_SCCB->DataLength -= p_SCCB->Sccb_ATC;
7166 /*---------------------------------------------------------------------
7168 * Function: Wait 1 Second
7170 * Description: Wait for 1 second.
7172 *---------------------------------------------------------------------*/
7174 static void FPT_Wait1Second(u32 p_port)
7178 for (i = 0; i < 4; i++) {
7180 FPT_Wait(p_port, TO_250ms);
7182 if ((RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST))
7185 if ((RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL))
7190 /*---------------------------------------------------------------------
7192 * Function: FPT_Wait
7194 * Description: Wait the desired delay.
7196 *---------------------------------------------------------------------*/
7198 static void FPT_Wait(u32 p_port, unsigned char p_delay)
7200 unsigned char old_timer;
7201 unsigned char green_flag;
7203 old_timer = RD_HARPOON(p_port + hp_seltimeout);
7205 green_flag = RD_HARPOON(p_port + hp_clkctrl_0);
7206 WR_HARPOON(p_port + hp_clkctrl_0, CLKCTRL_DEFAULT);
7208 WR_HARPOON(p_port + hp_seltimeout, p_delay);
7209 WRW_HARPOON((p_port + hp_intstat), TIMEOUT);
7210 WRW_HARPOON((p_port + hp_intena), (FPT_default_intena & ~TIMEOUT));
7212 WR_HARPOON(p_port + hp_portctrl_0,
7213 (RD_HARPOON(p_port + hp_portctrl_0) | START_TO));
7215 while (!(RDW_HARPOON((p_port + hp_intstat)) & TIMEOUT)) {
7217 if ((RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST))
7220 if ((RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL))
7224 WR_HARPOON(p_port + hp_portctrl_0,
7225 (RD_HARPOON(p_port + hp_portctrl_0) & ~START_TO));
7227 WRW_HARPOON((p_port + hp_intstat), TIMEOUT);
7228 WRW_HARPOON((p_port + hp_intena), FPT_default_intena);
7230 WR_HARPOON(p_port + hp_clkctrl_0, green_flag);
7232 WR_HARPOON(p_port + hp_seltimeout, old_timer);
7235 /*---------------------------------------------------------------------
7237 * Function: Enable/Disable Write to EEPROM
7239 * Description: The EEPROM must first be enabled for writes
7240 * A total of 9 clocks are needed.
7242 *---------------------------------------------------------------------*/
7244 static void FPT_utilEEWriteOnOff(u32 p_port, unsigned char p_mode)
7246 unsigned char ee_value;
7249 (unsigned char)(RD_HARPOON(p_port + hp_ee_ctrl) &
7250 (EXT_ARB_ACK | SCSI_TERM_ENA_H));
7254 FPT_utilEESendCmdAddr(p_port, EWEN, EWEN_ADDR);
7258 FPT_utilEESendCmdAddr(p_port, EWDS, EWDS_ADDR);
7260 WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS)); /*Turn off CS */
7261 WR_HARPOON(p_port + hp_ee_ctrl, ee_value); /*Turn off Master Select */
7264 /*---------------------------------------------------------------------
7266 * Function: Write EEPROM
7268 * Description: Write a word to the EEPROM at the specified
7271 *---------------------------------------------------------------------*/
7273 static void FPT_utilEEWrite(u32 p_port, unsigned short ee_data,
7274 unsigned short ee_addr)
7277 unsigned char ee_value;
7282 char)((RD_HARPOON(p_port + hp_ee_ctrl) &
7283 (EXT_ARB_ACK | SCSI_TERM_ENA_H)) | (SEE_MS | SEE_CS));
7285 FPT_utilEESendCmdAddr(p_port, EE_WRITE, ee_addr);
7287 ee_value |= (SEE_MS + SEE_CS);
7289 for (i = 0x8000; i != 0; i >>= 1) {
7294 ee_value &= ~SEE_DO;
7296 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7297 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7298 ee_value |= SEE_CLK; /* Clock data! */
7299 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7300 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7301 ee_value &= ~SEE_CLK;
7302 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7303 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7305 ee_value &= (EXT_ARB_ACK | SCSI_TERM_ENA_H);
7306 WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS));
7308 FPT_Wait(p_port, TO_10ms);
7310 WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS | SEE_CS)); /* Set CS to EEPROM */
7311 WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS)); /* Turn off CS */
7312 WR_HARPOON(p_port + hp_ee_ctrl, ee_value); /* Turn off Master Select */
7315 /*---------------------------------------------------------------------
7317 * Function: Read EEPROM
7319 * Description: Read a word from the EEPROM at the desired
7322 *---------------------------------------------------------------------*/
7324 static unsigned short FPT_utilEERead(u32 p_port,
7325 unsigned short ee_addr)
7327 unsigned short i, ee_data1, ee_data2;
7330 ee_data1 = FPT_utilEEReadOrg(p_port, ee_addr);
7332 ee_data2 = FPT_utilEEReadOrg(p_port, ee_addr);
7334 if (ee_data1 == ee_data2)
7337 ee_data1 = ee_data2;
7345 /*---------------------------------------------------------------------
7347 * Function: Read EEPROM Original
7349 * Description: Read a word from the EEPROM at the desired
7352 *---------------------------------------------------------------------*/
7354 static unsigned short FPT_utilEEReadOrg(u32 p_port, unsigned short ee_addr)
7357 unsigned char ee_value;
7358 unsigned short i, ee_data;
7362 char)((RD_HARPOON(p_port + hp_ee_ctrl) &
7363 (EXT_ARB_ACK | SCSI_TERM_ENA_H)) | (SEE_MS | SEE_CS));
7365 FPT_utilEESendCmdAddr(p_port, EE_READ, ee_addr);
7367 ee_value |= (SEE_MS + SEE_CS);
7370 for (i = 1; i <= 16; i++) {
7372 ee_value |= SEE_CLK; /* Clock data! */
7373 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7374 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7375 ee_value &= ~SEE_CLK;
7376 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7377 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7381 if (RD_HARPOON(p_port + hp_ee_ctrl) & SEE_DI)
7385 ee_value &= ~(SEE_MS + SEE_CS);
7386 WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS)); /*Turn off CS */
7387 WR_HARPOON(p_port + hp_ee_ctrl, ee_value); /*Turn off Master Select */
7392 /*---------------------------------------------------------------------
7394 * Function: Send EE command and Address to the EEPROM
7396 * Description: Transfers the correct command and sends the address
7399 *---------------------------------------------------------------------*/
7401 static void FPT_utilEESendCmdAddr(u32 p_port, unsigned char ee_cmd,
7402 unsigned short ee_addr)
7404 unsigned char ee_value;
7405 unsigned char narrow_flg;
7410 (unsigned char)(RD_HARPOON(p_port + hp_page_ctrl) &
7414 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7416 ee_value |= SEE_CS; /* Set CS to EEPROM */
7417 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7419 for (i = 0x04; i != 0; i >>= 1) {
7424 ee_value &= ~SEE_DO;
7426 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7427 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7428 ee_value |= SEE_CLK; /* Clock data! */
7429 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7430 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7431 ee_value &= ~SEE_CLK;
7432 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7433 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7447 ee_value &= ~SEE_DO;
7449 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7450 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7451 ee_value |= SEE_CLK; /* Clock data! */
7452 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7453 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7454 ee_value &= ~SEE_CLK;
7455 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7456 WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7462 static unsigned short FPT_CalcCrc16(unsigned char buffer[])
7464 unsigned short crc = 0;
7467 for (i = 0; i < ID_STRING_LENGTH; i++) {
7468 ch = (unsigned short)buffer[i];
7469 for (j = 0; j < 8; j++) {
7471 crc = (crc >> 1) ^ CRCMASK;
7480 static unsigned char FPT_CalcLrc(unsigned char buffer[])
7485 for (i = 0; i < ID_STRING_LENGTH; i++)
7491 The following inline definitions avoid type conflicts.
7494 static inline unsigned char
7495 FlashPoint__ProbeHostAdapter(struct fpoint_info *FlashPointInfo)
7497 return FlashPoint_ProbeHostAdapter((struct sccb_mgr_info *)
7501 static inline void *
7502 FlashPoint__HardwareResetHostAdapter(struct fpoint_info *FlashPointInfo)
7504 return FlashPoint_HardwareResetHostAdapter((struct sccb_mgr_info *)
7509 FlashPoint__ReleaseHostAdapter(void *CardHandle)
7511 FlashPoint_ReleaseHostAdapter(CardHandle);
7515 FlashPoint__StartCCB(void *CardHandle, struct blogic_ccb *CCB)
7517 FlashPoint_StartCCB(CardHandle, (struct sccb *)CCB);
7521 FlashPoint__AbortCCB(void *CardHandle, struct blogic_ccb *CCB)
7523 FlashPoint_AbortCCB(CardHandle, (struct sccb *)CCB);
7527 FlashPoint__InterruptPending(void *CardHandle)
7529 return FlashPoint_InterruptPending(CardHandle);
7533 FlashPoint__HandleInterrupt(void *CardHandle)
7535 return FlashPoint_HandleInterrupt(CardHandle);
7538 #define FlashPoint_ProbeHostAdapter FlashPoint__ProbeHostAdapter
7539 #define FlashPoint_HardwareResetHostAdapter FlashPoint__HardwareResetHostAdapter
7540 #define FlashPoint_ReleaseHostAdapter FlashPoint__ReleaseHostAdapter
7541 #define FlashPoint_StartCCB FlashPoint__StartCCB
7542 #define FlashPoint_AbortCCB FlashPoint__AbortCCB
7543 #define FlashPoint_InterruptPending FlashPoint__InterruptPending
7544 #define FlashPoint_HandleInterrupt FlashPoint__HandleInterrupt
7546 #else /* !CONFIG_SCSI_FLASHPOINT */
7549 Define prototypes for the FlashPoint SCCB Manager Functions.
7552 extern unsigned char FlashPoint_ProbeHostAdapter(struct fpoint_info *);
7553 extern void *FlashPoint_HardwareResetHostAdapter(struct fpoint_info *);
7554 extern void FlashPoint_StartCCB(void *, struct blogic_ccb *);
7555 extern int FlashPoint_AbortCCB(void *, struct blogic_ccb *);
7556 extern bool FlashPoint_InterruptPending(void *);
7557 extern int FlashPoint_HandleInterrupt(void *);
7558 extern void FlashPoint_ReleaseHostAdapter(void *);
7560 #endif /* CONFIG_SCSI_FLASHPOINT */