xref: /freebsd/sys/dev/ispfw/ispfw.c (revision b52b9d56d4e96089873a75f9e29062eec19fabba)
1 /* $FreeBSD$ */
2 /*
3  * ISP Firmware Helper Pseudo Device for FreeBSD
4  *
5  * Copyright (c) 2000, 2001, by Matthew Jacob
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice immediately at the beginning of the file, without modification,
13  *    this list of conditions, and the following disclaimer.
14  * 2. The name of the author may not be used to endorse or promote products
15  *    derived from this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
21  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  */
29 
30 #include <sys/param.h>
31 #include <sys/kernel.h>
32 #include <sys/malloc.h>
33 
34 #include <dev/ispfw/asm_1040.h>
35 #include <dev/ispfw/asm_1080.h>
36 #include <dev/ispfw/asm_12160.h>
37 #include <dev/ispfw/asm_2100.h>
38 #include <dev/ispfw/asm_2200.h>
39 #include <dev/ispfw/asm_2300.h>
40 #if	_MACHINE_ARCH == sparc64
41 #include <dev/ispfw/asm_1000.h>
42 #endif
43 
44 #define	ISPFW_VERSION	0
45 
46 #define	PCI_PRODUCT_QLOGIC_ISP1020	0x1020
47 #define	PCI_PRODUCT_QLOGIC_ISP1080	0x1080
48 #define	PCI_PRODUCT_QLOGIC_ISP12160	0x1216
49 #define	PCI_PRODUCT_QLOGIC_ISP1240	0x1240
50 #define	PCI_PRODUCT_QLOGIC_ISP1280	0x1280
51 #define	PCI_PRODUCT_QLOGIC_ISP2100	0x2100
52 #define	PCI_PRODUCT_QLOGIC_ISP2200	0x2200
53 #define	PCI_PRODUCT_QLOGIC_ISP2300	0x2300
54 #define	PCI_PRODUCT_QLOGIC_ISP2312	0x2312
55 #if	_MACHINE_ARCH == sparc64
56 #define	SBUS_PRODUCT_QLOGIC_ISP1000	0x1000
57 #endif
58 
59 typedef void ispfwfunc(int, int, int, const u_int16_t **);
60 extern ispfwfunc *isp_get_firmware_p;
61 static void isp_get_firmware(int, int, int, const u_int16_t **);
62 
63 static int ncallers = 0;
64 static const u_int16_t ***callp = NULL;
65 static int addcaller(const u_int16_t **);
66 
67 static int
68 addcaller(const u_int16_t **caller)
69 {
70 	const u_int16_t ***newcallp;
71 	int i;
72 	for (i = 0; i < ncallers; i++) {
73 		if (callp[i] == caller)
74 			return (1);
75 	}
76 	newcallp = malloc((ncallers + 1) * sizeof (const u_int16_t ***),
77 	    M_DEVBUF, M_NOWAIT);
78 	if (newcallp == NULL) {
79 		return (0);
80 	}
81 	for (i = 0; i < ncallers; i++) {
82 		newcallp[i] = callp[i];
83 	}
84 	newcallp[ncallers] = caller;
85 	if (ncallers++)
86 		free(callp, M_DEVBUF);
87 	callp = newcallp;
88 	return (1);
89 }
90 
91 static void
92 isp_get_firmware(int version, int tgtmode, int devid, const u_int16_t **ptrp)
93 {
94 	const u_int16_t *rp = NULL;
95 
96 	if (version == ISPFW_VERSION) {
97 		switch (devid) {
98 		case PCI_PRODUCT_QLOGIC_ISP1020:
99 			if (tgtmode)
100 				rp = isp_1040_risc_code_it;
101 			else
102 				rp = isp_1040_risc_code;
103 			break;
104 		case PCI_PRODUCT_QLOGIC_ISP1080:
105 		case PCI_PRODUCT_QLOGIC_ISP1240:
106 		case PCI_PRODUCT_QLOGIC_ISP1280:
107 			if (tgtmode)
108 				rp = isp_1080_risc_code_it;
109 			else
110 				rp = isp_1080_risc_code;
111 			break;
112 		case PCI_PRODUCT_QLOGIC_ISP12160:
113 			if (tgtmode)
114 				rp = isp_12160_risc_code_it;
115 			else
116 				rp = isp_12160_risc_code;
117 			break;
118 		case PCI_PRODUCT_QLOGIC_ISP2100:
119 			rp = isp_2100_risc_code;
120 			break;
121 		case PCI_PRODUCT_QLOGIC_ISP2200:
122 			rp = isp_2200_risc_code;
123 			break;
124 		case PCI_PRODUCT_QLOGIC_ISP2300:
125 		case PCI_PRODUCT_QLOGIC_ISP2312:
126 			rp = isp_2300_risc_code;
127 			break;
128 #if	_MACHINE_ARCH == sparc64
129 		case SBUS_PRODUCT_QLOGIC_ISP1000:
130 			if (tgtmode)
131 				break;
132 			rp = isp_1000_risc_code;
133 			break;
134 #endif
135 		default:
136 			break;
137 		}
138 	}
139 	if (rp && addcaller(ptrp)) {
140 		*ptrp = rp;
141 	}
142 }
143 
144 static int
145 isp_module_handler(module_t mod, int what, void *arg)
146 {
147 	switch (what) {
148 	case MOD_LOAD:
149 		isp_get_firmware_p = isp_get_firmware;
150 		break;
151 	case MOD_UNLOAD:
152 		isp_get_firmware_p = NULL;
153 		if (ncallers)  {
154 			int i;
155 			for (i = 0; i < ncallers; i++) {
156 				*callp[i] = NULL;
157 			}
158 			free(callp, M_DEVBUF);
159 		}
160 		break;
161 	default:
162 		break;
163 	}
164 	return (0);
165 }
166 static moduledata_t ispfw_mod = {
167 	"ispfw", isp_module_handler, NULL
168 };
169 DECLARE_MODULE(ispfw, ispfw_mod, SI_SUB_DRIVERS, SI_ORDER_THIRD);
170 MODULE_VERSION(ispfw, ISPFW_VERSION);
171 MODULE_DEPEND(ispfw, isp, 1, 1, 1);
172