3 * Bill Paul <wpaul@windriver.com>. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by Bill Paul.
16 * 4. Neither the name of the author nor the names of any co-contributors
17 * may be used to endorse or promote products derived from this software
18 * without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY Bill Paul AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL Bill Paul OR THE VOICES IN HIS HEAD
24 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
30 * THE POSSIBILITY OF SUCH DAMAGE.
33 #include <sys/cdefs.h>
34 __KERNEL_RCSID(0, "$NetBSD: if_ndis_pci.c,v 1.16 2009/08/02 20:22:34 dsl Exp $");
36 __FBSDID("$FreeBSD: src/sys/dev/if_ndis/if_ndis_pci.c,v 1.8.2.3 2005/03/31 04:24:36 wpaul Exp $");
39 #include <sys/param.h>
40 #include <sys/systm.h>
41 #include <sys/kernel.h>
42 #include <sys/module.h>
43 #include <sys/socket.h>
44 #include <sys/queue.h>
45 #include <sys/sysctl.h>
48 #include <net/if_arp.h>
49 #include <net/if_media.h>
53 #include <sys/kthread.h>
54 #include <net/if_ether.h>
56 #include <net80211/ieee80211_var.h>
58 #include <dev/pci/pcireg.h>
59 #include <dev/pci/pcivar.h>
61 #include <compat/ndis/pe_var.h>
62 #include <compat/ndis/resource_var.h>
63 #include <compat/ndis/ntoskrnl_var.h>
64 #include <compat/ndis/ndis_var.h>
65 #include <compat/ndis/cfg_var.h>
66 #include <dev/if_ndis/if_ndisvar.h>
68 #include "ndis_driver_data.h"
71 #include <compat/ndis/hal_var.h>
74 #ifdef NDIS_PCI_DEV_TABLE
78 * Various supported device vendors/types and their names.
79 * These are defined in the ndis_driver_data.h file.
81 static struct ndis_pci_type ndis_devs
[] = {
82 #ifdef NDIS_PCI_DEV_TABLE
88 /*static*/ int ndis_probe_pci(device_t parent
,
91 /*static*/ void ndis_attach_pci(device_t parent
,
94 extern void ndis_attach (void *);
95 extern int ndis_shutdown (device_t
);
96 extern int ndis_detach (device_t
, int);
97 extern int ndis_suspend (device_t
);
98 extern int ndis_resume (device_t
);
100 extern int ndis_intr(void *);
102 extern unsigned char drv_data
[];
105 //static funcptr ndis_txeof_wrap;
106 //static funcptr ndis_rxeof_wrap;
107 //static funcptr ndis_linksts_wrap;
108 //static funcptr ndis_linksts_done_wrap;
118 sizeof(struct ndis_softc
),
128 ndisdrv_modevent(module_t mod
, int cmd
);
131 * These are just for the in-kernel version, to delay calling
132 * these functions untill enough context is built up.
134 void load_ndisapi(void *);
135 void load_ndisdrv(void *);
137 void load_ndisapi(void *arg
)
139 ndis_lkm_handle(NULL
, MOD_LOAD
);
141 void load_ndisdrv(void *arg
)
143 ndisdrv_modevent(NULL
, MOD_LOAD
);
148 ndis_probe_pci(device_t parent
, cfdata_t match
, void *aux
)
150 struct pci_attach_args
*pa
= aux
;
151 int vendor
= PCI_VENDOR(pa
->pa_id
);
152 int product
= PCI_PRODUCT(pa
->pa_id
);
154 struct ndis_pci_type
*t
= ndis_devs
;
155 driver_object
*drv
= NULL
; /* = windrv_lookup(0, "PCI Bus");**/
158 printf("in ndis_probe_pci\n");
159 printf("vendor = %x, product = %x\n", vendor
, product
);
162 while(t
->ndis_name
!= NULL
) {
164 printf("t->ndis_vid = %x, t->ndis_did = %x\n",
165 t
->ndis_vid
, t
->ndis_did
);
167 if((vendor
== t
->ndis_vid
) && (product
== t
->ndis_did
)) {
169 ndisdrv_modevent(NULL
, MOD_LOAD
);
170 //kthread_create(load_ndisdrv, NULL);
173 drv
= windrv_lookup(0, "PCI Bus");
174 printf("Matching vendor: %x, product: %x, name: %s\n", vendor
, product
, t
->ndis_name
);
175 windrv_create_pdo(drv
, parent
);
181 return 0; /* dosen't match */
184 /* 6 BADR's + 1 IRQ (so far) */
185 #define MAX_RESOURCES 7
188 void ndis_attach_pci(device_t parent
, device_t self
, void *aux
)
190 struct ndis_softc
*sc
= (struct ndis_softc
*)self
;
191 struct pci_attach_args
*pa
= aux
;
195 pci_intr_handle_t ih
;
200 ndis_resource_list
*rl
= NULL
;
201 struct cm_partial_resource_desc
*prd
= NULL
;
203 struct pci_conf_state conf_state
;
208 printf("in ndis_attach_pci()\n");
210 /* initalize the softc */
211 //sc->ndis_hardware_type = NDIS_PCI;
213 sc
->ndis_iftype
= PCIBus
;
214 sc
->ndis_res_pc
= pa
->pa_pc
;
215 sc
->ndis_res_pctag
= pa
->pa_tag
;
216 /* TODO: is this correct? All are just pa->pa_dmat? */
217 sc
->ndis_mtag
= pa
->pa_dmat
;
218 sc
->ndis_ttag
= pa
->pa_dmat
;
219 sc
->ndis_parent_tag
= pa
->pa_dmat
;
220 sc
->ndis_res_io
= NULL
;
221 sc
->ndis_res_mem
= NULL
;
222 sc
->ndis_res_altmem
= NULL
;
223 sc
->ndis_block
= NULL
;
224 sc
->ndis_shlist
= NULL
;
228 printf("sc->ndis_mtag = %x\n", (unsigned int)sc
->ndis_mtag
);
230 rl
= malloc(sizeof(ndis_resource_list
) +
231 (sizeof(cm_partial_resource_desc
) * (MAX_RESOURCES
-1)),
232 M_DEVBUF
, M_NOWAIT
|M_ZERO
);
236 //printf("error: out of memory\n");
240 rl
->cprl_version
= 5;
241 rl
->cprl_version
= 1;
243 prd
= rl
->cprl_partial_descs
;
246 pci_devinfo(pa
->pa_id
, pa
->pa_class
, 0, devinfo
, sizeof devinfo
);
247 revision
= PCI_REVISION(pa
->pa_class
);
248 printf(": %s (rev. 0x%02x)\n", devinfo
, revision
);
250 pci_conf_print(sc
->ndis_res_pc
, sc
->ndis_res_pctag
, NULL
);
252 pci_conf_capture(sc
->ndis_res_pc
, sc
->ndis_res_pctag
, &conf_state
);
253 for(i
=0; i
<16; i
++) {
254 printf("conf_state.reg[%d] = %x\n", i
, conf_state
.reg
[i
]);
258 /* just do the conversion work in attach instead of calling ndis_convert_res() */
259 for(bar
= 0x10; bar
<= 0x24; bar
+= 0x04) {
260 type
= pci_mapreg_type(sc
->ndis_res_pc
, sc
->ndis_res_pctag
, bar
);
261 if(pci_mapreg_info(sc
->ndis_res_pc
, sc
->ndis_res_pctag
, bar
, type
, &base
,
263 printf("pci_mapreg_info() failed on BAR 0x%x!\n", bar
);
266 case PCI_MAPREG_TYPE_IO
:
267 prd
->cprd_type
= CmResourceTypePort
;
268 prd
->cprd_flags
= CM_RESOURCE_PORT_IO
;
269 prd
->u
.cprd_port
.cprd_start
.np_quad
= (uint64_t)base
;
270 prd
->u
.cprd_port
.cprd_len
= (uint32_t)size
;
271 if((sc
->ndis_res_io
=
272 malloc(sizeof(struct ndis_resource
), M_DEVBUF
, M_NOWAIT
| M_ZERO
)) == NULL
) {
273 //printf("error: out of memory\n");
277 sc
->ndis_res_io
->res_base
= base
;
278 sc
->ndis_res_io
->res_size
= size
;
279 sc
->ndis_res_io
->res_tag
= X86_BUS_SPACE_IO
;
280 bus_space_map(sc
->ndis_res_io
->res_tag
,
281 sc
->ndis_res_io
->res_base
,
282 sc
->ndis_res_io
->res_size
,
284 &sc
->ndis_res_io
->res_handle
);
286 case PCI_MAPREG_TYPE_MEM
:
287 prd
->cprd_type
= CmResourceTypeMemory
;
288 prd
->cprd_flags
= CM_RESOURCE_MEMORY_READ_WRITE
;
289 prd
->u
.cprd_mem
.cprd_start
.np_quad
= (uint64_t)base
;
290 prd
->u
.cprd_mem
.cprd_len
= (uint32_t)size
;
292 if(sc
->ndis_res_mem
!= NULL
&&
293 sc
->ndis_res_altmem
!= NULL
) {
294 printf("too many resources\n");
298 if(sc
->ndis_res_mem
) {
299 if((sc
->ndis_res_altmem
=
300 malloc(sizeof(struct ndis_resource
), M_DEVBUF
, M_NOWAIT
| M_ZERO
)) == NULL
) {
304 sc
->ndis_res_altmem
->res_base
= base
;
305 sc
->ndis_res_altmem
->res_size
= size
;
306 sc
->ndis_res_altmem
->res_tag
= X86_BUS_SPACE_MEM
;
309 if(bus_space_map(sc
->ndis_res_altmem
->res_tag
,
310 sc
->ndis_res_altmem
->res_base
,
311 sc
->ndis_res_altmem
->res_size
,
312 flags
|BUS_SPACE_MAP_LINEAR
,
313 &sc
->ndis_res_altmem
->res_handle
)) {
314 printf("bus_space_map failed\n");
317 if((sc
->ndis_res_mem
=
318 malloc(sizeof(struct ndis_resource
), M_DEVBUF
, M_NOWAIT
| M_ZERO
)) == NULL
) {
322 sc
->ndis_res_mem
->res_base
= base
;
323 sc
->ndis_res_mem
->res_size
= size
;
324 sc
->ndis_res_mem
->res_tag
= X86_BUS_SPACE_MEM
;
326 if(bus_space_map(sc
->ndis_res_mem
->res_tag
,
327 sc
->ndis_res_mem
->res_base
,
328 sc
->ndis_res_mem
->res_size
,
329 flags
|BUS_SPACE_MAP_LINEAR
,
330 &sc
->ndis_res_mem
->res_handle
)) {
331 printf("bus_space_map failed\n");
337 printf("unknown type\n");
339 prd
->cprd_sharedisp
= CmResourceShareDeviceExclusive
;
346 /* add the interrupt to the list */
347 prd
->cprd_type
= CmResourceTypeInterrupt
;
349 /* TODO: is this all we need to save for the interrupt? */
350 prd
->u
.cprd_intr
.cprd_level
= pa
->pa_intrline
;
351 prd
->u
.cprd_intr
.cprd_vector
= pa
->pa_intrline
;
352 prd
->u
.cprd_intr
.cprd_affinity
= 0;
355 pci_intr_map(pa
, &ih
);
356 sc
->ndis_intrhand
= pci_intr_establish(pa
->pa_pc
, ih
, IPL_NET
/*| PCATCH*/, ndis_intr
, sc
);
357 sc
->ndis_irq
= (void *)sc
->ndis_intrhand
;
359 printf("pci interrupt: %s\n", pci_intr_string(pa
->pa_pc
, ih
));
366 /* save resource list in the softc */
368 sc
->ndis_rescnt
= rl
->cprl_count
;
370 kthread_create(PRI_NONE
, 0, NULL
, ndis_attach
, (void *)sc
,
371 NULL
, "ndis_attach");
375 #endif /* NDIS_PCI_DEV_TABLE */