109c434b8SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2cd9ad58dSDavid S. Miller /* esp_scsi.c: ESP SCSI driver.
3cd9ad58dSDavid S. Miller *
4cd9ad58dSDavid S. Miller * Copyright (C) 2007 David S. Miller (davem@davemloft.net)
5cd9ad58dSDavid S. Miller */
6cd9ad58dSDavid S. Miller
7cd9ad58dSDavid S. Miller #include <linux/kernel.h>
8cd9ad58dSDavid S. Miller #include <linux/types.h>
9cd9ad58dSDavid S. Miller #include <linux/slab.h>
10cd9ad58dSDavid S. Miller #include <linux/delay.h>
11cd9ad58dSDavid S. Miller #include <linux/list.h>
12cd9ad58dSDavid S. Miller #include <linux/completion.h>
13cd9ad58dSDavid S. Miller #include <linux/kallsyms.h>
14cd9ad58dSDavid S. Miller #include <linux/module.h>
15cd9ad58dSDavid S. Miller #include <linux/moduleparam.h>
16cd9ad58dSDavid S. Miller #include <linux/init.h>
17e1f2a094SAlexey Dobriyan #include <linux/irqreturn.h>
18cd9ad58dSDavid S. Miller
19cd9ad58dSDavid S. Miller #include <asm/irq.h>
20cd9ad58dSDavid S. Miller #include <asm/io.h>
21cd9ad58dSDavid S. Miller #include <asm/dma.h>
22cd9ad58dSDavid S. Miller
23cd9ad58dSDavid S. Miller #include <scsi/scsi.h>
24cd9ad58dSDavid S. Miller #include <scsi/scsi_host.h>
25cd9ad58dSDavid S. Miller #include <scsi/scsi_cmnd.h>
26cd9ad58dSDavid S. Miller #include <scsi/scsi_device.h>
27cd9ad58dSDavid S. Miller #include <scsi/scsi_tcq.h>
28cd9ad58dSDavid S. Miller #include <scsi/scsi_dbg.h>
29cd9ad58dSDavid S. Miller #include <scsi/scsi_transport_spi.h>
30cd9ad58dSDavid S. Miller
31cd9ad58dSDavid S. Miller #include "esp_scsi.h"
32cd9ad58dSDavid S. Miller
33cd9ad58dSDavid S. Miller #define DRV_MODULE_NAME "esp"
34cd9ad58dSDavid S. Miller #define PFX DRV_MODULE_NAME ": "
35cd9ad58dSDavid S. Miller #define DRV_VERSION "2.000"
36cd9ad58dSDavid S. Miller #define DRV_MODULE_RELDATE "April 19, 2007"
37cd9ad58dSDavid S. Miller
38cd9ad58dSDavid S. Miller /* SCSI bus reset settle time in seconds. */
39cd9ad58dSDavid S. Miller static int esp_bus_reset_settle = 3;
40cd9ad58dSDavid S. Miller
41cd9ad58dSDavid S. Miller static u32 esp_debug;
42cd9ad58dSDavid S. Miller #define ESP_DEBUG_INTR 0x00000001
43cd9ad58dSDavid S. Miller #define ESP_DEBUG_SCSICMD 0x00000002
44cd9ad58dSDavid S. Miller #define ESP_DEBUG_RESET 0x00000004
45cd9ad58dSDavid S. Miller #define ESP_DEBUG_MSGIN 0x00000008
46cd9ad58dSDavid S. Miller #define ESP_DEBUG_MSGOUT 0x00000010
47cd9ad58dSDavid S. Miller #define ESP_DEBUG_CMDDONE 0x00000020
48cd9ad58dSDavid S. Miller #define ESP_DEBUG_DISCONNECT 0x00000040
49cd9ad58dSDavid S. Miller #define ESP_DEBUG_DATASTART 0x00000080
50cd9ad58dSDavid S. Miller #define ESP_DEBUG_DATADONE 0x00000100
51cd9ad58dSDavid S. Miller #define ESP_DEBUG_RECONNECT 0x00000200
52cd9ad58dSDavid S. Miller #define ESP_DEBUG_AUTOSENSE 0x00000400
531af6f603SHannes Reinecke #define ESP_DEBUG_EVENT 0x00000800
541af6f603SHannes Reinecke #define ESP_DEBUG_COMMAND 0x00001000
55cd9ad58dSDavid S. Miller
56cd9ad58dSDavid S. Miller #define esp_log_intr(f, a...) \
57cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_INTR) \
58a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
59cd9ad58dSDavid S. Miller } while (0)
60cd9ad58dSDavid S. Miller
61cd9ad58dSDavid S. Miller #define esp_log_reset(f, a...) \
62cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_RESET) \
63a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
64cd9ad58dSDavid S. Miller } while (0)
65cd9ad58dSDavid S. Miller
66cd9ad58dSDavid S. Miller #define esp_log_msgin(f, a...) \
67cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_MSGIN) \
68a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
69cd9ad58dSDavid S. Miller } while (0)
70cd9ad58dSDavid S. Miller
71cd9ad58dSDavid S. Miller #define esp_log_msgout(f, a...) \
72cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_MSGOUT) \
73a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
74cd9ad58dSDavid S. Miller } while (0)
75cd9ad58dSDavid S. Miller
76cd9ad58dSDavid S. Miller #define esp_log_cmddone(f, a...) \
77cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_CMDDONE) \
78a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
79cd9ad58dSDavid S. Miller } while (0)
80cd9ad58dSDavid S. Miller
81cd9ad58dSDavid S. Miller #define esp_log_disconnect(f, a...) \
82cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_DISCONNECT) \
83a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
84cd9ad58dSDavid S. Miller } while (0)
85cd9ad58dSDavid S. Miller
86cd9ad58dSDavid S. Miller #define esp_log_datastart(f, a...) \
87cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_DATASTART) \
88a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
89cd9ad58dSDavid S. Miller } while (0)
90cd9ad58dSDavid S. Miller
91cd9ad58dSDavid S. Miller #define esp_log_datadone(f, a...) \
92cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_DATADONE) \
93a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
94cd9ad58dSDavid S. Miller } while (0)
95cd9ad58dSDavid S. Miller
96cd9ad58dSDavid S. Miller #define esp_log_reconnect(f, a...) \
97cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_RECONNECT) \
98a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
99cd9ad58dSDavid S. Miller } while (0)
100cd9ad58dSDavid S. Miller
101cd9ad58dSDavid S. Miller #define esp_log_autosense(f, a...) \
102cd9ad58dSDavid S. Miller do { if (esp_debug & ESP_DEBUG_AUTOSENSE) \
103a1a75b35SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
104cd9ad58dSDavid S. Miller } while (0)
105cd9ad58dSDavid S. Miller
1061af6f603SHannes Reinecke #define esp_log_event(f, a...) \
1071af6f603SHannes Reinecke do { if (esp_debug & ESP_DEBUG_EVENT) \
1081af6f603SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
1091af6f603SHannes Reinecke } while (0)
1101af6f603SHannes Reinecke
1111af6f603SHannes Reinecke #define esp_log_command(f, a...) \
1121af6f603SHannes Reinecke do { if (esp_debug & ESP_DEBUG_COMMAND) \
1131af6f603SHannes Reinecke shost_printk(KERN_DEBUG, esp->host, f, ## a); \
1141af6f603SHannes Reinecke } while (0)
1151af6f603SHannes Reinecke
116cd9ad58dSDavid S. Miller #define esp_read8(REG) esp->ops->esp_read8(esp, REG)
117cd9ad58dSDavid S. Miller #define esp_write8(VAL,REG) esp->ops->esp_write8(esp, VAL, REG)
118cd9ad58dSDavid S. Miller
esp_log_fill_regs(struct esp * esp,struct esp_event_ent * p)119cd9ad58dSDavid S. Miller static void esp_log_fill_regs(struct esp *esp,
120cd9ad58dSDavid S. Miller struct esp_event_ent *p)
121cd9ad58dSDavid S. Miller {
122cd9ad58dSDavid S. Miller p->sreg = esp->sreg;
123cd9ad58dSDavid S. Miller p->seqreg = esp->seqreg;
124cd9ad58dSDavid S. Miller p->sreg2 = esp->sreg2;
125cd9ad58dSDavid S. Miller p->ireg = esp->ireg;
126cd9ad58dSDavid S. Miller p->select_state = esp->select_state;
127cd9ad58dSDavid S. Miller p->event = esp->event;
128cd9ad58dSDavid S. Miller }
129cd9ad58dSDavid S. Miller
scsi_esp_cmd(struct esp * esp,u8 val)130cd9ad58dSDavid S. Miller void scsi_esp_cmd(struct esp *esp, u8 val)
131cd9ad58dSDavid S. Miller {
132cd9ad58dSDavid S. Miller struct esp_event_ent *p;
133cd9ad58dSDavid S. Miller int idx = esp->esp_event_cur;
134cd9ad58dSDavid S. Miller
135cd9ad58dSDavid S. Miller p = &esp->esp_event_log[idx];
136cd9ad58dSDavid S. Miller p->type = ESP_EVENT_TYPE_CMD;
137cd9ad58dSDavid S. Miller p->val = val;
138cd9ad58dSDavid S. Miller esp_log_fill_regs(esp, p);
139cd9ad58dSDavid S. Miller
140cd9ad58dSDavid S. Miller esp->esp_event_cur = (idx + 1) & (ESP_EVENT_LOG_SZ - 1);
141cd9ad58dSDavid S. Miller
1421af6f603SHannes Reinecke esp_log_command("cmd[%02x]\n", val);
143cd9ad58dSDavid S. Miller esp_write8(val, ESP_CMD);
144cd9ad58dSDavid S. Miller }
145cd9ad58dSDavid S. Miller EXPORT_SYMBOL(scsi_esp_cmd);
146cd9ad58dSDavid S. Miller
esp_send_dma_cmd(struct esp * esp,int len,int max_len,int cmd)1473170866fSHannes Reinecke static void esp_send_dma_cmd(struct esp *esp, int len, int max_len, int cmd)
1483170866fSHannes Reinecke {
1493170866fSHannes Reinecke if (esp->flags & ESP_FLAG_USE_FIFO) {
1503170866fSHannes Reinecke int i;
1513170866fSHannes Reinecke
1523170866fSHannes Reinecke scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1533170866fSHannes Reinecke for (i = 0; i < len; i++)
1543170866fSHannes Reinecke esp_write8(esp->command_block[i], ESP_FDATA);
1553170866fSHannes Reinecke scsi_esp_cmd(esp, cmd);
1563170866fSHannes Reinecke } else {
1573170866fSHannes Reinecke if (esp->rev == FASHME)
1583170866fSHannes Reinecke scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1593170866fSHannes Reinecke cmd |= ESP_CMD_DMA;
1603170866fSHannes Reinecke esp->ops->send_dma_cmd(esp, esp->command_block_dma,
1613170866fSHannes Reinecke len, max_len, 0, cmd);
1623170866fSHannes Reinecke }
1633170866fSHannes Reinecke }
1643170866fSHannes Reinecke
esp_event(struct esp * esp,u8 val)165cd9ad58dSDavid S. Miller static void esp_event(struct esp *esp, u8 val)
166cd9ad58dSDavid S. Miller {
167cd9ad58dSDavid S. Miller struct esp_event_ent *p;
168cd9ad58dSDavid S. Miller int idx = esp->esp_event_cur;
169cd9ad58dSDavid S. Miller
170cd9ad58dSDavid S. Miller p = &esp->esp_event_log[idx];
171cd9ad58dSDavid S. Miller p->type = ESP_EVENT_TYPE_EVENT;
172cd9ad58dSDavid S. Miller p->val = val;
173cd9ad58dSDavid S. Miller esp_log_fill_regs(esp, p);
174cd9ad58dSDavid S. Miller
175cd9ad58dSDavid S. Miller esp->esp_event_cur = (idx + 1) & (ESP_EVENT_LOG_SZ - 1);
176cd9ad58dSDavid S. Miller
177cd9ad58dSDavid S. Miller esp->event = val;
178cd9ad58dSDavid S. Miller }
179cd9ad58dSDavid S. Miller
esp_dump_cmd_log(struct esp * esp)180cd9ad58dSDavid S. Miller static void esp_dump_cmd_log(struct esp *esp)
181cd9ad58dSDavid S. Miller {
182cd9ad58dSDavid S. Miller int idx = esp->esp_event_cur;
183cd9ad58dSDavid S. Miller int stop = idx;
184cd9ad58dSDavid S. Miller
185a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host, "Dumping command log\n");
186cd9ad58dSDavid S. Miller do {
187cd9ad58dSDavid S. Miller struct esp_event_ent *p = &esp->esp_event_log[idx];
188cd9ad58dSDavid S. Miller
189a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
190a1a75b35SHannes Reinecke "ent[%d] %s val[%02x] sreg[%02x] seqreg[%02x] "
191cd9ad58dSDavid S. Miller "sreg2[%02x] ireg[%02x] ss[%02x] event[%02x]\n",
192a1a75b35SHannes Reinecke idx,
193a1a75b35SHannes Reinecke p->type == ESP_EVENT_TYPE_CMD ? "CMD" : "EVENT",
194cd9ad58dSDavid S. Miller p->val, p->sreg, p->seqreg,
195cd9ad58dSDavid S. Miller p->sreg2, p->ireg, p->select_state, p->event);
196cd9ad58dSDavid S. Miller
197cd9ad58dSDavid S. Miller idx = (idx + 1) & (ESP_EVENT_LOG_SZ - 1);
198cd9ad58dSDavid S. Miller } while (idx != stop);
199cd9ad58dSDavid S. Miller }
200cd9ad58dSDavid S. Miller
esp_flush_fifo(struct esp * esp)201cd9ad58dSDavid S. Miller static void esp_flush_fifo(struct esp *esp)
202cd9ad58dSDavid S. Miller {
203cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
204cd9ad58dSDavid S. Miller if (esp->rev == ESP236) {
205cd9ad58dSDavid S. Miller int lim = 1000;
206cd9ad58dSDavid S. Miller
207cd9ad58dSDavid S. Miller while (esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES) {
208cd9ad58dSDavid S. Miller if (--lim == 0) {
209a1a75b35SHannes Reinecke shost_printk(KERN_ALERT, esp->host,
210a1a75b35SHannes Reinecke "ESP_FF_BYTES will not clear!\n");
211cd9ad58dSDavid S. Miller break;
212cd9ad58dSDavid S. Miller }
213cd9ad58dSDavid S. Miller udelay(1);
214cd9ad58dSDavid S. Miller }
215cd9ad58dSDavid S. Miller }
216cd9ad58dSDavid S. Miller }
217cd9ad58dSDavid S. Miller
hme_read_fifo(struct esp * esp)218cd9ad58dSDavid S. Miller static void hme_read_fifo(struct esp *esp)
219cd9ad58dSDavid S. Miller {
220cd9ad58dSDavid S. Miller int fcnt = esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES;
221cd9ad58dSDavid S. Miller int idx = 0;
222cd9ad58dSDavid S. Miller
223cd9ad58dSDavid S. Miller while (fcnt--) {
224cd9ad58dSDavid S. Miller esp->fifo[idx++] = esp_read8(ESP_FDATA);
225cd9ad58dSDavid S. Miller esp->fifo[idx++] = esp_read8(ESP_FDATA);
226cd9ad58dSDavid S. Miller }
227cd9ad58dSDavid S. Miller if (esp->sreg2 & ESP_STAT2_F1BYTE) {
228cd9ad58dSDavid S. Miller esp_write8(0, ESP_FDATA);
229cd9ad58dSDavid S. Miller esp->fifo[idx++] = esp_read8(ESP_FDATA);
230cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
231cd9ad58dSDavid S. Miller }
232cd9ad58dSDavid S. Miller esp->fifo_cnt = idx;
233cd9ad58dSDavid S. Miller }
234cd9ad58dSDavid S. Miller
esp_set_all_config3(struct esp * esp,u8 val)235cd9ad58dSDavid S. Miller static void esp_set_all_config3(struct esp *esp, u8 val)
236cd9ad58dSDavid S. Miller {
237cd9ad58dSDavid S. Miller int i;
238cd9ad58dSDavid S. Miller
239cd9ad58dSDavid S. Miller for (i = 0; i < ESP_MAX_TARGET; i++)
240cd9ad58dSDavid S. Miller esp->target[i].esp_config3 = val;
241cd9ad58dSDavid S. Miller }
242cd9ad58dSDavid S. Miller
243cd9ad58dSDavid S. Miller /* Reset the ESP chip, _not_ the SCSI bus. */
esp_reset_esp(struct esp * esp)244cd9ad58dSDavid S. Miller static void esp_reset_esp(struct esp *esp)
245cd9ad58dSDavid S. Miller {
246cd9ad58dSDavid S. Miller /* Now reset the ESP chip */
247cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RC);
248cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL | ESP_CMD_DMA);
249a793804fSDavid S. Miller if (esp->rev == FAST)
250a793804fSDavid S. Miller esp_write8(ESP_CONFIG2_FENAB, ESP_CFG2);
251cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL | ESP_CMD_DMA);
252cd9ad58dSDavid S. Miller
253cd9ad58dSDavid S. Miller /* This is the only point at which it is reliable to read
254cd9ad58dSDavid S. Miller * the ID-code for a fast ESP chip variants.
255cd9ad58dSDavid S. Miller */
256cd9ad58dSDavid S. Miller esp->max_period = ((35 * esp->ccycle) / 1000);
257cd9ad58dSDavid S. Miller if (esp->rev == FAST) {
258bd407261SKars de Jong u8 family_code = ESP_FAMILY(esp_read8(ESP_UID));
259bd407261SKars de Jong
260bd407261SKars de Jong if (family_code == ESP_UID_F236) {
261cd9ad58dSDavid S. Miller esp->rev = FAS236;
262bd407261SKars de Jong } else if (family_code == ESP_UID_HME) {
263cd9ad58dSDavid S. Miller esp->rev = FASHME; /* Version is usually '5'. */
264bd407261SKars de Jong } else if (family_code == ESP_UID_FSC) {
265bd407261SKars de Jong esp->rev = FSC;
266bd407261SKars de Jong /* Enable Active Negation */
267bd407261SKars de Jong esp_write8(ESP_CONFIG4_RADE, ESP_CFG4);
268bd407261SKars de Jong } else {
269cd9ad58dSDavid S. Miller esp->rev = FAS100A;
270bd407261SKars de Jong }
271cd9ad58dSDavid S. Miller esp->min_period = ((4 * esp->ccycle) / 1000);
272cd9ad58dSDavid S. Miller } else {
273cd9ad58dSDavid S. Miller esp->min_period = ((5 * esp->ccycle) / 1000);
274cd9ad58dSDavid S. Miller }
275eeea2f9cSHannes Reinecke if (esp->rev == FAS236) {
276eeea2f9cSHannes Reinecke /*
277eeea2f9cSHannes Reinecke * The AM53c974 chip returns the same ID as FAS236;
278eeea2f9cSHannes Reinecke * try to configure glitch eater.
279eeea2f9cSHannes Reinecke */
280eeea2f9cSHannes Reinecke u8 config4 = ESP_CONFIG4_GE1;
281eeea2f9cSHannes Reinecke esp_write8(config4, ESP_CFG4);
282eeea2f9cSHannes Reinecke config4 = esp_read8(ESP_CFG4);
283eeea2f9cSHannes Reinecke if (config4 & ESP_CONFIG4_GE1) {
284eeea2f9cSHannes Reinecke esp->rev = PCSCSI;
285eeea2f9cSHannes Reinecke esp_write8(esp->config4, ESP_CFG4);
286eeea2f9cSHannes Reinecke }
287eeea2f9cSHannes Reinecke }
288cd9ad58dSDavid S. Miller esp->max_period = (esp->max_period + 3)>>2;
289cd9ad58dSDavid S. Miller esp->min_period = (esp->min_period + 3)>>2;
290cd9ad58dSDavid S. Miller
291cd9ad58dSDavid S. Miller esp_write8(esp->config1, ESP_CFG1);
292cd9ad58dSDavid S. Miller switch (esp->rev) {
293cd9ad58dSDavid S. Miller case ESP100:
294cd9ad58dSDavid S. Miller /* nothing to do */
295cd9ad58dSDavid S. Miller break;
296cd9ad58dSDavid S. Miller
297cd9ad58dSDavid S. Miller case ESP100A:
298cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
299cd9ad58dSDavid S. Miller break;
300cd9ad58dSDavid S. Miller
301cd9ad58dSDavid S. Miller case ESP236:
302cd9ad58dSDavid S. Miller /* Slow 236 */
303cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
304cd9ad58dSDavid S. Miller esp->prev_cfg3 = esp->target[0].esp_config3;
305cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
306cd9ad58dSDavid S. Miller break;
307cd9ad58dSDavid S. Miller
308cd9ad58dSDavid S. Miller case FASHME:
309cd9ad58dSDavid S. Miller esp->config2 |= (ESP_CONFIG2_HME32 | ESP_CONFIG2_HMEFENAB);
310df561f66SGustavo A. R. Silva fallthrough;
311cd9ad58dSDavid S. Miller
312cd9ad58dSDavid S. Miller case FAS236:
313eeea2f9cSHannes Reinecke case PCSCSI:
314bd407261SKars de Jong case FSC:
315cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
316cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
317cd9ad58dSDavid S. Miller u8 cfg3 = esp->target[0].esp_config3;
318cd9ad58dSDavid S. Miller
319cd9ad58dSDavid S. Miller cfg3 |= ESP_CONFIG3_FCLOCK | ESP_CONFIG3_OBPUSH;
320cd9ad58dSDavid S. Miller if (esp->scsi_id >= 8)
321cd9ad58dSDavid S. Miller cfg3 |= ESP_CONFIG3_IDBIT3;
322cd9ad58dSDavid S. Miller esp_set_all_config3(esp, cfg3);
323cd9ad58dSDavid S. Miller } else {
324cd9ad58dSDavid S. Miller u32 cfg3 = esp->target[0].esp_config3;
325cd9ad58dSDavid S. Miller
326cd9ad58dSDavid S. Miller cfg3 |= ESP_CONFIG3_FCLK;
327cd9ad58dSDavid S. Miller esp_set_all_config3(esp, cfg3);
328cd9ad58dSDavid S. Miller }
329cd9ad58dSDavid S. Miller esp->prev_cfg3 = esp->target[0].esp_config3;
330cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
331cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
332cd9ad58dSDavid S. Miller esp->radelay = 80;
333cd9ad58dSDavid S. Miller } else {
334cd9ad58dSDavid S. Miller if (esp->flags & ESP_FLAG_DIFFERENTIAL)
335cd9ad58dSDavid S. Miller esp->radelay = 0;
336cd9ad58dSDavid S. Miller else
337cd9ad58dSDavid S. Miller esp->radelay = 96;
338cd9ad58dSDavid S. Miller }
339cd9ad58dSDavid S. Miller break;
340cd9ad58dSDavid S. Miller
341cd9ad58dSDavid S. Miller case FAS100A:
342cd9ad58dSDavid S. Miller /* Fast 100a */
343cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
344cd9ad58dSDavid S. Miller esp_set_all_config3(esp,
345cd9ad58dSDavid S. Miller (esp->target[0].esp_config3 |
346cd9ad58dSDavid S. Miller ESP_CONFIG3_FCLOCK));
347cd9ad58dSDavid S. Miller esp->prev_cfg3 = esp->target[0].esp_config3;
348cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
349cd9ad58dSDavid S. Miller esp->radelay = 32;
350cd9ad58dSDavid S. Miller break;
351cd9ad58dSDavid S. Miller
352cd9ad58dSDavid S. Miller default:
353cd9ad58dSDavid S. Miller break;
354cd9ad58dSDavid S. Miller }
355cd9ad58dSDavid S. Miller
356a793804fSDavid S. Miller /* Reload the configuration registers */
357a793804fSDavid S. Miller esp_write8(esp->cfact, ESP_CFACT);
358a793804fSDavid S. Miller
359a793804fSDavid S. Miller esp->prev_stp = 0;
360a793804fSDavid S. Miller esp_write8(esp->prev_stp, ESP_STP);
361a793804fSDavid S. Miller
362a793804fSDavid S. Miller esp->prev_soff = 0;
363a793804fSDavid S. Miller esp_write8(esp->prev_soff, ESP_SOFF);
364a793804fSDavid S. Miller
365a793804fSDavid S. Miller esp_write8(esp->neg_defp, ESP_TIMEO);
366a793804fSDavid S. Miller
367cd9ad58dSDavid S. Miller /* Eat any bitrot in the chip */
368cd9ad58dSDavid S. Miller esp_read8(ESP_INTRPT);
369cd9ad58dSDavid S. Miller udelay(100);
370cd9ad58dSDavid S. Miller }
371cd9ad58dSDavid S. Miller
esp_map_dma(struct esp * esp,struct scsi_cmnd * cmd)372cd9ad58dSDavid S. Miller static void esp_map_dma(struct esp *esp, struct scsi_cmnd *cmd)
373cd9ad58dSDavid S. Miller {
374cd9ad58dSDavid S. Miller struct esp_cmd_priv *spriv = ESP_CMD_PRIV(cmd);
3754c2baaafSFUJITA Tomonori struct scatterlist *sg = scsi_sglist(cmd);
3763f9295b6SChristoph Hellwig int total = 0, i;
377ee5a1dbfSMing Lei struct scatterlist *s;
378cd9ad58dSDavid S. Miller
3793f9295b6SChristoph Hellwig if (cmd->sc_data_direction == DMA_NONE)
380cd9ad58dSDavid S. Miller return;
381cd9ad58dSDavid S. Miller
3823f9295b6SChristoph Hellwig if (esp->flags & ESP_FLAG_NO_DMA_MAP) {
3833f9295b6SChristoph Hellwig /*
3843f9295b6SChristoph Hellwig * For pseudo DMA and PIO we need the virtual address instead of
3853f9295b6SChristoph Hellwig * a dma address, so perform an identity mapping.
3863f9295b6SChristoph Hellwig */
38786117d7fSChristoph Hellwig spriv->num_sg = scsi_sg_count(cmd);
388ee5a1dbfSMing Lei
389ee5a1dbfSMing Lei scsi_for_each_sg(cmd, s, spriv->num_sg, i) {
390ee5a1dbfSMing Lei s->dma_address = (uintptr_t)sg_virt(s);
391ee5a1dbfSMing Lei total += sg_dma_len(s);
3923f9295b6SChristoph Hellwig }
3933f9295b6SChristoph Hellwig } else {
39486117d7fSChristoph Hellwig spriv->num_sg = scsi_dma_map(cmd);
395ee5a1dbfSMing Lei scsi_for_each_sg(cmd, s, spriv->num_sg, i)
396ee5a1dbfSMing Lei total += sg_dma_len(s);
3973f9295b6SChristoph Hellwig }
3983f9295b6SChristoph Hellwig spriv->cur_residue = sg_dma_len(sg);
399ee5a1dbfSMing Lei spriv->prv_sg = NULL;
4003f9295b6SChristoph Hellwig spriv->cur_sg = sg;
401cd9ad58dSDavid S. Miller spriv->tot_residue = total;
402cd9ad58dSDavid S. Miller }
403cd9ad58dSDavid S. Miller
esp_cur_dma_addr(struct esp_cmd_entry * ent,struct scsi_cmnd * cmd)404cd9ad58dSDavid S. Miller static dma_addr_t esp_cur_dma_addr(struct esp_cmd_entry *ent,
405cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd)
406cd9ad58dSDavid S. Miller {
407cd9ad58dSDavid S. Miller struct esp_cmd_priv *p = ESP_CMD_PRIV(cmd);
408cd9ad58dSDavid S. Miller
409cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
410cd9ad58dSDavid S. Miller return ent->sense_dma +
411cd9ad58dSDavid S. Miller (ent->sense_ptr - cmd->sense_buffer);
412cd9ad58dSDavid S. Miller }
413cd9ad58dSDavid S. Miller
414cd9ad58dSDavid S. Miller return sg_dma_address(p->cur_sg) +
415cd9ad58dSDavid S. Miller (sg_dma_len(p->cur_sg) -
416cd9ad58dSDavid S. Miller p->cur_residue);
417cd9ad58dSDavid S. Miller }
418cd9ad58dSDavid S. Miller
esp_cur_dma_len(struct esp_cmd_entry * ent,struct scsi_cmnd * cmd)419cd9ad58dSDavid S. Miller static unsigned int esp_cur_dma_len(struct esp_cmd_entry *ent,
420cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd)
421cd9ad58dSDavid S. Miller {
422cd9ad58dSDavid S. Miller struct esp_cmd_priv *p = ESP_CMD_PRIV(cmd);
423cd9ad58dSDavid S. Miller
424cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
425cd9ad58dSDavid S. Miller return SCSI_SENSE_BUFFERSIZE -
426cd9ad58dSDavid S. Miller (ent->sense_ptr - cmd->sense_buffer);
427cd9ad58dSDavid S. Miller }
428cd9ad58dSDavid S. Miller return p->cur_residue;
429cd9ad58dSDavid S. Miller }
430cd9ad58dSDavid S. Miller
esp_advance_dma(struct esp * esp,struct esp_cmd_entry * ent,struct scsi_cmnd * cmd,unsigned int len)431cd9ad58dSDavid S. Miller static void esp_advance_dma(struct esp *esp, struct esp_cmd_entry *ent,
432cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd, unsigned int len)
433cd9ad58dSDavid S. Miller {
434cd9ad58dSDavid S. Miller struct esp_cmd_priv *p = ESP_CMD_PRIV(cmd);
435cd9ad58dSDavid S. Miller
436cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
437cd9ad58dSDavid S. Miller ent->sense_ptr += len;
438cd9ad58dSDavid S. Miller return;
439cd9ad58dSDavid S. Miller }
440cd9ad58dSDavid S. Miller
441cd9ad58dSDavid S. Miller p->cur_residue -= len;
442cd9ad58dSDavid S. Miller p->tot_residue -= len;
443cd9ad58dSDavid S. Miller if (p->cur_residue < 0 || p->tot_residue < 0) {
444a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
445a1a75b35SHannes Reinecke "Data transfer overflow.\n");
446a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
447a1a75b35SHannes Reinecke "cur_residue[%d] tot_residue[%d] len[%u]\n",
448cd9ad58dSDavid S. Miller p->cur_residue, p->tot_residue, len);
449cd9ad58dSDavid S. Miller p->cur_residue = 0;
450cd9ad58dSDavid S. Miller p->tot_residue = 0;
451cd9ad58dSDavid S. Miller }
452cd9ad58dSDavid S. Miller if (!p->cur_residue && p->tot_residue) {
453ee5a1dbfSMing Lei p->prv_sg = p->cur_sg;
454ee5a1dbfSMing Lei p->cur_sg = sg_next(p->cur_sg);
455cd9ad58dSDavid S. Miller p->cur_residue = sg_dma_len(p->cur_sg);
456cd9ad58dSDavid S. Miller }
457cd9ad58dSDavid S. Miller }
458cd9ad58dSDavid S. Miller
esp_unmap_dma(struct esp * esp,struct scsi_cmnd * cmd)459cd9ad58dSDavid S. Miller static void esp_unmap_dma(struct esp *esp, struct scsi_cmnd *cmd)
460cd9ad58dSDavid S. Miller {
4613f9295b6SChristoph Hellwig if (!(esp->flags & ESP_FLAG_NO_DMA_MAP))
4623f9295b6SChristoph Hellwig scsi_dma_unmap(cmd);
463cd9ad58dSDavid S. Miller }
464cd9ad58dSDavid S. Miller
esp_save_pointers(struct esp * esp,struct esp_cmd_entry * ent)465cd9ad58dSDavid S. Miller static void esp_save_pointers(struct esp *esp, struct esp_cmd_entry *ent)
466cd9ad58dSDavid S. Miller {
467cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
468cd9ad58dSDavid S. Miller struct esp_cmd_priv *spriv = ESP_CMD_PRIV(cmd);
469cd9ad58dSDavid S. Miller
470cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
471cd9ad58dSDavid S. Miller ent->saved_sense_ptr = ent->sense_ptr;
472cd9ad58dSDavid S. Miller return;
473cd9ad58dSDavid S. Miller }
474cd9ad58dSDavid S. Miller ent->saved_cur_residue = spriv->cur_residue;
475ee5a1dbfSMing Lei ent->saved_prv_sg = spriv->prv_sg;
476cd9ad58dSDavid S. Miller ent->saved_cur_sg = spriv->cur_sg;
477cd9ad58dSDavid S. Miller ent->saved_tot_residue = spriv->tot_residue;
478cd9ad58dSDavid S. Miller }
479cd9ad58dSDavid S. Miller
esp_restore_pointers(struct esp * esp,struct esp_cmd_entry * ent)480cd9ad58dSDavid S. Miller static void esp_restore_pointers(struct esp *esp, struct esp_cmd_entry *ent)
481cd9ad58dSDavid S. Miller {
482cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
483cd9ad58dSDavid S. Miller struct esp_cmd_priv *spriv = ESP_CMD_PRIV(cmd);
484cd9ad58dSDavid S. Miller
485cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
486cd9ad58dSDavid S. Miller ent->sense_ptr = ent->saved_sense_ptr;
487cd9ad58dSDavid S. Miller return;
488cd9ad58dSDavid S. Miller }
489cd9ad58dSDavid S. Miller spriv->cur_residue = ent->saved_cur_residue;
490ee5a1dbfSMing Lei spriv->prv_sg = ent->saved_prv_sg;
491cd9ad58dSDavid S. Miller spriv->cur_sg = ent->saved_cur_sg;
492cd9ad58dSDavid S. Miller spriv->tot_residue = ent->saved_tot_residue;
493cd9ad58dSDavid S. Miller }
494cd9ad58dSDavid S. Miller
esp_write_tgt_config3(struct esp * esp,int tgt)495cd9ad58dSDavid S. Miller static void esp_write_tgt_config3(struct esp *esp, int tgt)
496cd9ad58dSDavid S. Miller {
497cd9ad58dSDavid S. Miller if (esp->rev > ESP100A) {
498cd9ad58dSDavid S. Miller u8 val = esp->target[tgt].esp_config3;
499cd9ad58dSDavid S. Miller
500cd9ad58dSDavid S. Miller if (val != esp->prev_cfg3) {
501cd9ad58dSDavid S. Miller esp->prev_cfg3 = val;
502cd9ad58dSDavid S. Miller esp_write8(val, ESP_CFG3);
503cd9ad58dSDavid S. Miller }
504cd9ad58dSDavid S. Miller }
505cd9ad58dSDavid S. Miller }
506cd9ad58dSDavid S. Miller
esp_write_tgt_sync(struct esp * esp,int tgt)507cd9ad58dSDavid S. Miller static void esp_write_tgt_sync(struct esp *esp, int tgt)
508cd9ad58dSDavid S. Miller {
509cd9ad58dSDavid S. Miller u8 off = esp->target[tgt].esp_offset;
510cd9ad58dSDavid S. Miller u8 per = esp->target[tgt].esp_period;
511cd9ad58dSDavid S. Miller
512cd9ad58dSDavid S. Miller if (off != esp->prev_soff) {
513cd9ad58dSDavid S. Miller esp->prev_soff = off;
514cd9ad58dSDavid S. Miller esp_write8(off, ESP_SOFF);
515cd9ad58dSDavid S. Miller }
516cd9ad58dSDavid S. Miller if (per != esp->prev_stp) {
517cd9ad58dSDavid S. Miller esp->prev_stp = per;
518cd9ad58dSDavid S. Miller esp_write8(per, ESP_STP);
519cd9ad58dSDavid S. Miller }
520cd9ad58dSDavid S. Miller }
521cd9ad58dSDavid S. Miller
esp_dma_length_limit(struct esp * esp,u32 dma_addr,u32 dma_len)522cd9ad58dSDavid S. Miller static u32 esp_dma_length_limit(struct esp *esp, u32 dma_addr, u32 dma_len)
523cd9ad58dSDavid S. Miller {
524cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
525cd9ad58dSDavid S. Miller /* Arbitrary segment boundaries, 24-bit counts. */
526cd9ad58dSDavid S. Miller if (dma_len > (1U << 24))
527cd9ad58dSDavid S. Miller dma_len = (1U << 24);
528cd9ad58dSDavid S. Miller } else {
529cd9ad58dSDavid S. Miller u32 base, end;
530cd9ad58dSDavid S. Miller
531cd9ad58dSDavid S. Miller /* ESP chip limits other variants by 16-bits of transfer
532cd9ad58dSDavid S. Miller * count. Actually on FAS100A and FAS236 we could get
533cd9ad58dSDavid S. Miller * 24-bits of transfer count by enabling ESP_CONFIG2_FENAB
534cd9ad58dSDavid S. Miller * in the ESP_CFG2 register but that causes other unwanted
535cd9ad58dSDavid S. Miller * changes so we don't use it currently.
536cd9ad58dSDavid S. Miller */
537cd9ad58dSDavid S. Miller if (dma_len > (1U << 16))
538cd9ad58dSDavid S. Miller dma_len = (1U << 16);
539cd9ad58dSDavid S. Miller
540cd9ad58dSDavid S. Miller /* All of the DMA variants hooked up to these chips
541cd9ad58dSDavid S. Miller * cannot handle crossing a 24-bit address boundary.
542cd9ad58dSDavid S. Miller */
543cd9ad58dSDavid S. Miller base = dma_addr & ((1U << 24) - 1U);
544cd9ad58dSDavid S. Miller end = base + dma_len;
545cd9ad58dSDavid S. Miller if (end > (1U << 24))
546cd9ad58dSDavid S. Miller end = (1U <<24);
547cd9ad58dSDavid S. Miller dma_len = end - base;
548cd9ad58dSDavid S. Miller }
549cd9ad58dSDavid S. Miller return dma_len;
550cd9ad58dSDavid S. Miller }
551cd9ad58dSDavid S. Miller
esp_need_to_nego_wide(struct esp_target_data * tp)552cd9ad58dSDavid S. Miller static int esp_need_to_nego_wide(struct esp_target_data *tp)
553cd9ad58dSDavid S. Miller {
554cd9ad58dSDavid S. Miller struct scsi_target *target = tp->starget;
555cd9ad58dSDavid S. Miller
556cd9ad58dSDavid S. Miller return spi_width(target) != tp->nego_goal_width;
557cd9ad58dSDavid S. Miller }
558cd9ad58dSDavid S. Miller
esp_need_to_nego_sync(struct esp_target_data * tp)559cd9ad58dSDavid S. Miller static int esp_need_to_nego_sync(struct esp_target_data *tp)
560cd9ad58dSDavid S. Miller {
561cd9ad58dSDavid S. Miller struct scsi_target *target = tp->starget;
562cd9ad58dSDavid S. Miller
563cd9ad58dSDavid S. Miller /* When offset is zero, period is "don't care". */
564cd9ad58dSDavid S. Miller if (!spi_offset(target) && !tp->nego_goal_offset)
565cd9ad58dSDavid S. Miller return 0;
566cd9ad58dSDavid S. Miller
567cd9ad58dSDavid S. Miller if (spi_offset(target) == tp->nego_goal_offset &&
568cd9ad58dSDavid S. Miller spi_period(target) == tp->nego_goal_period)
569cd9ad58dSDavid S. Miller return 0;
570cd9ad58dSDavid S. Miller
571cd9ad58dSDavid S. Miller return 1;
572cd9ad58dSDavid S. Miller }
573cd9ad58dSDavid S. Miller
esp_alloc_lun_tag(struct esp_cmd_entry * ent,struct esp_lun_data * lp)574cd9ad58dSDavid S. Miller static int esp_alloc_lun_tag(struct esp_cmd_entry *ent,
575cd9ad58dSDavid S. Miller struct esp_lun_data *lp)
576cd9ad58dSDavid S. Miller {
57721af8107SDavid S. Miller if (!ent->orig_tag[0]) {
578cd9ad58dSDavid S. Miller /* Non-tagged, slot already taken? */
579cd9ad58dSDavid S. Miller if (lp->non_tagged_cmd)
580cd9ad58dSDavid S. Miller return -EBUSY;
581cd9ad58dSDavid S. Miller
582cd9ad58dSDavid S. Miller if (lp->hold) {
583cd9ad58dSDavid S. Miller /* We are being held by active tagged
584cd9ad58dSDavid S. Miller * commands.
585cd9ad58dSDavid S. Miller */
586cd9ad58dSDavid S. Miller if (lp->num_tagged)
587cd9ad58dSDavid S. Miller return -EBUSY;
588cd9ad58dSDavid S. Miller
589cd9ad58dSDavid S. Miller /* Tagged commands completed, we can unplug
590cd9ad58dSDavid S. Miller * the queue and run this untagged command.
591cd9ad58dSDavid S. Miller */
592cd9ad58dSDavid S. Miller lp->hold = 0;
593cd9ad58dSDavid S. Miller } else if (lp->num_tagged) {
594cd9ad58dSDavid S. Miller /* Plug the queue until num_tagged decreases
595cd9ad58dSDavid S. Miller * to zero in esp_free_lun_tag.
596cd9ad58dSDavid S. Miller */
597cd9ad58dSDavid S. Miller lp->hold = 1;
598cd9ad58dSDavid S. Miller return -EBUSY;
599cd9ad58dSDavid S. Miller }
600cd9ad58dSDavid S. Miller
601cd9ad58dSDavid S. Miller lp->non_tagged_cmd = ent;
602cd9ad58dSDavid S. Miller return 0;
603201c37d7SFinn Thain }
604201c37d7SFinn Thain
605201c37d7SFinn Thain /* Tagged command. Check that it isn't blocked by a non-tagged one. */
606cd9ad58dSDavid S. Miller if (lp->non_tagged_cmd || lp->hold)
607cd9ad58dSDavid S. Miller return -EBUSY;
608cd9ad58dSDavid S. Miller
60921af8107SDavid S. Miller BUG_ON(lp->tagged_cmds[ent->orig_tag[1]]);
610cd9ad58dSDavid S. Miller
61121af8107SDavid S. Miller lp->tagged_cmds[ent->orig_tag[1]] = ent;
612cd9ad58dSDavid S. Miller lp->num_tagged++;
613cd9ad58dSDavid S. Miller
614cd9ad58dSDavid S. Miller return 0;
615cd9ad58dSDavid S. Miller }
616cd9ad58dSDavid S. Miller
esp_free_lun_tag(struct esp_cmd_entry * ent,struct esp_lun_data * lp)617cd9ad58dSDavid S. Miller static void esp_free_lun_tag(struct esp_cmd_entry *ent,
618cd9ad58dSDavid S. Miller struct esp_lun_data *lp)
619cd9ad58dSDavid S. Miller {
62021af8107SDavid S. Miller if (ent->orig_tag[0]) {
62121af8107SDavid S. Miller BUG_ON(lp->tagged_cmds[ent->orig_tag[1]] != ent);
62221af8107SDavid S. Miller lp->tagged_cmds[ent->orig_tag[1]] = NULL;
623cd9ad58dSDavid S. Miller lp->num_tagged--;
624cd9ad58dSDavid S. Miller } else {
625cd9ad58dSDavid S. Miller BUG_ON(lp->non_tagged_cmd != ent);
626cd9ad58dSDavid S. Miller lp->non_tagged_cmd = NULL;
627cd9ad58dSDavid S. Miller }
628cd9ad58dSDavid S. Miller }
629cd9ad58dSDavid S. Miller
esp_map_sense(struct esp * esp,struct esp_cmd_entry * ent)6303f9295b6SChristoph Hellwig static void esp_map_sense(struct esp *esp, struct esp_cmd_entry *ent)
6313f9295b6SChristoph Hellwig {
6323f9295b6SChristoph Hellwig ent->sense_ptr = ent->cmd->sense_buffer;
6333f9295b6SChristoph Hellwig if (esp->flags & ESP_FLAG_NO_DMA_MAP) {
6343f9295b6SChristoph Hellwig ent->sense_dma = (uintptr_t)ent->sense_ptr;
6353f9295b6SChristoph Hellwig return;
6363f9295b6SChristoph Hellwig }
6373f9295b6SChristoph Hellwig
6383f9295b6SChristoph Hellwig ent->sense_dma = dma_map_single(esp->dev, ent->sense_ptr,
6393f9295b6SChristoph Hellwig SCSI_SENSE_BUFFERSIZE, DMA_FROM_DEVICE);
6403f9295b6SChristoph Hellwig }
6413f9295b6SChristoph Hellwig
esp_unmap_sense(struct esp * esp,struct esp_cmd_entry * ent)6423f9295b6SChristoph Hellwig static void esp_unmap_sense(struct esp *esp, struct esp_cmd_entry *ent)
6433f9295b6SChristoph Hellwig {
6443f9295b6SChristoph Hellwig if (!(esp->flags & ESP_FLAG_NO_DMA_MAP))
6453f9295b6SChristoph Hellwig dma_unmap_single(esp->dev, ent->sense_dma,
6463f9295b6SChristoph Hellwig SCSI_SENSE_BUFFERSIZE, DMA_FROM_DEVICE);
6473f9295b6SChristoph Hellwig ent->sense_ptr = NULL;
6483f9295b6SChristoph Hellwig }
6493f9295b6SChristoph Hellwig
650835b8c16SBhaskar Chowdhury /* When a contingent allegiance condition is created, we force feed a
651cd9ad58dSDavid S. Miller * REQUEST_SENSE command to the device to fetch the sense data. I
652cd9ad58dSDavid S. Miller * tried many other schemes, relying on the scsi error handling layer
653cd9ad58dSDavid S. Miller * to send out the REQUEST_SENSE automatically, but this was difficult
654cd9ad58dSDavid S. Miller * to get right especially in the presence of applications like smartd
655cd9ad58dSDavid S. Miller * which use SG_IO to send out their own REQUEST_SENSE commands.
656cd9ad58dSDavid S. Miller */
esp_autosense(struct esp * esp,struct esp_cmd_entry * ent)657cd9ad58dSDavid S. Miller static void esp_autosense(struct esp *esp, struct esp_cmd_entry *ent)
658cd9ad58dSDavid S. Miller {
659cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
660cd9ad58dSDavid S. Miller struct scsi_device *dev = cmd->device;
661cd9ad58dSDavid S. Miller int tgt, lun;
662cd9ad58dSDavid S. Miller u8 *p, val;
663cd9ad58dSDavid S. Miller
664cd9ad58dSDavid S. Miller tgt = dev->id;
665cd9ad58dSDavid S. Miller lun = dev->lun;
666cd9ad58dSDavid S. Miller
667cd9ad58dSDavid S. Miller
668cd9ad58dSDavid S. Miller if (!ent->sense_ptr) {
669a1a75b35SHannes Reinecke esp_log_autosense("Doing auto-sense for tgt[%d] lun[%d]\n",
670a1a75b35SHannes Reinecke tgt, lun);
6713f9295b6SChristoph Hellwig esp_map_sense(esp, ent);
672cd9ad58dSDavid S. Miller }
673cd9ad58dSDavid S. Miller ent->saved_sense_ptr = ent->sense_ptr;
674cd9ad58dSDavid S. Miller
675cd9ad58dSDavid S. Miller esp->active_cmd = ent;
676cd9ad58dSDavid S. Miller
677cd9ad58dSDavid S. Miller p = esp->command_block;
678cd9ad58dSDavid S. Miller esp->msg_out_len = 0;
679cd9ad58dSDavid S. Miller
680cd9ad58dSDavid S. Miller *p++ = IDENTIFY(0, lun);
681cd9ad58dSDavid S. Miller *p++ = REQUEST_SENSE;
682cd9ad58dSDavid S. Miller *p++ = ((dev->scsi_level <= SCSI_2) ?
683cd9ad58dSDavid S. Miller (lun << 5) : 0);
684cd9ad58dSDavid S. Miller *p++ = 0;
685cd9ad58dSDavid S. Miller *p++ = 0;
686cd9ad58dSDavid S. Miller *p++ = SCSI_SENSE_BUFFERSIZE;
687cd9ad58dSDavid S. Miller *p++ = 0;
688cd9ad58dSDavid S. Miller
689cd9ad58dSDavid S. Miller esp->select_state = ESP_SELECT_BASIC;
690cd9ad58dSDavid S. Miller
691cd9ad58dSDavid S. Miller val = tgt;
692cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
693cd9ad58dSDavid S. Miller val |= ESP_BUSID_RESELID | ESP_BUSID_CTR32BIT;
694cd9ad58dSDavid S. Miller esp_write8(val, ESP_BUSID);
695cd9ad58dSDavid S. Miller
696cd9ad58dSDavid S. Miller esp_write_tgt_sync(esp, tgt);
697cd9ad58dSDavid S. Miller esp_write_tgt_config3(esp, tgt);
698cd9ad58dSDavid S. Miller
699cd9ad58dSDavid S. Miller val = (p - esp->command_block);
700cd9ad58dSDavid S. Miller
7013170866fSHannes Reinecke esp_send_dma_cmd(esp, val, 16, ESP_CMD_SELA);
702cd9ad58dSDavid S. Miller }
703cd9ad58dSDavid S. Miller
find_and_prep_issuable_command(struct esp * esp)704cd9ad58dSDavid S. Miller static struct esp_cmd_entry *find_and_prep_issuable_command(struct esp *esp)
705cd9ad58dSDavid S. Miller {
706cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
707cd9ad58dSDavid S. Miller
708cd9ad58dSDavid S. Miller list_for_each_entry(ent, &esp->queued_cmds, list) {
709cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
710cd9ad58dSDavid S. Miller struct scsi_device *dev = cmd->device;
711cd9ad58dSDavid S. Miller struct esp_lun_data *lp = dev->hostdata;
712cd9ad58dSDavid S. Miller
713cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
714cd9ad58dSDavid S. Miller ent->tag[0] = 0;
715cd9ad58dSDavid S. Miller ent->tag[1] = 0;
716cd9ad58dSDavid S. Miller return ent;
717cd9ad58dSDavid S. Miller }
718cd9ad58dSDavid S. Miller
71950668633SChristoph Hellwig if (!spi_populate_tag_msg(&ent->tag[0], cmd)) {
720cd9ad58dSDavid S. Miller ent->tag[0] = 0;
721cd9ad58dSDavid S. Miller ent->tag[1] = 0;
722cd9ad58dSDavid S. Miller }
72321af8107SDavid S. Miller ent->orig_tag[0] = ent->tag[0];
72421af8107SDavid S. Miller ent->orig_tag[1] = ent->tag[1];
725cd9ad58dSDavid S. Miller
726cd9ad58dSDavid S. Miller if (esp_alloc_lun_tag(ent, lp) < 0)
727cd9ad58dSDavid S. Miller continue;
728cd9ad58dSDavid S. Miller
729cd9ad58dSDavid S. Miller return ent;
730cd9ad58dSDavid S. Miller }
731cd9ad58dSDavid S. Miller
732cd9ad58dSDavid S. Miller return NULL;
733cd9ad58dSDavid S. Miller }
734cd9ad58dSDavid S. Miller
esp_maybe_execute_command(struct esp * esp)735cd9ad58dSDavid S. Miller static void esp_maybe_execute_command(struct esp *esp)
736cd9ad58dSDavid S. Miller {
737cd9ad58dSDavid S. Miller struct esp_target_data *tp;
738cd9ad58dSDavid S. Miller struct scsi_device *dev;
739cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd;
740cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
7418bca2143SFinn Thain bool select_and_stop = false;
742cd9ad58dSDavid S. Miller int tgt, lun, i;
743cd9ad58dSDavid S. Miller u32 val, start_cmd;
744cd9ad58dSDavid S. Miller u8 *p;
745cd9ad58dSDavid S. Miller
746cd9ad58dSDavid S. Miller if (esp->active_cmd ||
747cd9ad58dSDavid S. Miller (esp->flags & ESP_FLAG_RESETTING))
748cd9ad58dSDavid S. Miller return;
749cd9ad58dSDavid S. Miller
750cd9ad58dSDavid S. Miller ent = find_and_prep_issuable_command(esp);
751cd9ad58dSDavid S. Miller if (!ent)
752cd9ad58dSDavid S. Miller return;
753cd9ad58dSDavid S. Miller
754cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
755cd9ad58dSDavid S. Miller esp_autosense(esp, ent);
756cd9ad58dSDavid S. Miller return;
757cd9ad58dSDavid S. Miller }
758cd9ad58dSDavid S. Miller
759cd9ad58dSDavid S. Miller cmd = ent->cmd;
760cd9ad58dSDavid S. Miller dev = cmd->device;
761cd9ad58dSDavid S. Miller tgt = dev->id;
762cd9ad58dSDavid S. Miller lun = dev->lun;
763cd9ad58dSDavid S. Miller tp = &esp->target[tgt];
764cd9ad58dSDavid S. Miller
76563ce2499SKirill A. Shutemov list_move(&ent->list, &esp->active_cmds);
766cd9ad58dSDavid S. Miller
767cd9ad58dSDavid S. Miller esp->active_cmd = ent;
768cd9ad58dSDavid S. Miller
769cd9ad58dSDavid S. Miller esp_map_dma(esp, cmd);
770cd9ad58dSDavid S. Miller esp_save_pointers(esp, ent);
771cd9ad58dSDavid S. Miller
7728bca2143SFinn Thain if (!(cmd->cmd_len == 6 || cmd->cmd_len == 10 || cmd->cmd_len == 12))
7738bca2143SFinn Thain select_and_stop = true;
774cd9ad58dSDavid S. Miller
775cd9ad58dSDavid S. Miller p = esp->command_block;
776cd9ad58dSDavid S. Miller
777cd9ad58dSDavid S. Miller esp->msg_out_len = 0;
778cd9ad58dSDavid S. Miller if (tp->flags & ESP_TGT_CHECK_NEGO) {
779cd9ad58dSDavid S. Miller /* Need to negotiate. If the target is broken
780cd9ad58dSDavid S. Miller * go for synchronous transfers and non-wide.
781cd9ad58dSDavid S. Miller */
782cd9ad58dSDavid S. Miller if (tp->flags & ESP_TGT_BROKEN) {
783cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_DISCONNECT;
784cd9ad58dSDavid S. Miller tp->nego_goal_period = 0;
785cd9ad58dSDavid S. Miller tp->nego_goal_offset = 0;
786cd9ad58dSDavid S. Miller tp->nego_goal_width = 0;
787cd9ad58dSDavid S. Miller tp->nego_goal_tags = 0;
788cd9ad58dSDavid S. Miller }
789cd9ad58dSDavid S. Miller
790cd9ad58dSDavid S. Miller /* If the settings are not changing, skip this. */
791cd9ad58dSDavid S. Miller if (spi_width(tp->starget) == tp->nego_goal_width &&
792cd9ad58dSDavid S. Miller spi_period(tp->starget) == tp->nego_goal_period &&
793cd9ad58dSDavid S. Miller spi_offset(tp->starget) == tp->nego_goal_offset) {
794cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_CHECK_NEGO;
795cd9ad58dSDavid S. Miller goto build_identify;
796cd9ad58dSDavid S. Miller }
797cd9ad58dSDavid S. Miller
798cd9ad58dSDavid S. Miller if (esp->rev == FASHME && esp_need_to_nego_wide(tp)) {
799cd9ad58dSDavid S. Miller esp->msg_out_len =
800cd9ad58dSDavid S. Miller spi_populate_width_msg(&esp->msg_out[0],
801cd9ad58dSDavid S. Miller (tp->nego_goal_width ?
802cd9ad58dSDavid S. Miller 1 : 0));
803cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_NEGO_WIDE;
804cd9ad58dSDavid S. Miller } else if (esp_need_to_nego_sync(tp)) {
805cd9ad58dSDavid S. Miller esp->msg_out_len =
806cd9ad58dSDavid S. Miller spi_populate_sync_msg(&esp->msg_out[0],
807cd9ad58dSDavid S. Miller tp->nego_goal_period,
808cd9ad58dSDavid S. Miller tp->nego_goal_offset);
809cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_NEGO_SYNC;
810cd9ad58dSDavid S. Miller } else {
811cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_CHECK_NEGO;
812cd9ad58dSDavid S. Miller }
813cd9ad58dSDavid S. Miller
8148bca2143SFinn Thain /* If there are multiple message bytes, use Select and Stop */
8158bca2143SFinn Thain if (esp->msg_out_len)
8168bca2143SFinn Thain select_and_stop = true;
817cd9ad58dSDavid S. Miller }
818cd9ad58dSDavid S. Miller
819cd9ad58dSDavid S. Miller build_identify:
82087c58ef5SFinn Thain *p++ = IDENTIFY(tp->flags & ESP_TGT_DISCONNECT, lun);
821cd9ad58dSDavid S. Miller
822cd9ad58dSDavid S. Miller if (ent->tag[0] && esp->rev == ESP100) {
823cd9ad58dSDavid S. Miller /* ESP100 lacks select w/atn3 command, use select
824cd9ad58dSDavid S. Miller * and stop instead.
825cd9ad58dSDavid S. Miller */
8268bca2143SFinn Thain select_and_stop = true;
827cd9ad58dSDavid S. Miller }
828cd9ad58dSDavid S. Miller
8298bca2143SFinn Thain if (select_and_stop) {
830cd9ad58dSDavid S. Miller esp->cmd_bytes_left = cmd->cmd_len;
831cd9ad58dSDavid S. Miller esp->cmd_bytes_ptr = &cmd->cmnd[0];
832cd9ad58dSDavid S. Miller
833cd9ad58dSDavid S. Miller if (ent->tag[0]) {
834cd9ad58dSDavid S. Miller for (i = esp->msg_out_len - 1;
835cd9ad58dSDavid S. Miller i >= 0; i--)
836cd9ad58dSDavid S. Miller esp->msg_out[i + 2] = esp->msg_out[i];
837cd9ad58dSDavid S. Miller esp->msg_out[0] = ent->tag[0];
838cd9ad58dSDavid S. Miller esp->msg_out[1] = ent->tag[1];
839cd9ad58dSDavid S. Miller esp->msg_out_len += 2;
840cd9ad58dSDavid S. Miller }
841cd9ad58dSDavid S. Miller
8423170866fSHannes Reinecke start_cmd = ESP_CMD_SELAS;
843cd9ad58dSDavid S. Miller esp->select_state = ESP_SELECT_MSGOUT;
8448bca2143SFinn Thain } else {
8458bca2143SFinn Thain start_cmd = ESP_CMD_SELA;
8468bca2143SFinn Thain if (ent->tag[0]) {
8478bca2143SFinn Thain *p++ = ent->tag[0];
8488bca2143SFinn Thain *p++ = ent->tag[1];
8498bca2143SFinn Thain
8508bca2143SFinn Thain start_cmd = ESP_CMD_SA3;
8518bca2143SFinn Thain }
8528bca2143SFinn Thain
8538bca2143SFinn Thain for (i = 0; i < cmd->cmd_len; i++)
8548bca2143SFinn Thain *p++ = cmd->cmnd[i];
8558bca2143SFinn Thain
8568bca2143SFinn Thain esp->select_state = ESP_SELECT_BASIC;
857cd9ad58dSDavid S. Miller }
858cd9ad58dSDavid S. Miller val = tgt;
859cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
860cd9ad58dSDavid S. Miller val |= ESP_BUSID_RESELID | ESP_BUSID_CTR32BIT;
861cd9ad58dSDavid S. Miller esp_write8(val, ESP_BUSID);
862cd9ad58dSDavid S. Miller
863cd9ad58dSDavid S. Miller esp_write_tgt_sync(esp, tgt);
864cd9ad58dSDavid S. Miller esp_write_tgt_config3(esp, tgt);
865cd9ad58dSDavid S. Miller
866cd9ad58dSDavid S. Miller val = (p - esp->command_block);
867cd9ad58dSDavid S. Miller
868cd9ad58dSDavid S. Miller if (esp_debug & ESP_DEBUG_SCSICMD) {
869cd9ad58dSDavid S. Miller printk("ESP: tgt[%d] lun[%d] scsi_cmd [ ", tgt, lun);
870cd9ad58dSDavid S. Miller for (i = 0; i < cmd->cmd_len; i++)
871cd9ad58dSDavid S. Miller printk("%02x ", cmd->cmnd[i]);
872cd9ad58dSDavid S. Miller printk("]\n");
873cd9ad58dSDavid S. Miller }
874cd9ad58dSDavid S. Miller
8753170866fSHannes Reinecke esp_send_dma_cmd(esp, val, 16, start_cmd);
876cd9ad58dSDavid S. Miller }
877cd9ad58dSDavid S. Miller
esp_get_ent(struct esp * esp)878cd9ad58dSDavid S. Miller static struct esp_cmd_entry *esp_get_ent(struct esp *esp)
879cd9ad58dSDavid S. Miller {
880cd9ad58dSDavid S. Miller struct list_head *head = &esp->esp_cmd_pool;
881cd9ad58dSDavid S. Miller struct esp_cmd_entry *ret;
882cd9ad58dSDavid S. Miller
883cd9ad58dSDavid S. Miller if (list_empty(head)) {
884cd9ad58dSDavid S. Miller ret = kzalloc(sizeof(struct esp_cmd_entry), GFP_ATOMIC);
885cd9ad58dSDavid S. Miller } else {
886cd9ad58dSDavid S. Miller ret = list_entry(head->next, struct esp_cmd_entry, list);
887cd9ad58dSDavid S. Miller list_del(&ret->list);
888cd9ad58dSDavid S. Miller memset(ret, 0, sizeof(*ret));
889cd9ad58dSDavid S. Miller }
890cd9ad58dSDavid S. Miller return ret;
891cd9ad58dSDavid S. Miller }
892cd9ad58dSDavid S. Miller
esp_put_ent(struct esp * esp,struct esp_cmd_entry * ent)893cd9ad58dSDavid S. Miller static void esp_put_ent(struct esp *esp, struct esp_cmd_entry *ent)
894cd9ad58dSDavid S. Miller {
895cd9ad58dSDavid S. Miller list_add(&ent->list, &esp->esp_cmd_pool);
896cd9ad58dSDavid S. Miller }
897cd9ad58dSDavid S. Miller
esp_cmd_is_done(struct esp * esp,struct esp_cmd_entry * ent,struct scsi_cmnd * cmd,unsigned char host_byte)898cd9ad58dSDavid S. Miller static void esp_cmd_is_done(struct esp *esp, struct esp_cmd_entry *ent,
8996b50529eSHannes Reinecke struct scsi_cmnd *cmd, unsigned char host_byte)
900cd9ad58dSDavid S. Miller {
901cd9ad58dSDavid S. Miller struct scsi_device *dev = cmd->device;
902cd9ad58dSDavid S. Miller int tgt = dev->id;
903cd9ad58dSDavid S. Miller int lun = dev->lun;
904cd9ad58dSDavid S. Miller
905cd9ad58dSDavid S. Miller esp->active_cmd = NULL;
906cd9ad58dSDavid S. Miller esp_unmap_dma(esp, cmd);
907cd9ad58dSDavid S. Miller esp_free_lun_tag(ent, dev->hostdata);
9086b50529eSHannes Reinecke cmd->result = 0;
9096b50529eSHannes Reinecke set_host_byte(cmd, host_byte);
910809dadb1SHannes Reinecke if (host_byte == DID_OK)
9116b50529eSHannes Reinecke set_status_byte(cmd, ent->status);
912cd9ad58dSDavid S. Miller
913cd9ad58dSDavid S. Miller if (ent->eh_done) {
914cd9ad58dSDavid S. Miller complete(ent->eh_done);
915cd9ad58dSDavid S. Miller ent->eh_done = NULL;
916cd9ad58dSDavid S. Miller }
917cd9ad58dSDavid S. Miller
918cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_AUTOSENSE) {
9193f9295b6SChristoph Hellwig esp_unmap_sense(esp, ent);
920cd9ad58dSDavid S. Miller
921cd9ad58dSDavid S. Miller /* Restore the message/status bytes to what we actually
922cd9ad58dSDavid S. Miller * saw originally. Also, report that we are providing
923cd9ad58dSDavid S. Miller * the sense data.
924cd9ad58dSDavid S. Miller */
925464a00c9SHannes Reinecke cmd->result = SAM_STAT_CHECK_CONDITION;
926cd9ad58dSDavid S. Miller
927cd9ad58dSDavid S. Miller ent->flags &= ~ESP_CMD_FLAG_AUTOSENSE;
928cd9ad58dSDavid S. Miller if (esp_debug & ESP_DEBUG_AUTOSENSE) {
929cd9ad58dSDavid S. Miller int i;
930cd9ad58dSDavid S. Miller
931cd9ad58dSDavid S. Miller printk("esp%d: tgt[%d] lun[%d] AUTO SENSE[ ",
932cd9ad58dSDavid S. Miller esp->host->unique_id, tgt, lun);
933cd9ad58dSDavid S. Miller for (i = 0; i < 18; i++)
934cd9ad58dSDavid S. Miller printk("%02x ", cmd->sense_buffer[i]);
935cd9ad58dSDavid S. Miller printk("]\n");
936cd9ad58dSDavid S. Miller }
937cd9ad58dSDavid S. Miller }
938cd9ad58dSDavid S. Miller
939f8ab27d9SBart Van Assche scsi_done(cmd);
940cd9ad58dSDavid S. Miller
941cd9ad58dSDavid S. Miller list_del(&ent->list);
942cd9ad58dSDavid S. Miller esp_put_ent(esp, ent);
943cd9ad58dSDavid S. Miller
944cd9ad58dSDavid S. Miller esp_maybe_execute_command(esp);
945cd9ad58dSDavid S. Miller }
946cd9ad58dSDavid S. Miller
esp_event_queue_full(struct esp * esp,struct esp_cmd_entry * ent)947cd9ad58dSDavid S. Miller static void esp_event_queue_full(struct esp *esp, struct esp_cmd_entry *ent)
948cd9ad58dSDavid S. Miller {
949cd9ad58dSDavid S. Miller struct scsi_device *dev = ent->cmd->device;
950cd9ad58dSDavid S. Miller struct esp_lun_data *lp = dev->hostdata;
951cd9ad58dSDavid S. Miller
952cd9ad58dSDavid S. Miller scsi_track_queue_full(dev, lp->num_tagged - 1);
953cd9ad58dSDavid S. Miller }
954cd9ad58dSDavid S. Miller
esp_queuecommand_lck(struct scsi_cmnd * cmd)955af049dfdSBart Van Assche static int esp_queuecommand_lck(struct scsi_cmnd *cmd)
956cd9ad58dSDavid S. Miller {
957cd9ad58dSDavid S. Miller struct scsi_device *dev = cmd->device;
9582b14ec78SChristoph Hellwig struct esp *esp = shost_priv(dev->host);
959cd9ad58dSDavid S. Miller struct esp_cmd_priv *spriv;
960cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
961cd9ad58dSDavid S. Miller
962cd9ad58dSDavid S. Miller ent = esp_get_ent(esp);
963cd9ad58dSDavid S. Miller if (!ent)
964cd9ad58dSDavid S. Miller return SCSI_MLQUEUE_HOST_BUSY;
965cd9ad58dSDavid S. Miller
966cd9ad58dSDavid S. Miller ent->cmd = cmd;
967cd9ad58dSDavid S. Miller
968cd9ad58dSDavid S. Miller spriv = ESP_CMD_PRIV(cmd);
96986117d7fSChristoph Hellwig spriv->num_sg = 0;
970cd9ad58dSDavid S. Miller
971cd9ad58dSDavid S. Miller list_add_tail(&ent->list, &esp->queued_cmds);
972cd9ad58dSDavid S. Miller
973cd9ad58dSDavid S. Miller esp_maybe_execute_command(esp);
974cd9ad58dSDavid S. Miller
975cd9ad58dSDavid S. Miller return 0;
976cd9ad58dSDavid S. Miller }
977cd9ad58dSDavid S. Miller
DEF_SCSI_QCMD(esp_queuecommand)978f281233dSJeff Garzik static DEF_SCSI_QCMD(esp_queuecommand)
979f281233dSJeff Garzik
980cd9ad58dSDavid S. Miller static int esp_check_gross_error(struct esp *esp)
981cd9ad58dSDavid S. Miller {
982cd9ad58dSDavid S. Miller if (esp->sreg & ESP_STAT_SPAM) {
983cd9ad58dSDavid S. Miller /* Gross Error, could be one of:
984cd9ad58dSDavid S. Miller * - top of fifo overwritten
985cd9ad58dSDavid S. Miller * - top of command register overwritten
986cd9ad58dSDavid S. Miller * - DMA programmed with wrong direction
987cd9ad58dSDavid S. Miller * - improper phase change
988cd9ad58dSDavid S. Miller */
989a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
990a1a75b35SHannes Reinecke "Gross error sreg[%02x]\n", esp->sreg);
991cd9ad58dSDavid S. Miller /* XXX Reset the chip. XXX */
992cd9ad58dSDavid S. Miller return 1;
993cd9ad58dSDavid S. Miller }
994cd9ad58dSDavid S. Miller return 0;
995cd9ad58dSDavid S. Miller }
996cd9ad58dSDavid S. Miller
esp_check_spur_intr(struct esp * esp)997cd9ad58dSDavid S. Miller static int esp_check_spur_intr(struct esp *esp)
998cd9ad58dSDavid S. Miller {
999cd9ad58dSDavid S. Miller switch (esp->rev) {
1000cd9ad58dSDavid S. Miller case ESP100:
1001cd9ad58dSDavid S. Miller case ESP100A:
1002cd9ad58dSDavid S. Miller /* The interrupt pending bit of the status register cannot
1003cd9ad58dSDavid S. Miller * be trusted on these revisions.
1004cd9ad58dSDavid S. Miller */
1005cd9ad58dSDavid S. Miller esp->sreg &= ~ESP_STAT_INTR;
1006cd9ad58dSDavid S. Miller break;
1007cd9ad58dSDavid S. Miller
1008cd9ad58dSDavid S. Miller default:
1009cd9ad58dSDavid S. Miller if (!(esp->sreg & ESP_STAT_INTR)) {
1010cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_SR)
1011cd9ad58dSDavid S. Miller return 1;
1012cd9ad58dSDavid S. Miller
1013cd9ad58dSDavid S. Miller /* If the DMA is indicating interrupt pending and the
1014cd9ad58dSDavid S. Miller * ESP is not, the only possibility is a DMA error.
1015cd9ad58dSDavid S. Miller */
1016cd9ad58dSDavid S. Miller if (!esp->ops->dma_error(esp)) {
1017a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1018a1a75b35SHannes Reinecke "Spurious irq, sreg=%02x.\n",
1019a1a75b35SHannes Reinecke esp->sreg);
1020cd9ad58dSDavid S. Miller return -1;
1021cd9ad58dSDavid S. Miller }
1022cd9ad58dSDavid S. Miller
1023a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host, "DMA error\n");
1024cd9ad58dSDavid S. Miller
1025cd9ad58dSDavid S. Miller /* XXX Reset the chip. XXX */
1026cd9ad58dSDavid S. Miller return -1;
1027cd9ad58dSDavid S. Miller }
1028cd9ad58dSDavid S. Miller break;
1029cd9ad58dSDavid S. Miller }
1030cd9ad58dSDavid S. Miller
1031cd9ad58dSDavid S. Miller return 0;
1032cd9ad58dSDavid S. Miller }
1033cd9ad58dSDavid S. Miller
esp_schedule_reset(struct esp * esp)1034cd9ad58dSDavid S. Miller static void esp_schedule_reset(struct esp *esp)
1035cd9ad58dSDavid S. Miller {
1036d75f773cSSakari Ailus esp_log_reset("esp_schedule_reset() from %ps\n",
1037cd9ad58dSDavid S. Miller __builtin_return_address(0));
1038cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_RESETTING;
1039cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_RESET);
1040cd9ad58dSDavid S. Miller }
1041cd9ad58dSDavid S. Miller
1042cd9ad58dSDavid S. Miller /* In order to avoid having to add a special half-reconnected state
1043cd9ad58dSDavid S. Miller * into the driver we just sit here and poll through the rest of
1044cd9ad58dSDavid S. Miller * the reselection process to get the tag message bytes.
1045cd9ad58dSDavid S. Miller */
esp_reconnect_with_tag(struct esp * esp,struct esp_lun_data * lp)1046cd9ad58dSDavid S. Miller static struct esp_cmd_entry *esp_reconnect_with_tag(struct esp *esp,
1047cd9ad58dSDavid S. Miller struct esp_lun_data *lp)
1048cd9ad58dSDavid S. Miller {
1049cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
1050cd9ad58dSDavid S. Miller int i;
1051cd9ad58dSDavid S. Miller
1052cd9ad58dSDavid S. Miller if (!lp->num_tagged) {
1053a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1054a1a75b35SHannes Reinecke "Reconnect w/num_tagged==0\n");
1055cd9ad58dSDavid S. Miller return NULL;
1056cd9ad58dSDavid S. Miller }
1057cd9ad58dSDavid S. Miller
1058a1a75b35SHannes Reinecke esp_log_reconnect("reconnect tag, ");
1059cd9ad58dSDavid S. Miller
1060cd9ad58dSDavid S. Miller for (i = 0; i < ESP_QUICKIRQ_LIMIT; i++) {
1061cd9ad58dSDavid S. Miller if (esp->ops->irq_pending(esp))
1062cd9ad58dSDavid S. Miller break;
1063cd9ad58dSDavid S. Miller }
1064cd9ad58dSDavid S. Miller if (i == ESP_QUICKIRQ_LIMIT) {
1065a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1066a1a75b35SHannes Reinecke "Reconnect IRQ1 timeout\n");
1067cd9ad58dSDavid S. Miller return NULL;
1068cd9ad58dSDavid S. Miller }
1069cd9ad58dSDavid S. Miller
1070cd9ad58dSDavid S. Miller esp->sreg = esp_read8(ESP_STATUS);
1071cd9ad58dSDavid S. Miller esp->ireg = esp_read8(ESP_INTRPT);
1072cd9ad58dSDavid S. Miller
1073cd9ad58dSDavid S. Miller esp_log_reconnect("IRQ(%d:%x:%x), ",
1074cd9ad58dSDavid S. Miller i, esp->ireg, esp->sreg);
1075cd9ad58dSDavid S. Miller
1076cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_DC) {
1077a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1078a1a75b35SHannes Reinecke "Reconnect, got disconnect.\n");
1079cd9ad58dSDavid S. Miller return NULL;
1080cd9ad58dSDavid S. Miller }
1081cd9ad58dSDavid S. Miller
1082cd9ad58dSDavid S. Miller if ((esp->sreg & ESP_STAT_PMASK) != ESP_MIP) {
1083a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1084a1a75b35SHannes Reinecke "Reconnect, not MIP sreg[%02x].\n", esp->sreg);
1085cd9ad58dSDavid S. Miller return NULL;
1086cd9ad58dSDavid S. Miller }
1087cd9ad58dSDavid S. Miller
1088cd9ad58dSDavid S. Miller /* DMA in the tag bytes... */
1089cd9ad58dSDavid S. Miller esp->command_block[0] = 0xff;
1090cd9ad58dSDavid S. Miller esp->command_block[1] = 0xff;
1091cd9ad58dSDavid S. Miller esp->ops->send_dma_cmd(esp, esp->command_block_dma,
1092cd9ad58dSDavid S. Miller 2, 2, 1, ESP_CMD_DMA | ESP_CMD_TI);
1093cd9ad58dSDavid S. Miller
109470f23fd6SJustin P. Mattock /* ACK the message. */
1095cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_MOK);
1096cd9ad58dSDavid S. Miller
1097cd9ad58dSDavid S. Miller for (i = 0; i < ESP_RESELECT_TAG_LIMIT; i++) {
1098cd9ad58dSDavid S. Miller if (esp->ops->irq_pending(esp)) {
1099cd9ad58dSDavid S. Miller esp->sreg = esp_read8(ESP_STATUS);
1100cd9ad58dSDavid S. Miller esp->ireg = esp_read8(ESP_INTRPT);
1101cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_FDONE)
1102cd9ad58dSDavid S. Miller break;
1103cd9ad58dSDavid S. Miller }
1104cd9ad58dSDavid S. Miller udelay(1);
1105cd9ad58dSDavid S. Miller }
1106cd9ad58dSDavid S. Miller if (i == ESP_RESELECT_TAG_LIMIT) {
1107a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host, "Reconnect IRQ2 timeout\n");
1108cd9ad58dSDavid S. Miller return NULL;
1109cd9ad58dSDavid S. Miller }
1110cd9ad58dSDavid S. Miller esp->ops->dma_drain(esp);
1111cd9ad58dSDavid S. Miller esp->ops->dma_invalidate(esp);
1112cd9ad58dSDavid S. Miller
1113cd9ad58dSDavid S. Miller esp_log_reconnect("IRQ2(%d:%x:%x) tag[%x:%x]\n",
1114cd9ad58dSDavid S. Miller i, esp->ireg, esp->sreg,
1115cd9ad58dSDavid S. Miller esp->command_block[0],
1116cd9ad58dSDavid S. Miller esp->command_block[1]);
1117cd9ad58dSDavid S. Miller
1118cd9ad58dSDavid S. Miller if (esp->command_block[0] < SIMPLE_QUEUE_TAG ||
1119cd9ad58dSDavid S. Miller esp->command_block[0] > ORDERED_QUEUE_TAG) {
1120a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1121a1a75b35SHannes Reinecke "Reconnect, bad tag type %02x.\n",
1122a1a75b35SHannes Reinecke esp->command_block[0]);
1123cd9ad58dSDavid S. Miller return NULL;
1124cd9ad58dSDavid S. Miller }
1125cd9ad58dSDavid S. Miller
1126cd9ad58dSDavid S. Miller ent = lp->tagged_cmds[esp->command_block[1]];
1127cd9ad58dSDavid S. Miller if (!ent) {
1128a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1129a1a75b35SHannes Reinecke "Reconnect, no entry for tag %02x.\n",
1130a1a75b35SHannes Reinecke esp->command_block[1]);
1131cd9ad58dSDavid S. Miller return NULL;
1132cd9ad58dSDavid S. Miller }
1133cd9ad58dSDavid S. Miller
1134cd9ad58dSDavid S. Miller return ent;
1135cd9ad58dSDavid S. Miller }
1136cd9ad58dSDavid S. Miller
esp_reconnect(struct esp * esp)1137cd9ad58dSDavid S. Miller static int esp_reconnect(struct esp *esp)
1138cd9ad58dSDavid S. Miller {
1139cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
1140cd9ad58dSDavid S. Miller struct esp_target_data *tp;
1141cd9ad58dSDavid S. Miller struct esp_lun_data *lp;
1142cd9ad58dSDavid S. Miller struct scsi_device *dev;
1143cd9ad58dSDavid S. Miller int target, lun;
1144cd9ad58dSDavid S. Miller
1145cd9ad58dSDavid S. Miller BUG_ON(esp->active_cmd);
1146cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
1147cd9ad58dSDavid S. Miller /* FASHME puts the target and lun numbers directly
1148cd9ad58dSDavid S. Miller * into the fifo.
1149cd9ad58dSDavid S. Miller */
1150cd9ad58dSDavid S. Miller target = esp->fifo[0];
1151cd9ad58dSDavid S. Miller lun = esp->fifo[1] & 0x7;
1152cd9ad58dSDavid S. Miller } else {
1153cd9ad58dSDavid S. Miller u8 bits = esp_read8(ESP_FDATA);
1154cd9ad58dSDavid S. Miller
1155cd9ad58dSDavid S. Miller /* Older chips put the lun directly into the fifo, but
1156cd9ad58dSDavid S. Miller * the target is given as a sample of the arbitration
1157cd9ad58dSDavid S. Miller * lines on the bus at reselection time. So we should
1158cd9ad58dSDavid S. Miller * see the ID of the ESP and the one reconnecting target
1159cd9ad58dSDavid S. Miller * set in the bitmap.
1160cd9ad58dSDavid S. Miller */
1161cd9ad58dSDavid S. Miller if (!(bits & esp->scsi_id_mask))
1162cd9ad58dSDavid S. Miller goto do_reset;
1163cd9ad58dSDavid S. Miller bits &= ~esp->scsi_id_mask;
1164cd9ad58dSDavid S. Miller if (!bits || (bits & (bits - 1)))
1165cd9ad58dSDavid S. Miller goto do_reset;
1166cd9ad58dSDavid S. Miller
1167cd9ad58dSDavid S. Miller target = ffs(bits) - 1;
1168cd9ad58dSDavid S. Miller lun = (esp_read8(ESP_FDATA) & 0x7);
1169cd9ad58dSDavid S. Miller
1170cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1171cd9ad58dSDavid S. Miller if (esp->rev == ESP100) {
1172cd9ad58dSDavid S. Miller u8 ireg = esp_read8(ESP_INTRPT);
1173cd9ad58dSDavid S. Miller /* This chip has a bug during reselection that can
1174cd9ad58dSDavid S. Miller * cause a spurious illegal-command interrupt, which
1175cd9ad58dSDavid S. Miller * we simply ACK here. Another possibility is a bus
1176cd9ad58dSDavid S. Miller * reset so we must check for that.
1177cd9ad58dSDavid S. Miller */
1178cd9ad58dSDavid S. Miller if (ireg & ESP_INTR_SR)
1179cd9ad58dSDavid S. Miller goto do_reset;
1180cd9ad58dSDavid S. Miller }
1181cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL);
1182cd9ad58dSDavid S. Miller }
1183cd9ad58dSDavid S. Miller
1184cd9ad58dSDavid S. Miller esp_write_tgt_sync(esp, target);
1185cd9ad58dSDavid S. Miller esp_write_tgt_config3(esp, target);
1186cd9ad58dSDavid S. Miller
1187cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_MOK);
1188cd9ad58dSDavid S. Miller
1189cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
1190cd9ad58dSDavid S. Miller esp_write8(target | ESP_BUSID_RESELID | ESP_BUSID_CTR32BIT,
1191cd9ad58dSDavid S. Miller ESP_BUSID);
1192cd9ad58dSDavid S. Miller
1193cd9ad58dSDavid S. Miller tp = &esp->target[target];
1194cd9ad58dSDavid S. Miller dev = __scsi_device_lookup_by_target(tp->starget, lun);
1195cd9ad58dSDavid S. Miller if (!dev) {
1196a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1197a1a75b35SHannes Reinecke "Reconnect, no lp tgt[%u] lun[%u]\n",
1198a1a75b35SHannes Reinecke target, lun);
1199cd9ad58dSDavid S. Miller goto do_reset;
1200cd9ad58dSDavid S. Miller }
1201cd9ad58dSDavid S. Miller lp = dev->hostdata;
1202cd9ad58dSDavid S. Miller
1203cd9ad58dSDavid S. Miller ent = lp->non_tagged_cmd;
1204cd9ad58dSDavid S. Miller if (!ent) {
1205cd9ad58dSDavid S. Miller ent = esp_reconnect_with_tag(esp, lp);
1206cd9ad58dSDavid S. Miller if (!ent)
1207cd9ad58dSDavid S. Miller goto do_reset;
1208cd9ad58dSDavid S. Miller }
1209cd9ad58dSDavid S. Miller
1210cd9ad58dSDavid S. Miller esp->active_cmd = ent;
1211cd9ad58dSDavid S. Miller
1212cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
1213cd9ad58dSDavid S. Miller esp_restore_pointers(esp, ent);
1214cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1215cd9ad58dSDavid S. Miller return 1;
1216cd9ad58dSDavid S. Miller
1217cd9ad58dSDavid S. Miller do_reset:
1218cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1219cd9ad58dSDavid S. Miller return 0;
1220cd9ad58dSDavid S. Miller }
1221cd9ad58dSDavid S. Miller
esp_finish_select(struct esp * esp)1222cd9ad58dSDavid S. Miller static int esp_finish_select(struct esp *esp)
1223cd9ad58dSDavid S. Miller {
1224cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
1225cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd;
1226cd9ad58dSDavid S. Miller
1227cd9ad58dSDavid S. Miller /* No longer selecting. */
1228cd9ad58dSDavid S. Miller esp->select_state = ESP_SELECT_NONE;
1229cd9ad58dSDavid S. Miller
1230cd9ad58dSDavid S. Miller esp->seqreg = esp_read8(ESP_SSTEP) & ESP_STEP_VBITS;
1231cd9ad58dSDavid S. Miller ent = esp->active_cmd;
1232cd9ad58dSDavid S. Miller cmd = ent->cmd;
1233cd9ad58dSDavid S. Miller
1234cd9ad58dSDavid S. Miller if (esp->ops->dma_error(esp)) {
1235cd9ad58dSDavid S. Miller /* If we see a DMA error during or as a result of selection,
1236cd9ad58dSDavid S. Miller * all bets are off.
1237cd9ad58dSDavid S. Miller */
1238cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
12396b50529eSHannes Reinecke esp_cmd_is_done(esp, ent, cmd, DID_ERROR);
1240cd9ad58dSDavid S. Miller return 0;
1241cd9ad58dSDavid S. Miller }
1242cd9ad58dSDavid S. Miller
1243cd9ad58dSDavid S. Miller esp->ops->dma_invalidate(esp);
1244cd9ad58dSDavid S. Miller
1245cd9ad58dSDavid S. Miller if (esp->ireg == (ESP_INTR_RSEL | ESP_INTR_FDONE)) {
1246cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[cmd->device->id];
1247cd9ad58dSDavid S. Miller
1248cd9ad58dSDavid S. Miller /* Carefully back out of the selection attempt. Release
1249cd9ad58dSDavid S. Miller * resources (such as DMA mapping & TAG) and reset state (such
1250cd9ad58dSDavid S. Miller * as message out and command delivery variables).
1251cd9ad58dSDavid S. Miller */
1252cd9ad58dSDavid S. Miller if (!(ent->flags & ESP_CMD_FLAG_AUTOSENSE)) {
1253cd9ad58dSDavid S. Miller esp_unmap_dma(esp, cmd);
1254cd9ad58dSDavid S. Miller esp_free_lun_tag(ent, cmd->device->hostdata);
1255cd9ad58dSDavid S. Miller tp->flags &= ~(ESP_TGT_NEGO_SYNC | ESP_TGT_NEGO_WIDE);
1256cd9ad58dSDavid S. Miller esp->cmd_bytes_ptr = NULL;
1257cd9ad58dSDavid S. Miller esp->cmd_bytes_left = 0;
1258cd9ad58dSDavid S. Miller } else {
12593f9295b6SChristoph Hellwig esp_unmap_sense(esp, ent);
1260cd9ad58dSDavid S. Miller }
1261cd9ad58dSDavid S. Miller
1262cd9ad58dSDavid S. Miller /* Now that the state is unwound properly, put back onto
1263cd9ad58dSDavid S. Miller * the issue queue. This command is no longer active.
1264cd9ad58dSDavid S. Miller */
126563ce2499SKirill A. Shutemov list_move(&ent->list, &esp->queued_cmds);
1266cd9ad58dSDavid S. Miller esp->active_cmd = NULL;
1267cd9ad58dSDavid S. Miller
1268cd9ad58dSDavid S. Miller /* Return value ignored by caller, it directly invokes
1269cd9ad58dSDavid S. Miller * esp_reconnect().
1270cd9ad58dSDavid S. Miller */
1271cd9ad58dSDavid S. Miller return 0;
1272cd9ad58dSDavid S. Miller }
1273cd9ad58dSDavid S. Miller
1274cd9ad58dSDavid S. Miller if (esp->ireg == ESP_INTR_DC) {
1275cd9ad58dSDavid S. Miller struct scsi_device *dev = cmd->device;
1276cd9ad58dSDavid S. Miller
1277cd9ad58dSDavid S. Miller /* Disconnect. Make sure we re-negotiate sync and
1278cd9ad58dSDavid S. Miller * wide parameters if this target starts responding
1279cd9ad58dSDavid S. Miller * again in the future.
1280cd9ad58dSDavid S. Miller */
1281cd9ad58dSDavid S. Miller esp->target[dev->id].flags |= ESP_TGT_CHECK_NEGO;
1282cd9ad58dSDavid S. Miller
1283cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_ESEL);
12846b50529eSHannes Reinecke esp_cmd_is_done(esp, ent, cmd, DID_BAD_TARGET);
1285cd9ad58dSDavid S. Miller return 1;
1286cd9ad58dSDavid S. Miller }
1287cd9ad58dSDavid S. Miller
1288cd9ad58dSDavid S. Miller if (esp->ireg == (ESP_INTR_FDONE | ESP_INTR_BSERV)) {
1289cd9ad58dSDavid S. Miller /* Selection successful. On pre-FAST chips we have
1290cd9ad58dSDavid S. Miller * to do a NOP and possibly clean out the FIFO.
1291cd9ad58dSDavid S. Miller */
1292cd9ad58dSDavid S. Miller if (esp->rev <= ESP236) {
1293cd9ad58dSDavid S. Miller int fcnt = esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES;
1294cd9ad58dSDavid S. Miller
1295cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL);
1296cd9ad58dSDavid S. Miller
1297cd9ad58dSDavid S. Miller if (!fcnt &&
1298cd9ad58dSDavid S. Miller (!esp->prev_soff ||
1299cd9ad58dSDavid S. Miller ((esp->sreg & ESP_STAT_PMASK) != ESP_DIP)))
1300cd9ad58dSDavid S. Miller esp_flush_fifo(esp);
1301cd9ad58dSDavid S. Miller }
1302cd9ad58dSDavid S. Miller
13038bca2143SFinn Thain /* If we are doing a Select And Stop command, negotiation, etc.
13048bca2143SFinn Thain * we'll do the right thing as we transition to the next phase.
1305cd9ad58dSDavid S. Miller */
1306cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
1307cd9ad58dSDavid S. Miller return 0;
1308cd9ad58dSDavid S. Miller }
1309cd9ad58dSDavid S. Miller
1310a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1311a1a75b35SHannes Reinecke "Unexpected selection completion ireg[%x]\n", esp->ireg);
1312cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1313cd9ad58dSDavid S. Miller return 0;
1314cd9ad58dSDavid S. Miller }
1315cd9ad58dSDavid S. Miller
esp_data_bytes_sent(struct esp * esp,struct esp_cmd_entry * ent,struct scsi_cmnd * cmd)1316cd9ad58dSDavid S. Miller static int esp_data_bytes_sent(struct esp *esp, struct esp_cmd_entry *ent,
1317cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd)
1318cd9ad58dSDavid S. Miller {
1319cd9ad58dSDavid S. Miller int fifo_cnt, ecount, bytes_sent, flush_fifo;
1320cd9ad58dSDavid S. Miller
1321cd9ad58dSDavid S. Miller fifo_cnt = esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES;
1322cd9ad58dSDavid S. Miller if (esp->prev_cfg3 & ESP_CONFIG3_EWIDE)
1323cd9ad58dSDavid S. Miller fifo_cnt <<= 1;
1324cd9ad58dSDavid S. Miller
1325cd9ad58dSDavid S. Miller ecount = 0;
1326cd9ad58dSDavid S. Miller if (!(esp->sreg & ESP_STAT_TCNT)) {
1327cd9ad58dSDavid S. Miller ecount = ((unsigned int)esp_read8(ESP_TCLOW) |
1328cd9ad58dSDavid S. Miller (((unsigned int)esp_read8(ESP_TCMED)) << 8));
1329cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
1330cd9ad58dSDavid S. Miller ecount |= ((unsigned int)esp_read8(FAS_RLO)) << 16;
1331e858d930SHannes Reinecke if (esp->rev == PCSCSI && (esp->config2 & ESP_CONFIG2_FENAB))
1332e858d930SHannes Reinecke ecount |= ((unsigned int)esp_read8(ESP_TCHI)) << 16;
1333cd9ad58dSDavid S. Miller }
1334cd9ad58dSDavid S. Miller
1335cd9ad58dSDavid S. Miller bytes_sent = esp->data_dma_len;
1336cd9ad58dSDavid S. Miller bytes_sent -= ecount;
1337fd47d919SFinn Thain bytes_sent -= esp->send_cmd_residual;
1338cd9ad58dSDavid S. Miller
13396df388f2SHannes Reinecke /*
1340835b8c16SBhaskar Chowdhury * The am53c974 has a DMA 'peculiarity'. The doc states:
13416df388f2SHannes Reinecke * In some odd byte conditions, one residual byte will
13426df388f2SHannes Reinecke * be left in the SCSI FIFO, and the FIFO Flags will
13436df388f2SHannes Reinecke * never count to '0 '. When this happens, the residual
13446df388f2SHannes Reinecke * byte should be retrieved via PIO following completion
13456df388f2SHannes Reinecke * of the BLAST operation.
13466df388f2SHannes Reinecke */
13476df388f2SHannes Reinecke if (fifo_cnt == 1 && ent->flags & ESP_CMD_FLAG_RESIDUAL) {
13486df388f2SHannes Reinecke size_t count = 1;
13496df388f2SHannes Reinecke size_t offset = bytes_sent;
13506df388f2SHannes Reinecke u8 bval = esp_read8(ESP_FDATA);
13516df388f2SHannes Reinecke
13526df388f2SHannes Reinecke if (ent->flags & ESP_CMD_FLAG_AUTOSENSE)
13536df388f2SHannes Reinecke ent->sense_ptr[bytes_sent] = bval;
13546df388f2SHannes Reinecke else {
13556df388f2SHannes Reinecke struct esp_cmd_priv *p = ESP_CMD_PRIV(cmd);
13566df388f2SHannes Reinecke u8 *ptr;
13576df388f2SHannes Reinecke
135886117d7fSChristoph Hellwig ptr = scsi_kmap_atomic_sg(p->cur_sg, p->num_sg,
13596df388f2SHannes Reinecke &offset, &count);
13606df388f2SHannes Reinecke if (likely(ptr)) {
13616df388f2SHannes Reinecke *(ptr + offset) = bval;
13626df388f2SHannes Reinecke scsi_kunmap_atomic_sg(ptr);
13636df388f2SHannes Reinecke }
13646df388f2SHannes Reinecke }
13656df388f2SHannes Reinecke bytes_sent += fifo_cnt;
13666df388f2SHannes Reinecke ent->flags &= ~ESP_CMD_FLAG_RESIDUAL;
13676df388f2SHannes Reinecke }
1368cd9ad58dSDavid S. Miller if (!(ent->flags & ESP_CMD_FLAG_WRITE))
1369cd9ad58dSDavid S. Miller bytes_sent -= fifo_cnt;
1370cd9ad58dSDavid S. Miller
1371cd9ad58dSDavid S. Miller flush_fifo = 0;
1372cd9ad58dSDavid S. Miller if (!esp->prev_soff) {
1373cd9ad58dSDavid S. Miller /* Synchronous data transfer, always flush fifo. */
1374cd9ad58dSDavid S. Miller flush_fifo = 1;
1375cd9ad58dSDavid S. Miller } else {
1376cd9ad58dSDavid S. Miller if (esp->rev == ESP100) {
1377cd9ad58dSDavid S. Miller u32 fflags, phase;
1378cd9ad58dSDavid S. Miller
1379cd9ad58dSDavid S. Miller /* ESP100 has a chip bug where in the synchronous data
1380cd9ad58dSDavid S. Miller * phase it can mistake a final long REQ pulse from the
1381cd9ad58dSDavid S. Miller * target as an extra data byte. Fun.
1382cd9ad58dSDavid S. Miller *
1383cd9ad58dSDavid S. Miller * To detect this case we resample the status register
1384cd9ad58dSDavid S. Miller * and fifo flags. If we're still in a data phase and
1385cd9ad58dSDavid S. Miller * we see spurious chunks in the fifo, we return error
1386cd9ad58dSDavid S. Miller * to the caller which should reset and set things up
1387cd9ad58dSDavid S. Miller * such that we only try future transfers to this
1388cd9ad58dSDavid S. Miller * target in synchronous mode.
1389cd9ad58dSDavid S. Miller */
1390cd9ad58dSDavid S. Miller esp->sreg = esp_read8(ESP_STATUS);
1391cd9ad58dSDavid S. Miller phase = esp->sreg & ESP_STAT_PMASK;
1392cd9ad58dSDavid S. Miller fflags = esp_read8(ESP_FFLAGS);
1393cd9ad58dSDavid S. Miller
1394cd9ad58dSDavid S. Miller if ((phase == ESP_DOP &&
1395cd9ad58dSDavid S. Miller (fflags & ESP_FF_ONOTZERO)) ||
1396cd9ad58dSDavid S. Miller (phase == ESP_DIP &&
1397cd9ad58dSDavid S. Miller (fflags & ESP_FF_FBYTES)))
1398cd9ad58dSDavid S. Miller return -1;
1399cd9ad58dSDavid S. Miller }
1400cd9ad58dSDavid S. Miller if (!(ent->flags & ESP_CMD_FLAG_WRITE))
1401cd9ad58dSDavid S. Miller flush_fifo = 1;
1402cd9ad58dSDavid S. Miller }
1403cd9ad58dSDavid S. Miller
1404cd9ad58dSDavid S. Miller if (flush_fifo)
1405cd9ad58dSDavid S. Miller esp_flush_fifo(esp);
1406cd9ad58dSDavid S. Miller
1407cd9ad58dSDavid S. Miller return bytes_sent;
1408cd9ad58dSDavid S. Miller }
1409cd9ad58dSDavid S. Miller
esp_setsync(struct esp * esp,struct esp_target_data * tp,u8 scsi_period,u8 scsi_offset,u8 esp_stp,u8 esp_soff)1410cd9ad58dSDavid S. Miller static void esp_setsync(struct esp *esp, struct esp_target_data *tp,
1411cd9ad58dSDavid S. Miller u8 scsi_period, u8 scsi_offset,
1412cd9ad58dSDavid S. Miller u8 esp_stp, u8 esp_soff)
1413cd9ad58dSDavid S. Miller {
1414cd9ad58dSDavid S. Miller spi_period(tp->starget) = scsi_period;
1415cd9ad58dSDavid S. Miller spi_offset(tp->starget) = scsi_offset;
1416cd9ad58dSDavid S. Miller spi_width(tp->starget) = (tp->flags & ESP_TGT_WIDE) ? 1 : 0;
1417cd9ad58dSDavid S. Miller
1418cd9ad58dSDavid S. Miller if (esp_soff) {
1419cd9ad58dSDavid S. Miller esp_stp &= 0x1f;
1420cd9ad58dSDavid S. Miller esp_soff |= esp->radelay;
1421cd9ad58dSDavid S. Miller if (esp->rev >= FAS236) {
1422cd9ad58dSDavid S. Miller u8 bit = ESP_CONFIG3_FSCSI;
1423cd9ad58dSDavid S. Miller if (esp->rev >= FAS100A)
1424cd9ad58dSDavid S. Miller bit = ESP_CONFIG3_FAST;
1425cd9ad58dSDavid S. Miller
1426cd9ad58dSDavid S. Miller if (scsi_period < 50) {
1427cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
1428cd9ad58dSDavid S. Miller esp_soff &= ~esp->radelay;
1429cd9ad58dSDavid S. Miller tp->esp_config3 |= bit;
1430cd9ad58dSDavid S. Miller } else {
1431cd9ad58dSDavid S. Miller tp->esp_config3 &= ~bit;
1432cd9ad58dSDavid S. Miller }
1433cd9ad58dSDavid S. Miller esp->prev_cfg3 = tp->esp_config3;
1434cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
1435cd9ad58dSDavid S. Miller }
1436cd9ad58dSDavid S. Miller }
1437cd9ad58dSDavid S. Miller
1438cd9ad58dSDavid S. Miller tp->esp_period = esp->prev_stp = esp_stp;
1439cd9ad58dSDavid S. Miller tp->esp_offset = esp->prev_soff = esp_soff;
1440cd9ad58dSDavid S. Miller
1441cd9ad58dSDavid S. Miller esp_write8(esp_soff, ESP_SOFF);
1442cd9ad58dSDavid S. Miller esp_write8(esp_stp, ESP_STP);
1443cd9ad58dSDavid S. Miller
1444cd9ad58dSDavid S. Miller tp->flags &= ~(ESP_TGT_NEGO_SYNC | ESP_TGT_CHECK_NEGO);
1445cd9ad58dSDavid S. Miller
1446cd9ad58dSDavid S. Miller spi_display_xfer_agreement(tp->starget);
1447cd9ad58dSDavid S. Miller }
1448cd9ad58dSDavid S. Miller
esp_msgin_reject(struct esp * esp)1449cd9ad58dSDavid S. Miller static void esp_msgin_reject(struct esp *esp)
1450cd9ad58dSDavid S. Miller {
1451cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1452cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
1453cd9ad58dSDavid S. Miller struct esp_target_data *tp;
1454cd9ad58dSDavid S. Miller int tgt;
1455cd9ad58dSDavid S. Miller
1456cd9ad58dSDavid S. Miller tgt = cmd->device->id;
1457cd9ad58dSDavid S. Miller tp = &esp->target[tgt];
1458cd9ad58dSDavid S. Miller
1459cd9ad58dSDavid S. Miller if (tp->flags & ESP_TGT_NEGO_WIDE) {
1460cd9ad58dSDavid S. Miller tp->flags &= ~(ESP_TGT_NEGO_WIDE | ESP_TGT_WIDE);
1461cd9ad58dSDavid S. Miller
1462cd9ad58dSDavid S. Miller if (!esp_need_to_nego_sync(tp)) {
1463cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_CHECK_NEGO;
1464cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RATN);
1465cd9ad58dSDavid S. Miller } else {
1466cd9ad58dSDavid S. Miller esp->msg_out_len =
1467cd9ad58dSDavid S. Miller spi_populate_sync_msg(&esp->msg_out[0],
1468cd9ad58dSDavid S. Miller tp->nego_goal_period,
1469cd9ad58dSDavid S. Miller tp->nego_goal_offset);
1470cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_NEGO_SYNC;
1471cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1472cd9ad58dSDavid S. Miller }
1473cd9ad58dSDavid S. Miller return;
1474cd9ad58dSDavid S. Miller }
1475cd9ad58dSDavid S. Miller
1476cd9ad58dSDavid S. Miller if (tp->flags & ESP_TGT_NEGO_SYNC) {
1477cd9ad58dSDavid S. Miller tp->flags &= ~(ESP_TGT_NEGO_SYNC | ESP_TGT_CHECK_NEGO);
1478cd9ad58dSDavid S. Miller tp->esp_period = 0;
1479cd9ad58dSDavid S. Miller tp->esp_offset = 0;
1480cd9ad58dSDavid S. Miller esp_setsync(esp, tp, 0, 0, 0, 0);
1481cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RATN);
1482cd9ad58dSDavid S. Miller return;
1483cd9ad58dSDavid S. Miller }
1484cd9ad58dSDavid S. Miller
1485c69edff5SFinn Thain shost_printk(KERN_INFO, esp->host, "Unexpected MESSAGE REJECT\n");
1486c69edff5SFinn Thain esp_schedule_reset(esp);
1487cd9ad58dSDavid S. Miller }
1488cd9ad58dSDavid S. Miller
esp_msgin_sdtr(struct esp * esp,struct esp_target_data * tp)1489cd9ad58dSDavid S. Miller static void esp_msgin_sdtr(struct esp *esp, struct esp_target_data *tp)
1490cd9ad58dSDavid S. Miller {
1491cd9ad58dSDavid S. Miller u8 period = esp->msg_in[3];
1492cd9ad58dSDavid S. Miller u8 offset = esp->msg_in[4];
1493cd9ad58dSDavid S. Miller u8 stp;
1494cd9ad58dSDavid S. Miller
1495cd9ad58dSDavid S. Miller if (!(tp->flags & ESP_TGT_NEGO_SYNC))
1496cd9ad58dSDavid S. Miller goto do_reject;
1497cd9ad58dSDavid S. Miller
1498cd9ad58dSDavid S. Miller if (offset > 15)
1499cd9ad58dSDavid S. Miller goto do_reject;
1500cd9ad58dSDavid S. Miller
1501cd9ad58dSDavid S. Miller if (offset) {
1502237abac6SJulia Lawall int one_clock;
1503cd9ad58dSDavid S. Miller
1504cd9ad58dSDavid S. Miller if (period > esp->max_period) {
1505cd9ad58dSDavid S. Miller period = offset = 0;
1506cd9ad58dSDavid S. Miller goto do_sdtr;
1507cd9ad58dSDavid S. Miller }
1508cd9ad58dSDavid S. Miller if (period < esp->min_period)
1509cd9ad58dSDavid S. Miller goto do_reject;
1510cd9ad58dSDavid S. Miller
1511cd9ad58dSDavid S. Miller one_clock = esp->ccycle / 1000;
1512237abac6SJulia Lawall stp = DIV_ROUND_UP(period << 2, one_clock);
1513cd9ad58dSDavid S. Miller if (stp && esp->rev >= FAS236) {
1514cd9ad58dSDavid S. Miller if (stp >= 50)
1515cd9ad58dSDavid S. Miller stp--;
1516cd9ad58dSDavid S. Miller }
1517cd9ad58dSDavid S. Miller } else {
1518cd9ad58dSDavid S. Miller stp = 0;
1519cd9ad58dSDavid S. Miller }
1520cd9ad58dSDavid S. Miller
1521cd9ad58dSDavid S. Miller esp_setsync(esp, tp, period, offset, stp, offset);
1522cd9ad58dSDavid S. Miller return;
1523cd9ad58dSDavid S. Miller
1524cd9ad58dSDavid S. Miller do_reject:
1525cd9ad58dSDavid S. Miller esp->msg_out[0] = MESSAGE_REJECT;
1526cd9ad58dSDavid S. Miller esp->msg_out_len = 1;
1527cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1528cd9ad58dSDavid S. Miller return;
1529cd9ad58dSDavid S. Miller
1530cd9ad58dSDavid S. Miller do_sdtr:
1531cd9ad58dSDavid S. Miller tp->nego_goal_period = period;
1532cd9ad58dSDavid S. Miller tp->nego_goal_offset = offset;
1533cd9ad58dSDavid S. Miller esp->msg_out_len =
1534cd9ad58dSDavid S. Miller spi_populate_sync_msg(&esp->msg_out[0],
1535cd9ad58dSDavid S. Miller tp->nego_goal_period,
1536cd9ad58dSDavid S. Miller tp->nego_goal_offset);
1537cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1538cd9ad58dSDavid S. Miller }
1539cd9ad58dSDavid S. Miller
esp_msgin_wdtr(struct esp * esp,struct esp_target_data * tp)1540cd9ad58dSDavid S. Miller static void esp_msgin_wdtr(struct esp *esp, struct esp_target_data *tp)
1541cd9ad58dSDavid S. Miller {
1542cd9ad58dSDavid S. Miller int size = 8 << esp->msg_in[3];
1543cd9ad58dSDavid S. Miller u8 cfg3;
1544cd9ad58dSDavid S. Miller
1545cd9ad58dSDavid S. Miller if (esp->rev != FASHME)
1546cd9ad58dSDavid S. Miller goto do_reject;
1547cd9ad58dSDavid S. Miller
1548cd9ad58dSDavid S. Miller if (size != 8 && size != 16)
1549cd9ad58dSDavid S. Miller goto do_reject;
1550cd9ad58dSDavid S. Miller
1551cd9ad58dSDavid S. Miller if (!(tp->flags & ESP_TGT_NEGO_WIDE))
1552cd9ad58dSDavid S. Miller goto do_reject;
1553cd9ad58dSDavid S. Miller
1554cd9ad58dSDavid S. Miller cfg3 = tp->esp_config3;
1555cd9ad58dSDavid S. Miller if (size == 16) {
1556cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_WIDE;
1557cd9ad58dSDavid S. Miller cfg3 |= ESP_CONFIG3_EWIDE;
1558cd9ad58dSDavid S. Miller } else {
1559cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_WIDE;
1560cd9ad58dSDavid S. Miller cfg3 &= ~ESP_CONFIG3_EWIDE;
1561cd9ad58dSDavid S. Miller }
1562cd9ad58dSDavid S. Miller tp->esp_config3 = cfg3;
1563cd9ad58dSDavid S. Miller esp->prev_cfg3 = cfg3;
1564cd9ad58dSDavid S. Miller esp_write8(cfg3, ESP_CFG3);
1565cd9ad58dSDavid S. Miller
1566cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_NEGO_WIDE;
1567cd9ad58dSDavid S. Miller
1568cd9ad58dSDavid S. Miller spi_period(tp->starget) = 0;
1569cd9ad58dSDavid S. Miller spi_offset(tp->starget) = 0;
1570cd9ad58dSDavid S. Miller if (!esp_need_to_nego_sync(tp)) {
1571cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_CHECK_NEGO;
1572cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RATN);
1573cd9ad58dSDavid S. Miller } else {
1574cd9ad58dSDavid S. Miller esp->msg_out_len =
1575cd9ad58dSDavid S. Miller spi_populate_sync_msg(&esp->msg_out[0],
1576cd9ad58dSDavid S. Miller tp->nego_goal_period,
1577cd9ad58dSDavid S. Miller tp->nego_goal_offset);
1578cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_NEGO_SYNC;
1579cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1580cd9ad58dSDavid S. Miller }
1581cd9ad58dSDavid S. Miller return;
1582cd9ad58dSDavid S. Miller
1583cd9ad58dSDavid S. Miller do_reject:
1584cd9ad58dSDavid S. Miller esp->msg_out[0] = MESSAGE_REJECT;
1585cd9ad58dSDavid S. Miller esp->msg_out_len = 1;
1586cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1587cd9ad58dSDavid S. Miller }
1588cd9ad58dSDavid S. Miller
esp_msgin_extended(struct esp * esp)1589cd9ad58dSDavid S. Miller static void esp_msgin_extended(struct esp *esp)
1590cd9ad58dSDavid S. Miller {
1591cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1592cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
1593cd9ad58dSDavid S. Miller struct esp_target_data *tp;
1594cd9ad58dSDavid S. Miller int tgt = cmd->device->id;
1595cd9ad58dSDavid S. Miller
1596cd9ad58dSDavid S. Miller tp = &esp->target[tgt];
1597cd9ad58dSDavid S. Miller if (esp->msg_in[2] == EXTENDED_SDTR) {
1598cd9ad58dSDavid S. Miller esp_msgin_sdtr(esp, tp);
1599cd9ad58dSDavid S. Miller return;
1600cd9ad58dSDavid S. Miller }
1601cd9ad58dSDavid S. Miller if (esp->msg_in[2] == EXTENDED_WDTR) {
1602cd9ad58dSDavid S. Miller esp_msgin_wdtr(esp, tp);
1603cd9ad58dSDavid S. Miller return;
1604cd9ad58dSDavid S. Miller }
1605cd9ad58dSDavid S. Miller
1606a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1607a1a75b35SHannes Reinecke "Unexpected extended msg type %x\n", esp->msg_in[2]);
1608cd9ad58dSDavid S. Miller
1609c69edff5SFinn Thain esp->msg_out[0] = MESSAGE_REJECT;
1610cd9ad58dSDavid S. Miller esp->msg_out_len = 1;
1611cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1612cd9ad58dSDavid S. Miller }
1613cd9ad58dSDavid S. Miller
1614cd9ad58dSDavid S. Miller /* Analyze msgin bytes received from target so far. Return non-zero
1615cd9ad58dSDavid S. Miller * if there are more bytes needed to complete the message.
1616cd9ad58dSDavid S. Miller */
esp_msgin_process(struct esp * esp)1617cd9ad58dSDavid S. Miller static int esp_msgin_process(struct esp *esp)
1618cd9ad58dSDavid S. Miller {
1619cd9ad58dSDavid S. Miller u8 msg0 = esp->msg_in[0];
1620cd9ad58dSDavid S. Miller int len = esp->msg_in_len;
1621cd9ad58dSDavid S. Miller
1622cd9ad58dSDavid S. Miller if (msg0 & 0x80) {
1623cd9ad58dSDavid S. Miller /* Identify */
1624a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1625a1a75b35SHannes Reinecke "Unexpected msgin identify\n");
1626cd9ad58dSDavid S. Miller return 0;
1627cd9ad58dSDavid S. Miller }
1628cd9ad58dSDavid S. Miller
1629cd9ad58dSDavid S. Miller switch (msg0) {
1630cd9ad58dSDavid S. Miller case EXTENDED_MESSAGE:
1631cd9ad58dSDavid S. Miller if (len == 1)
1632cd9ad58dSDavid S. Miller return 1;
1633cd9ad58dSDavid S. Miller if (len < esp->msg_in[1] + 2)
1634cd9ad58dSDavid S. Miller return 1;
1635cd9ad58dSDavid S. Miller esp_msgin_extended(esp);
1636cd9ad58dSDavid S. Miller return 0;
1637cd9ad58dSDavid S. Miller
1638cd9ad58dSDavid S. Miller case IGNORE_WIDE_RESIDUE: {
1639cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent;
1640cd9ad58dSDavid S. Miller struct esp_cmd_priv *spriv;
1641cd9ad58dSDavid S. Miller if (len == 1)
1642cd9ad58dSDavid S. Miller return 1;
1643cd9ad58dSDavid S. Miller
1644cd9ad58dSDavid S. Miller if (esp->msg_in[1] != 1)
1645cd9ad58dSDavid S. Miller goto do_reject;
1646cd9ad58dSDavid S. Miller
1647cd9ad58dSDavid S. Miller ent = esp->active_cmd;
1648cd9ad58dSDavid S. Miller spriv = ESP_CMD_PRIV(ent->cmd);
1649cd9ad58dSDavid S. Miller
1650cd9ad58dSDavid S. Miller if (spriv->cur_residue == sg_dma_len(spriv->cur_sg)) {
1651ee5a1dbfSMing Lei spriv->cur_sg = spriv->prv_sg;
1652cd9ad58dSDavid S. Miller spriv->cur_residue = 1;
1653cd9ad58dSDavid S. Miller } else
1654cd9ad58dSDavid S. Miller spriv->cur_residue++;
1655cd9ad58dSDavid S. Miller spriv->tot_residue++;
1656cd9ad58dSDavid S. Miller return 0;
1657cd9ad58dSDavid S. Miller }
1658cd9ad58dSDavid S. Miller case NOP:
1659cd9ad58dSDavid S. Miller return 0;
1660cd9ad58dSDavid S. Miller case RESTORE_POINTERS:
1661cd9ad58dSDavid S. Miller esp_restore_pointers(esp, esp->active_cmd);
1662cd9ad58dSDavid S. Miller return 0;
1663cd9ad58dSDavid S. Miller case SAVE_POINTERS:
1664cd9ad58dSDavid S. Miller esp_save_pointers(esp, esp->active_cmd);
1665cd9ad58dSDavid S. Miller return 0;
1666cd9ad58dSDavid S. Miller
1667cd9ad58dSDavid S. Miller case COMMAND_COMPLETE:
1668cd9ad58dSDavid S. Miller case DISCONNECT: {
1669cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1670cd9ad58dSDavid S. Miller
1671cd9ad58dSDavid S. Miller ent->message = msg0;
1672cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_FREE_BUS);
1673cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1674cd9ad58dSDavid S. Miller return 0;
1675cd9ad58dSDavid S. Miller }
1676cd9ad58dSDavid S. Miller case MESSAGE_REJECT:
1677cd9ad58dSDavid S. Miller esp_msgin_reject(esp);
1678cd9ad58dSDavid S. Miller return 0;
1679cd9ad58dSDavid S. Miller
1680cd9ad58dSDavid S. Miller default:
1681cd9ad58dSDavid S. Miller do_reject:
1682cd9ad58dSDavid S. Miller esp->msg_out[0] = MESSAGE_REJECT;
1683cd9ad58dSDavid S. Miller esp->msg_out_len = 1;
1684cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
1685cd9ad58dSDavid S. Miller return 0;
1686cd9ad58dSDavid S. Miller }
1687cd9ad58dSDavid S. Miller }
1688cd9ad58dSDavid S. Miller
esp_process_event(struct esp * esp)1689cd9ad58dSDavid S. Miller static int esp_process_event(struct esp *esp)
1690cd9ad58dSDavid S. Miller {
16913170866fSHannes Reinecke int write, i;
1692cd9ad58dSDavid S. Miller
1693cd9ad58dSDavid S. Miller again:
1694cd9ad58dSDavid S. Miller write = 0;
16951af6f603SHannes Reinecke esp_log_event("process event %d phase %x\n",
16961af6f603SHannes Reinecke esp->event, esp->sreg & ESP_STAT_PMASK);
1697cd9ad58dSDavid S. Miller switch (esp->event) {
1698cd9ad58dSDavid S. Miller case ESP_EVENT_CHECK_PHASE:
1699cd9ad58dSDavid S. Miller switch (esp->sreg & ESP_STAT_PMASK) {
1700cd9ad58dSDavid S. Miller case ESP_DOP:
1701cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_DATA_OUT);
1702cd9ad58dSDavid S. Miller break;
1703cd9ad58dSDavid S. Miller case ESP_DIP:
1704cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_DATA_IN);
1705cd9ad58dSDavid S. Miller break;
1706cd9ad58dSDavid S. Miller case ESP_STATP:
1707cd9ad58dSDavid S. Miller esp_flush_fifo(esp);
1708cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_ICCSEQ);
1709cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_STATUS);
1710cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1711cd9ad58dSDavid S. Miller return 1;
1712cd9ad58dSDavid S. Miller
1713cd9ad58dSDavid S. Miller case ESP_MOP:
1714cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_MSGOUT);
1715cd9ad58dSDavid S. Miller break;
1716cd9ad58dSDavid S. Miller
1717cd9ad58dSDavid S. Miller case ESP_MIP:
1718cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_MSGIN);
1719cd9ad58dSDavid S. Miller break;
1720cd9ad58dSDavid S. Miller
1721cd9ad58dSDavid S. Miller case ESP_CMDP:
1722cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CMD_START);
1723cd9ad58dSDavid S. Miller break;
1724cd9ad58dSDavid S. Miller
1725cd9ad58dSDavid S. Miller default:
1726a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1727a1a75b35SHannes Reinecke "Unexpected phase, sreg=%02x\n",
1728cd9ad58dSDavid S. Miller esp->sreg);
1729cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1730cd9ad58dSDavid S. Miller return 0;
1731cd9ad58dSDavid S. Miller }
1732cd9ad58dSDavid S. Miller goto again;
1733cd9ad58dSDavid S. Miller
1734cd9ad58dSDavid S. Miller case ESP_EVENT_DATA_IN:
1735cd9ad58dSDavid S. Miller write = 1;
1736df561f66SGustavo A. R. Silva fallthrough;
1737cd9ad58dSDavid S. Miller
1738cd9ad58dSDavid S. Miller case ESP_EVENT_DATA_OUT: {
1739cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1740cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
1741cd9ad58dSDavid S. Miller dma_addr_t dma_addr = esp_cur_dma_addr(ent, cmd);
1742cd9ad58dSDavid S. Miller unsigned int dma_len = esp_cur_dma_len(ent, cmd);
1743cd9ad58dSDavid S. Miller
1744cd9ad58dSDavid S. Miller if (esp->rev == ESP100)
1745cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL);
1746cd9ad58dSDavid S. Miller
1747cd9ad58dSDavid S. Miller if (write)
1748cd9ad58dSDavid S. Miller ent->flags |= ESP_CMD_FLAG_WRITE;
1749cd9ad58dSDavid S. Miller else
1750cd9ad58dSDavid S. Miller ent->flags &= ~ESP_CMD_FLAG_WRITE;
1751cd9ad58dSDavid S. Miller
17526fe07aafSFinn Thain if (esp->ops->dma_length_limit)
17536fe07aafSFinn Thain dma_len = esp->ops->dma_length_limit(esp, dma_addr,
17546fe07aafSFinn Thain dma_len);
17556fe07aafSFinn Thain else
1756cd9ad58dSDavid S. Miller dma_len = esp_dma_length_limit(esp, dma_addr, dma_len);
17576fe07aafSFinn Thain
1758cd9ad58dSDavid S. Miller esp->data_dma_len = dma_len;
1759cd9ad58dSDavid S. Miller
1760cd9ad58dSDavid S. Miller if (!dma_len) {
1761a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1762a1a75b35SHannes Reinecke "DMA length is zero!\n");
1763a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
1764a1a75b35SHannes Reinecke "cur adr[%08llx] len[%08x]\n",
1765e1f2a094SAlexey Dobriyan (unsigned long long)esp_cur_dma_addr(ent, cmd),
1766cd9ad58dSDavid S. Miller esp_cur_dma_len(ent, cmd));
1767cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1768cd9ad58dSDavid S. Miller return 0;
1769cd9ad58dSDavid S. Miller }
1770cd9ad58dSDavid S. Miller
1771a1a75b35SHannes Reinecke esp_log_datastart("start data addr[%08llx] len[%u] write(%d)\n",
1772e1f2a094SAlexey Dobriyan (unsigned long long)dma_addr, dma_len, write);
1773cd9ad58dSDavid S. Miller
1774cd9ad58dSDavid S. Miller esp->ops->send_dma_cmd(esp, dma_addr, dma_len, dma_len,
1775cd9ad58dSDavid S. Miller write, ESP_CMD_DMA | ESP_CMD_TI);
1776cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_DATA_DONE);
1777cd9ad58dSDavid S. Miller break;
1778cd9ad58dSDavid S. Miller }
1779cd9ad58dSDavid S. Miller case ESP_EVENT_DATA_DONE: {
1780cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1781cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
1782cd9ad58dSDavid S. Miller int bytes_sent;
1783cd9ad58dSDavid S. Miller
1784cd9ad58dSDavid S. Miller if (esp->ops->dma_error(esp)) {
1785a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1786a1a75b35SHannes Reinecke "data done, DMA error, resetting\n");
1787cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1788cd9ad58dSDavid S. Miller return 0;
1789cd9ad58dSDavid S. Miller }
1790cd9ad58dSDavid S. Miller
1791cd9ad58dSDavid S. Miller if (ent->flags & ESP_CMD_FLAG_WRITE) {
1792cd9ad58dSDavid S. Miller /* XXX parity errors, etc. XXX */
1793cd9ad58dSDavid S. Miller
1794cd9ad58dSDavid S. Miller esp->ops->dma_drain(esp);
1795cd9ad58dSDavid S. Miller }
1796cd9ad58dSDavid S. Miller esp->ops->dma_invalidate(esp);
1797cd9ad58dSDavid S. Miller
1798cd9ad58dSDavid S. Miller if (esp->ireg != ESP_INTR_BSERV) {
1799cd9ad58dSDavid S. Miller /* We should always see exactly a bus-service
1800cd9ad58dSDavid S. Miller * interrupt at the end of a successful transfer.
1801cd9ad58dSDavid S. Miller */
1802a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1803a1a75b35SHannes Reinecke "data done, not BSERV, resetting\n");
1804cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1805cd9ad58dSDavid S. Miller return 0;
1806cd9ad58dSDavid S. Miller }
1807cd9ad58dSDavid S. Miller
1808cd9ad58dSDavid S. Miller bytes_sent = esp_data_bytes_sent(esp, ent, cmd);
1809cd9ad58dSDavid S. Miller
1810a1a75b35SHannes Reinecke esp_log_datadone("data done flgs[%x] sent[%d]\n",
1811cd9ad58dSDavid S. Miller ent->flags, bytes_sent);
1812cd9ad58dSDavid S. Miller
1813cd9ad58dSDavid S. Miller if (bytes_sent < 0) {
1814cd9ad58dSDavid S. Miller /* XXX force sync mode for this target XXX */
1815cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1816cd9ad58dSDavid S. Miller return 0;
1817cd9ad58dSDavid S. Miller }
1818cd9ad58dSDavid S. Miller
1819cd9ad58dSDavid S. Miller esp_advance_dma(esp, ent, cmd, bytes_sent);
1820cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
1821cd9ad58dSDavid S. Miller goto again;
1822cd9ad58dSDavid S. Miller }
1823cd9ad58dSDavid S. Miller
1824cd9ad58dSDavid S. Miller case ESP_EVENT_STATUS: {
1825cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1826cd9ad58dSDavid S. Miller
1827cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_FDONE) {
1828cd9ad58dSDavid S. Miller ent->status = esp_read8(ESP_FDATA);
1829cd9ad58dSDavid S. Miller ent->message = esp_read8(ESP_FDATA);
1830cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_MOK);
1831cd9ad58dSDavid S. Miller } else if (esp->ireg == ESP_INTR_BSERV) {
1832cd9ad58dSDavid S. Miller ent->status = esp_read8(ESP_FDATA);
1833cd9ad58dSDavid S. Miller ent->message = 0xff;
1834cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_MSGIN);
1835cd9ad58dSDavid S. Miller return 0;
1836cd9ad58dSDavid S. Miller }
1837cd9ad58dSDavid S. Miller
1838cd9ad58dSDavid S. Miller if (ent->message != COMMAND_COMPLETE) {
1839a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1840a1a75b35SHannes Reinecke "Unexpected message %x in status\n",
1841cd9ad58dSDavid S. Miller ent->message);
1842cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1843cd9ad58dSDavid S. Miller return 0;
1844cd9ad58dSDavid S. Miller }
1845cd9ad58dSDavid S. Miller
1846cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_FREE_BUS);
1847cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1848cd9ad58dSDavid S. Miller break;
1849cd9ad58dSDavid S. Miller }
1850cd9ad58dSDavid S. Miller case ESP_EVENT_FREE_BUS: {
1851cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent = esp->active_cmd;
1852cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
1853cd9ad58dSDavid S. Miller
1854cd9ad58dSDavid S. Miller if (ent->message == COMMAND_COMPLETE ||
1855cd9ad58dSDavid S. Miller ent->message == DISCONNECT)
1856cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_ESEL);
1857cd9ad58dSDavid S. Miller
1858cd9ad58dSDavid S. Miller if (ent->message == COMMAND_COMPLETE) {
1859a1a75b35SHannes Reinecke esp_log_cmddone("Command done status[%x] message[%x]\n",
1860cd9ad58dSDavid S. Miller ent->status, ent->message);
1861cd9ad58dSDavid S. Miller if (ent->status == SAM_STAT_TASK_SET_FULL)
1862cd9ad58dSDavid S. Miller esp_event_queue_full(esp, ent);
1863cd9ad58dSDavid S. Miller
1864cd9ad58dSDavid S. Miller if (ent->status == SAM_STAT_CHECK_CONDITION &&
1865cd9ad58dSDavid S. Miller !(ent->flags & ESP_CMD_FLAG_AUTOSENSE)) {
1866cd9ad58dSDavid S. Miller ent->flags |= ESP_CMD_FLAG_AUTOSENSE;
1867cd9ad58dSDavid S. Miller esp_autosense(esp, ent);
1868cd9ad58dSDavid S. Miller } else {
18696b50529eSHannes Reinecke esp_cmd_is_done(esp, ent, cmd, DID_OK);
1870cd9ad58dSDavid S. Miller }
1871cd9ad58dSDavid S. Miller } else if (ent->message == DISCONNECT) {
1872a1a75b35SHannes Reinecke esp_log_disconnect("Disconnecting tgt[%d] tag[%x:%x]\n",
1873cd9ad58dSDavid S. Miller cmd->device->id,
1874cd9ad58dSDavid S. Miller ent->tag[0], ent->tag[1]);
1875cd9ad58dSDavid S. Miller
1876cd9ad58dSDavid S. Miller esp->active_cmd = NULL;
1877cd9ad58dSDavid S. Miller esp_maybe_execute_command(esp);
1878cd9ad58dSDavid S. Miller } else {
1879a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1880a1a75b35SHannes Reinecke "Unexpected message %x in freebus\n",
1881cd9ad58dSDavid S. Miller ent->message);
1882cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1883cd9ad58dSDavid S. Miller return 0;
1884cd9ad58dSDavid S. Miller }
1885cd9ad58dSDavid S. Miller if (esp->active_cmd)
1886cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1887cd9ad58dSDavid S. Miller break;
1888cd9ad58dSDavid S. Miller }
1889cd9ad58dSDavid S. Miller case ESP_EVENT_MSGOUT: {
1890cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1891cd9ad58dSDavid S. Miller
1892cd9ad58dSDavid S. Miller if (esp_debug & ESP_DEBUG_MSGOUT) {
1893cd9ad58dSDavid S. Miller int i;
1894cd9ad58dSDavid S. Miller printk("ESP: Sending message [ ");
1895cd9ad58dSDavid S. Miller for (i = 0; i < esp->msg_out_len; i++)
1896cd9ad58dSDavid S. Miller printk("%02x ", esp->msg_out[i]);
1897cd9ad58dSDavid S. Miller printk("]\n");
1898cd9ad58dSDavid S. Miller }
1899cd9ad58dSDavid S. Miller
1900cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
1901cd9ad58dSDavid S. Miller int i;
1902cd9ad58dSDavid S. Miller
1903cd9ad58dSDavid S. Miller /* Always use the fifo. */
1904cd9ad58dSDavid S. Miller for (i = 0; i < esp->msg_out_len; i++) {
1905cd9ad58dSDavid S. Miller esp_write8(esp->msg_out[i], ESP_FDATA);
1906cd9ad58dSDavid S. Miller esp_write8(0, ESP_FDATA);
1907cd9ad58dSDavid S. Miller }
1908cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_TI);
1909cd9ad58dSDavid S. Miller } else {
1910cd9ad58dSDavid S. Miller if (esp->msg_out_len == 1) {
1911cd9ad58dSDavid S. Miller esp_write8(esp->msg_out[0], ESP_FDATA);
1912cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_TI);
19133170866fSHannes Reinecke } else if (esp->flags & ESP_FLAG_USE_FIFO) {
19143170866fSHannes Reinecke for (i = 0; i < esp->msg_out_len; i++)
19153170866fSHannes Reinecke esp_write8(esp->msg_out[i], ESP_FDATA);
19163170866fSHannes Reinecke scsi_esp_cmd(esp, ESP_CMD_TI);
1917cd9ad58dSDavid S. Miller } else {
1918cd9ad58dSDavid S. Miller /* Use DMA. */
1919cd9ad58dSDavid S. Miller memcpy(esp->command_block,
1920cd9ad58dSDavid S. Miller esp->msg_out,
1921cd9ad58dSDavid S. Miller esp->msg_out_len);
1922cd9ad58dSDavid S. Miller
1923cd9ad58dSDavid S. Miller esp->ops->send_dma_cmd(esp,
1924cd9ad58dSDavid S. Miller esp->command_block_dma,
1925cd9ad58dSDavid S. Miller esp->msg_out_len,
1926cd9ad58dSDavid S. Miller esp->msg_out_len,
1927cd9ad58dSDavid S. Miller 0,
1928cd9ad58dSDavid S. Miller ESP_CMD_DMA|ESP_CMD_TI);
1929cd9ad58dSDavid S. Miller }
1930cd9ad58dSDavid S. Miller }
1931cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_MSGOUT_DONE);
1932cd9ad58dSDavid S. Miller break;
1933cd9ad58dSDavid S. Miller }
1934cd9ad58dSDavid S. Miller case ESP_EVENT_MSGOUT_DONE:
1935cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
1936cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1937cd9ad58dSDavid S. Miller } else {
1938cd9ad58dSDavid S. Miller if (esp->msg_out_len > 1)
1939cd9ad58dSDavid S. Miller esp->ops->dma_invalidate(esp);
1940cd9ad58dSDavid S. Miller
1941201c37d7SFinn Thain /* XXX if the chip went into disconnected mode,
1942201c37d7SFinn Thain * we can't run the phase state machine anyway.
1943201c37d7SFinn Thain */
1944201c37d7SFinn Thain if (!(esp->ireg & ESP_INTR_DC))
1945cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL);
1946cd9ad58dSDavid S. Miller }
1947201c37d7SFinn Thain
1948d60e9eecSFinn Thain esp->msg_out_len = 0;
1949d60e9eecSFinn Thain
1950cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
1951cd9ad58dSDavid S. Miller goto again;
1952cd9ad58dSDavid S. Miller case ESP_EVENT_MSGIN:
1953cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_BSERV) {
1954cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
1955cd9ad58dSDavid S. Miller if (!(esp_read8(ESP_STATUS2) &
1956cd9ad58dSDavid S. Miller ESP_STAT2_FEMPTY))
1957cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1958cd9ad58dSDavid S. Miller } else {
1959cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1960cd9ad58dSDavid S. Miller if (esp->rev == ESP100)
1961cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_NULL);
1962cd9ad58dSDavid S. Miller }
1963cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_TI);
1964cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
1965cd9ad58dSDavid S. Miller return 1;
1966cd9ad58dSDavid S. Miller }
1967cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_FDONE) {
1968cd9ad58dSDavid S. Miller u8 val;
1969cd9ad58dSDavid S. Miller
1970cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
1971cd9ad58dSDavid S. Miller val = esp->fifo[0];
1972cd9ad58dSDavid S. Miller else
1973cd9ad58dSDavid S. Miller val = esp_read8(ESP_FDATA);
1974cd9ad58dSDavid S. Miller esp->msg_in[esp->msg_in_len++] = val;
1975cd9ad58dSDavid S. Miller
1976a1a75b35SHannes Reinecke esp_log_msgin("Got msgin byte %x\n", val);
1977cd9ad58dSDavid S. Miller
1978cd9ad58dSDavid S. Miller if (!esp_msgin_process(esp))
1979cd9ad58dSDavid S. Miller esp->msg_in_len = 0;
1980cd9ad58dSDavid S. Miller
1981cd9ad58dSDavid S. Miller if (esp->rev == FASHME)
1982cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_FLUSH);
1983cd9ad58dSDavid S. Miller
1984cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_MOK);
1985cd9ad58dSDavid S. Miller
1986c69edff5SFinn Thain /* Check whether a bus reset is to be done next */
1987c69edff5SFinn Thain if (esp->event == ESP_EVENT_RESET)
1988c69edff5SFinn Thain return 0;
1989c69edff5SFinn Thain
1990cd9ad58dSDavid S. Miller if (esp->event != ESP_EVENT_FREE_BUS)
1991cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
1992cd9ad58dSDavid S. Miller } else {
1993a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
1994a1a75b35SHannes Reinecke "MSGIN neither BSERV not FDON, resetting");
1995cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
1996cd9ad58dSDavid S. Miller return 0;
1997cd9ad58dSDavid S. Miller }
1998cd9ad58dSDavid S. Miller break;
1999cd9ad58dSDavid S. Miller case ESP_EVENT_CMD_START:
2000cd9ad58dSDavid S. Miller memcpy(esp->command_block, esp->cmd_bytes_ptr,
2001cd9ad58dSDavid S. Miller esp->cmd_bytes_left);
20023170866fSHannes Reinecke esp_send_dma_cmd(esp, esp->cmd_bytes_left, 16, ESP_CMD_TI);
2003cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CMD_DONE);
2004cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_QUICKIRQ_CHECK;
2005cd9ad58dSDavid S. Miller break;
2006cd9ad58dSDavid S. Miller case ESP_EVENT_CMD_DONE:
2007cd9ad58dSDavid S. Miller esp->ops->dma_invalidate(esp);
2008cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_BSERV) {
2009cd9ad58dSDavid S. Miller esp_event(esp, ESP_EVENT_CHECK_PHASE);
2010cd9ad58dSDavid S. Miller goto again;
2011cd9ad58dSDavid S. Miller }
2012cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
2013cd9ad58dSDavid S. Miller return 0;
2014cd9ad58dSDavid S. Miller
2015cd9ad58dSDavid S. Miller case ESP_EVENT_RESET:
2016cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RS);
2017cd9ad58dSDavid S. Miller break;
2018cd9ad58dSDavid S. Miller
2019cd9ad58dSDavid S. Miller default:
2020a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
2021a1a75b35SHannes Reinecke "Unexpected event %x, resetting\n", esp->event);
2022cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
2023cd9ad58dSDavid S. Miller return 0;
2024cd9ad58dSDavid S. Miller }
2025cd9ad58dSDavid S. Miller return 1;
2026cd9ad58dSDavid S. Miller }
2027cd9ad58dSDavid S. Miller
esp_reset_cleanup_one(struct esp * esp,struct esp_cmd_entry * ent)2028cd9ad58dSDavid S. Miller static void esp_reset_cleanup_one(struct esp *esp, struct esp_cmd_entry *ent)
2029cd9ad58dSDavid S. Miller {
2030cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
2031cd9ad58dSDavid S. Miller
2032cd9ad58dSDavid S. Miller esp_unmap_dma(esp, cmd);
2033cd9ad58dSDavid S. Miller esp_free_lun_tag(ent, cmd->device->hostdata);
2034cd9ad58dSDavid S. Miller cmd->result = DID_RESET << 16;
2035cd9ad58dSDavid S. Miller
20363f9295b6SChristoph Hellwig if (ent->flags & ESP_CMD_FLAG_AUTOSENSE)
20373f9295b6SChristoph Hellwig esp_unmap_sense(esp, ent);
2038cd9ad58dSDavid S. Miller
2039f8ab27d9SBart Van Assche scsi_done(cmd);
2040cd9ad58dSDavid S. Miller list_del(&ent->list);
2041cd9ad58dSDavid S. Miller esp_put_ent(esp, ent);
2042cd9ad58dSDavid S. Miller }
2043cd9ad58dSDavid S. Miller
esp_clear_hold(struct scsi_device * dev,void * data)2044cd9ad58dSDavid S. Miller static void esp_clear_hold(struct scsi_device *dev, void *data)
2045cd9ad58dSDavid S. Miller {
2046cd9ad58dSDavid S. Miller struct esp_lun_data *lp = dev->hostdata;
2047cd9ad58dSDavid S. Miller
2048cd9ad58dSDavid S. Miller BUG_ON(lp->num_tagged);
2049cd9ad58dSDavid S. Miller lp->hold = 0;
2050cd9ad58dSDavid S. Miller }
2051cd9ad58dSDavid S. Miller
esp_reset_cleanup(struct esp * esp)2052cd9ad58dSDavid S. Miller static void esp_reset_cleanup(struct esp *esp)
2053cd9ad58dSDavid S. Miller {
2054cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent, *tmp;
2055cd9ad58dSDavid S. Miller int i;
2056cd9ad58dSDavid S. Miller
2057cd9ad58dSDavid S. Miller list_for_each_entry_safe(ent, tmp, &esp->queued_cmds, list) {
2058cd9ad58dSDavid S. Miller struct scsi_cmnd *cmd = ent->cmd;
2059cd9ad58dSDavid S. Miller
2060cd9ad58dSDavid S. Miller list_del(&ent->list);
2061cd9ad58dSDavid S. Miller cmd->result = DID_RESET << 16;
2062f8ab27d9SBart Van Assche scsi_done(cmd);
2063cd9ad58dSDavid S. Miller esp_put_ent(esp, ent);
2064cd9ad58dSDavid S. Miller }
2065cd9ad58dSDavid S. Miller
2066cd9ad58dSDavid S. Miller list_for_each_entry_safe(ent, tmp, &esp->active_cmds, list) {
2067cd9ad58dSDavid S. Miller if (ent == esp->active_cmd)
2068cd9ad58dSDavid S. Miller esp->active_cmd = NULL;
2069cd9ad58dSDavid S. Miller esp_reset_cleanup_one(esp, ent);
2070cd9ad58dSDavid S. Miller }
2071cd9ad58dSDavid S. Miller
2072cd9ad58dSDavid S. Miller BUG_ON(esp->active_cmd != NULL);
2073cd9ad58dSDavid S. Miller
2074cd9ad58dSDavid S. Miller /* Force renegotiation of sync/wide transfers. */
2075cd9ad58dSDavid S. Miller for (i = 0; i < ESP_MAX_TARGET; i++) {
2076cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[i];
2077cd9ad58dSDavid S. Miller
2078cd9ad58dSDavid S. Miller tp->esp_period = 0;
2079cd9ad58dSDavid S. Miller tp->esp_offset = 0;
2080cd9ad58dSDavid S. Miller tp->esp_config3 &= ~(ESP_CONFIG3_EWIDE |
2081cd9ad58dSDavid S. Miller ESP_CONFIG3_FSCSI |
2082cd9ad58dSDavid S. Miller ESP_CONFIG3_FAST);
2083cd9ad58dSDavid S. Miller tp->flags &= ~ESP_TGT_WIDE;
2084cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_CHECK_NEGO;
2085cd9ad58dSDavid S. Miller
2086cd9ad58dSDavid S. Miller if (tp->starget)
2087522939d4SMaciej W. Rozycki __starget_for_each_device(tp->starget, NULL,
2088cd9ad58dSDavid S. Miller esp_clear_hold);
2089cd9ad58dSDavid S. Miller }
2090204abf28SThomas Bogendoerfer esp->flags &= ~ESP_FLAG_RESETTING;
2091cd9ad58dSDavid S. Miller }
2092cd9ad58dSDavid S. Miller
2093cd9ad58dSDavid S. Miller /* Runs under host->lock */
__esp_interrupt(struct esp * esp)2094cd9ad58dSDavid S. Miller static void __esp_interrupt(struct esp *esp)
2095cd9ad58dSDavid S. Miller {
2096cd9ad58dSDavid S. Miller int finish_reset, intr_done;
2097cd9ad58dSDavid S. Miller u8 phase;
2098cd9ad58dSDavid S. Miller
20999535fff3SHannes Reinecke /*
21009535fff3SHannes Reinecke * Once INTRPT is read STATUS and SSTEP are cleared.
21019535fff3SHannes Reinecke */
2102cd9ad58dSDavid S. Miller esp->sreg = esp_read8(ESP_STATUS);
21039535fff3SHannes Reinecke esp->seqreg = esp_read8(ESP_SSTEP);
21049535fff3SHannes Reinecke esp->ireg = esp_read8(ESP_INTRPT);
2105cd9ad58dSDavid S. Miller
2106cd9ad58dSDavid S. Miller if (esp->flags & ESP_FLAG_RESETTING) {
2107cd9ad58dSDavid S. Miller finish_reset = 1;
2108cd9ad58dSDavid S. Miller } else {
2109cd9ad58dSDavid S. Miller if (esp_check_gross_error(esp))
2110cd9ad58dSDavid S. Miller return;
2111cd9ad58dSDavid S. Miller
2112cd9ad58dSDavid S. Miller finish_reset = esp_check_spur_intr(esp);
2113cd9ad58dSDavid S. Miller if (finish_reset < 0)
2114cd9ad58dSDavid S. Miller return;
2115cd9ad58dSDavid S. Miller }
2116cd9ad58dSDavid S. Miller
2117cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_SR)
2118cd9ad58dSDavid S. Miller finish_reset = 1;
2119cd9ad58dSDavid S. Miller
2120cd9ad58dSDavid S. Miller if (finish_reset) {
2121cd9ad58dSDavid S. Miller esp_reset_cleanup(esp);
2122cd9ad58dSDavid S. Miller if (esp->eh_reset) {
2123cd9ad58dSDavid S. Miller complete(esp->eh_reset);
2124cd9ad58dSDavid S. Miller esp->eh_reset = NULL;
2125cd9ad58dSDavid S. Miller }
2126cd9ad58dSDavid S. Miller return;
2127cd9ad58dSDavid S. Miller }
2128cd9ad58dSDavid S. Miller
2129cd9ad58dSDavid S. Miller phase = (esp->sreg & ESP_STAT_PMASK);
2130cd9ad58dSDavid S. Miller if (esp->rev == FASHME) {
2131cd9ad58dSDavid S. Miller if (((phase != ESP_DIP && phase != ESP_DOP) &&
2132cd9ad58dSDavid S. Miller esp->select_state == ESP_SELECT_NONE &&
2133cd9ad58dSDavid S. Miller esp->event != ESP_EVENT_STATUS &&
2134cd9ad58dSDavid S. Miller esp->event != ESP_EVENT_DATA_DONE) ||
2135cd9ad58dSDavid S. Miller (esp->ireg & ESP_INTR_RSEL)) {
2136cd9ad58dSDavid S. Miller esp->sreg2 = esp_read8(ESP_STATUS2);
2137cd9ad58dSDavid S. Miller if (!(esp->sreg2 & ESP_STAT2_FEMPTY) ||
2138cd9ad58dSDavid S. Miller (esp->sreg2 & ESP_STAT2_F1BYTE))
2139cd9ad58dSDavid S. Miller hme_read_fifo(esp);
2140cd9ad58dSDavid S. Miller }
2141cd9ad58dSDavid S. Miller }
2142cd9ad58dSDavid S. Miller
2143a1a75b35SHannes Reinecke esp_log_intr("intr sreg[%02x] seqreg[%02x] "
2144cd9ad58dSDavid S. Miller "sreg2[%02x] ireg[%02x]\n",
2145cd9ad58dSDavid S. Miller esp->sreg, esp->seqreg, esp->sreg2, esp->ireg);
2146cd9ad58dSDavid S. Miller
2147cd9ad58dSDavid S. Miller intr_done = 0;
2148cd9ad58dSDavid S. Miller
2149cd9ad58dSDavid S. Miller if (esp->ireg & (ESP_INTR_S | ESP_INTR_SATN | ESP_INTR_IC)) {
2150a1a75b35SHannes Reinecke shost_printk(KERN_INFO, esp->host,
2151a1a75b35SHannes Reinecke "unexpected IREG %02x\n", esp->ireg);
2152cd9ad58dSDavid S. Miller if (esp->ireg & ESP_INTR_IC)
2153cd9ad58dSDavid S. Miller esp_dump_cmd_log(esp);
2154cd9ad58dSDavid S. Miller
2155cd9ad58dSDavid S. Miller esp_schedule_reset(esp);
2156cd9ad58dSDavid S. Miller } else {
2157201c37d7SFinn Thain if (esp->ireg & ESP_INTR_RSEL) {
2158cd9ad58dSDavid S. Miller if (esp->active_cmd)
2159cd9ad58dSDavid S. Miller (void) esp_finish_select(esp);
2160cd9ad58dSDavid S. Miller intr_done = esp_reconnect(esp);
2161201c37d7SFinn Thain } else {
2162201c37d7SFinn Thain /* Some combination of FDONE, BSERV, DC. */
2163201c37d7SFinn Thain if (esp->select_state != ESP_SELECT_NONE)
2164201c37d7SFinn Thain intr_done = esp_finish_select(esp);
2165cd9ad58dSDavid S. Miller }
2166cd9ad58dSDavid S. Miller }
2167cd9ad58dSDavid S. Miller while (!intr_done)
2168cd9ad58dSDavid S. Miller intr_done = esp_process_event(esp);
2169cd9ad58dSDavid S. Miller }
2170cd9ad58dSDavid S. Miller
scsi_esp_intr(int irq,void * dev_id)2171cd9ad58dSDavid S. Miller irqreturn_t scsi_esp_intr(int irq, void *dev_id)
2172cd9ad58dSDavid S. Miller {
2173cd9ad58dSDavid S. Miller struct esp *esp = dev_id;
2174cd9ad58dSDavid S. Miller unsigned long flags;
2175cd9ad58dSDavid S. Miller irqreturn_t ret;
2176cd9ad58dSDavid S. Miller
2177cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2178cd9ad58dSDavid S. Miller ret = IRQ_NONE;
2179cd9ad58dSDavid S. Miller if (esp->ops->irq_pending(esp)) {
2180cd9ad58dSDavid S. Miller ret = IRQ_HANDLED;
2181cd9ad58dSDavid S. Miller for (;;) {
2182cd9ad58dSDavid S. Miller int i;
2183cd9ad58dSDavid S. Miller
2184cd9ad58dSDavid S. Miller __esp_interrupt(esp);
2185cd9ad58dSDavid S. Miller if (!(esp->flags & ESP_FLAG_QUICKIRQ_CHECK))
2186cd9ad58dSDavid S. Miller break;
2187cd9ad58dSDavid S. Miller esp->flags &= ~ESP_FLAG_QUICKIRQ_CHECK;
2188cd9ad58dSDavid S. Miller
2189cd9ad58dSDavid S. Miller for (i = 0; i < ESP_QUICKIRQ_LIMIT; i++) {
2190cd9ad58dSDavid S. Miller if (esp->ops->irq_pending(esp))
2191cd9ad58dSDavid S. Miller break;
2192cd9ad58dSDavid S. Miller }
2193cd9ad58dSDavid S. Miller if (i == ESP_QUICKIRQ_LIMIT)
2194cd9ad58dSDavid S. Miller break;
2195cd9ad58dSDavid S. Miller }
2196cd9ad58dSDavid S. Miller }
2197cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2198cd9ad58dSDavid S. Miller
2199cd9ad58dSDavid S. Miller return ret;
2200cd9ad58dSDavid S. Miller }
2201cd9ad58dSDavid S. Miller EXPORT_SYMBOL(scsi_esp_intr);
2202cd9ad58dSDavid S. Miller
esp_get_revision(struct esp * esp)220376246808SAdrian Bunk static void esp_get_revision(struct esp *esp)
2204cd9ad58dSDavid S. Miller {
2205cd9ad58dSDavid S. Miller u8 val;
2206cd9ad58dSDavid S. Miller
2207cd9ad58dSDavid S. Miller esp->config1 = (ESP_CONFIG1_PENABLE | (esp->scsi_id & 7));
22088a9aeb45SPaolo Bonzini if (esp->config2 == 0) {
2209cd9ad58dSDavid S. Miller esp->config2 = (ESP_CONFIG2_SCSI2ENAB | ESP_CONFIG2_REGPARITY);
2210cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
2211cd9ad58dSDavid S. Miller
2212cd9ad58dSDavid S. Miller val = esp_read8(ESP_CFG2);
2213cd9ad58dSDavid S. Miller val &= ~ESP_CONFIG2_MAGIC;
22148a9aeb45SPaolo Bonzini
22158a9aeb45SPaolo Bonzini esp->config2 = 0;
2216cd9ad58dSDavid S. Miller if (val != (ESP_CONFIG2_SCSI2ENAB | ESP_CONFIG2_REGPARITY)) {
22178a9aeb45SPaolo Bonzini /*
22188a9aeb45SPaolo Bonzini * If what we write to cfg2 does not come back,
22198a9aeb45SPaolo Bonzini * cfg2 is not implemented.
22208a9aeb45SPaolo Bonzini * Therefore this must be a plain esp100.
2221cd9ad58dSDavid S. Miller */
2222cd9ad58dSDavid S. Miller esp->rev = ESP100;
22238a9aeb45SPaolo Bonzini return;
22248a9aeb45SPaolo Bonzini }
22258a9aeb45SPaolo Bonzini }
22268a9aeb45SPaolo Bonzini
2227cd9ad58dSDavid S. Miller esp_set_all_config3(esp, 5);
2228cd9ad58dSDavid S. Miller esp->prev_cfg3 = 5;
2229cd9ad58dSDavid S. Miller esp_write8(esp->config2, ESP_CFG2);
2230cd9ad58dSDavid S. Miller esp_write8(0, ESP_CFG3);
2231cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
2232cd9ad58dSDavid S. Miller
2233cd9ad58dSDavid S. Miller val = esp_read8(ESP_CFG3);
2234cd9ad58dSDavid S. Miller if (val != 5) {
2235cd9ad58dSDavid S. Miller /* The cfg2 register is implemented, however
2236cd9ad58dSDavid S. Miller * cfg3 is not, must be esp100a.
2237cd9ad58dSDavid S. Miller */
2238cd9ad58dSDavid S. Miller esp->rev = ESP100A;
2239cd9ad58dSDavid S. Miller } else {
2240cd9ad58dSDavid S. Miller esp_set_all_config3(esp, 0);
2241cd9ad58dSDavid S. Miller esp->prev_cfg3 = 0;
2242cd9ad58dSDavid S. Miller esp_write8(esp->prev_cfg3, ESP_CFG3);
2243cd9ad58dSDavid S. Miller
2244cd9ad58dSDavid S. Miller /* All of cfg{1,2,3} implemented, must be one of
2245cd9ad58dSDavid S. Miller * the fas variants, figure out which one.
2246cd9ad58dSDavid S. Miller */
2247cd9ad58dSDavid S. Miller if (esp->cfact == 0 || esp->cfact > ESP_CCF_F5) {
2248cd9ad58dSDavid S. Miller esp->rev = FAST;
2249cd9ad58dSDavid S. Miller esp->sync_defp = SYNC_DEFP_FAST;
2250cd9ad58dSDavid S. Miller } else {
2251cd9ad58dSDavid S. Miller esp->rev = ESP236;
2252cd9ad58dSDavid S. Miller }
2253cd9ad58dSDavid S. Miller }
2254cd9ad58dSDavid S. Miller }
2255cd9ad58dSDavid S. Miller
esp_init_swstate(struct esp * esp)225676246808SAdrian Bunk static void esp_init_swstate(struct esp *esp)
2257cd9ad58dSDavid S. Miller {
2258cd9ad58dSDavid S. Miller int i;
2259cd9ad58dSDavid S. Miller
2260cd9ad58dSDavid S. Miller INIT_LIST_HEAD(&esp->queued_cmds);
2261cd9ad58dSDavid S. Miller INIT_LIST_HEAD(&esp->active_cmds);
2262cd9ad58dSDavid S. Miller INIT_LIST_HEAD(&esp->esp_cmd_pool);
2263cd9ad58dSDavid S. Miller
2264cd9ad58dSDavid S. Miller /* Start with a clear state, domain validation (via ->slave_configure,
2265cd9ad58dSDavid S. Miller * spi_dv_device()) will attempt to enable SYNC, WIDE, and tagged
2266cd9ad58dSDavid S. Miller * commands.
2267cd9ad58dSDavid S. Miller */
2268cd9ad58dSDavid S. Miller for (i = 0 ; i < ESP_MAX_TARGET; i++) {
2269cd9ad58dSDavid S. Miller esp->target[i].flags = 0;
2270cd9ad58dSDavid S. Miller esp->target[i].nego_goal_period = 0;
2271cd9ad58dSDavid S. Miller esp->target[i].nego_goal_offset = 0;
2272cd9ad58dSDavid S. Miller esp->target[i].nego_goal_width = 0;
2273cd9ad58dSDavid S. Miller esp->target[i].nego_goal_tags = 0;
2274cd9ad58dSDavid S. Miller }
2275cd9ad58dSDavid S. Miller }
2276cd9ad58dSDavid S. Miller
2277cd9ad58dSDavid S. Miller /* This places the ESP into a known state at boot time. */
esp_bootup_reset(struct esp * esp)2278d679f805SMartin Habets static void esp_bootup_reset(struct esp *esp)
2279cd9ad58dSDavid S. Miller {
2280cd9ad58dSDavid S. Miller u8 val;
2281cd9ad58dSDavid S. Miller
2282cd9ad58dSDavid S. Miller /* Reset the DMA */
2283cd9ad58dSDavid S. Miller esp->ops->reset_dma(esp);
2284cd9ad58dSDavid S. Miller
2285cd9ad58dSDavid S. Miller /* Reset the ESP */
2286cd9ad58dSDavid S. Miller esp_reset_esp(esp);
2287cd9ad58dSDavid S. Miller
2288cd9ad58dSDavid S. Miller /* Reset the SCSI bus, but tell ESP not to generate an irq */
2289cd9ad58dSDavid S. Miller val = esp_read8(ESP_CFG1);
2290cd9ad58dSDavid S. Miller val |= ESP_CONFIG1_SRRDISAB;
2291cd9ad58dSDavid S. Miller esp_write8(val, ESP_CFG1);
2292cd9ad58dSDavid S. Miller
2293cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RS);
2294cd9ad58dSDavid S. Miller udelay(400);
2295cd9ad58dSDavid S. Miller
2296cd9ad58dSDavid S. Miller esp_write8(esp->config1, ESP_CFG1);
2297cd9ad58dSDavid S. Miller
2298cd9ad58dSDavid S. Miller /* Eat any bitrot in the chip and we are done... */
2299cd9ad58dSDavid S. Miller esp_read8(ESP_INTRPT);
2300cd9ad58dSDavid S. Miller }
2301cd9ad58dSDavid S. Miller
esp_set_clock_params(struct esp * esp)230276246808SAdrian Bunk static void esp_set_clock_params(struct esp *esp)
2303cd9ad58dSDavid S. Miller {
23046fe07aafSFinn Thain int fhz;
2305cd9ad58dSDavid S. Miller u8 ccf;
2306cd9ad58dSDavid S. Miller
2307cd9ad58dSDavid S. Miller /* This is getting messy but it has to be done correctly or else
2308cd9ad58dSDavid S. Miller * you get weird behavior all over the place. We are trying to
2309cd9ad58dSDavid S. Miller * basically figure out three pieces of information.
2310cd9ad58dSDavid S. Miller *
2311cd9ad58dSDavid S. Miller * a) Clock Conversion Factor
2312cd9ad58dSDavid S. Miller *
2313cd9ad58dSDavid S. Miller * This is a representation of the input crystal clock frequency
2314cd9ad58dSDavid S. Miller * going into the ESP on this machine. Any operation whose timing
2315cd9ad58dSDavid S. Miller * is longer than 400ns depends on this value being correct. For
2316cd9ad58dSDavid S. Miller * example, you'll get blips for arbitration/selection during high
2317cd9ad58dSDavid S. Miller * load or with multiple targets if this is not set correctly.
2318cd9ad58dSDavid S. Miller *
2319cd9ad58dSDavid S. Miller * b) Selection Time-Out
2320cd9ad58dSDavid S. Miller *
2321cd9ad58dSDavid S. Miller * The ESP isn't very bright and will arbitrate for the bus and try
2322cd9ad58dSDavid S. Miller * to select a target forever if you let it. This value tells the
2323cd9ad58dSDavid S. Miller * ESP when it has taken too long to negotiate and that it should
2324cd9ad58dSDavid S. Miller * interrupt the CPU so we can see what happened. The value is
2325cd9ad58dSDavid S. Miller * computed as follows (from NCR/Symbios chip docs).
2326cd9ad58dSDavid S. Miller *
2327cd9ad58dSDavid S. Miller * (Time Out Period) * (Input Clock)
2328cd9ad58dSDavid S. Miller * STO = ----------------------------------
2329cd9ad58dSDavid S. Miller * (8192) * (Clock Conversion Factor)
2330cd9ad58dSDavid S. Miller *
2331cd9ad58dSDavid S. Miller * We use a time out period of 250ms (ESP_BUS_TIMEOUT).
2332cd9ad58dSDavid S. Miller *
2333cd9ad58dSDavid S. Miller * c) Imperical constants for synchronous offset and transfer period
2334cd9ad58dSDavid S. Miller * register values
2335cd9ad58dSDavid S. Miller *
2336cd9ad58dSDavid S. Miller * This entails the smallest and largest sync period we could ever
2337cd9ad58dSDavid S. Miller * handle on this ESP.
2338cd9ad58dSDavid S. Miller */
23396fe07aafSFinn Thain fhz = esp->cfreq;
2340cd9ad58dSDavid S. Miller
23416fe07aafSFinn Thain ccf = ((fhz / 1000000) + 4) / 5;
2342cd9ad58dSDavid S. Miller if (ccf == 1)
2343cd9ad58dSDavid S. Miller ccf = 2;
2344cd9ad58dSDavid S. Miller
2345cd9ad58dSDavid S. Miller /* If we can't find anything reasonable, just assume 20MHZ.
2346cd9ad58dSDavid S. Miller * This is the clock frequency of the older sun4c's where I've
2347cd9ad58dSDavid S. Miller * been unable to find the clock-frequency PROM property. All
2348cd9ad58dSDavid S. Miller * other machines provide useful values it seems.
2349cd9ad58dSDavid S. Miller */
23506fe07aafSFinn Thain if (fhz <= 5000000 || ccf < 1 || ccf > 8) {
23516fe07aafSFinn Thain fhz = 20000000;
2352cd9ad58dSDavid S. Miller ccf = 4;
2353cd9ad58dSDavid S. Miller }
2354cd9ad58dSDavid S. Miller
2355cd9ad58dSDavid S. Miller esp->cfact = (ccf == 8 ? 0 : ccf);
23566fe07aafSFinn Thain esp->cfreq = fhz;
23576fe07aafSFinn Thain esp->ccycle = ESP_HZ_TO_CYCLE(fhz);
2358cd9ad58dSDavid S. Miller esp->ctick = ESP_TICK(ccf, esp->ccycle);
23596fe07aafSFinn Thain esp->neg_defp = ESP_NEG_DEFP(fhz, ccf);
2360cd9ad58dSDavid S. Miller esp->sync_defp = SYNC_DEFP_SLOW;
2361cd9ad58dSDavid S. Miller }
2362cd9ad58dSDavid S. Miller
2363cd9ad58dSDavid S. Miller static const char *esp_chip_names[] = {
2364cd9ad58dSDavid S. Miller "ESP100",
2365cd9ad58dSDavid S. Miller "ESP100A",
2366cd9ad58dSDavid S. Miller "ESP236",
2367cd9ad58dSDavid S. Miller "FAS236",
23682086faaeSKars de Jong "AM53C974",
2369bd407261SKars de Jong "53CF9x-2",
2370cd9ad58dSDavid S. Miller "FAS100A",
2371cd9ad58dSDavid S. Miller "FAST",
2372cd9ad58dSDavid S. Miller "FASHME",
2373cd9ad58dSDavid S. Miller };
2374cd9ad58dSDavid S. Miller
2375cd9ad58dSDavid S. Miller static struct scsi_transport_template *esp_transport_template;
2376cd9ad58dSDavid S. Miller
scsi_esp_register(struct esp * esp)237744b1b4d2SChristoph Hellwig int scsi_esp_register(struct esp *esp)
2378cd9ad58dSDavid S. Miller {
2379cd9ad58dSDavid S. Miller static int instance;
2380cd9ad58dSDavid S. Miller int err;
2381cd9ad58dSDavid S. Miller
23823707a186SHannes Reinecke if (!esp->num_tags)
23833707a186SHannes Reinecke esp->num_tags = ESP_DEFAULT_TAGS;
2384cd9ad58dSDavid S. Miller esp->host->transportt = esp_transport_template;
2385cd9ad58dSDavid S. Miller esp->host->max_lun = ESP_MAX_LUN;
2386cd9ad58dSDavid S. Miller esp->host->cmd_per_lun = 2;
2387ff4abd6cSDavid Miller esp->host->unique_id = instance;
2388cd9ad58dSDavid S. Miller
2389cd9ad58dSDavid S. Miller esp_set_clock_params(esp);
2390cd9ad58dSDavid S. Miller
2391cd9ad58dSDavid S. Miller esp_get_revision(esp);
2392cd9ad58dSDavid S. Miller
2393cd9ad58dSDavid S. Miller esp_init_swstate(esp);
2394cd9ad58dSDavid S. Miller
2395cd9ad58dSDavid S. Miller esp_bootup_reset(esp);
2396cd9ad58dSDavid S. Miller
239744b1b4d2SChristoph Hellwig dev_printk(KERN_INFO, esp->dev, "esp%u: regs[%1p:%1p] irq[%u]\n",
2398cd9ad58dSDavid S. Miller esp->host->unique_id, esp->regs, esp->dma_regs,
2399cd9ad58dSDavid S. Miller esp->host->irq);
240044b1b4d2SChristoph Hellwig dev_printk(KERN_INFO, esp->dev,
2401a1a75b35SHannes Reinecke "esp%u: is a %s, %u MHz (ccf=%u), SCSI ID %u\n",
2402cd9ad58dSDavid S. Miller esp->host->unique_id, esp_chip_names[esp->rev],
2403cd9ad58dSDavid S. Miller esp->cfreq / 1000000, esp->cfact, esp->scsi_id);
2404cd9ad58dSDavid S. Miller
2405cd9ad58dSDavid S. Miller /* Let the SCSI bus reset settle. */
2406cd9ad58dSDavid S. Miller ssleep(esp_bus_reset_settle);
2407cd9ad58dSDavid S. Miller
240844b1b4d2SChristoph Hellwig err = scsi_add_host(esp->host, esp->dev);
2409cd9ad58dSDavid S. Miller if (err)
2410cd9ad58dSDavid S. Miller return err;
2411cd9ad58dSDavid S. Miller
2412ff4abd6cSDavid Miller instance++;
2413cd9ad58dSDavid S. Miller
2414cd9ad58dSDavid S. Miller scsi_scan_host(esp->host);
2415cd9ad58dSDavid S. Miller
2416cd9ad58dSDavid S. Miller return 0;
2417cd9ad58dSDavid S. Miller }
2418cd9ad58dSDavid S. Miller EXPORT_SYMBOL(scsi_esp_register);
2419cd9ad58dSDavid S. Miller
scsi_esp_unregister(struct esp * esp)242076246808SAdrian Bunk void scsi_esp_unregister(struct esp *esp)
2421cd9ad58dSDavid S. Miller {
2422cd9ad58dSDavid S. Miller scsi_remove_host(esp->host);
2423cd9ad58dSDavid S. Miller }
2424cd9ad58dSDavid S. Miller EXPORT_SYMBOL(scsi_esp_unregister);
2425cd9ad58dSDavid S. Miller
esp_target_alloc(struct scsi_target * starget)2426ec5e69f6SJames Bottomley static int esp_target_alloc(struct scsi_target *starget)
2427ec5e69f6SJames Bottomley {
2428ec5e69f6SJames Bottomley struct esp *esp = shost_priv(dev_to_shost(&starget->dev));
2429ec5e69f6SJames Bottomley struct esp_target_data *tp = &esp->target[starget->id];
2430ec5e69f6SJames Bottomley
2431ec5e69f6SJames Bottomley tp->starget = starget;
2432ec5e69f6SJames Bottomley
2433ec5e69f6SJames Bottomley return 0;
2434ec5e69f6SJames Bottomley }
2435ec5e69f6SJames Bottomley
esp_target_destroy(struct scsi_target * starget)2436ec5e69f6SJames Bottomley static void esp_target_destroy(struct scsi_target *starget)
2437ec5e69f6SJames Bottomley {
2438ec5e69f6SJames Bottomley struct esp *esp = shost_priv(dev_to_shost(&starget->dev));
2439ec5e69f6SJames Bottomley struct esp_target_data *tp = &esp->target[starget->id];
2440ec5e69f6SJames Bottomley
2441ec5e69f6SJames Bottomley tp->starget = NULL;
2442ec5e69f6SJames Bottomley }
2443ec5e69f6SJames Bottomley
esp_slave_alloc(struct scsi_device * dev)2444cd9ad58dSDavid S. Miller static int esp_slave_alloc(struct scsi_device *dev)
2445cd9ad58dSDavid S. Miller {
24462b14ec78SChristoph Hellwig struct esp *esp = shost_priv(dev->host);
2447cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[dev->id];
2448cd9ad58dSDavid S. Miller struct esp_lun_data *lp;
2449cd9ad58dSDavid S. Miller
2450cd9ad58dSDavid S. Miller lp = kzalloc(sizeof(*lp), GFP_KERNEL);
2451cd9ad58dSDavid S. Miller if (!lp)
2452cd9ad58dSDavid S. Miller return -ENOMEM;
2453cd9ad58dSDavid S. Miller dev->hostdata = lp;
2454cd9ad58dSDavid S. Miller
2455cd9ad58dSDavid S. Miller spi_min_period(tp->starget) = esp->min_period;
2456cd9ad58dSDavid S. Miller spi_max_offset(tp->starget) = 15;
2457cd9ad58dSDavid S. Miller
2458cd9ad58dSDavid S. Miller if (esp->flags & ESP_FLAG_WIDE_CAPABLE)
2459cd9ad58dSDavid S. Miller spi_max_width(tp->starget) = 1;
2460cd9ad58dSDavid S. Miller else
2461cd9ad58dSDavid S. Miller spi_max_width(tp->starget) = 0;
2462cd9ad58dSDavid S. Miller
2463cd9ad58dSDavid S. Miller return 0;
2464cd9ad58dSDavid S. Miller }
2465cd9ad58dSDavid S. Miller
esp_slave_configure(struct scsi_device * dev)2466cd9ad58dSDavid S. Miller static int esp_slave_configure(struct scsi_device *dev)
2467cd9ad58dSDavid S. Miller {
24682b14ec78SChristoph Hellwig struct esp *esp = shost_priv(dev->host);
2469cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[dev->id];
2470cd9ad58dSDavid S. Miller
24713707a186SHannes Reinecke if (dev->tagged_supported)
24723707a186SHannes Reinecke scsi_change_queue_depth(dev, esp->num_tags);
2473cd9ad58dSDavid S. Miller
2474cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_DISCONNECT;
2475cd9ad58dSDavid S. Miller
2476cd9ad58dSDavid S. Miller if (!spi_initial_dv(dev->sdev_target))
2477cd9ad58dSDavid S. Miller spi_dv_device(dev);
2478cd9ad58dSDavid S. Miller
2479cd9ad58dSDavid S. Miller return 0;
2480cd9ad58dSDavid S. Miller }
2481cd9ad58dSDavid S. Miller
esp_slave_destroy(struct scsi_device * dev)2482cd9ad58dSDavid S. Miller static void esp_slave_destroy(struct scsi_device *dev)
2483cd9ad58dSDavid S. Miller {
2484cd9ad58dSDavid S. Miller struct esp_lun_data *lp = dev->hostdata;
2485cd9ad58dSDavid S. Miller
2486cd9ad58dSDavid S. Miller kfree(lp);
2487cd9ad58dSDavid S. Miller dev->hostdata = NULL;
2488cd9ad58dSDavid S. Miller }
2489cd9ad58dSDavid S. Miller
esp_eh_abort_handler(struct scsi_cmnd * cmd)2490cd9ad58dSDavid S. Miller static int esp_eh_abort_handler(struct scsi_cmnd *cmd)
2491cd9ad58dSDavid S. Miller {
24922b14ec78SChristoph Hellwig struct esp *esp = shost_priv(cmd->device->host);
2493cd9ad58dSDavid S. Miller struct esp_cmd_entry *ent, *tmp;
2494cd9ad58dSDavid S. Miller struct completion eh_done;
2495cd9ad58dSDavid S. Miller unsigned long flags;
2496cd9ad58dSDavid S. Miller
2497cd9ad58dSDavid S. Miller /* XXX This helps a lot with debugging but might be a bit
2498cd9ad58dSDavid S. Miller * XXX much for the final driver.
2499cd9ad58dSDavid S. Miller */
2500cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2501a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host, "Aborting command [%p:%02x]\n",
2502a1a75b35SHannes Reinecke cmd, cmd->cmnd[0]);
2503cd9ad58dSDavid S. Miller ent = esp->active_cmd;
2504cd9ad58dSDavid S. Miller if (ent)
2505a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host,
2506a1a75b35SHannes Reinecke "Current command [%p:%02x]\n",
2507a1a75b35SHannes Reinecke ent->cmd, ent->cmd->cmnd[0]);
2508cd9ad58dSDavid S. Miller list_for_each_entry(ent, &esp->queued_cmds, list) {
2509a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host, "Queued command [%p:%02x]\n",
2510a1a75b35SHannes Reinecke ent->cmd, ent->cmd->cmnd[0]);
2511cd9ad58dSDavid S. Miller }
2512cd9ad58dSDavid S. Miller list_for_each_entry(ent, &esp->active_cmds, list) {
2513a1a75b35SHannes Reinecke shost_printk(KERN_ERR, esp->host, " Active command [%p:%02x]\n",
2514a1a75b35SHannes Reinecke ent->cmd, ent->cmd->cmnd[0]);
2515cd9ad58dSDavid S. Miller }
2516cd9ad58dSDavid S. Miller esp_dump_cmd_log(esp);
2517cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2518cd9ad58dSDavid S. Miller
2519cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2520cd9ad58dSDavid S. Miller
2521cd9ad58dSDavid S. Miller ent = NULL;
2522cd9ad58dSDavid S. Miller list_for_each_entry(tmp, &esp->queued_cmds, list) {
2523cd9ad58dSDavid S. Miller if (tmp->cmd == cmd) {
2524cd9ad58dSDavid S. Miller ent = tmp;
2525cd9ad58dSDavid S. Miller break;
2526cd9ad58dSDavid S. Miller }
2527cd9ad58dSDavid S. Miller }
2528cd9ad58dSDavid S. Miller
2529cd9ad58dSDavid S. Miller if (ent) {
2530cd9ad58dSDavid S. Miller /* Easiest case, we didn't even issue the command
2531cd9ad58dSDavid S. Miller * yet so it is trivial to abort.
2532cd9ad58dSDavid S. Miller */
2533cd9ad58dSDavid S. Miller list_del(&ent->list);
2534cd9ad58dSDavid S. Miller
2535cd9ad58dSDavid S. Miller cmd->result = DID_ABORT << 16;
2536f8ab27d9SBart Van Assche scsi_done(cmd);
2537cd9ad58dSDavid S. Miller
2538cd9ad58dSDavid S. Miller esp_put_ent(esp, ent);
2539cd9ad58dSDavid S. Miller
2540cd9ad58dSDavid S. Miller goto out_success;
2541cd9ad58dSDavid S. Miller }
2542cd9ad58dSDavid S. Miller
2543cd9ad58dSDavid S. Miller init_completion(&eh_done);
2544cd9ad58dSDavid S. Miller
2545cd9ad58dSDavid S. Miller ent = esp->active_cmd;
2546cd9ad58dSDavid S. Miller if (ent && ent->cmd == cmd) {
2547cd9ad58dSDavid S. Miller /* Command is the currently active command on
2548cd9ad58dSDavid S. Miller * the bus. If we already have an output message
2549cd9ad58dSDavid S. Miller * pending, no dice.
2550cd9ad58dSDavid S. Miller */
2551cd9ad58dSDavid S. Miller if (esp->msg_out_len)
2552cd9ad58dSDavid S. Miller goto out_failure;
2553cd9ad58dSDavid S. Miller
2554cd9ad58dSDavid S. Miller /* Send out an abort, encouraging the target to
2555cd9ad58dSDavid S. Miller * go to MSGOUT phase by asserting ATN.
2556cd9ad58dSDavid S. Miller */
2557cd9ad58dSDavid S. Miller esp->msg_out[0] = ABORT_TASK_SET;
2558cd9ad58dSDavid S. Miller esp->msg_out_len = 1;
2559cd9ad58dSDavid S. Miller ent->eh_done = &eh_done;
2560cd9ad58dSDavid S. Miller
2561cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_SATN);
2562cd9ad58dSDavid S. Miller } else {
2563cd9ad58dSDavid S. Miller /* The command is disconnected. This is not easy to
2564cd9ad58dSDavid S. Miller * abort. For now we fail and let the scsi error
2565cd9ad58dSDavid S. Miller * handling layer go try a scsi bus reset or host
2566cd9ad58dSDavid S. Miller * reset.
2567cd9ad58dSDavid S. Miller *
2568cd9ad58dSDavid S. Miller * What we could do is put together a scsi command
2569cd9ad58dSDavid S. Miller * solely for the purpose of sending an abort message
2570cd9ad58dSDavid S. Miller * to the target. Coming up with all the code to
2571cd9ad58dSDavid S. Miller * cook up scsi commands, special case them everywhere,
2572cd9ad58dSDavid S. Miller * etc. is for questionable gain and it would be better
2573cd9ad58dSDavid S. Miller * if the generic scsi error handling layer could do at
2574cd9ad58dSDavid S. Miller * least some of that for us.
2575cd9ad58dSDavid S. Miller *
2576cd9ad58dSDavid S. Miller * Anyways this is an area for potential future improvement
2577cd9ad58dSDavid S. Miller * in this driver.
2578cd9ad58dSDavid S. Miller */
2579cd9ad58dSDavid S. Miller goto out_failure;
2580cd9ad58dSDavid S. Miller }
2581cd9ad58dSDavid S. Miller
2582cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2583cd9ad58dSDavid S. Miller
2584cd9ad58dSDavid S. Miller if (!wait_for_completion_timeout(&eh_done, 5 * HZ)) {
2585cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2586cd9ad58dSDavid S. Miller ent->eh_done = NULL;
2587cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2588cd9ad58dSDavid S. Miller
2589cd9ad58dSDavid S. Miller return FAILED;
2590cd9ad58dSDavid S. Miller }
2591cd9ad58dSDavid S. Miller
2592cd9ad58dSDavid S. Miller return SUCCESS;
2593cd9ad58dSDavid S. Miller
2594cd9ad58dSDavid S. Miller out_success:
2595cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2596cd9ad58dSDavid S. Miller return SUCCESS;
2597cd9ad58dSDavid S. Miller
2598cd9ad58dSDavid S. Miller out_failure:
2599cd9ad58dSDavid S. Miller /* XXX This might be a good location to set ESP_TGT_BROKEN
2600cd9ad58dSDavid S. Miller * XXX since we know which target/lun in particular is
2601cd9ad58dSDavid S. Miller * XXX causing trouble.
2602cd9ad58dSDavid S. Miller */
2603cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2604cd9ad58dSDavid S. Miller return FAILED;
2605cd9ad58dSDavid S. Miller }
2606cd9ad58dSDavid S. Miller
esp_eh_bus_reset_handler(struct scsi_cmnd * cmd)2607cd9ad58dSDavid S. Miller static int esp_eh_bus_reset_handler(struct scsi_cmnd *cmd)
2608cd9ad58dSDavid S. Miller {
26092b14ec78SChristoph Hellwig struct esp *esp = shost_priv(cmd->device->host);
2610cd9ad58dSDavid S. Miller struct completion eh_reset;
2611cd9ad58dSDavid S. Miller unsigned long flags;
2612cd9ad58dSDavid S. Miller
2613cd9ad58dSDavid S. Miller init_completion(&eh_reset);
2614cd9ad58dSDavid S. Miller
2615cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2616cd9ad58dSDavid S. Miller
2617cd9ad58dSDavid S. Miller esp->eh_reset = &eh_reset;
2618cd9ad58dSDavid S. Miller
2619cd9ad58dSDavid S. Miller /* XXX This is too simple... We should add lots of
2620cd9ad58dSDavid S. Miller * XXX checks here so that if we find that the chip is
2621cd9ad58dSDavid S. Miller * XXX very wedged we return failure immediately so
2622cd9ad58dSDavid S. Miller * XXX that we can perform a full chip reset.
2623cd9ad58dSDavid S. Miller */
2624cd9ad58dSDavid S. Miller esp->flags |= ESP_FLAG_RESETTING;
2625cd9ad58dSDavid S. Miller scsi_esp_cmd(esp, ESP_CMD_RS);
2626cd9ad58dSDavid S. Miller
2627cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2628cd9ad58dSDavid S. Miller
2629cd9ad58dSDavid S. Miller ssleep(esp_bus_reset_settle);
2630cd9ad58dSDavid S. Miller
2631cd9ad58dSDavid S. Miller if (!wait_for_completion_timeout(&eh_reset, 5 * HZ)) {
2632cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2633cd9ad58dSDavid S. Miller esp->eh_reset = NULL;
2634cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2635cd9ad58dSDavid S. Miller
2636cd9ad58dSDavid S. Miller return FAILED;
2637cd9ad58dSDavid S. Miller }
2638cd9ad58dSDavid S. Miller
2639cd9ad58dSDavid S. Miller return SUCCESS;
2640cd9ad58dSDavid S. Miller }
2641cd9ad58dSDavid S. Miller
2642cd9ad58dSDavid S. Miller /* All bets are off, reset the entire device. */
esp_eh_host_reset_handler(struct scsi_cmnd * cmd)2643cd9ad58dSDavid S. Miller static int esp_eh_host_reset_handler(struct scsi_cmnd *cmd)
2644cd9ad58dSDavid S. Miller {
26452b14ec78SChristoph Hellwig struct esp *esp = shost_priv(cmd->device->host);
2646cd9ad58dSDavid S. Miller unsigned long flags;
2647cd9ad58dSDavid S. Miller
2648cd9ad58dSDavid S. Miller spin_lock_irqsave(esp->host->host_lock, flags);
2649cd9ad58dSDavid S. Miller esp_bootup_reset(esp);
2650cd9ad58dSDavid S. Miller esp_reset_cleanup(esp);
2651cd9ad58dSDavid S. Miller spin_unlock_irqrestore(esp->host->host_lock, flags);
2652cd9ad58dSDavid S. Miller
2653cd9ad58dSDavid S. Miller ssleep(esp_bus_reset_settle);
2654cd9ad58dSDavid S. Miller
2655cd9ad58dSDavid S. Miller return SUCCESS;
2656cd9ad58dSDavid S. Miller }
2657cd9ad58dSDavid S. Miller
esp_info(struct Scsi_Host * host)2658cd9ad58dSDavid S. Miller static const char *esp_info(struct Scsi_Host *host)
2659cd9ad58dSDavid S. Miller {
2660cd9ad58dSDavid S. Miller return "esp";
2661cd9ad58dSDavid S. Miller }
2662cd9ad58dSDavid S. Miller
2663*3b465a14SBart Van Assche const struct scsi_host_template scsi_esp_template = {
2664cd9ad58dSDavid S. Miller .module = THIS_MODULE,
2665cd9ad58dSDavid S. Miller .name = "esp",
2666cd9ad58dSDavid S. Miller .info = esp_info,
2667cd9ad58dSDavid S. Miller .queuecommand = esp_queuecommand,
2668ec5e69f6SJames Bottomley .target_alloc = esp_target_alloc,
2669ec5e69f6SJames Bottomley .target_destroy = esp_target_destroy,
2670cd9ad58dSDavid S. Miller .slave_alloc = esp_slave_alloc,
2671cd9ad58dSDavid S. Miller .slave_configure = esp_slave_configure,
2672cd9ad58dSDavid S. Miller .slave_destroy = esp_slave_destroy,
2673cd9ad58dSDavid S. Miller .eh_abort_handler = esp_eh_abort_handler,
2674cd9ad58dSDavid S. Miller .eh_bus_reset_handler = esp_eh_bus_reset_handler,
2675cd9ad58dSDavid S. Miller .eh_host_reset_handler = esp_eh_host_reset_handler,
2676cd9ad58dSDavid S. Miller .can_queue = 7,
2677cd9ad58dSDavid S. Miller .this_id = 7,
2678cd9ad58dSDavid S. Miller .sg_tablesize = SG_ALL,
2679cd9ad58dSDavid S. Miller .max_sectors = 0xffff,
2680cd9ad58dSDavid S. Miller .skip_settle_delay = 1,
2681dfae3987SBart Van Assche .cmd_size = sizeof(struct esp_cmd_priv),
2682cd9ad58dSDavid S. Miller };
2683cd9ad58dSDavid S. Miller EXPORT_SYMBOL(scsi_esp_template);
2684cd9ad58dSDavid S. Miller
esp_get_signalling(struct Scsi_Host * host)2685cd9ad58dSDavid S. Miller static void esp_get_signalling(struct Scsi_Host *host)
2686cd9ad58dSDavid S. Miller {
26872b14ec78SChristoph Hellwig struct esp *esp = shost_priv(host);
2688cd9ad58dSDavid S. Miller enum spi_signal_type type;
2689cd9ad58dSDavid S. Miller
2690cd9ad58dSDavid S. Miller if (esp->flags & ESP_FLAG_DIFFERENTIAL)
2691cd9ad58dSDavid S. Miller type = SPI_SIGNAL_HVD;
2692cd9ad58dSDavid S. Miller else
2693cd9ad58dSDavid S. Miller type = SPI_SIGNAL_SE;
2694cd9ad58dSDavid S. Miller
2695cd9ad58dSDavid S. Miller spi_signalling(host) = type;
2696cd9ad58dSDavid S. Miller }
2697cd9ad58dSDavid S. Miller
esp_set_offset(struct scsi_target * target,int offset)2698cd9ad58dSDavid S. Miller static void esp_set_offset(struct scsi_target *target, int offset)
2699cd9ad58dSDavid S. Miller {
2700cd9ad58dSDavid S. Miller struct Scsi_Host *host = dev_to_shost(target->dev.parent);
27012b14ec78SChristoph Hellwig struct esp *esp = shost_priv(host);
2702cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[target->id];
2703cd9ad58dSDavid S. Miller
270402507a80SFinn Thain if (esp->flags & ESP_FLAG_DISABLE_SYNC)
270502507a80SFinn Thain tp->nego_goal_offset = 0;
270602507a80SFinn Thain else
2707cd9ad58dSDavid S. Miller tp->nego_goal_offset = offset;
2708cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_CHECK_NEGO;
2709cd9ad58dSDavid S. Miller }
2710cd9ad58dSDavid S. Miller
esp_set_period(struct scsi_target * target,int period)2711cd9ad58dSDavid S. Miller static void esp_set_period(struct scsi_target *target, int period)
2712cd9ad58dSDavid S. Miller {
2713cd9ad58dSDavid S. Miller struct Scsi_Host *host = dev_to_shost(target->dev.parent);
27142b14ec78SChristoph Hellwig struct esp *esp = shost_priv(host);
2715cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[target->id];
2716cd9ad58dSDavid S. Miller
2717cd9ad58dSDavid S. Miller tp->nego_goal_period = period;
2718cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_CHECK_NEGO;
2719cd9ad58dSDavid S. Miller }
2720cd9ad58dSDavid S. Miller
esp_set_width(struct scsi_target * target,int width)2721cd9ad58dSDavid S. Miller static void esp_set_width(struct scsi_target *target, int width)
2722cd9ad58dSDavid S. Miller {
2723cd9ad58dSDavid S. Miller struct Scsi_Host *host = dev_to_shost(target->dev.parent);
27242b14ec78SChristoph Hellwig struct esp *esp = shost_priv(host);
2725cd9ad58dSDavid S. Miller struct esp_target_data *tp = &esp->target[target->id];
2726cd9ad58dSDavid S. Miller
2727cd9ad58dSDavid S. Miller tp->nego_goal_width = (width ? 1 : 0);
2728cd9ad58dSDavid S. Miller tp->flags |= ESP_TGT_CHECK_NEGO;
2729cd9ad58dSDavid S. Miller }
2730cd9ad58dSDavid S. Miller
2731cd9ad58dSDavid S. Miller static struct spi_function_template esp_transport_ops = {
2732cd9ad58dSDavid S. Miller .set_offset = esp_set_offset,
2733cd9ad58dSDavid S. Miller .show_offset = 1,
2734cd9ad58dSDavid S. Miller .set_period = esp_set_period,
2735cd9ad58dSDavid S. Miller .show_period = 1,
2736cd9ad58dSDavid S. Miller .set_width = esp_set_width,
2737cd9ad58dSDavid S. Miller .show_width = 1,
2738cd9ad58dSDavid S. Miller .get_signalling = esp_get_signalling,
2739cd9ad58dSDavid S. Miller };
2740cd9ad58dSDavid S. Miller
esp_init(void)2741cd9ad58dSDavid S. Miller static int __init esp_init(void)
2742cd9ad58dSDavid S. Miller {
2743cd9ad58dSDavid S. Miller esp_transport_template = spi_attach_transport(&esp_transport_ops);
2744cd9ad58dSDavid S. Miller if (!esp_transport_template)
2745cd9ad58dSDavid S. Miller return -ENODEV;
2746cd9ad58dSDavid S. Miller
2747cd9ad58dSDavid S. Miller return 0;
2748cd9ad58dSDavid S. Miller }
2749cd9ad58dSDavid S. Miller
esp_exit(void)2750cd9ad58dSDavid S. Miller static void __exit esp_exit(void)
2751cd9ad58dSDavid S. Miller {
2752cd9ad58dSDavid S. Miller spi_release_transport(esp_transport_template);
2753cd9ad58dSDavid S. Miller }
2754cd9ad58dSDavid S. Miller
2755cd9ad58dSDavid S. Miller MODULE_DESCRIPTION("ESP SCSI driver core");
2756cd9ad58dSDavid S. Miller MODULE_AUTHOR("David S. Miller (davem@davemloft.net)");
2757cd9ad58dSDavid S. Miller MODULE_LICENSE("GPL");
2758cd9ad58dSDavid S. Miller MODULE_VERSION(DRV_VERSION);
2759cd9ad58dSDavid S. Miller
2760cd9ad58dSDavid S. Miller module_param(esp_bus_reset_settle, int, 0);
2761cd9ad58dSDavid S. Miller MODULE_PARM_DESC(esp_bus_reset_settle,
2762cd9ad58dSDavid S. Miller "ESP scsi bus reset delay in seconds");
2763cd9ad58dSDavid S. Miller
2764cd9ad58dSDavid S. Miller module_param(esp_debug, int, 0);
2765cd9ad58dSDavid S. Miller MODULE_PARM_DESC(esp_debug,
2766cd9ad58dSDavid S. Miller "ESP bitmapped debugging message enable value:\n"
2767cd9ad58dSDavid S. Miller " 0x00000001 Log interrupt events\n"
2768cd9ad58dSDavid S. Miller " 0x00000002 Log scsi commands\n"
2769cd9ad58dSDavid S. Miller " 0x00000004 Log resets\n"
2770cd9ad58dSDavid S. Miller " 0x00000008 Log message in events\n"
2771cd9ad58dSDavid S. Miller " 0x00000010 Log message out events\n"
2772cd9ad58dSDavid S. Miller " 0x00000020 Log command completion\n"
2773cd9ad58dSDavid S. Miller " 0x00000040 Log disconnects\n"
2774cd9ad58dSDavid S. Miller " 0x00000080 Log data start\n"
2775cd9ad58dSDavid S. Miller " 0x00000100 Log data done\n"
2776cd9ad58dSDavid S. Miller " 0x00000200 Log reconnects\n"
2777cd9ad58dSDavid S. Miller " 0x00000400 Log auto-sense data\n"
2778cd9ad58dSDavid S. Miller );
2779cd9ad58dSDavid S. Miller
2780cd9ad58dSDavid S. Miller module_init(esp_init);
2781cd9ad58dSDavid S. Miller module_exit(esp_exit);
278253dce332SFinn Thain
278353dce332SFinn Thain #ifdef CONFIG_SCSI_ESP_PIO
esp_wait_for_fifo(struct esp * esp)278453dce332SFinn Thain static inline unsigned int esp_wait_for_fifo(struct esp *esp)
278553dce332SFinn Thain {
278653dce332SFinn Thain int i = 500000;
278753dce332SFinn Thain
278853dce332SFinn Thain do {
278953dce332SFinn Thain unsigned int fbytes = esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES;
279053dce332SFinn Thain
279153dce332SFinn Thain if (fbytes)
279253dce332SFinn Thain return fbytes;
279353dce332SFinn Thain
27948c6f803fSFinn Thain udelay(1);
279553dce332SFinn Thain } while (--i);
279653dce332SFinn Thain
279753dce332SFinn Thain shost_printk(KERN_ERR, esp->host, "FIFO is empty. sreg [%02x]\n",
279853dce332SFinn Thain esp_read8(ESP_STATUS));
279953dce332SFinn Thain return 0;
280053dce332SFinn Thain }
280153dce332SFinn Thain
esp_wait_for_intr(struct esp * esp)280253dce332SFinn Thain static inline int esp_wait_for_intr(struct esp *esp)
280353dce332SFinn Thain {
280453dce332SFinn Thain int i = 500000;
280553dce332SFinn Thain
280653dce332SFinn Thain do {
280753dce332SFinn Thain esp->sreg = esp_read8(ESP_STATUS);
280853dce332SFinn Thain if (esp->sreg & ESP_STAT_INTR)
280953dce332SFinn Thain return 0;
281053dce332SFinn Thain
28118c6f803fSFinn Thain udelay(1);
281253dce332SFinn Thain } while (--i);
281353dce332SFinn Thain
281453dce332SFinn Thain shost_printk(KERN_ERR, esp->host, "IRQ timeout. sreg [%02x]\n",
281553dce332SFinn Thain esp->sreg);
281653dce332SFinn Thain return 1;
281753dce332SFinn Thain }
281853dce332SFinn Thain
281953dce332SFinn Thain #define ESP_FIFO_SIZE 16
282053dce332SFinn Thain
esp_send_pio_cmd(struct esp * esp,u32 addr,u32 esp_count,u32 dma_count,int write,u8 cmd)282153dce332SFinn Thain void esp_send_pio_cmd(struct esp *esp, u32 addr, u32 esp_count,
282253dce332SFinn Thain u32 dma_count, int write, u8 cmd)
282353dce332SFinn Thain {
282453dce332SFinn Thain u8 phase = esp->sreg & ESP_STAT_PMASK;
282553dce332SFinn Thain
282653dce332SFinn Thain cmd &= ~ESP_CMD_DMA;
282753dce332SFinn Thain esp->send_cmd_error = 0;
282853dce332SFinn Thain
282953dce332SFinn Thain if (write) {
283053dce332SFinn Thain u8 *dst = (u8 *)addr;
283153dce332SFinn Thain u8 mask = ~(phase == ESP_MIP ? ESP_INTR_FDONE : ESP_INTR_BSERV);
283253dce332SFinn Thain
283353dce332SFinn Thain scsi_esp_cmd(esp, cmd);
283453dce332SFinn Thain
283553dce332SFinn Thain while (1) {
283653dce332SFinn Thain if (!esp_wait_for_fifo(esp))
283753dce332SFinn Thain break;
283853dce332SFinn Thain
28398c6f803fSFinn Thain *dst++ = readb(esp->fifo_reg);
284053dce332SFinn Thain --esp_count;
284153dce332SFinn Thain
284253dce332SFinn Thain if (!esp_count)
284353dce332SFinn Thain break;
284453dce332SFinn Thain
284553dce332SFinn Thain if (esp_wait_for_intr(esp)) {
284653dce332SFinn Thain esp->send_cmd_error = 1;
284753dce332SFinn Thain break;
284853dce332SFinn Thain }
284953dce332SFinn Thain
285053dce332SFinn Thain if ((esp->sreg & ESP_STAT_PMASK) != phase)
285153dce332SFinn Thain break;
285253dce332SFinn Thain
285353dce332SFinn Thain esp->ireg = esp_read8(ESP_INTRPT);
285453dce332SFinn Thain if (esp->ireg & mask) {
285553dce332SFinn Thain esp->send_cmd_error = 1;
285653dce332SFinn Thain break;
285753dce332SFinn Thain }
285853dce332SFinn Thain
285953dce332SFinn Thain if (phase == ESP_MIP)
28608c6f803fSFinn Thain esp_write8(ESP_CMD_MOK, ESP_CMD);
286153dce332SFinn Thain
28628c6f803fSFinn Thain esp_write8(ESP_CMD_TI, ESP_CMD);
286353dce332SFinn Thain }
286453dce332SFinn Thain } else {
286553dce332SFinn Thain unsigned int n = ESP_FIFO_SIZE;
286653dce332SFinn Thain u8 *src = (u8 *)addr;
286753dce332SFinn Thain
286853dce332SFinn Thain scsi_esp_cmd(esp, ESP_CMD_FLUSH);
286953dce332SFinn Thain
287053dce332SFinn Thain if (n > esp_count)
287153dce332SFinn Thain n = esp_count;
287253dce332SFinn Thain writesb(esp->fifo_reg, src, n);
287353dce332SFinn Thain src += n;
287453dce332SFinn Thain esp_count -= n;
287553dce332SFinn Thain
287653dce332SFinn Thain scsi_esp_cmd(esp, cmd);
287753dce332SFinn Thain
287853dce332SFinn Thain while (esp_count) {
287953dce332SFinn Thain if (esp_wait_for_intr(esp)) {
288053dce332SFinn Thain esp->send_cmd_error = 1;
288153dce332SFinn Thain break;
288253dce332SFinn Thain }
288353dce332SFinn Thain
288453dce332SFinn Thain if ((esp->sreg & ESP_STAT_PMASK) != phase)
288553dce332SFinn Thain break;
288653dce332SFinn Thain
288753dce332SFinn Thain esp->ireg = esp_read8(ESP_INTRPT);
288853dce332SFinn Thain if (esp->ireg & ~ESP_INTR_BSERV) {
288953dce332SFinn Thain esp->send_cmd_error = 1;
289053dce332SFinn Thain break;
289153dce332SFinn Thain }
289253dce332SFinn Thain
289353dce332SFinn Thain n = ESP_FIFO_SIZE -
289453dce332SFinn Thain (esp_read8(ESP_FFLAGS) & ESP_FF_FBYTES);
289553dce332SFinn Thain
289653dce332SFinn Thain if (n > esp_count)
289753dce332SFinn Thain n = esp_count;
289853dce332SFinn Thain writesb(esp->fifo_reg, src, n);
289953dce332SFinn Thain src += n;
290053dce332SFinn Thain esp_count -= n;
290153dce332SFinn Thain
29028c6f803fSFinn Thain esp_write8(ESP_CMD_TI, ESP_CMD);
290353dce332SFinn Thain }
290453dce332SFinn Thain }
290553dce332SFinn Thain
290653dce332SFinn Thain esp->send_cmd_residual = esp_count;
290753dce332SFinn Thain }
290853dce332SFinn Thain EXPORT_SYMBOL(esp_send_pio_cmd);
290953dce332SFinn Thain #endif
2910