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