Merge tag 'pull-work.iov_iter-rebased' of git://git.kernel.org/pub/scm/linux/kernel...
[platform/kernel/linux-rpi.git] / drivers / scsi / FlashPoint.c
1 /*
2
3   FlashPoint.c -- FlashPoint SCCB Manager for Linux
4
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.
10
11   Copyright 1995-1996 by Mylex Corporation.  All Rights Reserved
12
13   This file is available under both the GNU General Public License
14   and a BSD-style copyright; see LICENSE.FlashPoint for details.
15
16 */
17
18
19 #ifdef CONFIG_SCSI_FLASHPOINT
20
21 #define MAX_CARDS       8
22 #undef BUSTYPE_PCI
23
24 #define CRCMASK 0xA001
25
26 #define FAILURE         0xFFFFFFFFL
27
28 struct sccb;
29 typedef void (*CALL_BK_FN) (struct sccb *);
30
31 struct sccb_mgr_info {
32         u32 si_baseaddr;
33         unsigned char si_present;
34         unsigned char si_intvect;
35         unsigned char si_id;
36         unsigned char si_lun;
37         u16 si_fw_revision;
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;
43         u16 si_mflags;
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];
49         u32 si_OS_reserved;
50         unsigned char si_XlatInfo[4];
51         u32 si_reserved2[5];
52         u32 si_secondary_range;
53 };
54
55 #define SCSI_PARITY_ENA           0x0001
56 #define LOW_BYTE_TERM             0x0010
57 #define HIGH_BYTE_TERM            0x0020
58 #define BUSTYPE_PCI       0x3
59
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
66
67 #define HARPOON_FAMILY        0x02
68
69 /* SCCB struct used for both SCCB and UCB manager compiles! 
70  * The UCB Manager treats the SCCB as it's 'native hardware structure' 
71  */
72
73 /*#pragma pack(1)*/
74 struct sccb {
75         unsigned char OperationCode;
76         unsigned char ControlByte;
77         unsigned char CdbLength;
78         unsigned char RequestSenseLength;
79         u32 DataLength;
80         void *DataPointer;
81         unsigned char CcbRes[2];
82         unsigned char HostStatus;
83         unsigned char TargetStatus;
84         unsigned char TargID;
85         unsigned char Lun;
86         unsigned char Cdb[12];
87         unsigned char CcbRes1;
88         unsigned char Reserved1;
89         u32 Reserved2;
90         u32 SensePointer;
91
92         CALL_BK_FN SccbCallback;        /* VOID (*SccbCallback)(); */
93         u32 SccbIOPort;                 /* Identifies board base port */
94         unsigned char SccbStatus;
95         unsigned char SCCBRes2;
96         u16 SccbOSFlags;
97
98         u32 Sccb_XferCnt;       /* actual transfer count */
99         u32 Sccb_ATC;
100         u32 SccbVirtDataPtr;    /* virtual addr for OS/2 */
101         u32 Sccb_res1;
102         u16 Sccb_MGRFlags;
103         u16 Sccb_sgseg;
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;
110         u32 Sccb_savedATC;
111         unsigned char Save_Cdb[6];
112         unsigned char Save_CdbLen;
113         unsigned char Sccb_XferState;
114         u32 Sccb_SGoffset;
115 };
116
117 #pragma pack()
118
119 #define SCATTER_GATHER_COMMAND    0x02
120 #define RESIDUAL_COMMAND          0x03
121 #define RESIDUAL_SG_COMMAND       0x04
122 #define RESET_COMMAND             0x81
123
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 */
128
129 #define NO_AUTO_REQUEST_SENSE    0x01   /* No Request Sense Buffer */
130
131 #define BUS_FREE_ST     0
132 #define SELECT_ST       1
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 */
137 #define COMMAND_ST      6
138 #define DATA_OUT_ST     7
139 #define DATA_IN_ST      8
140 #define DISCONNECT_ST   9
141 #define ABORT_ST        11
142
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
149
150 #define F_STATUSLOADED                 0x01
151 #define F_DEV_SELECTED                 0x04
152
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 */
158
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 */
162
163 #define SCCB_IN_PROCESS            0x00
164 #define SCCB_SUCCESS               0x01
165 #define SCCB_ABORT                 0x02
166 #define SCCB_ERROR                 0x04
167
168 #define  ORION_FW_REV      3110
169
170 #define QUEUE_DEPTH     254+1   /*1 for Normal disconnect 32 for Q'ing. */
171
172 #define MAX_MB_CARDS    4       /* Max. no of cards suppoerted on Mother Board */
173
174 #define MAX_SCSI_TAR    16
175 #define MAX_LUN         32
176 #define LUN_MASK                        0x1f
177
178 #define SG_BUF_CNT      16      /*Number of prefetched elements. */
179
180 #define SG_ELEMENT_SIZE 8       /*Eight byte per element. */
181
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))
188
189 #define  TAR_SYNC_MASK     (BIT(7)+BIT(6))
190 #define  SYNC_TRYING               BIT(6)
191 #define  SYNC_SUPPORTED    (BIT(7)+BIT(6))
192
193 #define  TAR_WIDE_MASK     (BIT(5)+BIT(4))
194 #define  WIDE_ENABLED              BIT(4)
195 #define  WIDE_NEGOCIATED   BIT(5)
196
197 #define  TAR_TAG_Q_MASK    (BIT(3)+BIT(2))
198 #define  TAG_Q_TRYING              BIT(2)
199 #define  TAG_Q_REJECT      BIT(3)
200
201 #define  TAR_ALLOW_DISC    BIT(0)
202
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))
207
208 #define  EE_WIDE_SCSI      BIT(7)
209
210 struct sccb_mgr_tar_info {
211
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];
223 };
224
225 struct nvram_info {
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
238                                                            of targets */
239         unsigned char niScamTbl[MAX_SCSI_TAR][4];       /* Compressed Scam name
240                                                            string of Targets */
241 };
242
243 #define MODEL_LT                1
244 #define MODEL_DL                2
245 #define MODEL_LW                3
246 #define MODEL_DW                4
247
248 struct sccb_card {
249         struct sccb *currentSCCB;
250         struct sccb_mgr_info *cardInfo;
251
252         u32 ioPort;
253
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;
260         unsigned char ourId;
261         struct nvram_info *pNvRamInfo;
262         struct sccb *discQ_Tbl[QUEUE_DEPTH];
263
264 };
265
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
274
275 #define  ID_STRING_LENGTH  32
276 #define  TYPE_CODE0        0x63 /*Level2 Mstr (bits 7-6),  */
277
278 #define  SLV_TYPE_CODE0    0xA3 /*Priority Bit set (bits 7-6),  */
279
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
285
286 #define  ID_0_7      0x18
287 #define  ID_8_F      0x11
288 #define  MISC_CODE   0x14
289 #define  CLR_P_FLAG  0x18
290
291 #define  INIT_SELTD  0x01
292 #define  LEVEL2_TAR  0x02
293
294 enum scam_id_st { ID0, ID1, ID2, ID3, ID4, ID5, ID6, ID7, ID8, ID9, ID10, ID11,
295             ID12,
296         ID13, ID14, ID15, ID_UNUSED, ID_UNASSIGNED, ID_ASSIGNED, LEGACY,
297         CLR_PRIORITY, NO_ID_AVAIL
298 };
299
300 typedef struct SCCBscam_info {
301
302         unsigned char id_string[ID_STRING_LENGTH];
303         enum scam_id_st state;
304
305 } SCCBSCAM_INFO;
306
307
308 #define  SMIDENT                 0x80
309 #define  DISC_PRIV               0x40
310
311 #define  SM8BIT                  0x00
312 #define  SM16BIT                 0x01
313
314 #define  SIX_BYTE_CMD            0x06
315 #define  TWELVE_BYTE_CMD         0x0C
316
317 #define  ASYNC                   0x00
318 #define  MAX_OFFSET              0x0F   /* Maxbyteoffset for Sync Xfers */
319
320 #define  EEPROM_WD_CNT     256
321
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
332
333 #define  IGNORE_B_SCAN     32
334 #define  SEND_START_ENA    34
335 #define  DEVICE_ENABLE     36
336
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
346
347 #define  EE_SCAMBASE      256
348
349 #define  SCAM_ENABLED   BIT(2)
350 #define  SCAM_LEVEL2    BIT(3)
351
352 #define RENEGO_ENA              BIT(10)
353 #define CONNIO_ENA              BIT(11)
354 #define  GREEN_PC_ENA   BIT(12)
355
356 #define  AUTO_RATE_00   00
357 #define  AUTO_RATE_05   01
358 #define  AUTO_RATE_10   02
359 #define  AUTO_RATE_20   03
360
361 #define  WIDE_NEGO_BIT     BIT(7)
362 #define  DISC_ENABLE_BIT   BIT(6)
363
364 #define  hp_vendor_id_0       0x00      /* LSB */
365 #define  ORION_VEND_0   0x4B
366
367 #define  hp_vendor_id_1       0x01      /* MSB */
368 #define  ORION_VEND_1   0x10
369
370 #define  hp_device_id_0       0x02      /* LSB */
371 #define  ORION_DEV_0    0x30
372
373 #define  hp_device_id_1       0x03      /* MSB */
374 #define  ORION_DEV_1    0x81
375
376         /* Sub Vendor ID and Sub Device ID only available in
377            Harpoon Version 2 and higher */
378
379 #define  hp_sub_device_id_0   0x06      /* LSB */
380
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)
386
387 #define  hp_sys_ctrl          0x0F
388
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              */
393
394 #define  hp_host_blk_cnt      0x13
395
396 #define  XFER_BLK64        0x06 /*     1 1 0 64 byte per block */
397
398 #define  BM_THRESHOLD      0x40 /* PCI mode can only xfer 16 bytes */
399
400 #define  hp_int_mask          0x17
401
402 #define  INT_CMD_COMPL     BIT(0)       /* DMA command complete   */
403 #define  INT_EXT_STATUS    BIT(1)       /* Extended Status Set    */
404
405 #define  hp_xfer_cnt_lo       0x18
406 #define  hp_xfer_cnt_hi       0x1A
407 #define  hp_xfer_cmd          0x1B
408
409 #define  XFER_HOST_DMA     0x00 /*     0 0 0 Transfer Host -> DMA */
410 #define  XFER_DMA_HOST     0x01 /*     0 0 1 Transfer DMA  -> Host */
411
412 #define  XFER_HOST_AUTO    0x00 /*     0 0 Auto Transfer Size   */
413
414 #define  XFER_DMA_8BIT     0x20 /*     0 1 8 BIT  Transfer Size */
415
416 #define  DISABLE_INT       BIT(7)       /*Do not interrupt at end of cmd. */
417
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))
420
421 #define  hp_host_addr_lo      0x1C
422 #define  hp_host_addr_hmi     0x1E
423
424 #define  hp_ee_ctrl           0x22
425
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)
433
434 #define  EE_READ           0x06
435 #define  EE_WRITE          0x05
436 #define  EWEN              0x04
437 #define  EWEN_ADDR         0x03C0
438 #define  EWDS              0x04
439 #define  EWDS_ADDR         0x0000
440
441 #define  hp_bm_ctrl           0x26
442
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)       /*?? */
447
448 #define  BMCTRL_DEFAULT    (FORCE1_XFER|FAST_SINGLE|SCSI_TERM_ENA_L)
449
450 #define  hp_sg_addr           0x28
451 #define  hp_page_ctrl         0x29
452
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 */
457
458 #define  hp_pci_stat_cfg      0x2D
459
460 #define  REC_MASTER_ABORT  BIT(5)       /*received Master abort */
461
462 #define  hp_rev_num           0x33
463
464 #define  hp_stack_data        0x34
465 #define  hp_stack_addr        0x35
466
467 #define  hp_ext_status        0x36
468
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)
478
479 #define  hp_int_status        0x37
480
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)       /* */
484
485 #define  hp_fifo_cnt          0x38
486
487 #define  hp_intena               0x40
488
489 #define  RESET           BIT(7)
490 #define  PROG_HLT                BIT(6)
491 #define  PARITY          BIT(5)
492 #define  FIFO            BIT(4)
493 #define  SEL             BIT(3)
494 #define  SCAM_SEL                BIT(2)
495 #define  RSEL            BIT(1)
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
508
509 #define  hp_intstat              0x42
510
511 #define  hp_scsisig           0x44
512
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)
521
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)       )
528
529 #define  hp_scsictrl_0        0x45
530
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)
536
537 #define  hp_portctrl_0        0x46
538
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)
547
548 #define  hp_scsireset         0x47
549
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)
556
557 #define  hp_xfercnt_0         0x48
558 #define  hp_xfercnt_2         0x4A
559
560 #define  hp_fifodata_0        0x4C
561 #define  hp_addstat           0x4E
562
563 #define  SCAM_TIMER        BIT(7)
564 #define  SCSI_MODE8        BIT(3)
565 #define  SCSI_PAR_ERR      BIT(0)
566
567 #define  hp_prgmcnt_0         0x4F
568
569 #define  hp_selfid_0          0x50
570 #define  hp_selfid_1          0x51
571 #define  hp_arb_id            0x52
572
573 #define  hp_select_id         0x53
574
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
580
581 #define  hp_synctarg_8        0x58
582 #define  hp_synctarg_9        0x59
583 #define  hp_synctarg_10       0x5A
584 #define  hp_synctarg_11       0x5B
585
586 #define  hp_synctarg_4        0x5C
587 #define  hp_synctarg_5        0x5D
588 #define  hp_synctarg_6        0x5E
589 #define  hp_synctarg_7        0x5F
590
591 #define  hp_synctarg_0        0x60
592 #define  hp_synctarg_1        0x61
593 #define  hp_synctarg_2        0x62
594 #define  hp_synctarg_3        0x63
595
596 #define  NARROW_SCSI       BIT(4)
597 #define  DEFAULT_OFFSET    0x0F
598
599 #define  hp_autostart_0       0x64
600 #define  hp_autostart_1       0x65
601 #define  hp_autostart_3       0x67
602
603 #define  AUTO_IMMED    BIT(5)
604 #define  SELECT   BIT(6)
605 #define  END_DATA (BIT(7)+BIT(6))
606
607 #define  hp_gp_reg_0          0x68
608 #define  hp_gp_reg_1          0x69
609 #define  hp_gp_reg_3          0x6B
610
611 #define  hp_seltimeout        0x6C
612
613 #define  TO_4ms            0x67 /* 3.9959ms */
614
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 */
619
620 #define  hp_clkctrl_0         0x6D
621
622 #define  PWR_DWN           BIT(6)
623 #define  ACTdeassert       BIT(4)
624 #define  CLK_40MHZ         (BIT(1) + BIT(0))
625
626 #define  CLKCTRL_DEFAULT   (ACTdeassert | CLK_40MHZ)
627
628 #define  hp_fiforead          0x6E
629 #define  hp_fifowrite         0x6F
630
631 #define  hp_offsetctr         0x70
632 #define  hp_xferstat          0x71
633
634 #define  FIFO_EMPTY        BIT(6)
635
636 #define  hp_portctrl_1        0x72
637
638 #define  CHK_SCSI_P        BIT(3)
639 #define  HOST_MODE8        BIT(0)
640
641 #define  hp_xfer_pad          0x73
642
643 #define  ID_UNLOCK         BIT(3)
644
645 #define  hp_scsidata_0        0x74
646 #define  hp_scsidata_1        0x75
647
648 #define  hp_aramBase          0x80
649 #define  BIOS_DATA_OFFSET     0x60
650 #define  BIOS_RELATIVE_CARD   0x64
651
652 #define  AR3      (BIT(9) + BIT(8))
653 #define  SDATA    BIT(10)
654
655 #define  CRD_OP   BIT(11)       /* Cmp Reg. w/ Data */
656
657 #define  CRR_OP   BIT(12)       /* Cmp Reg. w. Reg. */
658
659 #define  CPE_OP   (BIT(14)+BIT(11))     /* Cmp SCSI phs & Branch EQ */
660
661 #define  CPN_OP   (BIT(14)+BIT(12))     /* Cmp SCSI phs & Branch NOT EQ */
662
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))
669
670 #define  BRH_OP   BIT(13)       /* Branch */
671
672 #define  ALWAYS   0x00
673 #define  EQUAL    BIT(8)
674 #define  NOT_EQ   BIT(9)
675
676 #define  TCB_OP   (BIT(13)+BIT(11))     /* Test condition & branch */
677
678 #define  FIFO_0      BIT(10)
679
680 #define  MPM_OP   BIT(15)       /* Match phase and move data */
681
682 #define  MRR_OP   BIT(14)       /* Move DReg. to Reg. */
683
684 #define  S_IDREG  (BIT(2)+BIT(1)+BIT(0))
685
686 #define  D_AR0    0x00
687 #define  D_AR1    BIT(0)
688 #define  D_BUCKET (BIT(2) + BIT(1) + BIT(0))
689
690 #define  RAT_OP      (BIT(14)+BIT(13)+BIT(11))
691
692 #define  SSI_OP      (BIT(15)+BIT(11))
693
694 #define  SSI_ITAR_DISC  (ITAR_DISC >> 8)
695 #define  SSI_IDO_STRT   (IDO_STRT >> 8)
696
697 #define  SSI_ICMD_COMP  (ICMD_COMP >> 8)
698 #define  SSI_ITICKLE    (ITICKLE >> 8)
699
700 #define  SSI_IUNKWN     (IUNKWN >> 8)
701 #define  SSI_INO_CC     (IUNKWN >> 8)
702 #define  SSI_IRFAIL     (IUNKWN >> 8)
703
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 */
715
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
720
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
726
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), \
729                                  xfercnt <<= 16,\
730                                  xfercnt |= RDW_HARPOON((unsigned short)(port+hp_xfercnt_0)))
731  */
732 #define HP_SETUP_ADDR_CNT(port,addr,count) (WRW_HARPOON((port+hp_host_addr_lo), (unsigned short)(addr & 0x0000FFFFL)),\
733          addr >>= 16,\
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)),\
737          count >>= 16,\
738          WR_HARPOON(port+hp_xfer_cnt_hi, (count & 0xFF)))
739
740 #define ACCEPT_MSG(port) {while(RD_HARPOON(port+hp_scsisig) & SCSI_REQ){}\
741                           WR_HARPOON(port+hp_scsisig, S_ILL_PH);}
742
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));}
745
746 #define DISABLE_AUTO(port) (WR_HARPOON(port+hp_scsireset, PROG_RESET),\
747                         WR_HARPOON(port+hp_scsireset, 0x00))
748
749 #define ARAM_ACCESS(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
750                              (RD_HARPOON(p_port+hp_page_ctrl) | SGRAM_ARAM)))
751
752 #define SGRAM_ACCESS(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
753                              (RD_HARPOON(p_port+hp_page_ctrl) & ~SGRAM_ARAM)))
754
755 #define MDISABLE_INT(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
756                              (RD_HARPOON(p_port+hp_page_ctrl) | G_INT_DISABLE)))
757
758 #define MENABLE_INT(p_port) (WR_HARPOON(p_port+hp_page_ctrl, \
759                              (RD_HARPOON(p_port+hp_page_ctrl) & ~G_INT_DISABLE)))
760
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,
779                         unsigned char data);
780 static unsigned char FPT_ChkIfChipInitialized(u32 ioPort);
781
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);
785
786 static void FPT_sinits(struct sccb *p_sccb, unsigned char p_card);
787 static void FPT_RNVRamData(struct nvram_info *pNvRamInfo);
788
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);
792
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[]);
807
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);
819
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);
827
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);
831
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);
835
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);
845
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);
850
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);
856
857 static void FPT_scini(unsigned char p_card, unsigned char p_our_id,
858                       unsigned char p_power_up);
859
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[]);
879
880 static void FPT_autoCmdCmplt(u32 p_port, unsigned char p_card);
881 static void FPT_autoLoadDefaultMap(u32 p_port);
882
883 static struct sccb_mgr_tar_info FPT_sccbMgrTbl[MAX_CARDS][MAX_SCSI_TAR] =
884     { {{0}} };
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} };
888
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
895 };
896
897 static unsigned short FPT_default_intena = 0;
898
899 static void (*FPT_s_PhaseTbl[8]) (u32, unsigned char) = {
900 0};
901
902 /*---------------------------------------------------------------------
903  *
904  * Function: FlashPoint_ProbeHostAdapter
905  *
906  * Description: Setup and/or Search for cards and return info to caller.
907  *
908  *---------------------------------------------------------------------*/
909
910 static int FlashPoint_ProbeHostAdapter(struct sccb_mgr_info *pCardInfo)
911 {
912         static unsigned char first_time = 1;
913
914         unsigned char i, j, id, ScamFlg;
915         unsigned short temp, temp2, temp3, temp4, temp5, temp6;
916         u32 ioport;
917         struct nvram_info *pCurrNvRam;
918
919         ioport = pCardInfo->si_baseaddr;
920
921         if (RD_HARPOON(ioport + hp_vendor_id_0) != ORION_VEND_0)
922                 return (int)FAILURE;
923
924         if ((RD_HARPOON(ioport + hp_vendor_id_1) != ORION_VEND_1))
925                 return (int)FAILURE;
926
927         if ((RD_HARPOON(ioport + hp_device_id_0) != ORION_DEV_0))
928                 return (int)FAILURE;
929
930         if ((RD_HARPOON(ioport + hp_device_id_1) != ORION_DEV_1))
931                 return (int)FAILURE;
932
933         if (RD_HARPOON(ioport + hp_rev_num) != 0x0f) {
934
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
938         device. */
939
940                 if (RD_HARPOON(ioport + hp_sub_device_id_0) & 0x0f)
941                         return (int)FAILURE;
942         }
943
944         if (first_time) {
945                 FPT_SccbMgrTableInitAll();
946                 first_time = 0;
947                 FPT_mbCards = 0;
948         }
949
950         if (FPT_RdStack(ioport, 0) != 0x00) {
951                 if (FPT_ChkIfChipInitialized(ioport) == 0) {
952                         pCurrNvRam = NULL;
953                         WR_HARPOON(ioport + hp_semaphore, 0x00);
954                         FPT_XbowInit(ioport, 0);        /*Must Init the SCSI before attempting */
955                         FPT_DiagEEPROM(ioport);
956                 } else {
957                         if (FPT_mbCards < MAX_MB_CARDS) {
958                                 pCurrNvRam = &FPT_nvRamInfo[FPT_mbCards];
959                                 FPT_mbCards++;
960                                 pCurrNvRam->niBaseAddr = ioport;
961                                 FPT_RNVRamData(pCurrNvRam);
962                         } else
963                                 return (int)FAILURE;
964                 }
965         } else
966                 pCurrNvRam = NULL;
967
968         WR_HARPOON(ioport + hp_clkctrl_0, CLKCTRL_DEFAULT);
969         WR_HARPOON(ioport + hp_sys_ctrl, 0x00);
970
971         if (pCurrNvRam)
972                 pCardInfo->si_id = pCurrNvRam->niAdapId;
973         else
974                 pCardInfo->si_id =
975                     (unsigned
976                      char)(FPT_utilEERead(ioport,
977                                           (ADAPTER_SCSI_ID /
978                                            2)) & (unsigned char)0x0FF);
979
980         pCardInfo->si_lun = 0x00;
981         pCardInfo->si_fw_revision = ORION_FW_REV;
982         temp2 = 0x0000;
983         temp3 = 0x0000;
984         temp4 = 0x0000;
985         temp5 = 0x0000;
986         temp6 = 0x0000;
987
988         for (id = 0; id < (16 / 2); id++) {
989
990                 if (pCurrNvRam) {
991                         temp = (unsigned short)pCurrNvRam->niSyncTbl[id];
992                         temp = ((temp & 0x03) + ((temp << 4) & 0xc0)) +
993                             (((temp << 4) & 0x0300) + ((temp << 8) & 0xc000));
994                 } else
995                         temp =
996                             FPT_utilEERead(ioport,
997                                            (unsigned short)((SYNC_RATE_TBL / 2)
998                                                             + id));
999
1000                 for (i = 0; i < 2; temp >>= 8, i++) {
1001
1002                         temp2 >>= 1;
1003                         temp3 >>= 1;
1004                         temp4 >>= 1;
1005                         temp5 >>= 1;
1006                         temp6 >>= 1;
1007                         switch (temp & 0x3) {
1008                         case AUTO_RATE_20:      /* Synchronous, 20 mega-transfers/second */
1009                                 temp6 |= 0x8000;
1010                                 fallthrough;
1011                         case AUTO_RATE_10:      /* Synchronous, 10 mega-transfers/second */
1012                                 temp5 |= 0x8000;
1013                                 fallthrough;
1014                         case AUTO_RATE_05:      /* Synchronous, 5 mega-transfers/second */
1015                                 temp2 |= 0x8000;
1016                                 fallthrough;
1017                         case AUTO_RATE_00:      /* Asynchronous */
1018                                 break;
1019                         }
1020
1021                         if (temp & DISC_ENABLE_BIT)
1022                                 temp3 |= 0x8000;
1023
1024                         if (temp & WIDE_NEGO_BIT)
1025                                 temp4 |= 0x8000;
1026
1027                 }
1028         }
1029
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;
1035
1036         if (pCurrNvRam)
1037                 i = pCurrNvRam->niSysConf;
1038         else
1039                 i = (unsigned
1040                      char)(FPT_utilEERead(ioport, (SYSTEM_CONFIG / 2)));
1041
1042         if (pCurrNvRam)
1043                 ScamFlg = pCurrNvRam->niScamConf;
1044         else
1045                 ScamFlg =
1046                     (unsigned char)FPT_utilEERead(ioport, SCAM_CONFIG / 2);
1047
1048         pCardInfo->si_mflags = 0x0000;
1049
1050         if (i & 0x01)
1051                 pCardInfo->si_mflags |= SCSI_PARITY_ENA;
1052
1053         if (!(i & 0x02))
1054                 pCardInfo->si_mflags |= SOFT_RESET;
1055
1056         if (i & 0x10)
1057                 pCardInfo->si_mflags |= EXTENDED_TRANSLATION;
1058
1059         if (ScamFlg & SCAM_ENABLED)
1060                 pCardInfo->si_mflags |= FLAG_SCAM_ENABLED;
1061
1062         if (ScamFlg & SCAM_LEVEL2)
1063                 pCardInfo->si_mflags |= FLAG_SCAM_LEVEL2;
1064
1065         j = (RD_HARPOON(ioport + hp_bm_ctrl) & ~SCSI_TERM_ENA_L);
1066         if (i & 0x04) {
1067                 j |= SCSI_TERM_ENA_L;
1068         }
1069         WR_HARPOON(ioport + hp_bm_ctrl, j);
1070
1071         j = (RD_HARPOON(ioport + hp_ee_ctrl) & ~SCSI_TERM_ENA_H);
1072         if (i & 0x08) {
1073                 j |= SCSI_TERM_ENA_H;
1074         }
1075         WR_HARPOON(ioport + hp_ee_ctrl, j);
1076
1077         if (!(RD_HARPOON(ioport + hp_page_ctrl) & NARROW_SCSI_CARD))
1078
1079                 pCardInfo->si_mflags |= SUPPORT_16TAR_32LUN;
1080
1081         pCardInfo->si_card_family = HARPOON_FAMILY;
1082         pCardInfo->si_bustype = BUSTYPE_PCI;
1083
1084         if (pCurrNvRam) {
1085                 pCardInfo->si_card_model[0] = '9';
1086                 switch (pCurrNvRam->niModel & 0x0f) {
1087                 case MODEL_LT:
1088                         pCardInfo->si_card_model[1] = '3';
1089                         pCardInfo->si_card_model[2] = '0';
1090                         break;
1091                 case MODEL_LW:
1092                         pCardInfo->si_card_model[1] = '5';
1093                         pCardInfo->si_card_model[2] = '0';
1094                         break;
1095                 case MODEL_DL:
1096                         pCardInfo->si_card_model[1] = '3';
1097                         pCardInfo->si_card_model[2] = '2';
1098                         break;
1099                 case MODEL_DW:
1100                         pCardInfo->si_card_model[1] = '5';
1101                         pCardInfo->si_card_model[2] = '2';
1102                         break;
1103                 }
1104         } else {
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));
1108
1109                 pCardInfo->si_card_model[1] = (unsigned char)(temp & 0x00FF);
1110                 pCardInfo->si_card_model[2] = (unsigned char)(temp >> 8);
1111         }
1112
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);
1125         } else {
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)));
1130                 temp3 = 0;
1131                 for (i = 0; i < 8; i++) {
1132                         temp3 <<= 1;
1133                         if (!(RD_HARPOON(ioport + hp_ee_ctrl) & BIT(7)))
1134                                 temp3 |= 1;
1135                         WR_HARPOON(ioport + hp_xfer_pad, (temp2 & ~BIT(4)));
1136                         WR_HARPOON(ioport + hp_xfer_pad, (temp2 | BIT(4)));
1137                 }
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;
1144         }
1145
1146         ARAM_ACCESS(ioport);
1147
1148         for (i = 0; i < 4; i++) {
1149
1150                 pCardInfo->si_XlatInfo[i] =
1151                     RD_HARPOON(ioport + hp_aramBase + BIOS_DATA_OFFSET + i);
1152         }
1153
1154         /* return with -1 if no sort, else return with
1155            logical card number sorted by BIOS (zero-based) */
1156
1157         pCardInfo->si_relative_cardnum =
1158             (unsigned
1159              char)(RD_HARPOON(ioport + hp_aramBase + BIOS_RELATIVE_CARD) - 1);
1160
1161         SGRAM_ACCESS(ioport);
1162
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;
1171
1172         pCardInfo->si_present = 0x01;
1173
1174         return 0;
1175 }
1176
1177 /*---------------------------------------------------------------------
1178  *
1179  * Function: FlashPoint_HardwareResetHostAdapter
1180  *
1181  * Description: Setup adapter for normal operation (hard reset).
1182  *
1183  *---------------------------------------------------------------------*/
1184
1185 static void *FlashPoint_HardwareResetHostAdapter(struct sccb_mgr_info
1186                                                          *pCardInfo)
1187 {
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;
1192         u32 ioport;
1193
1194         ioport = pCardInfo->si_baseaddr;
1195
1196         for (thisCard = 0; thisCard <= MAX_CARDS; thisCard++) {
1197
1198                 if (thisCard == MAX_CARDS)
1199                         return (void *)FAILURE;
1200
1201                 if (FPT_BL_Card[thisCard].ioPort == ioport) {
1202
1203                         CurrCard = &FPT_BL_Card[thisCard];
1204                         FPT_SccbMgrTableInitCard(CurrCard, thisCard);
1205                         break;
1206                 }
1207
1208                 else if (FPT_BL_Card[thisCard].ioPort == 0x00) {
1209
1210                         FPT_BL_Card[thisCard].ioPort = ioport;
1211                         CurrCard = &FPT_BL_Card[thisCard];
1212
1213                         if (FPT_mbCards)
1214                                 for (i = 0; i < FPT_mbCards; i++) {
1215                                         if (CurrCard->ioPort ==
1216                                             FPT_nvRamInfo[i].niBaseAddr)
1217                                                 CurrCard->pNvRamInfo =
1218                                                     &FPT_nvRamInfo[i];
1219                                 }
1220                         FPT_SccbMgrTableInitCard(CurrCard, thisCard);
1221                         CurrCard->cardIndex = thisCard;
1222                         CurrCard->cardInfo = pCardInfo;
1223
1224                         break;
1225                 }
1226         }
1227
1228         pCurrNvRam = CurrCard->pNvRamInfo;
1229
1230         if (pCurrNvRam) {
1231                 ScamFlg = pCurrNvRam->niScamConf;
1232         } else {
1233                 ScamFlg =
1234                     (unsigned char)FPT_utilEERead(ioport, SCAM_CONFIG / 2);
1235         }
1236
1237         FPT_BusMasterInit(ioport);
1238         FPT_XbowInit(ioport, ScamFlg);
1239
1240         FPT_autoLoadDefaultMap(ioport);
1241
1242         for (i = 0, id = 0x01; i != pCardInfo->si_id; i++, id <<= 1) {
1243         }
1244
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;
1249
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));
1253
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);
1258
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);
1263
1264         if (!(pCardInfo->si_mflags & SOFT_RESET)) {
1265
1266                 FPT_sresb(ioport, thisCard);
1267
1268                 FPT_scini(thisCard, pCardInfo->si_id, 0);
1269         }
1270
1271         if (pCardInfo->si_mflags & POST_ALL_UNDERRRUNS)
1272                 CurrCard->globalFlags |= F_NO_FILTER;
1273
1274         if (pCurrNvRam) {
1275                 if (pCurrNvRam->niSysConf & 0x10)
1276                         CurrCard->globalFlags |= F_GREEN_PC;
1277         } else {
1278                 if (FPT_utilEERead(ioport, (SYSTEM_CONFIG / 2)) & GREEN_PC_ENA)
1279                         CurrCard->globalFlags |= F_GREEN_PC;
1280         }
1281
1282         /* Set global flag to indicate Re-Negotiation to be done on all
1283            ckeck condition */
1284         if (pCurrNvRam) {
1285                 if (pCurrNvRam->niScsiConf & 0x04)
1286                         CurrCard->globalFlags |= F_DO_RENEGO;
1287         } else {
1288                 if (FPT_utilEERead(ioport, (SCSI_CONFIG / 2)) & RENEGO_ENA)
1289                         CurrCard->globalFlags |= F_DO_RENEGO;
1290         }
1291
1292         if (pCurrNvRam) {
1293                 if (pCurrNvRam->niScsiConf & 0x08)
1294                         CurrCard->globalFlags |= F_CONLUN_IO;
1295         } else {
1296                 if (FPT_utilEERead(ioport, (SCSI_CONFIG / 2)) & CONNIO_ENA)
1297                         CurrCard->globalFlags |= F_CONLUN_IO;
1298         }
1299
1300         temp = pCardInfo->si_per_targ_no_disc;
1301
1302         for (i = 0, id = 1; i < MAX_SCSI_TAR; i++, id <<= 1) {
1303
1304                 if (temp & id)
1305                         FPT_sccbMgrTbl[thisCard][i].TarStatus |= TAR_ALLOW_DISC;
1306         }
1307
1308         sync_bit_map = 0x0001;
1309
1310         for (id = 0; id < (MAX_SCSI_TAR / 2); id++) {
1311
1312                 if (pCurrNvRam) {
1313                         temp = (unsigned short)pCurrNvRam->niSyncTbl[id];
1314                         temp = ((temp & 0x03) + ((temp << 4) & 0xc0)) +
1315                             (((temp << 4) & 0x0300) + ((temp << 8) & 0xc000));
1316                 } else
1317                         temp =
1318                             FPT_utilEERead(ioport,
1319                                            (unsigned short)((SYNC_RATE_TBL / 2)
1320                                                             + id));
1321
1322                 for (i = 0; i < 2; temp >>= 8, i++) {
1323
1324                         if (pCardInfo->si_per_targ_init_sync & sync_bit_map) {
1325
1326                                 FPT_sccbMgrTbl[thisCard][id * 2 +
1327                                                          i].TarEEValue =
1328                                     (unsigned char)temp;
1329                         }
1330
1331                         else {
1332                                 FPT_sccbMgrTbl[thisCard][id * 2 +
1333                                                          i].TarStatus |=
1334                                     SYNC_SUPPORTED;
1335                                 FPT_sccbMgrTbl[thisCard][id * 2 +
1336                                                          i].TarEEValue =
1337                                     (unsigned char)(temp & ~EE_SYNC_MASK);
1338                         }
1339
1340 /*         if ((pCardInfo->si_per_targ_wide_nego & sync_bit_map) ||
1341             (id*2+i >= 8)){
1342 */
1343                         if (pCardInfo->si_per_targ_wide_nego & sync_bit_map) {
1344
1345                                 FPT_sccbMgrTbl[thisCard][id * 2 +
1346                                                          i].TarEEValue |=
1347                                     EE_WIDE_SCSI;
1348
1349                         }
1350
1351                         else {  /* NARROW SCSI */
1352                                 FPT_sccbMgrTbl[thisCard][id * 2 +
1353                                                          i].TarStatus |=
1354                                     WIDE_NEGOCIATED;
1355                         }
1356
1357                         sync_bit_map <<= 1;
1358
1359                 }
1360         }
1361
1362         WR_HARPOON((ioport + hp_semaphore),
1363                    (unsigned char)(RD_HARPOON((ioport + hp_semaphore)) |
1364                                    SCCB_MGR_PRESENT));
1365
1366         return (void *)CurrCard;
1367 }
1368
1369 static void FlashPoint_ReleaseHostAdapter(void *pCurrCard)
1370 {
1371         unsigned char i;
1372         u32 portBase;
1373         u32 regOffset;
1374         u32 scamData;
1375         u32 *pScamTbl;
1376         struct nvram_info *pCurrNvRam;
1377
1378         pCurrNvRam = ((struct sccb_card *)pCurrCard)->pNvRamInfo;
1379
1380         if (pCurrNvRam) {
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);
1386
1387                 for (i = 0; i < MAX_SCSI_TAR / 2; i++)
1388                         FPT_WrStack(pCurrNvRam->niBaseAddr,
1389                                     (unsigned char)(i + 5),
1390                                     pCurrNvRam->niSyncTbl[i]);
1391
1392                 portBase = pCurrNvRam->niBaseAddr;
1393
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);
1399                 }
1400
1401         } else {
1402                 FPT_WrStack(((struct sccb_card *)pCurrCard)->ioPort, 0, 0);
1403         }
1404 }
1405
1406 static void FPT_RNVRamData(struct nvram_info *pNvRamInfo)
1407 {
1408         unsigned char i;
1409         u32 portBase;
1410         u32 regOffset;
1411         u32 scamData;
1412         u32 *pScamTbl;
1413
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);
1419
1420         for (i = 0; i < MAX_SCSI_TAR / 2; i++)
1421                 pNvRamInfo->niSyncTbl[i] =
1422                     FPT_RdStack(pNvRamInfo->niBaseAddr, (unsigned char)(i + 5));
1423
1424         portBase = pNvRamInfo->niBaseAddr;
1425
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;
1431         }
1432
1433 }
1434
1435 static unsigned char FPT_RdStack(u32 portBase, unsigned char index)
1436 {
1437         WR_HARPOON(portBase + hp_stack_addr, index);
1438         return RD_HARPOON(portBase + hp_stack_data);
1439 }
1440
1441 static void FPT_WrStack(u32 portBase, unsigned char index, unsigned char data)
1442 {
1443         WR_HARPOON(portBase + hp_stack_addr, index);
1444         WR_HARPOON(portBase + hp_stack_data, data);
1445 }
1446
1447 static unsigned char FPT_ChkIfChipInitialized(u32 ioPort)
1448 {
1449         if ((RD_HARPOON(ioPort + hp_arb_id) & 0x0f) != FPT_RdStack(ioPort, 4))
1450                 return 0;
1451         if ((RD_HARPOON(ioPort + hp_clkctrl_0) & CLKCTRL_DEFAULT)
1452             != CLKCTRL_DEFAULT)
1453                 return 0;
1454         if ((RD_HARPOON(ioPort + hp_seltimeout) == TO_250ms) ||
1455             (RD_HARPOON(ioPort + hp_seltimeout) == TO_290ms))
1456                 return 1;
1457         return 0;
1458
1459 }
1460
1461 /*---------------------------------------------------------------------
1462  *
1463  * Function: FlashPoint_StartCCB
1464  *
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.
1468  *
1469  *---------------------------------------------------------------------*/
1470 static void FlashPoint_StartCCB(void *curr_card, struct sccb *p_Sccb)
1471 {
1472         u32 ioport;
1473         unsigned char thisCard, lun;
1474         struct sccb *pSaveSccb;
1475         CALL_BK_FN callback;
1476         struct sccb_card *pCurrCard = curr_card;
1477
1478         thisCard = pCurrCard->cardIndex;
1479         ioport = pCurrCard->ioPort;
1480
1481         if ((p_Sccb->TargID >= MAX_SCSI_TAR) || (p_Sccb->Lun >= MAX_LUN)) {
1482
1483                 p_Sccb->HostStatus = SCCB_COMPLETE;
1484                 p_Sccb->SccbStatus = SCCB_ERROR;
1485                 callback = (CALL_BK_FN) p_Sccb->SccbCallback;
1486                 if (callback)
1487                         callback(p_Sccb);
1488
1489                 return;
1490         }
1491
1492         FPT_sinits(p_Sccb, thisCard);
1493
1494         if (!pCurrCard->cmdCounter) {
1495                 WR_HARPOON(ioport + hp_semaphore,
1496                            (RD_HARPOON(ioport + hp_semaphore)
1497                             | SCCB_MGR_ACTIVE));
1498
1499                 if (pCurrCard->globalFlags & F_GREEN_PC) {
1500                         WR_HARPOON(ioport + hp_clkctrl_0, CLKCTRL_DEFAULT);
1501                         WR_HARPOON(ioport + hp_sys_ctrl, 0x00);
1502                 }
1503         }
1504
1505         pCurrCard->cmdCounter++;
1506
1507         if (RD_HARPOON(ioport + hp_semaphore) & BIOS_IN_USE) {
1508
1509                 WR_HARPOON(ioport + hp_semaphore,
1510                            (RD_HARPOON(ioport + hp_semaphore)
1511                             | TICKLE_ME));
1512                 if (p_Sccb->OperationCode == RESET_COMMAND) {
1513                         pSaveSccb =
1514                             pCurrCard->currentSCCB;
1515                         pCurrCard->currentSCCB = p_Sccb;
1516                         FPT_queueSelectFail(&FPT_BL_Card[thisCard], thisCard);
1517                         pCurrCard->currentSCCB =
1518                             pSaveSccb;
1519                 } else {
1520                         FPT_queueAddSccb(p_Sccb, thisCard);
1521                 }
1522         }
1523
1524         else if ((RD_HARPOON(ioport + hp_page_ctrl) & G_INT_DISABLE)) {
1525
1526                 if (p_Sccb->OperationCode == RESET_COMMAND) {
1527                         pSaveSccb =
1528                             pCurrCard->currentSCCB;
1529                         pCurrCard->currentSCCB = p_Sccb;
1530                         FPT_queueSelectFail(&FPT_BL_Card[thisCard], thisCard);
1531                         pCurrCard->currentSCCB =
1532                             pSaveSccb;
1533                 } else {
1534                         FPT_queueAddSccb(p_Sccb, thisCard);
1535                 }
1536         }
1537
1538         else {
1539
1540                 MDISABLE_INT(ioport);
1541
1542                 if ((pCurrCard->globalFlags & F_CONLUN_IO) &&
1543                     ((FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].
1544                       TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING))
1545                         lun = p_Sccb->Lun;
1546                 else
1547                         lun = 0;
1548                 if ((pCurrCard->currentSCCB == NULL) &&
1549                     (FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].TarSelQ_Cnt == 0)
1550                     && (FPT_sccbMgrTbl[thisCard][p_Sccb->TargID].TarLUNBusy[lun]
1551                         == 0)) {
1552
1553                         pCurrCard->currentSCCB = p_Sccb;
1554                         FPT_ssel(p_Sccb->SccbIOPort, thisCard);
1555                 }
1556
1557                 else {
1558
1559                         if (p_Sccb->OperationCode == RESET_COMMAND) {
1560                                 pSaveSccb = pCurrCard->currentSCCB;
1561                                 pCurrCard->currentSCCB = p_Sccb;
1562                                 FPT_queueSelectFail(&FPT_BL_Card[thisCard],
1563                                                     thisCard);
1564                                 pCurrCard->currentSCCB = pSaveSccb;
1565                         } else {
1566                                 FPT_queueAddSccb(p_Sccb, thisCard);
1567                         }
1568                 }
1569
1570                 MENABLE_INT(ioport);
1571         }
1572
1573 }
1574
1575 /*---------------------------------------------------------------------
1576  *
1577  * Function: FlashPoint_AbortCCB
1578  *
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.
1582  *
1583  *---------------------------------------------------------------------*/
1584 static int FlashPoint_AbortCCB(void *pCurrCard, struct sccb *p_Sccb)
1585 {
1586         u32 ioport;
1587
1588         unsigned char thisCard;
1589         CALL_BK_FN callback;
1590         struct sccb *pSaveSCCB;
1591         struct sccb_mgr_tar_info *currTar_Info;
1592
1593         ioport = ((struct sccb_card *)pCurrCard)->ioPort;
1594
1595         thisCard = ((struct sccb_card *)pCurrCard)->cardIndex;
1596
1597         if (!(RD_HARPOON(ioport + hp_page_ctrl) & G_INT_DISABLE)) {
1598
1599                 if (FPT_queueFindSccb(p_Sccb, thisCard)) {
1600
1601                         ((struct sccb_card *)pCurrCard)->cmdCounter--;
1602
1603                         if (!((struct sccb_card *)pCurrCard)->cmdCounter)
1604                                 WR_HARPOON(ioport + hp_semaphore,
1605                                            (RD_HARPOON(ioport + hp_semaphore)
1606                                             & (unsigned
1607                                                char)(~(SCCB_MGR_ACTIVE |
1608                                                        TICKLE_ME))));
1609
1610                         p_Sccb->SccbStatus = SCCB_ABORT;
1611                         callback = p_Sccb->SccbCallback;
1612                         callback(p_Sccb);
1613
1614                         return 0;
1615                 }
1616
1617                 else {
1618                         if (((struct sccb_card *)pCurrCard)->currentSCCB ==
1619                             p_Sccb) {
1620                                 p_Sccb->SccbStatus = SCCB_ABORT;
1621                                 return 0;
1622
1623                         }
1624
1625                         else {
1626                                 if (p_Sccb->Sccb_tag) {
1627                                         MDISABLE_INT(ioport);
1628                                         if (((struct sccb_card *)pCurrCard)->
1629                                             discQ_Tbl[p_Sccb->Sccb_tag] ==
1630                                             p_Sccb) {
1631                                                 p_Sccb->SccbStatus = SCCB_ABORT;
1632                                                 p_Sccb->Sccb_scsistat =
1633                                                     ABORT_ST;
1634                                                 p_Sccb->Sccb_scsimsg =
1635                                                     ABORT_TASK;
1636
1637                                                 if (((struct sccb_card *)
1638                                                      pCurrCard)->currentSCCB ==
1639                                                     NULL) {
1640                                                         ((struct sccb_card *)
1641                                                          pCurrCard)->
1642                                         currentSCCB = p_Sccb;
1643                                                         FPT_ssel(ioport,
1644                                                                  thisCard);
1645                                                 } else {
1646                                                         pSaveSCCB =
1647                                                             ((struct sccb_card
1648                                                               *)pCurrCard)->
1649                                                             currentSCCB;
1650                                                         ((struct sccb_card *)
1651                                                          pCurrCard)->
1652                                         currentSCCB = p_Sccb;
1653                                                         FPT_queueSelectFail((struct sccb_card *)pCurrCard, thisCard);
1654                                                         ((struct sccb_card *)
1655                                                          pCurrCard)->
1656                                         currentSCCB = pSaveSCCB;
1657                                                 }
1658                                         }
1659                                         MENABLE_INT(ioport);
1660                                         return 0;
1661                                 } else {
1662                                         currTar_Info =
1663                                             &FPT_sccbMgrTbl[thisCard][p_Sccb->
1664                                                                       TargID];
1665
1666                                         if (FPT_BL_Card[thisCard].
1667                                             discQ_Tbl[currTar_Info->
1668                                                       LunDiscQ_Idx[p_Sccb->Lun]]
1669                                             == p_Sccb) {
1670                                                 p_Sccb->SccbStatus = SCCB_ABORT;
1671                                                 return 0;
1672                                         }
1673                                 }
1674                         }
1675                 }
1676         }
1677         return -1;
1678 }
1679
1680 /*---------------------------------------------------------------------
1681  *
1682  * Function: FlashPoint_InterruptPending
1683  *
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.
1686  *
1687  *---------------------------------------------------------------------*/
1688 static unsigned char FlashPoint_InterruptPending(void *pCurrCard)
1689 {
1690         u32 ioport;
1691
1692         ioport = ((struct sccb_card *)pCurrCard)->ioPort;
1693
1694         if (RD_HARPOON(ioport + hp_int_status) & INT_ASSERTED) {
1695                 return 1;
1696         }
1697
1698         else
1699
1700                 return 0;
1701 }
1702
1703 /*---------------------------------------------------------------------
1704  *
1705  * Function: FlashPoint_HandleInterrupt
1706  *
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
1709  *              us.
1710  *
1711  *---------------------------------------------------------------------*/
1712 static int FlashPoint_HandleInterrupt(void *pcard)
1713 {
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;
1719         u32 ioport;
1720
1721         thisCard = pCurrCard->cardIndex;
1722         ioport = pCurrCard->ioPort;
1723
1724         MDISABLE_INT(ioport);
1725
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;
1729         else
1730                 bm_status = 0;
1731
1732         WR_HARPOON(ioport + hp_int_mask, (INT_CMD_COMPL | SCSI_INTERRUPT));
1733
1734         while ((hp_int = RDW_HARPOON((ioport + hp_intstat)) &
1735                                 FPT_default_intena) | bm_status) {
1736
1737                 currSCCB = pCurrCard->currentSCCB;
1738
1739                 if (hp_int & (FIFO | TIMEOUT | RESET | SCAM_SEL) || bm_status) {
1740                         result =
1741                             FPT_SccbMgr_bad_isr(ioport, thisCard, pCurrCard,
1742                                                 hp_int);
1743                         WRW_HARPOON((ioport + hp_intstat),
1744                                     (FIFO | TIMEOUT | RESET | SCAM_SEL));
1745                         bm_status = 0;
1746
1747                         if (result) {
1748
1749                                 MENABLE_INT(ioport);
1750                                 return result;
1751                         }
1752                 }
1753
1754                 else if (hp_int & ICMD_COMP) {
1755
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.
1761                                  */
1762                                 while (!
1763                                        (RDW_HARPOON((ioport + hp_intstat)) &
1764                                         (BUS_FREE | RSEL))) ;
1765                         }
1766
1767                         if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1768
1769                                 FPT_phaseChkFifo(ioport, thisCard);
1770
1771 /*         WRW_HARPOON((ioport+hp_intstat),
1772             (BUS_FREE | ICMD_COMP | ITAR_DISC | XFER_CNT_0));
1773          */
1774
1775                         WRW_HARPOON((ioport + hp_intstat), CLR_ALL_INT_1);
1776
1777                         FPT_autoCmdCmplt(ioport, thisCard);
1778
1779                 }
1780
1781                 else if (hp_int & ITAR_DISC) {
1782
1783                         if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1784                                 FPT_phaseChkFifo(ioport, thisCard);
1785
1786                         if (RD_HARPOON(ioport + hp_gp_reg_1) ==
1787                                         SAVE_POINTERS) {
1788
1789                                 WR_HARPOON(ioport + hp_gp_reg_1, 0x00);
1790                                 currSCCB->Sccb_XferState |= F_NO_DATA_YET;
1791
1792                                 currSCCB->Sccb_savedATC = currSCCB->Sccb_ATC;
1793                         }
1794
1795                         currSCCB->Sccb_scsistat = DISCONNECT_ST;
1796                         FPT_queueDisconnect(currSCCB, thisCard);
1797
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.
1802                          */
1803                         while (!
1804                                (RDW_HARPOON((ioport + hp_intstat)) &
1805                                 (BUS_FREE | RSEL))
1806                                && !((RDW_HARPOON((ioport + hp_intstat)) & PHASE)
1807                                     && RD_HARPOON((ioport + hp_scsisig)) ==
1808                                     (SCSI_BSY | SCSI_REQ | SCSI_CD | SCSI_MSG |
1809                                      SCSI_IOBIT))) ;
1810
1811                         /*
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.
1815                          */
1816                         if (!
1817                             (RDW_HARPOON((ioport + hp_intstat)) &
1818                              (BUS_FREE | RSEL))) {
1819                                 MENABLE_INT(ioport);
1820                                 return 0xFE;
1821                         }
1822
1823                         WRW_HARPOON((ioport + hp_intstat),
1824                                     (BUS_FREE | ITAR_DISC));
1825
1826                         pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
1827
1828                 }
1829
1830                 else if (hp_int & RSEL) {
1831
1832                         WRW_HARPOON((ioport + hp_intstat),
1833                                     (PROG_HLT | RSEL | PHASE | BUS_FREE));
1834
1835                         if (RDW_HARPOON((ioport + hp_intstat)) & ITAR_DISC) {
1836                                 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
1837                                         FPT_phaseChkFifo(ioport, thisCard);
1838
1839                                 if (RD_HARPOON(ioport + hp_gp_reg_1) ==
1840                                     SAVE_POINTERS) {
1841                                         WR_HARPOON(ioport + hp_gp_reg_1, 0x00);
1842                                         currSCCB->Sccb_XferState |=
1843                                             F_NO_DATA_YET;
1844                                         currSCCB->Sccb_savedATC =
1845                                             currSCCB->Sccb_ATC;
1846                                 }
1847
1848                                 WRW_HARPOON((ioport + hp_intstat),
1849                                             (BUS_FREE | ITAR_DISC));
1850                                 currSCCB->Sccb_scsistat = DISCONNECT_ST;
1851                                 FPT_queueDisconnect(currSCCB, thisCard);
1852                         }
1853
1854                         FPT_sres(ioport, thisCard, pCurrCard);
1855                         FPT_phaseDecode(ioport, thisCard);
1856
1857                 }
1858
1859                 else if ((hp_int & IDO_STRT) && (!(hp_int & BUS_FREE))) {
1860
1861                         WRW_HARPOON((ioport + hp_intstat),
1862                                     (IDO_STRT | XFER_CNT_0));
1863                         FPT_phaseDecode(ioport, thisCard);
1864
1865                 }
1866
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);
1873                         } else {
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 */
1880
1881                                 i = (unsigned
1882                                      char)(RD_HARPOON(ioport + hp_fifowrite));
1883                                 target =
1884                                     (unsigned
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 <<
1890                                                            4));
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));
1896                         }
1897                 }
1898
1899                 else if (hp_int & XFER_CNT_0) {
1900
1901                         WRW_HARPOON((ioport + hp_intstat), XFER_CNT_0);
1902
1903                         FPT_schkdd(ioport, thisCard);
1904
1905                 }
1906
1907                 else if (hp_int & BUS_FREE) {
1908
1909                         WRW_HARPOON((ioport + hp_intstat), BUS_FREE);
1910
1911                         if (pCurrCard->globalFlags & F_HOST_XFER_ACT) {
1912
1913                                 FPT_hostDataXferAbort(ioport, thisCard,
1914                                                       currSCCB);
1915                         }
1916
1917                         FPT_phaseBusFree(ioport, thisCard);
1918                 }
1919
1920                 else if (hp_int & ITICKLE) {
1921
1922                         WRW_HARPOON((ioport + hp_intstat), ITICKLE);
1923                         pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
1924                 }
1925
1926                 if (((struct sccb_card *)pCurrCard)->
1927                     globalFlags & F_NEW_SCCB_CMD) {
1928
1929                         pCurrCard->globalFlags &= ~F_NEW_SCCB_CMD;
1930
1931                         if (pCurrCard->currentSCCB == NULL)
1932                                 FPT_queueSearchSelect(pCurrCard, thisCard);
1933
1934                         if (pCurrCard->currentSCCB != NULL) {
1935                                 pCurrCard->globalFlags &= ~F_NEW_SCCB_CMD;
1936                                 FPT_ssel(ioport, thisCard);
1937                         }
1938
1939                         break;
1940
1941                 }
1942
1943         }                       /*end while */
1944
1945         MENABLE_INT(ioport);
1946
1947         return 0;
1948 }
1949
1950 /*---------------------------------------------------------------------
1951  *
1952  * Function: Sccb_bad_isr
1953  *
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
1957  *              processing time.
1958  *
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)
1963 {
1964         unsigned char temp, ScamFlg;
1965         struct sccb_mgr_tar_info *currTar_Info;
1966         struct nvram_info *pCurrNvRam;
1967
1968         if (RD_HARPOON(p_port + hp_ext_status) &
1969             (BM_FORCE_OFF | PCI_DEV_TMOUT | BM_PARITY_ERR | PIO_OVERRUN)) {
1970
1971                 if (pCurrCard->globalFlags & F_HOST_XFER_ACT) {
1972
1973                         FPT_hostDataXferAbort(p_port, p_card,
1974                                               pCurrCard->currentSCCB);
1975                 }
1976
1977                 if (RD_HARPOON(p_port + hp_pci_stat_cfg) & REC_MASTER_ABORT)
1978                 {
1979                         WR_HARPOON(p_port + hp_pci_stat_cfg,
1980                                    (RD_HARPOON(p_port + hp_pci_stat_cfg) &
1981                                     ~REC_MASTER_ABORT));
1982
1983                         WR_HARPOON(p_port + hp_host_blk_cnt, 0x00);
1984
1985                 }
1986
1987                 if (pCurrCard->currentSCCB != NULL) {
1988
1989                         if (!pCurrCard->currentSCCB->HostStatus)
1990                                 pCurrCard->currentSCCB->HostStatus =
1991                                     SCCB_BM_ERR;
1992
1993                         FPT_sxfrp(p_port, p_card);
1994
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);
2000
2001                         if (!
2002                             (RDW_HARPOON((p_port + hp_intstat)) &
2003                              (BUS_FREE | RESET))) {
2004                                 FPT_phaseDecode(p_port, p_card);
2005                         }
2006                 }
2007         }
2008
2009         else if (p_int & RESET) {
2010
2011                 WR_HARPOON(p_port + hp_clkctrl_0, CLKCTRL_DEFAULT);
2012                 WR_HARPOON(p_port + hp_sys_ctrl, 0x00);
2013                 if (pCurrCard->currentSCCB != NULL) {
2014
2015                         if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
2016
2017                                 FPT_hostDataXferAbort(p_port, p_card,
2018                                                       pCurrCard->currentSCCB);
2019                 }
2020
2021                 DISABLE_AUTO(p_port);
2022
2023                 FPT_sresb(p_port, p_card);
2024
2025                 while (RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST) {
2026                 }
2027
2028                 pCurrNvRam = pCurrCard->pNvRamInfo;
2029                 if (pCurrNvRam) {
2030                         ScamFlg = pCurrNvRam->niScamConf;
2031                 } else {
2032                         ScamFlg =
2033                             (unsigned char)FPT_utilEERead(p_port,
2034                                                           SCAM_CONFIG / 2);
2035                 }
2036
2037                 FPT_XbowInit(p_port, ScamFlg);
2038
2039                 FPT_scini(p_card, pCurrCard->ourId, 0);
2040
2041                 return 0xFF;
2042         }
2043
2044         else if (p_int & FIFO) {
2045
2046                 WRW_HARPOON((p_port + hp_intstat), FIFO);
2047
2048                 if (pCurrCard->currentSCCB != NULL)
2049                         FPT_sxfrp(p_port, p_card);
2050         }
2051
2052         else if (p_int & TIMEOUT) {
2053
2054                 DISABLE_AUTO(p_port);
2055
2056                 WRW_HARPOON((p_port + hp_intstat),
2057                             (PROG_HLT | TIMEOUT | SEL | BUS_FREE | PHASE |
2058                              IUNKWN));
2059
2060                 pCurrCard->currentSCCB->HostStatus = SCCB_SELECTION_TIMEOUT;
2061
2062                 currTar_Info =
2063                     &FPT_sccbMgrTbl[p_card][pCurrCard->currentSCCB->TargID];
2064                 if ((pCurrCard->globalFlags & F_CONLUN_IO)
2065                     && ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
2066                         TAG_Q_TRYING))
2067                         currTar_Info->TarLUNBusy[pCurrCard->currentSCCB->Lun] =
2068                             0;
2069                 else
2070                         currTar_Info->TarLUNBusy[0] = 0;
2071
2072                 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
2073                         currTar_Info->TarSyncCtrl = 0;
2074                         currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2075                 }
2076
2077                 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
2078                         currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
2079                 }
2080
2081                 FPT_sssyncv(p_port, pCurrCard->currentSCCB->TargID, NARROW_SCSI,
2082                             currTar_Info);
2083
2084                 FPT_queueCmdComplete(pCurrCard, pCurrCard->currentSCCB, p_card);
2085
2086         }
2087
2088         else if (p_int & SCAM_SEL) {
2089
2090                 FPT_scarb(p_port, LEVEL2_TAR);
2091                 FPT_scsel(p_port);
2092                 FPT_scasid(p_card, p_port);
2093
2094                 FPT_scbusf(p_port);
2095
2096                 WRW_HARPOON((p_port + hp_intstat), SCAM_SEL);
2097         }
2098
2099         return 0x00;
2100 }
2101
2102 /*---------------------------------------------------------------------
2103  *
2104  * Function: SccbMgrTableInit
2105  *
2106  * Description: Initialize all Sccb manager data structures.
2107  *
2108  *---------------------------------------------------------------------*/
2109
2110 static void FPT_SccbMgrTableInitAll(void)
2111 {
2112         unsigned char thisCard;
2113
2114         for (thisCard = 0; thisCard < MAX_CARDS; thisCard++) {
2115                 FPT_SccbMgrTableInitCard(&FPT_BL_Card[thisCard], thisCard);
2116
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;
2122         }
2123 }
2124
2125 /*---------------------------------------------------------------------
2126  *
2127  * Function: SccbMgrTableInit
2128  *
2129  * Description: Initialize all Sccb manager data structures.
2130  *
2131  *---------------------------------------------------------------------*/
2132
2133 static void FPT_SccbMgrTableInitCard(struct sccb_card *pCurrCard,
2134                                      unsigned char p_card)
2135 {
2136         unsigned char scsiID, qtag;
2137
2138         for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
2139                 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
2140         }
2141
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);
2146         }
2147
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;
2154
2155 }
2156
2157 /*---------------------------------------------------------------------
2158  *
2159  * Function: SccbMgrTableInit
2160  *
2161  * Description: Initialize all Sccb manager data structures.
2162  *
2163  *---------------------------------------------------------------------*/
2164
2165 static void FPT_SccbMgrTableInitTarget(unsigned char p_card,
2166                                        unsigned char target)
2167 {
2168
2169         unsigned char lun, qtag;
2170         struct sccb_mgr_tar_info *currTar_Info;
2171
2172         currTar_Info = &FPT_sccbMgrTbl[p_card][target];
2173
2174         currTar_Info->TarSelQ_Cnt = 0;
2175         currTar_Info->TarSyncCtrl = 0;
2176
2177         currTar_Info->TarSelQ_Head = NULL;
2178         currTar_Info->TarSelQ_Tail = NULL;
2179         currTar_Info->TarTagQ_Cnt = 0;
2180         currTar_Info->TarLUN_CA = 0;
2181
2182         for (lun = 0; lun < MAX_LUN; lun++) {
2183                 currTar_Info->TarLUNBusy[lun] = 0;
2184                 currTar_Info->LunDiscQ_Idx[lun] = 0;
2185         }
2186
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 ==
2190                             target) {
2191                                 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
2192                                 FPT_BL_Card[p_card].discQCount--;
2193                         }
2194                 }
2195         }
2196 }
2197
2198 /*---------------------------------------------------------------------
2199  *
2200  * Function: sfetm
2201  *
2202  * Description: Read in a message byte from the SCSI bus, and check
2203  *              for a parity error.
2204  *
2205  *---------------------------------------------------------------------*/
2206
2207 static unsigned char FPT_sfm(u32 port, struct sccb *pCurrSCCB)
2208 {
2209         unsigned char message;
2210         unsigned short TimeOutLoop;
2211
2212         TimeOutLoop = 0;
2213         while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2214                (TimeOutLoop++ < 20000)) {
2215         }
2216
2217         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
2218
2219         message = RD_HARPOON(port + hp_scsidata_0);
2220
2221         WR_HARPOON(port + hp_scsisig, SCSI_ACK + S_MSGI_PH);
2222
2223         if (TimeOutLoop > 20000)
2224                 message = 0x00; /* force message byte = 0 if Time Out on Req */
2225
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;
2234                 }
2235                 message = 0x00;
2236                 do {
2237                         ACCEPT_MSG_ATN(port);
2238                         TimeOutLoop = 0;
2239                         while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2240                                (TimeOutLoop++ < 20000)) {
2241                         }
2242                         if (TimeOutLoop > 20000) {
2243                                 WRW_HARPOON((port + hp_intstat), PARITY);
2244                                 return message;
2245                         }
2246                         if ((RD_HARPOON(port + hp_scsisig) & S_SCSI_PHZ) !=
2247                             S_MSGI_PH) {
2248                                 WRW_HARPOON((port + hp_intstat), PARITY);
2249                                 return message;
2250                         }
2251                         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
2252
2253                         RD_HARPOON(port + hp_scsidata_0);
2254
2255                         WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2256
2257                 } while (1);
2258
2259         }
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);
2264         return message;
2265 }
2266
2267 /*---------------------------------------------------------------------
2268  *
2269  * Function: FPT_ssel
2270  *
2271  * Description: Load up automation and select target device.
2272  *
2273  *---------------------------------------------------------------------*/
2274
2275 static void FPT_ssel(u32 port, unsigned char p_card)
2276 {
2277
2278         unsigned char auto_loaded, i, target, *theCCB;
2279
2280         u32 cdb_reg;
2281         struct sccb_card *CurrCard;
2282         struct sccb *currSCCB;
2283         struct sccb_mgr_tar_info *currTar_Info;
2284         unsigned char lastTag, lun;
2285
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;
2291
2292         ARAM_ACCESS(port);
2293
2294         if ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) == TAG_Q_REJECT)
2295                 currSCCB->ControlByte &= ~F_USE_CMD_Q;
2296
2297         if (((CurrCard->globalFlags & F_CONLUN_IO) &&
2298              ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) != TAG_Q_TRYING)))
2299
2300                 lun = currSCCB->Lun;
2301         else
2302                 lun = 0;
2303
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)
2308                                 == TAG_Q_TRYING)) {
2309
2310                                 if (currTar_Info->TarTagQ_Cnt != 0) {
2311                                         currTar_Info->TarLUNBusy[lun] = 1;
2312                                         FPT_queueSelectFail(CurrCard, p_card);
2313                                         SGRAM_ACCESS(port);
2314                                         return;
2315                                 }
2316
2317                                 else {
2318                                         currTar_Info->TarLUNBusy[lun] = 1;
2319                                 }
2320
2321                         }
2322                         /*End non-tagged */
2323                         else {
2324                                 currTar_Info->TarLUNBusy[lun] = 1;
2325                         }
2326
2327                 }
2328                 /*!Use cmd Q Tagged */
2329                 else {
2330                         if (currTar_Info->TarLUN_CA == 1) {
2331                                 FPT_queueSelectFail(CurrCard, p_card);
2332                                 SGRAM_ACCESS(port);
2333                                 return;
2334                         }
2335
2336                         currTar_Info->TarLUNBusy[lun] = 1;
2337
2338                 }               /*else use cmd Q tagged */
2339
2340         }
2341         /*if glob tagged started */
2342         else {
2343                 currTar_Info->TarLUNBusy[lun] = 1;
2344         }
2345
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);
2352                         SGRAM_ACCESS(port);
2353                         return;
2354                 }
2355                 for (i = 1; i < QUEUE_DEPTH; i++) {
2356                         if (++lastTag >= QUEUE_DEPTH)
2357                                 lastTag = 1;
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++;
2363                                 break;
2364                         }
2365                 }
2366                 if (i == QUEUE_DEPTH) {
2367                         currTar_Info->TarLUNBusy[lun] = 1;
2368                         FPT_queueSelectFail(CurrCard, p_card);
2369                         SGRAM_ACCESS(port);
2370                         return;
2371                 }
2372         }
2373
2374         auto_loaded = 0;
2375
2376         WR_HARPOON(port + hp_select_id, target);
2377         WR_HARPOON(port + hp_gp_reg_3, target); /* Use by new automation logic */
2378
2379         if (currSCCB->OperationCode == RESET_COMMAND) {
2380                 WRW_HARPOON((port + ID_MSG_STRT), (MPM_OP + AMSG_OUT +
2381                                                    (currSCCB->
2382                                                     Sccb_idmsg & ~DISC_PRIV)));
2383
2384                 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + NP);
2385
2386                 currSCCB->Sccb_scsimsg = TARGET_RESET;
2387
2388                 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
2389                 auto_loaded = 1;
2390                 currSCCB->Sccb_scsistat = SELECT_BDR_ST;
2391
2392                 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
2393                         currTar_Info->TarSyncCtrl = 0;
2394                         currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2395                 }
2396
2397                 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
2398                         currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
2399                 }
2400
2401                 FPT_sssyncv(port, target, NARROW_SCSI, currTar_Info);
2402                 FPT_SccbMgrTableInitTarget(p_card, target);
2403
2404         }
2405
2406         else if (currSCCB->Sccb_scsistat == ABORT_ST) {
2407                 WRW_HARPOON((port + ID_MSG_STRT), (MPM_OP + AMSG_OUT +
2408                                                    (currSCCB->
2409                                                     Sccb_idmsg & ~DISC_PRIV)));
2410
2411                 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
2412
2413                 WRW_HARPOON((port + SYNC_MSGS + 0), (MPM_OP + AMSG_OUT +
2414                                                      (((unsigned
2415                                                         char)(currSCCB->
2416                                                               ControlByte &
2417                                                               TAG_TYPE_MASK)
2418                                                        >> 6) | (unsigned char)
2419                                                       0x20)));
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));
2423
2424                 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
2425                 auto_loaded = 1;
2426
2427         }
2428
2429         else if (!(currTar_Info->TarStatus & WIDE_NEGOCIATED)) {
2430                 auto_loaded = FPT_siwidn(port, p_card);
2431                 currSCCB->Sccb_scsistat = SELECT_WN_ST;
2432         }
2433
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;
2438         }
2439
2440         if (!auto_loaded) {
2441
2442                 if (currSCCB->ControlByte & F_USE_CMD_Q) {
2443
2444                         CurrCard->globalFlags |= F_TAG_STARTED;
2445
2446                         if ((currTar_Info->TarStatus & TAR_TAG_Q_MASK)
2447                             == TAG_Q_REJECT) {
2448                                 currSCCB->ControlByte &= ~F_USE_CMD_Q;
2449
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);
2454
2455                                 WRW_HARPOON((port + NON_TAG_ID_MSG),
2456                                             (MPM_OP + AMSG_OUT +
2457                                              currSCCB->Sccb_idmsg));
2458
2459                                 WR_HARPOON(port + hp_autostart_3,
2460                                            (SELECT + SELCHK_STRT));
2461
2462                                 /* Setup our STATE so we know what happened when
2463                                    the wheels fall off. */
2464                                 currSCCB->Sccb_scsistat = SELECT_ST;
2465
2466                                 currTar_Info->TarLUNBusy[lun] = 1;
2467                         }
2468
2469                         else {
2470                                 WRW_HARPOON((port + ID_MSG_STRT),
2471                                             (MPM_OP + AMSG_OUT +
2472                                              currSCCB->Sccb_idmsg));
2473
2474                                 WRW_HARPOON((port + ID_MSG_STRT + 2),
2475                                             (MPM_OP + AMSG_OUT +
2476                                              (((unsigned char)(currSCCB->
2477                                                                ControlByte &
2478                                                                TAG_TYPE_MASK)
2479                                                >> 6) | (unsigned char)0x20)));
2480
2481                                 for (i = 1; i < QUEUE_DEPTH; i++) {
2482                                         if (++lastTag >= QUEUE_DEPTH)
2483                                                 lastTag = 1;
2484                                         if (CurrCard->discQ_Tbl[lastTag] ==
2485                                             NULL) {
2486                                                 WRW_HARPOON((port +
2487                                                              ID_MSG_STRT + 6),
2488                                                             (MPM_OP + AMSG_OUT +
2489                                                              lastTag));
2490                                                 CurrCard->tagQ_Lst = lastTag;
2491                                                 currSCCB->Sccb_tag = lastTag;
2492                                                 CurrCard->discQ_Tbl[lastTag] =
2493                                                     currSCCB;
2494                                                 CurrCard->discQCount++;
2495                                                 break;
2496                                         }
2497                                 }
2498
2499                                 if (i == QUEUE_DEPTH) {
2500                                         currTar_Info->TarLUNBusy[lun] = 1;
2501                                         FPT_queueSelectFail(CurrCard, p_card);
2502                                         SGRAM_ACCESS(port);
2503                                         return;
2504                                 }
2505
2506                                 currSCCB->Sccb_scsistat = SELECT_Q_ST;
2507
2508                                 WR_HARPOON(port + hp_autostart_3,
2509                                            (SELECT + SELCHK_STRT));
2510                         }
2511                 }
2512
2513                 else {
2514
2515                         WRW_HARPOON((port + ID_MSG_STRT),
2516                                     BRH_OP + ALWAYS + NTCMD);
2517
2518                         WRW_HARPOON((port + NON_TAG_ID_MSG),
2519                                     (MPM_OP + AMSG_OUT + currSCCB->Sccb_idmsg));
2520
2521                         currSCCB->Sccb_scsistat = SELECT_ST;
2522
2523                         WR_HARPOON(port + hp_autostart_3,
2524                                    (SELECT + SELCHK_STRT));
2525                 }
2526
2527                 theCCB = (unsigned char *)&currSCCB->Cdb[0];
2528
2529                 cdb_reg = port + CMD_STRT;
2530
2531                 for (i = 0; i < currSCCB->CdbLength; i++) {
2532                         WRW_HARPOON(cdb_reg, (MPM_OP + ACOMMAND + *theCCB));
2533                         cdb_reg += 2;
2534                         theCCB++;
2535                 }
2536
2537                 if (currSCCB->CdbLength != TWELVE_BYTE_CMD)
2538                         WRW_HARPOON(cdb_reg, (BRH_OP + ALWAYS + NP));
2539
2540         }
2541         /* auto_loaded */
2542         WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
2543         WR_HARPOON(port + hp_xferstat, 0x00);
2544
2545         WRW_HARPOON((port + hp_intstat), (PROG_HLT | TIMEOUT | SEL | BUS_FREE));
2546
2547         WR_HARPOON(port + hp_portctrl_0, (SCSI_PORT));
2548
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));
2552         } else {
2553
2554 /*      auto_loaded =  (RD_HARPOON(port+hp_autostart_3) & (unsigned char)0x1F);
2555       auto_loaded |= AUTO_IMMED; */
2556                 auto_loaded = AUTO_IMMED;
2557
2558                 DISABLE_AUTO(port);
2559
2560                 WR_HARPOON(port + hp_autostart_3, auto_loaded);
2561         }
2562
2563         SGRAM_ACCESS(port);
2564 }
2565
2566 /*---------------------------------------------------------------------
2567  *
2568  * Function: FPT_sres
2569  *
2570  * Description: Hookup the correct CCB and handle the incoming messages.
2571  *
2572  *---------------------------------------------------------------------*/
2573
2574 static void FPT_sres(u32 port, unsigned char p_card,
2575                      struct sccb_card *pCurrCard)
2576 {
2577
2578         unsigned char our_target, message, lun = 0, tag, msgRetryCount;
2579
2580         struct sccb_mgr_tar_info *currTar_Info;
2581         struct sccb *currSCCB;
2582
2583         if (pCurrCard->currentSCCB != NULL) {
2584                 currTar_Info =
2585                     &FPT_sccbMgrTbl[p_card][pCurrCard->currentSCCB->TargID];
2586                 DISABLE_AUTO(port);
2587
2588                 WR_HARPOON((port + hp_scsictrl_0), (ENA_RESEL | ENA_SCAM_SEL));
2589
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;
2594                 }
2595                 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
2596                         currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
2597                         currSCCB->Sccb_scsistat = BUS_FREE_ST;
2598                 }
2599                 if (((pCurrCard->globalFlags & F_CONLUN_IO) &&
2600                      ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
2601                       TAG_Q_TRYING))) {
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->
2607                                                                   Lun]]
2608                                     = NULL;
2609                         }
2610                 } else {
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->
2616                                                              Sccb_tag] = NULL;
2617                                 }
2618                         } else {
2619                                 if (currSCCB->Sccb_scsistat != ABORT_ST) {
2620                                         pCurrCard->discQCount--;
2621                                         pCurrCard->discQ_Tbl[currTar_Info->
2622                                                              LunDiscQ_Idx[0]] =
2623                                             NULL;
2624                                 }
2625                         }
2626                 }
2627
2628                 FPT_queueSelectFail(&FPT_BL_Card[p_card], p_card);
2629         }
2630
2631         WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
2632
2633         our_target = (unsigned char)(RD_HARPOON(port + hp_select_id) >> 4);
2634         currTar_Info = &FPT_sccbMgrTbl[p_card][our_target];
2635
2636         msgRetryCount = 0;
2637         do {
2638
2639                 currTar_Info = &FPT_sccbMgrTbl[p_card][our_target];
2640                 tag = 0;
2641
2642                 while (!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) {
2643                         if (!(RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) {
2644
2645                                 WRW_HARPOON((port + hp_intstat), PHASE);
2646                                 return;
2647                         }
2648                 }
2649
2650                 WRW_HARPOON((port + hp_intstat), PHASE);
2651                 if ((RD_HARPOON(port + hp_scsisig) & S_SCSI_PHZ) == S_MSGI_PH) {
2652
2653                         message = FPT_sfm(port, pCurrCard->currentSCCB);
2654                         if (message) {
2655
2656                                 if (message <= (0x80 | LUN_MASK)) {
2657                                         lun = message & (unsigned char)LUN_MASK;
2658
2659                                         if ((currTar_Info->
2660                                              TarStatus & TAR_TAG_Q_MASK) ==
2661                                             TAG_Q_TRYING) {
2662                                                 if (currTar_Info->TarTagQ_Cnt !=
2663                                                     0) {
2664
2665                                                         if (!
2666                                                             (currTar_Info->
2667                                                              TarLUN_CA)) {
2668                                                                 ACCEPT_MSG(port);       /*Release the ACK for ID msg. */
2669
2670                                                                 message =
2671                                                                     FPT_sfm
2672                                                                     (port,
2673                                                                      pCurrCard->
2674                                                                      currentSCCB);
2675                                                                 if (message) {
2676                                                                         ACCEPT_MSG
2677                                                                             (port);
2678                                                                 }
2679
2680                                                                 else
2681                                                                         message
2682                                                                             = 0;
2683
2684                                                                 if (message !=
2685                                                                     0) {
2686                                                                         tag =
2687                                                                             FPT_sfm
2688                                                                             (port,
2689                                                                              pCurrCard->
2690                                                                              currentSCCB);
2691
2692                                                                         if (!
2693                                                                             (tag))
2694                                                                                 message
2695                                                                                     =
2696                                                                                     0;
2697                                                                 }
2698
2699                                                         }
2700                                                         /*C.A. exists! */
2701                                                 }
2702                                                 /*End Q cnt != 0 */
2703                                         }
2704                                         /*End Tag cmds supported! */
2705                                 }
2706                                 /*End valid ID message.  */
2707                                 else {
2708
2709                                         ACCEPT_MSG_ATN(port);
2710                                 }
2711
2712                         }
2713                         /* End good id message. */
2714                         else {
2715
2716                                 message = 0;
2717                         }
2718                 } else {
2719                         ACCEPT_MSG_ATN(port);
2720
2721                         while (!
2722                                (RDW_HARPOON((port + hp_intstat)) &
2723                                 (PHASE | RESET))
2724                                && !(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)
2725                                && (RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) ;
2726
2727                         return;
2728                 }
2729
2730                 if (message == 0) {
2731                         msgRetryCount++;
2732                         if (msgRetryCount == 1) {
2733                                 FPT_SendMsg(port, MSG_PARITY_ERROR);
2734                         } else {
2735                                 FPT_SendMsg(port, TARGET_RESET);
2736
2737                                 FPT_sssyncv(port, our_target, NARROW_SCSI,
2738                                             currTar_Info);
2739
2740                                 if (FPT_sccbMgrTbl[p_card][our_target].
2741                                     TarEEValue & EE_SYNC_MASK) {
2742
2743                                         FPT_sccbMgrTbl[p_card][our_target].
2744                                             TarStatus &= ~TAR_SYNC_MASK;
2745
2746                                 }
2747
2748                                 if (FPT_sccbMgrTbl[p_card][our_target].
2749                                     TarEEValue & EE_WIDE_SCSI) {
2750
2751                                         FPT_sccbMgrTbl[p_card][our_target].
2752                                             TarStatus &= ~TAR_WIDE_MASK;
2753                                 }
2754
2755                                 FPT_queueFlushTargSccb(p_card, our_target,
2756                                                        SCCB_COMPLETE);
2757                                 FPT_SccbMgrTableInitTarget(p_card, our_target);
2758                                 return;
2759                         }
2760                 }
2761         } while (message == 0);
2762
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) {
2769                         ACCEPT_MSG(port);
2770                 } else {
2771                         ACCEPT_MSG_ATN(port);
2772                 }
2773         } else {
2774                 currTar_Info->TarLUNBusy[0] = 1;
2775
2776                 if (tag) {
2777                         if (pCurrCard->discQ_Tbl[tag] != NULL) {
2778                                 pCurrCard->currentSCCB =
2779                                     pCurrCard->discQ_Tbl[tag];
2780                                 currTar_Info->TarTagQ_Cnt--;
2781                                 ACCEPT_MSG(port);
2782                         } else {
2783                                 ACCEPT_MSG_ATN(port);
2784                         }
2785                 } else {
2786                         pCurrCard->currentSCCB =
2787                             pCurrCard->discQ_Tbl[currTar_Info->LunDiscQ_Idx[0]];
2788                         if (pCurrCard->currentSCCB != NULL) {
2789                                 ACCEPT_MSG(port);
2790                         } else {
2791                                 ACCEPT_MSG_ATN(port);
2792                         }
2793                 }
2794         }
2795
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);
2802                 }
2803         }
2804
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)) ;
2808 }
2809
2810 static void FPT_SendMsg(u32 port, unsigned char message)
2811 {
2812         while (!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) {
2813                 if (!(RD_HARPOON(port + hp_scsisig) & SCSI_BSY)) {
2814
2815                         WRW_HARPOON((port + hp_intstat), PHASE);
2816                         return;
2817                 }
2818         }
2819
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));
2824
2825                 WR_HARPOON(port + hp_portctrl_0, SCSI_BUS_EN);
2826
2827                 WR_HARPOON(port + hp_scsidata_0, message);
2828
2829                 WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
2830
2831                 ACCEPT_MSG(port);
2832
2833                 WR_HARPOON(port + hp_portctrl_0, 0x00);
2834
2835                 if ((message == ABORT_TASK_SET) || (message == TARGET_RESET) ||
2836                     (message == ABORT_TASK)) {
2837                         while (!
2838                                (RDW_HARPOON((port + hp_intstat)) &
2839                                 (BUS_FREE | PHASE))) {
2840                         }
2841
2842                         if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
2843                                 WRW_HARPOON((port + hp_intstat), BUS_FREE);
2844                         }
2845                 }
2846         }
2847 }
2848
2849 /*---------------------------------------------------------------------
2850  *
2851  * Function: FPT_sdecm
2852  *
2853  * Description: Determine the proper response to the message from the
2854  *              target device.
2855  *
2856  *---------------------------------------------------------------------*/
2857 static void FPT_sdecm(unsigned char message, u32 port, unsigned char p_card)
2858 {
2859         struct sccb *currSCCB;
2860         struct sccb_card *CurrCard;
2861         struct sccb_mgr_tar_info *currTar_Info;
2862
2863         CurrCard = &FPT_BL_Card[p_card];
2864         currSCCB = CurrCard->currentSCCB;
2865
2866         currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
2867
2868         if (message == RESTORE_POINTERS) {
2869                 if (!(currSCCB->Sccb_XferState & F_NO_DATA_YET)) {
2870                         currSCCB->Sccb_ATC = currSCCB->Sccb_savedATC;
2871
2872                         FPT_hostDataXferRestart(currSCCB);
2873                 }
2874
2875                 ACCEPT_MSG(port);
2876                 WR_HARPOON(port + hp_autostart_1,
2877                            (AUTO_IMMED + DISCONNECT_START));
2878         }
2879
2880         else if (message == COMMAND_COMPLETE) {
2881
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;
2886                 }
2887
2888                 ACCEPT_MSG(port);
2889
2890         }
2891
2892         else if ((message == NOP) || (message >= IDENTIFY_BASE) ||
2893                  (message == INITIATE_RECOVERY) ||
2894                  (message == RELEASE_RECOVERY)) {
2895
2896                 ACCEPT_MSG(port);
2897                 WR_HARPOON(port + hp_autostart_1,
2898                            (AUTO_IMMED + DISCONNECT_START));
2899         }
2900
2901         else if (message == MESSAGE_REJECT) {
2902
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) ==
2907                         TAG_Q_TRYING))
2908                 {
2909                         WRW_HARPOON((port + hp_intstat), BUS_FREE);
2910
2911                         ACCEPT_MSG(port);
2912
2913                         while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2914                                (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)))
2915                         {
2916                         }
2917
2918                         if (currSCCB->Lun == 0x00) {
2919                                 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
2920
2921                                         currTar_Info->TarStatus |=
2922                                             (unsigned char)SYNC_SUPPORTED;
2923
2924                                         currTar_Info->TarEEValue &=
2925                                             ~EE_SYNC_MASK;
2926                                 }
2927
2928                                 else if (currSCCB->Sccb_scsistat ==
2929                                           SELECT_WN_ST) {
2930
2931                                         currTar_Info->TarStatus =
2932                                             (currTar_Info->
2933                                              TarStatus & ~WIDE_ENABLED) |
2934                                             WIDE_NEGOCIATED;
2935
2936                                         currTar_Info->TarEEValue &=
2937                                             ~EE_WIDE_SCSI;
2938
2939                                 }
2940
2941                                 else if ((currTar_Info->
2942                                           TarStatus & TAR_TAG_Q_MASK) ==
2943                                          TAG_Q_TRYING) {
2944                                         currTar_Info->TarStatus =
2945                                             (currTar_Info->
2946                                              TarStatus & ~(unsigned char)
2947                                              TAR_TAG_Q_MASK) | TAG_Q_REJECT;
2948
2949                                         currSCCB->ControlByte &= ~F_USE_CMD_Q;
2950                                         CurrCard->discQCount--;
2951                                         CurrCard->discQ_Tbl[currSCCB->
2952                                                             Sccb_tag] = NULL;
2953                                         currSCCB->Sccb_tag = 0x00;
2954
2955                                 }
2956                         }
2957
2958                         if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
2959
2960                                 if (currSCCB->Lun == 0x00) {
2961                                         WRW_HARPOON((port + hp_intstat),
2962                                                     BUS_FREE);
2963                                         CurrCard->globalFlags |= F_NEW_SCCB_CMD;
2964                                 }
2965                         }
2966
2967                         else {
2968
2969                                 if ((CurrCard->globalFlags & F_CONLUN_IO) &&
2970                                     ((currTar_Info->
2971                                       TarStatus & TAR_TAG_Q_MASK) !=
2972                                      TAG_Q_TRYING))
2973                                         currTar_Info->TarLUNBusy[currSCCB->
2974                                                                  Lun] = 1;
2975                                 else
2976                                         currTar_Info->TarLUNBusy[0] = 1;
2977
2978                                 currSCCB->ControlByte &=
2979                                     ~(unsigned char)F_USE_CMD_Q;
2980
2981                                 WR_HARPOON(port + hp_autostart_1,
2982                                            (AUTO_IMMED + DISCONNECT_START));
2983
2984                         }
2985                 }
2986
2987                 else {
2988                         ACCEPT_MSG(port);
2989
2990                         while ((!(RD_HARPOON(port + hp_scsisig) & SCSI_REQ)) &&
2991                                (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)))
2992                         {
2993                         }
2994
2995                         if (!(RDW_HARPOON((port + hp_intstat)) & BUS_FREE)) {
2996                                 WR_HARPOON(port + hp_autostart_1,
2997                                            (AUTO_IMMED + DISCONNECT_START));
2998                         }
2999                 }
3000         }
3001
3002         else if (message == EXTENDED_MESSAGE) {
3003
3004                 ACCEPT_MSG(port);
3005                 FPT_shandem(port, p_card, currSCCB);
3006         }
3007
3008         else if (message == IGNORE_WIDE_RESIDUE) {
3009
3010                 ACCEPT_MSG(port);       /* ACK the RESIDUE MSG */
3011
3012                 message = FPT_sfm(port, currSCCB);
3013
3014                 if (currSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
3015                         ACCEPT_MSG(port);
3016                 WR_HARPOON(port + hp_autostart_1,
3017                            (AUTO_IMMED + DISCONNECT_START));
3018         }
3019
3020         else {
3021
3022                 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
3023                 currSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3024
3025                 ACCEPT_MSG_ATN(port);
3026                 WR_HARPOON(port + hp_autostart_1,
3027                            (AUTO_IMMED + DISCONNECT_START));
3028         }
3029 }
3030
3031 /*---------------------------------------------------------------------
3032  *
3033  * Function: FPT_shandem
3034  *
3035  * Description: Decide what to do with the extended message.
3036  *
3037  *---------------------------------------------------------------------*/
3038 static void FPT_shandem(u32 port, unsigned char p_card, struct sccb *pCurrSCCB)
3039 {
3040         unsigned char length, message;
3041
3042         length = FPT_sfm(port, pCurrSCCB);
3043         if (length) {
3044
3045                 ACCEPT_MSG(port);
3046                 message = FPT_sfm(port, pCurrSCCB);
3047                 if (message) {
3048
3049                         if (message == EXTENDED_SDTR) {
3050
3051                                 if (length == 0x03) {
3052
3053                                         ACCEPT_MSG(port);
3054                                         FPT_stsyncn(port, p_card);
3055                                 } else {
3056
3057                                         pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3058                                         ACCEPT_MSG_ATN(port);
3059                                 }
3060                         } else if (message == EXTENDED_WDTR) {
3061
3062                                 if (length == 0x02) {
3063
3064                                         ACCEPT_MSG(port);
3065                                         FPT_stwidn(port, p_card);
3066                                 } else {
3067
3068                                         pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3069                                         ACCEPT_MSG_ATN(port);
3070
3071                                         WR_HARPOON(port + hp_autostart_1,
3072                                                    (AUTO_IMMED +
3073                                                     DISCONNECT_START));
3074                                 }
3075                         } else {
3076
3077                                 pCurrSCCB->Sccb_scsimsg = MESSAGE_REJECT;
3078                                 ACCEPT_MSG_ATN(port);
3079
3080                                 WR_HARPOON(port + hp_autostart_1,
3081                                            (AUTO_IMMED + DISCONNECT_START));
3082                         }
3083                 } else {
3084                         if (pCurrSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
3085                                 ACCEPT_MSG(port);
3086                         WR_HARPOON(port + hp_autostart_1,
3087                                    (AUTO_IMMED + DISCONNECT_START));
3088                 }
3089         } else {
3090                 if (pCurrSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)
3091                         WR_HARPOON(port + hp_autostart_1,
3092                                    (AUTO_IMMED + DISCONNECT_START));
3093         }
3094 }
3095
3096 /*---------------------------------------------------------------------
3097  *
3098  * Function: FPT_sisyncn
3099  *
3100  * Description: Read in a message byte from the SCSI bus, and check
3101  *              for a parity error.
3102  *
3103  *---------------------------------------------------------------------*/
3104
3105 static unsigned char FPT_sisyncn(u32 port, unsigned char p_card,
3106                                  unsigned char syncFlag)
3107 {
3108         struct sccb *currSCCB;
3109         struct sccb_mgr_tar_info *currTar_Info;
3110
3111         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3112         currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3113
3114         if (!((currTar_Info->TarStatus & TAR_SYNC_MASK) == SYNC_TRYING)) {
3115
3116                 WRW_HARPOON((port + ID_MSG_STRT),
3117                             (MPM_OP + AMSG_OUT +
3118                              (currSCCB->
3119                               Sccb_idmsg & ~(unsigned char)DISC_PRIV)));
3120
3121                 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
3122
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));
3128
3129                 if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_20MB)
3130
3131                         WRW_HARPOON((port + SYNC_MSGS + 6),
3132                                     (MPM_OP + AMSG_OUT + 12));
3133
3134                 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) ==
3135                          EE_SYNC_10MB)
3136
3137                         WRW_HARPOON((port + SYNC_MSGS + 6),
3138                                     (MPM_OP + AMSG_OUT + 25));
3139
3140                 else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) ==
3141                          EE_SYNC_5MB)
3142
3143                         WRW_HARPOON((port + SYNC_MSGS + 6),
3144                                     (MPM_OP + AMSG_OUT + 50));
3145
3146                 else
3147                         WRW_HARPOON((port + SYNC_MSGS + 6),
3148                                     (MPM_OP + AMSG_OUT + 00));
3149
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));
3154
3155                 if (syncFlag == 0) {
3156                         WR_HARPOON(port + hp_autostart_3,
3157                                    (SELECT + SELCHK_STRT));
3158                         currTar_Info->TarStatus =
3159                             ((currTar_Info->
3160                               TarStatus & ~(unsigned char)TAR_SYNC_MASK) |
3161                              (unsigned char)SYNC_TRYING);
3162                 } else {
3163                         WR_HARPOON(port + hp_autostart_3,
3164                                    (AUTO_IMMED + CMD_ONLY_STRT));
3165                 }
3166
3167                 return 1;
3168         }
3169
3170         else {
3171
3172                 currTar_Info->TarStatus |= (unsigned char)SYNC_SUPPORTED;
3173                 currTar_Info->TarEEValue &= ~EE_SYNC_MASK;
3174                 return 0;
3175         }
3176 }
3177
3178 /*---------------------------------------------------------------------
3179  *
3180  * Function: FPT_stsyncn
3181  *
3182  * Description: The has sent us a Sync Nego message so handle it as
3183  *              necessary.
3184  *
3185  *---------------------------------------------------------------------*/
3186 static void FPT_stsyncn(u32 port, unsigned char p_card)
3187 {
3188         unsigned char sync_msg, offset, sync_reg, our_sync_msg;
3189         struct sccb *currSCCB;
3190         struct sccb_mgr_tar_info *currTar_Info;
3191
3192         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3193         currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3194
3195         sync_msg = FPT_sfm(port, currSCCB);
3196
3197         if ((sync_msg == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3198                 WR_HARPOON(port + hp_autostart_1,
3199                            (AUTO_IMMED + DISCONNECT_START));
3200                 return;
3201         }
3202
3203         ACCEPT_MSG(port);
3204
3205         offset = FPT_sfm(port, currSCCB);
3206
3207         if ((offset == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3208                 WR_HARPOON(port + hp_autostart_1,
3209                            (AUTO_IMMED + DISCONNECT_START));
3210                 return;
3211         }
3212
3213         if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_20MB)
3214
3215                 our_sync_msg = 12;      /* Setup our Message to 20mb/s */
3216
3217         else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_10MB)
3218
3219                 our_sync_msg = 25;      /* Setup our Message to 10mb/s */
3220
3221         else if ((currTar_Info->TarEEValue & EE_SYNC_MASK) == EE_SYNC_5MB)
3222
3223                 our_sync_msg = 50;      /* Setup our Message to 5mb/s */
3224         else
3225
3226                 our_sync_msg = 0;       /* Message = Async */
3227
3228         if (sync_msg < our_sync_msg) {
3229                 sync_msg = our_sync_msg;        /*if faster, then set to max. */
3230         }
3231
3232         if (offset == ASYNC)
3233                 sync_msg = ASYNC;
3234
3235         if (offset > MAX_OFFSET)
3236                 offset = MAX_OFFSET;
3237
3238         sync_reg = 0x00;
3239
3240         if (sync_msg > 12)
3241
3242                 sync_reg = 0x20;        /* Use 10MB/s */
3243
3244         if (sync_msg > 25)
3245
3246                 sync_reg = 0x40;        /* Use 6.6MB/s */
3247
3248         if (sync_msg > 38)
3249
3250                 sync_reg = 0x60;        /* Use 5MB/s */
3251
3252         if (sync_msg > 50)
3253
3254                 sync_reg = 0x80;        /* Use 4MB/s */
3255
3256         if (sync_msg > 62)
3257
3258                 sync_reg = 0xA0;        /* Use 3.33MB/s */
3259
3260         if (sync_msg > 75)
3261
3262                 sync_reg = 0xC0;        /* Use 2.85MB/s */
3263
3264         if (sync_msg > 87)
3265
3266                 sync_reg = 0xE0;        /* Use 2.5MB/s */
3267
3268         if (sync_msg > 100) {
3269
3270                 sync_reg = 0x00;        /* Use ASYNC */
3271                 offset = 0x00;
3272         }
3273
3274         if (currTar_Info->TarStatus & WIDE_ENABLED)
3275
3276                 sync_reg |= offset;
3277
3278         else
3279
3280                 sync_reg |= (offset | NARROW_SCSI);
3281
3282         FPT_sssyncv(port, currSCCB->TargID, sync_reg, currTar_Info);
3283
3284         if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
3285
3286                 ACCEPT_MSG(port);
3287
3288                 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3289                                             ~(unsigned char)TAR_SYNC_MASK) |
3290                                            (unsigned char)SYNC_SUPPORTED);
3291
3292                 WR_HARPOON(port + hp_autostart_1,
3293                            (AUTO_IMMED + DISCONNECT_START));
3294         }
3295
3296         else {
3297
3298                 ACCEPT_MSG_ATN(port);
3299
3300                 FPT_sisyncr(port, sync_msg, offset);
3301
3302                 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3303                                             ~(unsigned char)TAR_SYNC_MASK) |
3304                                            (unsigned char)SYNC_SUPPORTED);
3305         }
3306 }
3307
3308 /*---------------------------------------------------------------------
3309  *
3310  * Function: FPT_sisyncr
3311  *
3312  * Description: Answer the targets sync message.
3313  *
3314  *---------------------------------------------------------------------*/
3315 static void FPT_sisyncr(u32 port, unsigned char sync_pulse,
3316                         unsigned char offset)
3317 {
3318         ARAM_ACCESS(port);
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));
3328         SGRAM_ACCESS(port);
3329
3330         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3331         WRW_HARPOON((port + hp_intstat), CLR_ALL_INT_1);
3332
3333         WR_HARPOON(port + hp_autostart_3, (AUTO_IMMED + CMD_ONLY_STRT));
3334
3335         while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | AUTO_INT))) {
3336         }
3337 }
3338
3339 /*---------------------------------------------------------------------
3340  *
3341  * Function: FPT_siwidn
3342  *
3343  * Description: Read in a message byte from the SCSI bus, and check
3344  *              for a parity error.
3345  *
3346  *---------------------------------------------------------------------*/
3347
3348 static unsigned char FPT_siwidn(u32 port, unsigned char p_card)
3349 {
3350         struct sccb *currSCCB;
3351         struct sccb_mgr_tar_info *currTar_Info;
3352
3353         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3354         currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3355
3356         if (!((currTar_Info->TarStatus & TAR_WIDE_MASK) == WIDE_NEGOCIATED)) {
3357
3358                 WRW_HARPOON((port + ID_MSG_STRT),
3359                             (MPM_OP + AMSG_OUT +
3360                              (currSCCB->
3361                               Sccb_idmsg & ~(unsigned char)DISC_PRIV)));
3362
3363                 WRW_HARPOON((port + ID_MSG_STRT + 2), BRH_OP + ALWAYS + CMDPZ);
3364
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));
3374
3375                 WR_HARPOON(port + hp_autostart_3, (SELECT + SELCHK_STRT));
3376
3377                 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3378                                             ~(unsigned char)TAR_WIDE_MASK) |
3379                                            (unsigned char)WIDE_ENABLED);
3380
3381                 return 1;
3382         }
3383
3384         else {
3385
3386                 currTar_Info->TarStatus = ((currTar_Info->TarStatus &
3387                                             ~(unsigned char)TAR_WIDE_MASK) |
3388                                            WIDE_NEGOCIATED);
3389
3390                 currTar_Info->TarEEValue &= ~EE_WIDE_SCSI;
3391                 return 0;
3392         }
3393 }
3394
3395 /*---------------------------------------------------------------------
3396  *
3397  * Function: FPT_stwidn
3398  *
3399  * Description: The has sent us a Wide Nego message so handle it as
3400  *              necessary.
3401  *
3402  *---------------------------------------------------------------------*/
3403 static void FPT_stwidn(u32 port, unsigned char p_card)
3404 {
3405         unsigned char width;
3406         struct sccb *currSCCB;
3407         struct sccb_mgr_tar_info *currTar_Info;
3408
3409         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3410         currTar_Info = &FPT_sccbMgrTbl[p_card][currSCCB->TargID];
3411
3412         width = FPT_sfm(port, currSCCB);
3413
3414         if ((width == 0x00) && (currSCCB->Sccb_scsimsg == MSG_PARITY_ERROR)) {
3415                 WR_HARPOON(port + hp_autostart_1,
3416                            (AUTO_IMMED + DISCONNECT_START));
3417                 return;
3418         }
3419
3420         if (!(currTar_Info->TarEEValue & EE_WIDE_SCSI))
3421                 width = 0;
3422
3423         if (width) {
3424                 currTar_Info->TarStatus |= WIDE_ENABLED;
3425                 width = 0;
3426         } else {
3427                 width = NARROW_SCSI;
3428                 currTar_Info->TarStatus &= ~WIDE_ENABLED;
3429         }
3430
3431         FPT_sssyncv(port, currSCCB->TargID, width, currTar_Info);
3432
3433         if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
3434
3435                 currTar_Info->TarStatus |= WIDE_NEGOCIATED;
3436
3437                 if (!
3438                     ((currTar_Info->TarStatus & TAR_SYNC_MASK) ==
3439                      SYNC_SUPPORTED)) {
3440                         ACCEPT_MSG_ATN(port);
3441                         ARAM_ACCESS(port);
3442                         FPT_sisyncn(port, p_card, 1);
3443                         currSCCB->Sccb_scsistat = SELECT_SN_ST;
3444                         SGRAM_ACCESS(port);
3445                 } else {
3446                         ACCEPT_MSG(port);
3447                         WR_HARPOON(port + hp_autostart_1,
3448                                    (AUTO_IMMED + DISCONNECT_START));
3449                 }
3450         }
3451
3452         else {
3453
3454                 ACCEPT_MSG_ATN(port);
3455
3456                 if (currTar_Info->TarEEValue & EE_WIDE_SCSI)
3457                         width = SM16BIT;
3458                 else
3459                         width = SM8BIT;
3460
3461                 FPT_siwidr(port, width);
3462
3463                 currTar_Info->TarStatus |= (WIDE_NEGOCIATED | WIDE_ENABLED);
3464         }
3465 }
3466
3467 /*---------------------------------------------------------------------
3468  *
3469  * Function: FPT_siwidr
3470  *
3471  * Description: Answer the targets Wide nego message.
3472  *
3473  *---------------------------------------------------------------------*/
3474 static void FPT_siwidr(u32 port, unsigned char width)
3475 {
3476         ARAM_ACCESS(port);
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));
3485         SGRAM_ACCESS(port);
3486
3487         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3488         WRW_HARPOON((port + hp_intstat), CLR_ALL_INT_1);
3489
3490         WR_HARPOON(port + hp_autostart_3, (AUTO_IMMED + CMD_ONLY_STRT));
3491
3492         while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | AUTO_INT))) {
3493         }
3494 }
3495
3496 /*---------------------------------------------------------------------
3497  *
3498  * Function: FPT_sssyncv
3499  *
3500  * Description: Write the desired value to the Sync Register for the
3501  *              ID specified.
3502  *
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)
3507 {
3508         unsigned char index;
3509
3510         index = p_id;
3511
3512         switch (index) {
3513
3514         case 0:
3515                 index = 12;     /* hp_synctarg_0 */
3516                 break;
3517         case 1:
3518                 index = 13;     /* hp_synctarg_1 */
3519                 break;
3520         case 2:
3521                 index = 14;     /* hp_synctarg_2 */
3522                 break;
3523         case 3:
3524                 index = 15;     /* hp_synctarg_3 */
3525                 break;
3526         case 4:
3527                 index = 8;      /* hp_synctarg_4 */
3528                 break;
3529         case 5:
3530                 index = 9;      /* hp_synctarg_5 */
3531                 break;
3532         case 6:
3533                 index = 10;     /* hp_synctarg_6 */
3534                 break;
3535         case 7:
3536                 index = 11;     /* hp_synctarg_7 */
3537                 break;
3538         case 8:
3539                 index = 4;      /* hp_synctarg_8 */
3540                 break;
3541         case 9:
3542                 index = 5;      /* hp_synctarg_9 */
3543                 break;
3544         case 10:
3545                 index = 6;      /* hp_synctarg_10 */
3546                 break;
3547         case 11:
3548                 index = 7;      /* hp_synctarg_11 */
3549                 break;
3550         case 12:
3551                 index = 0;      /* hp_synctarg_12 */
3552                 break;
3553         case 13:
3554                 index = 1;      /* hp_synctarg_13 */
3555                 break;
3556         case 14:
3557                 index = 2;      /* hp_synctarg_14 */
3558                 break;
3559         case 15:
3560                 index = 3;      /* hp_synctarg_15 */
3561
3562         }
3563
3564         WR_HARPOON(p_port + hp_synctarg_base + index, p_sync_value);
3565
3566         currTar_Info->TarSyncCtrl = p_sync_value;
3567 }
3568
3569 /*---------------------------------------------------------------------
3570  *
3571  * Function: FPT_sresb
3572  *
3573  * Description: Reset the desired card's SCSI bus.
3574  *
3575  *---------------------------------------------------------------------*/
3576 static void FPT_sresb(u32 port, unsigned char p_card)
3577 {
3578         unsigned char scsiID, i;
3579
3580         struct sccb_mgr_tar_info *currTar_Info;
3581
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);
3585
3586         WR_HARPOON(port + hp_scsictrl_0, SCSI_RST);
3587
3588         scsiID = RD_HARPOON(port + hp_seltimeout);
3589         WR_HARPOON(port + hp_seltimeout, TO_5ms);
3590         WRW_HARPOON((port + hp_intstat), TIMEOUT);
3591
3592         WR_HARPOON(port + hp_portctrl_0, (SCSI_PORT | START_TO));
3593
3594         while (!(RDW_HARPOON((port + hp_intstat)) & TIMEOUT)) {
3595         }
3596
3597         WR_HARPOON(port + hp_seltimeout, scsiID);
3598
3599         WR_HARPOON(port + hp_scsictrl_0, ENA_SCAM_SEL);
3600
3601         FPT_Wait(port, TO_5ms);
3602
3603         WRW_HARPOON((port + hp_intstat), CLR_ALL_INT);
3604
3605         WR_HARPOON(port + hp_int_mask, (RD_HARPOON(port + hp_int_mask) | 0x00));
3606
3607         for (scsiID = 0; scsiID < MAX_SCSI_TAR; scsiID++) {
3608                 currTar_Info = &FPT_sccbMgrTbl[p_card][scsiID];
3609
3610                 if (currTar_Info->TarEEValue & EE_SYNC_MASK) {
3611                         currTar_Info->TarSyncCtrl = 0;
3612                         currTar_Info->TarStatus &= ~TAR_SYNC_MASK;
3613                 }
3614
3615                 if (currTar_Info->TarEEValue & EE_WIDE_SCSI) {
3616                         currTar_Info->TarStatus &= ~TAR_WIDE_MASK;
3617                 }
3618
3619                 FPT_sssyncv(port, scsiID, NARROW_SCSI, currTar_Info);
3620
3621                 FPT_SccbMgrTableInitTarget(p_card, scsiID);
3622         }
3623
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
3627                                              | F_NEW_SCCB_CMD);
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;
3631
3632         for (i = 0; i < QUEUE_DEPTH; i++)
3633                 FPT_BL_Card[p_card].discQ_Tbl[i] = NULL;
3634
3635         WR_HARPOON(port + hp_page_ctrl,
3636                    (RD_HARPOON(port + hp_page_ctrl) & ~G_INT_DISABLE));
3637
3638 }
3639
3640 /*---------------------------------------------------------------------
3641  *
3642  * Function: FPT_ssenss
3643  *
3644  * Description: Setup for the Auto Sense command.
3645  *
3646  *---------------------------------------------------------------------*/
3647 static void FPT_ssenss(struct sccb_card *pCurrCard)
3648 {
3649         unsigned char i;
3650         struct sccb *currSCCB;
3651
3652         currSCCB = pCurrCard->currentSCCB;
3653
3654         currSCCB->Save_CdbLen = currSCCB->CdbLength;
3655
3656         for (i = 0; i < 6; i++) {
3657
3658                 currSCCB->Save_Cdb[i] = currSCCB->Cdb[i];
3659         }
3660
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;
3668
3669         currSCCB->Sccb_XferCnt = (u32)currSCCB->RequestSenseLength;
3670
3671         currSCCB->Sccb_ATC = 0x00;
3672
3673         currSCCB->Sccb_XferState |= F_AUTO_SENSE;
3674
3675         currSCCB->Sccb_XferState &= ~F_SG_XFER;
3676
3677         currSCCB->Sccb_idmsg = currSCCB->Sccb_idmsg & ~(unsigned char)DISC_PRIV;
3678
3679         currSCCB->ControlByte = 0x00;
3680
3681         currSCCB->Sccb_MGRFlags &= F_STATUSLOADED;
3682 }
3683
3684 /*---------------------------------------------------------------------
3685  *
3686  * Function: FPT_sxfrp
3687  *
3688  * Description: Transfer data into the bit bucket until the device
3689  *              decides to switch phase.
3690  *
3691  *---------------------------------------------------------------------*/
3692
3693 static void FPT_sxfrp(u32 p_port, unsigned char p_card)
3694 {
3695         unsigned char curr_phz;
3696
3697         DISABLE_AUTO(p_port);
3698
3699         if (FPT_BL_Card[p_card].globalFlags & F_HOST_XFER_ACT) {
3700
3701                 FPT_hostDataXferAbort(p_port, p_card,
3702                                       FPT_BL_Card[p_card].currentSCCB);
3703
3704         }
3705
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.       */
3708
3709         if (RDW_HARPOON((p_port + hp_intstat)) &
3710             (BUS_FREE | XFER_CNT_0 | AUTO_INT))
3711                 return;
3712
3713         WR_HARPOON(p_port + hp_xfercnt_0, 0x00);
3714
3715         curr_phz = RD_HARPOON(p_port + hp_scsisig) & (unsigned char)S_SCSI_PHZ;
3716
3717         WRW_HARPOON((p_port + hp_intstat), XFER_CNT_0);
3718
3719         WR_HARPOON(p_port + hp_scsisig, curr_phz);
3720
3721         while (!(RDW_HARPOON((p_port + hp_intstat)) & (BUS_FREE | RESET)) &&
3722                (curr_phz ==
3723                 (RD_HARPOON(p_port + hp_scsisig) & (unsigned char)S_SCSI_PHZ)))
3724         {
3725                 if (curr_phz & (unsigned char)SCSI_IOBIT) {
3726                         WR_HARPOON(p_port + hp_portctrl_0,
3727                                    (SCSI_PORT | HOST_PORT | SCSI_INBIT));
3728
3729                         if (!(RD_HARPOON(p_port + hp_xferstat) & FIFO_EMPTY)) {
3730                                 RD_HARPOON(p_port + hp_fifodata_0);
3731                         }
3732                 } else {
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);
3737                         }
3738                 }
3739         }                       /* End of While loop for padding data I/O phase */
3740
3741         while (!(RDW_HARPOON((p_port + hp_intstat)) & (BUS_FREE | RESET))) {
3742                 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_REQ)
3743                         break;
3744         }
3745
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);
3750         }
3751
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)) {
3756                 }
3757
3758                 if (RDW_HARPOON((p_port + hp_intstat)) &
3759                     (ICMD_COMP | ITAR_DISC))
3760                         while (!
3761                                (RDW_HARPOON((p_port + hp_intstat)) &
3762                                 (BUS_FREE | RSEL))) ;
3763         }
3764 }
3765
3766 /*---------------------------------------------------------------------
3767  *
3768  * Function: FPT_schkdd
3769  *
3770  * Description: Make sure data has been flushed from both FIFOs and abort
3771  *              the operations if necessary.
3772  *
3773  *---------------------------------------------------------------------*/
3774
3775 static void FPT_schkdd(u32 port, unsigned char p_card)
3776 {
3777         unsigned short TimeOutLoop;
3778         unsigned char sPhase;
3779
3780         struct sccb *currSCCB;
3781
3782         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3783
3784         if ((currSCCB->Sccb_scsistat != DATA_OUT_ST) &&
3785             (currSCCB->Sccb_scsistat != DATA_IN_ST)) {
3786                 return;
3787         }
3788
3789         if (currSCCB->Sccb_XferState & F_ODD_BALL_CNT) {
3790
3791                 currSCCB->Sccb_ATC += (currSCCB->Sccb_XferCnt - 1);
3792
3793                 currSCCB->Sccb_XferCnt = 1;
3794
3795                 currSCCB->Sccb_XferState &= ~F_ODD_BALL_CNT;
3796                 WRW_HARPOON((port + hp_fiforead), (unsigned short)0x00);
3797                 WR_HARPOON(port + hp_xferstat, 0x00);
3798         }
3799
3800         else {
3801
3802                 currSCCB->Sccb_ATC += currSCCB->Sccb_XferCnt;
3803
3804                 currSCCB->Sccb_XferCnt = 0;
3805         }
3806
3807         if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
3808             (currSCCB->HostStatus == SCCB_COMPLETE)) {
3809
3810                 currSCCB->HostStatus = SCCB_PARITY_ERR;
3811                 WRW_HARPOON((port + hp_intstat), PARITY);
3812         }
3813
3814         FPT_hostDataXferAbort(port, p_card, currSCCB);
3815
3816         while (RD_HARPOON(port + hp_scsisig) & SCSI_ACK) {
3817         }
3818
3819         TimeOutLoop = 0;
3820
3821         while (RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY) {
3822                 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
3823                         return;
3824                 }
3825                 if (RD_HARPOON(port + hp_offsetctr) & (unsigned char)0x1F) {
3826                         break;
3827                 }
3828                 if (RDW_HARPOON((port + hp_intstat)) & RESET) {
3829                         return;
3830                 }
3831                 if ((RD_HARPOON(port + hp_scsisig) & SCSI_REQ)
3832                     || (TimeOutLoop++ > 0x3000))
3833                         break;
3834         }
3835
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))) {
3841
3842                 WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3843
3844                 if (!(currSCCB->Sccb_XferState & F_ALL_XFERRED)) {
3845                         if (currSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
3846                                 FPT_phaseDataIn(port, p_card);
3847                         }
3848
3849                         else {
3850                                 FPT_phaseDataOut(port, p_card);
3851                         }
3852                 } else {
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);
3858                         }
3859                 }
3860
3861         }
3862
3863         else {
3864                 WR_HARPOON(port + hp_portctrl_0, 0x00);
3865         }
3866 }
3867
3868 /*---------------------------------------------------------------------
3869  *
3870  * Function: FPT_sinits
3871  *
3872  * Description: Setup SCCB manager fields in this SCCB.
3873  *
3874  *---------------------------------------------------------------------*/
3875
3876 static void FPT_sinits(struct sccb *p_sccb, unsigned char p_card)
3877 {
3878         struct sccb_mgr_tar_info *currTar_Info;
3879
3880         if ((p_sccb->TargID >= MAX_SCSI_TAR) || (p_sccb->Lun >= MAX_LUN)) {
3881                 return;
3882         }
3883         currTar_Info = &FPT_sccbMgrTbl[p_card][p_sccb->TargID];
3884
3885         p_sccb->Sccb_XferState = 0x00;
3886         p_sccb->Sccb_XferCnt = p_sccb->DataLength;
3887
3888         if ((p_sccb->OperationCode == SCATTER_GATHER_COMMAND) ||
3889             (p_sccb->OperationCode == RESIDUAL_SG_COMMAND)) {
3890
3891                 p_sccb->Sccb_SGoffset = 0;
3892                 p_sccb->Sccb_XferState = F_SG_XFER;
3893                 p_sccb->Sccb_XferCnt = 0x00;
3894         }
3895
3896         if (p_sccb->DataLength == 0x00)
3897
3898                 p_sccb->Sccb_XferState |= F_ALL_XFERRED;
3899
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;
3903
3904                 else
3905                         currTar_Info->TarStatus |= TAG_Q_TRYING;
3906         }
3907
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 */
3911
3912 /*
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)) {
3916 */
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);
3920         } else {
3921                 p_sccb->Sccb_idmsg = IDENTIFY(false, p_sccb->Lun);
3922         }
3923
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;
3931 /*
3932    p_sccb->SccbVirtDataPtr    = 0x00;
3933    p_sccb->Sccb_forwardlink   = NULL;
3934    p_sccb->Sccb_backlink      = NULL;
3935  */
3936         p_sccb->Sccb_scsistat = BUS_FREE_ST;
3937         p_sccb->SccbStatus = SCCB_IN_PROCESS;
3938         p_sccb->Sccb_scsimsg = NOP;
3939
3940 }
3941
3942 /*---------------------------------------------------------------------
3943  *
3944  * Function: Phase Decode
3945  *
3946  * Description: Determine the phase and call the appropriate function.
3947  *
3948  *---------------------------------------------------------------------*/
3949
3950 static void FPT_phaseDecode(u32 p_port, unsigned char p_card)
3951 {
3952         unsigned char phase_ref;
3953         void (*phase) (u32, unsigned char);
3954
3955         DISABLE_AUTO(p_port);
3956
3957         phase_ref =
3958             (unsigned char)(RD_HARPOON(p_port + hp_scsisig) & S_SCSI_PHZ);
3959
3960         phase = FPT_s_PhaseTbl[phase_ref];
3961
3962         (*phase) (p_port, p_card);      /* Call the correct phase func */
3963 }
3964
3965 /*---------------------------------------------------------------------
3966  *
3967  * Function: Data Out Phase
3968  *
3969  * Description: Start up both the BusMaster and Xbow.
3970  *
3971  *---------------------------------------------------------------------*/
3972
3973 static void FPT_phaseDataOut(u32 port, unsigned char p_card)
3974 {
3975
3976         struct sccb *currSCCB;
3977
3978         currSCCB = FPT_BL_Card[p_card].currentSCCB;
3979         if (currSCCB == NULL) {
3980                 return;         /* Exit if No SCCB record */
3981         }
3982
3983         currSCCB->Sccb_scsistat = DATA_OUT_ST;
3984         currSCCB->Sccb_XferState &= ~(F_HOST_XFER_DIR | F_NO_DATA_YET);
3985
3986         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
3987
3988         WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
3989
3990         WR_HARPOON(port + hp_autostart_0, (END_DATA + END_DATA_START));
3991
3992         FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
3993
3994         if (currSCCB->Sccb_XferCnt == 0) {
3995
3996                 if ((currSCCB->ControlByte & SCCB_DATA_XFER_OUT) &&
3997                     (currSCCB->HostStatus == SCCB_COMPLETE))
3998                         currSCCB->HostStatus = SCCB_DATA_OVER_RUN;
3999
4000                 FPT_sxfrp(port, p_card);
4001                 if (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | RESET)))
4002                         FPT_phaseDecode(port, p_card);
4003         }
4004 }
4005
4006 /*---------------------------------------------------------------------
4007  *
4008  * Function: Data In Phase
4009  *
4010  * Description: Startup the BusMaster and the XBOW.
4011  *
4012  *---------------------------------------------------------------------*/
4013
4014 static void FPT_phaseDataIn(u32 port, unsigned char p_card)
4015 {
4016
4017         struct sccb *currSCCB;
4018
4019         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4020
4021         if (currSCCB == NULL) {
4022                 return;         /* Exit if No SCCB record */
4023         }
4024
4025         currSCCB->Sccb_scsistat = DATA_IN_ST;
4026         currSCCB->Sccb_XferState |= F_HOST_XFER_DIR;
4027         currSCCB->Sccb_XferState &= ~F_NO_DATA_YET;
4028
4029         WR_HARPOON(port + hp_portctrl_0, SCSI_PORT);
4030
4031         WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
4032
4033         WR_HARPOON(port + hp_autostart_0, (END_DATA + END_DATA_START));
4034
4035         FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
4036
4037         if (currSCCB->Sccb_XferCnt == 0) {
4038
4039                 if ((currSCCB->ControlByte & SCCB_DATA_XFER_IN) &&
4040                     (currSCCB->HostStatus == SCCB_COMPLETE))
4041                         currSCCB->HostStatus = SCCB_DATA_OVER_RUN;
4042
4043                 FPT_sxfrp(port, p_card);
4044                 if (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | RESET)))
4045                         FPT_phaseDecode(port, p_card);
4046
4047         }
4048 }
4049
4050 /*---------------------------------------------------------------------
4051  *
4052  * Function: Command Phase
4053  *
4054  * Description: Load the CDB into the automation and start it up.
4055  *
4056  *---------------------------------------------------------------------*/
4057
4058 static void FPT_phaseCommand(u32 p_port, unsigned char p_card)
4059 {
4060         struct sccb *currSCCB;
4061         u32 cdb_reg;
4062         unsigned char i;
4063
4064         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4065
4066         if (currSCCB->OperationCode == RESET_COMMAND) {
4067
4068                 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4069                 currSCCB->CdbLength = SIX_BYTE_CMD;
4070         }
4071
4072         WR_HARPOON(p_port + hp_scsisig, 0x00);
4073
4074         ARAM_ACCESS(p_port);
4075
4076         cdb_reg = p_port + CMD_STRT;
4077
4078         for (i = 0; i < currSCCB->CdbLength; i++) {
4079
4080                 if (currSCCB->OperationCode == RESET_COMMAND)
4081
4082                         WRW_HARPOON(cdb_reg, (MPM_OP + ACOMMAND + 0x00));
4083
4084                 else
4085                         WRW_HARPOON(cdb_reg,
4086                                     (MPM_OP + ACOMMAND + currSCCB->Cdb[i]));
4087                 cdb_reg += 2;
4088         }
4089
4090         if (currSCCB->CdbLength != TWELVE_BYTE_CMD)
4091                 WRW_HARPOON(cdb_reg, (BRH_OP + ALWAYS + NP));
4092
4093         WR_HARPOON(p_port + hp_portctrl_0, (SCSI_PORT));
4094
4095         currSCCB->Sccb_scsistat = COMMAND_ST;
4096
4097         WR_HARPOON(p_port + hp_autostart_3, (AUTO_IMMED | CMD_ONLY_STRT));
4098         SGRAM_ACCESS(p_port);
4099 }
4100
4101 /*---------------------------------------------------------------------
4102  *
4103  * Function: Status phase
4104  *
4105  * Description: Bring in the status and command complete message bytes
4106  *
4107  *---------------------------------------------------------------------*/
4108
4109 static void FPT_phaseStatus(u32 port, unsigned char p_card)
4110 {
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?
4114          */
4115
4116         WR_HARPOON(port + hp_scsisig, 0x00);
4117
4118         WR_HARPOON(port + hp_autostart_0, (AUTO_IMMED + END_DATA_START));
4119 }
4120
4121 /*---------------------------------------------------------------------
4122  *
4123  * Function: Phase Message Out
4124  *
4125  * Description: Send out our message (if we have one) and handle whatever
4126  *              else is involed.
4127  *
4128  *---------------------------------------------------------------------*/
4129
4130 static void FPT_phaseMsgOut(u32 port, unsigned char p_card)
4131 {
4132         unsigned char message, scsiID;
4133         struct sccb *currSCCB;
4134         struct sccb_mgr_tar_info *currTar_Info;
4135
4136         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4137
4138         if (currSCCB != NULL) {
4139
4140                 message = currSCCB->Sccb_scsimsg;
4141                 scsiID = currSCCB->TargID;
4142
4143                 if (message == TARGET_RESET) {
4144
4145                         currTar_Info = &FPT_sccbMgrTbl[p_card][scsiID];
4146                         currTar_Info->TarSyncCtrl = 0;
4147                         FPT_sssyncv(port, scsiID, NARROW_SCSI, currTar_Info);
4148
4149                         if (FPT_sccbMgrTbl[p_card][scsiID].
4150                             TarEEValue & EE_SYNC_MASK) {
4151
4152                                 FPT_sccbMgrTbl[p_card][scsiID].TarStatus &=
4153                                     ~TAR_SYNC_MASK;
4154
4155                         }
4156
4157                         if (FPT_sccbMgrTbl[p_card][scsiID].
4158                             TarEEValue & EE_WIDE_SCSI) {
4159
4160                                 FPT_sccbMgrTbl[p_card][scsiID].TarStatus &=
4161                                     ~TAR_WIDE_MASK;
4162                         }
4163
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] !=
4169                             NULL) {
4170                                 FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4171                                                               Sccb_tag] = NULL;
4172                                 FPT_sccbMgrTbl[p_card][scsiID].TarTagQ_Cnt--;
4173                         }
4174
4175                 }
4176
4177                 else if (currSCCB->Sccb_scsistat < COMMAND_ST) {
4178
4179                         if (message == NOP) {
4180                                 currSCCB->Sccb_MGRFlags |= F_DEV_SELECTED;
4181
4182                                 FPT_ssel(port, p_card);
4183                                 return;
4184                         }
4185                 } else {
4186
4187                         if (message == ABORT_TASK_SET)
4188
4189                                 FPT_queueFlushSccb(p_card, SCCB_COMPLETE);
4190                 }
4191
4192         } else {
4193                 message = ABORT_TASK_SET;
4194         }
4195
4196         WRW_HARPOON((port + hp_intstat), (BUS_FREE | PHASE | XFER_CNT_0));
4197
4198         WR_HARPOON(port + hp_portctrl_0, SCSI_BUS_EN);
4199
4200         WR_HARPOON(port + hp_scsidata_0, message);
4201
4202         WR_HARPOON(port + hp_scsisig, (SCSI_ACK + S_ILL_PH));
4203
4204         ACCEPT_MSG(port);
4205
4206         WR_HARPOON(port + hp_portctrl_0, 0x00);
4207
4208         if ((message == ABORT_TASK_SET) || (message == TARGET_RESET) ||
4209             (message == ABORT_TASK)) {
4210
4211                 while (!(RDW_HARPOON((port + hp_intstat)) & (BUS_FREE | PHASE))) {
4212                 }
4213
4214                 if (RDW_HARPOON((port + hp_intstat)) & BUS_FREE) {
4215                         WRW_HARPOON((port + hp_intstat), BUS_FREE);
4216
4217                         if (currSCCB != NULL) {
4218
4219                                 if ((FPT_BL_Card[p_card].
4220                                      globalFlags & F_CONLUN_IO)
4221                                     &&
4222                                     ((FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4223                                       TarStatus & TAR_TAG_Q_MASK) !=
4224                                      TAG_Q_TRYING))
4225                                         FPT_sccbMgrTbl[p_card][currSCCB->
4226                                                                TargID].
4227                                             TarLUNBusy[currSCCB->Lun] = 0;
4228                                 else
4229                                         FPT_sccbMgrTbl[p_card][currSCCB->
4230                                                                TargID].
4231                                             TarLUNBusy[0] = 0;
4232
4233                                 FPT_queueCmdComplete(&FPT_BL_Card[p_card],
4234                                                      currSCCB, p_card);
4235                         }
4236
4237                         else {
4238                                 FPT_BL_Card[p_card].globalFlags |=
4239                                     F_NEW_SCCB_CMD;
4240                         }
4241                 }
4242
4243                 else {
4244
4245                         FPT_sxfrp(port, p_card);
4246                 }
4247         }
4248
4249         else {
4250
4251                 if (message == MSG_PARITY_ERROR) {
4252                         currSCCB->Sccb_scsimsg = NOP;
4253                         WR_HARPOON(port + hp_autostart_1,
4254                                    (AUTO_IMMED + DISCONNECT_START));
4255                 } else {
4256                         FPT_sxfrp(port, p_card);
4257                 }
4258         }
4259 }
4260
4261 /*---------------------------------------------------------------------
4262  *
4263  * Function: Message In phase
4264  *
4265  * Description: Bring in the message and determine what to do with it.
4266  *
4267  *---------------------------------------------------------------------*/
4268
4269 static void FPT_phaseMsgIn(u32 port, unsigned char p_card)
4270 {
4271         unsigned char message;
4272         struct sccb *currSCCB;
4273
4274         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4275
4276         if (FPT_BL_Card[p_card].globalFlags & F_HOST_XFER_ACT) {
4277
4278                 FPT_phaseChkFifo(port, p_card);
4279         }
4280
4281         message = RD_HARPOON(port + hp_scsidata_0);
4282         if ((message == DISCONNECT) || (message == SAVE_POINTERS)) {
4283
4284                 WR_HARPOON(port + hp_autostart_1,
4285                            (AUTO_IMMED + END_DATA_START));
4286
4287         }
4288
4289         else {
4290
4291                 message = FPT_sfm(port, currSCCB);
4292                 if (message) {
4293
4294                         FPT_sdecm(message, port, p_card);
4295
4296                 } else {
4297                         if (currSCCB->Sccb_scsimsg != MSG_PARITY_ERROR)
4298                                 ACCEPT_MSG(port);
4299                         WR_HARPOON(port + hp_autostart_1,
4300                                    (AUTO_IMMED + DISCONNECT_START));
4301                 }
4302         }
4303
4304 }
4305
4306 /*---------------------------------------------------------------------
4307  *
4308  * Function: Illegal phase
4309  *
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.
4313  *
4314  *---------------------------------------------------------------------*/
4315
4316 static void FPT_phaseIllegal(u32 port, unsigned char p_card)
4317 {
4318         struct sccb *currSCCB;
4319
4320         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4321
4322         WR_HARPOON(port + hp_scsisig, RD_HARPOON(port + hp_scsisig));
4323         if (currSCCB != NULL) {
4324
4325                 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4326                 currSCCB->Sccb_scsistat = ABORT_ST;
4327                 currSCCB->Sccb_scsimsg = ABORT_TASK_SET;
4328         }
4329
4330         ACCEPT_MSG_ATN(port);
4331 }
4332
4333 /*---------------------------------------------------------------------
4334  *
4335  * Function: Phase Check FIFO
4336  *
4337  * Description: Make sure data has been flushed from both FIFOs and abort
4338  *              the operations if necessary.
4339  *
4340  *---------------------------------------------------------------------*/
4341
4342 static void FPT_phaseChkFifo(u32 port, unsigned char p_card)
4343 {
4344         u32 xfercnt;
4345         struct sccb *currSCCB;
4346
4347         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4348
4349         if (currSCCB->Sccb_scsistat == DATA_IN_ST) {
4350
4351                 while ((!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY)) &&
4352                        (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)) {
4353                 }
4354
4355                 if (!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY)) {
4356                         currSCCB->Sccb_ATC += currSCCB->Sccb_XferCnt;
4357
4358                         currSCCB->Sccb_XferCnt = 0;
4359
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);
4364                         }
4365
4366                         FPT_hostDataXferAbort(port, p_card, currSCCB);
4367
4368                         FPT_dataXferProcessor(port, &FPT_BL_Card[p_card]);
4369
4370                         while ((!(RD_HARPOON(port + hp_xferstat) & FIFO_EMPTY))
4371                                && (RD_HARPOON(port + hp_ext_status) &
4372                                    BM_CMD_BUSY)) {
4373                         }
4374
4375                 }
4376         }
4377
4378         /*End Data In specific code. */
4379         GET_XFER_CNT(port, xfercnt);
4380
4381         WR_HARPOON(port + hp_xfercnt_0, 0x00);
4382
4383         WR_HARPOON(port + hp_portctrl_0, 0x00);
4384
4385         currSCCB->Sccb_ATC += (currSCCB->Sccb_XferCnt - xfercnt);
4386
4387         currSCCB->Sccb_XferCnt = xfercnt;
4388
4389         if ((RDW_HARPOON((port + hp_intstat)) & PARITY) &&
4390             (currSCCB->HostStatus == SCCB_COMPLETE)) {
4391
4392                 currSCCB->HostStatus = SCCB_PARITY_ERR;
4393                 WRW_HARPOON((port + hp_intstat), PARITY);
4394         }
4395
4396         FPT_hostDataXferAbort(port, p_card, currSCCB);
4397
4398         WR_HARPOON(port + hp_fifowrite, 0x00);
4399         WR_HARPOON(port + hp_fiforead, 0x00);
4400         WR_HARPOON(port + hp_xferstat, 0x00);
4401
4402         WRW_HARPOON((port + hp_intstat), XFER_CNT_0);
4403 }
4404
4405 /*---------------------------------------------------------------------
4406  *
4407  * Function: Phase Bus Free
4408  *
4409  * Description: We just went bus free so figure out if it was
4410  *              because of command complete or from a disconnect.
4411  *
4412  *---------------------------------------------------------------------*/
4413 static void FPT_phaseBusFree(u32 port, unsigned char p_card)
4414 {
4415         struct sccb *currSCCB;
4416
4417         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4418
4419         if (currSCCB != NULL) {
4420
4421                 DISABLE_AUTO(port);
4422
4423                 if (currSCCB->OperationCode == RESET_COMMAND) {
4424
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;
4430                         else
4431                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4432                                     TarLUNBusy[0] = 0;
4433
4434                         FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB,
4435                                              p_card);
4436
4437                         FPT_queueSearchSelect(&FPT_BL_Card[p_card], p_card);
4438
4439                 }
4440
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 &=
4445                             ~EE_SYNC_MASK;
4446                 }
4447
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;
4452
4453                         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4454                             ~EE_WIDE_SCSI;
4455                 }
4456
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.     */
4461
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;
4468                         }
4469
4470                         else {
4471                                 return;
4472                         }
4473                 }
4474
4475                 else {
4476
4477                         currSCCB->Sccb_scsistat = BUS_FREE_ST;
4478
4479                         if (!currSCCB->HostStatus) {
4480                                 currSCCB->HostStatus = SCCB_PHASE_SEQUENCE_FAIL;
4481                         }
4482
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;
4488                         else
4489                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4490                                     TarLUNBusy[0] = 0;
4491
4492                         FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB,
4493                                              p_card);
4494                         return;
4495                 }
4496
4497                 FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4498
4499         }                       /*end if !=null */
4500 }
4501
4502 /*---------------------------------------------------------------------
4503  *
4504  * Function: Auto Load Default Map
4505  *
4506  * Description: Load the Automation RAM with the default map values.
4507  *
4508  *---------------------------------------------------------------------*/
4509 static void FPT_autoLoadDefaultMap(u32 p_port)
4510 {
4511         u32 map_addr;
4512
4513         ARAM_ACCESS(p_port);
4514         map_addr = p_port + hp_aramBase;
4515
4516         WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0xC0));      /*ID MESSAGE */
4517         map_addr += 2;
4518         WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0x20));      /*SIMPLE TAG QUEUEING MSG */
4519         map_addr += 2;
4520         WRW_HARPOON(map_addr, RAT_OP);  /*RESET ATTENTION */
4521         map_addr += 2;
4522         WRW_HARPOON(map_addr, (MPM_OP + AMSG_OUT + 0x00));      /*TAG ID MSG */
4523         map_addr += 2;
4524         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 0 */
4525         map_addr += 2;
4526         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 1 */
4527         map_addr += 2;
4528         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 2 */
4529         map_addr += 2;
4530         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 3 */
4531         map_addr += 2;
4532         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 4 */
4533         map_addr += 2;
4534         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 5 */
4535         map_addr += 2;
4536         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 6 */
4537         map_addr += 2;
4538         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 7 */
4539         map_addr += 2;
4540         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 8 */
4541         map_addr += 2;
4542         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 9 */
4543         map_addr += 2;
4544         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 10 */
4545         map_addr += 2;
4546         WRW_HARPOON(map_addr, (MPM_OP + ACOMMAND + 0x00));      /*CDB BYTE 11 */
4547         map_addr += 2;
4548         WRW_HARPOON(map_addr, (CPE_OP + ADATA_OUT + DINT));     /*JUMP IF DATA OUT */
4549         map_addr += 2;
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 */
4553         map_addr += 2;
4554         WRW_HARPOON(map_addr, (CPE_OP + ADATA_IN + DINT));      /*JUMP IF NOT DATA IN PHZ */
4555         map_addr += 2;
4556         WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + ST)); /*IF NOT MSG IN CHECK 4 DATA IN */
4557         map_addr += 2;
4558         WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x02)); /*SAVE DATA PTR MSG? */
4559         map_addr += 2;
4560         WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + DC));  /*GO CHECK FOR DISCONNECT MSG */
4561         map_addr += 2;
4562         WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_AR1));        /*SAVE DATA PTRS MSG */
4563         map_addr += 2;
4564         WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + ST)); /*IF NOT MSG IN CHECK DATA IN */
4565         map_addr += 2;
4566         WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x04)); /*DISCONNECT MSG? */
4567         map_addr += 2;
4568         WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + UNKNWN));      /*UKNKNOWN MSG */
4569         map_addr += 2;
4570         WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_BUCKET));     /*XFER DISCONNECT MSG */
4571         map_addr += 2;
4572         WRW_HARPOON(map_addr, (SSI_OP + SSI_ITAR_DISC));        /*STOP AND INTERRUPT */
4573         map_addr += 2;
4574         WRW_HARPOON(map_addr, (CPN_OP + ASTATUS + UNKNWN));     /*JUMP IF NOT STATUS PHZ. */
4575         map_addr += 2;
4576         WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_AR0));        /*GET STATUS BYTE */
4577         map_addr += 2;
4578         WRW_HARPOON(map_addr, (CPN_OP + AMSG_IN + CC)); /*ERROR IF NOT MSG IN PHZ */
4579         map_addr += 2;
4580         WRW_HARPOON(map_addr, (CRD_OP + SDATA + 0x00)); /*CHECK FOR CMD COMPLETE MSG. */
4581         map_addr += 2;
4582         WRW_HARPOON(map_addr, (BRH_OP + NOT_EQ + CC));  /*ERROR IF NOT CMD COMPLETE MSG. */
4583         map_addr += 2;
4584         WRW_HARPOON(map_addr, (MRR_OP + SDATA + D_BUCKET));     /*GET CMD COMPLETE MSG */
4585         map_addr += 2;
4586         WRW_HARPOON(map_addr, (SSI_OP + SSI_ICMD_COMP));        /*END OF COMMAND */
4587         map_addr += 2;
4588
4589         WRW_HARPOON(map_addr, (SSI_OP + SSI_IUNKWN));   /*RECEIVED UNKNOWN MSG BYTE */
4590         map_addr += 2;
4591         WRW_HARPOON(map_addr, (SSI_OP + SSI_INO_CC));   /*NO COMMAND COMPLETE AFTER STATUS */
4592         map_addr += 2;
4593         WRW_HARPOON(map_addr, (SSI_OP + SSI_ITICKLE));  /*BIOS Tickled the Mgr */
4594         map_addr += 2;
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 */
4598         map_addr += 2;
4599         WRW_HARPOON(map_addr, (BRH_OP + EQUAL + 0x00)); /*SEL ID OK then Conti. */
4600         map_addr += 2;
4601         WRW_HARPOON(map_addr, (SSI_OP + SSI_INO_CC));   /*NO COMMAND COMPLETE AFTER STATUS */
4602
4603         SGRAM_ACCESS(p_port);
4604 }
4605
4606 /*---------------------------------------------------------------------
4607  *
4608  * Function: Auto Command Complete
4609  *
4610  * Description: Post command back to host and find another command
4611  *              to execute.
4612  *
4613  *---------------------------------------------------------------------*/
4614
4615 static void FPT_autoCmdCmplt(u32 p_port, unsigned char p_card)
4616 {
4617         struct sccb *currSCCB;
4618         unsigned char status_byte;
4619
4620         currSCCB = FPT_BL_Card[p_card].currentSCCB;
4621
4622         status_byte = RD_HARPOON(p_port + hp_gp_reg_0);
4623
4624         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarLUN_CA = 0;
4625
4626         if (status_byte != SAM_STAT_GOOD) {
4627
4628                 if (status_byte == SAM_STAT_TASK_SET_FULL) {
4629
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]
4639                                               [currSCCB->TargID].
4640                                               LunDiscQ_Idx[currSCCB->Lun]] =
4641                                     NULL;
4642                         } else {
4643                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4644                                     TarLUNBusy[0] = 1;
4645                                 if (currSCCB->Sccb_tag) {
4646                                         if (FPT_BL_Card[p_card].discQCount != 0)
4647                                                 FPT_BL_Card[p_card].
4648                                                     discQCount--;
4649                                         FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4650                                                                       Sccb_tag]
4651                                             = NULL;
4652                                 } else {
4653                                         if (FPT_BL_Card[p_card].discQCount != 0)
4654                                                 FPT_BL_Card[p_card].
4655                                                     discQCount--;
4656                                         FPT_BL_Card[p_card].
4657                                             discQ_Tbl[FPT_sccbMgrTbl[p_card]
4658                                                       [currSCCB->TargID].
4659                                                       LunDiscQ_Idx[0]] = NULL;
4660                                 }
4661                         }
4662
4663                         currSCCB->Sccb_MGRFlags |= F_STATUSLOADED;
4664
4665                         FPT_queueSelectFail(&FPT_BL_Card[p_card], p_card);
4666
4667                         return;
4668                 }
4669
4670                 if (currSCCB->Sccb_scsistat == SELECT_SN_ST) {
4671                         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus |=
4672                             (unsigned char)SYNC_SUPPORTED;
4673
4674                         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4675                             ~EE_SYNC_MASK;
4676                         FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4677
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]
4687                                               [currSCCB->TargID].
4688                                               LunDiscQ_Idx[currSCCB->Lun]] =
4689                                     NULL;
4690                         } else {
4691                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4692                                     TarLUNBusy[0] = 1;
4693                                 if (currSCCB->Sccb_tag) {
4694                                         if (FPT_BL_Card[p_card].discQCount != 0)
4695                                                 FPT_BL_Card[p_card].
4696                                                     discQCount--;
4697                                         FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4698                                                                       Sccb_tag]
4699                                             = NULL;
4700                                 } else {
4701                                         if (FPT_BL_Card[p_card].discQCount != 0)
4702                                                 FPT_BL_Card[p_card].
4703                                                     discQCount--;
4704                                         FPT_BL_Card[p_card].
4705                                             discQ_Tbl[FPT_sccbMgrTbl[p_card]
4706                                                       [currSCCB->TargID].
4707                                                       LunDiscQ_Idx[0]] = NULL;
4708                                 }
4709                         }
4710                         return;
4711
4712                 }
4713
4714                 if (currSCCB->Sccb_scsistat == SELECT_WN_ST) {
4715
4716                         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarStatus =
4717                             (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4718                              TarStatus & ~WIDE_ENABLED) | WIDE_NEGOCIATED;
4719
4720                         FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarEEValue &=
4721                             ~EE_WIDE_SCSI;
4722                         FPT_BL_Card[p_card].globalFlags |= F_NEW_SCCB_CMD;
4723
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]
4733                                               [currSCCB->TargID].
4734                                               LunDiscQ_Idx[currSCCB->Lun]] =
4735                                     NULL;
4736                         } else {
4737                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4738                                     TarLUNBusy[0] = 1;
4739                                 if (currSCCB->Sccb_tag) {
4740                                         if (FPT_BL_Card[p_card].discQCount != 0)
4741                                                 FPT_BL_Card[p_card].
4742                                                     discQCount--;
4743                                         FPT_BL_Card[p_card].discQ_Tbl[currSCCB->
4744                                                                       Sccb_tag]
4745                                             = NULL;
4746                                 } else {
4747                                         if (FPT_BL_Card[p_card].discQCount != 0)
4748                                                 FPT_BL_Card[p_card].
4749                                                     discQCount--;
4750                                         FPT_BL_Card[p_card].
4751                                             discQ_Tbl[FPT_sccbMgrTbl[p_card]
4752                                                       [currSCCB->TargID].
4753                                                       LunDiscQ_Idx[0]] = NULL;
4754                                 }
4755                         }
4756                         return;
4757
4758                 }
4759
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->
4765                                                                TargID].
4766                                             TarStatus &= ~TAR_SYNC_MASK;
4767                                 }
4768                                 if (FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4769                                     TarEEValue & EE_WIDE_SCSI) {
4770                                         FPT_sccbMgrTbl[p_card][currSCCB->
4771                                                                TargID].
4772                                             TarStatus &= ~TAR_WIDE_MASK;
4773                                 }
4774                         }
4775                 }
4776
4777                 if (!(currSCCB->Sccb_XferState & F_AUTO_SENSE)) {
4778
4779                         currSCCB->SccbStatus = SCCB_ERROR;
4780                         currSCCB->TargetStatus = status_byte;
4781
4782                         if (status_byte == SAM_STAT_CHECK_CONDITION) {
4783
4784                                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].
4785                                     TarLUN_CA = 1;
4786
4787                                 if (currSCCB->RequestSenseLength !=
4788                                     NO_AUTO_REQUEST_SENSE) {
4789
4790                                         if (currSCCB->RequestSenseLength == 0)
4791                                                 currSCCB->RequestSenseLength =
4792                                                     14;
4793
4794                                         FPT_ssenss(&FPT_BL_Card[p_card]);
4795                                         FPT_BL_Card[p_card].globalFlags |=
4796                                             F_NEW_SCCB_CMD;
4797
4798                                         if (((FPT_BL_Card[p_card].
4799                                               globalFlags & F_CONLUN_IO)
4800                                              &&
4801                                              ((FPT_sccbMgrTbl[p_card]
4802                                                [currSCCB->TargID].
4803                                                TarStatus & TAR_TAG_Q_MASK) !=
4804                                               TAG_Q_TRYING))) {
4805                                                 FPT_sccbMgrTbl[p_card]
4806                                                     [currSCCB->TargID].
4807                                                     TarLUNBusy[currSCCB->Lun] =
4808                                                     1;
4809                                                 if (FPT_BL_Card[p_card].
4810                                                     discQCount != 0)
4811                                                         FPT_BL_Card[p_card].
4812                                                             discQCount--;
4813                                                 FPT_BL_Card[p_card].
4814                                                     discQ_Tbl[FPT_sccbMgrTbl
4815                                                               [p_card]
4816                                                               [currSCCB->
4817                                                                TargID].
4818                                                               LunDiscQ_Idx
4819                                                               [currSCCB->Lun]] =
4820                                                     NULL;
4821                                         } else {
4822                                                 FPT_sccbMgrTbl[p_card]
4823                                                     [currSCCB->TargID].
4824                                                     TarLUNBusy[0] = 1;
4825                                                 if (currSCCB->Sccb_tag) {
4826                                                         if (FPT_BL_Card[p_card].
4827                                                             discQCount != 0)
4828                                                                 FPT_BL_Card
4829                                                                     [p_card].
4830                                                                     discQCount--;
4831                                                         FPT_BL_Card[p_card].
4832                                                             discQ_Tbl[currSCCB->
4833                                                                       Sccb_tag]
4834                                                             = NULL;
4835                                                 } else {
4836                                                         if (FPT_BL_Card[p_card].
4837                                                             discQCount != 0)
4838                                                                 FPT_BL_Card
4839                                                                     [p_card].
4840                                                                     discQCount--;
4841                                                         FPT_BL_Card[p_card].
4842                                                             discQ_Tbl
4843                                                             [FPT_sccbMgrTbl
4844                                                              [p_card][currSCCB->
4845                                                                       TargID].
4846                                                              LunDiscQ_Idx[0]] =
4847                                                             NULL;
4848                                                 }
4849                                         }
4850                                         return;
4851                                 }
4852                         }
4853                 }
4854         }
4855
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->
4860                                                                     Lun] = 0;
4861         else
4862                 FPT_sccbMgrTbl[p_card][currSCCB->TargID].TarLUNBusy[0] = 0;
4863
4864         FPT_queueCmdComplete(&FPT_BL_Card[p_card], currSCCB, p_card);
4865 }
4866
4867 #define SHORT_WAIT   0x0000000F
4868 #define LONG_WAIT    0x0000FFFFL
4869
4870 /*---------------------------------------------------------------------
4871  *
4872  * Function: Data Transfer Processor
4873  *
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.
4885  *              
4886  *---------------------------------------------------------------------*/
4887
4888 static void FPT_dataXferProcessor(u32 port, struct sccb_card *pCurrCard)
4889 {
4890         struct sccb *currSCCB;
4891
4892         currSCCB = pCurrCard->currentSCCB;
4893
4894         if (currSCCB->Sccb_XferState & F_SG_XFER) {
4895                 if (pCurrCard->globalFlags & F_HOST_XFER_ACT)
4896                 {
4897                         currSCCB->Sccb_sgseg += (unsigned char)SG_BUF_CNT;
4898                         currSCCB->Sccb_SGoffset = 0x00;
4899                 }
4900                 pCurrCard->globalFlags |= F_HOST_XFER_ACT;
4901
4902                 FPT_busMstrSGDataXferStart(port, currSCCB);
4903         }
4904
4905         else {
4906                 if (!(pCurrCard->globalFlags & F_HOST_XFER_ACT)) {
4907                         pCurrCard->globalFlags |= F_HOST_XFER_ACT;
4908
4909                         FPT_busMstrDataXferStart(port, currSCCB);
4910                 }
4911         }
4912 }
4913
4914 /*---------------------------------------------------------------------
4915  *
4916  * Function: BusMaster Scatter Gather Data Transfer Start
4917  *
4918  * Description:
4919  *
4920  *---------------------------------------------------------------------*/
4921 static void FPT_busMstrSGDataXferStart(u32 p_port, struct sccb *pcurrSCCB)
4922 {
4923         u32 count, addr, tmpSGCnt;
4924         unsigned int sg_index;
4925         unsigned char sg_count, i;
4926         u32 reg_offset;
4927         struct blogic_sg_seg *segp;
4928
4929         if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR)
4930                 count = ((u32)HOST_RD_CMD) << 24;
4931         else
4932                 count = ((u32)HOST_WRT_CMD) << 24;
4933
4934         sg_count = 0;
4935         tmpSGCnt = 0;
4936         sg_index = pcurrSCCB->Sccb_sgseg;
4937         reg_offset = hp_aramBase;
4938
4939         i = (unsigned char)(RD_HARPOON(p_port + hp_page_ctrl) &
4940                             ~(SGRAM_ARAM | SCATTER_EN));
4941
4942         WR_HARPOON(p_port + hp_page_ctrl, i);
4943
4944         while ((sg_count < (unsigned char)SG_BUF_CNT) &&
4945                         ((sg_index * (unsigned int)SG_ELEMENT_SIZE) <
4946                         pcurrSCCB->DataLength)) {
4947
4948                 segp = (struct blogic_sg_seg *)(pcurrSCCB->DataPointer) +
4949                                 sg_index;
4950                 tmpSGCnt += segp->segbytes;
4951                 count |= segp->segbytes;
4952                 addr = segp->segdata;
4953
4954                 if ((!sg_count) && (pcurrSCCB->Sccb_SGoffset)) {
4955                         addr +=
4956                             ((count & 0x00FFFFFFL) - pcurrSCCB->Sccb_SGoffset);
4957                         count =
4958                             (count & 0xFF000000L) | pcurrSCCB->Sccb_SGoffset;
4959                         tmpSGCnt = count & 0x00FFFFFFL;
4960                 }
4961
4962                 WR_HARP32(p_port, reg_offset, addr);
4963                 reg_offset += 4;
4964
4965                 WR_HARP32(p_port, reg_offset, count);
4966                 reg_offset += 4;
4967
4968                 count &= 0xFF000000L;
4969                 sg_index++;
4970                 sg_count++;
4971
4972         }                       /*End While */
4973
4974         pcurrSCCB->Sccb_XferCnt = tmpSGCnt;
4975
4976         WR_HARPOON(p_port + hp_sg_addr, (sg_count << 4));
4977
4978         if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
4979
4980                 WR_HARP32(p_port, hp_xfercnt_0, tmpSGCnt);
4981
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);
4985         }
4986
4987         else {
4988
4989                 if ((!(RD_HARPOON(p_port + hp_synctarg_0) & NARROW_SCSI)) &&
4990                     (tmpSGCnt & 0x000000001)) {
4991
4992                         pcurrSCCB->Sccb_XferState |= F_ODD_BALL_CNT;
4993                         tmpSGCnt--;
4994                 }
4995
4996                 WR_HARP32(p_port, hp_xfercnt_0, tmpSGCnt);
4997
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);
5001         }
5002
5003         WR_HARPOON(p_port + hp_page_ctrl, (unsigned char)(i | SCATTER_EN));
5004
5005 }
5006
5007 /*---------------------------------------------------------------------
5008  *
5009  * Function: BusMaster Data Transfer Start
5010  *
5011  * Description: 
5012  *
5013  *---------------------------------------------------------------------*/
5014 static void FPT_busMstrDataXferStart(u32 p_port, struct sccb *pcurrSCCB)
5015 {
5016         u32 addr, count;
5017
5018         if (!(pcurrSCCB->Sccb_XferState & F_AUTO_SENSE)) {
5019
5020                 count = pcurrSCCB->Sccb_XferCnt;
5021
5022                 addr = (u32)(unsigned long)pcurrSCCB->DataPointer + pcurrSCCB->Sccb_ATC;
5023         }
5024
5025         else {
5026                 addr = pcurrSCCB->SensePointer;
5027                 count = pcurrSCCB->RequestSenseLength;
5028
5029         }
5030
5031         HP_SETUP_ADDR_CNT(p_port, addr, count);
5032
5033         if (pcurrSCCB->Sccb_XferState & F_HOST_XFER_DIR) {
5034
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);
5038
5039                 WR_HARPOON(p_port + hp_xfer_cmd,
5040                            (XFER_DMA_HOST | XFER_HOST_AUTO | XFER_DMA_8BIT));
5041         }
5042
5043         else {
5044
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);
5048
5049                 WR_HARPOON(p_port + hp_xfer_cmd,
5050                            (XFER_HOST_DMA | XFER_HOST_AUTO | XFER_DMA_8BIT));
5051
5052         }
5053 }
5054
5055 /*---------------------------------------------------------------------
5056  *
5057  * Function: BusMaster Timeout Handler
5058  *
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.
5065  *
5066  *---------------------------------------------------------------------*/
5067 static unsigned char FPT_busMstrTimeOut(u32 p_port)
5068 {
5069         unsigned long timeout;
5070
5071         timeout = LONG_WAIT;
5072
5073         WR_HARPOON(p_port + hp_sys_ctrl, HALT_MACH);
5074
5075         while ((!(RD_HARPOON(p_port + hp_ext_status) & CMD_ABORTED))
5076                && timeout--) {
5077         }
5078
5079         if (RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY) {
5080                 WR_HARPOON(p_port + hp_sys_ctrl, HARD_ABORT);
5081
5082                 timeout = LONG_WAIT;
5083                 while ((RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY)
5084                        && timeout--) {
5085                 }
5086         }
5087
5088         RD_HARPOON(p_port + hp_int_status);     /*Clear command complete */
5089
5090         if (RD_HARPOON(p_port + hp_ext_status) & BM_CMD_BUSY) {
5091                 return 1;
5092         }
5093
5094         else {
5095                 return 0;
5096         }
5097 }
5098
5099 /*---------------------------------------------------------------------
5100  *
5101  * Function: Host Data Transfer Abort
5102  *
5103  * Description: Abort any in progress transfer.
5104  *
5105  *---------------------------------------------------------------------*/
5106 static void FPT_hostDataXferAbort(u32 port, unsigned char p_card,
5107                                   struct sccb *pCurrSCCB)
5108 {
5109
5110         unsigned long timeout;
5111         unsigned long remain_cnt;
5112         u32 sg_ptr;
5113         struct blogic_sg_seg *segp;
5114
5115         FPT_BL_Card[p_card].globalFlags &= ~F_HOST_XFER_ACT;
5116
5117         if (pCurrSCCB->Sccb_XferState & F_AUTO_SENSE) {
5118
5119                 if (!(RD_HARPOON(port + hp_int_status) & INT_CMD_COMPL)) {
5120
5121                         WR_HARPOON(port + hp_bm_ctrl,
5122                                    (RD_HARPOON(port + hp_bm_ctrl) |
5123                                     FLUSH_XFER_CNTR));
5124                         timeout = LONG_WAIT;
5125
5126                         while ((RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)
5127                                && timeout--) {
5128                         }
5129
5130                         WR_HARPOON(port + hp_bm_ctrl,
5131                                    (RD_HARPOON(port + hp_bm_ctrl) &
5132                                     ~FLUSH_XFER_CNTR));
5133
5134                         if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5135
5136                                 if (FPT_busMstrTimeOut(port)) {
5137
5138                                         if (pCurrSCCB->HostStatus == 0x00)
5139
5140                                                 pCurrSCCB->HostStatus =
5141                                                     SCCB_BM_ERR;
5142
5143                                 }
5144
5145                                 if (RD_HARPOON(port + hp_int_status) &
5146                                     INT_EXT_STATUS)
5147
5148                                         if (RD_HARPOON(port + hp_ext_status) &
5149                                             BAD_EXT_STATUS)
5150
5151                                                 if (pCurrSCCB->HostStatus ==
5152                                                     0x00)
5153                                                 {
5154                                                         pCurrSCCB->HostStatus =
5155                                                             SCCB_BM_ERR;
5156                                                 }
5157                         }
5158                 }
5159         }
5160
5161         else if (pCurrSCCB->Sccb_XferCnt) {
5162
5163                 if (pCurrSCCB->Sccb_XferState & F_SG_XFER) {
5164
5165                         WR_HARPOON(port + hp_page_ctrl,
5166                                    (RD_HARPOON(port + hp_page_ctrl) &
5167                                     ~SCATTER_EN));
5168
5169                         WR_HARPOON(port + hp_sg_addr, 0x00);
5170
5171                         sg_ptr = pCurrSCCB->Sccb_sgseg + SG_BUF_CNT;
5172
5173                         if (sg_ptr >
5174                             (unsigned int)(pCurrSCCB->DataLength /
5175                                            SG_ELEMENT_SIZE)) {
5176
5177                                 sg_ptr = (u32)(pCurrSCCB->DataLength /
5178                                                         SG_ELEMENT_SIZE);
5179                         }
5180
5181                         remain_cnt = pCurrSCCB->Sccb_XferCnt;
5182
5183                         while (remain_cnt < 0x01000000L) {
5184
5185                                 sg_ptr--;
5186                                 segp = (struct blogic_sg_seg *)(pCurrSCCB->
5187                                                 DataPointer) + (sg_ptr * 2);
5188                                 if (remain_cnt > (unsigned long)segp->segbytes)
5189                                         remain_cnt -=
5190                                                 (unsigned long)segp->segbytes;
5191                                 else
5192                                         break;
5193                         }
5194
5195                         if (remain_cnt < 0x01000000L) {
5196
5197                                 pCurrSCCB->Sccb_SGoffset = remain_cnt;
5198
5199                                 pCurrSCCB->Sccb_sgseg = (unsigned short)sg_ptr;
5200
5201                                 if ((unsigned long)(sg_ptr * SG_ELEMENT_SIZE) ==
5202                                     pCurrSCCB->DataLength && (remain_cnt == 0))
5203
5204                                         pCurrSCCB->Sccb_XferState |=
5205                                             F_ALL_XFERRED;
5206                         }
5207
5208                         else {
5209
5210                                 if (pCurrSCCB->HostStatus == 0x00) {
5211
5212                                         pCurrSCCB->HostStatus =
5213                                             SCCB_GROSS_FW_ERR;
5214                                 }
5215                         }
5216                 }
5217
5218                 if (!(pCurrSCCB->Sccb_XferState & F_HOST_XFER_DIR)) {
5219
5220                         if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5221
5222                                 FPT_busMstrTimeOut(port);
5223                         }
5224
5225                         else {
5226
5227                                 if (RD_HARPOON(port + hp_int_status) &
5228                                     INT_EXT_STATUS) {
5229
5230                                         if (RD_HARPOON(port + hp_ext_status) &
5231                                             BAD_EXT_STATUS) {
5232
5233                                                 if (pCurrSCCB->HostStatus ==
5234                                                     0x00) {
5235
5236                                                         pCurrSCCB->HostStatus =
5237                                                             SCCB_BM_ERR;
5238                                                 }
5239                                         }
5240                                 }
5241
5242                         }
5243                 }
5244
5245                 else {
5246
5247                         if ((RD_HARPOON(port + hp_fifo_cnt)) >= BM_THRESHOLD) {
5248
5249                                 timeout = SHORT_WAIT;
5250
5251                                 while ((RD_HARPOON(port + hp_ext_status) &
5252                                         BM_CMD_BUSY)
5253                                        && ((RD_HARPOON(port + hp_fifo_cnt)) >=
5254                                            BM_THRESHOLD) && timeout--) {
5255                                 }
5256                         }
5257
5258                         if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5259
5260                                 WR_HARPOON(port + hp_bm_ctrl,
5261                                            (RD_HARPOON(port + hp_bm_ctrl) |
5262                                             FLUSH_XFER_CNTR));
5263
5264                                 timeout = LONG_WAIT;
5265
5266                                 while ((RD_HARPOON(port + hp_ext_status) &
5267                                         BM_CMD_BUSY) && timeout--) {
5268                                 }
5269
5270                                 WR_HARPOON(port + hp_bm_ctrl,
5271                                            (RD_HARPOON(port + hp_bm_ctrl) &
5272                                             ~FLUSH_XFER_CNTR));
5273
5274                                 if (RD_HARPOON(port + hp_ext_status) &
5275                                     BM_CMD_BUSY) {
5276
5277                                         if (pCurrSCCB->HostStatus == 0x00) {
5278
5279                                                 pCurrSCCB->HostStatus =
5280                                                     SCCB_BM_ERR;
5281                                         }
5282
5283                                         FPT_busMstrTimeOut(port);
5284                                 }
5285                         }
5286
5287                         if (RD_HARPOON(port + hp_int_status) & INT_EXT_STATUS) {
5288
5289                                 if (RD_HARPOON(port + hp_ext_status) &
5290                                     BAD_EXT_STATUS) {
5291
5292                                         if (pCurrSCCB->HostStatus == 0x00) {
5293
5294                                                 pCurrSCCB->HostStatus =
5295                                                     SCCB_BM_ERR;
5296                                         }
5297                                 }
5298                         }
5299                 }
5300
5301         }
5302
5303         else {
5304
5305                 if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5306
5307                         timeout = LONG_WAIT;
5308
5309                         while ((RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY)
5310                                && timeout--) {
5311                         }
5312
5313                         if (RD_HARPOON(port + hp_ext_status) & BM_CMD_BUSY) {
5314
5315                                 if (pCurrSCCB->HostStatus == 0x00) {
5316
5317                                         pCurrSCCB->HostStatus = SCCB_BM_ERR;
5318                                 }
5319
5320                                 FPT_busMstrTimeOut(port);
5321                         }
5322                 }
5323
5324                 if (RD_HARPOON(port + hp_int_status) & INT_EXT_STATUS) {
5325
5326                         if (RD_HARPOON(port + hp_ext_status) & BAD_EXT_STATUS) {
5327
5328                                 if (pCurrSCCB->HostStatus == 0x00) {
5329
5330                                         pCurrSCCB->HostStatus = SCCB_BM_ERR;
5331                                 }
5332                         }
5333
5334                 }
5335
5336                 if (pCurrSCCB->Sccb_XferState & F_SG_XFER) {
5337
5338                         WR_HARPOON(port + hp_page_ctrl,
5339                                    (RD_HARPOON(port + hp_page_ctrl) &
5340                                     ~SCATTER_EN));
5341
5342                         WR_HARPOON(port + hp_sg_addr, 0x00);
5343
5344                         pCurrSCCB->Sccb_sgseg += SG_BUF_CNT;
5345
5346                         pCurrSCCB->Sccb_SGoffset = 0x00;
5347
5348                         if ((u32)(pCurrSCCB->Sccb_sgseg * SG_ELEMENT_SIZE) >=
5349                                         pCurrSCCB->DataLength) {
5350
5351                                 pCurrSCCB->Sccb_XferState |= F_ALL_XFERRED;
5352                                 pCurrSCCB->Sccb_sgseg =
5353                                     (unsigned short)(pCurrSCCB->DataLength /
5354                                                      SG_ELEMENT_SIZE);
5355                         }
5356                 }
5357
5358                 else {
5359                         if (!(pCurrSCCB->Sccb_XferState & F_AUTO_SENSE))
5360                                 pCurrSCCB->Sccb_XferState |= F_ALL_XFERRED;
5361                 }
5362         }
5363
5364         WR_HARPOON(port + hp_int_mask, (INT_CMD_COMPL | SCSI_INTERRUPT));
5365 }
5366
5367 /*---------------------------------------------------------------------
5368  *
5369  * Function: Host Data Transfer Restart
5370  *
5371  * Description: Reset the available count due to a restore data
5372  *              pointers message.
5373  *
5374  *---------------------------------------------------------------------*/
5375 static void FPT_hostDataXferRestart(struct sccb *currSCCB)
5376 {
5377         unsigned long data_count;
5378         unsigned int sg_index;
5379         struct blogic_sg_seg *segp;
5380
5381         if (currSCCB->Sccb_XferState & F_SG_XFER) {
5382
5383                 currSCCB->Sccb_XferCnt = 0;
5384
5385                 sg_index = 0xffff;      /*Index by long words into sg list. */
5386                 data_count = 0;         /*Running count of SG xfer counts. */
5387
5388
5389                 while (data_count < currSCCB->Sccb_ATC) {
5390
5391                         sg_index++;
5392                         segp = (struct blogic_sg_seg *)(currSCCB->DataPointer) +
5393                                                 (sg_index * 2);
5394                         data_count += segp->segbytes;
5395                 }
5396
5397                 if (data_count == currSCCB->Sccb_ATC) {
5398
5399                         currSCCB->Sccb_SGoffset = 0;
5400                         sg_index++;
5401                 }
5402
5403                 else {
5404                         currSCCB->Sccb_SGoffset =
5405                             data_count - currSCCB->Sccb_ATC;
5406                 }
5407
5408                 currSCCB->Sccb_sgseg = (unsigned short)sg_index;
5409         }
5410
5411         else {
5412                 currSCCB->Sccb_XferCnt =
5413                     currSCCB->DataLength - currSCCB->Sccb_ATC;
5414         }
5415 }
5416
5417 /*---------------------------------------------------------------------
5418  *
5419  * Function: FPT_scini
5420  *
5421  * Description: Setup all data structures necessary for SCAM selection.
5422  *
5423  *---------------------------------------------------------------------*/
5424
5425 static void FPT_scini(unsigned char p_card, unsigned char p_our_id,
5426                       unsigned char p_power_up)
5427 {
5428
5429         unsigned char loser, assigned_id;
5430         u32 p_port;
5431
5432         unsigned char i, k, ScamFlg;
5433         struct sccb_card *currCard;
5434         struct nvram_info *pCurrNvRam;
5435
5436         currCard = &FPT_BL_Card[p_card];
5437         p_port = currCard->ioPort;
5438         pCurrNvRam = currCard->pNvRamInfo;
5439
5440         if (pCurrNvRam) {
5441                 ScamFlg = pCurrNvRam->niScamConf;
5442                 i = pCurrNvRam->niSysConf;
5443         } else {
5444                 ScamFlg =
5445                     (unsigned char)FPT_utilEERead(p_port, SCAM_CONFIG / 2);
5446                 i = (unsigned
5447                      char)(FPT_utilEERead(p_port, (SYSTEM_CONFIG / 2)));
5448         }
5449         if (!(i & 0x02))        /* check if reset bus in AutoSCSI parameter set */
5450                 return;
5451
5452         FPT_inisci(p_card, p_port, p_our_id);
5453
5454         /* Force to wait 1 sec after SCSI bus reset. Some SCAM device FW
5455            too slow to return to SCAM selection */
5456
5457         /* if (p_power_up)
5458            FPT_Wait1Second(p_port);
5459            else
5460            FPT_Wait(p_port, TO_250ms); */
5461
5462         FPT_Wait1Second(p_port);
5463
5464         if ((ScamFlg & SCAM_ENABLED) && (ScamFlg & SCAM_LEVEL2)) {
5465                 while (!(FPT_scarb(p_port, INIT_SELTD))) {
5466                 }
5467
5468                 FPT_scsel(p_port);
5469
5470                 do {
5471                         FPT_scxferc(p_port, SYNC_PTRN);
5472                         FPT_scxferc(p_port, DOM_MSTR);
5473                         loser =
5474                             FPT_scsendi(p_port,
5475                                         &FPT_scamInfo[p_our_id].id_string[0]);
5476                 } while (loser == 0xFF);
5477
5478                 FPT_scbusf(p_port);
5479
5480                 if ((p_power_up) && (!loser)) {
5481                         FPT_sresb(p_port, p_card);
5482                         FPT_Wait(p_port, TO_250ms);
5483
5484                         while (!(FPT_scarb(p_port, INIT_SELTD))) {
5485                         }
5486
5487                         FPT_scsel(p_port);
5488
5489                         do {
5490                                 FPT_scxferc(p_port, SYNC_PTRN);
5491                                 FPT_scxferc(p_port, DOM_MSTR);
5492                                 loser =
5493                                     FPT_scsendi(p_port,
5494                                                 &FPT_scamInfo[p_our_id].
5495                                                 id_string[0]);
5496                         } while (loser == 0xFF);
5497
5498                         FPT_scbusf(p_port);
5499                 }
5500         }
5501
5502         else {
5503                 loser = 0;
5504         }
5505
5506         if (!loser) {
5507
5508                 FPT_scamInfo[p_our_id].state = ID_ASSIGNED;
5509
5510                 if (ScamFlg & SCAM_ENABLED) {
5511
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)) {
5521
5522                                                         FPT_scamInfo[i].
5523                                                             id_string[0] = 0xFF;
5524                                                         FPT_scamInfo[i].
5525                                                             id_string[1] = 0xFA;
5526                                                         if (pCurrNvRam == NULL)
5527                                                                 currCard->
5528                                                                     globalFlags
5529                                                                     |=
5530                                                                     F_UPDATE_EEPROM;
5531                                                 }
5532                                         }
5533                                 }
5534                         }
5535
5536                         FPT_sresb(p_port, p_card);
5537                         FPT_Wait1Second(p_port);
5538                         while (!(FPT_scarb(p_port, INIT_SELTD))) {
5539                         }
5540                         FPT_scsel(p_port);
5541                         FPT_scasid(p_card, p_port);
5542                 }
5543
5544         }
5545
5546         else if ((loser) && (ScamFlg & SCAM_ENABLED)) {
5547                 FPT_scamInfo[p_our_id].id_string[0] = SLV_TYPE_CODE0;
5548                 assigned_id = 0;
5549                 FPT_scwtsel(p_port);
5550
5551                 do {
5552                         while (FPT_scxferc(p_port, 0x00) != SYNC_PTRN) {
5553                         }
5554
5555                         i = FPT_scxferc(p_port, 0x00);
5556                         if (i == ASSIGN_ID) {
5557                                 if (!
5558                                     (FPT_scsendi
5559                                      (p_port,
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);
5564
5565                                                 if (FPT_scvalq(k)) {
5566                                                         currCard->ourId =
5567                                                             ((unsigned char)(i
5568                                                                              <<
5569                                                                              3)
5570                                                              +
5571                                                              (k &
5572                                                               (unsigned char)7))
5573                                                             & (unsigned char)
5574                                                             0x3F;
5575                                                         FPT_inisci(p_card,
5576                                                                    p_port,
5577                                                                    p_our_id);
5578                                                         FPT_scamInfo[currCard->
5579                                                                      ourId].
5580                                                             state = ID_ASSIGNED;
5581                                                         FPT_scamInfo[currCard->
5582                                                                      ourId].
5583                                                             id_string[0]
5584                                                             = SLV_TYPE_CODE0;
5585                                                         assigned_id = 1;
5586                                                 }
5587                                         }
5588                                 }
5589                         }
5590
5591                         else if (i == SET_P_FLAG) {
5592                                 if (!(FPT_scsendi(p_port,
5593                                                   &FPT_scamInfo[p_our_id].
5594                                                   id_string[0])))
5595                                         FPT_scamInfo[p_our_id].id_string[0] |=
5596                                             0x80;
5597                         }
5598                 } while (!assigned_id);
5599
5600                 while (FPT_scxferc(p_port, 0x00) != CFG_CMPLT) {
5601                 }
5602         }
5603
5604         if (ScamFlg & SCAM_ENABLED) {
5605                 FPT_scbusf(p_port);
5606                 if (currCard->globalFlags & F_UPDATE_EEPROM) {
5607                         FPT_scsavdi(p_card, p_port);
5608                         currCard->globalFlags &= ~F_UPDATE_EEPROM;
5609                 }
5610         }
5611
5612 /*
5613    for (i=0,k=0; i < MAX_SCSI_TAR; i++)
5614       {
5615       if ((FPT_scamInfo[i].state == ID_ASSIGNED) ||
5616          (FPT_scamInfo[i].state == LEGACY))
5617          k++;
5618       }
5619
5620    if (k==2)
5621       currCard->globalFlags |= F_SINGLE_DEVICE;
5622    else
5623       currCard->globalFlags &= ~F_SINGLE_DEVICE;
5624 */
5625 }
5626
5627 /*---------------------------------------------------------------------
5628  *
5629  * Function: FPT_scarb
5630  *
5631  * Description: Gain control of the bus and wait SCAM select time (250ms)
5632  *
5633  *---------------------------------------------------------------------*/
5634
5635 static int FPT_scarb(u32 p_port, unsigned char p_sel_type)
5636 {
5637         if (p_sel_type == INIT_SELTD) {
5638
5639                 while (RD_HARPOON(p_port + hp_scsisig) & (SCSI_SEL | SCSI_BSY)) {
5640                 }
5641
5642                 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_SEL)
5643                         return 0;
5644
5645                 if (RD_HARPOON(p_port + hp_scsidata_0) != 00)
5646                         return 0;
5647
5648                 WR_HARPOON(p_port + hp_scsisig,
5649                            (RD_HARPOON(p_port + hp_scsisig) | SCSI_BSY));
5650
5651                 if (RD_HARPOON(p_port + hp_scsisig) & SCSI_SEL) {
5652
5653                         WR_HARPOON(p_port + hp_scsisig,
5654                                    (RD_HARPOON(p_port + hp_scsisig) &
5655                                     ~SCSI_BSY));
5656                         return 0;
5657                 }
5658
5659                 WR_HARPOON(p_port + hp_scsisig,
5660                            (RD_HARPOON(p_port + hp_scsisig) | SCSI_SEL));
5661
5662                 if (RD_HARPOON(p_port + hp_scsidata_0) != 00) {
5663
5664                         WR_HARPOON(p_port + hp_scsisig,
5665                                    (RD_HARPOON(p_port + hp_scsisig) &
5666                                     ~(SCSI_BSY | SCSI_SEL)));
5667                         return 0;
5668                 }
5669         }
5670
5671         WR_HARPOON(p_port + hp_clkctrl_0, (RD_HARPOON(p_port + hp_clkctrl_0)
5672                                            & ~ACTdeassert));
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);
5677
5678         WR_HARPOON(p_port + hp_scsisig,
5679                    (RD_HARPOON(p_port + hp_scsisig) | SCSI_MSG));
5680
5681         WR_HARPOON(p_port + hp_scsisig, (RD_HARPOON(p_port + hp_scsisig)
5682                                          & ~SCSI_BSY));
5683
5684         FPT_Wait(p_port, TO_250ms);
5685
5686         return 1;
5687 }
5688
5689 /*---------------------------------------------------------------------
5690  *
5691  * Function: FPT_scbusf
5692  *
5693  * Description: Release the SCSI bus and disable SCAM selection.
5694  *
5695  *---------------------------------------------------------------------*/
5696
5697 static void FPT_scbusf(u32 p_port)
5698 {
5699         WR_HARPOON(p_port + hp_page_ctrl,
5700                    (RD_HARPOON(p_port + hp_page_ctrl) | G_INT_DISABLE));
5701
5702         WR_HARPOON(p_port + hp_scsidata_0, 0x00);
5703
5704         WR_HARPOON(p_port + hp_portctrl_0, (RD_HARPOON(p_port + hp_portctrl_0)
5705                                             & ~SCSI_BUS_EN));
5706
5707         WR_HARPOON(p_port + hp_scsisig, 0x00);
5708
5709         WR_HARPOON(p_port + hp_scsireset, (RD_HARPOON(p_port + hp_scsireset)
5710                                            & ~SCAM_EN));
5711
5712         WR_HARPOON(p_port + hp_clkctrl_0, (RD_HARPOON(p_port + hp_clkctrl_0)
5713                                            | ACTdeassert));
5714
5715         WRW_HARPOON((p_port + hp_intstat), (BUS_FREE | AUTO_INT | SCAM_SEL));
5716
5717         WR_HARPOON(p_port + hp_page_ctrl,
5718                    (RD_HARPOON(p_port + hp_page_ctrl) & ~G_INT_DISABLE));
5719 }
5720
5721 /*---------------------------------------------------------------------
5722  *
5723  * Function: FPT_scasid
5724  *
5725  * Description: Assign an ID to all the SCAM devices.
5726  *
5727  *---------------------------------------------------------------------*/
5728
5729 static void FPT_scasid(unsigned char p_card, u32 p_port)
5730 {
5731         unsigned char temp_id_string[ID_STRING_LENGTH];
5732
5733         unsigned char i, k, scam_id;
5734         unsigned char crcBytes[3];
5735         struct nvram_info *pCurrNvRam;
5736         unsigned short *pCrcBytes;
5737
5738         pCurrNvRam = FPT_BL_Card[p_card].pNvRamInfo;
5739
5740         i = 0;
5741
5742         while (!i) {
5743
5744                 for (k = 0; k < ID_STRING_LENGTH; k++) {
5745                         temp_id_string[k] = (unsigned char)0x00;
5746                 }
5747
5748                 FPT_scxferc(p_port, SYNC_PTRN);
5749                 FPT_scxferc(p_port, ASSIGN_ID);
5750
5751                 if (!(FPT_sciso(p_port, &temp_id_string[0]))) {
5752                         if (pCurrNvRam) {
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;
5761                         }
5762                         i = FPT_scmachid(p_card, temp_id_string);
5763
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. */
5768                         }
5769
5770                         else if (i != NO_ID_AVAIL) {
5771                                 if (i < 8)
5772                                         FPT_scxferc(p_port, ID_0_7);
5773                                 else
5774                                         FPT_scxferc(p_port, ID_8_F);
5775
5776                                 scam_id = (i & (unsigned char)0x07);
5777
5778                                 for (k = 1; k < 0x08; k <<= 1)
5779                                         if (!(k & i))
5780                                                 scam_id += 0x08;        /*Count number of zeros in DB0-3. */
5781
5782                                 FPT_scxferc(p_port, scam_id);
5783
5784                                 i = 0;  /*Not the last ID yet. */
5785                         }
5786                 }
5787
5788                 else {
5789                         i = 1;
5790                 }
5791
5792         }                       /*End while */
5793
5794         FPT_scxferc(p_port, SYNC_PTRN);
5795         FPT_scxferc(p_port, CFG_CMPLT);
5796 }
5797
5798 /*---------------------------------------------------------------------
5799  *
5800  * Function: FPT_scsel
5801  *
5802  * Description: Select all the SCAM devices.
5803  *
5804  *---------------------------------------------------------------------*/
5805
5806 static void FPT_scsel(u32 p_port)
5807 {
5808
5809         WR_HARPOON(p_port + hp_scsisig, SCSI_SEL);
5810         FPT_scwiros(p_port, SCSI_MSG);
5811
5812         WR_HARPOON(p_port + hp_scsisig, (SCSI_SEL | SCSI_BSY));
5813
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))));
5819
5820         WR_HARPOON(p_port + hp_scsisig, (SCSI_BSY | SCSI_IOBIT | SCSI_CD));
5821         FPT_scwiros(p_port, SCSI_SEL);
5822
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));
5827
5828         WR_HARPOON(p_port + hp_scsisig,
5829                    (SCSI_SEL | SCSI_BSY | SCSI_IOBIT | SCSI_CD));
5830 }
5831
5832 /*---------------------------------------------------------------------
5833  *
5834  * Function: FPT_scxferc
5835  *
5836  * Description: Handshake the p_data (DB4-0) across the bus.
5837  *
5838  *---------------------------------------------------------------------*/
5839
5840 static unsigned char FPT_scxferc(u32 p_port, unsigned char p_data)
5841 {
5842         unsigned char curr_data, ret_data;
5843
5844         curr_data = p_data | BIT(7) | BIT(5);   /*Start with DB7 & DB5 asserted. */
5845
5846         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5847
5848         curr_data &= ~BIT(7);
5849
5850         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5851
5852         FPT_scwirod(p_port, BIT(7));    /*Wait for DB7 to be released. */
5853         while (!(RD_HARPOON(p_port + hp_scsidata_0) & BIT(5))) ;
5854
5855         ret_data = (RD_HARPOON(p_port + hp_scsidata_0) & (unsigned char)0x1F);
5856
5857         curr_data |= BIT(6);
5858
5859         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5860
5861         curr_data &= ~BIT(5);
5862
5863         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5864
5865         FPT_scwirod(p_port, BIT(5));    /*Wait for DB5 to be released. */
5866
5867         curr_data &= ~(BIT(4) | BIT(3) | BIT(2) | BIT(1) | BIT(0));     /*Release data bits */
5868         curr_data |= BIT(7);
5869
5870         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5871
5872         curr_data &= ~BIT(6);
5873
5874         WR_HARPOON(p_port + hp_scsidata_0, curr_data);
5875
5876         FPT_scwirod(p_port, BIT(6));    /*Wait for DB6 to be released. */
5877
5878         return ret_data;
5879 }
5880
5881 /*---------------------------------------------------------------------
5882  *
5883  * Function: FPT_scsendi
5884  *
5885  * Description: Transfer our Identification string to determine if we
5886  *              will be the dominant master.
5887  *
5888  *---------------------------------------------------------------------*/
5889
5890 static unsigned char FPT_scsendi(u32 p_port, unsigned char p_id_string[])
5891 {
5892         unsigned char ret_data, byte_cnt, bit_cnt, defer;
5893
5894         defer = 0;
5895
5896         for (byte_cnt = 0; byte_cnt < ID_STRING_LENGTH; byte_cnt++) {
5897
5898                 for (bit_cnt = 0x80; bit_cnt != 0; bit_cnt >>= 1) {
5899
5900                         if (defer)
5901                                 ret_data = FPT_scxferc(p_port, 00);
5902
5903                         else if (p_id_string[byte_cnt] & bit_cnt)
5904
5905                                 ret_data = FPT_scxferc(p_port, 02);
5906
5907                         else {
5908
5909                                 ret_data = FPT_scxferc(p_port, 01);
5910                                 if (ret_data & 02)
5911                                         defer = 1;
5912                         }
5913
5914                         if ((ret_data & 0x1C) == 0x10)
5915                                 return 0x00;    /*End of isolation stage, we won! */
5916
5917                         if (ret_data & 0x1C)
5918                                 return 0xFF;
5919
5920                         if ((defer) && (!(ret_data & 0x1F)))
5921                                 return 0x01;    /*End of isolation stage, we lost. */
5922
5923                 }               /*bit loop */
5924
5925         }                       /*byte loop */
5926
5927         if (defer)
5928                 return 0x01;    /*We lost */
5929         else
5930                 return 0;       /*We WON! Yeeessss! */
5931 }
5932
5933 /*---------------------------------------------------------------------
5934  *
5935  * Function: FPT_sciso
5936  *
5937  * Description: Transfer the Identification string.
5938  *
5939  *---------------------------------------------------------------------*/
5940
5941 static unsigned char FPT_sciso(u32 p_port, unsigned char p_id_string[])
5942 {
5943         unsigned char ret_data, the_data, byte_cnt, bit_cnt;
5944
5945         the_data = 0;
5946
5947         for (byte_cnt = 0; byte_cnt < ID_STRING_LENGTH; byte_cnt++) {
5948
5949                 for (bit_cnt = 0; bit_cnt < 8; bit_cnt++) {
5950
5951                         ret_data = FPT_scxferc(p_port, 0);
5952
5953                         if (ret_data & 0xFC)
5954                                 return 0xFF;
5955
5956                         else {
5957
5958                                 the_data <<= 1;
5959                                 if (ret_data & BIT(1)) {
5960                                         the_data |= 1;
5961                                 }
5962                         }
5963
5964                         if ((ret_data & 0x1F) == 0) {
5965 /*
5966                                 if(bit_cnt != 0 || bit_cnt != 8)
5967                                 {
5968                                         byte_cnt = 0;
5969                                         bit_cnt = 0;
5970                                         FPT_scxferc(p_port, SYNC_PTRN);
5971                                         FPT_scxferc(p_port, ASSIGN_ID);
5972                                         continue;
5973                                 }
5974 */
5975                                 if (byte_cnt)
5976                                         return 0x00;
5977                                 else
5978                                         return 0xFF;
5979                         }
5980
5981                 }               /*bit loop */
5982
5983                 p_id_string[byte_cnt] = the_data;
5984
5985         }                       /*byte loop */
5986
5987         return 0;
5988 }
5989
5990 /*---------------------------------------------------------------------
5991  *
5992  * Function: FPT_scwirod
5993  *
5994  * Description: Sample the SCSI data bus making sure the signal has been
5995  *              deasserted for the correct number of consecutive samples.
5996  *
5997  *---------------------------------------------------------------------*/
5998
5999 static void FPT_scwirod(u32 p_port, unsigned char p_data_bit)
6000 {
6001         unsigned char i;
6002
6003         i = 0;
6004         while (i < MAX_SCSI_TAR) {
6005
6006                 if (RD_HARPOON(p_port + hp_scsidata_0) & p_data_bit)
6007
6008                         i = 0;
6009
6010                 else
6011
6012                         i++;
6013
6014         }
6015 }
6016
6017 /*---------------------------------------------------------------------
6018  *
6019  * Function: FPT_scwiros
6020  *
6021  * Description: Sample the SCSI Signal lines making sure the signal has been
6022  *              deasserted for the correct number of consecutive samples.
6023  *
6024  *---------------------------------------------------------------------*/
6025
6026 static void FPT_scwiros(u32 p_port, unsigned char p_data_bit)
6027 {
6028         unsigned char i;
6029
6030         i = 0;
6031         while (i < MAX_SCSI_TAR) {
6032
6033                 if (RD_HARPOON(p_port + hp_scsisig) & p_data_bit)
6034
6035                         i = 0;
6036
6037                 else
6038
6039                         i++;
6040
6041         }
6042 }
6043
6044 /*---------------------------------------------------------------------
6045  *
6046  * Function: FPT_scvalq
6047  *
6048  * Description: Make sure we received a valid data byte.
6049  *
6050  *---------------------------------------------------------------------*/
6051
6052 static unsigned char FPT_scvalq(unsigned char p_quintet)
6053 {
6054         unsigned char count;
6055
6056         for (count = 1; count < 0x08; count <<= 1) {
6057                 if (!(p_quintet & count))
6058                         p_quintet -= 0x80;
6059         }
6060
6061         if (p_quintet & 0x18)
6062                 return 0;
6063
6064         else
6065                 return 1;
6066 }
6067
6068 /*---------------------------------------------------------------------
6069  *
6070  * Function: FPT_scsell
6071  *
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.
6075  *
6076  *---------------------------------------------------------------------*/
6077
6078 static unsigned char FPT_scsell(u32 p_port, unsigned char targ_id)
6079 {
6080         unsigned long i;
6081
6082         WR_HARPOON(p_port + hp_page_ctrl,
6083                    (RD_HARPOON(p_port + hp_page_ctrl) | G_INT_DISABLE));
6084
6085         ARAM_ACCESS(p_port);
6086
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);
6090
6091         for (i = p_port + CMD_STRT; i < p_port + CMD_STRT + 12; i += 2) {
6092                 WRW_HARPOON(i, (MPM_OP + ACOMMAND));
6093         }
6094         WRW_HARPOON(i, (BRH_OP + ALWAYS + NP));
6095
6096         WRW_HARPOON((p_port + hp_intstat),
6097                     (RESET | TIMEOUT | SEL | BUS_FREE | AUTO_INT));
6098
6099         WR_HARPOON(p_port + hp_select_id, targ_id);
6100
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));
6104
6105         while (!(RDW_HARPOON((p_port + hp_intstat)) &
6106                  (RESET | PROG_HLT | TIMEOUT | AUTO_INT))) {
6107         }
6108
6109         if (RDW_HARPOON((p_port + hp_intstat)) & RESET)
6110                 FPT_Wait(p_port, TO_250ms);
6111
6112         DISABLE_AUTO(p_port);
6113
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);
6117
6118         SGRAM_ACCESS(p_port);
6119
6120         if (RDW_HARPOON((p_port + hp_intstat)) & (RESET | TIMEOUT)) {
6121
6122                 WRW_HARPOON((p_port + hp_intstat),
6123                             (RESET | TIMEOUT | SEL | BUS_FREE | PHASE));
6124
6125                 WR_HARPOON(p_port + hp_page_ctrl,
6126                            (RD_HARPOON(p_port + hp_page_ctrl) &
6127                             ~G_INT_DISABLE));
6128
6129                 return 0;       /*No legacy device */
6130         }
6131
6132         else {
6133
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));
6138                                 ACCEPT_MSG(p_port);
6139                         }
6140                 }
6141
6142                 WRW_HARPOON((p_port + hp_intstat), CLR_ALL_INT_1);
6143
6144                 WR_HARPOON(p_port + hp_page_ctrl,
6145                            (RD_HARPOON(p_port + hp_page_ctrl) &
6146                             ~G_INT_DISABLE));
6147
6148                 return 1;       /*Found one of them oldies! */
6149         }
6150 }
6151
6152 /*---------------------------------------------------------------------
6153  *
6154  * Function: FPT_scwtsel
6155  *
6156  * Description: Wait to be selected by another SCAM initiator.
6157  *
6158  *---------------------------------------------------------------------*/
6159
6160 static void FPT_scwtsel(u32 p_port)
6161 {
6162         while (!(RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL)) {
6163         }
6164 }
6165
6166 /*---------------------------------------------------------------------
6167  *
6168  * Function: FPT_inisci
6169  *
6170  * Description: Setup the data Structure with the info from the EEPROM.
6171  *
6172  *---------------------------------------------------------------------*/
6173
6174 static void FPT_inisci(unsigned char p_card, u32 p_port, unsigned char p_our_id)
6175 {
6176         unsigned char i, k, max_id;
6177         unsigned short ee_data;
6178         struct nvram_info *pCurrNvRam;
6179
6180         pCurrNvRam = FPT_BL_Card[p_card].pNvRamInfo;
6181
6182         if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6183                 max_id = 0x08;
6184
6185         else
6186                 max_id = 0x10;
6187
6188         if (pCurrNvRam) {
6189                 for (i = 0; i < max_id; i++) {
6190
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;
6197
6198                         if (FPT_scamInfo[i].id_string[0] == 0x00)
6199                                 FPT_scamInfo[i].state = ID_UNUSED;      /*Default to unused ID. */
6200                         else
6201                                 FPT_scamInfo[i].state = ID_UNASSIGNED;  /*Default to unassigned ID. */
6202
6203                 }
6204         } else {
6205                 for (i = 0; i < max_id; i++) {
6206                         for (k = 0; k < ID_STRING_LENGTH; k += 2) {
6207                                 ee_data =
6208                                     FPT_utilEERead(p_port,
6209                                                    (unsigned
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;
6215                                 ee_data >>= 8;
6216                                 FPT_scamInfo[i].id_string[k + 1] =
6217                                     (unsigned char)ee_data;
6218                         }
6219
6220                         if ((FPT_scamInfo[i].id_string[0] == 0x00) ||
6221                             (FPT_scamInfo[i].id_string[0] == 0xFF))
6222
6223                                 FPT_scamInfo[i].state = ID_UNUSED;      /*Default to unused ID. */
6224
6225                         else
6226                                 FPT_scamInfo[i].state = ID_UNASSIGNED;  /*Default to unassigned ID. */
6227
6228                 }
6229         }
6230         for (k = 0; k < ID_STRING_LENGTH; k++)
6231                 FPT_scamInfo[p_our_id].id_string[k] = FPT_scamHAString[k];
6232
6233 }
6234
6235 /*---------------------------------------------------------------------
6236  *
6237  * Function: FPT_scmachid
6238  *
6239  * Description: Match the Device ID string with our values stored in
6240  *              the EEPROM.
6241  *
6242  *---------------------------------------------------------------------*/
6243
6244 static unsigned char FPT_scmachid(unsigned char p_card,
6245                                   unsigned char p_id_string[])
6246 {
6247
6248         unsigned char i, k, match;
6249
6250         for (i = 0; i < MAX_SCSI_TAR; i++) {
6251
6252                 match = 1;
6253
6254                 for (k = 0; k < ID_STRING_LENGTH; k++) {
6255                         if (p_id_string[k] != FPT_scamInfo[i].id_string[k])
6256                                 match = 0;
6257                 }
6258
6259                 if (match) {
6260                         FPT_scamInfo[i].state = ID_ASSIGNED;
6261                         return i;
6262                 }
6263
6264         }
6265
6266         if (p_id_string[0] & BIT(5))
6267                 i = 8;
6268         else
6269                 i = MAX_SCSI_TAR;
6270
6271         if (((p_id_string[0] & 0x06) == 0x02)
6272             || ((p_id_string[0] & 0x06) == 0x04))
6273                 match = p_id_string[1] & (unsigned char)0x1F;
6274         else
6275                 match = 7;
6276
6277         while (i > 0) {
6278                 i--;
6279
6280                 if (FPT_scamInfo[match].state == ID_UNUSED) {
6281                         for (k = 0; k < ID_STRING_LENGTH; k++) {
6282                                 FPT_scamInfo[match].id_string[k] =
6283                                     p_id_string[k];
6284                         }
6285
6286                         FPT_scamInfo[match].state = ID_ASSIGNED;
6287
6288                         if (FPT_BL_Card[p_card].pNvRamInfo == NULL)
6289                                 FPT_BL_Card[p_card].globalFlags |=
6290                                     F_UPDATE_EEPROM;
6291                         return match;
6292
6293                 }
6294
6295                 match--;
6296
6297                 if (match == 0xFF) {
6298                         if (p_id_string[0] & BIT(5))
6299                                 match = 7;
6300                         else
6301                                 match = MAX_SCSI_TAR - 1;
6302                 }
6303         }
6304
6305         if (p_id_string[0] & BIT(7)) {
6306                 return CLR_PRIORITY;
6307         }
6308
6309         if (p_id_string[0] & BIT(5))
6310                 i = 8;
6311         else
6312                 i = MAX_SCSI_TAR;
6313
6314         if (((p_id_string[0] & 0x06) == 0x02)
6315             || ((p_id_string[0] & 0x06) == 0x04))
6316                 match = p_id_string[1] & (unsigned char)0x1F;
6317         else
6318                 match = 7;
6319
6320         while (i > 0) {
6321
6322                 i--;
6323
6324                 if (FPT_scamInfo[match].state == ID_UNASSIGNED) {
6325                         for (k = 0; k < ID_STRING_LENGTH; k++) {
6326                                 FPT_scamInfo[match].id_string[k] =
6327                                     p_id_string[k];
6328                         }
6329
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 |=
6334                                     F_UPDATE_EEPROM;
6335                         return match;
6336
6337                 }
6338
6339                 match--;
6340
6341                 if (match == 0xFF) {
6342                         if (p_id_string[0] & BIT(5))
6343                                 match = 7;
6344                         else
6345                                 match = MAX_SCSI_TAR - 1;
6346                 }
6347         }
6348
6349         return NO_ID_AVAIL;
6350 }
6351
6352 /*---------------------------------------------------------------------
6353  *
6354  * Function: FPT_scsavdi
6355  *
6356  * Description: Save off the device SCAM ID strings.
6357  *
6358  *---------------------------------------------------------------------*/
6359
6360 static void FPT_scsavdi(unsigned char p_card, u32 p_port)
6361 {
6362         unsigned char i, k, max_id;
6363         unsigned short ee_data, sum_data;
6364
6365         sum_data = 0x0000;
6366
6367         for (i = 1; i < EE_SCAMBASE / 2; i++) {
6368                 sum_data += FPT_utilEERead(p_port, i);
6369         }
6370
6371         FPT_utilEEWriteOnOff(p_port, 1);        /* Enable write access to the EEPROM */
6372
6373         if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6374                 max_id = 0x08;
6375
6376         else
6377                 max_id = 0x10;
6378
6379         for (i = 0; i < max_id; i++) {
6380
6381                 for (k = 0; k < ID_STRING_LENGTH; k += 2) {
6382                         ee_data = FPT_scamInfo[i].id_string[k + 1];
6383                         ee_data <<= 8;
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)));
6390                 }
6391         }
6392
6393         FPT_utilEEWrite(p_port, sum_data, EEPROM_CHECK_SUM / 2);
6394         FPT_utilEEWriteOnOff(p_port, 0);        /* Turn off write access */
6395 }
6396
6397 /*---------------------------------------------------------------------
6398  *
6399  * Function: FPT_XbowInit
6400  *
6401  * Description: Setup the Xbow for normal operation.
6402  *
6403  *---------------------------------------------------------------------*/
6404
6405 static void FPT_XbowInit(u32 port, unsigned char ScamFlg)
6406 {
6407         unsigned char i;
6408
6409         i = RD_HARPOON(port + hp_page_ctrl);
6410         WR_HARPOON(port + hp_page_ctrl, (unsigned char)(i | G_INT_DISABLE));
6411
6412         WR_HARPOON(port + hp_scsireset, 0x00);
6413         WR_HARPOON(port + hp_portctrl_1, HOST_MODE8);
6414
6415         WR_HARPOON(port + hp_scsireset, (DMA_RESET | HPSCSI_RESET | PROG_RESET |
6416                                          FIFO_CLR));
6417
6418         WR_HARPOON(port + hp_scsireset, SCSI_INI);
6419
6420         WR_HARPOON(port + hp_clkctrl_0, CLKCTRL_DEFAULT);
6421
6422         WR_HARPOON(port + hp_scsisig, 0x00);    /*  Clear any signals we might */
6423         WR_HARPOON(port + hp_scsictrl_0, ENA_SCAM_SEL);
6424
6425         WRW_HARPOON((port + hp_intstat), CLR_ALL_INT);
6426
6427         FPT_default_intena = RESET | RSEL | PROG_HLT | TIMEOUT |
6428             BUS_FREE | XFER_CNT_0 | AUTO_INT;
6429
6430         if ((ScamFlg & SCAM_ENABLED) && (ScamFlg & SCAM_LEVEL2))
6431                 FPT_default_intena |= SCAM_SEL;
6432
6433         WRW_HARPOON((port + hp_intena), FPT_default_intena);
6434
6435         WR_HARPOON(port + hp_seltimeout, TO_290ms);
6436
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);
6441
6442         WR_HARPOON(port + hp_page_ctrl, i);
6443
6444 }
6445
6446 /*---------------------------------------------------------------------
6447  *
6448  * Function: FPT_BusMasterInit
6449  *
6450  * Description: Initialize the BusMaster for normal operations.
6451  *
6452  *---------------------------------------------------------------------*/
6453
6454 static void FPT_BusMasterInit(u32 p_port)
6455 {
6456
6457         WR_HARPOON(p_port + hp_sys_ctrl, DRVR_RST);
6458         WR_HARPOON(p_port + hp_sys_ctrl, 0x00);
6459
6460         WR_HARPOON(p_port + hp_host_blk_cnt, XFER_BLK64);
6461
6462         WR_HARPOON(p_port + hp_bm_ctrl, (BMCTRL_DEFAULT));
6463
6464         WR_HARPOON(p_port + hp_ee_ctrl, (SCSI_TERM_ENA_H));
6465
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) &
6469                                            ~SCATTER_EN));
6470 }
6471
6472 /*---------------------------------------------------------------------
6473  *
6474  * Function: FPT_DiagEEPROM
6475  *
6476  * Description: Verfiy checksum and 'Key' and initialize the EEPROM if
6477  *              necessary.
6478  *
6479  *---------------------------------------------------------------------*/
6480
6481 static void FPT_DiagEEPROM(u32 p_port)
6482 {
6483         unsigned short index, temp, max_wd_cnt;
6484
6485         if (RD_HARPOON(p_port + hp_page_ctrl) & NARROW_SCSI_CARD)
6486                 max_wd_cnt = EEPROM_WD_CNT;
6487         else
6488                 max_wd_cnt = EEPROM_WD_CNT * 2;
6489
6490         temp = FPT_utilEERead(p_port, FW_SIGNATURE / 2);
6491
6492         if (temp == 0x4641) {
6493
6494                 for (index = 2; index < max_wd_cnt; index++) {
6495
6496                         temp += FPT_utilEERead(p_port, index);
6497
6498                 }
6499
6500                 if (temp == FPT_utilEERead(p_port, EEPROM_CHECK_SUM / 2)) {
6501
6502                         return; /*EEPROM is Okay so return now! */
6503                 }
6504         }
6505
6506         FPT_utilEEWriteOnOff(p_port, (unsigned char)1);
6507
6508         for (index = 0; index < max_wd_cnt; index++) {
6509
6510                 FPT_utilEEWrite(p_port, 0x0000, index);
6511         }
6512
6513         temp = 0;
6514
6515         FPT_utilEEWrite(p_port, 0x4641, FW_SIGNATURE / 2);
6516         temp += 0x4641;
6517         FPT_utilEEWrite(p_port, 0x3920, MODEL_NUMB_0 / 2);
6518         temp += 0x3920;
6519         FPT_utilEEWrite(p_port, 0x3033, MODEL_NUMB_2 / 2);
6520         temp += 0x3033;
6521         FPT_utilEEWrite(p_port, 0x2020, MODEL_NUMB_4 / 2);
6522         temp += 0x2020;
6523         FPT_utilEEWrite(p_port, 0x70D3, SYSTEM_CONFIG / 2);
6524         temp += 0x70D3;
6525         FPT_utilEEWrite(p_port, 0x0010, BIOS_CONFIG / 2);
6526         temp += 0x0010;
6527         FPT_utilEEWrite(p_port, 0x0003, SCAM_CONFIG / 2);
6528         temp += 0x0003;
6529         FPT_utilEEWrite(p_port, 0x0007, ADAPTER_SCSI_ID / 2);
6530         temp += 0x0007;
6531
6532         FPT_utilEEWrite(p_port, 0x0000, IGNORE_B_SCAN / 2);
6533         temp += 0x0000;
6534         FPT_utilEEWrite(p_port, 0x0000, SEND_START_ENA / 2);
6535         temp += 0x0000;
6536         FPT_utilEEWrite(p_port, 0x0000, DEVICE_ENABLE / 2);
6537         temp += 0x0000;
6538
6539         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL01 / 2);
6540         temp += 0x4242;
6541         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL23 / 2);
6542         temp += 0x4242;
6543         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL45 / 2);
6544         temp += 0x4242;
6545         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL67 / 2);
6546         temp += 0x4242;
6547         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBL89 / 2);
6548         temp += 0x4242;
6549         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLab / 2);
6550         temp += 0x4242;
6551         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLcd / 2);
6552         temp += 0x4242;
6553         FPT_utilEEWrite(p_port, 0x4242, SYNC_RATE_TBLef / 2);
6554         temp += 0x4242;
6555
6556         FPT_utilEEWrite(p_port, 0x6C46, 64 / 2);        /*PRODUCT ID */
6557         temp += 0x6C46;
6558         FPT_utilEEWrite(p_port, 0x7361, 66 / 2);        /* FlashPoint LT   */
6559         temp += 0x7361;
6560         FPT_utilEEWrite(p_port, 0x5068, 68 / 2);
6561         temp += 0x5068;
6562         FPT_utilEEWrite(p_port, 0x696F, 70 / 2);
6563         temp += 0x696F;
6564         FPT_utilEEWrite(p_port, 0x746E, 72 / 2);
6565         temp += 0x746E;
6566         FPT_utilEEWrite(p_port, 0x4C20, 74 / 2);
6567         temp += 0x4C20;
6568         FPT_utilEEWrite(p_port, 0x2054, 76 / 2);
6569         temp += 0x2054;
6570         FPT_utilEEWrite(p_port, 0x2020, 78 / 2);
6571         temp += 0x2020;
6572
6573         index = ((EE_SCAMBASE / 2) + (7 * 16));
6574         FPT_utilEEWrite(p_port, (0x0700 + TYPE_CODE0), index);
6575         temp += (0x0700 + TYPE_CODE0);
6576         index++;
6577         FPT_utilEEWrite(p_port, 0x5542, index); /*Vendor ID code */
6578         temp += 0x5542;         /* BUSLOGIC      */
6579         index++;
6580         FPT_utilEEWrite(p_port, 0x4C53, index);
6581         temp += 0x4C53;
6582         index++;
6583         FPT_utilEEWrite(p_port, 0x474F, index);
6584         temp += 0x474F;
6585         index++;
6586         FPT_utilEEWrite(p_port, 0x4349, index);
6587         temp += 0x4349;
6588         index++;
6589         FPT_utilEEWrite(p_port, 0x5442, index); /*Vendor unique code */
6590         temp += 0x5442;         /* BT- 930           */
6591         index++;
6592         FPT_utilEEWrite(p_port, 0x202D, index);
6593         temp += 0x202D;
6594         index++;
6595         FPT_utilEEWrite(p_port, 0x3339, index);
6596         temp += 0x3339;
6597         index++;                /*Serial #          */
6598         FPT_utilEEWrite(p_port, 0x2030, index); /* 01234567         */
6599         temp += 0x2030;
6600         index++;
6601         FPT_utilEEWrite(p_port, 0x5453, index);
6602         temp += 0x5453;
6603         index++;
6604         FPT_utilEEWrite(p_port, 0x5645, index);
6605         temp += 0x5645;
6606         index++;
6607         FPT_utilEEWrite(p_port, 0x2045, index);
6608         temp += 0x2045;
6609         index++;
6610         FPT_utilEEWrite(p_port, 0x202F, index);
6611         temp += 0x202F;
6612         index++;
6613         FPT_utilEEWrite(p_port, 0x4F4A, index);
6614         temp += 0x4F4A;
6615         index++;
6616         FPT_utilEEWrite(p_port, 0x204E, index);
6617         temp += 0x204E;
6618         index++;
6619         FPT_utilEEWrite(p_port, 0x3539, index);
6620         temp += 0x3539;
6621
6622         FPT_utilEEWrite(p_port, temp, EEPROM_CHECK_SUM / 2);
6623
6624         FPT_utilEEWriteOnOff(p_port, (unsigned char)0);
6625
6626 }
6627
6628 /*---------------------------------------------------------------------
6629  *
6630  * Function: Queue Search Select
6631  *
6632  * Description: Try to find a new command to execute.
6633  *
6634  *---------------------------------------------------------------------*/
6635
6636 static void FPT_queueSearchSelect(struct sccb_card *pCurrCard,
6637                                   unsigned char p_card)
6638 {
6639         unsigned char scan_ptr, lun;
6640         struct sccb_mgr_tar_info *currTar_Info;
6641         struct sccb *pOldSccb;
6642
6643         scan_ptr = pCurrCard->scanIndex;
6644         do {
6645                 currTar_Info = &FPT_sccbMgrTbl[p_card][scan_ptr];
6646                 if ((pCurrCard->globalFlags & F_CONLUN_IO) &&
6647                     ((currTar_Info->TarStatus & TAR_TAG_Q_MASK) !=
6648                      TAG_Q_TRYING)) {
6649                         if (currTar_Info->TarSelQ_Cnt != 0) {
6650
6651                                 scan_ptr++;
6652                                 if (scan_ptr == MAX_SCSI_TAR)
6653                                         scan_ptr = 0;
6654
6655                                 for (lun = 0; lun < MAX_LUN; lun++) {
6656                                         if (currTar_Info->TarLUNBusy[lun] == 0) {
6657
6658                                                 pCurrCard->currentSCCB =
6659                                                     currTar_Info->TarSelQ_Head;
6660                                                 pOldSccb = NULL;
6661
6662                                                 while ((pCurrCard->
6663                                                         currentSCCB != NULL)
6664                                                        && (lun !=
6665                                                            pCurrCard->
6666                                                            currentSCCB->Lun)) {
6667                                                         pOldSccb =
6668                                                             pCurrCard->
6669                                                             currentSCCB;
6670                                                         pCurrCard->currentSCCB =
6671                                                             (struct sccb
6672                                                              *)(pCurrCard->
6673                                                                 currentSCCB)->
6674                                                             Sccb_forwardlink;
6675                                                 }
6676                                                 if (pCurrCard->currentSCCB ==
6677                                                     NULL)
6678                                                         continue;
6679                                                 if (pOldSccb != NULL) {
6680                                                         pOldSccb->
6681                                                             Sccb_forwardlink =
6682                                                             (struct sccb
6683                                                              *)(pCurrCard->
6684                                                                 currentSCCB)->
6685                                                             Sccb_forwardlink;
6686                                                         pOldSccb->
6687                                                             Sccb_backlink =
6688                                                             (struct sccb
6689                                                              *)(pCurrCard->
6690                                                                 currentSCCB)->
6691                                                             Sccb_backlink;
6692                                                         currTar_Info->
6693                                                             TarSelQ_Cnt--;
6694                                                 } else {
6695                                                         currTar_Info->
6696                                                             TarSelQ_Head =
6697                                                             (struct sccb
6698                                                              *)(pCurrCard->
6699                                                                 currentSCCB)->
6700                                                             Sccb_forwardlink;
6701
6702                                                         if (currTar_Info->
6703                                                             TarSelQ_Head ==
6704                                                             NULL) {
6705                                                                 currTar_Info->
6706                                                                     TarSelQ_Tail
6707                                                                     = NULL;
6708                                                                 currTar_Info->
6709                                                                     TarSelQ_Cnt
6710                                                                     = 0;
6711                                                         } else {
6712                                                                 currTar_Info->
6713                                                                     TarSelQ_Cnt--;
6714                                                                 currTar_Info->
6715                                                                     TarSelQ_Head->
6716                                                                     Sccb_backlink
6717                                                                     =
6718                                                                     (struct sccb
6719                                                                      *)NULL;
6720                                                         }
6721                                                 }
6722                                                 pCurrCard->scanIndex = scan_ptr;
6723
6724                                                 pCurrCard->globalFlags |=
6725                                                     F_NEW_SCCB_CMD;
6726
6727                                                 break;
6728                                         }
6729                                 }
6730                         }
6731
6732                         else {
6733                                 scan_ptr++;
6734                                 if (scan_ptr == MAX_SCSI_TAR) {
6735                                         scan_ptr = 0;
6736                                 }
6737                         }
6738
6739                 } else {
6740                         if ((currTar_Info->TarSelQ_Cnt != 0) &&
6741                             (currTar_Info->TarLUNBusy[0] == 0)) {
6742
6743                                 pCurrCard->currentSCCB =
6744                                     currTar_Info->TarSelQ_Head;
6745
6746                                 currTar_Info->TarSelQ_Head =
6747                                     (struct sccb *)(pCurrCard->currentSCCB)->
6748                                     Sccb_forwardlink;
6749
6750                                 if (currTar_Info->TarSelQ_Head == NULL) {
6751                                         currTar_Info->TarSelQ_Tail = NULL;
6752                                         currTar_Info->TarSelQ_Cnt = 0;
6753                                 } else {
6754                                         currTar_Info->TarSelQ_Cnt--;
6755                                         currTar_Info->TarSelQ_Head->
6756                                             Sccb_backlink = (struct sccb *)NULL;
6757                                 }
6758
6759                                 scan_ptr++;
6760                                 if (scan_ptr == MAX_SCSI_TAR)
6761                                         scan_ptr = 0;
6762
6763                                 pCurrCard->scanIndex = scan_ptr;
6764
6765                                 pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
6766
6767                                 break;
6768                         }
6769
6770                         else {
6771                                 scan_ptr++;
6772                                 if (scan_ptr == MAX_SCSI_TAR) {
6773                                         scan_ptr = 0;
6774                                 }
6775                         }
6776                 }
6777         } while (scan_ptr != pCurrCard->scanIndex);
6778 }
6779
6780 /*---------------------------------------------------------------------
6781  *
6782  * Function: Queue Select Fail
6783  *
6784  * Description: Add the current SCCB to the head of the Queue.
6785  *
6786  *---------------------------------------------------------------------*/
6787
6788 static void FPT_queueSelectFail(struct sccb_card *pCurrCard,
6789                                 unsigned char p_card)
6790 {
6791         unsigned char thisTarg;
6792         struct sccb_mgr_tar_info *currTar_Info;
6793
6794         if (pCurrCard->currentSCCB != NULL) {
6795                 thisTarg =
6796                     (unsigned char)(((struct sccb *)(pCurrCard->currentSCCB))->
6797                                     TargID);
6798                 currTar_Info = &FPT_sccbMgrTbl[p_card][thisTarg];
6799
6800                 pCurrCard->currentSCCB->Sccb_backlink = (struct sccb *)NULL;
6801
6802                 pCurrCard->currentSCCB->Sccb_forwardlink =
6803                     currTar_Info->TarSelQ_Head;
6804
6805                 if (currTar_Info->TarSelQ_Cnt == 0) {
6806                         currTar_Info->TarSelQ_Tail = pCurrCard->currentSCCB;
6807                 }
6808
6809                 else {
6810                         currTar_Info->TarSelQ_Head->Sccb_backlink =
6811                             pCurrCard->currentSCCB;
6812                 }
6813
6814                 currTar_Info->TarSelQ_Head = pCurrCard->currentSCCB;
6815
6816                 pCurrCard->currentSCCB = NULL;
6817                 currTar_Info->TarSelQ_Cnt++;
6818         }
6819 }
6820
6821 /*---------------------------------------------------------------------
6822  *
6823  * Function: Queue Command Complete
6824  *
6825  * Description: Call the callback function with the current SCCB.
6826  *
6827  *---------------------------------------------------------------------*/
6828
6829 static void FPT_queueCmdComplete(struct sccb_card *pCurrCard,
6830                                  struct sccb *p_sccb, unsigned char p_card)
6831 {
6832
6833         unsigned char i, SCSIcmd;
6834         CALL_BK_FN callback;
6835         struct sccb_mgr_tar_info *currTar_Info;
6836
6837         SCSIcmd = p_sccb->Cdb[0];
6838
6839         if (!(p_sccb->Sccb_XferState & F_ALL_XFERRED)) {
6840
6841                 if ((p_sccb->
6842                      ControlByte & (SCCB_DATA_XFER_OUT | SCCB_DATA_XFER_IN))
6843                     && (p_sccb->HostStatus == SCCB_COMPLETE)
6844                     && (p_sccb->TargetStatus != SAM_STAT_CHECK_CONDITION))
6845
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)
6853                             )
6854                                 p_sccb->HostStatus = SCCB_DATA_UNDER_RUN;
6855         }
6856
6857         if (p_sccb->SccbStatus == SCCB_IN_PROCESS) {
6858                 if (p_sccb->HostStatus || p_sccb->TargetStatus)
6859                         p_sccb->SccbStatus = SCCB_ERROR;
6860                 else
6861                         p_sccb->SccbStatus = SCCB_SUCCESS;
6862         }
6863
6864         if (p_sccb->Sccb_XferState & F_AUTO_SENSE) {
6865
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];
6869                 }
6870         }
6871
6872         if ((p_sccb->OperationCode == RESIDUAL_SG_COMMAND) ||
6873             (p_sccb->OperationCode == RESIDUAL_COMMAND)) {
6874
6875                 FPT_utilUpdateResidual(p_sccb);
6876         }
6877
6878         pCurrCard->cmdCounter--;
6879         if (!pCurrCard->cmdCounter) {
6880
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);
6885                 }
6886
6887                 WR_HARPOON(pCurrCard->ioPort + hp_semaphore,
6888                            (RD_HARPOON(pCurrCard->ioPort + hp_semaphore) &
6889                             ~SCCB_MGR_ACTIVE));
6890
6891         }
6892
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) !=
6897                       TAG_Q_TRYING))) {
6898                         pCurrCard->discQCount--;
6899                         pCurrCard->discQ_Tbl[currTar_Info->
6900                                              LunDiscQ_Idx[p_sccb->Lun]] = NULL;
6901                 } else {
6902                         if (p_sccb->Sccb_tag) {
6903                                 pCurrCard->discQCount--;
6904                                 pCurrCard->discQ_Tbl[p_sccb->Sccb_tag] = NULL;
6905                         } else {
6906                                 pCurrCard->discQCount--;
6907                                 pCurrCard->discQ_Tbl[currTar_Info->
6908                                                      LunDiscQ_Idx[0]] = NULL;
6909                         }
6910                 }
6911
6912         }
6913
6914         callback = (CALL_BK_FN) p_sccb->SccbCallback;
6915         callback(p_sccb);
6916         pCurrCard->globalFlags |= F_NEW_SCCB_CMD;
6917         pCurrCard->currentSCCB = NULL;
6918 }
6919
6920 /*---------------------------------------------------------------------
6921  *
6922  * Function: Queue Disconnect
6923  *
6924  * Description: Add SCCB to our disconnect array.
6925  *
6926  *---------------------------------------------------------------------*/
6927 static void FPT_queueDisconnect(struct sccb *p_sccb, unsigned char p_card)
6928 {
6929         struct sccb_mgr_tar_info *currTar_Info;
6930
6931         currTar_Info = &FPT_sccbMgrTbl[p_card][p_sccb->TargID];
6932
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]] =
6937                     p_sccb;
6938         } else {
6939                 if (p_sccb->Sccb_tag) {
6940                         FPT_BL_Card[p_card].discQ_Tbl[p_sccb->Sccb_tag] =
6941                             p_sccb;
6942                         FPT_sccbMgrTbl[p_card][p_sccb->TargID].TarLUNBusy[0] =
6943                             0;
6944                         FPT_sccbMgrTbl[p_card][p_sccb->TargID].TarTagQ_Cnt++;
6945                 } else {
6946                         FPT_BL_Card[p_card].discQ_Tbl[currTar_Info->
6947                                                       LunDiscQ_Idx[0]] = p_sccb;
6948                 }
6949         }
6950         FPT_BL_Card[p_card].currentSCCB = NULL;
6951 }
6952
6953 /*---------------------------------------------------------------------
6954  *
6955  * Function: Queue Flush SCCB
6956  *
6957  * Description: Flush all SCCB's back to the host driver for this target.
6958  *
6959  *---------------------------------------------------------------------*/
6960
6961 static void FPT_queueFlushSccb(unsigned char p_card, unsigned char error_code)
6962 {
6963         unsigned char qtag, thisTarg;
6964         struct sccb *currSCCB;
6965         struct sccb_mgr_tar_info *currTar_Info;
6966
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];
6971
6972                 for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
6973
6974                         if (FPT_BL_Card[p_card].discQ_Tbl[qtag] &&
6975                             (FPT_BL_Card[p_card].discQ_Tbl[qtag]->TargID ==
6976                              thisTarg)) {
6977
6978                                 FPT_BL_Card[p_card].discQ_Tbl[qtag]->
6979                                     HostStatus = (unsigned char)error_code;
6980
6981                                 FPT_queueCmdComplete(&FPT_BL_Card[p_card],
6982                                                      FPT_BL_Card[p_card].
6983                                                      discQ_Tbl[qtag], p_card);
6984
6985                                 FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
6986                                 currTar_Info->TarTagQ_Cnt--;
6987
6988                         }
6989                 }
6990         }
6991
6992 }
6993
6994 /*---------------------------------------------------------------------
6995  *
6996  * Function: Queue Flush Target SCCB
6997  *
6998  * Description: Flush all SCCB's back to the host driver for this target.
6999  *
7000  *---------------------------------------------------------------------*/
7001
7002 static void FPT_queueFlushTargSccb(unsigned char p_card, unsigned char thisTarg,
7003                                    unsigned char error_code)
7004 {
7005         unsigned char qtag;
7006         struct sccb_mgr_tar_info *currTar_Info;
7007
7008         currTar_Info = &FPT_sccbMgrTbl[p_card][thisTarg];
7009
7010         for (qtag = 0; qtag < QUEUE_DEPTH; qtag++) {
7011
7012                 if (FPT_BL_Card[p_card].discQ_Tbl[qtag] &&
7013                     (FPT_BL_Card[p_card].discQ_Tbl[qtag]->TargID == thisTarg)) {
7014
7015                         FPT_BL_Card[p_card].discQ_Tbl[qtag]->HostStatus =
7016                             (unsigned char)error_code;
7017
7018                         FPT_queueCmdComplete(&FPT_BL_Card[p_card],
7019                                              FPT_BL_Card[p_card].
7020                                              discQ_Tbl[qtag], p_card);
7021
7022                         FPT_BL_Card[p_card].discQ_Tbl[qtag] = NULL;
7023                         currTar_Info->TarTagQ_Cnt--;
7024
7025                 }
7026         }
7027
7028 }
7029
7030 static void FPT_queueAddSccb(struct sccb *p_SCCB, unsigned char p_card)
7031 {
7032         struct sccb_mgr_tar_info *currTar_Info;
7033         currTar_Info = &FPT_sccbMgrTbl[p_card][p_SCCB->TargID];
7034
7035         p_SCCB->Sccb_forwardlink = NULL;
7036
7037         p_SCCB->Sccb_backlink = currTar_Info->TarSelQ_Tail;
7038
7039         if (currTar_Info->TarSelQ_Cnt == 0) {
7040
7041                 currTar_Info->TarSelQ_Head = p_SCCB;
7042         }
7043
7044         else {
7045
7046                 currTar_Info->TarSelQ_Tail->Sccb_forwardlink = p_SCCB;
7047         }
7048
7049         currTar_Info->TarSelQ_Tail = p_SCCB;
7050         currTar_Info->TarSelQ_Cnt++;
7051 }
7052
7053 /*---------------------------------------------------------------------
7054  *
7055  * Function: Queue Find SCCB
7056  *
7057  * Description: Search the target select Queue for this SCCB, and
7058  *              remove it if found.
7059  *
7060  *---------------------------------------------------------------------*/
7061
7062 static unsigned char FPT_queueFindSccb(struct sccb *p_SCCB,
7063                                        unsigned char p_card)
7064 {
7065         struct sccb *q_ptr;
7066         struct sccb_mgr_tar_info *currTar_Info;
7067
7068         currTar_Info = &FPT_sccbMgrTbl[p_card][p_SCCB->TargID];
7069
7070         q_ptr = currTar_Info->TarSelQ_Head;
7071
7072         while (q_ptr != NULL) {
7073
7074                 if (q_ptr == p_SCCB) {
7075
7076                         if (currTar_Info->TarSelQ_Head == q_ptr) {
7077
7078                                 currTar_Info->TarSelQ_Head =
7079                                     q_ptr->Sccb_forwardlink;
7080                         }
7081
7082                         if (currTar_Info->TarSelQ_Tail == q_ptr) {
7083
7084                                 currTar_Info->TarSelQ_Tail =
7085                                     q_ptr->Sccb_backlink;
7086                         }
7087
7088                         if (q_ptr->Sccb_forwardlink != NULL) {
7089                                 q_ptr->Sccb_forwardlink->Sccb_backlink =
7090                                     q_ptr->Sccb_backlink;
7091                         }
7092
7093                         if (q_ptr->Sccb_backlink != NULL) {
7094                                 q_ptr->Sccb_backlink->Sccb_forwardlink =
7095                                     q_ptr->Sccb_forwardlink;
7096                         }
7097
7098                         currTar_Info->TarSelQ_Cnt--;
7099
7100                         return 1;
7101                 }
7102
7103                 else {
7104                         q_ptr = q_ptr->Sccb_forwardlink;
7105                 }
7106         }
7107
7108         return 0;
7109
7110 }
7111
7112 /*---------------------------------------------------------------------
7113  *
7114  * Function: Utility Update Residual Count
7115  *
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
7121  *              element.
7122  *
7123  *---------------------------------------------------------------------*/
7124
7125 static void FPT_utilUpdateResidual(struct sccb *p_SCCB)
7126 {
7127         unsigned long partial_cnt;
7128         unsigned int sg_index;
7129         struct blogic_sg_seg *segp;
7130
7131         if (p_SCCB->Sccb_XferState & F_ALL_XFERRED) {
7132
7133                 p_SCCB->DataLength = 0x0000;
7134         }
7135
7136         else if (p_SCCB->Sccb_XferState & F_SG_XFER) {
7137
7138                 partial_cnt = 0x0000;
7139
7140                 sg_index = p_SCCB->Sccb_sgseg;
7141
7142
7143                 if (p_SCCB->Sccb_SGoffset) {
7144
7145                         partial_cnt = p_SCCB->Sccb_SGoffset;
7146                         sg_index++;
7147                 }
7148
7149                 while (((unsigned long)sg_index *
7150                         (unsigned long)SG_ELEMENT_SIZE) < p_SCCB->DataLength) {
7151                         segp = (struct blogic_sg_seg *)(p_SCCB->DataPointer) +
7152                                         (sg_index * 2);
7153                         partial_cnt += segp->segbytes;
7154                         sg_index++;
7155                 }
7156
7157                 p_SCCB->DataLength = partial_cnt;
7158         }
7159
7160         else {
7161
7162                 p_SCCB->DataLength -= p_SCCB->Sccb_ATC;
7163         }
7164 }
7165
7166 /*---------------------------------------------------------------------
7167  *
7168  * Function: Wait 1 Second
7169  *
7170  * Description: Wait for 1 second.
7171  *
7172  *---------------------------------------------------------------------*/
7173
7174 static void FPT_Wait1Second(u32 p_port)
7175 {
7176         unsigned char i;
7177
7178         for (i = 0; i < 4; i++) {
7179
7180                 FPT_Wait(p_port, TO_250ms);
7181
7182                 if ((RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST))
7183                         break;
7184
7185                 if ((RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL))
7186                         break;
7187         }
7188 }
7189
7190 /*---------------------------------------------------------------------
7191  *
7192  * Function: FPT_Wait
7193  *
7194  * Description: Wait the desired delay.
7195  *
7196  *---------------------------------------------------------------------*/
7197
7198 static void FPT_Wait(u32 p_port, unsigned char p_delay)
7199 {
7200         unsigned char old_timer;
7201         unsigned char green_flag;
7202
7203         old_timer = RD_HARPOON(p_port + hp_seltimeout);
7204
7205         green_flag = RD_HARPOON(p_port + hp_clkctrl_0);
7206         WR_HARPOON(p_port + hp_clkctrl_0, CLKCTRL_DEFAULT);
7207
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));
7211
7212         WR_HARPOON(p_port + hp_portctrl_0,
7213                    (RD_HARPOON(p_port + hp_portctrl_0) | START_TO));
7214
7215         while (!(RDW_HARPOON((p_port + hp_intstat)) & TIMEOUT)) {
7216
7217                 if ((RD_HARPOON(p_port + hp_scsictrl_0) & SCSI_RST))
7218                         break;
7219
7220                 if ((RDW_HARPOON((p_port + hp_intstat)) & SCAM_SEL))
7221                         break;
7222         }
7223
7224         WR_HARPOON(p_port + hp_portctrl_0,
7225                    (RD_HARPOON(p_port + hp_portctrl_0) & ~START_TO));
7226
7227         WRW_HARPOON((p_port + hp_intstat), TIMEOUT);
7228         WRW_HARPOON((p_port + hp_intena), FPT_default_intena);
7229
7230         WR_HARPOON(p_port + hp_clkctrl_0, green_flag);
7231
7232         WR_HARPOON(p_port + hp_seltimeout, old_timer);
7233 }
7234
7235 /*---------------------------------------------------------------------
7236  *
7237  * Function: Enable/Disable Write to EEPROM
7238  *
7239  * Description: The EEPROM must first be enabled for writes
7240  *              A total of 9 clocks are needed.
7241  *
7242  *---------------------------------------------------------------------*/
7243
7244 static void FPT_utilEEWriteOnOff(u32 p_port, unsigned char p_mode)
7245 {
7246         unsigned char ee_value;
7247
7248         ee_value =
7249             (unsigned char)(RD_HARPOON(p_port + hp_ee_ctrl) &
7250                             (EXT_ARB_ACK | SCSI_TERM_ENA_H));
7251
7252         if (p_mode)
7253
7254                 FPT_utilEESendCmdAddr(p_port, EWEN, EWEN_ADDR);
7255
7256         else
7257
7258                 FPT_utilEESendCmdAddr(p_port, EWDS, EWDS_ADDR);
7259
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 */
7262 }
7263
7264 /*---------------------------------------------------------------------
7265  *
7266  * Function: Write EEPROM
7267  *
7268  * Description: Write a word to the EEPROM at the specified
7269  *              address.
7270  *
7271  *---------------------------------------------------------------------*/
7272
7273 static void FPT_utilEEWrite(u32 p_port, unsigned short ee_data,
7274                             unsigned short ee_addr)
7275 {
7276
7277         unsigned char ee_value;
7278         unsigned short i;
7279
7280         ee_value =
7281             (unsigned
7282              char)((RD_HARPOON(p_port + hp_ee_ctrl) &
7283                     (EXT_ARB_ACK | SCSI_TERM_ENA_H)) | (SEE_MS | SEE_CS));
7284
7285         FPT_utilEESendCmdAddr(p_port, EE_WRITE, ee_addr);
7286
7287         ee_value |= (SEE_MS + SEE_CS);
7288
7289         for (i = 0x8000; i != 0; i >>= 1) {
7290
7291                 if (i & ee_data)
7292                         ee_value |= SEE_DO;
7293                 else
7294                         ee_value &= ~SEE_DO;
7295
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);
7304         }
7305         ee_value &= (EXT_ARB_ACK | SCSI_TERM_ENA_H);
7306         WR_HARPOON(p_port + hp_ee_ctrl, (ee_value | SEE_MS));
7307
7308         FPT_Wait(p_port, TO_10ms);
7309
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 */
7313 }
7314
7315 /*---------------------------------------------------------------------
7316  *
7317  * Function: Read EEPROM
7318  *
7319  * Description: Read a word from the EEPROM at the desired
7320  *              address.
7321  *
7322  *---------------------------------------------------------------------*/
7323
7324 static unsigned short FPT_utilEERead(u32 p_port,
7325                                      unsigned short ee_addr)
7326 {
7327         unsigned short i, ee_data1, ee_data2;
7328
7329         i = 0;
7330         ee_data1 = FPT_utilEEReadOrg(p_port, ee_addr);
7331         do {
7332                 ee_data2 = FPT_utilEEReadOrg(p_port, ee_addr);
7333
7334                 if (ee_data1 == ee_data2)
7335                         return ee_data1;
7336
7337                 ee_data1 = ee_data2;
7338                 i++;
7339
7340         } while (i < 4);
7341
7342         return ee_data1;
7343 }
7344
7345 /*---------------------------------------------------------------------
7346  *
7347  * Function: Read EEPROM Original 
7348  *
7349  * Description: Read a word from the EEPROM at the desired
7350  *              address.
7351  *
7352  *---------------------------------------------------------------------*/
7353
7354 static unsigned short FPT_utilEEReadOrg(u32 p_port, unsigned short ee_addr)
7355 {
7356
7357         unsigned char ee_value;
7358         unsigned short i, ee_data;
7359
7360         ee_value =
7361             (unsigned
7362              char)((RD_HARPOON(p_port + hp_ee_ctrl) &
7363                     (EXT_ARB_ACK | SCSI_TERM_ENA_H)) | (SEE_MS | SEE_CS));
7364
7365         FPT_utilEESendCmdAddr(p_port, EE_READ, ee_addr);
7366
7367         ee_value |= (SEE_MS + SEE_CS);
7368         ee_data = 0;
7369
7370         for (i = 1; i <= 16; i++) {
7371
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);
7378
7379                 ee_data <<= 1;
7380
7381                 if (RD_HARPOON(p_port + hp_ee_ctrl) & SEE_DI)
7382                         ee_data |= 1;
7383         }
7384
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 */
7388
7389         return ee_data;
7390 }
7391
7392 /*---------------------------------------------------------------------
7393  *
7394  * Function: Send EE command and Address to the EEPROM
7395  *
7396  * Description: Transfers the correct command and sends the address
7397  *              to the eeprom.
7398  *
7399  *---------------------------------------------------------------------*/
7400
7401 static void FPT_utilEESendCmdAddr(u32 p_port, unsigned char ee_cmd,
7402                                   unsigned short ee_addr)
7403 {
7404         unsigned char ee_value;
7405         unsigned char narrow_flg;
7406
7407         unsigned short i;
7408
7409         narrow_flg =
7410             (unsigned char)(RD_HARPOON(p_port + hp_page_ctrl) &
7411                             NARROW_SCSI_CARD);
7412
7413         ee_value = SEE_MS;
7414         WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7415
7416         ee_value |= SEE_CS;     /* Set CS to EEPROM */
7417         WR_HARPOON(p_port + hp_ee_ctrl, ee_value);
7418
7419         for (i = 0x04; i != 0; i >>= 1) {
7420
7421                 if (i & ee_cmd)
7422                         ee_value |= SEE_DO;
7423                 else
7424                         ee_value &= ~SEE_DO;
7425
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);
7434         }
7435
7436         if (narrow_flg)
7437                 i = 0x0080;
7438
7439         else
7440                 i = 0x0200;
7441
7442         while (i != 0) {
7443
7444                 if (i & ee_addr)
7445                         ee_value |= SEE_DO;
7446                 else
7447                         ee_value &= ~SEE_DO;
7448
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);
7457
7458                 i >>= 1;
7459         }
7460 }
7461
7462 static unsigned short FPT_CalcCrc16(unsigned char buffer[])
7463 {
7464         unsigned short crc = 0;
7465         int i, j;
7466         unsigned short ch;
7467         for (i = 0; i < ID_STRING_LENGTH; i++) {
7468                 ch = (unsigned short)buffer[i];
7469                 for (j = 0; j < 8; j++) {
7470                         if ((crc ^ ch) & 1)
7471                                 crc = (crc >> 1) ^ CRCMASK;
7472                         else
7473                                 crc >>= 1;
7474                         ch >>= 1;
7475                 }
7476         }
7477         return crc;
7478 }
7479
7480 static unsigned char FPT_CalcLrc(unsigned char buffer[])
7481 {
7482         int i;
7483         unsigned char lrc;
7484         lrc = 0;
7485         for (i = 0; i < ID_STRING_LENGTH; i++)
7486                 lrc ^= buffer[i];
7487         return lrc;
7488 }
7489
7490 /*
7491   The following inline definitions avoid type conflicts.
7492 */
7493
7494 static inline unsigned char
7495 FlashPoint__ProbeHostAdapter(struct fpoint_info *FlashPointInfo)
7496 {
7497         return FlashPoint_ProbeHostAdapter((struct sccb_mgr_info *)
7498                                            FlashPointInfo);
7499 }
7500
7501 static inline void *
7502 FlashPoint__HardwareResetHostAdapter(struct fpoint_info *FlashPointInfo)
7503 {
7504         return FlashPoint_HardwareResetHostAdapter((struct sccb_mgr_info *)
7505                                                    FlashPointInfo);
7506 }
7507
7508 static inline void
7509 FlashPoint__ReleaseHostAdapter(void *CardHandle)
7510 {
7511         FlashPoint_ReleaseHostAdapter(CardHandle);
7512 }
7513
7514 static inline void
7515 FlashPoint__StartCCB(void *CardHandle, struct blogic_ccb *CCB)
7516 {
7517         FlashPoint_StartCCB(CardHandle, (struct sccb *)CCB);
7518 }
7519
7520 static inline void
7521 FlashPoint__AbortCCB(void *CardHandle, struct blogic_ccb *CCB)
7522 {
7523         FlashPoint_AbortCCB(CardHandle, (struct sccb *)CCB);
7524 }
7525
7526 static inline bool
7527 FlashPoint__InterruptPending(void *CardHandle)
7528 {
7529         return FlashPoint_InterruptPending(CardHandle);
7530 }
7531
7532 static inline int
7533 FlashPoint__HandleInterrupt(void *CardHandle)
7534 {
7535         return FlashPoint_HandleInterrupt(CardHandle);
7536 }
7537
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
7545
7546 #else                           /* !CONFIG_SCSI_FLASHPOINT */
7547
7548 /*
7549   Define prototypes for the FlashPoint SCCB Manager Functions.
7550 */
7551
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 *);
7559
7560 #endif                          /* CONFIG_SCSI_FLASHPOINT */