xref: /freebsd/sys/dev/cardbus/cardbus.c (revision 7660b554bc59a07be0431c17e0e33815818baa69)
1 /*
2  * Copyright (c) 2003 M. Warner Losh.  All Rights Reserved.
3  * Copyright (c) 2000,2001 Jonathan Chen.  All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions, and the following disclaimer,
10  *    without modification, immediately at the beginning of the file.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in
13  *    the documentation and/or other materials provided with the
14  *    distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
20  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  *
28  */
29 
30 #include <sys/cdefs.h>
31 __FBSDID("$FreeBSD$");
32 
33 #include <sys/param.h>
34 #include <sys/systm.h>
35 #include <sys/malloc.h>
36 #include <sys/kernel.h>
37 #include <sys/sysctl.h>
38 
39 #include <sys/bus.h>
40 #include <machine/bus.h>
41 #include <sys/rman.h>
42 #include <machine/resource.h>
43 
44 #include <sys/pciio.h>
45 #include <dev/pci/pcivar.h>
46 #include <dev/pci/pcireg.h>
47 #include <dev/pci/pci_private.h>
48 
49 #include <dev/cardbus/cardbusreg.h>
50 #include <dev/cardbus/cardbusvar.h>
51 #include <dev/cardbus/cardbus_cis.h>
52 #include <dev/pccard/pccardvar.h>
53 
54 #include "power_if.h"
55 #include "pcib_if.h"
56 
57 /* sysctl vars */
58 SYSCTL_NODE(_hw, OID_AUTO, cardbus, CTLFLAG_RD, 0, "CardBus parameters");
59 
60 int    cardbus_debug = 0;
61 TUNABLE_INT("hw.cardbus.debug", &cardbus_debug);
62 SYSCTL_INT(_hw_cardbus, OID_AUTO, debug, CTLFLAG_RW,
63     &cardbus_debug, 0,
64   "CardBus debug");
65 
66 int    cardbus_cis_debug = 0;
67 TUNABLE_INT("hw.cardbus.cis_debug", &cardbus_cis_debug);
68 SYSCTL_INT(_hw_cardbus, OID_AUTO, cis_debug, CTLFLAG_RW,
69     &cardbus_cis_debug, 0,
70   "CardBus CIS debug");
71 
72 #define	DPRINTF(a) if (cardbus_debug) printf a
73 #define	DEVPRINTF(x) if (cardbus_debug) device_printf x
74 
75 
76 static int	cardbus_attach(device_t cbdev);
77 static int	cardbus_attach_card(device_t cbdev);
78 static int	cardbus_detach(device_t cbdev);
79 static int	cardbus_detach_card(device_t cbdev);
80 static void	cardbus_device_setup_regs(device_t brdev, int b, int s, int f,
81 		    pcicfgregs *cfg);
82 static void	cardbus_driver_added(device_t cbdev, driver_t *driver);
83 static int	cardbus_probe(device_t cbdev);
84 static int	cardbus_read_ivar(device_t cbdev, device_t child, int which,
85 		    uintptr_t *result);
86 static void	cardbus_release_all_resources(device_t cbdev,
87 		    struct cardbus_devinfo *dinfo);
88 static int	cardbus_write_ivar(device_t cbdev, device_t child, int which,
89 		    uintptr_t value);
90 
91 /************************************************************************/
92 /* Probe/Attach								*/
93 /************************************************************************/
94 
95 static int
96 cardbus_probe(device_t cbdev)
97 {
98 	device_set_desc(cbdev, "CardBus bus");
99 	return 0;
100 }
101 
102 static int
103 cardbus_attach(device_t cbdev)
104 {
105 	return 0;
106 }
107 
108 static int
109 cardbus_detach(device_t cbdev)
110 {
111 	cardbus_detach_card(cbdev);
112 	return 0;
113 }
114 
115 static int
116 cardbus_suspend(device_t self)
117 {
118 	cardbus_detach_card(self);
119 	return (0);
120 }
121 
122 static int
123 cardbus_resume(device_t self)
124 {
125 	return (0);
126 }
127 
128 /************************************************************************/
129 /* Attach/Detach card							*/
130 /************************************************************************/
131 
132 static void
133 cardbus_device_setup_regs(device_t brdev, int b, int s, int f, pcicfgregs *cfg)
134 {
135 	PCIB_WRITE_CONFIG(brdev, b, s, f, PCIR_INTLINE,
136 	    pci_get_irq(device_get_parent(brdev)), 1);
137 	cfg->intline = PCIB_READ_CONFIG(brdev, b, s, f, PCIR_INTLINE, 1);
138 
139 	PCIB_WRITE_CONFIG(brdev, b, s, f, PCIR_CACHELNSZ, 0x08, 1);
140 	cfg->cachelnsz = PCIB_READ_CONFIG(brdev, b, s, f, PCIR_CACHELNSZ, 1);
141 
142 	PCIB_WRITE_CONFIG(brdev, b, s, f, PCIR_LATTIMER, 0xa8, 1);
143 	cfg->lattimer = PCIB_READ_CONFIG(brdev, b, s, f, PCIR_LATTIMER, 1);
144 
145 	PCIB_WRITE_CONFIG(brdev, b, s, f, PCIR_MINGNT, 0x14, 1);
146 	cfg->mingnt = PCIB_READ_CONFIG(brdev, b, s, f, PCIR_MINGNT, 1);
147 
148 	PCIB_WRITE_CONFIG(brdev, b, s, f, PCIR_MAXLAT, 0x14, 1);
149 	cfg->maxlat = PCIB_READ_CONFIG(brdev, b, s, f, PCIR_MAXLAT, 1);
150 }
151 
152 static int
153 cardbus_attach_card(device_t cbdev)
154 {
155 	device_t brdev = device_get_parent(cbdev);
156 	int cardattached = 0;
157 	static int curr_bus_number = 2; /* XXX EVILE BAD (see below) */
158 	int bus, slot, func;
159 
160 	cardbus_detach_card(cbdev); /* detach existing cards */
161 
162 	POWER_ENABLE_SOCKET(brdev, cbdev);
163 	bus = pcib_get_bus(cbdev);
164 	if (bus == 0) {
165 		/*
166 		 * XXX EVILE BAD XXX
167 		 * Not all BIOSes initialize the secondary bus number properly,
168 		 * so if the default is bad, we just put one in and hope it
169 		 * works.
170 		 */
171 		bus = curr_bus_number;
172 		pci_write_config(brdev, PCIR_SECBUS_2, curr_bus_number, 1);
173 		pci_write_config(brdev, PCIR_SUBBUS_2, curr_bus_number + 2, 1);
174 		curr_bus_number += 3;
175 	}
176 	/* For each function, set it up and try to attach a driver to it */
177 	for (slot = 0; slot <= CARDBUS_SLOTMAX; slot++) {
178 		int cardbusfunchigh = 0;
179 		for (func = 0; func <= cardbusfunchigh; func++) {
180 			struct cardbus_devinfo *dinfo;
181 
182 			dinfo = (struct cardbus_devinfo *)
183 			    pci_read_device(brdev, bus, slot, func,
184 				sizeof(struct cardbus_devinfo));
185 			if (dinfo == NULL)
186 				continue;
187 			if (dinfo->pci.cfg.mfdev)
188 				cardbusfunchigh = CARDBUS_FUNCMAX;
189 
190 			cardbus_device_setup_regs(brdev, bus, slot, func,
191 			    &dinfo->pci.cfg);
192 			dinfo->pci.cfg.dev = device_add_child(cbdev, NULL, -1);
193 			if (!dinfo->pci.cfg.dev) {
194 				DEVPRINTF((cbdev, "Cannot add child!\n"));
195 				pci_freecfg((struct pci_devinfo *)dinfo);
196 				continue;
197 			}
198 			resource_list_init(&dinfo->pci.resources);
199 			device_set_ivars(dinfo->pci.cfg.dev, dinfo);
200 			cardbus_do_cis(cbdev, dinfo->pci.cfg.dev);
201 			pci_print_verbose(&dinfo->pci);
202 			if (device_probe_and_attach(dinfo->pci.cfg.dev) != 0)
203 				cardbus_release_all_resources(cbdev, dinfo);
204 			else
205 				cardattached++;
206 		}
207 	}
208 
209 	if (cardattached > 0)
210 		return (0);
211 	POWER_DISABLE_SOCKET(brdev, cbdev);
212 	return (ENOENT);
213 }
214 
215 static int
216 cardbus_detach_card(device_t cbdev)
217 {
218 	int numdevs;
219 	device_t *devlist;
220 	int tmp;
221 	int err = 0;
222 
223 	device_get_children(cbdev, &devlist, &numdevs);
224 
225 	if (numdevs == 0) {
226 		free(devlist, M_TEMP);
227 		return (ENOENT);
228 	}
229 
230 	for (tmp = 0; tmp < numdevs; tmp++) {
231 		struct cardbus_devinfo *dinfo = device_get_ivars(devlist[tmp]);
232 		int status = device_get_state(devlist[tmp]);
233 
234 		if (dinfo->pci.cfg.dev != devlist[tmp])
235 			device_printf(cbdev, "devinfo dev mismatch\n");
236 		if (status == DS_ATTACHED || status == DS_BUSY)
237 			device_detach(devlist[tmp]);
238 		cardbus_release_all_resources(cbdev, dinfo);
239 		device_delete_child(cbdev, devlist[tmp]);
240 		pci_freecfg((struct pci_devinfo *)dinfo);
241 	}
242 	POWER_DISABLE_SOCKET(device_get_parent(cbdev), cbdev);
243 	free(devlist, M_TEMP);
244 	return (err);
245 }
246 
247 static void
248 cardbus_driver_added(device_t cbdev, driver_t *driver)
249 {
250 	int numdevs;
251 	device_t *devlist;
252 	device_t dev;
253 	int i;
254 	struct cardbus_devinfo *dinfo;
255 
256 	DEVICE_IDENTIFY(driver, cbdev);
257 	device_get_children(cbdev, &devlist, &numdevs);
258 	/*
259 	 * If there are no drivers attached, but there are children,
260 	 * then power the card up.
261 	 */
262 	for (i = 0; i < numdevs; i++) {
263 		dev = devlist[i];
264 		if (device_get_state(dev) != DS_NOTPRESENT)
265 		    break;
266 	}
267 	if (i > 0 && i == numdevs)
268 		POWER_ENABLE_SOCKET(device_get_parent(cbdev), cbdev);
269 	for (i = 0; i < numdevs; i++) {
270 		dev = devlist[i];
271 		if (device_get_state(dev) != DS_NOTPRESENT)
272 			continue;
273 		dinfo = device_get_ivars(dev);
274 		pci_print_verbose(&dinfo->pci);
275 		resource_list_init(&dinfo->pci.resources);
276 		cardbus_do_cis(cbdev, dev);
277 		if (device_probe_and_attach(dev) != 0)
278 			cardbus_release_all_resources(cbdev, dinfo);
279 	}
280 	free(devlist, M_TEMP);
281 }
282 
283 static void
284 cardbus_release_all_resources(device_t cbdev, struct cardbus_devinfo *dinfo)
285 {
286 	struct resource_list_entry *rle;
287 
288 	/* Free all allocated resources */
289 	SLIST_FOREACH(rle, &dinfo->pci.resources, link) {
290 		if (rle->res) {
291 			if (rman_get_device(rle->res) != cbdev)
292 				device_printf(cbdev, "release_all_resource: "
293 				    "Resource still owned by child, oops. "
294 				    "(type=%d, rid=%d, addr=%lx)\n",
295 				    rle->type, rle->rid,
296 				    rman_get_start(rle->res));
297 			BUS_RELEASE_RESOURCE(device_get_parent(cbdev),
298 			    cbdev, rle->type, rle->rid, rle->res);
299 			rle->res = NULL;
300 			/*
301 			 * zero out config so the card won't acknowledge
302 			 * access to the space anymore
303 			 */
304 			pci_write_config(dinfo->pci.cfg.dev, rle->rid, 0, 4);
305 		}
306 	}
307 	resource_list_free(&dinfo->pci.resources);
308 }
309 
310 /************************************************************************/
311 /* Other Bus Methods							*/
312 /************************************************************************/
313 
314 static int
315 cardbus_read_ivar(device_t cbdev, device_t child, int which, uintptr_t *result)
316 {
317 	struct cardbus_devinfo *dinfo;
318 	pcicfgregs *cfg;
319 
320 	dinfo = device_get_ivars(child);
321 	cfg = &dinfo->pci.cfg;
322 
323 	switch (which) {
324 	case PCI_IVAR_ETHADDR:
325 		/*
326 		 * The generic accessor doesn't deal with failure, so
327 		 * we set the return value, then return an error.
328 		 */
329 		if ((dinfo->fepresent & (1 << TPL_FUNCE_LAN_NID)) == 0) {
330 			*((uint8_t **) result) = NULL;
331 			return (EINVAL);
332 		}
333 		*((uint8_t **) result) = dinfo->funce.lan.nid;
334 		break;
335 	default:
336 		return (pci_read_ivar(cbdev, child, which, result));
337 	}
338 	return 0;
339 }
340 
341 static int
342 cardbus_write_ivar(device_t cbdev, device_t child, int which, uintptr_t value)
343 {
344 	return(pci_write_ivar(cbdev, child, which, value));
345 }
346 
347 static device_method_t cardbus_methods[] = {
348 	/* Device interface */
349 	DEVMETHOD(device_probe,		cardbus_probe),
350 	DEVMETHOD(device_attach,	cardbus_attach),
351 	DEVMETHOD(device_detach,	cardbus_detach),
352 	DEVMETHOD(device_shutdown,	bus_generic_shutdown),
353 	DEVMETHOD(device_suspend,	cardbus_suspend),
354 	DEVMETHOD(device_resume,	cardbus_resume),
355 
356 	/* Bus interface */
357 	DEVMETHOD(bus_print_child,	pci_print_child),
358 	DEVMETHOD(bus_probe_nomatch,	pci_probe_nomatch),
359 	DEVMETHOD(bus_read_ivar,	cardbus_read_ivar),
360 	DEVMETHOD(bus_write_ivar,	cardbus_write_ivar),
361 	DEVMETHOD(bus_driver_added,	cardbus_driver_added),
362 	DEVMETHOD(bus_activate_resource, bus_generic_activate_resource),
363 	DEVMETHOD(bus_deactivate_resource, bus_generic_deactivate_resource),
364 	DEVMETHOD(bus_get_resource_list,pci_get_resource_list),
365 	DEVMETHOD(bus_set_resource,	bus_generic_rl_set_resource),
366 	DEVMETHOD(bus_get_resource,	bus_generic_rl_get_resource),
367 	DEVMETHOD(bus_delete_resource,	pci_delete_resource),
368 	DEVMETHOD(bus_alloc_resource,	pci_alloc_resource),
369 	DEVMETHOD(bus_release_resource,	bus_generic_rl_release_resource),
370 	DEVMETHOD(bus_activate_resource, bus_generic_activate_resource),
371 	DEVMETHOD(bus_deactivate_resource, bus_generic_deactivate_resource),
372 	DEVMETHOD(bus_setup_intr,	bus_generic_setup_intr),
373 	DEVMETHOD(bus_teardown_intr,	bus_generic_teardown_intr),
374 	DEVMETHOD(bus_child_pnpinfo_str, pci_child_pnpinfo_str_method),
375 	DEVMETHOD(bus_child_location_str, pci_child_location_str_method),
376 
377 	/* Card Interface */
378 	DEVMETHOD(card_attach_card,	cardbus_attach_card),
379 	DEVMETHOD(card_detach_card,	cardbus_detach_card),
380 	DEVMETHOD(card_cis_read,	cardbus_cis_read),
381 	DEVMETHOD(card_cis_free,	cardbus_cis_free),
382 
383 	/* Cardbus/PCI interface */
384 	DEVMETHOD(pci_read_config,	pci_read_config_method),
385 	DEVMETHOD(pci_write_config,	pci_write_config_method),
386 	DEVMETHOD(pci_enable_busmaster,	pci_enable_busmaster_method),
387 	DEVMETHOD(pci_disable_busmaster, pci_disable_busmaster_method),
388 	DEVMETHOD(pci_enable_io,	pci_enable_io_method),
389 	DEVMETHOD(pci_disable_io,	pci_disable_io_method),
390 	DEVMETHOD(pci_get_powerstate,	pci_get_powerstate_method),
391 	DEVMETHOD(pci_set_powerstate,	pci_set_powerstate_method),
392 	DEVMETHOD(pci_assign_interrupt, pci_assign_interrupt_method),
393 
394 	{0,0}
395 };
396 
397 static driver_t cardbus_driver = {
398 	"cardbus",
399 	cardbus_methods,
400 	0 /* no softc */
401 };
402 
403 static devclass_t cardbus_devclass;
404 
405 DRIVER_MODULE(cardbus, cbb, cardbus_driver, cardbus_devclass, 0, 0);
406 MODULE_VERSION(cardbus, 1);
407