xref: /OK3568_Linux_fs/kernel/drivers/scsi/hptiop.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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