1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * HighPoint RR3xxx/4xxx controller driver for Linux
4*4882a593Smuzhiyun * Copyright (C) 2006-2015 HighPoint Technologies, Inc. All Rights Reserved.
5*4882a593Smuzhiyun *
6*4882a593Smuzhiyun * Please report bugs/comments/suggestions to linux@highpoint-tech.com
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * For more information, visit http://www.highpoint-tech.com
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun #include <linux/module.h>
11*4882a593Smuzhiyun #include <linux/types.h>
12*4882a593Smuzhiyun #include <linux/string.h>
13*4882a593Smuzhiyun #include <linux/kernel.h>
14*4882a593Smuzhiyun #include <linux/pci.h>
15*4882a593Smuzhiyun #include <linux/interrupt.h>
16*4882a593Smuzhiyun #include <linux/errno.h>
17*4882a593Smuzhiyun #include <linux/delay.h>
18*4882a593Smuzhiyun #include <linux/timer.h>
19*4882a593Smuzhiyun #include <linux/spinlock.h>
20*4882a593Smuzhiyun #include <linux/gfp.h>
21*4882a593Smuzhiyun #include <linux/uaccess.h>
22*4882a593Smuzhiyun #include <asm/io.h>
23*4882a593Smuzhiyun #include <asm/div64.h>
24*4882a593Smuzhiyun #include <scsi/scsi_cmnd.h>
25*4882a593Smuzhiyun #include <scsi/scsi_device.h>
26*4882a593Smuzhiyun #include <scsi/scsi.h>
27*4882a593Smuzhiyun #include <scsi/scsi_tcq.h>
28*4882a593Smuzhiyun #include <scsi/scsi_host.h>
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun #include "hptiop.h"
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun MODULE_AUTHOR("HighPoint Technologies, Inc.");
33*4882a593Smuzhiyun MODULE_DESCRIPTION("HighPoint RocketRAID 3xxx/4xxx Controller Driver");
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun static char driver_name[] = "hptiop";
36*4882a593Smuzhiyun static const char driver_name_long[] = "RocketRAID 3xxx/4xxx Controller driver";
37*4882a593Smuzhiyun static const char driver_ver[] = "v1.10.0";
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun static int iop_send_sync_msg(struct hptiop_hba *hba, u32 msg, u32 millisec);
40*4882a593Smuzhiyun static void hptiop_finish_scsi_req(struct hptiop_hba *hba, u32 tag,
41*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req);
42*4882a593Smuzhiyun static void hptiop_host_request_callback_itl(struct hptiop_hba *hba, u32 tag);
43*4882a593Smuzhiyun static void hptiop_iop_request_callback_itl(struct hptiop_hba *hba, u32 tag);
44*4882a593Smuzhiyun static void hptiop_message_callback(struct hptiop_hba *hba, u32 msg);
45*4882a593Smuzhiyun
iop_wait_ready_itl(struct hptiop_hba * hba,u32 millisec)46*4882a593Smuzhiyun static int iop_wait_ready_itl(struct hptiop_hba *hba, u32 millisec)
47*4882a593Smuzhiyun {
48*4882a593Smuzhiyun u32 req = 0;
49*4882a593Smuzhiyun int i;
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun for (i = 0; i < millisec; i++) {
52*4882a593Smuzhiyun req = readl(&hba->u.itl.iop->inbound_queue);
53*4882a593Smuzhiyun if (req != IOPMU_QUEUE_EMPTY)
54*4882a593Smuzhiyun break;
55*4882a593Smuzhiyun msleep(1);
56*4882a593Smuzhiyun }
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun if (req != IOPMU_QUEUE_EMPTY) {
59*4882a593Smuzhiyun writel(req, &hba->u.itl.iop->outbound_queue);
60*4882a593Smuzhiyun readl(&hba->u.itl.iop->outbound_intstatus);
61*4882a593Smuzhiyun return 0;
62*4882a593Smuzhiyun }
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun return -1;
65*4882a593Smuzhiyun }
66*4882a593Smuzhiyun
iop_wait_ready_mv(struct hptiop_hba * hba,u32 millisec)67*4882a593Smuzhiyun static int iop_wait_ready_mv(struct hptiop_hba *hba, u32 millisec)
68*4882a593Smuzhiyun {
69*4882a593Smuzhiyun return iop_send_sync_msg(hba, IOPMU_INBOUND_MSG0_NOP, millisec);
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun
iop_wait_ready_mvfrey(struct hptiop_hba * hba,u32 millisec)72*4882a593Smuzhiyun static int iop_wait_ready_mvfrey(struct hptiop_hba *hba, u32 millisec)
73*4882a593Smuzhiyun {
74*4882a593Smuzhiyun return iop_send_sync_msg(hba, IOPMU_INBOUND_MSG0_NOP, millisec);
75*4882a593Smuzhiyun }
76*4882a593Smuzhiyun
hptiop_request_callback_itl(struct hptiop_hba * hba,u32 tag)77*4882a593Smuzhiyun static void hptiop_request_callback_itl(struct hptiop_hba *hba, u32 tag)
78*4882a593Smuzhiyun {
79*4882a593Smuzhiyun if (tag & IOPMU_QUEUE_ADDR_HOST_BIT)
80*4882a593Smuzhiyun hptiop_host_request_callback_itl(hba,
81*4882a593Smuzhiyun tag & ~IOPMU_QUEUE_ADDR_HOST_BIT);
82*4882a593Smuzhiyun else
83*4882a593Smuzhiyun hptiop_iop_request_callback_itl(hba, tag);
84*4882a593Smuzhiyun }
85*4882a593Smuzhiyun
hptiop_drain_outbound_queue_itl(struct hptiop_hba * hba)86*4882a593Smuzhiyun static void hptiop_drain_outbound_queue_itl(struct hptiop_hba *hba)
87*4882a593Smuzhiyun {
88*4882a593Smuzhiyun u32 req;
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun while ((req = readl(&hba->u.itl.iop->outbound_queue)) !=
91*4882a593Smuzhiyun IOPMU_QUEUE_EMPTY) {
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun if (req & IOPMU_QUEUE_MASK_HOST_BITS)
94*4882a593Smuzhiyun hptiop_request_callback_itl(hba, req);
95*4882a593Smuzhiyun else {
96*4882a593Smuzhiyun struct hpt_iop_request_header __iomem * p;
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun p = (struct hpt_iop_request_header __iomem *)
99*4882a593Smuzhiyun ((char __iomem *)hba->u.itl.iop + req);
100*4882a593Smuzhiyun
101*4882a593Smuzhiyun if (readl(&p->flags) & IOP_REQUEST_FLAG_SYNC_REQUEST) {
102*4882a593Smuzhiyun if (readl(&p->context))
103*4882a593Smuzhiyun hptiop_request_callback_itl(hba, req);
104*4882a593Smuzhiyun else
105*4882a593Smuzhiyun writel(1, &p->context);
106*4882a593Smuzhiyun }
107*4882a593Smuzhiyun else
108*4882a593Smuzhiyun hptiop_request_callback_itl(hba, req);
109*4882a593Smuzhiyun }
110*4882a593Smuzhiyun }
111*4882a593Smuzhiyun }
112*4882a593Smuzhiyun
iop_intr_itl(struct hptiop_hba * hba)113*4882a593Smuzhiyun static int iop_intr_itl(struct hptiop_hba *hba)
114*4882a593Smuzhiyun {
115*4882a593Smuzhiyun struct hpt_iopmu_itl __iomem *iop = hba->u.itl.iop;
116*4882a593Smuzhiyun void __iomem *plx = hba->u.itl.plx;
117*4882a593Smuzhiyun u32 status;
118*4882a593Smuzhiyun int ret = 0;
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun if (plx && readl(plx + 0x11C5C) & 0xf)
121*4882a593Smuzhiyun writel(1, plx + 0x11C60);
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun status = readl(&iop->outbound_intstatus);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun if (status & IOPMU_OUTBOUND_INT_MSG0) {
126*4882a593Smuzhiyun u32 msg = readl(&iop->outbound_msgaddr0);
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun dprintk("received outbound msg %x\n", msg);
129*4882a593Smuzhiyun writel(IOPMU_OUTBOUND_INT_MSG0, &iop->outbound_intstatus);
130*4882a593Smuzhiyun hptiop_message_callback(hba, msg);
131*4882a593Smuzhiyun ret = 1;
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun if (status & IOPMU_OUTBOUND_INT_POSTQUEUE) {
135*4882a593Smuzhiyun hptiop_drain_outbound_queue_itl(hba);
136*4882a593Smuzhiyun ret = 1;
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun return ret;
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun
mv_outbound_read(struct hpt_iopmu_mv __iomem * mu)142*4882a593Smuzhiyun static u64 mv_outbound_read(struct hpt_iopmu_mv __iomem *mu)
143*4882a593Smuzhiyun {
144*4882a593Smuzhiyun u32 outbound_tail = readl(&mu->outbound_tail);
145*4882a593Smuzhiyun u32 outbound_head = readl(&mu->outbound_head);
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun if (outbound_tail != outbound_head) {
148*4882a593Smuzhiyun u64 p;
149*4882a593Smuzhiyun
150*4882a593Smuzhiyun memcpy_fromio(&p, &mu->outbound_q[mu->outbound_tail], 8);
151*4882a593Smuzhiyun outbound_tail++;
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun if (outbound_tail == MVIOP_QUEUE_LEN)
154*4882a593Smuzhiyun outbound_tail = 0;
155*4882a593Smuzhiyun writel(outbound_tail, &mu->outbound_tail);
156*4882a593Smuzhiyun return p;
157*4882a593Smuzhiyun } else
158*4882a593Smuzhiyun return 0;
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun
mv_inbound_write(u64 p,struct hptiop_hba * hba)161*4882a593Smuzhiyun static void mv_inbound_write(u64 p, struct hptiop_hba *hba)
162*4882a593Smuzhiyun {
163*4882a593Smuzhiyun u32 inbound_head = readl(&hba->u.mv.mu->inbound_head);
164*4882a593Smuzhiyun u32 head = inbound_head + 1;
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun if (head == MVIOP_QUEUE_LEN)
167*4882a593Smuzhiyun head = 0;
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun memcpy_toio(&hba->u.mv.mu->inbound_q[inbound_head], &p, 8);
170*4882a593Smuzhiyun writel(head, &hba->u.mv.mu->inbound_head);
171*4882a593Smuzhiyun writel(MVIOP_MU_INBOUND_INT_POSTQUEUE,
172*4882a593Smuzhiyun &hba->u.mv.regs->inbound_doorbell);
173*4882a593Smuzhiyun }
174*4882a593Smuzhiyun
hptiop_request_callback_mv(struct hptiop_hba * hba,u64 tag)175*4882a593Smuzhiyun static void hptiop_request_callback_mv(struct hptiop_hba *hba, u64 tag)
176*4882a593Smuzhiyun {
177*4882a593Smuzhiyun u32 req_type = (tag >> 5) & 0x7;
178*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req;
179*4882a593Smuzhiyun
180*4882a593Smuzhiyun dprintk("hptiop_request_callback_mv: tag=%llx\n", tag);
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun BUG_ON((tag & MVIOP_MU_QUEUE_REQUEST_RETURN_CONTEXT) == 0);
183*4882a593Smuzhiyun
184*4882a593Smuzhiyun switch (req_type) {
185*4882a593Smuzhiyun case IOP_REQUEST_TYPE_GET_CONFIG:
186*4882a593Smuzhiyun case IOP_REQUEST_TYPE_SET_CONFIG:
187*4882a593Smuzhiyun hba->msg_done = 1;
188*4882a593Smuzhiyun break;
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun case IOP_REQUEST_TYPE_SCSI_COMMAND:
191*4882a593Smuzhiyun req = hba->reqs[tag >> 8].req_virt;
192*4882a593Smuzhiyun if (likely(tag & MVIOP_MU_QUEUE_REQUEST_RESULT_BIT))
193*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_SUCCESS);
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun hptiop_finish_scsi_req(hba, tag>>8, req);
196*4882a593Smuzhiyun break;
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun default:
199*4882a593Smuzhiyun break;
200*4882a593Smuzhiyun }
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun
iop_intr_mv(struct hptiop_hba * hba)203*4882a593Smuzhiyun static int iop_intr_mv(struct hptiop_hba *hba)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun u32 status;
206*4882a593Smuzhiyun int ret = 0;
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun status = readl(&hba->u.mv.regs->outbound_doorbell);
209*4882a593Smuzhiyun writel(~status, &hba->u.mv.regs->outbound_doorbell);
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun if (status & MVIOP_MU_OUTBOUND_INT_MSG) {
212*4882a593Smuzhiyun u32 msg;
213*4882a593Smuzhiyun msg = readl(&hba->u.mv.mu->outbound_msg);
214*4882a593Smuzhiyun dprintk("received outbound msg %x\n", msg);
215*4882a593Smuzhiyun hptiop_message_callback(hba, msg);
216*4882a593Smuzhiyun ret = 1;
217*4882a593Smuzhiyun }
218*4882a593Smuzhiyun
219*4882a593Smuzhiyun if (status & MVIOP_MU_OUTBOUND_INT_POSTQUEUE) {
220*4882a593Smuzhiyun u64 tag;
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun while ((tag = mv_outbound_read(hba->u.mv.mu)))
223*4882a593Smuzhiyun hptiop_request_callback_mv(hba, tag);
224*4882a593Smuzhiyun ret = 1;
225*4882a593Smuzhiyun }
226*4882a593Smuzhiyun
227*4882a593Smuzhiyun return ret;
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun
hptiop_request_callback_mvfrey(struct hptiop_hba * hba,u32 _tag)230*4882a593Smuzhiyun static void hptiop_request_callback_mvfrey(struct hptiop_hba *hba, u32 _tag)
231*4882a593Smuzhiyun {
232*4882a593Smuzhiyun u32 req_type = _tag & 0xf;
233*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req;
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun switch (req_type) {
236*4882a593Smuzhiyun case IOP_REQUEST_TYPE_GET_CONFIG:
237*4882a593Smuzhiyun case IOP_REQUEST_TYPE_SET_CONFIG:
238*4882a593Smuzhiyun hba->msg_done = 1;
239*4882a593Smuzhiyun break;
240*4882a593Smuzhiyun
241*4882a593Smuzhiyun case IOP_REQUEST_TYPE_SCSI_COMMAND:
242*4882a593Smuzhiyun req = hba->reqs[(_tag >> 4) & 0xff].req_virt;
243*4882a593Smuzhiyun if (likely(_tag & IOPMU_QUEUE_REQUEST_RESULT_BIT))
244*4882a593Smuzhiyun req->header.result = IOP_RESULT_SUCCESS;
245*4882a593Smuzhiyun hptiop_finish_scsi_req(hba, (_tag >> 4) & 0xff, req);
246*4882a593Smuzhiyun break;
247*4882a593Smuzhiyun
248*4882a593Smuzhiyun default:
249*4882a593Smuzhiyun break;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun }
252*4882a593Smuzhiyun
iop_intr_mvfrey(struct hptiop_hba * hba)253*4882a593Smuzhiyun static int iop_intr_mvfrey(struct hptiop_hba *hba)
254*4882a593Smuzhiyun {
255*4882a593Smuzhiyun u32 _tag, status, cptr, cur_rptr;
256*4882a593Smuzhiyun int ret = 0;
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun if (hba->initialized)
259*4882a593Smuzhiyun writel(0, &(hba->u.mvfrey.mu->pcie_f0_int_enable));
260*4882a593Smuzhiyun
261*4882a593Smuzhiyun status = readl(&(hba->u.mvfrey.mu->f0_doorbell));
262*4882a593Smuzhiyun if (status) {
263*4882a593Smuzhiyun writel(status, &(hba->u.mvfrey.mu->f0_doorbell));
264*4882a593Smuzhiyun if (status & CPU_TO_F0_DRBL_MSG_BIT) {
265*4882a593Smuzhiyun u32 msg = readl(&(hba->u.mvfrey.mu->cpu_to_f0_msg_a));
266*4882a593Smuzhiyun dprintk("received outbound msg %x\n", msg);
267*4882a593Smuzhiyun hptiop_message_callback(hba, msg);
268*4882a593Smuzhiyun }
269*4882a593Smuzhiyun ret = 1;
270*4882a593Smuzhiyun }
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun status = readl(&(hba->u.mvfrey.mu->isr_cause));
273*4882a593Smuzhiyun if (status) {
274*4882a593Smuzhiyun writel(status, &(hba->u.mvfrey.mu->isr_cause));
275*4882a593Smuzhiyun do {
276*4882a593Smuzhiyun cptr = *hba->u.mvfrey.outlist_cptr & 0xff;
277*4882a593Smuzhiyun cur_rptr = hba->u.mvfrey.outlist_rptr;
278*4882a593Smuzhiyun while (cur_rptr != cptr) {
279*4882a593Smuzhiyun cur_rptr++;
280*4882a593Smuzhiyun if (cur_rptr == hba->u.mvfrey.list_count)
281*4882a593Smuzhiyun cur_rptr = 0;
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun _tag = hba->u.mvfrey.outlist[cur_rptr].val;
284*4882a593Smuzhiyun BUG_ON(!(_tag & IOPMU_QUEUE_MASK_HOST_BITS));
285*4882a593Smuzhiyun hptiop_request_callback_mvfrey(hba, _tag);
286*4882a593Smuzhiyun ret = 1;
287*4882a593Smuzhiyun }
288*4882a593Smuzhiyun hba->u.mvfrey.outlist_rptr = cur_rptr;
289*4882a593Smuzhiyun } while (cptr != (*hba->u.mvfrey.outlist_cptr & 0xff));
290*4882a593Smuzhiyun }
291*4882a593Smuzhiyun
292*4882a593Smuzhiyun if (hba->initialized)
293*4882a593Smuzhiyun writel(0x1010, &(hba->u.mvfrey.mu->pcie_f0_int_enable));
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun return ret;
296*4882a593Smuzhiyun }
297*4882a593Smuzhiyun
iop_send_sync_request_itl(struct hptiop_hba * hba,void __iomem * _req,u32 millisec)298*4882a593Smuzhiyun static int iop_send_sync_request_itl(struct hptiop_hba *hba,
299*4882a593Smuzhiyun void __iomem *_req, u32 millisec)
300*4882a593Smuzhiyun {
301*4882a593Smuzhiyun struct hpt_iop_request_header __iomem *req = _req;
302*4882a593Smuzhiyun u32 i;
303*4882a593Smuzhiyun
304*4882a593Smuzhiyun writel(readl(&req->flags) | IOP_REQUEST_FLAG_SYNC_REQUEST, &req->flags);
305*4882a593Smuzhiyun writel(0, &req->context);
306*4882a593Smuzhiyun writel((unsigned long)req - (unsigned long)hba->u.itl.iop,
307*4882a593Smuzhiyun &hba->u.itl.iop->inbound_queue);
308*4882a593Smuzhiyun readl(&hba->u.itl.iop->outbound_intstatus);
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun for (i = 0; i < millisec; i++) {
311*4882a593Smuzhiyun iop_intr_itl(hba);
312*4882a593Smuzhiyun if (readl(&req->context))
313*4882a593Smuzhiyun return 0;
314*4882a593Smuzhiyun msleep(1);
315*4882a593Smuzhiyun }
316*4882a593Smuzhiyun
317*4882a593Smuzhiyun return -1;
318*4882a593Smuzhiyun }
319*4882a593Smuzhiyun
iop_send_sync_request_mv(struct hptiop_hba * hba,u32 size_bits,u32 millisec)320*4882a593Smuzhiyun static int iop_send_sync_request_mv(struct hptiop_hba *hba,
321*4882a593Smuzhiyun u32 size_bits, u32 millisec)
322*4882a593Smuzhiyun {
323*4882a593Smuzhiyun struct hpt_iop_request_header *reqhdr = hba->u.mv.internal_req;
324*4882a593Smuzhiyun u32 i;
325*4882a593Smuzhiyun
326*4882a593Smuzhiyun hba->msg_done = 0;
327*4882a593Smuzhiyun reqhdr->flags |= cpu_to_le32(IOP_REQUEST_FLAG_SYNC_REQUEST);
328*4882a593Smuzhiyun mv_inbound_write(hba->u.mv.internal_req_phy |
329*4882a593Smuzhiyun MVIOP_MU_QUEUE_ADDR_HOST_BIT | size_bits, hba);
330*4882a593Smuzhiyun
331*4882a593Smuzhiyun for (i = 0; i < millisec; i++) {
332*4882a593Smuzhiyun iop_intr_mv(hba);
333*4882a593Smuzhiyun if (hba->msg_done)
334*4882a593Smuzhiyun return 0;
335*4882a593Smuzhiyun msleep(1);
336*4882a593Smuzhiyun }
337*4882a593Smuzhiyun return -1;
338*4882a593Smuzhiyun }
339*4882a593Smuzhiyun
iop_send_sync_request_mvfrey(struct hptiop_hba * hba,u32 size_bits,u32 millisec)340*4882a593Smuzhiyun static int iop_send_sync_request_mvfrey(struct hptiop_hba *hba,
341*4882a593Smuzhiyun u32 size_bits, u32 millisec)
342*4882a593Smuzhiyun {
343*4882a593Smuzhiyun struct hpt_iop_request_header *reqhdr =
344*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_virt;
345*4882a593Smuzhiyun u32 i;
346*4882a593Smuzhiyun
347*4882a593Smuzhiyun hba->msg_done = 0;
348*4882a593Smuzhiyun reqhdr->flags |= cpu_to_le32(IOP_REQUEST_FLAG_SYNC_REQUEST);
349*4882a593Smuzhiyun hba->ops->post_req(hba, &(hba->u.mvfrey.internal_req));
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun for (i = 0; i < millisec; i++) {
352*4882a593Smuzhiyun iop_intr_mvfrey(hba);
353*4882a593Smuzhiyun if (hba->msg_done)
354*4882a593Smuzhiyun break;
355*4882a593Smuzhiyun msleep(1);
356*4882a593Smuzhiyun }
357*4882a593Smuzhiyun return hba->msg_done ? 0 : -1;
358*4882a593Smuzhiyun }
359*4882a593Smuzhiyun
hptiop_post_msg_itl(struct hptiop_hba * hba,u32 msg)360*4882a593Smuzhiyun static void hptiop_post_msg_itl(struct hptiop_hba *hba, u32 msg)
361*4882a593Smuzhiyun {
362*4882a593Smuzhiyun writel(msg, &hba->u.itl.iop->inbound_msgaddr0);
363*4882a593Smuzhiyun readl(&hba->u.itl.iop->outbound_intstatus);
364*4882a593Smuzhiyun }
365*4882a593Smuzhiyun
hptiop_post_msg_mv(struct hptiop_hba * hba,u32 msg)366*4882a593Smuzhiyun static void hptiop_post_msg_mv(struct hptiop_hba *hba, u32 msg)
367*4882a593Smuzhiyun {
368*4882a593Smuzhiyun writel(msg, &hba->u.mv.mu->inbound_msg);
369*4882a593Smuzhiyun writel(MVIOP_MU_INBOUND_INT_MSG, &hba->u.mv.regs->inbound_doorbell);
370*4882a593Smuzhiyun readl(&hba->u.mv.regs->inbound_doorbell);
371*4882a593Smuzhiyun }
372*4882a593Smuzhiyun
hptiop_post_msg_mvfrey(struct hptiop_hba * hba,u32 msg)373*4882a593Smuzhiyun static void hptiop_post_msg_mvfrey(struct hptiop_hba *hba, u32 msg)
374*4882a593Smuzhiyun {
375*4882a593Smuzhiyun writel(msg, &(hba->u.mvfrey.mu->f0_to_cpu_msg_a));
376*4882a593Smuzhiyun readl(&(hba->u.mvfrey.mu->f0_to_cpu_msg_a));
377*4882a593Smuzhiyun }
378*4882a593Smuzhiyun
iop_send_sync_msg(struct hptiop_hba * hba,u32 msg,u32 millisec)379*4882a593Smuzhiyun static int iop_send_sync_msg(struct hptiop_hba *hba, u32 msg, u32 millisec)
380*4882a593Smuzhiyun {
381*4882a593Smuzhiyun u32 i;
382*4882a593Smuzhiyun
383*4882a593Smuzhiyun hba->msg_done = 0;
384*4882a593Smuzhiyun hba->ops->disable_intr(hba);
385*4882a593Smuzhiyun hba->ops->post_msg(hba, msg);
386*4882a593Smuzhiyun
387*4882a593Smuzhiyun for (i = 0; i < millisec; i++) {
388*4882a593Smuzhiyun spin_lock_irq(hba->host->host_lock);
389*4882a593Smuzhiyun hba->ops->iop_intr(hba);
390*4882a593Smuzhiyun spin_unlock_irq(hba->host->host_lock);
391*4882a593Smuzhiyun if (hba->msg_done)
392*4882a593Smuzhiyun break;
393*4882a593Smuzhiyun msleep(1);
394*4882a593Smuzhiyun }
395*4882a593Smuzhiyun
396*4882a593Smuzhiyun hba->ops->enable_intr(hba);
397*4882a593Smuzhiyun return hba->msg_done? 0 : -1;
398*4882a593Smuzhiyun }
399*4882a593Smuzhiyun
iop_get_config_itl(struct hptiop_hba * hba,struct hpt_iop_request_get_config * config)400*4882a593Smuzhiyun static int iop_get_config_itl(struct hptiop_hba *hba,
401*4882a593Smuzhiyun struct hpt_iop_request_get_config *config)
402*4882a593Smuzhiyun {
403*4882a593Smuzhiyun u32 req32;
404*4882a593Smuzhiyun struct hpt_iop_request_get_config __iomem *req;
405*4882a593Smuzhiyun
406*4882a593Smuzhiyun req32 = readl(&hba->u.itl.iop->inbound_queue);
407*4882a593Smuzhiyun if (req32 == IOPMU_QUEUE_EMPTY)
408*4882a593Smuzhiyun return -1;
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun req = (struct hpt_iop_request_get_config __iomem *)
411*4882a593Smuzhiyun ((unsigned long)hba->u.itl.iop + req32);
412*4882a593Smuzhiyun
413*4882a593Smuzhiyun writel(0, &req->header.flags);
414*4882a593Smuzhiyun writel(IOP_REQUEST_TYPE_GET_CONFIG, &req->header.type);
415*4882a593Smuzhiyun writel(sizeof(struct hpt_iop_request_get_config), &req->header.size);
416*4882a593Smuzhiyun writel(IOP_RESULT_PENDING, &req->header.result);
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun if (iop_send_sync_request_itl(hba, req, 20000)) {
419*4882a593Smuzhiyun dprintk("Get config send cmd failed\n");
420*4882a593Smuzhiyun return -1;
421*4882a593Smuzhiyun }
422*4882a593Smuzhiyun
423*4882a593Smuzhiyun memcpy_fromio(config, req, sizeof(*config));
424*4882a593Smuzhiyun writel(req32, &hba->u.itl.iop->outbound_queue);
425*4882a593Smuzhiyun return 0;
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun
iop_get_config_mv(struct hptiop_hba * hba,struct hpt_iop_request_get_config * config)428*4882a593Smuzhiyun static int iop_get_config_mv(struct hptiop_hba *hba,
429*4882a593Smuzhiyun struct hpt_iop_request_get_config *config)
430*4882a593Smuzhiyun {
431*4882a593Smuzhiyun struct hpt_iop_request_get_config *req = hba->u.mv.internal_req;
432*4882a593Smuzhiyun
433*4882a593Smuzhiyun req->header.flags = cpu_to_le32(IOP_REQUEST_FLAG_OUTPUT_CONTEXT);
434*4882a593Smuzhiyun req->header.type = cpu_to_le32(IOP_REQUEST_TYPE_GET_CONFIG);
435*4882a593Smuzhiyun req->header.size =
436*4882a593Smuzhiyun cpu_to_le32(sizeof(struct hpt_iop_request_get_config));
437*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_PENDING);
438*4882a593Smuzhiyun req->header.context = cpu_to_le32(IOP_REQUEST_TYPE_GET_CONFIG<<5);
439*4882a593Smuzhiyun req->header.context_hi32 = 0;
440*4882a593Smuzhiyun
441*4882a593Smuzhiyun if (iop_send_sync_request_mv(hba, 0, 20000)) {
442*4882a593Smuzhiyun dprintk("Get config send cmd failed\n");
443*4882a593Smuzhiyun return -1;
444*4882a593Smuzhiyun }
445*4882a593Smuzhiyun
446*4882a593Smuzhiyun memcpy(config, req, sizeof(struct hpt_iop_request_get_config));
447*4882a593Smuzhiyun return 0;
448*4882a593Smuzhiyun }
449*4882a593Smuzhiyun
iop_get_config_mvfrey(struct hptiop_hba * hba,struct hpt_iop_request_get_config * config)450*4882a593Smuzhiyun static int iop_get_config_mvfrey(struct hptiop_hba *hba,
451*4882a593Smuzhiyun struct hpt_iop_request_get_config *config)
452*4882a593Smuzhiyun {
453*4882a593Smuzhiyun struct hpt_iop_request_get_config *info = hba->u.mvfrey.config;
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun if (info->header.size != sizeof(struct hpt_iop_request_get_config) ||
456*4882a593Smuzhiyun info->header.type != IOP_REQUEST_TYPE_GET_CONFIG)
457*4882a593Smuzhiyun return -1;
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun config->interface_version = info->interface_version;
460*4882a593Smuzhiyun config->firmware_version = info->firmware_version;
461*4882a593Smuzhiyun config->max_requests = info->max_requests;
462*4882a593Smuzhiyun config->request_size = info->request_size;
463*4882a593Smuzhiyun config->max_sg_count = info->max_sg_count;
464*4882a593Smuzhiyun config->data_transfer_length = info->data_transfer_length;
465*4882a593Smuzhiyun config->alignment_mask = info->alignment_mask;
466*4882a593Smuzhiyun config->max_devices = info->max_devices;
467*4882a593Smuzhiyun config->sdram_size = info->sdram_size;
468*4882a593Smuzhiyun
469*4882a593Smuzhiyun return 0;
470*4882a593Smuzhiyun }
471*4882a593Smuzhiyun
iop_set_config_itl(struct hptiop_hba * hba,struct hpt_iop_request_set_config * config)472*4882a593Smuzhiyun static int iop_set_config_itl(struct hptiop_hba *hba,
473*4882a593Smuzhiyun struct hpt_iop_request_set_config *config)
474*4882a593Smuzhiyun {
475*4882a593Smuzhiyun u32 req32;
476*4882a593Smuzhiyun struct hpt_iop_request_set_config __iomem *req;
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun req32 = readl(&hba->u.itl.iop->inbound_queue);
479*4882a593Smuzhiyun if (req32 == IOPMU_QUEUE_EMPTY)
480*4882a593Smuzhiyun return -1;
481*4882a593Smuzhiyun
482*4882a593Smuzhiyun req = (struct hpt_iop_request_set_config __iomem *)
483*4882a593Smuzhiyun ((unsigned long)hba->u.itl.iop + req32);
484*4882a593Smuzhiyun
485*4882a593Smuzhiyun memcpy_toio((u8 __iomem *)req + sizeof(struct hpt_iop_request_header),
486*4882a593Smuzhiyun (u8 *)config + sizeof(struct hpt_iop_request_header),
487*4882a593Smuzhiyun sizeof(struct hpt_iop_request_set_config) -
488*4882a593Smuzhiyun sizeof(struct hpt_iop_request_header));
489*4882a593Smuzhiyun
490*4882a593Smuzhiyun writel(0, &req->header.flags);
491*4882a593Smuzhiyun writel(IOP_REQUEST_TYPE_SET_CONFIG, &req->header.type);
492*4882a593Smuzhiyun writel(sizeof(struct hpt_iop_request_set_config), &req->header.size);
493*4882a593Smuzhiyun writel(IOP_RESULT_PENDING, &req->header.result);
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun if (iop_send_sync_request_itl(hba, req, 20000)) {
496*4882a593Smuzhiyun dprintk("Set config send cmd failed\n");
497*4882a593Smuzhiyun return -1;
498*4882a593Smuzhiyun }
499*4882a593Smuzhiyun
500*4882a593Smuzhiyun writel(req32, &hba->u.itl.iop->outbound_queue);
501*4882a593Smuzhiyun return 0;
502*4882a593Smuzhiyun }
503*4882a593Smuzhiyun
iop_set_config_mv(struct hptiop_hba * hba,struct hpt_iop_request_set_config * config)504*4882a593Smuzhiyun static int iop_set_config_mv(struct hptiop_hba *hba,
505*4882a593Smuzhiyun struct hpt_iop_request_set_config *config)
506*4882a593Smuzhiyun {
507*4882a593Smuzhiyun struct hpt_iop_request_set_config *req = hba->u.mv.internal_req;
508*4882a593Smuzhiyun
509*4882a593Smuzhiyun memcpy(req, config, sizeof(struct hpt_iop_request_set_config));
510*4882a593Smuzhiyun req->header.flags = cpu_to_le32(IOP_REQUEST_FLAG_OUTPUT_CONTEXT);
511*4882a593Smuzhiyun req->header.type = cpu_to_le32(IOP_REQUEST_TYPE_SET_CONFIG);
512*4882a593Smuzhiyun req->header.size =
513*4882a593Smuzhiyun cpu_to_le32(sizeof(struct hpt_iop_request_set_config));
514*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_PENDING);
515*4882a593Smuzhiyun req->header.context = cpu_to_le32(IOP_REQUEST_TYPE_SET_CONFIG<<5);
516*4882a593Smuzhiyun req->header.context_hi32 = 0;
517*4882a593Smuzhiyun
518*4882a593Smuzhiyun if (iop_send_sync_request_mv(hba, 0, 20000)) {
519*4882a593Smuzhiyun dprintk("Set config send cmd failed\n");
520*4882a593Smuzhiyun return -1;
521*4882a593Smuzhiyun }
522*4882a593Smuzhiyun
523*4882a593Smuzhiyun return 0;
524*4882a593Smuzhiyun }
525*4882a593Smuzhiyun
iop_set_config_mvfrey(struct hptiop_hba * hba,struct hpt_iop_request_set_config * config)526*4882a593Smuzhiyun static int iop_set_config_mvfrey(struct hptiop_hba *hba,
527*4882a593Smuzhiyun struct hpt_iop_request_set_config *config)
528*4882a593Smuzhiyun {
529*4882a593Smuzhiyun struct hpt_iop_request_set_config *req =
530*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_virt;
531*4882a593Smuzhiyun
532*4882a593Smuzhiyun memcpy(req, config, sizeof(struct hpt_iop_request_set_config));
533*4882a593Smuzhiyun req->header.flags = cpu_to_le32(IOP_REQUEST_FLAG_OUTPUT_CONTEXT);
534*4882a593Smuzhiyun req->header.type = cpu_to_le32(IOP_REQUEST_TYPE_SET_CONFIG);
535*4882a593Smuzhiyun req->header.size =
536*4882a593Smuzhiyun cpu_to_le32(sizeof(struct hpt_iop_request_set_config));
537*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_PENDING);
538*4882a593Smuzhiyun req->header.context = cpu_to_le32(IOP_REQUEST_TYPE_SET_CONFIG<<5);
539*4882a593Smuzhiyun req->header.context_hi32 = 0;
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun if (iop_send_sync_request_mvfrey(hba, 0, 20000)) {
542*4882a593Smuzhiyun dprintk("Set config send cmd failed\n");
543*4882a593Smuzhiyun return -1;
544*4882a593Smuzhiyun }
545*4882a593Smuzhiyun
546*4882a593Smuzhiyun return 0;
547*4882a593Smuzhiyun }
548*4882a593Smuzhiyun
hptiop_enable_intr_itl(struct hptiop_hba * hba)549*4882a593Smuzhiyun static void hptiop_enable_intr_itl(struct hptiop_hba *hba)
550*4882a593Smuzhiyun {
551*4882a593Smuzhiyun writel(~(IOPMU_OUTBOUND_INT_POSTQUEUE | IOPMU_OUTBOUND_INT_MSG0),
552*4882a593Smuzhiyun &hba->u.itl.iop->outbound_intmask);
553*4882a593Smuzhiyun }
554*4882a593Smuzhiyun
hptiop_enable_intr_mv(struct hptiop_hba * hba)555*4882a593Smuzhiyun static void hptiop_enable_intr_mv(struct hptiop_hba *hba)
556*4882a593Smuzhiyun {
557*4882a593Smuzhiyun writel(MVIOP_MU_OUTBOUND_INT_POSTQUEUE | MVIOP_MU_OUTBOUND_INT_MSG,
558*4882a593Smuzhiyun &hba->u.mv.regs->outbound_intmask);
559*4882a593Smuzhiyun }
560*4882a593Smuzhiyun
hptiop_enable_intr_mvfrey(struct hptiop_hba * hba)561*4882a593Smuzhiyun static void hptiop_enable_intr_mvfrey(struct hptiop_hba *hba)
562*4882a593Smuzhiyun {
563*4882a593Smuzhiyun writel(CPU_TO_F0_DRBL_MSG_BIT, &(hba->u.mvfrey.mu->f0_doorbell_enable));
564*4882a593Smuzhiyun writel(0x1, &(hba->u.mvfrey.mu->isr_enable));
565*4882a593Smuzhiyun writel(0x1010, &(hba->u.mvfrey.mu->pcie_f0_int_enable));
566*4882a593Smuzhiyun }
567*4882a593Smuzhiyun
hptiop_initialize_iop(struct hptiop_hba * hba)568*4882a593Smuzhiyun static int hptiop_initialize_iop(struct hptiop_hba *hba)
569*4882a593Smuzhiyun {
570*4882a593Smuzhiyun /* enable interrupts */
571*4882a593Smuzhiyun hba->ops->enable_intr(hba);
572*4882a593Smuzhiyun
573*4882a593Smuzhiyun hba->initialized = 1;
574*4882a593Smuzhiyun
575*4882a593Smuzhiyun /* start background tasks */
576*4882a593Smuzhiyun if (iop_send_sync_msg(hba,
577*4882a593Smuzhiyun IOPMU_INBOUND_MSG0_START_BACKGROUND_TASK, 5000)) {
578*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: fail to start background task\n",
579*4882a593Smuzhiyun hba->host->host_no);
580*4882a593Smuzhiyun return -1;
581*4882a593Smuzhiyun }
582*4882a593Smuzhiyun return 0;
583*4882a593Smuzhiyun }
584*4882a593Smuzhiyun
hptiop_map_pci_bar(struct hptiop_hba * hba,int index)585*4882a593Smuzhiyun static void __iomem *hptiop_map_pci_bar(struct hptiop_hba *hba, int index)
586*4882a593Smuzhiyun {
587*4882a593Smuzhiyun u32 mem_base_phy, length;
588*4882a593Smuzhiyun void __iomem *mem_base_virt;
589*4882a593Smuzhiyun
590*4882a593Smuzhiyun struct pci_dev *pcidev = hba->pcidev;
591*4882a593Smuzhiyun
592*4882a593Smuzhiyun
593*4882a593Smuzhiyun if (!(pci_resource_flags(pcidev, index) & IORESOURCE_MEM)) {
594*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: pci resource invalid\n",
595*4882a593Smuzhiyun hba->host->host_no);
596*4882a593Smuzhiyun return NULL;
597*4882a593Smuzhiyun }
598*4882a593Smuzhiyun
599*4882a593Smuzhiyun mem_base_phy = pci_resource_start(pcidev, index);
600*4882a593Smuzhiyun length = pci_resource_len(pcidev, index);
601*4882a593Smuzhiyun mem_base_virt = ioremap(mem_base_phy, length);
602*4882a593Smuzhiyun
603*4882a593Smuzhiyun if (!mem_base_virt) {
604*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: Fail to ioremap memory space\n",
605*4882a593Smuzhiyun hba->host->host_no);
606*4882a593Smuzhiyun return NULL;
607*4882a593Smuzhiyun }
608*4882a593Smuzhiyun return mem_base_virt;
609*4882a593Smuzhiyun }
610*4882a593Smuzhiyun
hptiop_map_pci_bar_itl(struct hptiop_hba * hba)611*4882a593Smuzhiyun static int hptiop_map_pci_bar_itl(struct hptiop_hba *hba)
612*4882a593Smuzhiyun {
613*4882a593Smuzhiyun struct pci_dev *pcidev = hba->pcidev;
614*4882a593Smuzhiyun hba->u.itl.iop = hptiop_map_pci_bar(hba, 0);
615*4882a593Smuzhiyun if (hba->u.itl.iop == NULL)
616*4882a593Smuzhiyun return -1;
617*4882a593Smuzhiyun if ((pcidev->device & 0xff00) == 0x4400) {
618*4882a593Smuzhiyun hba->u.itl.plx = hba->u.itl.iop;
619*4882a593Smuzhiyun hba->u.itl.iop = hptiop_map_pci_bar(hba, 2);
620*4882a593Smuzhiyun if (hba->u.itl.iop == NULL) {
621*4882a593Smuzhiyun iounmap(hba->u.itl.plx);
622*4882a593Smuzhiyun return -1;
623*4882a593Smuzhiyun }
624*4882a593Smuzhiyun }
625*4882a593Smuzhiyun return 0;
626*4882a593Smuzhiyun }
627*4882a593Smuzhiyun
hptiop_unmap_pci_bar_itl(struct hptiop_hba * hba)628*4882a593Smuzhiyun static void hptiop_unmap_pci_bar_itl(struct hptiop_hba *hba)
629*4882a593Smuzhiyun {
630*4882a593Smuzhiyun if (hba->u.itl.plx)
631*4882a593Smuzhiyun iounmap(hba->u.itl.plx);
632*4882a593Smuzhiyun iounmap(hba->u.itl.iop);
633*4882a593Smuzhiyun }
634*4882a593Smuzhiyun
hptiop_map_pci_bar_mv(struct hptiop_hba * hba)635*4882a593Smuzhiyun static int hptiop_map_pci_bar_mv(struct hptiop_hba *hba)
636*4882a593Smuzhiyun {
637*4882a593Smuzhiyun hba->u.mv.regs = hptiop_map_pci_bar(hba, 0);
638*4882a593Smuzhiyun if (hba->u.mv.regs == NULL)
639*4882a593Smuzhiyun return -1;
640*4882a593Smuzhiyun
641*4882a593Smuzhiyun hba->u.mv.mu = hptiop_map_pci_bar(hba, 2);
642*4882a593Smuzhiyun if (hba->u.mv.mu == NULL) {
643*4882a593Smuzhiyun iounmap(hba->u.mv.regs);
644*4882a593Smuzhiyun return -1;
645*4882a593Smuzhiyun }
646*4882a593Smuzhiyun
647*4882a593Smuzhiyun return 0;
648*4882a593Smuzhiyun }
649*4882a593Smuzhiyun
hptiop_map_pci_bar_mvfrey(struct hptiop_hba * hba)650*4882a593Smuzhiyun static int hptiop_map_pci_bar_mvfrey(struct hptiop_hba *hba)
651*4882a593Smuzhiyun {
652*4882a593Smuzhiyun hba->u.mvfrey.config = hptiop_map_pci_bar(hba, 0);
653*4882a593Smuzhiyun if (hba->u.mvfrey.config == NULL)
654*4882a593Smuzhiyun return -1;
655*4882a593Smuzhiyun
656*4882a593Smuzhiyun hba->u.mvfrey.mu = hptiop_map_pci_bar(hba, 2);
657*4882a593Smuzhiyun if (hba->u.mvfrey.mu == NULL) {
658*4882a593Smuzhiyun iounmap(hba->u.mvfrey.config);
659*4882a593Smuzhiyun return -1;
660*4882a593Smuzhiyun }
661*4882a593Smuzhiyun
662*4882a593Smuzhiyun return 0;
663*4882a593Smuzhiyun }
664*4882a593Smuzhiyun
hptiop_unmap_pci_bar_mv(struct hptiop_hba * hba)665*4882a593Smuzhiyun static void hptiop_unmap_pci_bar_mv(struct hptiop_hba *hba)
666*4882a593Smuzhiyun {
667*4882a593Smuzhiyun iounmap(hba->u.mv.regs);
668*4882a593Smuzhiyun iounmap(hba->u.mv.mu);
669*4882a593Smuzhiyun }
670*4882a593Smuzhiyun
hptiop_unmap_pci_bar_mvfrey(struct hptiop_hba * hba)671*4882a593Smuzhiyun static void hptiop_unmap_pci_bar_mvfrey(struct hptiop_hba *hba)
672*4882a593Smuzhiyun {
673*4882a593Smuzhiyun iounmap(hba->u.mvfrey.config);
674*4882a593Smuzhiyun iounmap(hba->u.mvfrey.mu);
675*4882a593Smuzhiyun }
676*4882a593Smuzhiyun
hptiop_message_callback(struct hptiop_hba * hba,u32 msg)677*4882a593Smuzhiyun static void hptiop_message_callback(struct hptiop_hba *hba, u32 msg)
678*4882a593Smuzhiyun {
679*4882a593Smuzhiyun dprintk("iop message 0x%x\n", msg);
680*4882a593Smuzhiyun
681*4882a593Smuzhiyun if (msg == IOPMU_INBOUND_MSG0_NOP ||
682*4882a593Smuzhiyun msg == IOPMU_INBOUND_MSG0_RESET_COMM)
683*4882a593Smuzhiyun hba->msg_done = 1;
684*4882a593Smuzhiyun
685*4882a593Smuzhiyun if (!hba->initialized)
686*4882a593Smuzhiyun return;
687*4882a593Smuzhiyun
688*4882a593Smuzhiyun if (msg == IOPMU_INBOUND_MSG0_RESET) {
689*4882a593Smuzhiyun atomic_set(&hba->resetting, 0);
690*4882a593Smuzhiyun wake_up(&hba->reset_wq);
691*4882a593Smuzhiyun }
692*4882a593Smuzhiyun else if (msg <= IOPMU_INBOUND_MSG0_MAX)
693*4882a593Smuzhiyun hba->msg_done = 1;
694*4882a593Smuzhiyun }
695*4882a593Smuzhiyun
get_req(struct hptiop_hba * hba)696*4882a593Smuzhiyun static struct hptiop_request *get_req(struct hptiop_hba *hba)
697*4882a593Smuzhiyun {
698*4882a593Smuzhiyun struct hptiop_request *ret;
699*4882a593Smuzhiyun
700*4882a593Smuzhiyun dprintk("get_req : req=%p\n", hba->req_list);
701*4882a593Smuzhiyun
702*4882a593Smuzhiyun ret = hba->req_list;
703*4882a593Smuzhiyun if (ret)
704*4882a593Smuzhiyun hba->req_list = ret->next;
705*4882a593Smuzhiyun
706*4882a593Smuzhiyun return ret;
707*4882a593Smuzhiyun }
708*4882a593Smuzhiyun
free_req(struct hptiop_hba * hba,struct hptiop_request * req)709*4882a593Smuzhiyun static void free_req(struct hptiop_hba *hba, struct hptiop_request *req)
710*4882a593Smuzhiyun {
711*4882a593Smuzhiyun dprintk("free_req(%d, %p)\n", req->index, req);
712*4882a593Smuzhiyun req->next = hba->req_list;
713*4882a593Smuzhiyun hba->req_list = req;
714*4882a593Smuzhiyun }
715*4882a593Smuzhiyun
hptiop_finish_scsi_req(struct hptiop_hba * hba,u32 tag,struct hpt_iop_request_scsi_command * req)716*4882a593Smuzhiyun static void hptiop_finish_scsi_req(struct hptiop_hba *hba, u32 tag,
717*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req)
718*4882a593Smuzhiyun {
719*4882a593Smuzhiyun struct scsi_cmnd *scp;
720*4882a593Smuzhiyun
721*4882a593Smuzhiyun dprintk("hptiop_finish_scsi_req: req=%p, type=%d, "
722*4882a593Smuzhiyun "result=%d, context=0x%x tag=%d\n",
723*4882a593Smuzhiyun req, req->header.type, req->header.result,
724*4882a593Smuzhiyun req->header.context, tag);
725*4882a593Smuzhiyun
726*4882a593Smuzhiyun BUG_ON(!req->header.result);
727*4882a593Smuzhiyun BUG_ON(req->header.type != cpu_to_le32(IOP_REQUEST_TYPE_SCSI_COMMAND));
728*4882a593Smuzhiyun
729*4882a593Smuzhiyun scp = hba->reqs[tag].scp;
730*4882a593Smuzhiyun
731*4882a593Smuzhiyun if (HPT_SCP(scp)->mapped)
732*4882a593Smuzhiyun scsi_dma_unmap(scp);
733*4882a593Smuzhiyun
734*4882a593Smuzhiyun switch (le32_to_cpu(req->header.result)) {
735*4882a593Smuzhiyun case IOP_RESULT_SUCCESS:
736*4882a593Smuzhiyun scsi_set_resid(scp,
737*4882a593Smuzhiyun scsi_bufflen(scp) - le32_to_cpu(req->dataxfer_length));
738*4882a593Smuzhiyun scp->result = (DID_OK<<16);
739*4882a593Smuzhiyun break;
740*4882a593Smuzhiyun case IOP_RESULT_BAD_TARGET:
741*4882a593Smuzhiyun scp->result = (DID_BAD_TARGET<<16);
742*4882a593Smuzhiyun break;
743*4882a593Smuzhiyun case IOP_RESULT_BUSY:
744*4882a593Smuzhiyun scp->result = (DID_BUS_BUSY<<16);
745*4882a593Smuzhiyun break;
746*4882a593Smuzhiyun case IOP_RESULT_RESET:
747*4882a593Smuzhiyun scp->result = (DID_RESET<<16);
748*4882a593Smuzhiyun break;
749*4882a593Smuzhiyun case IOP_RESULT_FAIL:
750*4882a593Smuzhiyun scp->result = (DID_ERROR<<16);
751*4882a593Smuzhiyun break;
752*4882a593Smuzhiyun case IOP_RESULT_INVALID_REQUEST:
753*4882a593Smuzhiyun scp->result = (DID_ABORT<<16);
754*4882a593Smuzhiyun break;
755*4882a593Smuzhiyun case IOP_RESULT_CHECK_CONDITION:
756*4882a593Smuzhiyun scsi_set_resid(scp,
757*4882a593Smuzhiyun scsi_bufflen(scp) - le32_to_cpu(req->dataxfer_length));
758*4882a593Smuzhiyun scp->result = SAM_STAT_CHECK_CONDITION;
759*4882a593Smuzhiyun memcpy(scp->sense_buffer, &req->sg_list, SCSI_SENSE_BUFFERSIZE);
760*4882a593Smuzhiyun goto skip_resid;
761*4882a593Smuzhiyun break;
762*4882a593Smuzhiyun
763*4882a593Smuzhiyun default:
764*4882a593Smuzhiyun scp->result = DRIVER_INVALID << 24 | DID_ABORT << 16;
765*4882a593Smuzhiyun break;
766*4882a593Smuzhiyun }
767*4882a593Smuzhiyun
768*4882a593Smuzhiyun scsi_set_resid(scp,
769*4882a593Smuzhiyun scsi_bufflen(scp) - le32_to_cpu(req->dataxfer_length));
770*4882a593Smuzhiyun
771*4882a593Smuzhiyun skip_resid:
772*4882a593Smuzhiyun dprintk("scsi_done(%p)\n", scp);
773*4882a593Smuzhiyun scp->scsi_done(scp);
774*4882a593Smuzhiyun free_req(hba, &hba->reqs[tag]);
775*4882a593Smuzhiyun }
776*4882a593Smuzhiyun
hptiop_host_request_callback_itl(struct hptiop_hba * hba,u32 _tag)777*4882a593Smuzhiyun static void hptiop_host_request_callback_itl(struct hptiop_hba *hba, u32 _tag)
778*4882a593Smuzhiyun {
779*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req;
780*4882a593Smuzhiyun u32 tag;
781*4882a593Smuzhiyun
782*4882a593Smuzhiyun if (hba->iopintf_v2) {
783*4882a593Smuzhiyun tag = _tag & ~IOPMU_QUEUE_REQUEST_RESULT_BIT;
784*4882a593Smuzhiyun req = hba->reqs[tag].req_virt;
785*4882a593Smuzhiyun if (likely(_tag & IOPMU_QUEUE_REQUEST_RESULT_BIT))
786*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_SUCCESS);
787*4882a593Smuzhiyun } else {
788*4882a593Smuzhiyun tag = _tag;
789*4882a593Smuzhiyun req = hba->reqs[tag].req_virt;
790*4882a593Smuzhiyun }
791*4882a593Smuzhiyun
792*4882a593Smuzhiyun hptiop_finish_scsi_req(hba, tag, req);
793*4882a593Smuzhiyun }
794*4882a593Smuzhiyun
hptiop_iop_request_callback_itl(struct hptiop_hba * hba,u32 tag)795*4882a593Smuzhiyun static void hptiop_iop_request_callback_itl(struct hptiop_hba *hba, u32 tag)
796*4882a593Smuzhiyun {
797*4882a593Smuzhiyun struct hpt_iop_request_header __iomem *req;
798*4882a593Smuzhiyun struct hpt_iop_request_ioctl_command __iomem *p;
799*4882a593Smuzhiyun struct hpt_ioctl_k *arg;
800*4882a593Smuzhiyun
801*4882a593Smuzhiyun req = (struct hpt_iop_request_header __iomem *)
802*4882a593Smuzhiyun ((unsigned long)hba->u.itl.iop + tag);
803*4882a593Smuzhiyun dprintk("hptiop_iop_request_callback_itl: req=%p, type=%d, "
804*4882a593Smuzhiyun "result=%d, context=0x%x tag=%d\n",
805*4882a593Smuzhiyun req, readl(&req->type), readl(&req->result),
806*4882a593Smuzhiyun readl(&req->context), tag);
807*4882a593Smuzhiyun
808*4882a593Smuzhiyun BUG_ON(!readl(&req->result));
809*4882a593Smuzhiyun BUG_ON(readl(&req->type) != IOP_REQUEST_TYPE_IOCTL_COMMAND);
810*4882a593Smuzhiyun
811*4882a593Smuzhiyun p = (struct hpt_iop_request_ioctl_command __iomem *)req;
812*4882a593Smuzhiyun arg = (struct hpt_ioctl_k *)(unsigned long)
813*4882a593Smuzhiyun (readl(&req->context) |
814*4882a593Smuzhiyun ((u64)readl(&req->context_hi32)<<32));
815*4882a593Smuzhiyun
816*4882a593Smuzhiyun if (readl(&req->result) == IOP_RESULT_SUCCESS) {
817*4882a593Smuzhiyun arg->result = HPT_IOCTL_RESULT_OK;
818*4882a593Smuzhiyun
819*4882a593Smuzhiyun if (arg->outbuf_size)
820*4882a593Smuzhiyun memcpy_fromio(arg->outbuf,
821*4882a593Smuzhiyun &p->buf[(readl(&p->inbuf_size) + 3)& ~3],
822*4882a593Smuzhiyun arg->outbuf_size);
823*4882a593Smuzhiyun
824*4882a593Smuzhiyun if (arg->bytes_returned)
825*4882a593Smuzhiyun *arg->bytes_returned = arg->outbuf_size;
826*4882a593Smuzhiyun }
827*4882a593Smuzhiyun else
828*4882a593Smuzhiyun arg->result = HPT_IOCTL_RESULT_FAILED;
829*4882a593Smuzhiyun
830*4882a593Smuzhiyun arg->done(arg);
831*4882a593Smuzhiyun writel(tag, &hba->u.itl.iop->outbound_queue);
832*4882a593Smuzhiyun }
833*4882a593Smuzhiyun
hptiop_intr(int irq,void * dev_id)834*4882a593Smuzhiyun static irqreturn_t hptiop_intr(int irq, void *dev_id)
835*4882a593Smuzhiyun {
836*4882a593Smuzhiyun struct hptiop_hba *hba = dev_id;
837*4882a593Smuzhiyun int handled;
838*4882a593Smuzhiyun unsigned long flags;
839*4882a593Smuzhiyun
840*4882a593Smuzhiyun spin_lock_irqsave(hba->host->host_lock, flags);
841*4882a593Smuzhiyun handled = hba->ops->iop_intr(hba);
842*4882a593Smuzhiyun spin_unlock_irqrestore(hba->host->host_lock, flags);
843*4882a593Smuzhiyun
844*4882a593Smuzhiyun return handled;
845*4882a593Smuzhiyun }
846*4882a593Smuzhiyun
hptiop_buildsgl(struct scsi_cmnd * scp,struct hpt_iopsg * psg)847*4882a593Smuzhiyun static int hptiop_buildsgl(struct scsi_cmnd *scp, struct hpt_iopsg *psg)
848*4882a593Smuzhiyun {
849*4882a593Smuzhiyun struct Scsi_Host *host = scp->device->host;
850*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)host->hostdata;
851*4882a593Smuzhiyun struct scatterlist *sg;
852*4882a593Smuzhiyun int idx, nseg;
853*4882a593Smuzhiyun
854*4882a593Smuzhiyun nseg = scsi_dma_map(scp);
855*4882a593Smuzhiyun BUG_ON(nseg < 0);
856*4882a593Smuzhiyun if (!nseg)
857*4882a593Smuzhiyun return 0;
858*4882a593Smuzhiyun
859*4882a593Smuzhiyun HPT_SCP(scp)->sgcnt = nseg;
860*4882a593Smuzhiyun HPT_SCP(scp)->mapped = 1;
861*4882a593Smuzhiyun
862*4882a593Smuzhiyun BUG_ON(HPT_SCP(scp)->sgcnt > hba->max_sg_descriptors);
863*4882a593Smuzhiyun
864*4882a593Smuzhiyun scsi_for_each_sg(scp, sg, HPT_SCP(scp)->sgcnt, idx) {
865*4882a593Smuzhiyun psg[idx].pci_address = cpu_to_le64(sg_dma_address(sg)) |
866*4882a593Smuzhiyun hba->ops->host_phy_flag;
867*4882a593Smuzhiyun psg[idx].size = cpu_to_le32(sg_dma_len(sg));
868*4882a593Smuzhiyun psg[idx].eot = (idx == HPT_SCP(scp)->sgcnt - 1) ?
869*4882a593Smuzhiyun cpu_to_le32(1) : 0;
870*4882a593Smuzhiyun }
871*4882a593Smuzhiyun return HPT_SCP(scp)->sgcnt;
872*4882a593Smuzhiyun }
873*4882a593Smuzhiyun
hptiop_post_req_itl(struct hptiop_hba * hba,struct hptiop_request * _req)874*4882a593Smuzhiyun static void hptiop_post_req_itl(struct hptiop_hba *hba,
875*4882a593Smuzhiyun struct hptiop_request *_req)
876*4882a593Smuzhiyun {
877*4882a593Smuzhiyun struct hpt_iop_request_header *reqhdr = _req->req_virt;
878*4882a593Smuzhiyun
879*4882a593Smuzhiyun reqhdr->context = cpu_to_le32(IOPMU_QUEUE_ADDR_HOST_BIT |
880*4882a593Smuzhiyun (u32)_req->index);
881*4882a593Smuzhiyun reqhdr->context_hi32 = 0;
882*4882a593Smuzhiyun
883*4882a593Smuzhiyun if (hba->iopintf_v2) {
884*4882a593Smuzhiyun u32 size, size_bits;
885*4882a593Smuzhiyun
886*4882a593Smuzhiyun size = le32_to_cpu(reqhdr->size);
887*4882a593Smuzhiyun if (size < 256)
888*4882a593Smuzhiyun size_bits = IOPMU_QUEUE_REQUEST_SIZE_BIT;
889*4882a593Smuzhiyun else if (size < 512)
890*4882a593Smuzhiyun size_bits = IOPMU_QUEUE_ADDR_HOST_BIT;
891*4882a593Smuzhiyun else
892*4882a593Smuzhiyun size_bits = IOPMU_QUEUE_REQUEST_SIZE_BIT |
893*4882a593Smuzhiyun IOPMU_QUEUE_ADDR_HOST_BIT;
894*4882a593Smuzhiyun writel(_req->req_shifted_phy | size_bits,
895*4882a593Smuzhiyun &hba->u.itl.iop->inbound_queue);
896*4882a593Smuzhiyun } else
897*4882a593Smuzhiyun writel(_req->req_shifted_phy | IOPMU_QUEUE_ADDR_HOST_BIT,
898*4882a593Smuzhiyun &hba->u.itl.iop->inbound_queue);
899*4882a593Smuzhiyun }
900*4882a593Smuzhiyun
hptiop_post_req_mv(struct hptiop_hba * hba,struct hptiop_request * _req)901*4882a593Smuzhiyun static void hptiop_post_req_mv(struct hptiop_hba *hba,
902*4882a593Smuzhiyun struct hptiop_request *_req)
903*4882a593Smuzhiyun {
904*4882a593Smuzhiyun struct hpt_iop_request_header *reqhdr = _req->req_virt;
905*4882a593Smuzhiyun u32 size, size_bit;
906*4882a593Smuzhiyun
907*4882a593Smuzhiyun reqhdr->context = cpu_to_le32(_req->index<<8 |
908*4882a593Smuzhiyun IOP_REQUEST_TYPE_SCSI_COMMAND<<5);
909*4882a593Smuzhiyun reqhdr->context_hi32 = 0;
910*4882a593Smuzhiyun size = le32_to_cpu(reqhdr->size);
911*4882a593Smuzhiyun
912*4882a593Smuzhiyun if (size <= 256)
913*4882a593Smuzhiyun size_bit = 0;
914*4882a593Smuzhiyun else if (size <= 256*2)
915*4882a593Smuzhiyun size_bit = 1;
916*4882a593Smuzhiyun else if (size <= 256*3)
917*4882a593Smuzhiyun size_bit = 2;
918*4882a593Smuzhiyun else
919*4882a593Smuzhiyun size_bit = 3;
920*4882a593Smuzhiyun
921*4882a593Smuzhiyun mv_inbound_write((_req->req_shifted_phy << 5) |
922*4882a593Smuzhiyun MVIOP_MU_QUEUE_ADDR_HOST_BIT | size_bit, hba);
923*4882a593Smuzhiyun }
924*4882a593Smuzhiyun
hptiop_post_req_mvfrey(struct hptiop_hba * hba,struct hptiop_request * _req)925*4882a593Smuzhiyun static void hptiop_post_req_mvfrey(struct hptiop_hba *hba,
926*4882a593Smuzhiyun struct hptiop_request *_req)
927*4882a593Smuzhiyun {
928*4882a593Smuzhiyun struct hpt_iop_request_header *reqhdr = _req->req_virt;
929*4882a593Smuzhiyun u32 index;
930*4882a593Smuzhiyun
931*4882a593Smuzhiyun reqhdr->flags |= cpu_to_le32(IOP_REQUEST_FLAG_OUTPUT_CONTEXT |
932*4882a593Smuzhiyun IOP_REQUEST_FLAG_ADDR_BITS |
933*4882a593Smuzhiyun ((_req->req_shifted_phy >> 11) & 0xffff0000));
934*4882a593Smuzhiyun reqhdr->context = cpu_to_le32(IOPMU_QUEUE_ADDR_HOST_BIT |
935*4882a593Smuzhiyun (_req->index << 4) | reqhdr->type);
936*4882a593Smuzhiyun reqhdr->context_hi32 = cpu_to_le32((_req->req_shifted_phy << 5) &
937*4882a593Smuzhiyun 0xffffffff);
938*4882a593Smuzhiyun
939*4882a593Smuzhiyun hba->u.mvfrey.inlist_wptr++;
940*4882a593Smuzhiyun index = hba->u.mvfrey.inlist_wptr & 0x3fff;
941*4882a593Smuzhiyun
942*4882a593Smuzhiyun if (index == hba->u.mvfrey.list_count) {
943*4882a593Smuzhiyun index = 0;
944*4882a593Smuzhiyun hba->u.mvfrey.inlist_wptr &= ~0x3fff;
945*4882a593Smuzhiyun hba->u.mvfrey.inlist_wptr ^= CL_POINTER_TOGGLE;
946*4882a593Smuzhiyun }
947*4882a593Smuzhiyun
948*4882a593Smuzhiyun hba->u.mvfrey.inlist[index].addr =
949*4882a593Smuzhiyun (dma_addr_t)_req->req_shifted_phy << 5;
950*4882a593Smuzhiyun hba->u.mvfrey.inlist[index].intrfc_len = (reqhdr->size + 3) / 4;
951*4882a593Smuzhiyun writel(hba->u.mvfrey.inlist_wptr,
952*4882a593Smuzhiyun &(hba->u.mvfrey.mu->inbound_write_ptr));
953*4882a593Smuzhiyun readl(&(hba->u.mvfrey.mu->inbound_write_ptr));
954*4882a593Smuzhiyun }
955*4882a593Smuzhiyun
hptiop_reset_comm_itl(struct hptiop_hba * hba)956*4882a593Smuzhiyun static int hptiop_reset_comm_itl(struct hptiop_hba *hba)
957*4882a593Smuzhiyun {
958*4882a593Smuzhiyun return 0;
959*4882a593Smuzhiyun }
960*4882a593Smuzhiyun
hptiop_reset_comm_mv(struct hptiop_hba * hba)961*4882a593Smuzhiyun static int hptiop_reset_comm_mv(struct hptiop_hba *hba)
962*4882a593Smuzhiyun {
963*4882a593Smuzhiyun return 0;
964*4882a593Smuzhiyun }
965*4882a593Smuzhiyun
hptiop_reset_comm_mvfrey(struct hptiop_hba * hba)966*4882a593Smuzhiyun static int hptiop_reset_comm_mvfrey(struct hptiop_hba *hba)
967*4882a593Smuzhiyun {
968*4882a593Smuzhiyun u32 list_count = hba->u.mvfrey.list_count;
969*4882a593Smuzhiyun
970*4882a593Smuzhiyun if (iop_send_sync_msg(hba, IOPMU_INBOUND_MSG0_RESET_COMM, 3000))
971*4882a593Smuzhiyun return -1;
972*4882a593Smuzhiyun
973*4882a593Smuzhiyun /* wait 100ms for MCU ready */
974*4882a593Smuzhiyun msleep(100);
975*4882a593Smuzhiyun
976*4882a593Smuzhiyun writel(cpu_to_le32(hba->u.mvfrey.inlist_phy & 0xffffffff),
977*4882a593Smuzhiyun &(hba->u.mvfrey.mu->inbound_base));
978*4882a593Smuzhiyun writel(cpu_to_le32((hba->u.mvfrey.inlist_phy >> 16) >> 16),
979*4882a593Smuzhiyun &(hba->u.mvfrey.mu->inbound_base_high));
980*4882a593Smuzhiyun
981*4882a593Smuzhiyun writel(cpu_to_le32(hba->u.mvfrey.outlist_phy & 0xffffffff),
982*4882a593Smuzhiyun &(hba->u.mvfrey.mu->outbound_base));
983*4882a593Smuzhiyun writel(cpu_to_le32((hba->u.mvfrey.outlist_phy >> 16) >> 16),
984*4882a593Smuzhiyun &(hba->u.mvfrey.mu->outbound_base_high));
985*4882a593Smuzhiyun
986*4882a593Smuzhiyun writel(cpu_to_le32(hba->u.mvfrey.outlist_cptr_phy & 0xffffffff),
987*4882a593Smuzhiyun &(hba->u.mvfrey.mu->outbound_shadow_base));
988*4882a593Smuzhiyun writel(cpu_to_le32((hba->u.mvfrey.outlist_cptr_phy >> 16) >> 16),
989*4882a593Smuzhiyun &(hba->u.mvfrey.mu->outbound_shadow_base_high));
990*4882a593Smuzhiyun
991*4882a593Smuzhiyun hba->u.mvfrey.inlist_wptr = (list_count - 1) | CL_POINTER_TOGGLE;
992*4882a593Smuzhiyun *hba->u.mvfrey.outlist_cptr = (list_count - 1) | CL_POINTER_TOGGLE;
993*4882a593Smuzhiyun hba->u.mvfrey.outlist_rptr = list_count - 1;
994*4882a593Smuzhiyun return 0;
995*4882a593Smuzhiyun }
996*4882a593Smuzhiyun
hptiop_queuecommand_lck(struct scsi_cmnd * scp,void (* done)(struct scsi_cmnd *))997*4882a593Smuzhiyun static int hptiop_queuecommand_lck(struct scsi_cmnd *scp,
998*4882a593Smuzhiyun void (*done)(struct scsi_cmnd *))
999*4882a593Smuzhiyun {
1000*4882a593Smuzhiyun struct Scsi_Host *host = scp->device->host;
1001*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)host->hostdata;
1002*4882a593Smuzhiyun struct hpt_iop_request_scsi_command *req;
1003*4882a593Smuzhiyun int sg_count = 0;
1004*4882a593Smuzhiyun struct hptiop_request *_req;
1005*4882a593Smuzhiyun
1006*4882a593Smuzhiyun BUG_ON(!done);
1007*4882a593Smuzhiyun scp->scsi_done = done;
1008*4882a593Smuzhiyun
1009*4882a593Smuzhiyun _req = get_req(hba);
1010*4882a593Smuzhiyun if (_req == NULL) {
1011*4882a593Smuzhiyun dprintk("hptiop_queuecmd : no free req\n");
1012*4882a593Smuzhiyun return SCSI_MLQUEUE_HOST_BUSY;
1013*4882a593Smuzhiyun }
1014*4882a593Smuzhiyun
1015*4882a593Smuzhiyun _req->scp = scp;
1016*4882a593Smuzhiyun
1017*4882a593Smuzhiyun dprintk("hptiop_queuecmd(scp=%p) %d/%d/%d/%llu cdb=(%08x-%08x-%08x-%08x) "
1018*4882a593Smuzhiyun "req_index=%d, req=%p\n",
1019*4882a593Smuzhiyun scp,
1020*4882a593Smuzhiyun host->host_no, scp->device->channel,
1021*4882a593Smuzhiyun scp->device->id, scp->device->lun,
1022*4882a593Smuzhiyun cpu_to_be32(((u32 *)scp->cmnd)[0]),
1023*4882a593Smuzhiyun cpu_to_be32(((u32 *)scp->cmnd)[1]),
1024*4882a593Smuzhiyun cpu_to_be32(((u32 *)scp->cmnd)[2]),
1025*4882a593Smuzhiyun cpu_to_be32(((u32 *)scp->cmnd)[3]),
1026*4882a593Smuzhiyun _req->index, _req->req_virt);
1027*4882a593Smuzhiyun
1028*4882a593Smuzhiyun scp->result = 0;
1029*4882a593Smuzhiyun
1030*4882a593Smuzhiyun if (scp->device->channel ||
1031*4882a593Smuzhiyun (scp->device->id > hba->max_devices) ||
1032*4882a593Smuzhiyun ((scp->device->id == (hba->max_devices-1)) && scp->device->lun)) {
1033*4882a593Smuzhiyun scp->result = DID_BAD_TARGET << 16;
1034*4882a593Smuzhiyun free_req(hba, _req);
1035*4882a593Smuzhiyun goto cmd_done;
1036*4882a593Smuzhiyun }
1037*4882a593Smuzhiyun
1038*4882a593Smuzhiyun req = _req->req_virt;
1039*4882a593Smuzhiyun
1040*4882a593Smuzhiyun /* build S/G table */
1041*4882a593Smuzhiyun sg_count = hptiop_buildsgl(scp, req->sg_list);
1042*4882a593Smuzhiyun if (!sg_count)
1043*4882a593Smuzhiyun HPT_SCP(scp)->mapped = 0;
1044*4882a593Smuzhiyun
1045*4882a593Smuzhiyun req->header.flags = cpu_to_le32(IOP_REQUEST_FLAG_OUTPUT_CONTEXT);
1046*4882a593Smuzhiyun req->header.type = cpu_to_le32(IOP_REQUEST_TYPE_SCSI_COMMAND);
1047*4882a593Smuzhiyun req->header.result = cpu_to_le32(IOP_RESULT_PENDING);
1048*4882a593Smuzhiyun req->dataxfer_length = cpu_to_le32(scsi_bufflen(scp));
1049*4882a593Smuzhiyun req->channel = scp->device->channel;
1050*4882a593Smuzhiyun req->target = scp->device->id;
1051*4882a593Smuzhiyun req->lun = scp->device->lun;
1052*4882a593Smuzhiyun req->header.size = cpu_to_le32(
1053*4882a593Smuzhiyun sizeof(struct hpt_iop_request_scsi_command)
1054*4882a593Smuzhiyun - sizeof(struct hpt_iopsg)
1055*4882a593Smuzhiyun + sg_count * sizeof(struct hpt_iopsg));
1056*4882a593Smuzhiyun
1057*4882a593Smuzhiyun memcpy(req->cdb, scp->cmnd, sizeof(req->cdb));
1058*4882a593Smuzhiyun hba->ops->post_req(hba, _req);
1059*4882a593Smuzhiyun return 0;
1060*4882a593Smuzhiyun
1061*4882a593Smuzhiyun cmd_done:
1062*4882a593Smuzhiyun dprintk("scsi_done(scp=%p)\n", scp);
1063*4882a593Smuzhiyun scp->scsi_done(scp);
1064*4882a593Smuzhiyun return 0;
1065*4882a593Smuzhiyun }
1066*4882a593Smuzhiyun
DEF_SCSI_QCMD(hptiop_queuecommand)1067*4882a593Smuzhiyun static DEF_SCSI_QCMD(hptiop_queuecommand)
1068*4882a593Smuzhiyun
1069*4882a593Smuzhiyun static const char *hptiop_info(struct Scsi_Host *host)
1070*4882a593Smuzhiyun {
1071*4882a593Smuzhiyun return driver_name_long;
1072*4882a593Smuzhiyun }
1073*4882a593Smuzhiyun
hptiop_reset_hba(struct hptiop_hba * hba)1074*4882a593Smuzhiyun static int hptiop_reset_hba(struct hptiop_hba *hba)
1075*4882a593Smuzhiyun {
1076*4882a593Smuzhiyun if (atomic_xchg(&hba->resetting, 1) == 0) {
1077*4882a593Smuzhiyun atomic_inc(&hba->reset_count);
1078*4882a593Smuzhiyun hba->ops->post_msg(hba, IOPMU_INBOUND_MSG0_RESET);
1079*4882a593Smuzhiyun }
1080*4882a593Smuzhiyun
1081*4882a593Smuzhiyun wait_event_timeout(hba->reset_wq,
1082*4882a593Smuzhiyun atomic_read(&hba->resetting) == 0, 60 * HZ);
1083*4882a593Smuzhiyun
1084*4882a593Smuzhiyun if (atomic_read(&hba->resetting)) {
1085*4882a593Smuzhiyun /* IOP is in unknown state, abort reset */
1086*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: reset failed\n", hba->host->host_no);
1087*4882a593Smuzhiyun return -1;
1088*4882a593Smuzhiyun }
1089*4882a593Smuzhiyun
1090*4882a593Smuzhiyun if (iop_send_sync_msg(hba,
1091*4882a593Smuzhiyun IOPMU_INBOUND_MSG0_START_BACKGROUND_TASK, 5000)) {
1092*4882a593Smuzhiyun dprintk("scsi%d: fail to start background task\n",
1093*4882a593Smuzhiyun hba->host->host_no);
1094*4882a593Smuzhiyun }
1095*4882a593Smuzhiyun
1096*4882a593Smuzhiyun return 0;
1097*4882a593Smuzhiyun }
1098*4882a593Smuzhiyun
hptiop_reset(struct scsi_cmnd * scp)1099*4882a593Smuzhiyun static int hptiop_reset(struct scsi_cmnd *scp)
1100*4882a593Smuzhiyun {
1101*4882a593Smuzhiyun struct hptiop_hba * hba = (struct hptiop_hba *)scp->device->host->hostdata;
1102*4882a593Smuzhiyun
1103*4882a593Smuzhiyun printk(KERN_WARNING "hptiop_reset(%d/%d/%d)\n",
1104*4882a593Smuzhiyun scp->device->host->host_no, -1, -1);
1105*4882a593Smuzhiyun
1106*4882a593Smuzhiyun return hptiop_reset_hba(hba)? FAILED : SUCCESS;
1107*4882a593Smuzhiyun }
1108*4882a593Smuzhiyun
hptiop_adjust_disk_queue_depth(struct scsi_device * sdev,int queue_depth)1109*4882a593Smuzhiyun static int hptiop_adjust_disk_queue_depth(struct scsi_device *sdev,
1110*4882a593Smuzhiyun int queue_depth)
1111*4882a593Smuzhiyun {
1112*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)sdev->host->hostdata;
1113*4882a593Smuzhiyun
1114*4882a593Smuzhiyun if (queue_depth > hba->max_requests)
1115*4882a593Smuzhiyun queue_depth = hba->max_requests;
1116*4882a593Smuzhiyun return scsi_change_queue_depth(sdev, queue_depth);
1117*4882a593Smuzhiyun }
1118*4882a593Smuzhiyun
hptiop_show_version(struct device * dev,struct device_attribute * attr,char * buf)1119*4882a593Smuzhiyun static ssize_t hptiop_show_version(struct device *dev,
1120*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
1121*4882a593Smuzhiyun {
1122*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%s\n", driver_ver);
1123*4882a593Smuzhiyun }
1124*4882a593Smuzhiyun
hptiop_show_fw_version(struct device * dev,struct device_attribute * attr,char * buf)1125*4882a593Smuzhiyun static ssize_t hptiop_show_fw_version(struct device *dev,
1126*4882a593Smuzhiyun struct device_attribute *attr, char *buf)
1127*4882a593Smuzhiyun {
1128*4882a593Smuzhiyun struct Scsi_Host *host = class_to_shost(dev);
1129*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)host->hostdata;
1130*4882a593Smuzhiyun
1131*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d.%d.%d.%d\n",
1132*4882a593Smuzhiyun hba->firmware_version >> 24,
1133*4882a593Smuzhiyun (hba->firmware_version >> 16) & 0xff,
1134*4882a593Smuzhiyun (hba->firmware_version >> 8) & 0xff,
1135*4882a593Smuzhiyun hba->firmware_version & 0xff);
1136*4882a593Smuzhiyun }
1137*4882a593Smuzhiyun
1138*4882a593Smuzhiyun static struct device_attribute hptiop_attr_version = {
1139*4882a593Smuzhiyun .attr = {
1140*4882a593Smuzhiyun .name = "driver-version",
1141*4882a593Smuzhiyun .mode = S_IRUGO,
1142*4882a593Smuzhiyun },
1143*4882a593Smuzhiyun .show = hptiop_show_version,
1144*4882a593Smuzhiyun };
1145*4882a593Smuzhiyun
1146*4882a593Smuzhiyun static struct device_attribute hptiop_attr_fw_version = {
1147*4882a593Smuzhiyun .attr = {
1148*4882a593Smuzhiyun .name = "firmware-version",
1149*4882a593Smuzhiyun .mode = S_IRUGO,
1150*4882a593Smuzhiyun },
1151*4882a593Smuzhiyun .show = hptiop_show_fw_version,
1152*4882a593Smuzhiyun };
1153*4882a593Smuzhiyun
1154*4882a593Smuzhiyun static struct device_attribute *hptiop_attrs[] = {
1155*4882a593Smuzhiyun &hptiop_attr_version,
1156*4882a593Smuzhiyun &hptiop_attr_fw_version,
1157*4882a593Smuzhiyun NULL
1158*4882a593Smuzhiyun };
1159*4882a593Smuzhiyun
hptiop_slave_config(struct scsi_device * sdev)1160*4882a593Smuzhiyun static int hptiop_slave_config(struct scsi_device *sdev)
1161*4882a593Smuzhiyun {
1162*4882a593Smuzhiyun if (sdev->type == TYPE_TAPE)
1163*4882a593Smuzhiyun blk_queue_max_hw_sectors(sdev->request_queue, 8192);
1164*4882a593Smuzhiyun
1165*4882a593Smuzhiyun return 0;
1166*4882a593Smuzhiyun }
1167*4882a593Smuzhiyun
1168*4882a593Smuzhiyun static struct scsi_host_template driver_template = {
1169*4882a593Smuzhiyun .module = THIS_MODULE,
1170*4882a593Smuzhiyun .name = driver_name,
1171*4882a593Smuzhiyun .queuecommand = hptiop_queuecommand,
1172*4882a593Smuzhiyun .eh_host_reset_handler = hptiop_reset,
1173*4882a593Smuzhiyun .info = hptiop_info,
1174*4882a593Smuzhiyun .emulated = 0,
1175*4882a593Smuzhiyun .proc_name = driver_name,
1176*4882a593Smuzhiyun .shost_attrs = hptiop_attrs,
1177*4882a593Smuzhiyun .slave_configure = hptiop_slave_config,
1178*4882a593Smuzhiyun .this_id = -1,
1179*4882a593Smuzhiyun .change_queue_depth = hptiop_adjust_disk_queue_depth,
1180*4882a593Smuzhiyun };
1181*4882a593Smuzhiyun
hptiop_internal_memalloc_itl(struct hptiop_hba * hba)1182*4882a593Smuzhiyun static int hptiop_internal_memalloc_itl(struct hptiop_hba *hba)
1183*4882a593Smuzhiyun {
1184*4882a593Smuzhiyun return 0;
1185*4882a593Smuzhiyun }
1186*4882a593Smuzhiyun
hptiop_internal_memalloc_mv(struct hptiop_hba * hba)1187*4882a593Smuzhiyun static int hptiop_internal_memalloc_mv(struct hptiop_hba *hba)
1188*4882a593Smuzhiyun {
1189*4882a593Smuzhiyun hba->u.mv.internal_req = dma_alloc_coherent(&hba->pcidev->dev,
1190*4882a593Smuzhiyun 0x800, &hba->u.mv.internal_req_phy, GFP_KERNEL);
1191*4882a593Smuzhiyun if (hba->u.mv.internal_req)
1192*4882a593Smuzhiyun return 0;
1193*4882a593Smuzhiyun else
1194*4882a593Smuzhiyun return -1;
1195*4882a593Smuzhiyun }
1196*4882a593Smuzhiyun
hptiop_internal_memalloc_mvfrey(struct hptiop_hba * hba)1197*4882a593Smuzhiyun static int hptiop_internal_memalloc_mvfrey(struct hptiop_hba *hba)
1198*4882a593Smuzhiyun {
1199*4882a593Smuzhiyun u32 list_count = readl(&hba->u.mvfrey.mu->inbound_conf_ctl);
1200*4882a593Smuzhiyun char *p;
1201*4882a593Smuzhiyun dma_addr_t phy;
1202*4882a593Smuzhiyun
1203*4882a593Smuzhiyun BUG_ON(hba->max_request_size == 0);
1204*4882a593Smuzhiyun
1205*4882a593Smuzhiyun if (list_count == 0) {
1206*4882a593Smuzhiyun BUG_ON(1);
1207*4882a593Smuzhiyun return -1;
1208*4882a593Smuzhiyun }
1209*4882a593Smuzhiyun
1210*4882a593Smuzhiyun list_count >>= 16;
1211*4882a593Smuzhiyun
1212*4882a593Smuzhiyun hba->u.mvfrey.list_count = list_count;
1213*4882a593Smuzhiyun hba->u.mvfrey.internal_mem_size = 0x800 +
1214*4882a593Smuzhiyun list_count * sizeof(struct mvfrey_inlist_entry) +
1215*4882a593Smuzhiyun list_count * sizeof(struct mvfrey_outlist_entry) +
1216*4882a593Smuzhiyun sizeof(int);
1217*4882a593Smuzhiyun
1218*4882a593Smuzhiyun p = dma_alloc_coherent(&hba->pcidev->dev,
1219*4882a593Smuzhiyun hba->u.mvfrey.internal_mem_size, &phy, GFP_KERNEL);
1220*4882a593Smuzhiyun if (!p)
1221*4882a593Smuzhiyun return -1;
1222*4882a593Smuzhiyun
1223*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_virt = p;
1224*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_shifted_phy = phy >> 5;
1225*4882a593Smuzhiyun hba->u.mvfrey.internal_req.scp = NULL;
1226*4882a593Smuzhiyun hba->u.mvfrey.internal_req.next = NULL;
1227*4882a593Smuzhiyun
1228*4882a593Smuzhiyun p += 0x800;
1229*4882a593Smuzhiyun phy += 0x800;
1230*4882a593Smuzhiyun
1231*4882a593Smuzhiyun hba->u.mvfrey.inlist = (struct mvfrey_inlist_entry *)p;
1232*4882a593Smuzhiyun hba->u.mvfrey.inlist_phy = phy;
1233*4882a593Smuzhiyun
1234*4882a593Smuzhiyun p += list_count * sizeof(struct mvfrey_inlist_entry);
1235*4882a593Smuzhiyun phy += list_count * sizeof(struct mvfrey_inlist_entry);
1236*4882a593Smuzhiyun
1237*4882a593Smuzhiyun hba->u.mvfrey.outlist = (struct mvfrey_outlist_entry *)p;
1238*4882a593Smuzhiyun hba->u.mvfrey.outlist_phy = phy;
1239*4882a593Smuzhiyun
1240*4882a593Smuzhiyun p += list_count * sizeof(struct mvfrey_outlist_entry);
1241*4882a593Smuzhiyun phy += list_count * sizeof(struct mvfrey_outlist_entry);
1242*4882a593Smuzhiyun
1243*4882a593Smuzhiyun hba->u.mvfrey.outlist_cptr = (__le32 *)p;
1244*4882a593Smuzhiyun hba->u.mvfrey.outlist_cptr_phy = phy;
1245*4882a593Smuzhiyun
1246*4882a593Smuzhiyun return 0;
1247*4882a593Smuzhiyun }
1248*4882a593Smuzhiyun
hptiop_internal_memfree_itl(struct hptiop_hba * hba)1249*4882a593Smuzhiyun static int hptiop_internal_memfree_itl(struct hptiop_hba *hba)
1250*4882a593Smuzhiyun {
1251*4882a593Smuzhiyun return 0;
1252*4882a593Smuzhiyun }
1253*4882a593Smuzhiyun
hptiop_internal_memfree_mv(struct hptiop_hba * hba)1254*4882a593Smuzhiyun static int hptiop_internal_memfree_mv(struct hptiop_hba *hba)
1255*4882a593Smuzhiyun {
1256*4882a593Smuzhiyun if (hba->u.mv.internal_req) {
1257*4882a593Smuzhiyun dma_free_coherent(&hba->pcidev->dev, 0x800,
1258*4882a593Smuzhiyun hba->u.mv.internal_req, hba->u.mv.internal_req_phy);
1259*4882a593Smuzhiyun return 0;
1260*4882a593Smuzhiyun } else
1261*4882a593Smuzhiyun return -1;
1262*4882a593Smuzhiyun }
1263*4882a593Smuzhiyun
hptiop_internal_memfree_mvfrey(struct hptiop_hba * hba)1264*4882a593Smuzhiyun static int hptiop_internal_memfree_mvfrey(struct hptiop_hba *hba)
1265*4882a593Smuzhiyun {
1266*4882a593Smuzhiyun if (hba->u.mvfrey.internal_req.req_virt) {
1267*4882a593Smuzhiyun dma_free_coherent(&hba->pcidev->dev,
1268*4882a593Smuzhiyun hba->u.mvfrey.internal_mem_size,
1269*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_virt,
1270*4882a593Smuzhiyun (dma_addr_t)
1271*4882a593Smuzhiyun hba->u.mvfrey.internal_req.req_shifted_phy << 5);
1272*4882a593Smuzhiyun return 0;
1273*4882a593Smuzhiyun } else
1274*4882a593Smuzhiyun return -1;
1275*4882a593Smuzhiyun }
1276*4882a593Smuzhiyun
hptiop_probe(struct pci_dev * pcidev,const struct pci_device_id * id)1277*4882a593Smuzhiyun static int hptiop_probe(struct pci_dev *pcidev, const struct pci_device_id *id)
1278*4882a593Smuzhiyun {
1279*4882a593Smuzhiyun struct Scsi_Host *host = NULL;
1280*4882a593Smuzhiyun struct hptiop_hba *hba;
1281*4882a593Smuzhiyun struct hptiop_adapter_ops *iop_ops;
1282*4882a593Smuzhiyun struct hpt_iop_request_get_config iop_config;
1283*4882a593Smuzhiyun struct hpt_iop_request_set_config set_config;
1284*4882a593Smuzhiyun dma_addr_t start_phy;
1285*4882a593Smuzhiyun void *start_virt;
1286*4882a593Smuzhiyun u32 offset, i, req_size;
1287*4882a593Smuzhiyun int rc;
1288*4882a593Smuzhiyun
1289*4882a593Smuzhiyun dprintk("hptiop_probe(%p)\n", pcidev);
1290*4882a593Smuzhiyun
1291*4882a593Smuzhiyun if (pci_enable_device(pcidev)) {
1292*4882a593Smuzhiyun printk(KERN_ERR "hptiop: fail to enable pci device\n");
1293*4882a593Smuzhiyun return -ENODEV;
1294*4882a593Smuzhiyun }
1295*4882a593Smuzhiyun
1296*4882a593Smuzhiyun printk(KERN_INFO "adapter at PCI %d:%d:%d, IRQ %d\n",
1297*4882a593Smuzhiyun pcidev->bus->number, pcidev->devfn >> 3, pcidev->devfn & 7,
1298*4882a593Smuzhiyun pcidev->irq);
1299*4882a593Smuzhiyun
1300*4882a593Smuzhiyun pci_set_master(pcidev);
1301*4882a593Smuzhiyun
1302*4882a593Smuzhiyun /* Enable 64bit DMA if possible */
1303*4882a593Smuzhiyun iop_ops = (struct hptiop_adapter_ops *)id->driver_data;
1304*4882a593Smuzhiyun rc = dma_set_mask(&pcidev->dev,
1305*4882a593Smuzhiyun DMA_BIT_MASK(iop_ops->hw_dma_bit_mask));
1306*4882a593Smuzhiyun if (rc)
1307*4882a593Smuzhiyun rc = dma_set_mask(&pcidev->dev, DMA_BIT_MASK(32));
1308*4882a593Smuzhiyun
1309*4882a593Smuzhiyun if (rc) {
1310*4882a593Smuzhiyun printk(KERN_ERR "hptiop: fail to set dma_mask\n");
1311*4882a593Smuzhiyun goto disable_pci_device;
1312*4882a593Smuzhiyun }
1313*4882a593Smuzhiyun
1314*4882a593Smuzhiyun if (pci_request_regions(pcidev, driver_name)) {
1315*4882a593Smuzhiyun printk(KERN_ERR "hptiop: pci_request_regions failed\n");
1316*4882a593Smuzhiyun goto disable_pci_device;
1317*4882a593Smuzhiyun }
1318*4882a593Smuzhiyun
1319*4882a593Smuzhiyun host = scsi_host_alloc(&driver_template, sizeof(struct hptiop_hba));
1320*4882a593Smuzhiyun if (!host) {
1321*4882a593Smuzhiyun printk(KERN_ERR "hptiop: fail to alloc scsi host\n");
1322*4882a593Smuzhiyun goto free_pci_regions;
1323*4882a593Smuzhiyun }
1324*4882a593Smuzhiyun
1325*4882a593Smuzhiyun hba = (struct hptiop_hba *)host->hostdata;
1326*4882a593Smuzhiyun memset(hba, 0, sizeof(struct hptiop_hba));
1327*4882a593Smuzhiyun
1328*4882a593Smuzhiyun hba->ops = iop_ops;
1329*4882a593Smuzhiyun hba->pcidev = pcidev;
1330*4882a593Smuzhiyun hba->host = host;
1331*4882a593Smuzhiyun hba->initialized = 0;
1332*4882a593Smuzhiyun hba->iopintf_v2 = 0;
1333*4882a593Smuzhiyun
1334*4882a593Smuzhiyun atomic_set(&hba->resetting, 0);
1335*4882a593Smuzhiyun atomic_set(&hba->reset_count, 0);
1336*4882a593Smuzhiyun
1337*4882a593Smuzhiyun init_waitqueue_head(&hba->reset_wq);
1338*4882a593Smuzhiyun init_waitqueue_head(&hba->ioctl_wq);
1339*4882a593Smuzhiyun
1340*4882a593Smuzhiyun host->max_lun = 128;
1341*4882a593Smuzhiyun host->max_channel = 0;
1342*4882a593Smuzhiyun host->io_port = 0;
1343*4882a593Smuzhiyun host->n_io_port = 0;
1344*4882a593Smuzhiyun host->irq = pcidev->irq;
1345*4882a593Smuzhiyun
1346*4882a593Smuzhiyun if (hba->ops->map_pci_bar(hba))
1347*4882a593Smuzhiyun goto free_scsi_host;
1348*4882a593Smuzhiyun
1349*4882a593Smuzhiyun if (hba->ops->iop_wait_ready(hba, 20000)) {
1350*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: firmware not ready\n",
1351*4882a593Smuzhiyun hba->host->host_no);
1352*4882a593Smuzhiyun goto unmap_pci_bar;
1353*4882a593Smuzhiyun }
1354*4882a593Smuzhiyun
1355*4882a593Smuzhiyun if (hba->ops->family == MV_BASED_IOP) {
1356*4882a593Smuzhiyun if (hba->ops->internal_memalloc(hba)) {
1357*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: internal_memalloc failed\n",
1358*4882a593Smuzhiyun hba->host->host_no);
1359*4882a593Smuzhiyun goto unmap_pci_bar;
1360*4882a593Smuzhiyun }
1361*4882a593Smuzhiyun }
1362*4882a593Smuzhiyun
1363*4882a593Smuzhiyun if (hba->ops->get_config(hba, &iop_config)) {
1364*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: get config failed\n",
1365*4882a593Smuzhiyun hba->host->host_no);
1366*4882a593Smuzhiyun goto unmap_pci_bar;
1367*4882a593Smuzhiyun }
1368*4882a593Smuzhiyun
1369*4882a593Smuzhiyun hba->max_requests = min(le32_to_cpu(iop_config.max_requests),
1370*4882a593Smuzhiyun HPTIOP_MAX_REQUESTS);
1371*4882a593Smuzhiyun hba->max_devices = le32_to_cpu(iop_config.max_devices);
1372*4882a593Smuzhiyun hba->max_request_size = le32_to_cpu(iop_config.request_size);
1373*4882a593Smuzhiyun hba->max_sg_descriptors = le32_to_cpu(iop_config.max_sg_count);
1374*4882a593Smuzhiyun hba->firmware_version = le32_to_cpu(iop_config.firmware_version);
1375*4882a593Smuzhiyun hba->interface_version = le32_to_cpu(iop_config.interface_version);
1376*4882a593Smuzhiyun hba->sdram_size = le32_to_cpu(iop_config.sdram_size);
1377*4882a593Smuzhiyun
1378*4882a593Smuzhiyun if (hba->ops->family == MVFREY_BASED_IOP) {
1379*4882a593Smuzhiyun if (hba->ops->internal_memalloc(hba)) {
1380*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: internal_memalloc failed\n",
1381*4882a593Smuzhiyun hba->host->host_no);
1382*4882a593Smuzhiyun goto unmap_pci_bar;
1383*4882a593Smuzhiyun }
1384*4882a593Smuzhiyun if (hba->ops->reset_comm(hba)) {
1385*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: reset comm failed\n",
1386*4882a593Smuzhiyun hba->host->host_no);
1387*4882a593Smuzhiyun goto unmap_pci_bar;
1388*4882a593Smuzhiyun }
1389*4882a593Smuzhiyun }
1390*4882a593Smuzhiyun
1391*4882a593Smuzhiyun if (hba->firmware_version > 0x01020000 ||
1392*4882a593Smuzhiyun hba->interface_version > 0x01020000)
1393*4882a593Smuzhiyun hba->iopintf_v2 = 1;
1394*4882a593Smuzhiyun
1395*4882a593Smuzhiyun host->max_sectors = le32_to_cpu(iop_config.data_transfer_length) >> 9;
1396*4882a593Smuzhiyun host->max_id = le32_to_cpu(iop_config.max_devices);
1397*4882a593Smuzhiyun host->sg_tablesize = le32_to_cpu(iop_config.max_sg_count);
1398*4882a593Smuzhiyun host->can_queue = le32_to_cpu(iop_config.max_requests);
1399*4882a593Smuzhiyun host->cmd_per_lun = le32_to_cpu(iop_config.max_requests);
1400*4882a593Smuzhiyun host->max_cmd_len = 16;
1401*4882a593Smuzhiyun
1402*4882a593Smuzhiyun req_size = sizeof(struct hpt_iop_request_scsi_command)
1403*4882a593Smuzhiyun + sizeof(struct hpt_iopsg) * (hba->max_sg_descriptors - 1);
1404*4882a593Smuzhiyun if ((req_size & 0x1f) != 0)
1405*4882a593Smuzhiyun req_size = (req_size + 0x1f) & ~0x1f;
1406*4882a593Smuzhiyun
1407*4882a593Smuzhiyun memset(&set_config, 0, sizeof(struct hpt_iop_request_set_config));
1408*4882a593Smuzhiyun set_config.iop_id = cpu_to_le32(host->host_no);
1409*4882a593Smuzhiyun set_config.vbus_id = cpu_to_le16(host->host_no);
1410*4882a593Smuzhiyun set_config.max_host_request_size = cpu_to_le16(req_size);
1411*4882a593Smuzhiyun
1412*4882a593Smuzhiyun if (hba->ops->set_config(hba, &set_config)) {
1413*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: set config failed\n",
1414*4882a593Smuzhiyun hba->host->host_no);
1415*4882a593Smuzhiyun goto unmap_pci_bar;
1416*4882a593Smuzhiyun }
1417*4882a593Smuzhiyun
1418*4882a593Smuzhiyun pci_set_drvdata(pcidev, host);
1419*4882a593Smuzhiyun
1420*4882a593Smuzhiyun if (request_irq(pcidev->irq, hptiop_intr, IRQF_SHARED,
1421*4882a593Smuzhiyun driver_name, hba)) {
1422*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: request irq %d failed\n",
1423*4882a593Smuzhiyun hba->host->host_no, pcidev->irq);
1424*4882a593Smuzhiyun goto unmap_pci_bar;
1425*4882a593Smuzhiyun }
1426*4882a593Smuzhiyun
1427*4882a593Smuzhiyun /* Allocate request mem */
1428*4882a593Smuzhiyun
1429*4882a593Smuzhiyun dprintk("req_size=%d, max_requests=%d\n", req_size, hba->max_requests);
1430*4882a593Smuzhiyun
1431*4882a593Smuzhiyun hba->req_size = req_size;
1432*4882a593Smuzhiyun hba->req_list = NULL;
1433*4882a593Smuzhiyun
1434*4882a593Smuzhiyun for (i = 0; i < hba->max_requests; i++) {
1435*4882a593Smuzhiyun start_virt = dma_alloc_coherent(&pcidev->dev,
1436*4882a593Smuzhiyun hba->req_size + 0x20,
1437*4882a593Smuzhiyun &start_phy, GFP_KERNEL);
1438*4882a593Smuzhiyun
1439*4882a593Smuzhiyun if (!start_virt) {
1440*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: fail to alloc request mem\n",
1441*4882a593Smuzhiyun hba->host->host_no);
1442*4882a593Smuzhiyun goto free_request_mem;
1443*4882a593Smuzhiyun }
1444*4882a593Smuzhiyun
1445*4882a593Smuzhiyun hba->dma_coherent[i] = start_virt;
1446*4882a593Smuzhiyun hba->dma_coherent_handle[i] = start_phy;
1447*4882a593Smuzhiyun
1448*4882a593Smuzhiyun if ((start_phy & 0x1f) != 0) {
1449*4882a593Smuzhiyun offset = ((start_phy + 0x1f) & ~0x1f) - start_phy;
1450*4882a593Smuzhiyun start_phy += offset;
1451*4882a593Smuzhiyun start_virt += offset;
1452*4882a593Smuzhiyun }
1453*4882a593Smuzhiyun
1454*4882a593Smuzhiyun hba->reqs[i].next = NULL;
1455*4882a593Smuzhiyun hba->reqs[i].req_virt = start_virt;
1456*4882a593Smuzhiyun hba->reqs[i].req_shifted_phy = start_phy >> 5;
1457*4882a593Smuzhiyun hba->reqs[i].index = i;
1458*4882a593Smuzhiyun free_req(hba, &hba->reqs[i]);
1459*4882a593Smuzhiyun }
1460*4882a593Smuzhiyun
1461*4882a593Smuzhiyun /* Enable Interrupt and start background task */
1462*4882a593Smuzhiyun if (hptiop_initialize_iop(hba))
1463*4882a593Smuzhiyun goto free_request_mem;
1464*4882a593Smuzhiyun
1465*4882a593Smuzhiyun if (scsi_add_host(host, &pcidev->dev)) {
1466*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: scsi_add_host failed\n",
1467*4882a593Smuzhiyun hba->host->host_no);
1468*4882a593Smuzhiyun goto free_request_mem;
1469*4882a593Smuzhiyun }
1470*4882a593Smuzhiyun
1471*4882a593Smuzhiyun scsi_scan_host(host);
1472*4882a593Smuzhiyun
1473*4882a593Smuzhiyun dprintk("scsi%d: hptiop_probe successfully\n", hba->host->host_no);
1474*4882a593Smuzhiyun return 0;
1475*4882a593Smuzhiyun
1476*4882a593Smuzhiyun free_request_mem:
1477*4882a593Smuzhiyun for (i = 0; i < hba->max_requests; i++) {
1478*4882a593Smuzhiyun if (hba->dma_coherent[i] && hba->dma_coherent_handle[i])
1479*4882a593Smuzhiyun dma_free_coherent(&hba->pcidev->dev,
1480*4882a593Smuzhiyun hba->req_size + 0x20,
1481*4882a593Smuzhiyun hba->dma_coherent[i],
1482*4882a593Smuzhiyun hba->dma_coherent_handle[i]);
1483*4882a593Smuzhiyun else
1484*4882a593Smuzhiyun break;
1485*4882a593Smuzhiyun }
1486*4882a593Smuzhiyun
1487*4882a593Smuzhiyun free_irq(hba->pcidev->irq, hba);
1488*4882a593Smuzhiyun
1489*4882a593Smuzhiyun unmap_pci_bar:
1490*4882a593Smuzhiyun hba->ops->internal_memfree(hba);
1491*4882a593Smuzhiyun
1492*4882a593Smuzhiyun hba->ops->unmap_pci_bar(hba);
1493*4882a593Smuzhiyun
1494*4882a593Smuzhiyun free_scsi_host:
1495*4882a593Smuzhiyun scsi_host_put(host);
1496*4882a593Smuzhiyun
1497*4882a593Smuzhiyun free_pci_regions:
1498*4882a593Smuzhiyun pci_release_regions(pcidev);
1499*4882a593Smuzhiyun
1500*4882a593Smuzhiyun disable_pci_device:
1501*4882a593Smuzhiyun pci_disable_device(pcidev);
1502*4882a593Smuzhiyun
1503*4882a593Smuzhiyun dprintk("scsi%d: hptiop_probe fail\n", host ? host->host_no : 0);
1504*4882a593Smuzhiyun return -ENODEV;
1505*4882a593Smuzhiyun }
1506*4882a593Smuzhiyun
hptiop_shutdown(struct pci_dev * pcidev)1507*4882a593Smuzhiyun static void hptiop_shutdown(struct pci_dev *pcidev)
1508*4882a593Smuzhiyun {
1509*4882a593Smuzhiyun struct Scsi_Host *host = pci_get_drvdata(pcidev);
1510*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)host->hostdata;
1511*4882a593Smuzhiyun
1512*4882a593Smuzhiyun dprintk("hptiop_shutdown(%p)\n", hba);
1513*4882a593Smuzhiyun
1514*4882a593Smuzhiyun /* stop the iop */
1515*4882a593Smuzhiyun if (iop_send_sync_msg(hba, IOPMU_INBOUND_MSG0_SHUTDOWN, 60000))
1516*4882a593Smuzhiyun printk(KERN_ERR "scsi%d: shutdown the iop timeout\n",
1517*4882a593Smuzhiyun hba->host->host_no);
1518*4882a593Smuzhiyun
1519*4882a593Smuzhiyun /* disable all outbound interrupts */
1520*4882a593Smuzhiyun hba->ops->disable_intr(hba);
1521*4882a593Smuzhiyun }
1522*4882a593Smuzhiyun
hptiop_disable_intr_itl(struct hptiop_hba * hba)1523*4882a593Smuzhiyun static void hptiop_disable_intr_itl(struct hptiop_hba *hba)
1524*4882a593Smuzhiyun {
1525*4882a593Smuzhiyun u32 int_mask;
1526*4882a593Smuzhiyun
1527*4882a593Smuzhiyun int_mask = readl(&hba->u.itl.iop->outbound_intmask);
1528*4882a593Smuzhiyun writel(int_mask |
1529*4882a593Smuzhiyun IOPMU_OUTBOUND_INT_MSG0 | IOPMU_OUTBOUND_INT_POSTQUEUE,
1530*4882a593Smuzhiyun &hba->u.itl.iop->outbound_intmask);
1531*4882a593Smuzhiyun readl(&hba->u.itl.iop->outbound_intmask);
1532*4882a593Smuzhiyun }
1533*4882a593Smuzhiyun
hptiop_disable_intr_mv(struct hptiop_hba * hba)1534*4882a593Smuzhiyun static void hptiop_disable_intr_mv(struct hptiop_hba *hba)
1535*4882a593Smuzhiyun {
1536*4882a593Smuzhiyun writel(0, &hba->u.mv.regs->outbound_intmask);
1537*4882a593Smuzhiyun readl(&hba->u.mv.regs->outbound_intmask);
1538*4882a593Smuzhiyun }
1539*4882a593Smuzhiyun
hptiop_disable_intr_mvfrey(struct hptiop_hba * hba)1540*4882a593Smuzhiyun static void hptiop_disable_intr_mvfrey(struct hptiop_hba *hba)
1541*4882a593Smuzhiyun {
1542*4882a593Smuzhiyun writel(0, &(hba->u.mvfrey.mu->f0_doorbell_enable));
1543*4882a593Smuzhiyun readl(&(hba->u.mvfrey.mu->f0_doorbell_enable));
1544*4882a593Smuzhiyun writel(0, &(hba->u.mvfrey.mu->isr_enable));
1545*4882a593Smuzhiyun readl(&(hba->u.mvfrey.mu->isr_enable));
1546*4882a593Smuzhiyun writel(0, &(hba->u.mvfrey.mu->pcie_f0_int_enable));
1547*4882a593Smuzhiyun readl(&(hba->u.mvfrey.mu->pcie_f0_int_enable));
1548*4882a593Smuzhiyun }
1549*4882a593Smuzhiyun
hptiop_remove(struct pci_dev * pcidev)1550*4882a593Smuzhiyun static void hptiop_remove(struct pci_dev *pcidev)
1551*4882a593Smuzhiyun {
1552*4882a593Smuzhiyun struct Scsi_Host *host = pci_get_drvdata(pcidev);
1553*4882a593Smuzhiyun struct hptiop_hba *hba = (struct hptiop_hba *)host->hostdata;
1554*4882a593Smuzhiyun u32 i;
1555*4882a593Smuzhiyun
1556*4882a593Smuzhiyun dprintk("scsi%d: hptiop_remove\n", hba->host->host_no);
1557*4882a593Smuzhiyun
1558*4882a593Smuzhiyun scsi_remove_host(host);
1559*4882a593Smuzhiyun
1560*4882a593Smuzhiyun hptiop_shutdown(pcidev);
1561*4882a593Smuzhiyun
1562*4882a593Smuzhiyun free_irq(hba->pcidev->irq, hba);
1563*4882a593Smuzhiyun
1564*4882a593Smuzhiyun for (i = 0; i < hba->max_requests; i++) {
1565*4882a593Smuzhiyun if (hba->dma_coherent[i] && hba->dma_coherent_handle[i])
1566*4882a593Smuzhiyun dma_free_coherent(&hba->pcidev->dev,
1567*4882a593Smuzhiyun hba->req_size + 0x20,
1568*4882a593Smuzhiyun hba->dma_coherent[i],
1569*4882a593Smuzhiyun hba->dma_coherent_handle[i]);
1570*4882a593Smuzhiyun else
1571*4882a593Smuzhiyun break;
1572*4882a593Smuzhiyun }
1573*4882a593Smuzhiyun
1574*4882a593Smuzhiyun hba->ops->internal_memfree(hba);
1575*4882a593Smuzhiyun
1576*4882a593Smuzhiyun hba->ops->unmap_pci_bar(hba);
1577*4882a593Smuzhiyun
1578*4882a593Smuzhiyun pci_release_regions(hba->pcidev);
1579*4882a593Smuzhiyun pci_set_drvdata(hba->pcidev, NULL);
1580*4882a593Smuzhiyun pci_disable_device(hba->pcidev);
1581*4882a593Smuzhiyun
1582*4882a593Smuzhiyun scsi_host_put(host);
1583*4882a593Smuzhiyun }
1584*4882a593Smuzhiyun
1585*4882a593Smuzhiyun static struct hptiop_adapter_ops hptiop_itl_ops = {
1586*4882a593Smuzhiyun .family = INTEL_BASED_IOP,
1587*4882a593Smuzhiyun .iop_wait_ready = iop_wait_ready_itl,
1588*4882a593Smuzhiyun .internal_memalloc = hptiop_internal_memalloc_itl,
1589*4882a593Smuzhiyun .internal_memfree = hptiop_internal_memfree_itl,
1590*4882a593Smuzhiyun .map_pci_bar = hptiop_map_pci_bar_itl,
1591*4882a593Smuzhiyun .unmap_pci_bar = hptiop_unmap_pci_bar_itl,
1592*4882a593Smuzhiyun .enable_intr = hptiop_enable_intr_itl,
1593*4882a593Smuzhiyun .disable_intr = hptiop_disable_intr_itl,
1594*4882a593Smuzhiyun .get_config = iop_get_config_itl,
1595*4882a593Smuzhiyun .set_config = iop_set_config_itl,
1596*4882a593Smuzhiyun .iop_intr = iop_intr_itl,
1597*4882a593Smuzhiyun .post_msg = hptiop_post_msg_itl,
1598*4882a593Smuzhiyun .post_req = hptiop_post_req_itl,
1599*4882a593Smuzhiyun .hw_dma_bit_mask = 64,
1600*4882a593Smuzhiyun .reset_comm = hptiop_reset_comm_itl,
1601*4882a593Smuzhiyun .host_phy_flag = cpu_to_le64(0),
1602*4882a593Smuzhiyun };
1603*4882a593Smuzhiyun
1604*4882a593Smuzhiyun static struct hptiop_adapter_ops hptiop_mv_ops = {
1605*4882a593Smuzhiyun .family = MV_BASED_IOP,
1606*4882a593Smuzhiyun .iop_wait_ready = iop_wait_ready_mv,
1607*4882a593Smuzhiyun .internal_memalloc = hptiop_internal_memalloc_mv,
1608*4882a593Smuzhiyun .internal_memfree = hptiop_internal_memfree_mv,
1609*4882a593Smuzhiyun .map_pci_bar = hptiop_map_pci_bar_mv,
1610*4882a593Smuzhiyun .unmap_pci_bar = hptiop_unmap_pci_bar_mv,
1611*4882a593Smuzhiyun .enable_intr = hptiop_enable_intr_mv,
1612*4882a593Smuzhiyun .disable_intr = hptiop_disable_intr_mv,
1613*4882a593Smuzhiyun .get_config = iop_get_config_mv,
1614*4882a593Smuzhiyun .set_config = iop_set_config_mv,
1615*4882a593Smuzhiyun .iop_intr = iop_intr_mv,
1616*4882a593Smuzhiyun .post_msg = hptiop_post_msg_mv,
1617*4882a593Smuzhiyun .post_req = hptiop_post_req_mv,
1618*4882a593Smuzhiyun .hw_dma_bit_mask = 33,
1619*4882a593Smuzhiyun .reset_comm = hptiop_reset_comm_mv,
1620*4882a593Smuzhiyun .host_phy_flag = cpu_to_le64(0),
1621*4882a593Smuzhiyun };
1622*4882a593Smuzhiyun
1623*4882a593Smuzhiyun static struct hptiop_adapter_ops hptiop_mvfrey_ops = {
1624*4882a593Smuzhiyun .family = MVFREY_BASED_IOP,
1625*4882a593Smuzhiyun .iop_wait_ready = iop_wait_ready_mvfrey,
1626*4882a593Smuzhiyun .internal_memalloc = hptiop_internal_memalloc_mvfrey,
1627*4882a593Smuzhiyun .internal_memfree = hptiop_internal_memfree_mvfrey,
1628*4882a593Smuzhiyun .map_pci_bar = hptiop_map_pci_bar_mvfrey,
1629*4882a593Smuzhiyun .unmap_pci_bar = hptiop_unmap_pci_bar_mvfrey,
1630*4882a593Smuzhiyun .enable_intr = hptiop_enable_intr_mvfrey,
1631*4882a593Smuzhiyun .disable_intr = hptiop_disable_intr_mvfrey,
1632*4882a593Smuzhiyun .get_config = iop_get_config_mvfrey,
1633*4882a593Smuzhiyun .set_config = iop_set_config_mvfrey,
1634*4882a593Smuzhiyun .iop_intr = iop_intr_mvfrey,
1635*4882a593Smuzhiyun .post_msg = hptiop_post_msg_mvfrey,
1636*4882a593Smuzhiyun .post_req = hptiop_post_req_mvfrey,
1637*4882a593Smuzhiyun .hw_dma_bit_mask = 64,
1638*4882a593Smuzhiyun .reset_comm = hptiop_reset_comm_mvfrey,
1639*4882a593Smuzhiyun .host_phy_flag = cpu_to_le64(1),
1640*4882a593Smuzhiyun };
1641*4882a593Smuzhiyun
1642*4882a593Smuzhiyun static struct pci_device_id hptiop_id_table[] = {
1643*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3220), (kernel_ulong_t)&hptiop_itl_ops },
1644*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3320), (kernel_ulong_t)&hptiop_itl_ops },
1645*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3410), (kernel_ulong_t)&hptiop_itl_ops },
1646*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3510), (kernel_ulong_t)&hptiop_itl_ops },
1647*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3511), (kernel_ulong_t)&hptiop_itl_ops },
1648*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3520), (kernel_ulong_t)&hptiop_itl_ops },
1649*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3521), (kernel_ulong_t)&hptiop_itl_ops },
1650*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3522), (kernel_ulong_t)&hptiop_itl_ops },
1651*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3530), (kernel_ulong_t)&hptiop_itl_ops },
1652*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3540), (kernel_ulong_t)&hptiop_itl_ops },
1653*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3560), (kernel_ulong_t)&hptiop_itl_ops },
1654*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4210), (kernel_ulong_t)&hptiop_itl_ops },
1655*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4211), (kernel_ulong_t)&hptiop_itl_ops },
1656*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4310), (kernel_ulong_t)&hptiop_itl_ops },
1657*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4311), (kernel_ulong_t)&hptiop_itl_ops },
1658*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4320), (kernel_ulong_t)&hptiop_itl_ops },
1659*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4321), (kernel_ulong_t)&hptiop_itl_ops },
1660*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4322), (kernel_ulong_t)&hptiop_itl_ops },
1661*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4400), (kernel_ulong_t)&hptiop_itl_ops },
1662*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3120), (kernel_ulong_t)&hptiop_mv_ops },
1663*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3122), (kernel_ulong_t)&hptiop_mv_ops },
1664*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3020), (kernel_ulong_t)&hptiop_mv_ops },
1665*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4520), (kernel_ulong_t)&hptiop_mvfrey_ops },
1666*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x4522), (kernel_ulong_t)&hptiop_mvfrey_ops },
1667*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3610), (kernel_ulong_t)&hptiop_mvfrey_ops },
1668*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3611), (kernel_ulong_t)&hptiop_mvfrey_ops },
1669*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3620), (kernel_ulong_t)&hptiop_mvfrey_ops },
1670*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3622), (kernel_ulong_t)&hptiop_mvfrey_ops },
1671*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3640), (kernel_ulong_t)&hptiop_mvfrey_ops },
1672*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3660), (kernel_ulong_t)&hptiop_mvfrey_ops },
1673*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3680), (kernel_ulong_t)&hptiop_mvfrey_ops },
1674*4882a593Smuzhiyun { PCI_VDEVICE(TTI, 0x3690), (kernel_ulong_t)&hptiop_mvfrey_ops },
1675*4882a593Smuzhiyun {},
1676*4882a593Smuzhiyun };
1677*4882a593Smuzhiyun
1678*4882a593Smuzhiyun MODULE_DEVICE_TABLE(pci, hptiop_id_table);
1679*4882a593Smuzhiyun
1680*4882a593Smuzhiyun static struct pci_driver hptiop_pci_driver = {
1681*4882a593Smuzhiyun .name = driver_name,
1682*4882a593Smuzhiyun .id_table = hptiop_id_table,
1683*4882a593Smuzhiyun .probe = hptiop_probe,
1684*4882a593Smuzhiyun .remove = hptiop_remove,
1685*4882a593Smuzhiyun .shutdown = hptiop_shutdown,
1686*4882a593Smuzhiyun };
1687*4882a593Smuzhiyun
hptiop_module_init(void)1688*4882a593Smuzhiyun static int __init hptiop_module_init(void)
1689*4882a593Smuzhiyun {
1690*4882a593Smuzhiyun printk(KERN_INFO "%s %s\n", driver_name_long, driver_ver);
1691*4882a593Smuzhiyun return pci_register_driver(&hptiop_pci_driver);
1692*4882a593Smuzhiyun }
1693*4882a593Smuzhiyun
hptiop_module_exit(void)1694*4882a593Smuzhiyun static void __exit hptiop_module_exit(void)
1695*4882a593Smuzhiyun {
1696*4882a593Smuzhiyun pci_unregister_driver(&hptiop_pci_driver);
1697*4882a593Smuzhiyun }
1698*4882a593Smuzhiyun
1699*4882a593Smuzhiyun
1700*4882a593Smuzhiyun module_init(hptiop_module_init);
1701*4882a593Smuzhiyun module_exit(hptiop_module_exit);
1702*4882a593Smuzhiyun
1703*4882a593Smuzhiyun MODULE_LICENSE("GPL");
1704*4882a593Smuzhiyun
1705