aic79xx.seq revision 1.5 1 1.5 thorpej /* $NetBSD: aic79xx.seq,v 1.5 2003/08/29 00:46:07 thorpej Exp $ */
2 1.4 thorpej
3 1.1 fvdl /*
4 1.1 fvdl * Adaptec U320 device driver firmware for Linux and FreeBSD.
5 1.1 fvdl *
6 1.1 fvdl * Copyright (c) 1994-2001 Justin T. Gibbs.
7 1.1 fvdl * Copyright (c) 2000-2002 Adaptec Inc.
8 1.1 fvdl * All rights reserved.
9 1.1 fvdl *
10 1.1 fvdl * Redistribution and use in source and binary forms, with or without
11 1.1 fvdl * modification, are permitted provided that the following conditions
12 1.1 fvdl * are met:
13 1.1 fvdl * 1. Redistributions of source code must retain the above copyright
14 1.1 fvdl * notice, this list of conditions, and the following disclaimer,
15 1.1 fvdl * without modification.
16 1.1 fvdl * 2. Redistributions in binary form must reproduce at minimum a disclaimer
17 1.1 fvdl * substantially similar to the "NO WARRANTY" disclaimer below
18 1.1 fvdl * ("Disclaimer") and any redistribution must be conditioned upon
19 1.1 fvdl * including a substantially similar Disclaimer requirement for further
20 1.1 fvdl * binary redistribution.
21 1.1 fvdl * 3. Neither the names of the above-listed copyright holders nor the names
22 1.1 fvdl * of any contributors may be used to endorse or promote products derived
23 1.1 fvdl * from this software without specific prior written permission.
24 1.1 fvdl *
25 1.1 fvdl * Alternatively, this software may be distributed under the terms of the
26 1.1 fvdl * GNU General Public License ("GPL") version 2 as published by the Free
27 1.1 fvdl * Software Foundation.
28 1.1 fvdl *
29 1.1 fvdl * NO WARRANTY
30 1.1 fvdl * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
31 1.1 fvdl * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
32 1.1 fvdl * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
33 1.1 fvdl * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
34 1.1 fvdl * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
35 1.1 fvdl * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
36 1.1 fvdl * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
37 1.1 fvdl * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
38 1.1 fvdl * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
39 1.1 fvdl * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
40 1.1 fvdl * POSSIBILITY OF SUCH DAMAGES.
41 1.1 fvdl *
42 1.5 thorpej * $FreeBSD: src/sys/dev/aic7xxx/aic79xx.seq,v 1.10 2003/05/26 21:18:48 gibbs Exp $
43 1.1 fvdl */
44 1.1 fvdl
45 1.5 thorpej VERSION = "Id: //depot/aic7xxx/aic7xxx/aic79xx.seq#93 $"
46 1.1 fvdl PATCH_ARG_LIST = "struct ahd_softc *ahd"
47 1.1 fvdl PREFIX = "ahd_"
48 1.1 fvdl
49 1.1 fvdl #include <dev/microcode/aic7xxx/aic79xx.reg>
50 1.1 fvdl #include <dev/scsipi/scsi_message.h>
51 1.1 fvdl
52 1.1 fvdl restart:
53 1.1 fvdl if ((ahd->bugs & AHD_INTCOLLISION_BUG) != 0) {
54 1.1 fvdl test SEQINTCODE, 0xFF jz idle_loop;
55 1.1 fvdl SET_SEQINTCODE(NO_SEQINT)
56 1.1 fvdl }
57 1.1 fvdl
58 1.1 fvdl idle_loop:
59 1.1 fvdl
60 1.1 fvdl if ((ahd->bugs & AHD_INTCOLLISION_BUG) != 0) {
61 1.1 fvdl /*
62 1.1 fvdl * Convert ERROR status into a sequencer
63 1.1 fvdl * interrupt to handle the case of an
64 1.1 fvdl * interrupt collision on the hardware
65 1.1 fvdl * setting of HWERR.
66 1.1 fvdl */
67 1.1 fvdl test ERROR, 0xFF jz no_error_set;
68 1.1 fvdl SET_SEQINTCODE(SAW_HWERR)
69 1.1 fvdl no_error_set:
70 1.1 fvdl }
71 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
72 1.1 fvdl test SCSISEQ0, ENSELO|ENARBO jnz idle_loop_checkbus;
73 1.1 fvdl test SEQ_FLAGS2, SELECTOUT_QFROZEN jnz idle_loop_checkbus;
74 1.1 fvdl cmp WAITING_TID_HEAD[1], SCB_LIST_NULL je idle_loop_checkbus;
75 1.1 fvdl /*
76 1.1 fvdl * ENSELO is cleared by a SELDO, so we must test for SELDO
77 1.1 fvdl * one last time.
78 1.1 fvdl */
79 1.1 fvdl BEGIN_CRITICAL;
80 1.1 fvdl test SSTAT0, SELDO jnz select_out;
81 1.1 fvdl END_CRITICAL;
82 1.1 fvdl call start_selection;
83 1.1 fvdl idle_loop_checkbus:
84 1.1 fvdl BEGIN_CRITICAL;
85 1.1 fvdl test SSTAT0, SELDO jnz select_out;
86 1.1 fvdl END_CRITICAL;
87 1.1 fvdl test SSTAT0, SELDI jnz select_in;
88 1.1 fvdl test SCSIPHASE, ~DATA_PHASE_MASK jz idle_loop_check_nonpackreq;
89 1.1 fvdl test SCSISIGO, ATNO jz idle_loop_check_nonpackreq;
90 1.1 fvdl call unexpected_nonpkt_phase_find_ctxt;
91 1.1 fvdl idle_loop_check_nonpackreq:
92 1.1 fvdl test SSTAT2, NONPACKREQ jz . + 2;
93 1.1 fvdl call unexpected_nonpkt_phase_find_ctxt;
94 1.4 thorpej if ((ahd->bugs & AHD_FAINT_LED_BUG) != 0) {
95 1.4 thorpej and A, FIFO0FREE|FIFO1FREE, DFFSTAT;
96 1.4 thorpej cmp A, FIFO0FREE|FIFO1FREE jne . + 3;
97 1.4 thorpej and SBLKCTL, ~DIAGLEDEN|DIAGLEDON;
98 1.4 thorpej jmp . + 2;
99 1.4 thorpej or SBLKCTL, DIAGLEDEN|DIAGLEDON;
100 1.4 thorpej }
101 1.1 fvdl call idle_loop_gsfifo_in_scsi_mode;
102 1.1 fvdl call idle_loop_service_fifos;
103 1.1 fvdl call idle_loop_cchan;
104 1.1 fvdl jmp idle_loop;
105 1.1 fvdl
106 1.1 fvdl BEGIN_CRITICAL;
107 1.1 fvdl idle_loop_gsfifo:
108 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
109 1.1 fvdl idle_loop_gsfifo_in_scsi_mode:
110 1.1 fvdl test LQISTAT2, LQIGSAVAIL jz return;
111 1.1 fvdl /*
112 1.1 fvdl * We have received good status for this transaction. There may
113 1.4 thorpej * still be data in our FIFOs draining to the host. Complete
114 1.4 thorpej * the SCB only if all data has transferred to the host.
115 1.1 fvdl */
116 1.1 fvdl good_status_IU_done:
117 1.1 fvdl bmov SCBPTR, GSFIFO, 2;
118 1.1 fvdl clr SCB_SCSI_STATUS;
119 1.1 fvdl /*
120 1.1 fvdl * If a command completed before an attempted task management
121 1.1 fvdl * function completed, notify the host after disabling any
122 1.1 fvdl * pending select-outs.
123 1.1 fvdl */
124 1.1 fvdl test SCB_TASK_MANAGEMENT, 0xFF jz gsfifo_complete_normally;
125 1.1 fvdl test SSTAT0, SELDO|SELINGO jnz . + 2;
126 1.1 fvdl and SCSISEQ0, ~ENSELO;
127 1.1 fvdl SET_SEQINTCODE(TASKMGMT_CMD_CMPLT_OKAY)
128 1.1 fvdl gsfifo_complete_normally:
129 1.1 fvdl or SCB_CONTROL, STATUS_RCVD;
130 1.1 fvdl
131 1.1 fvdl /*
132 1.1 fvdl * Since this status did not consume a FIFO, we have to
133 1.1 fvdl * be a bit more dilligent in how we check for FIFOs pertaining
134 1.1 fvdl * to this transaction. There are two states that a FIFO still
135 1.1 fvdl * transferring data may be in.
136 1.1 fvdl *
137 1.1 fvdl * 1) Configured and draining to the host, with a FIFO handler.
138 1.1 fvdl * 2) Pending cfg4data, fifo not empty.
139 1.1 fvdl *
140 1.4 thorpej * Case 1 can be detected by noticing a non-zero FIFO active
141 1.4 thorpej * count in the SCB. In this case, we allow the routine servicing
142 1.4 thorpej * the FIFO to complete the SCB.
143 1.1 fvdl *
144 1.1 fvdl * Case 2 implies either a pending or yet to occur save data
145 1.1 fvdl * pointers for this same context in the other FIFO. So, if
146 1.1 fvdl * we detect case 1, we will properly defer the post of the SCB
147 1.1 fvdl * and achieve the desired result. The pending cfg4data will
148 1.1 fvdl * notice that status has been received and complete the SCB.
149 1.1 fvdl */
150 1.4 thorpej test SCB_FIFO_USE_COUNT, 0xFF jnz idle_loop_gsfifo_in_scsi_mode;
151 1.1 fvdl call complete;
152 1.1 fvdl END_CRITICAL;
153 1.1 fvdl jmp idle_loop_gsfifo_in_scsi_mode;
154 1.1 fvdl
155 1.1 fvdl idle_loop_service_fifos:
156 1.1 fvdl SET_MODE(M_DFF0, M_DFF0)
157 1.1 fvdl test LONGJMP_ADDR[1], INVALID_ADDR jnz idle_loop_next_fifo;
158 1.1 fvdl call longjmp;
159 1.1 fvdl idle_loop_next_fifo:
160 1.1 fvdl SET_MODE(M_DFF1, M_DFF1)
161 1.1 fvdl test LONGJMP_ADDR[1], INVALID_ADDR jz longjmp;
162 1.4 thorpej return:
163 1.1 fvdl ret;
164 1.1 fvdl
165 1.1 fvdl idle_loop_cchan:
166 1.1 fvdl SET_MODE(M_CCHAN, M_CCHAN)
167 1.1 fvdl test QOFF_CTLSTA, HS_MAILBOX_ACT jz hs_mailbox_empty;
168 1.1 fvdl mov LOCAL_HS_MAILBOX, HS_MAILBOX;
169 1.1 fvdl or QOFF_CTLSTA, HS_MAILBOX_ACT;
170 1.1 fvdl hs_mailbox_empty:
171 1.1 fvdl BEGIN_CRITICAL;
172 1.1 fvdl test CCSCBCTL, CCARREN|CCSCBEN jz scbdma_idle;
173 1.1 fvdl test CCSCBCTL, CCSCBDIR jnz fetch_new_scb_inprog;
174 1.1 fvdl test CCSCBCTL, CCSCBDONE jz return;
175 1.1 fvdl END_CRITICAL;
176 1.1 fvdl /* FALLTHROUGH */
177 1.1 fvdl scbdma_tohost_done:
178 1.1 fvdl test CCSCBCTL, CCARREN jz fill_qoutfifo_dmadone;
179 1.1 fvdl /*
180 1.4 thorpej * An SCB has been succesfully uploaded to the host.
181 1.4 thorpej * If the SCB was uploaded for some reason other than
182 1.4 thorpej * bad SCSI status (currently only for underruns), we
183 1.4 thorpej * queue the SCB for normal completion. Otherwise, we
184 1.4 thorpej * wait until any select-out activity has halted, and
185 1.4 thorpej * then notify the host so that the transaction can be
186 1.4 thorpej * dealt with.
187 1.1 fvdl */
188 1.4 thorpej test SCB_SCSI_STATUS, 0xff jnz scbdma_notify_host;
189 1.4 thorpej and CCSCBCTL, ~(CCARREN|CCSCBEN);
190 1.4 thorpej bmov COMPLETE_DMA_SCB_HEAD, SCB_NEXT_COMPLETE, 2;
191 1.4 thorpej bmov SCB_NEXT_COMPLETE, COMPLETE_SCB_HEAD, 2;
192 1.4 thorpej bmov COMPLETE_SCB_HEAD, SCBPTR, 2 ret;
193 1.4 thorpej scbdma_notify_host:
194 1.4 thorpej SET_MODE(M_SCSI, M_SCSI)
195 1.4 thorpej test SCSISEQ0, ENSELO jnz return;
196 1.4 thorpej test SSTAT0, (SELDO|SELINGO) jnz return;
197 1.4 thorpej SET_MODE(M_CCHAN, M_CCHAN)
198 1.4 thorpej /*
199 1.4 thorpej * Remove SCB and notify host.
200 1.4 thorpej */
201 1.4 thorpej and CCSCBCTL, ~(CCARREN|CCSCBEN);
202 1.4 thorpej bmov COMPLETE_DMA_SCB_HEAD, SCB_NEXT_COMPLETE, 2;
203 1.4 thorpej SET_SEQINTCODE(BAD_SCB_STATUS)
204 1.4 thorpej ret;
205 1.1 fvdl fill_qoutfifo_dmadone:
206 1.1 fvdl and CCSCBCTL, ~(CCARREN|CCSCBEN);
207 1.1 fvdl call qoutfifo_updated;
208 1.1 fvdl mvi COMPLETE_SCB_DMAINPROG_HEAD[1], SCB_LIST_NULL;
209 1.1 fvdl bmov QOUTFIFO_NEXT_ADDR, SCBHADDR, 4;
210 1.1 fvdl test QOFF_CTLSTA, SDSCB_ROLLOVR jz return;
211 1.1 fvdl bmov QOUTFIFO_NEXT_ADDR, SHARED_DATA_ADDR, 4;
212 1.1 fvdl xor QOUTFIFO_ENTRY_VALID_TAG, QOUTFIFO_ENTRY_VALID_TOGGLE ret;
213 1.1 fvdl
214 1.1 fvdl qoutfifo_updated:
215 1.1 fvdl /*
216 1.2 wiz * If there are more commands waiting to be DMA'ed
217 1.3 wiz * to the host, always coalesce. Otherwise honor the
218 1.1 fvdl * host's wishes.
219 1.1 fvdl */
220 1.3 wiz cmp COMPLETE_DMA_SCB_HEAD[1], SCB_LIST_NULL jne coalesce_by_count;
221 1.3 wiz cmp COMPLETE_SCB_HEAD[1], SCB_LIST_NULL jne coalesce_by_count;
222 1.3 wiz test LOCAL_HS_MAILBOX, ENINT_COALESCE jz issue_cmdcmplt;
223 1.1 fvdl
224 1.1 fvdl /*
225 1.1 fvdl * If we have relatively few commands outstanding, don't
226 1.1 fvdl * bother waiting for another command to complete.
227 1.1 fvdl */
228 1.3 wiz test CMDS_PENDING[1], 0xFF jnz coalesce_by_count;
229 1.1 fvdl /* Add -1 so that jnc means <= not just < */
230 1.3 wiz add A, -1, INT_COALESCING_MINCMDS;
231 1.1 fvdl add NONE, A, CMDS_PENDING;
232 1.1 fvdl jnc issue_cmdcmplt;
233 1.1 fvdl
234 1.1 fvdl /*
235 1.3 wiz * If coalescing, only coalesce up to the limit
236 1.1 fvdl * provided by the host driver.
237 1.1 fvdl */
238 1.3 wiz coalesce_by_count:
239 1.3 wiz mov A, INT_COALESCING_MAXCMDS;
240 1.3 wiz add NONE, A, INT_COALESCING_CMDCOUNT;
241 1.1 fvdl jc issue_cmdcmplt;
242 1.1 fvdl /*
243 1.1 fvdl * If the timer is not currently active,
244 1.1 fvdl * fire it up.
245 1.1 fvdl */
246 1.1 fvdl test INTCTL, SWTMINTMASK jz return;
247 1.3 wiz bmov SWTIMER, INT_COALESCING_TIMER, 2;
248 1.1 fvdl mvi CLRSEQINTSTAT, CLRSEQ_SWTMRTO;
249 1.1 fvdl or INTCTL, SWTMINTEN|SWTIMER_START;
250 1.1 fvdl and INTCTL, ~SWTMINTMASK ret;
251 1.1 fvdl
252 1.1 fvdl issue_cmdcmplt:
253 1.1 fvdl mvi INTSTAT, CMDCMPLT;
254 1.3 wiz clr INT_COALESCING_CMDCOUNT;
255 1.1 fvdl or INTCTL, SWTMINTMASK ret;
256 1.1 fvdl
257 1.1 fvdl BEGIN_CRITICAL;
258 1.1 fvdl fetch_new_scb_inprog:
259 1.1 fvdl test CCSCBCTL, ARRDONE jz return;
260 1.1 fvdl fetch_new_scb_done:
261 1.1 fvdl and CCSCBCTL, ~(CCARREN|CCSCBEN);
262 1.1 fvdl bmov REG0, SCBPTR, 2;
263 1.1 fvdl clr A;
264 1.1 fvdl add CMDS_PENDING, 1;
265 1.1 fvdl adc CMDS_PENDING[1], A;
266 1.5 thorpej if ((ahd->bugs & AHD_PKT_LUN_BUG) != 0) {
267 1.5 thorpej /*
268 1.5 thorpej * "Short Luns" are not placed into outgoing LQ
269 1.5 thorpej * packets in the correct byte order. Use a full
270 1.5 thorpej * sized lun field instead and fill it with the
271 1.5 thorpej * one byte of lun information we support.
272 1.5 thorpej */
273 1.5 thorpej mov SCB_PKT_LUN[6], SCB_LUN;
274 1.5 thorpej }
275 1.4 thorpej /*
276 1.4 thorpej * The FIFO use count field is shared with the
277 1.4 thorpej * tag set by the host so that our SCB dma engine
278 1.4 thorpej * knows the correct location to store the SCB.
279 1.4 thorpej * Set it to zero before processing the SCB.
280 1.4 thorpej */
281 1.4 thorpej mov SCB_FIFO_USE_COUNT, ALLZEROS;
282 1.1 fvdl /* Update the next SCB address to download. */
283 1.1 fvdl bmov NEXT_QUEUED_SCB_ADDR, SCB_NEXT_SCB_BUSADDR, 4;
284 1.1 fvdl mvi SCB_NEXT[1], SCB_LIST_NULL;
285 1.1 fvdl mvi SCB_NEXT2[1], SCB_LIST_NULL;
286 1.1 fvdl /* Increment our position in the QINFIFO. */
287 1.1 fvdl mov NONE, SNSCB_QOFF;
288 1.1 fvdl /*
289 1.1 fvdl * SCBs that want to send messages are always
290 1.1 fvdl * queued independently. This ensures that they
291 1.1 fvdl * are at the head of the SCB list to select out
292 1.1 fvdl * to a target and we will see the MK_MESSAGE flag.
293 1.1 fvdl */
294 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jnz first_new_target_scb;
295 1.1 fvdl shr SINDEX, 3, SCB_SCSIID;
296 1.1 fvdl and SINDEX, ~0x1;
297 1.1 fvdl mvi SINDEX[1], (WAITING_SCB_TAILS >> 8);
298 1.1 fvdl bmov DINDEX, SINDEX, 2;
299 1.1 fvdl bmov SCBPTR, SINDIR, 2;
300 1.1 fvdl bmov DINDIR, REG0, 2;
301 1.1 fvdl cmp SCBPTR[1], SCB_LIST_NULL je first_new_target_scb;
302 1.1 fvdl bmov SCB_NEXT, REG0, 2 ret;
303 1.1 fvdl first_new_target_scb:
304 1.1 fvdl cmp WAITING_TID_HEAD[1], SCB_LIST_NULL je first_new_scb;
305 1.1 fvdl bmov SCBPTR, WAITING_TID_TAIL, 2;
306 1.1 fvdl bmov SCB_NEXT2, REG0, 2;
307 1.1 fvdl bmov WAITING_TID_TAIL, REG0, 2 ret;
308 1.1 fvdl first_new_scb:
309 1.1 fvdl bmov WAITING_TID_HEAD, REG0, 2;
310 1.1 fvdl bmov WAITING_TID_TAIL, REG0, 2 ret;
311 1.1 fvdl END_CRITICAL;
312 1.1 fvdl
313 1.1 fvdl scbdma_idle:
314 1.1 fvdl /*
315 1.1 fvdl * Give precedence to downloading new SCBs to execute
316 1.1 fvdl * unless select-outs are currently frozen.
317 1.1 fvdl */
318 1.1 fvdl test SEQ_FLAGS2, SELECTOUT_QFROZEN jnz . + 2;
319 1.1 fvdl BEGIN_CRITICAL;
320 1.1 fvdl test QOFF_CTLSTA, NEW_SCB_AVAIL jnz fetch_new_scb;
321 1.1 fvdl cmp COMPLETE_DMA_SCB_HEAD[1], SCB_LIST_NULL jne dma_complete_scb;
322 1.1 fvdl cmp COMPLETE_SCB_HEAD[1], SCB_LIST_NULL je return;
323 1.1 fvdl /* FALLTHROUGH */
324 1.1 fvdl fill_qoutfifo:
325 1.1 fvdl /*
326 1.2 wiz * Keep track of the SCBs we are DMA'ing just
327 1.1 fvdl * in case the DMA fails or is aborted.
328 1.1 fvdl */
329 1.1 fvdl mov A, QOUTFIFO_ENTRY_VALID_TAG;
330 1.1 fvdl bmov COMPLETE_SCB_DMAINPROG_HEAD, COMPLETE_SCB_HEAD, 2;
331 1.1 fvdl mvi CCSCBCTL, CCSCBRESET;
332 1.1 fvdl bmov SCBHADDR, QOUTFIFO_NEXT_ADDR, 4;
333 1.1 fvdl bmov SCBPTR, COMPLETE_SCB_HEAD, 2;
334 1.1 fvdl fill_qoutfifo_loop:
335 1.1 fvdl mov CCSCBRAM, SCBPTR;
336 1.1 fvdl or CCSCBRAM, A, SCBPTR[1];
337 1.1 fvdl mov NONE, SDSCB_QOFF;
338 1.3 wiz inc INT_COALESCING_CMDCOUNT;
339 1.1 fvdl add CMDS_PENDING, -1;
340 1.1 fvdl adc CMDS_PENDING[1], -1;
341 1.1 fvdl cmp SCB_NEXT_COMPLETE[1], SCB_LIST_NULL je fill_qoutfifo_done;
342 1.1 fvdl cmp CCSCBADDR, CCSCBADDR_MAX je fill_qoutfifo_done;
343 1.1 fvdl test QOFF_CTLSTA, SDSCB_ROLLOVR jnz fill_qoutfifo_done;
344 1.1 fvdl bmov SCBPTR, SCB_NEXT_COMPLETE, 2;
345 1.1 fvdl jmp fill_qoutfifo_loop;
346 1.1 fvdl fill_qoutfifo_done:
347 1.1 fvdl mov SCBHCNT, CCSCBADDR;
348 1.1 fvdl mvi CCSCBCTL, CCSCBEN|CCSCBRESET;
349 1.1 fvdl bmov COMPLETE_SCB_HEAD, SCB_NEXT_COMPLETE, 2;
350 1.1 fvdl mvi SCB_NEXT_COMPLETE[1], SCB_LIST_NULL ret;
351 1.1 fvdl
352 1.1 fvdl fetch_new_scb:
353 1.1 fvdl bmov SCBHADDR, NEXT_QUEUED_SCB_ADDR, 4;
354 1.1 fvdl mvi CCARREN|CCSCBEN|CCSCBDIR|CCSCBRESET jmp dma_scb;
355 1.1 fvdl dma_complete_scb:
356 1.1 fvdl bmov SCBPTR, COMPLETE_DMA_SCB_HEAD, 2;
357 1.1 fvdl bmov SCBHADDR, SCB_BUSADDR, 4;
358 1.4 thorpej mvi CCARREN|CCSCBEN|CCSCBRESET jmp dma_scb;
359 1.1 fvdl END_CRITICAL;
360 1.1 fvdl
361 1.1 fvdl /*
362 1.1 fvdl * Either post or fetch an SCB from host memory. The caller
363 1.1 fvdl * is responsible for polling for transfer completion.
364 1.1 fvdl *
365 1.1 fvdl * Prerequisits: Mode == M_CCHAN
366 1.1 fvdl * SINDEX contains CCSCBCTL flags
367 1.1 fvdl * SCBHADDR set to Host SCB address
368 1.1 fvdl * SCBPTR set to SCB src location on "push" operations
369 1.1 fvdl */
370 1.1 fvdl SET_SRC_MODE M_CCHAN;
371 1.1 fvdl SET_DST_MODE M_CCHAN;
372 1.1 fvdl dma_scb:
373 1.1 fvdl mvi SCBHCNT, SCB_TRANSFER_SIZE;
374 1.1 fvdl mov CCSCBCTL, SINDEX ret;
375 1.1 fvdl
376 1.1 fvdl BEGIN_CRITICAL;
377 1.1 fvdl setjmp:
378 1.1 fvdl bmov LONGJMP_ADDR, STACK, 2 ret;
379 1.1 fvdl setjmp_inline:
380 1.1 fvdl bmov LONGJMP_ADDR, STACK, 2;
381 1.1 fvdl longjmp:
382 1.1 fvdl bmov STACK, LONGJMP_ADDR, 2 ret;
383 1.1 fvdl END_CRITICAL;
384 1.1 fvdl
385 1.1 fvdl /*************************** Chip Bug Work Arounds ****************************/
386 1.1 fvdl /*
387 1.1 fvdl * Must disable interrupts when setting the mode pointer
388 1.1 fvdl * register as an interrupt occurring mid update will
389 1.1 fvdl * fail to store the new mode value for restoration on
390 1.1 fvdl * an iret.
391 1.1 fvdl */
392 1.1 fvdl if ((ahd->bugs & AHD_SET_MODE_BUG) != 0) {
393 1.1 fvdl set_mode_work_around:
394 1.1 fvdl mvi SEQINTCTL, INTVEC1DSL;
395 1.1 fvdl mov MODE_PTR, SINDEX;
396 1.1 fvdl clr SEQINTCTL ret;
397 1.1 fvdl
398 1.1 fvdl toggle_dff_mode_work_around:
399 1.1 fvdl mvi SEQINTCTL, INTVEC1DSL;
400 1.1 fvdl xor MODE_PTR, MK_MODE(M_DFF1, M_DFF1);
401 1.1 fvdl clr SEQINTCTL ret;
402 1.1 fvdl }
403 1.1 fvdl
404 1.1 fvdl
405 1.1 fvdl if ((ahd->bugs & AHD_INTCOLLISION_BUG) != 0) {
406 1.1 fvdl set_seqint_work_around:
407 1.1 fvdl mov SEQINTCODE, SINDEX;
408 1.1 fvdl mvi SEQINTCODE, NO_SEQINT ret;
409 1.1 fvdl }
410 1.1 fvdl
411 1.1 fvdl /************************ Packetized LongJmp Routines *************************/
412 1.1 fvdl SET_SRC_MODE M_SCSI;
413 1.1 fvdl SET_DST_MODE M_SCSI;
414 1.1 fvdl start_selection:
415 1.1 fvdl BEGIN_CRITICAL;
416 1.1 fvdl if ((ahd->bugs & AHD_SENT_SCB_UPDATE_BUG) != 0) {
417 1.1 fvdl /*
418 1.1 fvdl * Razor #494
419 1.1 fvdl * Rev A hardware fails to update LAST/CURR/NEXTSCB
420 1.1 fvdl * correctly after a packetized selection in several
421 1.1 fvdl * situations:
422 1.1 fvdl *
423 1.1 fvdl * 1) If only one command existed in the queue, the
424 1.1 fvdl * LAST/CURR/NEXTSCB are unchanged.
425 1.1 fvdl *
426 1.1 fvdl * 2) In a non QAS, protocol allowed phase change,
427 1.1 fvdl * the queue is shifted 1 too far. LASTSCB is
428 1.1 fvdl * the last SCB that was correctly processed.
429 1.1 fvdl *
430 1.1 fvdl * 3) In the QAS case, if the full list of commands
431 1.1 fvdl * was successfully sent, NEXTSCB is NULL and neither
432 1.1 fvdl * CURRSCB nor LASTSCB can be trusted. We must
433 1.1 fvdl * manually walk the list counting MAXCMDCNT elements
434 1.1 fvdl * to find the last SCB that was sent correctly.
435 1.1 fvdl *
436 1.1 fvdl * To simplify the workaround for this bug in SELDO
437 1.1 fvdl * handling, we initialize LASTSCB prior to enabling
438 1.1 fvdl * selection so we can rely on it even for case #1 above.
439 1.1 fvdl */
440 1.1 fvdl bmov LASTSCB, WAITING_TID_HEAD, 2;
441 1.1 fvdl }
442 1.1 fvdl bmov CURRSCB, WAITING_TID_HEAD, 2;
443 1.1 fvdl bmov SCBPTR, WAITING_TID_HEAD, 2;
444 1.1 fvdl shr SELOID, 4, SCB_SCSIID;
445 1.1 fvdl /*
446 1.1 fvdl * If we want to send a message to the device, ensure
447 1.1 fvdl * we are selecting with atn irregardless of our packetized
448 1.1 fvdl * agreement. Since SPI4 only allows target reset or PPR
449 1.1 fvdl * messages if this is a packetized connection, the change
450 1.1 fvdl * to our negotiation table entry for this selection will
451 1.1 fvdl * be cleared when the message is acted on.
452 1.1 fvdl */
453 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jz . + 3;
454 1.1 fvdl mov NEGOADDR, SELOID;
455 1.1 fvdl or NEGCONOPTS, ENAUTOATNO;
456 1.1 fvdl or SCSISEQ0, ENSELO ret;
457 1.1 fvdl END_CRITICAL;
458 1.1 fvdl
459 1.1 fvdl /*
460 1.1 fvdl * Allocate a FIFO for a non-packetized transaction.
461 1.1 fvdl * In RevA hardware, both FIFOs must be free before we
462 1.1 fvdl * can allocate a FIFO for a non-packetized transaction.
463 1.1 fvdl */
464 1.1 fvdl allocate_fifo_loop:
465 1.1 fvdl /*
466 1.1 fvdl * Do whatever work is required to free a FIFO.
467 1.1 fvdl */
468 1.1 fvdl call idle_loop_service_fifos;
469 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
470 1.1 fvdl allocate_fifo:
471 1.1 fvdl if ((ahd->bugs & AHD_NONPACKFIFO_BUG) != 0) {
472 1.1 fvdl and A, FIFO0FREE|FIFO1FREE, DFFSTAT;
473 1.1 fvdl cmp A, FIFO0FREE|FIFO1FREE jne allocate_fifo_loop;
474 1.1 fvdl } else {
475 1.1 fvdl test DFFSTAT, FIFO1FREE jnz allocate_fifo1;
476 1.1 fvdl test DFFSTAT, FIFO0FREE jz allocate_fifo_loop;
477 1.1 fvdl mvi DFFSTAT, B_CURRFIFO_0;
478 1.1 fvdl SET_MODE(M_DFF0, M_DFF0)
479 1.1 fvdl bmov SCBPTR, ALLOCFIFO_SCBPTR, 2 ret;
480 1.1 fvdl }
481 1.1 fvdl SET_SRC_MODE M_SCSI;
482 1.1 fvdl SET_DST_MODE M_SCSI;
483 1.1 fvdl allocate_fifo1:
484 1.1 fvdl mvi DFFSTAT, CURRFIFO_1;
485 1.1 fvdl SET_MODE(M_DFF1, M_DFF1)
486 1.1 fvdl bmov SCBPTR, ALLOCFIFO_SCBPTR, 2 ret;
487 1.1 fvdl
488 1.1 fvdl /*
489 1.1 fvdl * We have been reselected as an initiator
490 1.1 fvdl * or selected as a target.
491 1.1 fvdl */
492 1.1 fvdl SET_SRC_MODE M_SCSI;
493 1.1 fvdl SET_DST_MODE M_SCSI;
494 1.1 fvdl select_in:
495 1.1 fvdl if ((ahd->bugs & AHD_BUSFREEREV_BUG) != 0) {
496 1.1 fvdl /*
497 1.1 fvdl * This exposes a window whereby a
498 1.1 fvdl * busfree just after a selection will
499 1.1 fvdl * be missed, but there is no other safe
500 1.1 fvdl * way to enable busfree detection if
501 1.1 fvdl * the busfreerev function is broken.
502 1.1 fvdl */
503 1.1 fvdl mvi CLRSINT1,CLRBUSFREE;
504 1.1 fvdl or SIMODE1, ENBUSFREE;
505 1.1 fvdl }
506 1.1 fvdl or SXFRCTL0, SPIOEN;
507 1.1 fvdl and SAVED_SCSIID, SELID_MASK, SELID;
508 1.1 fvdl and A, OID, IOWNID;
509 1.1 fvdl or SAVED_SCSIID, A;
510 1.1 fvdl mvi CLRSINT0, CLRSELDI;
511 1.1 fvdl jmp ITloop;
512 1.1 fvdl
513 1.1 fvdl /*
514 1.1 fvdl * We have successfully selected out.
515 1.1 fvdl *
516 1.1 fvdl * Clear SELDO.
517 1.1 fvdl * Dequeue all SCBs sent from the waiting queue
518 1.1 fvdl * Requeue all SCBs *not* sent to the tail of the waiting queue
519 1.1 fvdl * Take Razor #494 into account for above.
520 1.1 fvdl *
521 1.1 fvdl * In Packetized Mode:
522 1.1 fvdl * Return to the idle loop. Our interrupt handler will take
523 1.1 fvdl * care of any incoming L_Qs.
524 1.1 fvdl *
525 1.1 fvdl * In Non-Packetize Mode:
526 1.1 fvdl * Continue to our normal state machine.
527 1.1 fvdl */
528 1.1 fvdl SET_SRC_MODE M_SCSI;
529 1.1 fvdl SET_DST_MODE M_SCSI;
530 1.1 fvdl select_out:
531 1.1 fvdl BEGIN_CRITICAL;
532 1.1 fvdl /* Clear out all SCBs that have been successfully sent. */
533 1.1 fvdl if ((ahd->bugs & AHD_SENT_SCB_UPDATE_BUG) != 0) {
534 1.1 fvdl /*
535 1.1 fvdl * For packetized, the LQO manager clears ENSELO on
536 1.1 fvdl * the assertion of SELDO. If we are non-packetized,
537 1.1 fvdl * LASTSCB and CURRSCB are accurate.
538 1.1 fvdl */
539 1.1 fvdl test SCSISEQ0, ENSELO jnz use_lastscb;
540 1.1 fvdl
541 1.1 fvdl /*
542 1.1 fvdl * The update is correct for LQOSTAT1 errors. All
543 1.1 fvdl * but LQOBUSFREE are handled by kernel interrupts.
544 1.1 fvdl * If we see LQOBUSFREE, return to the idle loop.
545 1.1 fvdl * Once we are out of the select_out critical section,
546 1.1 fvdl * the kernel will cleanup the LQOBUSFREE and we will
547 1.1 fvdl * eventually restart the selection if appropriate.
548 1.1 fvdl */
549 1.1 fvdl test LQOSTAT1, LQOBUSFREE jnz idle_loop;
550 1.1 fvdl
551 1.1 fvdl /*
552 1.1 fvdl * On a phase change oustside of packet boundaries,
553 1.1 fvdl * LASTSCB points to the currently active SCB context
554 1.1 fvdl * on the bus.
555 1.1 fvdl */
556 1.1 fvdl test LQOSTAT2, LQOPHACHGOUTPKT jnz use_lastscb;
557 1.1 fvdl
558 1.1 fvdl /*
559 1.1 fvdl * If the hardware has traversed the whole list, NEXTSCB
560 1.1 fvdl * will be NULL, CURRSCB and LASTSCB cannot be trusted,
561 1.1 fvdl * but MAXCMDCNT is accurate. If we stop part way through
562 1.1 fvdl * the list or only had one command to issue, NEXTSCB[1] is
563 1.1 fvdl * not NULL and LASTSCB is the last command to go out.
564 1.1 fvdl */
565 1.1 fvdl cmp NEXTSCB[1], SCB_LIST_NULL jne use_lastscb;
566 1.1 fvdl
567 1.1 fvdl /*
568 1.1 fvdl * Brute force walk.
569 1.1 fvdl */
570 1.1 fvdl bmov SCBPTR, WAITING_TID_HEAD, 2;
571 1.1 fvdl mvi SEQINTCTL, INTVEC1DSL;
572 1.1 fvdl mvi MODE_PTR, MK_MODE(M_CFG, M_CFG);
573 1.1 fvdl mov A, MAXCMDCNT;
574 1.1 fvdl mvi MODE_PTR, MK_MODE(M_SCSI, M_SCSI);
575 1.1 fvdl clr SEQINTCTL;
576 1.1 fvdl find_lastscb_loop:
577 1.1 fvdl dec A;
578 1.1 fvdl test A, 0xFF jz found_last_sent_scb;
579 1.1 fvdl bmov SCBPTR, SCB_NEXT, 2;
580 1.1 fvdl jmp find_lastscb_loop;
581 1.1 fvdl use_lastscb:
582 1.1 fvdl bmov SCBPTR, LASTSCB, 2;
583 1.1 fvdl found_last_sent_scb:
584 1.1 fvdl bmov CURRSCB, SCBPTR, 2;
585 1.1 fvdl curscb_ww_done:
586 1.1 fvdl } else {
587 1.1 fvdl /*
588 1.1 fvdl * Untested - Verify with Rev B.
589 1.1 fvdl */
590 1.1 fvdl bmov SCBPTR, CURRSCB, 2;
591 1.1 fvdl }
592 1.1 fvdl
593 1.1 fvdl /*
594 1.1 fvdl * Requeue any SCBs not sent, to the tail of the waiting Q.
595 1.1 fvdl */
596 1.1 fvdl cmp SCB_NEXT[1], SCB_LIST_NULL je select_out_list_done;
597 1.1 fvdl
598 1.1 fvdl /*
599 1.1 fvdl * We know that neither the per-TID list nor the list of
600 1.1 fvdl * TIDs is empty. Use this knowledge to our advantage.
601 1.1 fvdl */
602 1.1 fvdl bmov REG0, SCB_NEXT, 2;
603 1.1 fvdl bmov SCBPTR, WAITING_TID_TAIL, 2;
604 1.1 fvdl bmov SCB_NEXT2, REG0, 2;
605 1.1 fvdl bmov WAITING_TID_TAIL, REG0, 2;
606 1.1 fvdl jmp select_out_inc_tid_q;
607 1.1 fvdl
608 1.1 fvdl select_out_list_done:
609 1.1 fvdl /*
610 1.1 fvdl * The whole list made it. Just clear our TID's tail pointer
611 1.1 fvdl * unless we were queued independently due to our need to
612 1.1 fvdl * send a message.
613 1.1 fvdl */
614 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jnz select_out_inc_tid_q;
615 1.1 fvdl shr DINDEX, 3, SCB_SCSIID;
616 1.1 fvdl or DINDEX, 1; /* Want only the second byte */
617 1.1 fvdl mvi DINDEX[1], ((WAITING_SCB_TAILS) >> 8);
618 1.1 fvdl mvi DINDIR, SCB_LIST_NULL;
619 1.1 fvdl select_out_inc_tid_q:
620 1.1 fvdl bmov SCBPTR, WAITING_TID_HEAD, 2;
621 1.1 fvdl bmov WAITING_TID_HEAD, SCB_NEXT2, 2;
622 1.1 fvdl cmp WAITING_TID_HEAD[1], SCB_LIST_NULL jne . + 2;
623 1.1 fvdl mvi WAITING_TID_TAIL[1], SCB_LIST_NULL;
624 1.1 fvdl bmov SCBPTR, CURRSCB, 2;
625 1.1 fvdl mvi CLRSINT0, CLRSELDO;
626 1.1 fvdl test LQOSTAT2, LQOPHACHGOUTPKT jnz unexpected_nonpkt_phase;
627 1.1 fvdl test LQOSTAT1, LQOPHACHGINPKT jnz unexpected_nonpkt_phase;
628 1.1 fvdl
629 1.1 fvdl /*
630 1.1 fvdl * If this is a packetized connection, return to our
631 1.1 fvdl * idle_loop and let our interrupt handler deal with
632 1.1 fvdl * any connection setup/teardown issues. The only
633 1.1 fvdl * exceptions are the case of MK_MESSAGE and task management
634 1.1 fvdl * SCBs.
635 1.1 fvdl */
636 1.1 fvdl if ((ahd->bugs & AHD_LQO_ATNO_BUG) != 0) {
637 1.1 fvdl /*
638 1.1 fvdl * In the A, the LQO manager transitions to LQOSTOP0 even if
639 1.1 fvdl * we have selected out with ATN asserted and the target
640 1.1 fvdl * REQs in a non-packet phase.
641 1.1 fvdl */
642 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jz select_out_no_message;
643 1.1 fvdl test SCSISIGO, ATNO jnz select_out_non_packetized;
644 1.1 fvdl select_out_no_message:
645 1.1 fvdl }
646 1.1 fvdl test LQOSTAT2, LQOSTOP0 jz select_out_non_packetized;
647 1.1 fvdl test SCB_TASK_MANAGEMENT, 0xFF jz idle_loop;
648 1.1 fvdl SET_SEQINTCODE(TASKMGMT_FUNC_COMPLETE)
649 1.1 fvdl jmp idle_loop;
650 1.1 fvdl
651 1.1 fvdl select_out_non_packetized:
652 1.1 fvdl /* Non packetized request. */
653 1.1 fvdl and SCSISEQ0, ~ENSELO;
654 1.1 fvdl if ((ahd->bugs & AHD_BUSFREEREV_BUG) != 0) {
655 1.1 fvdl /*
656 1.1 fvdl * This exposes a window whereby a
657 1.1 fvdl * busfree just after a selection will
658 1.1 fvdl * be missed, but there is no other safe
659 1.1 fvdl * way to enable busfree detection if
660 1.1 fvdl * the busfreerev function is broken.
661 1.1 fvdl */
662 1.1 fvdl mvi CLRSINT1,CLRBUSFREE;
663 1.1 fvdl or SIMODE1, ENBUSFREE;
664 1.1 fvdl }
665 1.1 fvdl mov SAVED_SCSIID, SCB_SCSIID;
666 1.1 fvdl mov SAVED_LUN, SCB_LUN;
667 1.1 fvdl mvi SEQ_FLAGS, NO_CDB_SENT;
668 1.1 fvdl END_CRITICAL;
669 1.1 fvdl or SXFRCTL0, SPIOEN;
670 1.1 fvdl
671 1.1 fvdl /*
672 1.1 fvdl * As soon as we get a successful selection, the target
673 1.1 fvdl * should go into the message out phase since we have ATN
674 1.1 fvdl * asserted.
675 1.1 fvdl */
676 1.1 fvdl mvi MSG_OUT, MSG_IDENTIFYFLAG;
677 1.1 fvdl
678 1.1 fvdl /*
679 1.1 fvdl * Main loop for information transfer phases. Wait for the
680 1.1 fvdl * target to assert REQ before checking MSG, C/D and I/O for
681 1.1 fvdl * the bus phase.
682 1.1 fvdl */
683 1.1 fvdl mesgin_phasemis:
684 1.1 fvdl ITloop:
685 1.1 fvdl call phase_lock;
686 1.1 fvdl
687 1.1 fvdl mov A, LASTPHASE;
688 1.1 fvdl
689 1.1 fvdl test A, ~P_DATAIN_DT jz p_data;
690 1.1 fvdl cmp A,P_COMMAND je p_command;
691 1.1 fvdl cmp A,P_MESGOUT je p_mesgout;
692 1.1 fvdl cmp A,P_STATUS je p_status;
693 1.1 fvdl cmp A,P_MESGIN je p_mesgin;
694 1.1 fvdl
695 1.1 fvdl SET_SEQINTCODE(BAD_PHASE)
696 1.1 fvdl jmp ITloop; /* Try reading the bus again. */
697 1.1 fvdl
698 1.1 fvdl /*
699 1.1 fvdl * Command phase. Set up the DMA registers and let 'er rip.
700 1.1 fvdl */
701 1.1 fvdl p_command:
702 1.1 fvdl test SEQ_FLAGS, NOT_IDENTIFIED jz p_command_okay;
703 1.1 fvdl SET_SEQINTCODE(PROTO_VIOLATION)
704 1.1 fvdl p_command_okay:
705 1.1 fvdl test MODE_PTR, ~(MK_MODE(M_DFF1, M_DFF1))
706 1.1 fvdl jnz p_command_allocate_fifo;
707 1.1 fvdl /*
708 1.1 fvdl * Command retry. Free our current FIFO and
709 1.1 fvdl * re-allocate a FIFO so transfer state is
710 1.1 fvdl * reset.
711 1.1 fvdl */
712 1.1 fvdl SET_SRC_MODE M_DFF1;
713 1.1 fvdl SET_DST_MODE M_DFF1;
714 1.1 fvdl mvi DFFSXFRCTL, RSTCHN|CLRSHCNT;
715 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
716 1.1 fvdl p_command_allocate_fifo:
717 1.1 fvdl bmov ALLOCFIFO_SCBPTR, SCBPTR, 2;
718 1.1 fvdl call allocate_fifo;
719 1.1 fvdl SET_SRC_MODE M_DFF1;
720 1.1 fvdl SET_DST_MODE M_DFF1;
721 1.1 fvdl add NONE, -17, SCB_CDB_LEN;
722 1.1 fvdl jnc p_command_embedded;
723 1.1 fvdl p_command_from_host:
724 1.1 fvdl bmov HADDR[0], SCB_HOST_CDB_PTR, 9;
725 1.1 fvdl mvi SG_CACHE_PRE, LAST_SEG;
726 1.1 fvdl mvi DFCNTRL, (PRELOADEN|SCSIEN|HDMAEN);
727 1.1 fvdl jmp p_command_xfer;
728 1.1 fvdl p_command_embedded:
729 1.1 fvdl bmov SHCNT[0], SCB_CDB_LEN, 1;
730 1.1 fvdl bmov DFDAT, SCB_CDB_STORE, 16;
731 1.1 fvdl mvi DFCNTRL, SCSIEN;
732 1.1 fvdl p_command_xfer:
733 1.1 fvdl and SEQ_FLAGS, ~NO_CDB_SENT;
734 1.1 fvdl test DFCNTRL, SCSIEN jnz .;
735 1.1 fvdl /*
736 1.1 fvdl * DMA Channel automatically disabled.
737 1.1 fvdl * Don't allow a data phase if the command
738 1.1 fvdl * was not fully transferred.
739 1.1 fvdl */
740 1.1 fvdl test SSTAT2, SDONE jnz ITloop;
741 1.1 fvdl or SEQ_FLAGS, NO_CDB_SENT;
742 1.1 fvdl jmp ITloop;
743 1.1 fvdl
744 1.1 fvdl
745 1.1 fvdl /*
746 1.1 fvdl * Status phase. Wait for the data byte to appear, then read it
747 1.1 fvdl * and store it into the SCB.
748 1.1 fvdl */
749 1.1 fvdl SET_SRC_MODE M_SCSI;
750 1.1 fvdl SET_DST_MODE M_SCSI;
751 1.1 fvdl p_status:
752 1.1 fvdl test SEQ_FLAGS,NOT_IDENTIFIED jnz mesgin_proto_violation;
753 1.1 fvdl p_status_okay:
754 1.1 fvdl mov SCB_SCSI_STATUS, SCSIDAT;
755 1.1 fvdl or SCB_CONTROL, STATUS_RCVD;
756 1.1 fvdl jmp ITloop;
757 1.1 fvdl
758 1.1 fvdl /*
759 1.1 fvdl * Message out phase. If MSG_OUT is MSG_IDENTIFYFLAG, build a full
760 1.1 fvdl * indentify message sequence and send it to the target. The host may
761 1.1 fvdl * override this behavior by setting the MK_MESSAGE bit in the SCB
762 1.1 fvdl * control byte. This will cause us to interrupt the host and allow
763 1.1 fvdl * it to handle the message phase completely on its own. If the bit
764 1.1 fvdl * associated with this target is set, we will also interrupt the host,
765 1.1 fvdl * thereby allowing it to send a message on the next selection regardless
766 1.1 fvdl * of the transaction being sent.
767 1.1 fvdl *
768 1.1 fvdl * If MSG_OUT is == HOST_MSG, also interrupt the host and take a message.
769 1.1 fvdl * This is done to allow the host to send messages outside of an identify
770 1.1 fvdl * sequence while protecting the seqencer from testing the MK_MESSAGE bit
771 1.1 fvdl * on an SCB that might not be for the current nexus. (For example, a
772 1.1 fvdl * BDR message in responce to a bad reselection would leave us pointed to
773 1.1 fvdl * an SCB that doesn't have anything to do with the current target).
774 1.1 fvdl *
775 1.1 fvdl * Otherwise, treat MSG_OUT as a 1 byte message to send (abort, abort tag,
776 1.1 fvdl * bus device reset).
777 1.1 fvdl *
778 1.1 fvdl * When there are no messages to send, MSG_OUT should be set to MSG_NOOP,
779 1.1 fvdl * in case the target decides to put us in this phase for some strange
780 1.1 fvdl * reason.
781 1.1 fvdl */
782 1.1 fvdl p_mesgout_retry:
783 1.1 fvdl /* Turn on ATN for the retry */
784 1.1 fvdl mvi SCSISIGO, ATNO;
785 1.1 fvdl p_mesgout:
786 1.1 fvdl mov SINDEX, MSG_OUT;
787 1.1 fvdl cmp SINDEX, MSG_IDENTIFYFLAG jne p_mesgout_from_host;
788 1.1 fvdl test SCB_CONTROL,MK_MESSAGE jnz host_message_loop;
789 1.1 fvdl p_mesgout_identify:
790 1.1 fvdl or SINDEX, MSG_IDENTIFYFLAG|DISCENB, SCB_LUN;
791 1.1 fvdl test SCB_CONTROL, DISCENB jnz . + 2;
792 1.1 fvdl and SINDEX, ~DISCENB;
793 1.1 fvdl /*
794 1.1 fvdl * Send a tag message if TAG_ENB is set in the SCB control block.
795 1.1 fvdl * Use SCB_NONPACKET_TAG as the tag value.
796 1.1 fvdl */
797 1.1 fvdl p_mesgout_tag:
798 1.1 fvdl test SCB_CONTROL,TAG_ENB jz p_mesgout_onebyte;
799 1.1 fvdl mov SCSIDAT, SINDEX; /* Send the identify message */
800 1.1 fvdl call phase_lock;
801 1.1 fvdl cmp LASTPHASE, P_MESGOUT jne p_mesgout_done;
802 1.1 fvdl and SCSIDAT,TAG_ENB|SCB_TAG_TYPE,SCB_CONTROL;
803 1.1 fvdl call phase_lock;
804 1.1 fvdl cmp LASTPHASE, P_MESGOUT jne p_mesgout_done;
805 1.1 fvdl mov SCBPTR jmp p_mesgout_onebyte;
806 1.1 fvdl /*
807 1.1 fvdl * Interrupt the driver, and allow it to handle this message
808 1.1 fvdl * phase and any required retries.
809 1.1 fvdl */
810 1.1 fvdl p_mesgout_from_host:
811 1.1 fvdl cmp SINDEX, HOST_MSG jne p_mesgout_onebyte;
812 1.1 fvdl jmp host_message_loop;
813 1.1 fvdl
814 1.1 fvdl p_mesgout_onebyte:
815 1.1 fvdl mvi CLRSINT1, CLRATNO;
816 1.1 fvdl mov SCSIDAT, SINDEX;
817 1.1 fvdl
818 1.1 fvdl /*
819 1.1 fvdl * If the next bus phase after ATN drops is message out, it means
820 1.1 fvdl * that the target is requesting that the last message(s) be resent.
821 1.1 fvdl */
822 1.1 fvdl call phase_lock;
823 1.1 fvdl cmp LASTPHASE, P_MESGOUT je p_mesgout_retry;
824 1.1 fvdl
825 1.1 fvdl p_mesgout_done:
826 1.1 fvdl mvi CLRSINT1,CLRATNO; /* Be sure to turn ATNO off */
827 1.1 fvdl mov LAST_MSG, MSG_OUT;
828 1.1 fvdl mvi MSG_OUT, MSG_NOOP; /* No message left */
829 1.1 fvdl jmp ITloop;
830 1.1 fvdl
831 1.1 fvdl /*
832 1.1 fvdl * Message in phase. Bytes are read using Automatic PIO mode.
833 1.1 fvdl */
834 1.1 fvdl p_mesgin:
835 1.1 fvdl /* read the 1st message byte */
836 1.1 fvdl mvi ACCUM call inb_first;
837 1.1 fvdl
838 1.1 fvdl test A,MSG_IDENTIFYFLAG jnz mesgin_identify;
839 1.1 fvdl cmp A,MSG_DISCONNECT je mesgin_disconnect;
840 1.1 fvdl cmp A,MSG_SAVEDATAPOINTER je mesgin_sdptrs;
841 1.1 fvdl cmp ALLZEROS,A je mesgin_complete;
842 1.1 fvdl cmp A,MSG_RESTOREPOINTERS je mesgin_rdptrs;
843 1.1 fvdl cmp A,MSG_IGN_WIDE_RESIDUE je mesgin_ign_wide_residue;
844 1.1 fvdl cmp A,MSG_NOOP je mesgin_done;
845 1.1 fvdl
846 1.1 fvdl /*
847 1.1 fvdl * Pushed message loop to allow the kernel to
848 1.1 fvdl * run it's own message state engine. To avoid an
849 1.1 fvdl * extra nop instruction after signaling the kernel,
850 1.1 fvdl * we perform the phase_lock before checking to see
851 1.1 fvdl * if we should exit the loop and skip the phase_lock
852 1.1 fvdl * in the ITloop. Performing back to back phase_locks
853 1.1 fvdl * shouldn't hurt, but why do it twice...
854 1.1 fvdl */
855 1.1 fvdl host_message_loop:
856 1.1 fvdl call phase_lock; /* Benign the first time through. */
857 1.1 fvdl SET_SEQINTCODE(HOST_MSG_LOOP)
858 1.1 fvdl cmp RETURN_1, EXIT_MSG_LOOP je ITloop;
859 1.1 fvdl cmp RETURN_1, CONT_MSG_LOOP_WRITE jne . + 3;
860 1.1 fvdl mov SCSIDAT, RETURN_2;
861 1.1 fvdl jmp host_message_loop;
862 1.1 fvdl /* Must be CONT_MSG_LOOP_READ */
863 1.1 fvdl mov NONE, SCSIDAT; /* ACK Byte */
864 1.1 fvdl jmp host_message_loop;
865 1.1 fvdl
866 1.1 fvdl mesgin_ign_wide_residue:
867 1.1 fvdl mov SAVED_MODE, MODE_PTR;
868 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
869 1.1 fvdl shr NEGOADDR, 4, SAVED_SCSIID;
870 1.1 fvdl mov A, NEGCONOPTS;
871 1.1 fvdl RESTORE_MODE(SAVED_MODE)
872 1.1 fvdl test A, WIDEXFER jz mesgin_reject;
873 1.1 fvdl /* Pull the residue byte */
874 1.1 fvdl mvi REG0 call inb_next;
875 1.1 fvdl cmp REG0, 0x01 jne mesgin_reject;
876 1.1 fvdl test SCB_RESIDUAL_SGPTR[0], SG_LIST_NULL jz . + 2;
877 1.1 fvdl test DATA_COUNT_ODD, 0x1 jz mesgin_done;
878 1.1 fvdl jmp mesgin_done;
879 1.1 fvdl
880 1.1 fvdl mesgin_proto_violation:
881 1.1 fvdl SET_SEQINTCODE(PROTO_VIOLATION)
882 1.1 fvdl jmp mesgin_done;
883 1.1 fvdl mesgin_reject:
884 1.1 fvdl mvi MSG_MESSAGE_REJECT call mk_mesg;
885 1.1 fvdl mesgin_done:
886 1.1 fvdl mov NONE,SCSIDAT; /*dummy read from latch to ACK*/
887 1.1 fvdl jmp ITloop;
888 1.1 fvdl
889 1.1 fvdl #define INDEX_DISC_LIST(scsiid, lun) \
890 1.1 fvdl and A, 0xC0, scsiid; \
891 1.1 fvdl or SCBPTR, A, lun; \
892 1.1 fvdl clr SCBPTR[1]; \
893 1.1 fvdl and SINDEX, 0x30, scsiid; \
894 1.1 fvdl shr SINDEX, 3; /* Multiply by 2 */ \
895 1.1 fvdl add SINDEX, (SCB_DISCONNECTED_LISTS & 0xFF); \
896 1.1 fvdl mvi SINDEX[1], ((SCB_DISCONNECTED_LISTS >> 8) & 0xFF)
897 1.1 fvdl
898 1.1 fvdl mesgin_identify:
899 1.1 fvdl /*
900 1.1 fvdl * Determine whether a target is using tagged or non-tagged
901 1.1 fvdl * transactions by first looking at the transaction stored in
902 1.1 fvdl * the per-device, disconnected array. If there is no untagged
903 1.1 fvdl * transaction for this target, this must be a tagged transaction.
904 1.1 fvdl */
905 1.1 fvdl and SAVED_LUN, MSG_IDENTIFY_LUNMASK, A;
906 1.1 fvdl INDEX_DISC_LIST(SAVED_SCSIID, SAVED_LUN);
907 1.1 fvdl bmov DINDEX, SINDEX, 2;
908 1.1 fvdl bmov REG0, SINDIR, 2;
909 1.1 fvdl cmp REG0[1], SCB_LIST_NULL je snoop_tag;
910 1.1 fvdl /* Untagged. Clear the busy table entry and setup the SCB. */
911 1.1 fvdl bmov DINDIR, ALLONES, 2;
912 1.1 fvdl bmov SCBPTR, REG0, 2;
913 1.1 fvdl jmp setup_SCB;
914 1.1 fvdl
915 1.1 fvdl /*
916 1.1 fvdl * Here we "snoop" the bus looking for a SIMPLE QUEUE TAG message.
917 1.1 fvdl * If we get one, we use the tag returned to find the proper
918 1.1 fvdl * SCB. After receiving the tag, look for the SCB at SCB locations tag and
919 1.1 fvdl * tag + 256.
920 1.1 fvdl */
921 1.1 fvdl snoop_tag:
922 1.1 fvdl if ((ahd->flags & AHD_SEQUENCER_DEBUG) != 0) {
923 1.1 fvdl or SEQ_FLAGS, 0x80;
924 1.1 fvdl }
925 1.1 fvdl mov NONE, SCSIDAT; /* ACK Identify MSG */
926 1.1 fvdl call phase_lock;
927 1.1 fvdl if ((ahd->flags & AHD_SEQUENCER_DEBUG) != 0) {
928 1.1 fvdl or SEQ_FLAGS, 0x1;
929 1.1 fvdl }
930 1.1 fvdl cmp LASTPHASE, P_MESGIN jne not_found_ITloop;
931 1.1 fvdl if ((ahd->flags & AHD_SEQUENCER_DEBUG) != 0) {
932 1.1 fvdl or SEQ_FLAGS, 0x2;
933 1.1 fvdl }
934 1.1 fvdl cmp SCSIBUS, MSG_SIMPLE_Q_TAG jne not_found;
935 1.1 fvdl get_tag:
936 1.1 fvdl clr SCBPTR[1];
937 1.1 fvdl mvi SCBPTR call inb_next; /* tag value */
938 1.1 fvdl verify_scb:
939 1.1 fvdl test SCB_CONTROL,DISCONNECTED jz verify_other_scb;
940 1.1 fvdl mov A, SAVED_SCSIID;
941 1.1 fvdl cmp SCB_SCSIID, A jne verify_other_scb;
942 1.1 fvdl mov A, SAVED_LUN;
943 1.1 fvdl cmp SCB_LUN, A je setup_SCB_disconnected;
944 1.1 fvdl verify_other_scb:
945 1.1 fvdl xor SCBPTR[1], 1;
946 1.1 fvdl test SCBPTR[1], 0xFF jnz verify_scb;
947 1.1 fvdl jmp not_found;
948 1.1 fvdl
949 1.1 fvdl /*
950 1.1 fvdl * Ensure that the SCB the tag points to is for
951 1.1 fvdl * an SCB transaction to the reconnecting target.
952 1.1 fvdl */
953 1.1 fvdl setup_SCB:
954 1.1 fvdl if ((ahd->flags & AHD_SEQUENCER_DEBUG) != 0) {
955 1.1 fvdl or SEQ_FLAGS, 0x10;
956 1.1 fvdl }
957 1.1 fvdl test SCB_CONTROL,DISCONNECTED jz not_found;
958 1.1 fvdl setup_SCB_disconnected:
959 1.1 fvdl and SCB_CONTROL,~DISCONNECTED;
960 1.1 fvdl clr SEQ_FLAGS; /* make note of IDENTIFY */
961 1.1 fvdl test SCB_SGPTR, SG_LIST_NULL jnz . + 3;
962 1.1 fvdl bmov ALLOCFIFO_SCBPTR, SCBPTR, 2;
963 1.1 fvdl call allocate_fifo;
964 1.1 fvdl /* See if the host wants to send a message upon reconnection */
965 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jz mesgin_done;
966 1.1 fvdl mvi HOST_MSG call mk_mesg;
967 1.1 fvdl jmp mesgin_done;
968 1.1 fvdl
969 1.1 fvdl not_found:
970 1.1 fvdl SET_SEQINTCODE(NO_MATCH)
971 1.1 fvdl jmp mesgin_done;
972 1.1 fvdl
973 1.1 fvdl not_found_ITloop:
974 1.1 fvdl SET_SEQINTCODE(NO_MATCH)
975 1.1 fvdl jmp ITloop;
976 1.1 fvdl
977 1.1 fvdl /*
978 1.1 fvdl * We received a "command complete" message. Put the SCB on the complete
979 1.1 fvdl * queue and trigger a completion interrupt via the idle loop. Before doing
980 1.1 fvdl * so, check to see if there
981 1.1 fvdl * is a residual or the status byte is something other than STATUS_GOOD (0).
982 1.1 fvdl * In either of these conditions, we upload the SCB back to the host so it can
983 1.1 fvdl * process this information. In the case of a non zero status byte, we
984 1.1 fvdl * additionally interrupt the kernel driver synchronously, allowing it to
985 1.1 fvdl * decide if sense should be retrieved. If the kernel driver wishes to request
986 1.1 fvdl * sense, it will fill the kernel SCB with a request sense command, requeue
987 1.1 fvdl * it to the QINFIFO and tell us not to post to the QOUTFIFO by setting
988 1.1 fvdl * RETURN_1 to SEND_SENSE.
989 1.1 fvdl */
990 1.1 fvdl mesgin_complete:
991 1.1 fvdl
992 1.1 fvdl /*
993 1.1 fvdl * If ATN is raised, we still want to give the target a message.
994 1.1 fvdl * Perhaps there was a parity error on this last message byte.
995 1.1 fvdl * Either way, the target should take us to message out phase
996 1.1 fvdl * and then attempt to complete the command again. We should use a
997 1.1 fvdl * critical section here to guard against a timeout triggering
998 1.1 fvdl * for this command and setting ATN while we are still processing
999 1.1 fvdl * the completion.
1000 1.1 fvdl test SCSISIGI, ATNI jnz mesgin_done;
1001 1.1 fvdl */
1002 1.1 fvdl
1003 1.1 fvdl /*
1004 1.1 fvdl * If we are identified and have successfully sent the CDB,
1005 1.1 fvdl * any status will do. Optimize this fast path.
1006 1.1 fvdl */
1007 1.1 fvdl test SCB_CONTROL, STATUS_RCVD jz mesgin_proto_violation;
1008 1.1 fvdl test SEQ_FLAGS, NOT_IDENTIFIED|NO_CDB_SENT jz complete_accepted;
1009 1.1 fvdl
1010 1.1 fvdl /*
1011 1.1 fvdl * If the target never sent an identify message but instead went
1012 1.1 fvdl * to mesgin to give an invalid message, let the host abort us.
1013 1.1 fvdl */
1014 1.1 fvdl test SEQ_FLAGS, NOT_IDENTIFIED jnz mesgin_proto_violation;
1015 1.1 fvdl
1016 1.1 fvdl /*
1017 1.1 fvdl * If we recevied good status but never successfully sent the
1018 1.1 fvdl * cdb, abort the command.
1019 1.1 fvdl */
1020 1.1 fvdl test SCB_SCSI_STATUS,0xff jnz complete_accepted;
1021 1.1 fvdl test SEQ_FLAGS, NO_CDB_SENT jnz mesgin_proto_violation;
1022 1.1 fvdl complete_accepted:
1023 1.1 fvdl
1024 1.1 fvdl /*
1025 1.1 fvdl * See if we attempted to deliver a message but the target ingnored us.
1026 1.1 fvdl */
1027 1.1 fvdl test SCB_CONTROL, MK_MESSAGE jz complete_nomsg;
1028 1.1 fvdl SET_SEQINTCODE(MKMSG_FAILED)
1029 1.1 fvdl complete_nomsg:
1030 1.1 fvdl call queue_scb_completion;
1031 1.1 fvdl jmp await_busfree;
1032 1.1 fvdl
1033 1.1 fvdl freeze_queue:
1034 1.1 fvdl /* Cancel any pending select-out. */
1035 1.1 fvdl test SSTAT0, SELDO|SELINGO jnz . + 2;
1036 1.1 fvdl and SCSISEQ0, ~ENSELO;
1037 1.1 fvdl mov ACCUM_SAVE, A;
1038 1.1 fvdl clr A;
1039 1.1 fvdl add QFREEZE_COUNT, 1;
1040 1.1 fvdl adc QFREEZE_COUNT[1], A;
1041 1.1 fvdl or SEQ_FLAGS2, SELECTOUT_QFROZEN;
1042 1.1 fvdl mov A, ACCUM_SAVE ret;
1043 1.1 fvdl
1044 1.4 thorpej /*
1045 1.4 thorpej * Complete the current FIFO's SCB if data for this same
1046 1.4 thorpej * SCB is not transferring in the other FIFO.
1047 1.4 thorpej */
1048 1.4 thorpej SET_SRC_MODE M_DFF1;
1049 1.4 thorpej SET_DST_MODE M_DFF1;
1050 1.4 thorpej pkt_complete_scb_if_fifos_idle:
1051 1.4 thorpej bmov ARG_1, SCBPTR, 2;
1052 1.4 thorpej mvi DFFSXFRCTL, CLRCHN;
1053 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
1054 1.1 fvdl bmov SCBPTR, ARG_1, 2;
1055 1.4 thorpej test SCB_FIFO_USE_COUNT, 0xFF jnz return;
1056 1.1 fvdl queue_scb_completion:
1057 1.1 fvdl test SCB_SCSI_STATUS,0xff jnz bad_status;
1058 1.1 fvdl /*
1059 1.1 fvdl * Check for residuals
1060 1.1 fvdl */
1061 1.1 fvdl test SCB_SGPTR, SG_LIST_NULL jnz complete; /* No xfer */
1062 1.1 fvdl test SCB_SGPTR, SG_FULL_RESID jnz upload_scb;/* Never xfered */
1063 1.1 fvdl test SCB_RESIDUAL_SGPTR, SG_LIST_NULL jz upload_scb;
1064 1.1 fvdl complete:
1065 1.1 fvdl bmov SCB_NEXT_COMPLETE, COMPLETE_SCB_HEAD, 2;
1066 1.1 fvdl bmov COMPLETE_SCB_HEAD, SCBPTR, 2 ret;
1067 1.1 fvdl bad_status:
1068 1.1 fvdl cmp SCB_SCSI_STATUS, STATUS_PKT_SENSE je upload_scb;
1069 1.1 fvdl call freeze_queue;
1070 1.1 fvdl upload_scb:
1071 1.4 thorpej /*
1072 1.4 thorpej * Restore SCB TAG since we reuse this field
1073 1.4 thorpej * in the sequencer. We don't want to corrupt
1074 1.4 thorpej * it on the host.
1075 1.4 thorpej */
1076 1.4 thorpej bmov SCB_TAG, SCBPTR, 2;
1077 1.1 fvdl bmov SCB_NEXT_COMPLETE, COMPLETE_DMA_SCB_HEAD, 2;
1078 1.1 fvdl bmov COMPLETE_DMA_SCB_HEAD, SCBPTR, 2;
1079 1.1 fvdl or SCB_SGPTR, SG_STATUS_VALID ret;
1080 1.1 fvdl
1081 1.1 fvdl /*
1082 1.1 fvdl * Is it a disconnect message? Set a flag in the SCB to remind us
1083 1.1 fvdl * and await the bus going free. If this is an untagged transaction
1084 1.1 fvdl * store the SCB id for it in our untagged target table for lookup on
1085 1.1 fvdl * a reselction.
1086 1.1 fvdl */
1087 1.1 fvdl mesgin_disconnect:
1088 1.1 fvdl /*
1089 1.1 fvdl * If ATN is raised, we still want to give the target a message.
1090 1.1 fvdl * Perhaps there was a parity error on this last message byte
1091 1.1 fvdl * or we want to abort this command. Either way, the target
1092 1.1 fvdl * should take us to message out phase and then attempt to
1093 1.1 fvdl * disconnect again.
1094 1.1 fvdl * XXX - Wait for more testing.
1095 1.1 fvdl test SCSISIGI, ATNI jnz mesgin_done;
1096 1.1 fvdl */
1097 1.1 fvdl test SEQ_FLAGS, NOT_IDENTIFIED|NO_CDB_SENT
1098 1.1 fvdl jnz mesgin_proto_violation;
1099 1.1 fvdl or SCB_CONTROL,DISCONNECTED;
1100 1.1 fvdl test SCB_CONTROL, TAG_ENB jnz await_busfree;
1101 1.1 fvdl queue_disc_scb:
1102 1.1 fvdl bmov REG0, SCBPTR, 2;
1103 1.1 fvdl INDEX_DISC_LIST(SAVED_SCSIID, SAVED_LUN);
1104 1.1 fvdl bmov DINDEX, SINDEX, 2;
1105 1.1 fvdl bmov DINDIR, REG0, 2;
1106 1.1 fvdl bmov SCBPTR, REG0, 2;
1107 1.1 fvdl /* FALLTHROUGH */
1108 1.1 fvdl await_busfree:
1109 1.1 fvdl and SIMODE1, ~ENBUSFREE;
1110 1.1 fvdl if ((ahd->bugs & AHD_BUSFREEREV_BUG) == 0) {
1111 1.1 fvdl /*
1112 1.1 fvdl * In the BUSFREEREV_BUG case, the
1113 1.1 fvdl * busfree status was cleared at the
1114 1.1 fvdl * beginning of the connection.
1115 1.1 fvdl */
1116 1.1 fvdl mvi CLRSINT1,CLRBUSFREE;
1117 1.1 fvdl }
1118 1.1 fvdl mov NONE, SCSIDAT; /* Ack the last byte */
1119 1.1 fvdl test MODE_PTR, ~(MK_MODE(M_DFF1, M_DFF1))
1120 1.1 fvdl jnz await_busfree_not_m_dff;
1121 1.1 fvdl SET_SRC_MODE M_DFF1;
1122 1.1 fvdl SET_DST_MODE M_DFF1;
1123 1.1 fvdl await_busfree_clrchn:
1124 1.1 fvdl mvi DFFSXFRCTL, CLRCHN;
1125 1.1 fvdl await_busfree_not_m_dff:
1126 1.1 fvdl call clear_target_state;
1127 1.1 fvdl test SSTAT1,REQINIT|BUSFREE jz .;
1128 1.1 fvdl test SSTAT1, BUSFREE jnz idle_loop;
1129 1.1 fvdl SET_SEQINTCODE(MISSED_BUSFREE)
1130 1.1 fvdl
1131 1.1 fvdl
1132 1.1 fvdl /*
1133 1.1 fvdl * Save data pointers message:
1134 1.1 fvdl * Copying RAM values back to SCB, for Save Data Pointers message, but
1135 1.1 fvdl * only if we've actually been into a data phase to change them. This
1136 1.1 fvdl * protects against bogus data in scratch ram and the residual counts
1137 1.1 fvdl * since they are only initialized when we go into data_in or data_out.
1138 1.1 fvdl * Ack the message as soon as possible.
1139 1.1 fvdl */
1140 1.1 fvdl SET_SRC_MODE M_DFF1;
1141 1.1 fvdl SET_DST_MODE M_DFF1;
1142 1.1 fvdl mesgin_sdptrs:
1143 1.1 fvdl mov NONE,SCSIDAT; /*dummy read from latch to ACK*/
1144 1.1 fvdl test SEQ_FLAGS, DPHASE jz ITloop;
1145 1.1 fvdl call save_pointers;
1146 1.1 fvdl jmp ITloop;
1147 1.1 fvdl
1148 1.1 fvdl save_pointers:
1149 1.1 fvdl /*
1150 1.1 fvdl * If we are asked to save our position at the end of the
1151 1.1 fvdl * transfer, just mark us at the end rather than perform a
1152 1.1 fvdl * full save.
1153 1.1 fvdl */
1154 1.1 fvdl test SCB_RESIDUAL_SGPTR[0], SG_LIST_NULL jz save_pointers_full;
1155 1.1 fvdl or SCB_SGPTR, SG_LIST_NULL ret;
1156 1.1 fvdl
1157 1.1 fvdl save_pointers_full:
1158 1.1 fvdl /*
1159 1.1 fvdl * The SCB_DATAPTR becomes the current SHADDR.
1160 1.1 fvdl * All other information comes directly from our residual
1161 1.1 fvdl * state.
1162 1.1 fvdl */
1163 1.1 fvdl bmov SCB_DATAPTR, SHADDR, 8;
1164 1.1 fvdl bmov SCB_DATACNT, SCB_RESIDUAL_DATACNT, 8 ret;
1165 1.1 fvdl
1166 1.1 fvdl /*
1167 1.1 fvdl * Restore pointers message? Data pointers are recopied from the
1168 1.1 fvdl * SCB anytime we enter a data phase for the first time, so all
1169 1.1 fvdl * we need to do is clear the DPHASE flag and let the data phase
1170 1.1 fvdl * code do the rest. We also reset/reallocate the FIFO to make
1171 1.1 fvdl * sure we have a clean start for the next data or command phase.
1172 1.1 fvdl */
1173 1.1 fvdl mesgin_rdptrs:
1174 1.1 fvdl and SEQ_FLAGS, ~DPHASE;
1175 1.1 fvdl test MODE_PTR, ~(MK_MODE(M_DFF1, M_DFF1)) jnz msgin_rdptrs_get_fifo;
1176 1.1 fvdl mvi DFFSXFRCTL, RSTCHN|CLRSHCNT;
1177 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
1178 1.1 fvdl msgin_rdptrs_get_fifo:
1179 1.1 fvdl call allocate_fifo;
1180 1.1 fvdl jmp mesgin_done;
1181 1.1 fvdl
1182 1.1 fvdl clear_target_state:
1183 1.1 fvdl mvi LASTPHASE, P_BUSFREE;
1184 1.1 fvdl /* clear target specific flags */
1185 1.1 fvdl mvi SEQ_FLAGS, NOT_IDENTIFIED|NO_CDB_SENT ret;
1186 1.1 fvdl
1187 1.1 fvdl phase_lock:
1188 1.1 fvdl if ((ahd->bugs & AHD_EARLY_REQ_BUG) != 0) {
1189 1.1 fvdl /*
1190 1.1 fvdl * Don't ignore persistent REQ assertions just because
1191 1.1 fvdl * they were asserted within the bus settle delay window.
1192 1.1 fvdl * This allows us to tolerate devices like the GEM318
1193 1.1 fvdl * that violate the SCSI spec. We are careful not to
1194 1.1 fvdl * count REQ while we are waiting for it to fall during
1195 1.1 fvdl * an async phase due to our asserted ACK. Each
1196 1.1 fvdl * sequencer instruction takes ~25ns, so the REQ must
1197 1.1 fvdl * last at least 100ns in order to be counted as a true
1198 1.1 fvdl * REQ.
1199 1.1 fvdl */
1200 1.1 fvdl test SCSIPHASE, 0xFF jnz phase_locked;
1201 1.1 fvdl test SCSISIGI, ACKI jnz phase_lock;
1202 1.1 fvdl test SCSISIGI, REQI jz phase_lock;
1203 1.1 fvdl test SCSIPHASE, 0xFF jnz phase_locked;
1204 1.1 fvdl test SCSISIGI, ACKI jnz phase_lock;
1205 1.1 fvdl test SCSISIGI, REQI jz phase_lock;
1206 1.1 fvdl phase_locked:
1207 1.1 fvdl } else {
1208 1.1 fvdl test SCSIPHASE, 0xFF jz .;
1209 1.1 fvdl }
1210 1.1 fvdl test SSTAT1, SCSIPERR jnz phase_lock;
1211 1.1 fvdl phase_lock_latch_phase:
1212 1.1 fvdl and LASTPHASE, PHASE_MASK, SCSISIGI ret;
1213 1.1 fvdl
1214 1.1 fvdl /*
1215 1.1 fvdl * Functions to read data in Automatic PIO mode.
1216 1.1 fvdl *
1217 1.1 fvdl * An ACK is not sent on input from the target until SCSIDATL is read from.
1218 1.1 fvdl * So we wait until SCSIDATL is latched (the usual way), then read the data
1219 1.1 fvdl * byte directly off the bus using SCSIBUSL. When we have pulled the ATN
1220 1.1 fvdl * line, or we just want to acknowledge the byte, then we do a dummy read
1221 1.1 fvdl * from SCISDATL. The SCSI spec guarantees that the target will hold the
1222 1.1 fvdl * data byte on the bus until we send our ACK.
1223 1.1 fvdl *
1224 1.1 fvdl * The assumption here is that these are called in a particular sequence,
1225 1.1 fvdl * and that REQ is already set when inb_first is called. inb_{first,next}
1226 1.1 fvdl * use the same calling convention as inb.
1227 1.1 fvdl */
1228 1.1 fvdl inb_next:
1229 1.1 fvdl mov NONE,SCSIDAT; /*dummy read from latch to ACK*/
1230 1.1 fvdl inb_next_wait:
1231 1.1 fvdl /*
1232 1.1 fvdl * If there is a parity error, wait for the kernel to
1233 1.1 fvdl * see the interrupt and prepare our message response
1234 1.1 fvdl * before continuing.
1235 1.1 fvdl */
1236 1.1 fvdl test SCSIPHASE, 0xFF jz .;
1237 1.1 fvdl test SSTAT1, SCSIPERR jnz inb_next_wait;
1238 1.1 fvdl inb_next_check_phase:
1239 1.1 fvdl and LASTPHASE, PHASE_MASK, SCSISIGI;
1240 1.1 fvdl cmp LASTPHASE, P_MESGIN jne mesgin_phasemis;
1241 1.1 fvdl inb_first:
1242 1.1 fvdl clr DINDEX[1];
1243 1.1 fvdl mov DINDEX,SINDEX;
1244 1.1 fvdl mov DINDIR,SCSIBUS ret; /*read byte directly from bus*/
1245 1.1 fvdl inb_last:
1246 1.1 fvdl mov NONE,SCSIDAT ret; /*dummy read from latch to ACK*/
1247 1.1 fvdl
1248 1.1 fvdl mk_mesg:
1249 1.1 fvdl mvi SCSISIGO, ATNO;
1250 1.1 fvdl mov MSG_OUT,SINDEX ret;
1251 1.1 fvdl
1252 1.1 fvdl SET_SRC_MODE M_DFF1;
1253 1.1 fvdl SET_DST_MODE M_DFF1;
1254 1.1 fvdl disable_ccsgen:
1255 1.1 fvdl test SG_STATE, FETCH_INPROG jz disable_ccsgen_fetch_done;
1256 1.1 fvdl clr CCSGCTL;
1257 1.1 fvdl disable_ccsgen_fetch_done:
1258 1.1 fvdl clr SG_STATE ret;
1259 1.1 fvdl
1260 1.1 fvdl service_fifo:
1261 1.1 fvdl /*
1262 1.1 fvdl * Do we have any prefetch left???
1263 1.1 fvdl */
1264 1.1 fvdl test SG_STATE, SEGS_AVAIL jnz idle_sg_avail;
1265 1.1 fvdl
1266 1.1 fvdl /*
1267 1.1 fvdl * Can this FIFO have access to the S/G cache yet?
1268 1.1 fvdl */
1269 1.1 fvdl test CCSGCTL, SG_CACHE_AVAIL jz return;
1270 1.1 fvdl
1271 1.1 fvdl /* Did we just finish fetching segs? */
1272 1.1 fvdl test CCSGCTL, CCSGDONE jnz idle_sgfetch_complete;
1273 1.1 fvdl
1274 1.1 fvdl /* Are we actively fetching segments? */
1275 1.1 fvdl test CCSGCTL, CCSGENACK jnz return;
1276 1.1 fvdl
1277 1.1 fvdl /*
1278 1.1 fvdl * We fetch a "cacheline aligned" and sized amount of data
1279 1.1 fvdl * so we don't end up referencing a non-existant page.
1280 1.1 fvdl * Cacheline aligned is in quotes because the kernel will
1281 1.1 fvdl * set the prefetch amount to a reasonable level if the
1282 1.1 fvdl * cacheline size is unknown.
1283 1.1 fvdl */
1284 1.1 fvdl bmov SGHADDR, SCB_RESIDUAL_SGPTR, 4;
1285 1.1 fvdl mvi SGHCNT, SG_PREFETCH_CNT;
1286 1.1 fvdl if ((ahd->bugs & AHD_REG_SLOW_SETTLE_BUG) != 0) {
1287 1.1 fvdl /*
1288 1.1 fvdl * Need two instruction between "touches" of SGHADDR.
1289 1.1 fvdl */
1290 1.1 fvdl nop;
1291 1.1 fvdl }
1292 1.1 fvdl and SGHADDR[0], SG_PREFETCH_ALIGN_MASK, SCB_RESIDUAL_SGPTR;
1293 1.1 fvdl mvi CCSGCTL, CCSGEN|CCSGRESET;
1294 1.1 fvdl or SG_STATE, FETCH_INPROG ret;
1295 1.1 fvdl idle_sgfetch_complete:
1296 1.1 fvdl /*
1297 1.1 fvdl * Guard against SG_CACHE_AVAIL activating during sg fetch
1298 1.1 fvdl * request in the other FIFO.
1299 1.1 fvdl */
1300 1.1 fvdl test SG_STATE, FETCH_INPROG jz return;
1301 1.1 fvdl clr CCSGCTL;
1302 1.1 fvdl and CCSGADDR, SG_PREFETCH_ADDR_MASK, SCB_RESIDUAL_SGPTR;
1303 1.1 fvdl mvi SG_STATE, SEGS_AVAIL|LOADING_NEEDED;
1304 1.1 fvdl idle_sg_avail:
1305 1.1 fvdl /* Does the hardware have space for another SG entry? */
1306 1.1 fvdl test DFSTATUS, PRELOAD_AVAIL jz return;
1307 1.1 fvdl /*
1308 1.1 fvdl * On the A, preloading a segment before HDMAENACK
1309 1.1 fvdl * comes true can clobber the shaddow address of the
1310 1.1 fvdl * first segment in the S/G FIFO. Wait until it is
1311 1.1 fvdl * safe to proceed.
1312 1.1 fvdl */
1313 1.1 fvdl if ((ahd->features & AHD_NEW_DFCNTRL_OPTS) == 0) {
1314 1.1 fvdl test DFCNTRL, HDMAENACK jz return;
1315 1.1 fvdl }
1316 1.1 fvdl if ((ahd->flags & AHD_64BIT_ADDRESSING) != 0) {
1317 1.1 fvdl bmov HADDR, CCSGRAM, 8;
1318 1.1 fvdl } else {
1319 1.1 fvdl bmov HADDR, CCSGRAM, 4;
1320 1.1 fvdl }
1321 1.1 fvdl bmov HCNT, CCSGRAM, 3;
1322 1.1 fvdl test HCNT[0], 0x1 jz . + 2;
1323 1.1 fvdl xor DATA_COUNT_ODD, 0x1;
1324 1.1 fvdl bmov SCB_RESIDUAL_DATACNT[3], CCSGRAM, 1;
1325 1.1 fvdl if ((ahd->flags & AHD_39BIT_ADDRESSING) != 0) {
1326 1.1 fvdl and HADDR[4], SG_HIGH_ADDR_BITS, SCB_RESIDUAL_DATACNT[3];
1327 1.1 fvdl }
1328 1.1 fvdl if ((ahd->flags & AHD_64BIT_ADDRESSING) != 0) {
1329 1.1 fvdl /* Skip 4 bytes of pad. */
1330 1.1 fvdl add CCSGADDR, 4;
1331 1.1 fvdl }
1332 1.1 fvdl sg_advance:
1333 1.1 fvdl clr A; /* add sizeof(struct scatter) */
1334 1.1 fvdl add SCB_RESIDUAL_SGPTR[0],SG_SIZEOF;
1335 1.1 fvdl adc SCB_RESIDUAL_SGPTR[1],A;
1336 1.1 fvdl adc SCB_RESIDUAL_SGPTR[2],A;
1337 1.1 fvdl adc SCB_RESIDUAL_SGPTR[3],A;
1338 1.1 fvdl mov SINDEX, SCB_RESIDUAL_SGPTR[0];
1339 1.1 fvdl test DATA_COUNT_ODD, 0x1 jz . + 2;
1340 1.1 fvdl or SINDEX, ODD_SEG;
1341 1.1 fvdl test SCB_RESIDUAL_DATACNT[3], SG_LAST_SEG jz . + 3;
1342 1.1 fvdl or SINDEX, LAST_SEG;
1343 1.1 fvdl clr SG_STATE;
1344 1.1 fvdl mov SG_CACHE_PRE, SINDEX;
1345 1.1 fvdl if ((ahd->features & AHD_NEW_DFCNTRL_OPTS) != 0) {
1346 1.1 fvdl /*
1347 1.1 fvdl * Use SCSIENWRDIS so that SCSIEN is never
1348 1.1 fvdl * modified by this operation.
1349 1.1 fvdl */
1350 1.1 fvdl or DFCNTRL, PRELOADEN|HDMAEN|SCSIENWRDIS;
1351 1.1 fvdl } else {
1352 1.1 fvdl or DFCNTRL, PRELOADEN|HDMAEN;
1353 1.1 fvdl }
1354 1.1 fvdl /*
1355 1.1 fvdl * Do we have another segment in the cache?
1356 1.1 fvdl */
1357 1.1 fvdl add NONE, SG_PREFETCH_CNT_LIMIT, CCSGADDR;
1358 1.1 fvdl jnc return;
1359 1.1 fvdl and SG_STATE, ~SEGS_AVAIL ret;
1360 1.1 fvdl
1361 1.1 fvdl /*
1362 1.1 fvdl * Initialize the DMA address and counter from the SCB.
1363 1.1 fvdl */
1364 1.1 fvdl load_first_seg:
1365 1.1 fvdl bmov HADDR, SCB_DATAPTR, 11;
1366 1.1 fvdl and DATA_COUNT_ODD, 0x1, SCB_DATACNT[0];
1367 1.1 fvdl and REG_ISR, ~SG_FULL_RESID, SCB_SGPTR[0];
1368 1.1 fvdl test SCB_DATACNT[3], SG_LAST_SEG jz . + 2;
1369 1.1 fvdl or REG_ISR, LAST_SEG;
1370 1.1 fvdl test DATA_COUNT_ODD, 0x1 jz . + 2;
1371 1.1 fvdl or REG_ISR, ODD_SEG;
1372 1.1 fvdl mov SG_CACHE_PRE, REG_ISR;
1373 1.1 fvdl mvi DFCNTRL, (PRELOADEN|SCSIEN|HDMAEN);
1374 1.1 fvdl /*
1375 1.1 fvdl * Since we've are entering a data phase, we will
1376 1.1 fvdl * rely on the SCB_RESID* fields. Initialize the
1377 1.1 fvdl * residual and clear the full residual flag.
1378 1.1 fvdl */
1379 1.1 fvdl and SCB_SGPTR[0], ~SG_FULL_RESID;
1380 1.1 fvdl bmov SCB_RESIDUAL_DATACNT[3], SCB_DATACNT[3], 5;
1381 1.1 fvdl /* If we need more S/G elements, tell the idle loop */
1382 1.1 fvdl test SCB_RESIDUAL_DATACNT[3], SG_LAST_SEG jnz . + 2;
1383 1.1 fvdl mvi SG_STATE, LOADING_NEEDED ret;
1384 1.1 fvdl clr SG_STATE ret;
1385 1.1 fvdl
1386 1.1 fvdl p_data_handle_xfer:
1387 1.4 thorpej call setjmp;
1388 1.1 fvdl test SG_STATE, LOADING_NEEDED jnz service_fifo;
1389 1.1 fvdl p_data_clear_handler:
1390 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR ret;
1391 1.1 fvdl
1392 1.1 fvdl p_data:
1393 1.1 fvdl test SEQ_FLAGS, NOT_IDENTIFIED|NO_CDB_SENT jz p_data_allowed;
1394 1.1 fvdl SET_SEQINTCODE(PROTO_VIOLATION)
1395 1.1 fvdl p_data_allowed:
1396 1.1 fvdl
1397 1.1 fvdl test SEQ_FLAGS, DPHASE jz data_phase_initialize;
1398 1.1 fvdl
1399 1.1 fvdl /*
1400 1.1 fvdl * If we re-enter the data phase after going through another
1401 1.1 fvdl * phase, our transfer location has almost certainly been
1402 1.1 fvdl * corrupted by the interveining, non-data, transfers. Ask
1403 1.1 fvdl * the host driver to fix us up based on the transfer residual
1404 1.1 fvdl * unless we already know that we should be bitbucketing.
1405 1.1 fvdl */
1406 1.1 fvdl test SCB_RESIDUAL_SGPTR[0], SG_LIST_NULL jnz p_data_bitbucket;
1407 1.1 fvdl SET_SEQINTCODE(PDATA_REINIT)
1408 1.1 fvdl jmp data_phase_inbounds;
1409 1.1 fvdl
1410 1.1 fvdl p_data_bitbucket:
1411 1.1 fvdl /*
1412 1.1 fvdl * Turn on `Bit Bucket' mode, wait until the target takes
1413 1.1 fvdl * us to another phase, and then notify the host.
1414 1.1 fvdl */
1415 1.1 fvdl mov SAVED_MODE, MODE_PTR;
1416 1.1 fvdl test MODE_PTR, ~(MK_MODE(M_DFF1, M_DFF1))
1417 1.1 fvdl jnz bitbucket_not_m_dff;
1418 1.1 fvdl /*
1419 1.1 fvdl * Ensure that any FIFO contents are cleared out and the
1420 1.1 fvdl * FIFO free'd prior to starting the BITBUCKET. BITBUCKET
1421 1.1 fvdl * doesn't discard data already in the FIFO.
1422 1.1 fvdl */
1423 1.1 fvdl mvi DFFSXFRCTL, RSTCHN|CLRSHCNT;
1424 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
1425 1.1 fvdl bitbucket_not_m_dff:
1426 1.1 fvdl or SXFRCTL1,BITBUCKET;
1427 1.1 fvdl /* Wait for non-data phase. */
1428 1.1 fvdl test SCSIPHASE, ~DATA_PHASE_MASK jz .;
1429 1.1 fvdl and SXFRCTL1, ~BITBUCKET;
1430 1.1 fvdl RESTORE_MODE(SAVED_MODE)
1431 1.1 fvdl SET_SRC_MODE M_DFF1;
1432 1.1 fvdl SET_DST_MODE M_DFF1;
1433 1.1 fvdl SET_SEQINTCODE(DATA_OVERRUN)
1434 1.1 fvdl jmp ITloop;
1435 1.1 fvdl
1436 1.1 fvdl data_phase_initialize:
1437 1.1 fvdl test SCB_SGPTR[0], SG_LIST_NULL jnz p_data_bitbucket;
1438 1.1 fvdl call load_first_seg;
1439 1.1 fvdl data_phase_inbounds:
1440 1.1 fvdl /* We have seen a data phase at least once. */
1441 1.1 fvdl or SEQ_FLAGS, DPHASE;
1442 1.1 fvdl mov SAVED_MODE, MODE_PTR;
1443 1.1 fvdl test SG_STATE, LOADING_NEEDED jz data_group_dma_loop;
1444 1.1 fvdl call p_data_handle_xfer;
1445 1.1 fvdl data_group_dma_loop:
1446 1.1 fvdl /*
1447 1.1 fvdl * The transfer is complete if either the last segment
1448 1.1 fvdl * completes or the target changes phase. Both conditions
1449 1.1 fvdl * will clear SCSIEN.
1450 1.1 fvdl */
1451 1.1 fvdl call idle_loop_service_fifos;
1452 1.1 fvdl call idle_loop_cchan;
1453 1.1 fvdl call idle_loop_gsfifo;
1454 1.1 fvdl RESTORE_MODE(SAVED_MODE)
1455 1.1 fvdl test DFCNTRL, SCSIEN jnz data_group_dma_loop;
1456 1.1 fvdl
1457 1.1 fvdl data_group_dmafinish:
1458 1.1 fvdl /*
1459 1.1 fvdl * The transfer has terminated either due to a phase
1460 1.1 fvdl * change, and/or the completion of the last segment.
1461 1.1 fvdl * We have two goals here. Do as much other work
1462 1.1 fvdl * as possible while the data fifo drains on a read
1463 1.1 fvdl * and respond as quickly as possible to the standard
1464 1.1 fvdl * messages (save data pointers/disconnect and command
1465 1.1 fvdl * complete) that usually follow a data phase.
1466 1.1 fvdl */
1467 1.1 fvdl call calc_residual;
1468 1.1 fvdl
1469 1.1 fvdl /*
1470 1.1 fvdl * Go ahead and shut down the DMA engine now.
1471 1.1 fvdl */
1472 1.1 fvdl test DFCNTRL, DIRECTION jnz data_phase_finish;
1473 1.1 fvdl data_group_fifoflush:
1474 1.1 fvdl if ((ahd->bugs & AHD_AUTOFLUSH_BUG) != 0) {
1475 1.1 fvdl or DFCNTRL, FIFOFLUSH;
1476 1.1 fvdl }
1477 1.1 fvdl /*
1478 1.1 fvdl * We have enabled the auto-ack feature. This means
1479 1.1 fvdl * that the controller may have already transferred
1480 1.1 fvdl * some overrun bytes into the data FIFO and acked them
1481 1.1 fvdl * on the bus. The only way to detect this situation is
1482 1.1 fvdl * to wait for LAST_SEG_DONE to come true on a completed
1483 1.1 fvdl * transfer and then test to see if the data FIFO is
1484 1.1 fvdl * non-empty. We know there is more data yet to transfer
1485 1.1 fvdl * if SG_LIST_NULL is not yet set, thus there cannot be
1486 1.1 fvdl * an overrun.
1487 1.1 fvdl */
1488 1.1 fvdl test SCB_RESIDUAL_SGPTR[0], SG_LIST_NULL jz data_phase_finish;
1489 1.1 fvdl test SG_CACHE_SHADOW, LAST_SEG_DONE jz .;
1490 1.1 fvdl test DFSTATUS, FIFOEMP jnz data_phase_finish;
1491 1.1 fvdl /* Overrun */
1492 1.1 fvdl jmp p_data;
1493 1.1 fvdl data_phase_finish:
1494 1.1 fvdl /*
1495 1.1 fvdl * If the target has left us in data phase, loop through
1496 1.2 wiz * the DMA code again. We will only loop if there is a
1497 1.1 fvdl * data overrun.
1498 1.1 fvdl */
1499 1.1 fvdl if ((ahd->flags & AHD_TARGETROLE) != 0) {
1500 1.1 fvdl test SSTAT0, TARGET jnz data_phase_done;
1501 1.1 fvdl }
1502 1.1 fvdl if ((ahd->flags & AHD_INITIATORROLE) != 0) {
1503 1.1 fvdl test SSTAT1, REQINIT jz .;
1504 1.1 fvdl test SCSIPHASE, DATA_PHASE_MASK jnz p_data;
1505 1.1 fvdl }
1506 1.1 fvdl
1507 1.1 fvdl data_phase_done:
1508 1.1 fvdl /* Kill off any pending prefetch */
1509 1.1 fvdl call disable_ccsgen;
1510 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
1511 1.1 fvdl
1512 1.1 fvdl if ((ahd->flags & AHD_TARGETROLE) != 0) {
1513 1.1 fvdl test SEQ_FLAGS, DPHASE_PENDING jz ITloop;
1514 1.1 fvdl /*
1515 1.1 fvdl and SEQ_FLAGS, ~DPHASE_PENDING;
1516 1.1 fvdl * For data-in phases, wait for any pending acks from the
1517 1.1 fvdl * initiator before changing phase. We only need to
1518 1.1 fvdl * send Ignore Wide Residue messages for data-in phases.
1519 1.1 fvdl test DFCNTRL, DIRECTION jz target_ITloop;
1520 1.1 fvdl test SSTAT1, REQINIT jnz .;
1521 1.1 fvdl test DATA_COUNT_ODD, 0x1 jz target_ITloop;
1522 1.1 fvdl SET_MODE(M_SCSI, M_SCSI)
1523 1.1 fvdl test NEGCONOPTS, WIDEXFER jz target_ITloop;
1524 1.1 fvdl */
1525 1.1 fvdl /*
1526 1.1 fvdl * Issue an Ignore Wide Residue Message.
1527 1.1 fvdl mvi P_MESGIN|BSYO call change_phase;
1528 1.1 fvdl mvi MSG_IGN_WIDE_RESIDUE call target_outb;
1529 1.1 fvdl mvi 1 call target_outb;
1530 1.1 fvdl jmp target_ITloop;
1531 1.1 fvdl */
1532 1.1 fvdl } else {
1533 1.1 fvdl jmp ITloop;
1534 1.1 fvdl }
1535 1.1 fvdl
1536 1.1 fvdl /*
1537 1.1 fvdl * We assume that, even though data may still be
1538 1.1 fvdl * transferring to the host, that the SCSI side of
1539 1.1 fvdl * the DMA engine is now in a static state. This
1540 1.1 fvdl * allows us to update our notion of where we are
1541 1.1 fvdl * in this transfer.
1542 1.1 fvdl *
1543 1.1 fvdl * If, by chance, we stopped before being able
1544 1.1 fvdl * to fetch additional segments for this transfer,
1545 1.1 fvdl * yet the last S/G was completely exhausted,
1546 1.1 fvdl * call our idle loop until it is able to load
1547 1.1 fvdl * another segment. This will allow us to immediately
1548 1.1 fvdl * pickup on the next segment on the next data phase.
1549 1.1 fvdl *
1550 1.1 fvdl * If we happened to stop on the last segment, then
1551 1.1 fvdl * our residual information is still correct from
1552 1.1 fvdl * the idle loop and there is no need to perform
1553 1.1 fvdl * any fixups.
1554 1.1 fvdl */
1555 1.1 fvdl residual_before_last_seg:
1556 1.1 fvdl test MDFFSTAT, SHVALID jnz sgptr_fixup;
1557 1.1 fvdl /*
1558 1.1 fvdl * Can never happen from an interrupt as the packetized
1559 1.1 fvdl * hardware will only interrupt us once SHVALID or
1560 1.1 fvdl * LAST_SEG_DONE.
1561 1.1 fvdl */
1562 1.1 fvdl call idle_loop_service_fifos;
1563 1.1 fvdl RESTORE_MODE(SAVED_MODE)
1564 1.1 fvdl /* FALLTHROUGH */
1565 1.1 fvdl calc_residual:
1566 1.1 fvdl test SG_CACHE_SHADOW, LAST_SEG jz residual_before_last_seg;
1567 1.1 fvdl /* Record if we've consumed all S/G entries */
1568 1.1 fvdl test MDFFSTAT, SHVALID jz . + 2;
1569 1.1 fvdl bmov SCB_RESIDUAL_DATACNT, SHCNT, 3 ret;
1570 1.1 fvdl or SCB_RESIDUAL_SGPTR[0], SG_LIST_NULL ret;
1571 1.1 fvdl
1572 1.1 fvdl sgptr_fixup:
1573 1.1 fvdl /*
1574 1.1 fvdl * Fixup the residual next S/G pointer. The S/G preload
1575 1.1 fvdl * feature of the chip allows us to load two elements
1576 1.1 fvdl * in addition to the currently active element. We
1577 1.1 fvdl * store the bottom byte of the next S/G pointer in
1578 1.1 fvdl * the SG_CACHE_PTR register so we can restore the
1579 1.1 fvdl * correct value when the DMA completes. If the next
1580 1.1 fvdl * sg ptr value has advanced to the point where higher
1581 1.1 fvdl * bytes in the address have been affected, fix them
1582 1.1 fvdl * too.
1583 1.1 fvdl */
1584 1.1 fvdl test SG_CACHE_SHADOW, 0x80 jz sgptr_fixup_done;
1585 1.1 fvdl test SCB_RESIDUAL_SGPTR[0], 0x80 jnz sgptr_fixup_done;
1586 1.1 fvdl add SCB_RESIDUAL_SGPTR[1], -1;
1587 1.1 fvdl adc SCB_RESIDUAL_SGPTR[2], -1;
1588 1.1 fvdl adc SCB_RESIDUAL_SGPTR[3], -1;
1589 1.1 fvdl sgptr_fixup_done:
1590 1.1 fvdl and SCB_RESIDUAL_SGPTR[0], SG_ADDR_MASK, SG_CACHE_SHADOW;
1591 1.1 fvdl clr DATA_COUNT_ODD;
1592 1.1 fvdl test SG_CACHE_SHADOW, ODD_SEG jz . + 2;
1593 1.1 fvdl or DATA_COUNT_ODD, 0x1;
1594 1.1 fvdl clr SCB_RESIDUAL_DATACNT[3]; /* We are not the last seg */
1595 1.1 fvdl bmov SCB_RESIDUAL_DATACNT, SHCNT, 3 ret;
1596 1.1 fvdl
1597 1.1 fvdl export timer_isr:
1598 1.1 fvdl call issue_cmdcmplt;
1599 1.1 fvdl mvi CLRSEQINTSTAT, CLRSEQ_SWTMRTO;
1600 1.1 fvdl if ((ahd->bugs & AHD_SET_MODE_BUG) != 0) {
1601 1.1 fvdl /*
1602 1.1 fvdl * In H2A4, the mode pointer is not saved
1603 1.1 fvdl * for intvec2, but is restored on iret.
1604 1.1 fvdl * This can lead to the restoration of a
1605 1.1 fvdl * bogus mode ptr. Manually clear the
1606 1.1 fvdl * intmask bits and do a normal return
1607 1.1 fvdl * to compensate.
1608 1.1 fvdl */
1609 1.1 fvdl and SEQINTCTL, ~(INTMASK2|INTMASK1) ret;
1610 1.1 fvdl } else {
1611 1.1 fvdl or SEQINTCTL, IRET ret;
1612 1.1 fvdl }
1613 1.1 fvdl
1614 1.1 fvdl export seq_isr:
1615 1.1 fvdl if ((ahd->features & AHD_RTI) == 0) {
1616 1.1 fvdl /*
1617 1.1 fvdl * On RevA Silicon, if the target returns us to data-out
1618 1.1 fvdl * after we have already trained for data-out, it is
1619 1.1 fvdl * possible for us to transition the free running clock to
1620 1.1 fvdl * data-valid before the required 100ns P1 setup time (8 P1
1621 1.1 fvdl * assertions in fast-160 mode). This will only happen if
1622 1.1 fvdl * this L-Q is a continuation of a data transfer for which
1623 1.1 fvdl * we have already prefetched data into our FIFO (LQ/Data
1624 1.1 fvdl * followed by LQ/Data for the same write transaction).
1625 1.1 fvdl * This can cause some target implementations to miss the
1626 1.1 fvdl * first few data transfers on the bus. We detect this
1627 1.1 fvdl * situation by noticing that this is the first data transfer
1628 1.1 fvdl * after an LQ (LQIWORKONLQ true), that the data transfer is
1629 1.1 fvdl * a continuation of a transfer already setup in our FIFO
1630 1.1 fvdl * (SAVEPTRS interrupt), and that the transaction is a write
1631 1.1 fvdl * (DIRECTION set in DFCNTRL). The delay is performed by
1632 1.1 fvdl * disabling SCSIEN until we see the first REQ from the
1633 1.1 fvdl * target.
1634 1.1 fvdl *
1635 1.1 fvdl * First instruction in an ISR cannot be a branch on
1636 1.1 fvdl * Rev A. Snapshot LQISTAT2 so the status is not missed
1637 1.1 fvdl * and deffer the test by one instruction.
1638 1.1 fvdl */
1639 1.1 fvdl mov REG_ISR, LQISTAT2;
1640 1.4 thorpej test REG_ISR, LQIWORKONLQ jz main_isr;
1641 1.4 thorpej test SEQINTSRC, SAVEPTRS jz main_isr;
1642 1.1 fvdl test LONGJMP_ADDR[1], INVALID_ADDR jz saveptr_active_fifo;
1643 1.1 fvdl /*
1644 1.4 thorpej * Switch to the active FIFO after clearing the snapshot
1645 1.4 thorpej * savepointer in the current FIFO. We do this so that
1646 1.4 thorpej * a pending CTXTDONE or SAVEPTR is visible in the active
1647 1.4 thorpej * FIFO. This status is the only way we can detect if we
1648 1.4 thorpej * have lost the race (e.g. host paused us) and our attepts
1649 1.4 thorpej * to disable the channel occurred after all REQs were
1650 1.4 thorpej * already seen and acked (REQINIT never comes true).
1651 1.1 fvdl */
1652 1.4 thorpej mvi DFFSXFRCTL, CLRCHN;
1653 1.1 fvdl xor MODE_PTR, MK_MODE(M_DFF1, M_DFF1);
1654 1.4 thorpej test DFCNTRL, DIRECTION jz interrupt_return;
1655 1.1 fvdl and DFCNTRL, ~SCSIEN;
1656 1.4 thorpej snapshot_wait_data_valid:
1657 1.4 thorpej test SEQINTSRC, (CTXTDONE|SAVEPTRS) jnz snapshot_data_valid;
1658 1.4 thorpej test SSTAT1, REQINIT jz snapshot_wait_data_valid;
1659 1.4 thorpej snapshot_data_valid:
1660 1.1 fvdl or DFCNTRL, SCSIEN;
1661 1.4 thorpej or SEQINTCTL, IRET ret;
1662 1.1 fvdl snapshot_saveptr:
1663 1.1 fvdl mvi DFFSXFRCTL, CLRCHN;
1664 1.1 fvdl or SEQINTCTL, IRET ret;
1665 1.4 thorpej main_isr:
1666 1.1 fvdl }
1667 1.1 fvdl test SEQINTSRC, CFG4DATA jnz cfg4data_intr;
1668 1.1 fvdl test SEQINTSRC, CFG4ISTAT jnz cfg4istat_intr;
1669 1.1 fvdl test SEQINTSRC, SAVEPTRS jnz saveptr_intr;
1670 1.1 fvdl test SEQINTSRC, CFG4ICMD jnz cfg4icmd_intr;
1671 1.1 fvdl SET_SEQINTCODE(INVALID_SEQINT)
1672 1.1 fvdl
1673 1.1 fvdl /*
1674 1.1 fvdl * There are two types of save pointers interrupts:
1675 1.1 fvdl * The first is a snapshot save pointers where the current FIFO is not
1676 1.1 fvdl * active and contains a snapshot of the current poniter information.
1677 1.1 fvdl * This happens between packets in a stream for a single L_Q. Since we
1678 1.1 fvdl * are not performing a pointer save, we can safely clear the channel
1679 1.1 fvdl * so it can be used for other transactions. On RTI capable controllers,
1680 1.1 fvdl * where snapshots can, and are, disabled, the code to handle this type
1681 1.1 fvdl * of snapshot is not active.
1682 1.1 fvdl *
1683 1.1 fvdl * The second case is a save pointers on an active FIFO which occurs
1684 1.1 fvdl * if the target changes to a new L_Q or busfrees/QASes and the transfer
1685 1.1 fvdl * has a residual. This should occur coincident with a ctxtdone. We
1686 1.1 fvdl * disable the interrupt and allow our active routine to handle the
1687 1.1 fvdl * save.
1688 1.1 fvdl */
1689 1.1 fvdl saveptr_intr:
1690 1.1 fvdl if ((ahd->features & AHD_RTI) == 0) {
1691 1.1 fvdl test LONGJMP_ADDR[1], INVALID_ADDR jnz snapshot_saveptr;
1692 1.1 fvdl }
1693 1.1 fvdl saveptr_active_fifo:
1694 1.1 fvdl and SEQIMODE, ~ENSAVEPTRS;
1695 1.1 fvdl or SEQINTCTL, IRET ret;
1696 1.1 fvdl
1697 1.1 fvdl cfg4data_intr:
1698 1.4 thorpej test SCB_SGPTR[0], SG_LIST_NULL jnz pkt_handle_overrun_inc_use_count;
1699 1.1 fvdl call load_first_seg;
1700 1.1 fvdl call pkt_handle_xfer;
1701 1.4 thorpej inc SCB_FIFO_USE_COUNT;
1702 1.4 thorpej interrupt_return:
1703 1.1 fvdl or SEQINTCTL, IRET ret;
1704 1.1 fvdl
1705 1.1 fvdl cfg4istat_intr:
1706 1.1 fvdl call freeze_queue;
1707 1.1 fvdl add NONE, -13, SCB_CDB_LEN;
1708 1.1 fvdl jnc cfg4istat_have_sense_addr;
1709 1.1 fvdl test SCB_CDB_LEN, SCB_CDB_LEN_PTR jnz cfg4istat_have_sense_addr;
1710 1.1 fvdl /*
1711 1.1 fvdl * Host sets up address/count and enables transfer.
1712 1.1 fvdl */
1713 1.1 fvdl SET_SEQINTCODE(CFG4ISTAT_INTR)
1714 1.1 fvdl jmp cfg4istat_setup_handler;
1715 1.1 fvdl cfg4istat_have_sense_addr:
1716 1.1 fvdl bmov HADDR, SCB_SENSE_BUSADDR, 4;
1717 1.1 fvdl mvi HCNT[1], (AHD_SENSE_BUFSIZE >> 8);
1718 1.1 fvdl mvi SG_CACHE_PRE, LAST_SEG;
1719 1.1 fvdl mvi DFCNTRL, PRELOADEN|SCSIEN|HDMAEN;
1720 1.1 fvdl cfg4istat_setup_handler:
1721 1.1 fvdl /*
1722 1.1 fvdl * Status pkt is transferring to host.
1723 1.1 fvdl * Wait in idle loop for transfer to complete.
1724 1.1 fvdl * If a command completed before an attempted
1725 1.1 fvdl * task management function completed, notify the host.
1726 1.1 fvdl */
1727 1.1 fvdl test SCB_TASK_MANAGEMENT, 0xFF jz cfg4istat_no_taskmgmt_func;
1728 1.1 fvdl SET_SEQINTCODE(TASKMGMT_CMD_CMPLT_OKAY)
1729 1.1 fvdl cfg4istat_no_taskmgmt_func:
1730 1.1 fvdl call pkt_handle_status;
1731 1.1 fvdl or SEQINTCTL, IRET ret;
1732 1.1 fvdl
1733 1.1 fvdl cfg4icmd_intr:
1734 1.1 fvdl /*
1735 1.1 fvdl * In the case of DMAing a CDB from the host, the normal
1736 1.1 fvdl * CDB buffer is formatted with an 8 byte address followed
1737 1.1 fvdl * by a 1 byte count.
1738 1.1 fvdl */
1739 1.1 fvdl bmov HADDR[0], SCB_HOST_CDB_PTR, 9;
1740 1.1 fvdl mvi SG_CACHE_PRE, LAST_SEG;
1741 1.1 fvdl mvi DFCNTRL, (PRELOADEN|SCSIEN|HDMAEN);
1742 1.1 fvdl call pkt_handle_cdb;
1743 1.1 fvdl or SEQINTCTL, IRET ret;
1744 1.1 fvdl
1745 1.1 fvdl /*
1746 1.1 fvdl * See if the target has gone on in this context creating an
1747 1.1 fvdl * overrun condition. For the write case, the hardware cannot
1748 1.1 fvdl * ack bytes until data are provided. So, if the target begins
1749 1.1 fvdl * another packet without changing contexts, implying we are
1750 1.1 fvdl * not sitting on a packet boundary, we are in an overrun
1751 1.1 fvdl * situation. For the read case, the hardware will continue to
1752 1.1 fvdl * ack bytes into the FIFO, and may even ack the last overrun packet
1753 1.1 fvdl * into the FIFO. If the FIFO should become non-empty, we are in
1754 1.1 fvdl * a read overrun case.
1755 1.1 fvdl */
1756 1.1 fvdl #define check_overrun \
1757 1.1 fvdl /* Not on a packet boundary. */ \
1758 1.1 fvdl test MDFFSTAT, DLZERO jz pkt_handle_overrun; \
1759 1.1 fvdl test DFSTATUS, FIFOEMP jz pkt_handle_overrun
1760 1.1 fvdl
1761 1.1 fvdl pkt_handle_xfer:
1762 1.1 fvdl test SG_STATE, LOADING_NEEDED jz pkt_last_seg;
1763 1.1 fvdl call setjmp;
1764 1.1 fvdl test SEQINTSRC, SAVEPTRS jnz pkt_saveptrs;
1765 1.1 fvdl test SCSIPHASE, ~DATA_PHASE_MASK jz . + 2;
1766 1.1 fvdl test SCSISIGO, ATNO jnz . + 2;
1767 1.1 fvdl test SSTAT2, NONPACKREQ jz pkt_service_fifo;
1768 1.1 fvdl /*
1769 1.1 fvdl * Defer handling of this NONPACKREQ until we
1770 1.1 fvdl * can be sure it pertains to this FIFO. SAVEPTRS
1771 1.1 fvdl * will not be asserted if the NONPACKREQ is for us,
1772 1.1 fvdl * so we must simulate it if shaddow is valid. If
1773 1.1 fvdl * shaddow is not valid, keep running this FIFO until we
1774 1.1 fvdl * have satisfied the transfer by loading segments and
1775 1.1 fvdl * waiting for either shaddow valid or last_seg_done.
1776 1.1 fvdl */
1777 1.1 fvdl test MDFFSTAT, SHVALID jnz pkt_saveptrs;
1778 1.1 fvdl pkt_service_fifo:
1779 1.1 fvdl test SG_STATE, LOADING_NEEDED jnz service_fifo;
1780 1.1 fvdl pkt_last_seg:
1781 1.1 fvdl call setjmp;
1782 1.1 fvdl test SEQINTSRC, SAVEPTRS jnz pkt_saveptrs;
1783 1.4 thorpej test SG_CACHE_SHADOW, LAST_SEG_DONE jnz pkt_last_seg_done;
1784 1.1 fvdl test SCSIPHASE, ~DATA_PHASE_MASK jz . + 2;
1785 1.1 fvdl test SCSISIGO, ATNO jnz . + 2;
1786 1.1 fvdl test SSTAT2, NONPACKREQ jz return;
1787 1.1 fvdl test MDFFSTAT, SHVALID jz return;
1788 1.1 fvdl /* FALLTHROUGH */
1789 1.1 fvdl
1790 1.1 fvdl /*
1791 1.1 fvdl * Either a SAVEPTRS interrupt condition is pending for this FIFO
1792 1.4 thorpej * or we have a pending NONPACKREQ for this FIFO. We differentiate
1793 1.1 fvdl * between the two by capturing the state of the SAVEPTRS interrupt
1794 1.1 fvdl * prior to clearing this status and executing the common code for
1795 1.1 fvdl * these two cases.
1796 1.1 fvdl */
1797 1.1 fvdl pkt_saveptrs:
1798 1.1 fvdl BEGIN_CRITICAL;
1799 1.1 fvdl if ((ahd->bugs & AHD_AUTOFLUSH_BUG) != 0) {
1800 1.1 fvdl or DFCNTRL, FIFOFLUSH;
1801 1.1 fvdl }
1802 1.1 fvdl mov REG0, SEQINTSRC;
1803 1.1 fvdl call calc_residual;
1804 1.1 fvdl call save_pointers;
1805 1.1 fvdl mvi CLRSEQINTSRC, CLRSAVEPTRS;
1806 1.1 fvdl call disable_ccsgen;
1807 1.1 fvdl or SEQIMODE, ENSAVEPTRS;
1808 1.1 fvdl test DFCNTRL, DIRECTION jnz pkt_saveptrs_check_status;
1809 1.1 fvdl test DFSTATUS, FIFOEMP jnz pkt_saveptrs_check_status;
1810 1.1 fvdl /*
1811 1.1 fvdl * Keep a handler around for this FIFO until it drains
1812 1.1 fvdl * to the host to guarantee that we don't complete the
1813 1.1 fvdl * command to the host before the data arrives.
1814 1.1 fvdl */
1815 1.1 fvdl pkt_saveptrs_wait_fifoemp:
1816 1.1 fvdl call setjmp;
1817 1.1 fvdl test DFSTATUS, FIFOEMP jz return;
1818 1.1 fvdl pkt_saveptrs_check_status:
1819 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
1820 1.1 fvdl test REG0, SAVEPTRS jz unexpected_nonpkt_phase;
1821 1.4 thorpej dec SCB_FIFO_USE_COUNT;
1822 1.4 thorpej test SCB_CONTROL, STATUS_RCVD jnz pkt_complete_scb_if_fifos_idle;
1823 1.1 fvdl mvi DFFSXFRCTL, CLRCHN ret;
1824 1.1 fvdl END_CRITICAL;
1825 1.1 fvdl
1826 1.4 thorpej /*
1827 1.4 thorpej * LAST_SEG_DONE status has been seen in the current FIFO.
1828 1.4 thorpej * This indicates that all of the allowed data for this
1829 1.4 thorpej * command has transferred across the SCSI and host buses.
1830 1.4 thorpej * Check for overrun and see if we can complete this command.
1831 1.4 thorpej */
1832 1.4 thorpej pkt_last_seg_done:
1833 1.1 fvdl BEGIN_CRITICAL;
1834 1.4 thorpej /*
1835 1.4 thorpej * Mark transfer as completed.
1836 1.4 thorpej */
1837 1.1 fvdl or SCB_SGPTR, SG_LIST_NULL;
1838 1.4 thorpej
1839 1.1 fvdl /*
1840 1.4 thorpej * Wait for the current context to finish to verify that
1841 1.4 thorpej * no overrun condition has occurred.
1842 1.1 fvdl */
1843 1.4 thorpej test SEQINTSRC, CTXTDONE jnz pkt_ctxt_done;
1844 1.1 fvdl call setjmp;
1845 1.4 thorpej pkt_wait_ctxt_done_loop:
1846 1.4 thorpej test SEQINTSRC, CTXTDONE jnz pkt_ctxt_done;
1847 1.4 thorpej /*
1848 1.4 thorpej * A sufficiently large overrun or a NONPACKREQ may
1849 1.4 thorpej * prevent CTXTDONE from ever asserting, so we must
1850 1.4 thorpej * poll for these statuses too.
1851 1.4 thorpej */
1852 1.1 fvdl check_overrun;
1853 1.1 fvdl test SSTAT2, NONPACKREQ jz return;
1854 1.1 fvdl test SEQINTSRC, CTXTDONE jz unexpected_nonpkt_phase;
1855 1.4 thorpej /* FALLTHROUGH */
1856 1.4 thorpej
1857 1.4 thorpej pkt_ctxt_done:
1858 1.1 fvdl check_overrun;
1859 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
1860 1.4 thorpej /*
1861 1.4 thorpej * If status has been received, it is safe to skip
1862 1.4 thorpej * the check to see if another FIFO is active because
1863 1.4 thorpej * LAST_SEG_DONE has been observed. However, we check
1864 1.4 thorpej * the FIFO anyway since it costs us only one extra
1865 1.4 thorpej * instruction to leverage common code to perform the
1866 1.4 thorpej * SCB completion.
1867 1.4 thorpej */
1868 1.4 thorpej dec SCB_FIFO_USE_COUNT;
1869 1.4 thorpej test SCB_CONTROL, STATUS_RCVD jnz pkt_complete_scb_if_fifos_idle;
1870 1.1 fvdl mvi DFFSXFRCTL, CLRCHN ret;
1871 1.1 fvdl END_CRITICAL;
1872 1.1 fvdl
1873 1.1 fvdl /*
1874 1.4 thorpej * Must wait until CDB xfer is over before issuing the
1875 1.4 thorpej * clear channel.
1876 1.4 thorpej */
1877 1.4 thorpej pkt_handle_cdb:
1878 1.4 thorpej call setjmp;
1879 1.4 thorpej test SG_CACHE_SHADOW, LAST_SEG_DONE jz return;
1880 1.4 thorpej or LONGJMP_ADDR[1], INVALID_ADDR;
1881 1.4 thorpej mvi DFFSXFRCTL, CLRCHN ret;
1882 1.4 thorpej
1883 1.4 thorpej /*
1884 1.1 fvdl * Watch over the status transfer. Our host sense buffer is
1885 1.1 fvdl * large enough to take the maximum allowed status packet.
1886 1.1 fvdl * None-the-less, we must still catch and report overruns to
1887 1.4 thorpej * the host. Additionally, properly catch unexpected non-packet
1888 1.4 thorpej * phases that are typically caused by CRC errors in status packet
1889 1.4 thorpej * transmission.
1890 1.1 fvdl */
1891 1.1 fvdl pkt_handle_status:
1892 1.4 thorpej call setjmp;
1893 1.4 thorpej test SG_CACHE_SHADOW, LAST_SEG_DONE jnz pkt_status_check_overrun;
1894 1.4 thorpej test SEQINTSRC, CTXTDONE jz pkt_status_check_nonpackreq;
1895 1.4 thorpej test SG_CACHE_SHADOW, LAST_SEG_DONE jnz pkt_status_check_overrun;
1896 1.4 thorpej pkt_status_IU_done:
1897 1.1 fvdl if ((ahd->bugs & AHD_AUTOFLUSH_BUG) != 0) {
1898 1.1 fvdl or DFCNTRL, FIFOFLUSH;
1899 1.1 fvdl }
1900 1.4 thorpej test DFSTATUS, FIFOEMP jz return;
1901 1.4 thorpej BEGIN_CRITICAL;
1902 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
1903 1.1 fvdl mvi SCB_SCSI_STATUS, STATUS_PKT_SENSE;
1904 1.1 fvdl or SCB_CONTROL, STATUS_RCVD;
1905 1.4 thorpej jmp pkt_complete_scb_if_fifos_idle;
1906 1.1 fvdl END_CRITICAL;
1907 1.4 thorpej pkt_status_check_overrun:
1908 1.1 fvdl /*
1909 1.4 thorpej * Status PKT overruns are uncerimoniously recovered with a
1910 1.4 thorpej * bus reset. If we've overrun, let the host know so that
1911 1.4 thorpej * recovery can be performed.
1912 1.4 thorpej *
1913 1.4 thorpej * LAST_SEG_DONE has been observed. If either CTXTDONE or
1914 1.4 thorpej * a NONPACKREQ phase change have occurred and the FIFO is
1915 1.4 thorpej * empty, there is no overrun.
1916 1.4 thorpej */
1917 1.4 thorpej test DFSTATUS, FIFOEMP jz pkt_status_report_overrun;
1918 1.4 thorpej test SEQINTSRC, CTXTDONE jz . + 2;
1919 1.4 thorpej test DFSTATUS, FIFOEMP jnz pkt_status_IU_done;
1920 1.4 thorpej test SCSIPHASE, ~DATA_PHASE_MASK jz return;
1921 1.4 thorpej test DFSTATUS, FIFOEMP jnz pkt_status_check_nonpackreq;
1922 1.4 thorpej pkt_status_report_overrun:
1923 1.1 fvdl SET_SEQINTCODE(STATUS_OVERRUN)
1924 1.4 thorpej /* SEQUENCER RESTARTED */
1925 1.4 thorpej pkt_status_check_nonpackreq:
1926 1.4 thorpej /*
1927 1.4 thorpej * CTXTDONE may be held off if a NONPACKREQ is associated with
1928 1.4 thorpej * the current context. If a NONPACKREQ is observed, decide
1929 1.4 thorpej * if it is for the current context. If it is for the current
1930 1.4 thorpej * context, we must defer NONPACKREQ processing until all data
1931 1.4 thorpej * has transferred to the host.
1932 1.4 thorpej */
1933 1.4 thorpej test SCSIPHASE, ~DATA_PHASE_MASK jz return;
1934 1.4 thorpej test SCSISIGO, ATNO jnz . + 2;
1935 1.4 thorpej test SSTAT2, NONPACKREQ jz return;
1936 1.4 thorpej test SEQINTSRC, CTXTDONE jnz pkt_status_IU_done;
1937 1.4 thorpej test DFSTATUS, FIFOEMP jz return;
1938 1.4 thorpej /*
1939 1.4 thorpej * The unexpected nonpkt phase handler assumes that any
1940 1.4 thorpej * data channel use will have a FIFO reference count. It
1941 1.4 thorpej * turns out that the status handler doesn't need a refernce
1942 1.4 thorpej * count since the status received flag, and thus completion
1943 1.4 thorpej * processing, cannot be set until the handler is finished.
1944 1.4 thorpej * We increment the count here to make the nonpkt handler
1945 1.4 thorpej * happy.
1946 1.4 thorpej */
1947 1.4 thorpej inc SCB_FIFO_USE_COUNT;
1948 1.4 thorpej /* FALLTHROUGH */
1949 1.1 fvdl
1950 1.1 fvdl /*
1951 1.1 fvdl * Nonpackreq is a polled status. It can come true in three situations:
1952 1.1 fvdl * we have received an L_Q, we have sent one or more L_Qs, or there is no
1953 1.1 fvdl * L_Q context associated with this REQ (REQ occurs immediately after a
1954 1.1 fvdl * (re)selection). Routines that know that the context responsible for this
1955 1.1 fvdl * nonpackreq call directly into unexpected_nonpkt_phase. In the case of the
1956 1.1 fvdl * top level idle loop, we exhaust all active contexts prior to determining that
1957 1.1 fvdl * we simply do not have the full I_T_L_Q for this phase.
1958 1.1 fvdl */
1959 1.1 fvdl unexpected_nonpkt_phase_find_ctxt:
1960 1.1 fvdl /*
1961 1.1 fvdl * This nonpackreq is most likely associated with one of the tags
1962 1.1 fvdl * in a FIFO or an outgoing LQ. Only treat it as an I_T only
1963 1.1 fvdl * nonpackreq if we've cleared out the FIFOs and handled any
1964 1.1 fvdl * pending SELDO.
1965 1.1 fvdl */
1966 1.1 fvdl SET_SRC_MODE M_SCSI;
1967 1.1 fvdl SET_DST_MODE M_SCSI;
1968 1.1 fvdl and A, FIFO1FREE|FIFO0FREE, DFFSTAT;
1969 1.1 fvdl cmp A, FIFO1FREE|FIFO0FREE jne return;
1970 1.1 fvdl test SSTAT0, SELDO jnz return;
1971 1.1 fvdl mvi SCBPTR[1], SCB_LIST_NULL;
1972 1.1 fvdl unexpected_nonpkt_phase:
1973 1.1 fvdl test MODE_PTR, ~(MK_MODE(M_DFF1, M_DFF1)) jnz . + 3;
1974 1.1 fvdl SET_SRC_MODE M_DFF0;
1975 1.1 fvdl SET_DST_MODE M_DFF0;
1976 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
1977 1.4 thorpej dec SCB_FIFO_USE_COUNT;
1978 1.1 fvdl mvi DFFSXFRCTL, CLRCHN;
1979 1.1 fvdl mvi CLRSINT2, CLRNONPACKREQ;
1980 1.1 fvdl test SCSIPHASE, ~(MSG_IN_PHASE|MSG_OUT_PHASE) jnz illegal_phase;
1981 1.1 fvdl SET_SEQINTCODE(ENTERING_NONPACK)
1982 1.1 fvdl jmp ITloop;
1983 1.1 fvdl
1984 1.1 fvdl illegal_phase:
1985 1.1 fvdl SET_SEQINTCODE(ILLEGAL_PHASE)
1986 1.1 fvdl jmp ITloop;
1987 1.1 fvdl
1988 1.1 fvdl /*
1989 1.1 fvdl * We have entered an overrun situation. If we have working
1990 1.1 fvdl * BITBUCKET, flip that on and let the hardware eat any overrun
1991 1.1 fvdl * data. Otherwise use an overrun buffer in the host to simulate
1992 1.1 fvdl * BITBUCKET.
1993 1.1 fvdl */
1994 1.4 thorpej pkt_handle_overrun_inc_use_count:
1995 1.4 thorpej inc SCB_FIFO_USE_COUNT;
1996 1.1 fvdl pkt_handle_overrun:
1997 1.1 fvdl SET_SEQINTCODE(CFG4OVERRUN)
1998 1.1 fvdl call freeze_queue;
1999 1.1 fvdl if ((ahd->bugs & AHD_PKT_BITBUCKET_BUG) == 0) {
2000 1.1 fvdl or DFFSXFRCTL, DFFBITBUCKET;
2001 1.1 fvdl SET_SRC_MODE M_DFF1;
2002 1.1 fvdl SET_DST_MODE M_DFF1;
2003 1.1 fvdl } else {
2004 1.1 fvdl call load_overrun_buf;
2005 1.1 fvdl mvi DFCNTRL, (HDMAEN|SCSIEN|PRELOADEN);
2006 1.1 fvdl }
2007 1.1 fvdl call setjmp;
2008 1.1 fvdl if ((ahd->bugs & AHD_PKT_BITBUCKET_BUG) != 0) {
2009 1.1 fvdl test DFSTATUS, PRELOAD_AVAIL jz overrun_load_done;
2010 1.1 fvdl call load_overrun_buf;
2011 1.1 fvdl or DFCNTRL, PRELOADEN;
2012 1.1 fvdl overrun_load_done:
2013 1.1 fvdl test SEQINTSRC, CTXTDONE jnz pkt_overrun_end;
2014 1.1 fvdl } else {
2015 1.1 fvdl test DFFSXFRCTL, DFFBITBUCKET jz pkt_overrun_end;
2016 1.1 fvdl }
2017 1.1 fvdl test SSTAT2, NONPACKREQ jz return;
2018 1.1 fvdl pkt_overrun_end:
2019 1.1 fvdl or SCB_RESIDUAL_SGPTR, SG_OVERRUN_RESID;
2020 1.1 fvdl test SEQINTSRC, CTXTDONE jz unexpected_nonpkt_phase;
2021 1.4 thorpej dec SCB_FIFO_USE_COUNT;
2022 1.1 fvdl or LONGJMP_ADDR[1], INVALID_ADDR;
2023 1.4 thorpej test SCB_CONTROL, STATUS_RCVD jnz pkt_complete_scb_if_fifos_idle;
2024 1.1 fvdl mvi DFFSXFRCTL, CLRCHN ret;
2025 1.1 fvdl
2026 1.1 fvdl if ((ahd->bugs & AHD_PKT_BITBUCKET_BUG) != 0) {
2027 1.1 fvdl load_overrun_buf:
2028 1.1 fvdl /*
2029 1.1 fvdl * Load a dummy segment if preload space is available.
2030 1.1 fvdl */
2031 1.1 fvdl mov HADDR[0], SHARED_DATA_ADDR;
2032 1.1 fvdl add HADDR[1], PKT_OVERRUN_BUFOFFSET, SHARED_DATA_ADDR[1];
2033 1.1 fvdl mov ACCUM_SAVE, A;
2034 1.1 fvdl clr A;
2035 1.1 fvdl adc HADDR[2], A, SHARED_DATA_ADDR[2];
2036 1.1 fvdl adc HADDR[3], A, SHARED_DATA_ADDR[3];
2037 1.1 fvdl mov A, ACCUM_SAVE;
2038 1.1 fvdl bmov HADDR[4], ALLZEROS, 4;
2039 1.1 fvdl /* PKT_OVERRUN_BUFSIZE is a multiple of 256 */
2040 1.1 fvdl clr HCNT[0];
2041 1.1 fvdl mvi HCNT[1], ((PKT_OVERRUN_BUFSIZE >> 8) & 0xFF);
2042 1.1 fvdl clr HCNT[2] ret;
2043 1.1 fvdl }
2044