xref: /freebsd/sys/dev/pccbb/pccbb.c (revision 5521ff5a4d1929056e7ffc982fac3341ca54df7c)
1 /*
2  * Copyright (c) 2000,2001 Jonathan Chen.
3  * 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  * $FreeBSD$
29  */
30 
31 /*
32  * Driver for PCI to Cardbus Bridge chips
33  *
34  * References:
35  *  TI Datasheets:
36  *   http://www-s.ti.com/cgi-bin/sc/generic2.cgi?family=PCI+CARDBUS+CONTROLLERS
37  * Much of the 16-bit PC Card compatibility code stolen from dev/pcic/i82365.c
38  * XXX and should be cleaned up to share as much as possible.
39  *
40  * Written by Jonathan Chen <jon@freebsd.org>
41  * The author would like to acknowledge:
42  *  * HAYAKAWA Koichi: Author of the NetBSD code for the same thing
43  *  * Warner Losh: Newbus/newcard guru and author of the pccard side of things
44  *  * YAMAMOTO Shigeru: Author of another FreeBSD cardbus driver
45  *  * David Cross: Author of the initial ugly hack for a specific cardbus card
46  */
47 
48 #define CBB_DEBUG
49 
50 #include <sys/param.h>
51 #include <sys/systm.h>
52 #include <sys/errno.h>
53 #include <sys/kernel.h>
54 #include <sys/kthread.h>
55 #include <sys/lock.h>
56 #include <sys/malloc.h>
57 #include <sys/mutex.h>
58 
59 #include <sys/bus.h>
60 #include <machine/bus.h>
61 #include <sys/rman.h>
62 #include <machine/resource.h>
63 
64 #include <pci/pcireg.h>
65 #include <pci/pcivar.h>
66 #include <machine/clock.h>
67 
68 #include <dev/pccard/pccardreg.h>
69 #include <dev/pccard/pccardvar.h>
70 #include <dev/pcic/i82365reg.h>
71 
72 #include <dev/pccbb/pccbbreg.h>
73 #include <dev/pccbb/pccbbvar.h>
74 
75 #include "power_if.h"
76 #include "card_if.h"
77 #include "pcib_if.h"
78 
79 #if defined CBB_DEBUG
80 #define	DPRINTF(x) printf x
81 #define	DEVPRINTF(x) device_printf x
82 #else
83 #define	DPRINTF(x)
84 #define	DEVPRINTF(x)
85 #endif
86 
87 #define	PCI_MASK_CONFIG(DEV,REG,MASK,SIZE) 				\
88 	pci_write_config(DEV, REG, pci_read_config(DEV, REG, SIZE) MASK, SIZE)
89 #define	PCI_MASK2_CONFIG(DEV,REG,MASK1,MASK2,SIZE)			\
90 	pci_write_config(DEV, REG, (					\
91 		pci_read_config(DEV, REG, SIZE) MASK1) MASK2, SIZE)
92 
93 /*
94  * XXX all the pcic code really doesn't belong here and needs to be
95  * XXX migrated to its own file, shared with the 16-bit code
96  */
97 #define	PCIC_READ(SC,REG)						\
98 	(((u_int8_t*)((SC)->sc_socketreg))[0x800+(REG)])
99 #define	PCIC_WRITE(SC,REG,val)						\
100 	(((u_int8_t*)((SC)->sc_socketreg))[0x800+(REG)]) = (val)
101 #define	PCIC_MASK(SC,REG,MASK)						\
102 	PCIC_WRITE(SC,REG,PCIC_READ(SC,REG) MASK)
103 #define	PCIC_MASK2(SC,REG,MASK,MASK2)					\
104 	PCIC_WRITE(SC,REG,(PCIC_READ(SC,REG) MASK) MASK2)
105 
106 struct pccbb_sclist {
107 	struct pccbb_softc *sc;
108 	STAILQ_ENTRY(pccbb_sclist) entries;
109 };
110 
111 static STAILQ_HEAD(, pccbb_sclist) softcs;
112 static int softcs_init = 0;
113 
114 
115 struct yenta_chipinfo {
116 	u_int32_t yc_id;
117 	const char *yc_name;
118 	int yc_chiptype;
119 	int yc_flags;
120 } yc_chipsets[] = {
121 	/* Texas Instruments chips */
122 	{PCI_DEVICE_ID_PCIC_TI1130, "TI1130 PCI-CardBus Bridge", CB_TI113X,
123 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
124 	{PCI_DEVICE_ID_PCIC_TI1131, "TI1131 PCI-CardBus Bridge", CB_TI113X,
125 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
126 
127 	{PCI_DEVICE_ID_PCIC_TI1211, "TI1211 PCI-CardBus Bridge", CB_TI12XX,
128 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
129 	{PCI_DEVICE_ID_PCIC_TI1220, "TI1220 PCI-CardBus Bridge", CB_TI12XX,
130 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
131 	{PCI_DEVICE_ID_PCIC_TI1221, "TI1221 PCI-CardBus Bridge", CB_TI12XX,
132 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
133 	{PCI_DEVICE_ID_PCIC_TI1225, "TI1225 PCI-CardBus Bridge", CB_TI12XX,
134 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
135 	{PCI_DEVICE_ID_PCIC_TI1250, "TI1250 PCI-CardBus Bridge", CB_TI12XX,
136 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
137 	{PCI_DEVICE_ID_PCIC_TI1251, "TI1251 PCI-CardBus Bridge", CB_TI12XX,
138 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
139 	{PCI_DEVICE_ID_PCIC_TI1251B,"TI1251B PCI-CardBus Bridge",CB_TI12XX,
140 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
141 	{PCI_DEVICE_ID_PCIC_TI1410, "TI1410 PCI-CardBus Bridge", CB_TI12XX,
142 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
143 	{PCI_DEVICE_ID_PCIC_TI1420, "TI1420 PCI-CardBus Bridge", CB_TI12XX,
144 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
145 	{PCI_DEVICE_ID_PCIC_TI1450, "TI1450 PCI-CardBus Bridge", CB_TI12XX,
146 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
147 	{PCI_DEVICE_ID_PCIC_TI1451, "TI1451 PCI-CardBus Bridge", CB_TI12XX,
148 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
149 	{PCI_DEVICE_ID_PCIC_TI4451, "TI4451 PCI-CardBus Bridge", CB_TI12XX,
150 		PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32},
151 
152 	/* Ricoh chips */
153 	{PCI_DEVICE_ID_RICOH_RL5C465, "RF5C465 PCI-CardBus Bridge",
154 		CB_RF5C46X, PCCBB_PCIC_MEM_32},
155 	{PCI_DEVICE_ID_RICOH_RL5C466, "RF5C466 PCI-CardBus Bridge",
156 		CB_RF5C46X, PCCBB_PCIC_MEM_32},
157 	{PCI_DEVICE_ID_RICOH_RL5C475, "RF5C475 PCI-CardBus Bridge",
158 		CB_RF5C47X, PCCBB_PCIC_MEM_32},
159 	{PCI_DEVICE_ID_RICOH_RL5C476, "RF5C476 PCI-CardBus Bridge",
160 		CB_RF5C47X, PCCBB_PCIC_MEM_32},
161 	{PCI_DEVICE_ID_RICOH_RL5C478, "RF5C478 PCI-CardBus Bridge",
162 		CB_RF5C47X, PCCBB_PCIC_MEM_32},
163 
164 	/* Toshiba products */
165 	{PCI_DEVICE_ID_TOSHIBA_TOPIC95, "ToPIC95 PCI-CardBus Bridge",
166 		CB_TOPIC95, PCCBB_PCIC_MEM_32},
167 	{PCI_DEVICE_ID_TOSHIBA_TOPIC95B, "ToPIC95B PCI-CardBus Bridge",
168 		CB_TOPIC95B, PCCBB_PCIC_MEM_32},
169 	{PCI_DEVICE_ID_TOSHIBA_TOPIC97, "ToPIC97 PCI-CardBus Bridge",
170 		CB_TOPIC97, PCCBB_PCIC_MEM_32},
171 	{PCI_DEVICE_ID_TOSHIBA_TOPIC100, "ToPIC100 PCI-CardBus Bridge",
172 		CB_TOPIC97, PCCBB_PCIC_MEM_32},
173 
174 	/* Cirrus Logic */
175 	{PCI_DEVICE_ID_PCIC_CLPD6832, "CLPD6832 PCI-CardBus Bridge",
176 		CB_CIRRUS, PCCBB_PCIC_MEM_32},
177 	{PCI_DEVICE_ID_PCIC_CLPD6833, "CLPD6833 PCI-CardBus Bridge",
178 		CB_CIRRUS, PCCBB_PCIC_MEM_32},
179 
180 	/* 02Micro */
181 	{PCI_DEVICE_ID_PCIC_OZ6832, "O2Mirco OZ6832/6833 PCI-CardBus Bridge",
182 		CB_CIRRUS, PCCBB_PCIC_MEM_32},
183 	{PCI_DEVICE_ID_PCIC_OZ6860, "O2Mirco OZ6836/6860 PCI-CardBus Bridge",
184 		CB_CIRRUS, PCCBB_PCIC_MEM_32},
185 	{PCI_DEVICE_ID_PCIC_OZ6872, "O2Mirco OZ6812/6872 PCI-CardBus Bridge",
186 		CB_CIRRUS, PCCBB_PCIC_MEM_32},
187 
188 	/* sentinel */
189 	{0 /* null id */, "unknown",
190 		CB_UNKNOWN, 0},
191 };
192 
193 
194 static int cb_chipset(u_int32_t pci_id, const char** namep, int* flagp);
195 static int pccbb_probe(device_t dev);
196 static void pccbb_chipinit(struct pccbb_softc* sc);
197 static int pccbb_attach(device_t dev);
198 static int pccbb_detach(device_t dev);
199 static void pccbb_driver_added(device_t dev, driver_t *driver);
200 static void pccbb_child_detached(device_t dev, device_t child);
201 static void pccbb_event_thread (void *arg);
202 static void pccbb_create_event_thread (struct pccbb_softc *sc);
203 static void pccbb_start_threads(void *arg);
204 static void pccbb_insert (struct pccbb_softc *sc);
205 static void pccbb_removal (struct pccbb_softc *sc);
206 static void pccbb_intr(void* arg);
207 static int pccbb_detect_voltage(device_t dev);
208 static int pccbb_power(device_t dev, int volts);
209 static void pccbb_cardbus_reset(device_t dev);
210 static int pccbb_cardbus_power_enable_socket(device_t self, device_t child);
211 static void pccbb_cardbus_power_disable_socket(device_t self, device_t child);
212 static int pccbb_cardbus_io_open(device_t dev, int win,
213 				 u_int32_t start, u_int32_t end);
214 static int pccbb_cardbus_mem_open(device_t dev, int win,
215 				  u_int32_t start, u_int32_t end);
216 static void pccbb_cardbus_auto_open(struct pccbb_softc *sc, int type);
217 static int pccbb_cardbus_activate_resource(device_t self, device_t child,
218 					   int type, int rid,
219 					   struct resource *r);
220 static int pccbb_cardbus_deactivate_resource(device_t self, device_t child,
221 					     int type, int rid,
222 					     struct resource *r);
223 static struct resource* pccbb_cardbus_alloc_resource(device_t self,
224 			     device_t child, int type, int* rid,
225 			     u_long start, u_long end, u_long count,
226 			     u_int flags);
227 static int pccbb_cardbus_release_resource(device_t self, device_t child,
228 					  int type,int rid,
229 					  struct resource *r);
230 static int pccbb_pcic_power_enable_socket(device_t self, device_t child);
231 static void pccbb_pcic_power_disable_socket(device_t self, device_t child);
232 static void pccbb_pcic_wait_ready(struct pccbb_softc *sc);
233 static void pccbb_pcic_do_mem_map(struct pccbb_softc *sc, int win);
234 static int pccbb_pcic_mem_map(struct pccbb_softc *sc, int kind,
235 			      struct resource *r, bus_addr_t card_addr,
236 			      int *win);
237 static void pccbb_pcic_mem_unmap(struct pccbb_softc *sc, int window);
238 static void pccbb_pcic_do_io_map(struct pccbb_softc *sc, int win);
239 static int pccbb_pcic_io_map(struct pccbb_softc *sc, int width,
240 			     struct resource *r, bus_addr_t card_addr,
241 			     int *win);
242 static void pccbb_pcic_io_unmap(struct pccbb_softc *sc, int window);
243 static int pccbb_pcic_activate_resource(device_t self, device_t child,
244 					int type, int rid, struct resource *r);
245 static int pccbb_pcic_deactivate_resource(device_t self, device_t child,
246 					 int type,int rid, struct resource *r);
247 static struct resource* pccbb_pcic_alloc_resource(device_t self,device_t child,
248 				int type, int* rid, u_long start,
249 				u_long end, u_long count, u_int flags);
250 static int pccbb_pcic_release_resource(device_t self, device_t child, int type,
251 				       int rid, struct resource *res);
252 static int pccbb_pcic_set_res_flags(device_t self, device_t child, int type,
253 				    int rid, u_int32_t flags);
254 static int pccbb_pcic_set_memory_offset(device_t self, device_t child, int rid,
255 					u_int32_t offset, u_int32_t *deltap);
256 static int pccbb_power_enable_socket(device_t self, device_t child);
257 static void pccbb_power_disable_socket(device_t self, device_t child);
258 static int pccbb_activate_resource(device_t self, device_t child, int type,
259 				   int rid, struct resource *r);
260 static int pccbb_deactivate_resource(device_t self, device_t child, int type,
261 				     int rid, struct resource *r);
262 static struct resource* pccbb_alloc_resource(device_t self, device_t child,
263 					     int type, int* rid, u_long start,
264 					     u_long end, u_long count,
265 					     u_int flags);
266 static int pccbb_release_resource(device_t self, device_t child,
267 				  int type, int rid, struct resource *r);
268 static int pccbb_maxslots(device_t dev);
269 static u_int32_t pccbb_read_config(device_t dev, int b, int s, int f,
270 				   int reg, int width);
271 static void pccbb_write_config(device_t dev, int b, int s, int f, int reg,
272 			       u_int32_t val, int width);
273 
274 
275 /************************************************************************/
276 /* Probe/Attach								*/
277 /************************************************************************/
278 
279 static int
280 cb_chipset(u_int32_t pci_id, const char** namep, int* flagp)
281 {
282 	int loopend = sizeof(yc_chipsets)/sizeof(yc_chipsets[0]);
283 	struct yenta_chipinfo *ycp, *ycend;
284 	ycend = yc_chipsets + loopend;
285 
286 	for (ycp = yc_chipsets; ycp < ycend && pci_id != ycp->yc_id; ++ycp);
287 	if (ycp == ycend) {
288 		/* not found */
289 		ycp = yc_chipsets + loopend - 1; /* to point the sentinel */
290 	}
291 	if (namep != NULL) {
292 		*namep = ycp->yc_name;
293 	}
294 	if (flagp != NULL) {
295 		*flagp = ycp->yc_flags;
296 	}
297 	return ycp->yc_chiptype;
298 }
299 
300 static int
301 pccbb_probe(device_t dev)
302 {
303 	const char *name;
304 
305 	if (cb_chipset(pci_get_devid(dev), &name, NULL) == CB_UNKNOWN)
306 		return ENXIO;
307 	device_set_desc(dev, name);
308 	return 0;
309 }
310 
311 static void
312 pccbb_chipinit(struct pccbb_softc* sc)
313 {
314 	/* Set CardBus latency timer */
315 	if (pci_read_config(sc->sc_dev, PCIR_SECLAT_1, 1) < 0x20)
316 		pci_write_config(sc->sc_dev, PCIR_SECLAT_1, 0x20, 1);
317 
318 	/* Set PCI latency timer */
319 	if (pci_read_config(sc->sc_dev, PCIR_LATTIMER, 1) < 0x20)
320 		pci_write_config(sc->sc_dev, PCIR_LATTIMER, 0x20, 1);
321 
322 	/* Enable memory access */
323 	PCI_MASK_CONFIG(sc->sc_dev, PCIR_COMMAND,
324 			| PCIM_CMD_MEMEN
325 			| PCIM_CMD_PORTEN
326 			| PCIM_CMD_BUSMASTEREN, 2);
327 
328 	/* disable Legacy IO */
329 	switch (sc->sc_chipset) {
330 	case CB_RF5C46X:
331 	case CB_RF5C47X:
332 		PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_BRIDGECTRL,
333 				& ~(PCCBBM_BRIDGECTRL_RL_3E0_EN|
334 				    PCCBBM_BRIDGECTRL_RL_3E2_EN), 2);
335 		break;
336 	default:
337 		pci_write_config(sc->sc_dev, PCCBBR_LEGACY, 0x0, 4);
338 		break;
339 	}
340 
341 	/* Use PCI interrupt for interrupt routing */
342 	PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_BRIDGECTRL,
343 			 & ~(PCCBBM_BRIDGECTRL_MASTER_ABORT |
344 			     PCCBBM_BRIDGECTRL_INTR_IREQ_EN),
345 			 | PCCBBM_BRIDGECTRL_WRITE_POST_EN,
346 			 2);
347 
348 	switch (sc->sc_chipset) {
349 	case CB_TI113X:
350 		PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_CBCTRL,
351 				 & ~PCCBBM_CBCTRL_113X_PCI_INTR,
352 				 | PCCBBM_CBCTRL_113X_PCI_CSC
353 				 | PCCBBM_CBCTRL_113X_PCI_IRQ_EN, 1);
354 		PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_DEVCTRL,
355 				& ~(PCCBBM_DEVCTRL_INT_SERIAL|
356 				    PCCBBM_DEVCTRL_INT_PCI), 1);
357 		PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE);
358 		PCIC_WRITE(sc, PCIC_CSC_INTR, 0);
359 		break;
360 	case CB_TI12XX:
361 		PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE);
362 		PCIC_WRITE(sc, PCIC_CSC_INTR, 0);
363 		break;
364 	case CB_TOPIC95B:
365 		PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_TOPIC_SOCKETCTRL,
366 				| PCCBBM_TOPIC_SOCKETCTRL_SCR_IRQSEL, 4);
367 		PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_TOPIC_SLOTCTRL,
368 				 | PCCBBM_TOPIC_SLOTCTRL_SLOTON
369 				 | PCCBBM_TOPIC_SLOTCTRL_SLOTEN
370 				 | PCCBBM_TOPIC_SLOTCTRL_ID_LOCK
371 				 | PCCBBM_TOPIC_SLOTCTRL_CARDBUS,
372 				 & ~PCCBBM_TOPIC_SLOTCTRL_SWDETECT, 4);
373 		break;
374 	}
375 
376 	/* close all memory and io windows */
377 	pci_write_config(sc->sc_dev, PCCBBR_MEMBASE0, 0xffffffff, 4);
378 	pci_write_config(sc->sc_dev, PCCBBR_MEMLIMIT0, 0, 4);
379 	pci_write_config(sc->sc_dev, PCCBBR_MEMBASE1, 0xffffffff, 4);
380 	pci_write_config(sc->sc_dev, PCCBBR_MEMLIMIT1, 0, 4);
381 	pci_write_config(sc->sc_dev, PCCBBR_IOBASE0, 0xffffffff, 4);
382 	pci_write_config(sc->sc_dev, PCCBBR_IOLIMIT0, 0, 4);
383 	pci_write_config(sc->sc_dev, PCCBBR_IOBASE1, 0xffffffff, 4);
384 	pci_write_config(sc->sc_dev, PCCBBR_IOLIMIT1, 0, 4);
385 }
386 
387 static int
388 pccbb_attach(device_t dev)
389 {
390 	struct pccbb_softc *sc = (struct pccbb_softc *)device_get_softc(dev);
391 	int rid;
392 	u_int32_t tmp;
393 
394 	if (!softcs_init) {
395 		softcs_init = 1;
396 		STAILQ_INIT(&softcs);
397 	}
398 	mtx_init(&sc->sc_mtx, device_get_nameunit(dev), MTX_DEF);
399 	sc->sc_chipset = cb_chipset(pci_get_devid(dev), NULL, &sc->sc_flags);
400 	sc->sc_dev = dev;
401 	sc->sc_cbdev = NULL;
402 	sc->sc_pccarddev = NULL;
403 	sc->sc_secbus = pci_read_config(dev, PCIR_SECBUS_2, 1);
404 	sc->sc_subbus = pci_read_config(dev, PCIR_SUBBUS_2, 1);
405 	sc->memalloc = 0;
406 	sc->ioalloc = 0;
407 	SLIST_INIT(&sc->rl);
408 
409 	/* Ths PCI bus should have given me memory... right? */
410 	rid=PCCBBR_SOCKBASE;
411 	sc->sc_base_res=bus_alloc_resource(dev, SYS_RES_MEMORY, &rid,
412 					   0,~0,1, RF_ACTIVE);
413 	if (!sc->sc_base_res) {
414 		/*
415 		 * XXX eVILE HACK BAD THING! XXX
416 		 * The pci bus device should do this for us.
417 		 * Some BIOSes doesn't assign a memory space properly.
418 		 * So we try to manually put one in...
419 		 */
420 		u_int32_t sockbase;
421 
422 		sockbase = pci_read_config(dev, rid, 4);
423 		if (sockbase < 0x100000 || sockbase >= 0xfffffff0) {
424 			pci_write_config(dev, rid, 0xffffffff, 4);
425 			sockbase = pci_read_config(dev, rid, 4);
426 			sockbase = (sockbase & 0xfffffff0) &
427 				-(sockbase & 0xfffffff0);
428 			sc->sc_base_res = bus_generic_alloc_resource(
429 				device_get_parent(dev), dev, SYS_RES_MEMORY,
430 				&rid, CARDBUS_SYS_RES_MEMORY_START,
431 				CARDBUS_SYS_RES_MEMORY_END, sockbase,
432 				RF_ACTIVE|rman_make_alignment_flags(sockbase));
433 			if (!sc->sc_base_res){
434 				device_printf(dev,
435 					"Could not grab register memory\n");
436 				mtx_destroy(&sc->sc_mtx);
437 				return ENOMEM;
438 			}
439 			pci_write_config(dev, PCCBBR_SOCKBASE,
440 					 rman_get_start(sc->sc_base_res), 4);
441 			DEVPRINTF((dev, "PCI Memory allocated: %08lx\n",
442 				   rman_get_start(sc->sc_base_res)));
443 		} else {
444 			device_printf(dev, "Could not map register memory\n");
445 			mtx_destroy(&sc->sc_mtx);
446 			return ENOMEM;
447 		}
448 	}
449 
450 	sc->sc_socketreg =
451 		(struct pccbb_socketreg *)rman_get_virtual(sc->sc_base_res);
452 	pccbb_chipinit(sc);
453 
454 	/* CSC Interrupt: Card detect interrupt on */
455 	sc->sc_socketreg->socket_mask |= PCCBB_SOCKET_MASK_CD;
456 
457 	/* reset interrupt */
458 	tmp = sc->sc_socketreg->socket_event;
459 	sc->sc_socketreg->socket_event = tmp;
460 
461 	/* Map and establish the interrupt. */
462 	rid=0;
463 	sc->sc_irq_res=bus_alloc_resource(dev, SYS_RES_IRQ, &rid, 0, ~0, 1,
464 					  RF_SHAREABLE | RF_ACTIVE);
465 	if (sc->sc_irq_res == NULL)  {
466 		printf("pccbb: Unable to map IRQ...\n");
467 		bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE,
468 				     sc->sc_base_res);
469 		mtx_destroy(&sc->sc_mtx);
470 		return ENOMEM;
471 	}
472 
473 	if (bus_setup_intr(dev, sc->sc_irq_res, INTR_TYPE_BIO, pccbb_intr, sc,
474 		     &(sc->sc_intrhand))) {
475 		device_printf(dev, "couldn't establish interrupt");
476 		bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res);
477 		bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE,
478 				     sc->sc_base_res);
479 		mtx_destroy(&sc->sc_mtx);
480 		return ENOMEM;
481 	}
482 
483 	/* attach children */
484 	sc->sc_cbdev = device_add_child(dev, "cardbus", -1);
485 	if (sc->sc_cbdev == NULL)
486 		DEVPRINTF((dev, "WARNING: cannot add cardbus bus.\n"));
487 	else if (device_probe_and_attach(sc->sc_cbdev) != 0) {
488 		DEVPRINTF((dev, "WARNING: cannot attach cardbus bus!\n"));
489 		sc->sc_cbdev = NULL;
490 	}
491 
492 	sc->sc_pccarddev = device_add_child(dev, "pccard", -1);
493 	if (sc->sc_pccarddev == NULL)
494 		DEVPRINTF((dev, "WARNING: cannot add pccard bus.\n"));
495 	else if (device_probe_and_attach(sc->sc_pccarddev) != 0) {
496 		DEVPRINTF((dev, "WARNING: cannot attach pccard bus.\n"));
497 		sc->sc_pccarddev = NULL;
498 	}
499 
500 #ifndef KLD_MODULE
501 	if (sc->sc_cbdev == NULL && sc->sc_pccarddev == NULL) {
502 		device_printf(dev, "ERROR: Failed to attach cardbus/pccard bus!\n");
503 		bus_teardown_intr(dev, sc->sc_irq_res, sc->sc_intrhand);
504 		bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res);
505 		bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE,
506 				     sc->sc_base_res);
507 		mtx_destroy(&sc->sc_mtx);
508 		return ENOMEM;
509 	}
510 #endif
511 
512 	{
513 		struct pccbb_sclist *sclist;
514 		sclist = malloc(sizeof(struct pccbb_sclist), M_DEVBUF,
515 				M_WAITOK);
516 		sclist->sc = sc;
517 		STAILQ_INSERT_TAIL(&softcs, sclist, entries);
518 	}
519 	return 0;
520 }
521 
522 static int
523 pccbb_detach(device_t dev)
524 {
525 	struct pccbb_softc *sc = device_get_softc(dev);
526 	int numdevs;
527 	device_t *devlist;
528 	int tmp;
529 	int error;
530 
531 	device_get_children(dev, &devlist, &numdevs);
532 
533 	error = 0;
534 	for (tmp = 0; tmp < numdevs; tmp++) {
535 		if (device_detach(devlist[tmp]) == 0)
536 			device_delete_child(dev, devlist[tmp]);
537 		else
538 			error++;
539 	}
540 	free(devlist, M_TEMP);
541 	if (error > 0)
542 		return ENXIO;
543 
544 	mtx_lock(&sc->sc_mtx);
545 	bus_teardown_intr(dev, sc->sc_irq_res, sc->sc_intrhand);
546 
547 	sc->sc_flags |= PCCBB_KTHREAD_DONE;
548 	if (sc->sc_flags & PCCBB_KTHREAD_RUNNING) {
549 		wakeup(sc);
550 		mtx_unlock(&sc->sc_mtx);
551 		DEVPRINTF((dev, "waiting for kthread exit..."));
552 		error = tsleep(sc, PWAIT, "pccbb-detach-wait", 60 * hz);
553 		if (error)
554 			DPRINTF(("timeout\n"));
555 		else
556 			DPRINTF(("done\n"));
557 	} else
558 		mtx_unlock(&sc->sc_mtx);
559 
560 	bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res);
561 	bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE,
562 			     sc->sc_base_res);
563 	mtx_destroy(&sc->sc_mtx);
564 	return 0;
565 }
566 
567 static void
568 pccbb_driver_added(device_t dev, driver_t *driver)
569 {
570 	struct pccbb_softc *sc = device_get_softc(dev);
571 	device_t *devlist;
572 	int tmp;
573 	int numdevs;
574 
575 	DEVICE_IDENTIFY(driver, dev);
576 	device_get_children(dev, &devlist, &numdevs);
577 	for (tmp = 0; tmp < numdevs; tmp++) {
578 		if (device_get_state(devlist[tmp]) == DS_NOTPRESENT &&
579 		    device_probe_and_attach(devlist[tmp]) == 0) {
580 			if (devlist[tmp] == NULL)
581 				/* NOTHING */;
582 			else if (strcmp(driver->name, "cardbus") == 0) {
583 				sc->sc_cbdev = devlist[tmp];
584 				if ((sc->sc_socketreg->socket_state
585 				     & PCCBB_SOCKET_STAT_CD) == 0) {
586 					mtx_lock(&sc->sc_mtx);
587 					wakeup(sc);
588 					mtx_unlock(&sc->sc_mtx);
589 				}
590 			} else if (strcmp(driver->name, "pccard") == 0) {
591 				sc->sc_pccarddev = devlist[tmp];
592 				if ((sc->sc_socketreg->socket_state
593 				     & PCCBB_SOCKET_STAT_CD) == 0) {
594 					mtx_lock(&sc->sc_mtx);
595 					wakeup(sc);
596 					mtx_unlock(&sc->sc_mtx);
597 				}
598 			} else
599 				device_printf(dev,
600 					      "Unsupported child bus: %s\n",
601 					      driver->name);
602 		}
603 	}
604 	free(devlist, M_TEMP);
605 }
606 
607 static void
608 pccbb_child_detached(device_t dev, device_t child)
609 {
610 	struct pccbb_softc *sc = device_get_softc(dev);
611 	if (child == sc->sc_cbdev)
612 		sc->sc_cbdev = NULL;
613 	else if (child == sc->sc_pccarddev)
614 		sc->sc_pccarddev = NULL;
615 	else
616 		device_printf(dev, "Unknown child detached: %s %p/%p\n",
617 			      device_get_nameunit(child), sc->sc_cbdev, sc->sc_pccarddev);
618 }
619 
620 /************************************************************************/
621 /* Kthreads								*/
622 /************************************************************************/
623 
624 static void
625 pccbb_event_thread (void *arg)
626 {
627 	struct pccbb_softc *sc = arg;
628 	u_int32_t status;
629 
630 	mtx_lock(&Giant);
631 	for(;;) {
632 		if (!(sc->sc_flags & PCCBB_KTHREAD_RUNNING))
633 			sc->sc_flags |= PCCBB_KTHREAD_RUNNING;
634 		else {
635 			tsleep (sc, PWAIT, "pccbbev", 0);
636 			/*
637 			 * Delay 1 second, make sure the user is done with
638 			 * whatever he is doing.  We tsleep on sc->sc_flags,
639 			 * which should never be woken up.
640 			 */
641 			tsleep (&sc->sc_flags, PWAIT, "pccbbev", 1*hz);
642 		}
643 		mtx_lock(&sc->sc_mtx);
644 		if (sc->sc_flags & PCCBB_KTHREAD_DONE)
645 			break;
646 
647 		status = sc->sc_socketreg->socket_state;
648 		if ((status & PCCBB_SOCKET_STAT_CD) == 0) {
649 			pccbb_insert(sc);
650 		} else {
651 			pccbb_removal(sc);
652 		}
653 		mtx_unlock(&sc->sc_mtx);
654 	}
655 	mtx_unlock(&sc->sc_mtx);
656 	sc->sc_flags &= ~PCCBB_KTHREAD_RUNNING;
657 	wakeup(sc);
658 	kthread_exit(0);
659 }
660 
661 static void
662 pccbb_create_event_thread (struct pccbb_softc *sc)
663 {
664 	if (kthread_create(pccbb_event_thread, sc, &sc->event_thread,
665 			   0, "%s%d", device_get_name(sc->sc_dev),
666 			   device_get_unit(sc->sc_dev))) {
667 		device_printf (sc->sc_dev, "unable to create event thread.\n");
668 		panic ("pccbb_create_event_thread");
669 	}
670 }
671 
672 static void
673 pccbb_start_threads(void *arg)
674 {
675 	struct pccbb_sclist *sclist;
676 
677 	STAILQ_FOREACH(sclist, &softcs, entries) {
678 		pccbb_create_event_thread(sclist->sc);
679 	}
680 }
681 
682 /************************************************************************/
683 /* Insert/removal							*/
684 /************************************************************************/
685 
686 static void
687 pccbb_insert (struct pccbb_softc *sc)
688 {
689 	u_int32_t sockevent, sockstate;
690 	int timeout = 30;
691 
692 	do {
693 		sockevent = sc->sc_socketreg->socket_event;
694 		sockstate = sc->sc_socketreg->socket_state;
695 	} while (sockstate & PCCBB_SOCKET_STAT_CD && --timeout > 0);
696 
697 	if (timeout < 0) {
698 		device_printf (sc->sc_dev, "insert timeout");
699 		return;
700 	}
701 
702 	DEVPRINTF((sc->sc_dev, "card inserted: event=0x%08x, state=%08x\n",
703 		   sockevent, sockstate));
704 
705 	if (sockstate & PCCBB_SOCKET_STAT_16BIT && sc->sc_pccarddev != NULL) {
706 		sc->sc_flags |= PCCBB_16BIT_CARD;
707 		if (CARD_ATTACH_CARD(sc->sc_pccarddev) != 0)
708 			device_printf(sc->sc_dev, "card activation failed\n");
709 	} else if (sockstate & PCCBB_SOCKET_STAT_CB && sc->sc_cbdev != NULL) {
710 		sc->sc_flags &= ~PCCBB_16BIT_CARD;
711 		if (CARD_ATTACH_CARD(sc->sc_cbdev) != 0)
712 			device_printf(sc->sc_dev, "card activation failed\n");
713 	} else {
714 		device_printf (sc->sc_dev, "Unsupported card type detected\n");
715 	}
716 }
717 
718 static void
719 pccbb_removal (struct pccbb_softc *sc)
720 {
721 	u_int32_t sockstate;
722 	struct pccbb_reslist *rle;
723 
724 	sockstate = sc->sc_socketreg->socket_state;
725 
726 	if (sockstate & PCCBB_16BIT_CARD && sc->sc_pccarddev != NULL)
727 		CARD_DETACH_CARD(sc->sc_pccarddev, DETACH_FORCE);
728 	else if ((!(sockstate & PCCBB_16BIT_CARD)) && sc->sc_cbdev != NULL)
729 		CARD_DETACH_CARD(sc->sc_cbdev, DETACH_FORCE);
730 
731 	while (NULL != (rle = SLIST_FIRST(&sc->rl))) {
732 		device_printf(sc->sc_dev, "Danger Will Robinson: Resource "
733 		    "left allocated!  This is a bug... "
734 		    "(rid=%x, type=%d, addr=%x)\n", rle->rid, rle->type,
735 		    rle->start);
736 		SLIST_REMOVE_HEAD(&sc->rl, entries);
737 	}
738 }
739 
740 /************************************************************************/
741 /* Interrupt Handler							*/
742 /************************************************************************/
743 
744 static void
745 pccbb_intr(void* arg)
746 {
747 	struct pccbb_softc *sc = arg;
748 	u_int32_t sockevent;
749 
750 	if (!(sockevent = sc->sc_socketreg->socket_event)) {
751 		/* not for me. */
752 		return;
753 	}
754 
755 	/* reset bit */
756 	sc->sc_socketreg->socket_event = sockevent | 0x01;
757 
758 	if (sockevent & PCCBB_SOCKET_EVENT_CD) {
759 		mtx_lock(&sc->sc_mtx);
760 		wakeup(sc);
761 		mtx_unlock(&sc->sc_mtx);
762 	} else {
763 		if (sockevent & PCCBB_SOCKET_EVENT_CSTS) {
764 			DPRINTF((" cstsevent occures, 0x%08x\n",
765 				 sc->sc_socketreg->socket_state));
766 		}
767 		if (sockevent & PCCBB_SOCKET_EVENT_POWER) {
768 			DPRINTF((" pwrevent occures, 0x%08x\n",
769 				 sc->sc_socketreg->socket_state));
770 		}
771 	}
772 
773 	return;
774 }
775 
776 /************************************************************************/
777 /* Generic Power functions						*/
778 /************************************************************************/
779 
780 static int
781 pccbb_detect_voltage(device_t dev)
782 {
783 	struct pccbb_softc *sc = device_get_softc(dev);
784 	u_int32_t psr;
785 	int vol = CARD_UKN_CARD;
786 
787 	psr = sc->sc_socketreg->socket_state;
788 
789 	if (psr & PCCBB_SOCKET_STAT_5VCARD) {
790 		vol |= CARD_5V_CARD;
791 	}
792 	if (psr & PCCBB_SOCKET_STAT_3VCARD) {
793 		vol |= CARD_3V_CARD;
794 	}
795 	if (psr & PCCBB_SOCKET_STAT_XVCARD) {
796 		vol |= CARD_XV_CARD;
797 	}
798 	if (psr & PCCBB_SOCKET_STAT_YVCARD) {
799 		vol |= CARD_YV_CARD;
800 	}
801 
802 	return vol;
803 }
804 
805 static int
806 pccbb_power(device_t dev, int volts)
807 {
808 	u_int32_t status, sock_ctrl;
809 	struct pccbb_softc *sc = device_get_softc(dev);
810 
811 	DEVPRINTF((sc->sc_dev, "pccbb_power: %s and %s [%x]\n",
812 		   (volts & CARD_VCCMASK) == CARD_VCC_UC ? "CARD_VCC_UC" :
813 		   (volts & CARD_VCCMASK) == CARD_VCC_5V ? "CARD_VCC_5V" :
814 		   (volts & CARD_VCCMASK) == CARD_VCC_3V ? "CARD_VCC_3V" :
815 		   (volts & CARD_VCCMASK) == CARD_VCC_XV ? "CARD_VCC_XV" :
816 		   (volts & CARD_VCCMASK) == CARD_VCC_YV ? "CARD_VCC_YV" :
817 		   (volts & CARD_VCCMASK) == CARD_VCC_0V ? "CARD_VCC_0V" :
818 		   "VCC-UNKNOWN",
819 		   (volts & CARD_VPPMASK) == CARD_VPP_UC ? "CARD_VPP_UC" :
820 		   (volts & CARD_VPPMASK) == CARD_VPP_12V ? "CARD_VPP_12V" :
821 		   (volts & CARD_VPPMASK) == CARD_VPP_VCC ? "CARD_VPP_VCC" :
822 		   (volts & CARD_VPPMASK) == CARD_VPP_0V ? "CARD_VPP_0V" :
823 		   "VPP-UNKNOWN",
824 		   volts));
825 
826 	status = sc->sc_socketreg->socket_state;
827 	sock_ctrl = sc->sc_socketreg->socket_control;
828 
829 	switch (volts & CARD_VCCMASK) {
830 	case CARD_VCC_UC:
831 		break;
832 	case CARD_VCC_5V:
833 		if (PCCBB_SOCKET_STAT_5VCARD & status) { /* check 5 V card */
834 			sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK;
835 			sock_ctrl |= PCCBB_SOCKET_CTRL_VCC_5V;
836 		} else {
837 			device_printf(sc->sc_dev,
838 				      "BAD voltage request: no 5 V card\n");
839 		}
840 		break;
841 	case CARD_VCC_3V:
842 		if (PCCBB_SOCKET_STAT_3VCARD & status) {
843 			sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK;
844 			sock_ctrl |= PCCBB_SOCKET_CTRL_VCC_3V;
845 		} else {
846 			device_printf(sc->sc_dev,
847 				      "BAD voltage request: no 3.3 V card\n");
848 		}
849 		break;
850 	case CARD_VCC_0V:
851 		sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK;
852 		break;
853 	default:
854 		return 0;			/* power NEVER changed */
855 		break;
856 	}
857 
858 	switch (volts & CARD_VPPMASK) {
859 	case CARD_VPP_UC:
860 		break;
861 	case CARD_VPP_0V:
862 		sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK;
863 		break;
864 	case CARD_VPP_VCC:
865 		sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK;
866 		sock_ctrl |= ((sock_ctrl >> 4) & 0x07);
867 		break;
868 	case CARD_VPP_12V:
869 		sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK;
870 		sock_ctrl |= PCCBB_SOCKET_CTRL_VPP_12V;
871 		break;
872 	}
873 
874 	if (sc->sc_socketreg->socket_control == sock_ctrl)
875 		return 1; /* no change necessary */
876 
877 	sc->sc_socketreg->socket_control = sock_ctrl;
878 	status = sc->sc_socketreg->socket_state;
879 
880 	{
881 		int timeout = 20;
882 		u_int32_t sockevent;
883 		do {
884 			DELAY(20*1000);
885 			sockevent = sc->sc_socketreg->socket_event;
886 		} while (!(sockevent & PCCBB_SOCKET_EVENT_POWER) &&
887 			 --timeout > 0);
888 		/* reset event status */
889 		sc->sc_socketreg->socket_event = sockevent;
890 		if ( timeout < 0 ) {
891 			printf ("VCC supply failed.\n");
892 			return 0;
893 		}
894 	}
895 	/* XXX
896 	 * delay 400 ms: thgough the standard defines that the Vcc set-up time
897 	 * is 20 ms, some PC-Card bridge requires longer duration.
898 	 */
899 	DELAY(400*1000);
900 
901 	if (status & PCCBB_SOCKET_STAT_BADVCC) {
902 		device_printf(sc->sc_dev,
903 			      "bad Vcc request. ctrl=0x%x, status=0x%x\n",
904 			      sock_ctrl ,status);
905 		printf("pccbb_power: %s and %s [%x]\n",
906 		       (volts & CARD_VCCMASK) == CARD_VCC_UC ? "CARD_VCC_UC" :
907 		       (volts & CARD_VCCMASK) == CARD_VCC_5V ? "CARD_VCC_5V" :
908 		       (volts & CARD_VCCMASK) == CARD_VCC_3V ? "CARD_VCC_3V" :
909 		       (volts & CARD_VCCMASK) == CARD_VCC_XV ? "CARD_VCC_XV" :
910 		       (volts & CARD_VCCMASK) == CARD_VCC_YV ? "CARD_VCC_YV" :
911 		       (volts & CARD_VCCMASK) == CARD_VCC_0V ? "CARD_VCC_0V" :
912 		       "VCC-UNKNOWN",
913 		       (volts & CARD_VPPMASK) == CARD_VPP_UC ? "CARD_VPP_UC" :
914 		       (volts & CARD_VPPMASK) == CARD_VPP_12V ? "CARD_VPP_12V":
915 		       (volts & CARD_VPPMASK) == CARD_VPP_VCC ? "CARD_VPP_VCC":
916 		       (volts & CARD_VPPMASK) == CARD_VPP_0V ? "CARD_VPP_0V" :
917 		       "VPP-UNKNOWN",
918 		       volts);
919 		return 0;
920 	}
921 	return 1;		/* power changed correctly */
922 }
923 
924 /************************************************************************/
925 /* Cardbus power functions						*/
926 /************************************************************************/
927 
928 static void
929 pccbb_cardbus_reset(device_t dev)
930 {
931 	struct pccbb_softc *sc = device_get_softc(dev);
932 	int delay_us;
933 
934 	delay_us = sc->sc_chipset == CB_RF5C47X ? 400*1000 : 20*1000;
935 
936 	PCI_MASK_CONFIG(dev, PCCBBR_BRIDGECTRL, |PCCBBM_BRIDGECTRL_RESET, 2);
937 
938 	DELAY(delay_us);
939 
940 	/* If a card exists, unreset it! */
941 	if ((sc->sc_socketreg->socket_state & PCCBB_SOCKET_STAT_CD) == 0) {
942 		PCI_MASK_CONFIG(dev, PCCBBR_BRIDGECTRL,
943 				&~PCCBBM_BRIDGECTRL_RESET, 2);
944 		DELAY(delay_us);
945 	}
946 }
947 
948 static int
949 pccbb_cardbus_power_enable_socket(device_t self, device_t child)
950 {
951 	struct pccbb_softc *sc = device_get_softc(self);
952 	int voltage;
953 
954 	if ((sc->sc_socketreg->socket_state & PCCBB_SOCKET_STAT_CD)
955 	    == PCCBB_SOCKET_STAT_CD)
956 		return ENODEV;
957 
958 	voltage = pccbb_detect_voltage(self);
959 
960 	pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V);
961 	if (voltage & CARD_5V_CARD)
962 		pccbb_power(self, CARD_VCC_5V | CARD_VPP_VCC);
963 	else if (voltage & CARD_3V_CARD)
964 		pccbb_power(self, CARD_VCC_3V | CARD_VPP_VCC);
965 	else {
966 		device_printf(self, "Unknown card voltage\n");
967 		return ENXIO;
968 	}
969 
970 	pccbb_cardbus_reset(self);
971 	return 0;
972 }
973 
974 static void
975 pccbb_cardbus_power_disable_socket(device_t self, device_t child)
976 {
977 	pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V);
978 	pccbb_cardbus_reset(self);
979 }
980 
981 /************************************************************************/
982 /* Cardbus Resource							*/
983 /************************************************************************/
984 
985 static int
986 pccbb_cardbus_io_open(device_t dev, int win, u_int32_t start, u_int32_t end)
987 {
988 	int basereg;
989 	int limitreg;
990 
991 	if ((win < 0) || (win > 1)) {
992 		DEVPRINTF((dev,
993 			   "pccbb_cardbus_io_open: window out of range %d\n",
994 			   win));
995 		return EINVAL;
996 	}
997 
998 	basereg = win*8 + PCCBBR_IOBASE0;
999 	limitreg = win*8 + PCCBBR_IOLIMIT0;
1000 
1001 	pci_write_config(dev, basereg, start, 4);
1002 	pci_write_config(dev, limitreg, end, 4);
1003 	return 0;
1004 }
1005 
1006 static int
1007 pccbb_cardbus_mem_open(device_t dev, int win, u_int32_t start, u_int32_t end)
1008 {
1009 	int basereg;
1010 	int limitreg;
1011 
1012 	if ((win < 0) || (win > 1)) {
1013 		DEVPRINTF((dev,
1014 			   "pccbb_cardbus_mem_open: window out of range %d\n",
1015 			   win));
1016 		return EINVAL;
1017 	}
1018 
1019 	basereg = win*8 + PCCBBR_MEMBASE0;
1020 	limitreg = win*8 + PCCBBR_MEMLIMIT0;
1021 
1022 	pci_write_config(dev, basereg, start, 4);
1023 	pci_write_config(dev, limitreg, end, 4);
1024 	return 0;
1025 }
1026 
1027 static void
1028 pccbb_cardbus_auto_open(struct pccbb_softc *sc, int type)
1029 {
1030 	u_int32_t starts[2];
1031 	u_int32_t ends[2];
1032 	struct pccbb_reslist *rle;
1033 	int align;
1034 
1035 	starts[0] = starts[1] = 0xffffffff;
1036 	ends[0] = ends[1] = 0;
1037 
1038 	SLIST_FOREACH(rle, &sc->rl, entries) {
1039 		if (rle->type != type)
1040 			;
1041 		else if (starts[0] == 0xffffffff) {
1042 			starts[0] = rle->start;
1043 			ends[0] = rle->end;
1044 			rle->win = 0;
1045 		} else if (rle->end > ends[0] &&
1046 			   rle->start - ends[0] < PCCBB_AUTO_OPEN_SMALLHOLE) {
1047 			ends[0] = rle->end;
1048 			rle->win = 0;
1049 		} else if (rle->start < starts[0] &&
1050 			   starts[0] - rle->end < PCCBB_AUTO_OPEN_SMALLHOLE) {
1051 			starts[0] = rle->start;
1052 			rle->win = 0;
1053 		} else if (starts[1] == 0xffffffff) {
1054 			starts[1] = rle->start;
1055 			ends[1] = rle->end;
1056 			rle->win = 1;
1057 		} else if (rle->end > ends[1] &&
1058 			   rle->start - ends[1] < PCCBB_AUTO_OPEN_SMALLHOLE) {
1059 			ends[1] = rle->end;
1060 			rle->win = 1;
1061 		} else if (rle->start < starts[1] &&
1062 			   starts[1] - rle->end < PCCBB_AUTO_OPEN_SMALLHOLE) {
1063 			starts[1] = rle->start;
1064 			rle->win = 1;
1065 		} else {
1066 			u_int32_t diffs[2];
1067 
1068 			diffs[0] = diffs[1] = 0xffffffff;
1069 			if (rle->start > ends[0])
1070 				diffs[0] = rle->start - ends[0];
1071 			else if (rle->end < starts[0])
1072 				diffs[0] = starts[0] - rle->end;
1073 			if (rle->start > ends[1])
1074 				diffs[1] = rle->start - ends[1];
1075 			else if (rle->end < starts[1])
1076 				diffs[1] = starts[1] - rle->end;
1077 
1078 			rle->win = (diffs[0] <= diffs[1])?0:1;
1079 			if (rle->start > ends[rle->win])
1080 				ends[rle->win] = rle->end;
1081 			else if (rle->end < starts[rle->win])
1082 				starts[rle->win] = rle->start;
1083 		}
1084 	}
1085 	if (type == SYS_RES_MEMORY)
1086 		align = PCCBB_MEMALIGN;
1087 	else if (type == SYS_RES_IOPORT)
1088 		align = PCCBB_IOALIGN;
1089 	else
1090 		align = 1;
1091 
1092 	if (starts[0] != 0xffffffff)
1093 		starts[0] -= starts[0] % align;
1094 	if (starts[1] != 0xffffffff)
1095 		starts[1] -= starts[1] % align;
1096 	if (ends[0] % align != 0)
1097 		ends[0] += align - ends[0]%align - 1;
1098 	if (ends[1] % align != 0)
1099 		ends[1] += align - ends[1]%align - 1;
1100 
1101 	if (type == SYS_RES_MEMORY) {
1102 		pccbb_cardbus_mem_open(sc->sc_dev, 0, starts[0], ends[0]);
1103 		pccbb_cardbus_mem_open(sc->sc_dev, 1, starts[1], ends[1]);
1104 	} else if (type == SYS_RES_IOPORT) {
1105 		pccbb_cardbus_io_open(sc->sc_dev, 0, starts[0], ends[0]);
1106 		pccbb_cardbus_io_open(sc->sc_dev, 1, starts[1], ends[1]);
1107 	}
1108 }
1109 
1110 static int
1111 pccbb_cardbus_activate_resource(device_t self, device_t child, int type,
1112 				int rid, struct resource *r)
1113 {
1114 	struct pccbb_softc *sc = device_get_softc(self);
1115 	struct pccbb_reslist *rle;
1116 
1117 	if (type == SYS_RES_MEMORY || type == SYS_RES_IOPORT) {
1118 		SLIST_FOREACH(rle, &sc->rl, entries) {
1119 			if (type == rle->type && rid == rle->rid &&
1120 			    child == rle->odev)
1121 				return bus_generic_activate_resource(
1122 					self, child, type, rid, r);
1123 		}
1124 		rle = malloc(sizeof(struct pccbb_reslist), M_DEVBUF, M_WAITOK);
1125 		rle->type = type;
1126 		rle->rid = rid;
1127 		rle->start = rman_get_start(r);
1128 		rle->end = rman_get_end(r);
1129 		rle->odev = child;
1130 		rle->win = -1;
1131 		SLIST_INSERT_HEAD(&sc->rl, rle, entries);
1132 
1133 		pccbb_cardbus_auto_open(sc, type);
1134 	}
1135 	return bus_generic_activate_resource(self, child, type, rid, r);
1136 }
1137 
1138 static int
1139 pccbb_cardbus_deactivate_resource(device_t self, device_t child, int type,
1140 				  int rid, struct resource *r)
1141 {
1142 	struct pccbb_softc *sc = device_get_softc(self);
1143 	struct pccbb_reslist *rle;
1144 
1145 	SLIST_FOREACH(rle, &sc->rl, entries) {
1146 		if (type == rle->type && rid == rle->rid &&
1147 		    child == rle->odev) {
1148 			SLIST_REMOVE(&sc->rl, rle, pccbb_reslist, entries);
1149 			if (type == SYS_RES_IOPORT ||
1150 			    type == SYS_RES_MEMORY)
1151 				pccbb_cardbus_auto_open(sc, type);
1152 			free(rle, M_DEVBUF);
1153 			break;
1154 		}
1155 	}
1156 	return bus_generic_deactivate_resource(self, child, type, rid, r);
1157 }
1158 
1159 static struct resource*
1160 pccbb_cardbus_alloc_resource(device_t self, device_t child, int type, int* rid,
1161 			     u_long start, u_long end, u_long count,
1162 			     u_int flags)
1163 {
1164 	if (type == SYS_RES_IRQ) {
1165 		struct pccbb_softc *sc = device_get_softc(self);
1166 		if (start == 0) {
1167 			start = end = rman_get_start(sc->sc_irq_res);
1168 		}
1169 		return bus_generic_alloc_resource(self, child, type, rid,
1170 						  start, end, count, flags);
1171 	} else {
1172 		if (type == SYS_RES_MEMORY && start == 0 && end == ~0) {
1173 			start = CARDBUS_SYS_RES_MEMORY_START;
1174 			end = CARDBUS_SYS_RES_MEMORY_END;
1175 		} else if (type == SYS_RES_IOPORT && start == 0 && end == ~0) {
1176 			start = CARDBUS_SYS_RES_IOPORT_START;
1177 			end = CARDBUS_SYS_RES_IOPORT_END;
1178 		}
1179 		return bus_generic_alloc_resource(self, child, type, rid,
1180 						  start, end, count, flags);
1181 	}
1182 }
1183 
1184 static int
1185 pccbb_cardbus_release_resource(device_t self, device_t child, int type,
1186 			       int rid, struct resource *r)
1187 {
1188 	return bus_generic_release_resource(self, child, type, rid, r);
1189 }
1190 
1191 /************************************************************************/
1192 /* PC Card Power Functions						*/
1193 /************************************************************************/
1194 
1195 static int
1196 pccbb_pcic_power_enable_socket(device_t self, device_t child)
1197 {
1198 	struct pccbb_softc *sc = device_get_softc(self);
1199 
1200 	DPRINTF(("pccbb_pcic_socket_enable:\n"));
1201 
1202 	/* power down/up the socket to reset */
1203 	{
1204 		int voltage = pccbb_detect_voltage(self);
1205 
1206 		pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V);
1207 		if (voltage & CARD_5V_CARD)
1208 			pccbb_power(self, CARD_VCC_5V | CARD_VPP_VCC);
1209 		else if (voltage & CARD_3V_CARD)
1210 			pccbb_power(self, CARD_VCC_3V | CARD_VPP_VCC);
1211 		else {
1212 			device_printf(self, "Unknown card voltage\n");
1213 			return ENXIO;
1214 		}
1215 	}
1216 
1217 	/* enable socket i/o */
1218 	PCIC_MASK(sc, PCIC_PWRCTL, | PCIC_PWRCTL_OE);
1219 
1220 	PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE);
1221 	/* hold reset for 30ms */
1222 	DELAY(30*1000);
1223 	/* clear the reset flag */
1224 	PCIC_MASK(sc, PCIC_INTR, | PCIC_INTR_RESET);
1225 	/* wait 20ms as per pc card standard (r2.01) section 4.3.6 */
1226 	DELAY(20*1000);
1227 
1228 	pccbb_pcic_wait_ready(sc);
1229 
1230 	/* disable all address windows */
1231 	PCIC_WRITE(sc, PCIC_ADDRWIN_ENABLE, 0);
1232 
1233 	{
1234 		int cardtype;
1235 		CARD_GET_TYPE(child, &cardtype);
1236 		PCIC_MASK(sc, PCIC_INTR, | ((cardtype == PCCARD_IFTYPE_IO) ?
1237 					    PCIC_INTR_CARDTYPE_IO :
1238 					    PCIC_INTR_CARDTYPE_MEM));
1239 		DEVPRINTF((sc->sc_dev, "card type is %s\n",
1240 			   (cardtype == PCCARD_IFTYPE_IO) ? "io" : "mem"));
1241 	}
1242 
1243 	/* reinstall all the memory and io mappings */
1244 	{
1245 		int win;
1246 
1247 		for (win = 0; win < PCIC_MEM_WINS; ++win) {
1248 			if (sc->memalloc & (1 << win)) {
1249 				pccbb_pcic_do_mem_map(sc, win);
1250 			}
1251 		}
1252 		for (win = 0; win < PCIC_IO_WINS; ++win) {
1253 			if (sc->ioalloc & (1 << win)) {
1254 				pccbb_pcic_do_io_map(sc, win);
1255 			}
1256 		}
1257 	}
1258 	return 0;
1259 }
1260 
1261 static void
1262 pccbb_pcic_power_disable_socket(device_t self, device_t child)
1263 {
1264 	struct pccbb_softc *sc = device_get_softc(self);
1265 
1266 	DPRINTF(("pccbb_pcic_socket_disable\n"));
1267 
1268 	/* reset signal asserting... */
1269 	PCIC_MASK(sc, PCIC_INTR, & ~PCIC_INTR_RESET);
1270 	DELAY(2*1000);
1271 
1272 	/* power down the socket */
1273 	PCIC_MASK(sc, PCIC_PWRCTL, &~PCIC_PWRCTL_OE);
1274 	pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V);
1275 
1276 	/* wait 300ms until power fails (Tpf). */
1277 	DELAY(300 * 1000);
1278 }
1279 
1280 /************************************************************************/
1281 /* PC Card Resource Functions						*/
1282 /************************************************************************/
1283 
1284 static void
1285 pccbb_pcic_wait_ready(struct pccbb_softc *sc)
1286 {
1287 	int i;
1288 	DEVPRINTF((sc->sc_dev, "pccbb_pcic_wait_ready: status 0x%02x\n",
1289 		   PCIC_READ(sc, PCIC_IF_STATUS)));
1290 	for (i = 0; i < 10000; i++) {
1291 		if (PCIC_READ(sc, PCIC_IF_STATUS) & PCIC_IF_STATUS_READY) {
1292 			return;
1293 		}
1294 		DELAY(500);
1295 	}
1296 	device_printf(sc->sc_dev, "ready never happened, status = %02x\n",
1297 		      PCIC_READ(sc, PCIC_IF_STATUS));
1298 }
1299 
1300 #define PCIC_MEMINFO(NUM) { \
1301 	PCIC_SYSMEM_ADDR ## NUM ## _START_LSB, \
1302 	PCIC_SYSMEM_ADDR ## NUM ## _START_MSB, \
1303 	PCIC_SYSMEM_ADDR ## NUM ## _STOP_LSB, \
1304 	PCIC_SYSMEM_ADDR ## NUM ## _STOP_MSB, \
1305 	PCIC_SYSMEM_ADDR ## NUM ## _WIN, \
1306 	PCIC_CARDMEM_ADDR ## NUM ## _LSB, \
1307 	PCIC_CARDMEM_ADDR ## NUM ## _MSB, \
1308 	PCIC_ADDRWIN_ENABLE_MEM ## NUM ## , \
1309 }
1310 
1311 static struct mem_map_index_st {
1312 	int sysmem_start_lsb;
1313 	int sysmem_start_msb;
1314 	int sysmem_stop_lsb;
1315 	int sysmem_stop_msb;
1316 	int sysmem_win;
1317 	int cardmem_lsb;
1318 	int cardmem_msb;
1319 	int memenable;
1320 } mem_map_index[] = {
1321 	PCIC_MEMINFO(0),
1322 	PCIC_MEMINFO(1),
1323 	PCIC_MEMINFO(2),
1324 	PCIC_MEMINFO(3),
1325 	PCIC_MEMINFO(4),
1326 };
1327 #undef PCIC_MEMINFO
1328 
1329 static void
1330 pccbb_pcic_do_mem_map(struct pccbb_softc *sc, int win)
1331 {
1332 	PCIC_WRITE(sc, mem_map_index[win].sysmem_start_lsb,
1333 		   (sc->mem[win].addr >> PCIC_SYSMEM_ADDRX_SHIFT) & 0xff);
1334 	PCIC_WRITE(sc, mem_map_index[win].sysmem_start_msb,
1335 		   ((sc->mem[win].addr >> (PCIC_SYSMEM_ADDRX_SHIFT + 8)) &
1336 		    PCIC_SYSMEM_ADDRX_START_MSB_ADDR_MASK) | 0x80);
1337 
1338 	PCIC_WRITE(sc, mem_map_index[win].sysmem_stop_lsb,
1339 		   ((sc->mem[win].addr + sc->mem[win].realsize - 1) >>
1340 		    PCIC_SYSMEM_ADDRX_SHIFT) & 0xff);
1341 	PCIC_WRITE(sc, mem_map_index[win].sysmem_stop_msb,
1342 		   (((sc->mem[win].addr + sc->mem[win].realsize - 1) >>
1343 		     (PCIC_SYSMEM_ADDRX_SHIFT + 8)) &
1344 		    PCIC_SYSMEM_ADDRX_STOP_MSB_ADDR_MASK) |
1345 		   PCIC_SYSMEM_ADDRX_STOP_MSB_WAIT2);
1346 
1347 	PCIC_WRITE(sc, mem_map_index[win].sysmem_win,
1348 		   (sc->mem[win].addr >> PCIC_MEMREG_WIN_SHIFT) & 0xff);
1349 
1350 	PCIC_WRITE(sc, mem_map_index[win].cardmem_lsb,
1351 		   (sc->mem[win].offset >> PCIC_CARDMEM_ADDRX_SHIFT) & 0xff);
1352 	PCIC_WRITE(sc, mem_map_index[win].cardmem_msb,
1353 		   ((sc->mem[win].offset >> (PCIC_CARDMEM_ADDRX_SHIFT + 8)) &
1354 		    PCIC_CARDMEM_ADDRX_MSB_ADDR_MASK) |
1355 		   ((sc->mem[win].kind == PCCARD_MEM_ATTR) ?
1356 		    PCIC_CARDMEM_ADDRX_MSB_REGACTIVE_ATTR : 0));
1357 
1358 	PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, | PCIC_ADDRWIN_ENABLE_MEMCS16
1359 		  			   | mem_map_index[win].memenable);
1360 
1361 	DELAY(100);
1362 
1363 #ifdef CBB_DEBUG
1364 	{
1365 		int r1, r2, r3, r4, r5, r6, r7;
1366 		r1 = PCIC_READ(sc, mem_map_index[win].sysmem_start_msb);
1367 		r2 = PCIC_READ(sc, mem_map_index[win].sysmem_start_lsb);
1368 		r3 = PCIC_READ(sc, mem_map_index[win].sysmem_stop_msb);
1369 		r4 = PCIC_READ(sc, mem_map_index[win].sysmem_stop_lsb);
1370 		r5 = PCIC_READ(sc, mem_map_index[win].cardmem_msb);
1371 		r6 = PCIC_READ(sc, mem_map_index[win].cardmem_lsb);
1372 		r7 = PCIC_READ(sc, mem_map_index[win].sysmem_win);
1373 		DPRINTF(("pccbb_pcic_do_mem_map window %d: %02x%02x %02x%02x "
1374 			 "%02x%02x %02x (%08x+%08x.%08x*%08lx)\n", win, r1, r2, r3, r4, r5, r6, r7,
1375 			 sc->mem[win].addr, sc->mem[win].size, sc->mem[win].realsize, sc->mem[win].offset));
1376 	}
1377 #endif
1378 }
1379 
1380 static int
1381 pccbb_pcic_mem_map(struct pccbb_softc *sc, int kind,
1382 		   struct resource *r, bus_addr_t card_addr, int *win)
1383 {
1384 	int i;
1385 
1386 	*win = -1;
1387 	for (i = 0; i < PCIC_MEM_WINS; i++) {
1388 		if ((sc->memalloc & (1 << i)) == 0) {
1389 			*win = i;
1390 			sc->memalloc |= (1 << i);
1391 			break;
1392 		}
1393 	}
1394 	if (*win == -1)
1395 		return (1);
1396 
1397 	card_addr = card_addr - card_addr % PCIC_MEM_PAGESIZE;
1398 	sc->mem[*win].memt = rman_get_bustag(r);
1399 	sc->mem[*win].memh = rman_get_bushandle(r);
1400 	sc->mem[*win].addr = rman_get_start(r);
1401 	sc->mem[*win].size = rman_get_end(r) - sc->mem[*win].addr + 1;
1402 	sc->mem[*win].realsize = sc->mem[*win].size + PCIC_MEM_PAGESIZE - 1;
1403 	sc->mem[*win].realsize = sc->mem[*win].realsize -
1404 				 (sc->mem[*win].realsize % PCIC_MEM_PAGESIZE);
1405 	sc->mem[*win].offset = ((long)card_addr) -
1406 			       ((long)(sc->mem[*win].addr));
1407 	sc->mem[*win].kind = kind;
1408 
1409 	DPRINTF(("pccbb_pcic_mem_map window %d bus %x+%x+%lx card addr %x\n",
1410 		 *win, sc->mem[*win].addr, sc->mem[*win].size,
1411 		 sc->mem[*win].offset, card_addr));
1412 
1413 	pccbb_pcic_do_mem_map(sc, *win);
1414 
1415 	return (0);
1416 }
1417 
1418 static void
1419 pccbb_pcic_mem_unmap(struct pccbb_softc *sc, int window)
1420 {
1421 	if (window >= PCIC_MEM_WINS)
1422 		panic("pccbb_pcic_mem_unmap: window out of range");
1423 
1424 	PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, & ~mem_map_index[window].memenable);
1425 	sc->memalloc &= ~(1 << window);
1426 }
1427 
1428 #define PCIC_IOINFO(NUM) { \
1429 	PCIC_IOADDR ## NUM ## _START_LSB, \
1430 	PCIC_IOADDR ## NUM ## _START_MSB, \
1431 	PCIC_IOADDR ## NUM ## _STOP_LSB, \
1432 	PCIC_IOADDR ## NUM ## _STOP_MSB, \
1433 	PCIC_ADDRWIN_ENABLE_IO ## NUM ## , \
1434 	PCIC_IOCTL_IO ## NUM ## _WAITSTATE \
1435 	| PCIC_IOCTL_IO ## NUM ## _ZEROWAIT \
1436 	| PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_MASK \
1437 	| PCIC_IOCTL_IO ## NUM ## _DATASIZE_MASK, \
1438 	{ \
1439 		PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_CARD, \
1440 		PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_DATASIZE \
1441 		| PCIC_IOCTL_IO ## NUM ## _DATASIZE_8BIT, \
1442 		PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_DATASIZE \
1443 		| PCIC_IOCTL_IO ## NUM ## _DATASIZE_16BIT, \
1444 	} \
1445 }
1446 
1447 static struct io_map_index_st {
1448 	int start_lsb;
1449 	int start_msb;
1450 	int stop_lsb;
1451 	int stop_msb;
1452 	int ioenable;
1453 	int ioctlmask;
1454 	int ioctlbits[3]; /* indexed by PCCARD_WIDTH_* */
1455 } io_map_index[] = {
1456 	PCIC_IOINFO(0),
1457 	PCIC_IOINFO(1),
1458 };
1459 #undef PCIC_IOINFO
1460 
1461 static void pccbb_pcic_do_io_map(struct pccbb_softc *sc, int win)
1462 {
1463 	PCIC_WRITE(sc, io_map_index[win].start_lsb, sc->io[win].addr & 0xff);
1464 	PCIC_WRITE(sc, io_map_index[win].start_msb,
1465 		   (sc->io[win].addr >> 8) & 0xff);
1466 
1467 	PCIC_WRITE(sc, io_map_index[win].stop_lsb,
1468 		   (sc->io[win].addr + sc->io[win].size - 1) & 0xff);
1469 	PCIC_WRITE(sc, io_map_index[win].stop_msb,
1470 		   ((sc->io[win].addr + sc->io[win].size - 1) >> 8) & 0xff);
1471 
1472 	PCIC_MASK2(sc, PCIC_IOCTL,
1473 		   & ~io_map_index[win].ioctlmask,
1474 		   | io_map_index[win].ioctlbits[sc->io[win].width]);
1475 
1476 	PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, | io_map_index[win].ioenable);
1477 #ifdef CBB_DEBUG
1478 	{
1479 		int r1, r2, r3, r4;
1480 		r1 = PCIC_READ(sc, io_map_index[win].start_msb);
1481 		r2 = PCIC_READ(sc, io_map_index[win].start_lsb);
1482 		r3 = PCIC_READ(sc, io_map_index[win].stop_msb);
1483 		r4 = PCIC_READ(sc, io_map_index[win].stop_lsb);
1484 		DPRINTF(("pccbb_pcic_do_io_map window %d: %02x%02x %02x%02x "
1485 			 "(%08x+%08x)\n", win, r1, r2, r3, r4,
1486 			 sc->io[win].addr, sc->io[win].size));
1487 	}
1488 #endif
1489 }
1490 
1491 static int
1492 pccbb_pcic_io_map(struct pccbb_softc *sc, int width,
1493 		  struct resource *r, bus_addr_t card_addr, int *win)
1494 {
1495 	int i;
1496 #ifdef CBB_DEBUG
1497 	static char *width_names[] = { "auto", "io8", "io16"};
1498 #endif
1499 
1500 	*win = -1;
1501 	for (i=0; i < PCIC_IO_WINS; i++) {
1502 		if ((sc->ioalloc & (1 << i)) == 0) {
1503 			*win = i;
1504 			sc->ioalloc |= (1 << i);
1505 			break;
1506 		}
1507 	}
1508 	if (*win == -1)
1509 		return (1);
1510 
1511 	sc->io[*win].iot = rman_get_bustag(r);
1512 	sc->io[*win].ioh = rman_get_bushandle(r);
1513 	sc->io[*win].addr = rman_get_start(r);
1514 	sc->io[*win].size = rman_get_end(r) - sc->io[*win].addr + 1;
1515 	sc->io[*win].flags = 0;
1516 	sc->io[*win].width = width;
1517 
1518 	DPRINTF(("pccbb_pcic_io_map window %d %s port %x+%x\n",
1519 		 *win, width_names[width], sc->io[*win].addr,
1520 		 sc->io[*win].size));
1521 
1522 	pccbb_pcic_do_io_map(sc, *win);
1523 
1524 	return (0);
1525 }
1526 
1527 static void
1528 pccbb_pcic_io_unmap(struct pccbb_softc *sc, int window)
1529 {
1530 	if (window >= PCIC_IO_WINS)
1531 		panic("pccbb_pcic_io_unmap: window out of range");
1532 
1533 	PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, & ~io_map_index[window].ioenable);
1534 
1535 	sc->ioalloc &= ~(1 << window);
1536 
1537 	sc->io[window].iot = 0;
1538 	sc->io[window].ioh = 0;
1539 	sc->io[window].addr = 0;
1540 	sc->io[window].size = 0;
1541 	sc->io[window].flags = 0;
1542 	sc->io[window].width = 0;
1543 }
1544 
1545 static int
1546 pccbb_pcic_activate_resource(device_t self, device_t child, int type, int rid,
1547 			     struct resource *r)
1548 {
1549 	int err;
1550 	int win;
1551 	struct pccbb_reslist *rle;
1552 	struct pccbb_softc *sc = device_get_softc(self);
1553 
1554 	if (rman_get_flags(r) & RF_ACTIVE)
1555 		return 0;
1556 
1557 	switch (type) {
1558 	case SYS_RES_IOPORT:
1559 		err = pccbb_pcic_io_map(sc, 0, r, 0, &win);
1560 		if (err)
1561 			return err;
1562 		break;
1563 	case SYS_RES_MEMORY:
1564 		err = pccbb_pcic_mem_map(sc, 0, r, 0, &win);
1565 		if (err)
1566 			return err;
1567 		break;
1568 	default:
1569 		break;
1570 	}
1571 	SLIST_FOREACH(rle, &sc->rl, entries) {
1572 		if (type == rle->type && rid == rle->rid &&
1573 		    child == rle->odev) {
1574 			rle->win = win;
1575 			break;
1576 		}
1577 	}
1578 	err = bus_generic_activate_resource(self, child, type, rid, r);
1579 	return (err);
1580 }
1581 
1582 static int
1583 pccbb_pcic_deactivate_resource(device_t self, device_t child, int type,
1584 			       int rid, struct resource *r)
1585 {
1586 	struct pccbb_softc *sc = device_get_softc(self);
1587 	int win;
1588 	struct pccbb_reslist *rle;
1589 	win = -1;
1590 	SLIST_FOREACH(rle, &sc->rl, entries) {
1591 		if (type == rle->type && rid == rle->rid &&
1592 		    child == rle->odev) {
1593 			win = rle->win;
1594 			break;
1595 		}
1596 	}
1597 	if (win == -1) {
1598 		panic("pccbb_pcic: deactivating bogus resoure");
1599 		return 1;
1600 	}
1601 
1602 	switch (type) {
1603 	case SYS_RES_IOPORT:
1604 		pccbb_pcic_io_unmap(sc, win);
1605 		break;
1606 	case SYS_RES_MEMORY:
1607 		pccbb_pcic_mem_unmap(sc, win);
1608 		break;
1609 	default:
1610 		break;
1611 	}
1612 	return bus_generic_deactivate_resource(self, child, type, rid, r);
1613 }
1614 
1615 static struct resource*
1616 pccbb_pcic_alloc_resource(device_t self, device_t child, int type, int* rid,
1617 			  u_long start, u_long end, u_long count, u_int flags)
1618 {
1619 	struct resource *r = NULL;
1620 	struct pccbb_softc *sc = device_get_softc(self);
1621 	struct pccbb_reslist *rle;
1622 
1623 	if ((sc->sc_flags & PCCBB_PCIC_MEM_32) == 0) {
1624 		panic("PCCBB bridge cannot handle non MEM_32 bridges\n");
1625 	}
1626 
1627 	switch (type) {
1628 	case SYS_RES_MEMORY:
1629 		/* Nearly default */
1630 		if (start == 0 && end == ~0 && count != 1) {
1631 			start = CARDBUS_SYS_RES_MEMORY_START; /* XXX -- should be tweakable*/
1632 			end = CARDBUS_SYS_RES_MEMORY_END;
1633 		}
1634 		flags = (flags & ~RF_ALIGNMENT_MASK)
1635 			| rman_make_alignment_flags(PCCBB_MEMALIGN);
1636 		break;
1637 	case SYS_RES_IOPORT:
1638 		if (start < 0x100)
1639 			start = 0x100;		/* XXX tweakable? */
1640 		if (end < start)
1641 			end = start;
1642 		break;
1643 	case SYS_RES_IRQ:
1644 		flags |= RF_SHAREABLE;
1645 		start = end = rman_get_start(sc->sc_irq_res);
1646 		break;
1647 	}
1648 	r = bus_generic_alloc_resource(self, child, type, rid, start, end,
1649 				       count, flags & ~RF_ACTIVE);
1650 	if (r == NULL)
1651 		return NULL;
1652 
1653 	rle = malloc(sizeof(struct pccbb_reslist), M_DEVBUF, M_WAITOK);
1654 	rle->type = type;
1655 	rle->rid = *rid;
1656 	rle->start = rman_get_start(r);
1657 	rle->end = rman_get_end(r);
1658 	rle->odev = child;
1659 	rle->win = -1;
1660 	SLIST_INSERT_HEAD(&sc->rl, rle, entries);
1661 
1662 	if (flags & RF_ACTIVE) {
1663 		if (bus_activate_resource(child, type, *rid, r) != 0) {
1664 			BUS_RELEASE_RESOURCE(self, child, type, *rid, r);
1665 			return NULL;
1666 		}
1667 	}
1668 
1669 	return r;
1670 }
1671 
1672 static int
1673 pccbb_pcic_release_resource(device_t self, device_t child, int type,
1674 			    int rid, struct resource *res)
1675 {
1676 	struct pccbb_softc *sc = device_get_softc(self);
1677 	struct pccbb_reslist *rle;
1678 	int count = 0;
1679 
1680 	if (rman_get_flags(res) & RF_ACTIVE) {
1681 		int error;
1682 		error = bus_deactivate_resource(child, type, rid, res);
1683 		if (error != 0)
1684 			return error;
1685 	}
1686 
1687 	SLIST_FOREACH(rle, &sc->rl, entries) {
1688 		if (type == rle->type && rid == rle->rid &&
1689 		    child == rle->odev) {
1690 			SLIST_REMOVE(&sc->rl, rle, pccbb_reslist, entries);
1691 			free(rle, M_DEVBUF);
1692 			count++;
1693 			break;
1694 		}
1695 	}
1696 	if (count == 0) {
1697 		panic("pccbb_pcic: releasing bogus resource");
1698 	}
1699 
1700 	return bus_generic_release_resource(self, child, type, rid, res);
1701 }
1702 
1703 /************************************************************************/
1704 /* PC Card methods							*/
1705 /************************************************************************/
1706 
1707 static int
1708 pccbb_pcic_set_res_flags(device_t self, device_t child, int type, int rid,
1709 			 u_int32_t flags)
1710 {
1711 	struct pccbb_softc *sc = device_get_softc(self);
1712 
1713 	if (type != SYS_RES_MEMORY)
1714 		return (EINVAL);
1715 	sc->mem[rid].kind = flags;
1716 	pccbb_pcic_do_mem_map(sc, rid);
1717 	return 0;
1718 }
1719 
1720 static int
1721 pccbb_pcic_set_memory_offset(device_t self, device_t child, int rid,
1722     u_int32_t cardaddr, u_int32_t *deltap)
1723 {
1724 	struct pccbb_softc *sc = device_get_softc(self);
1725 	int win;
1726 	struct pccbb_reslist *rle;
1727 	u_int32_t delta;
1728 
1729 	win = -1;
1730 
1731 	SLIST_FOREACH(rle, &sc->rl, entries) {
1732 		if (SYS_RES_MEMORY == rle->type && rid == rle->rid &&
1733 		    child == rle->odev) {
1734 			win = rle->win;
1735 			break;
1736 		}
1737 	}
1738 	if (win == -1) {
1739 		panic("pccbb_pcic: setting memory offset of bogus resource");
1740 		return 1;
1741 	}
1742 
1743 	delta = cardaddr % PCIC_MEM_PAGESIZE;
1744 	if (deltap)
1745 		*deltap = delta;
1746 	cardaddr -= delta;
1747 	sc->mem[win].realsize = sc->mem[win].size + delta +
1748 	    PCIC_MEM_PAGESIZE - 1;
1749 	sc->mem[win].realsize = sc->mem[win].realsize -
1750 	    (sc->mem[win].realsize % PCIC_MEM_PAGESIZE);
1751 	sc->mem[win].offset = cardaddr - sc->mem[win].addr;
1752 	pccbb_pcic_do_mem_map(sc, win);
1753 
1754 	return 0;
1755 }
1756 
1757 /************************************************************************/
1758 /* POWER methods							*/
1759 /************************************************************************/
1760 
1761 static int
1762 pccbb_power_enable_socket(device_t self, device_t child)
1763 {
1764 	struct pccbb_softc *sc = device_get_softc(self);
1765 
1766 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1767 		return pccbb_pcic_power_enable_socket(self, child);
1768 	else
1769 		return pccbb_cardbus_power_enable_socket(self, child);
1770 }
1771 
1772 static void
1773 pccbb_power_disable_socket(device_t self, device_t child)
1774 {
1775 	struct pccbb_softc *sc = device_get_softc(self);
1776 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1777 		pccbb_pcic_power_disable_socket(self, child);
1778 	else
1779 		pccbb_cardbus_power_disable_socket(self, child);
1780 }
1781 /************************************************************************/
1782 /* BUS Methods								*/
1783 /************************************************************************/
1784 
1785 
1786 static int
1787 pccbb_activate_resource(device_t self, device_t child, int type, int rid,
1788 			struct resource *r)
1789 {
1790 	struct pccbb_softc *sc = device_get_softc(self);
1791 
1792 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1793 		return pccbb_pcic_activate_resource(self, child, type, rid, r);
1794 	else
1795 		return pccbb_cardbus_activate_resource(self, child, type, rid,
1796 						       r);
1797 }
1798 
1799 static int
1800 pccbb_deactivate_resource(device_t self, device_t child, int type,
1801 			  int rid, struct resource *r)
1802 {
1803 	struct pccbb_softc *sc = device_get_softc(self);
1804 
1805 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1806 		return pccbb_pcic_deactivate_resource(self, child, type,
1807 						      rid, r);
1808 	else
1809 		return pccbb_cardbus_deactivate_resource(self, child, type,
1810 						      rid, r);
1811 }
1812 
1813 static struct resource*
1814 pccbb_alloc_resource(device_t self, device_t child, int type, int* rid,
1815 		     u_long start, u_long end, u_long count, u_int flags)
1816 {
1817 	struct pccbb_softc *sc = device_get_softc(self);
1818 
1819 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1820 		return pccbb_pcic_alloc_resource(self, child, type, rid,
1821 						 start, end, count, flags);
1822 	else
1823 		return pccbb_cardbus_alloc_resource(self, child, type, rid,
1824 						 start, end, count, flags);
1825 }
1826 
1827 static int
1828 pccbb_release_resource(device_t self, device_t child, int type, int rid,
1829 		       struct resource *r)
1830 {
1831 	struct pccbb_softc *sc = device_get_softc(self);
1832 
1833 	if (sc->sc_flags & PCCBB_16BIT_CARD)
1834 		return pccbb_pcic_release_resource(self, child, type,
1835 						   rid, r);
1836 	else
1837 		return pccbb_cardbus_release_resource(self, child, type,
1838 						      rid, r);
1839 }
1840 
1841 static int
1842 pccbb_read_ivar(device_t dev, device_t child, int which, uintptr_t *result)
1843 {
1844 	struct pccbb_softc	*sc = device_get_softc(dev);
1845 
1846 	switch (which) {
1847 	case PCIB_IVAR_BUS:
1848 		*result = sc->sc_secbus;
1849 		return(0);
1850 	}
1851 	return(ENOENT);
1852 }
1853 
1854 static int
1855 pccbb_write_ivar(device_t dev, device_t child, int which, uintptr_t value)
1856 {
1857 	struct pccbb_softc	*sc = device_get_softc(dev);
1858 
1859 	switch (which) {
1860 	case PCIB_IVAR_BUS:
1861 		sc->sc_secbus = value;
1862 		break;
1863 	}
1864 	return(ENOENT);
1865 }
1866 
1867 /************************************************************************/
1868 /* PCI compat methods							*/
1869 /************************************************************************/
1870 
1871 static int
1872 pccbb_maxslots(device_t dev)
1873 {
1874 	return 0;
1875 }
1876 
1877 static u_int32_t
1878 pccbb_read_config(device_t dev, int b, int s, int f, int reg, int width)
1879 {
1880 	/*
1881 	 * Pass through to the next ppb up the chain (i.e. our grandparent).
1882 	 */
1883 	return PCIB_READ_CONFIG(device_get_parent(device_get_parent(dev)),
1884 				b, s, f, reg, width);
1885 }
1886 
1887 static void
1888 pccbb_write_config(device_t dev, int b, int s, int f, int reg, u_int32_t val,
1889 		   int width)
1890 {
1891 	/*
1892 	 * Pass through to the next ppb up the chain (i.e. our grandparent).
1893 	 */
1894 	PCIB_WRITE_CONFIG(device_get_parent(device_get_parent(dev)),
1895 			  b, s, f, reg, val, width);
1896 }
1897 
1898 static device_method_t pccbb_methods[] = {
1899 	/* Device interface */
1900 	DEVMETHOD(device_probe,			pccbb_probe),
1901 	DEVMETHOD(device_attach,		pccbb_attach),
1902 	DEVMETHOD(device_detach,		pccbb_detach),
1903 	DEVMETHOD(device_suspend,		bus_generic_suspend),
1904 	DEVMETHOD(device_resume,		bus_generic_resume),
1905 
1906 	/* bus methods */
1907 	DEVMETHOD(bus_print_child,		bus_generic_print_child),
1908 	DEVMETHOD(bus_read_ivar,		pccbb_read_ivar),
1909 	DEVMETHOD(bus_write_ivar,		pccbb_write_ivar),
1910 	DEVMETHOD(bus_alloc_resource,		pccbb_alloc_resource),
1911 	DEVMETHOD(bus_release_resource,		pccbb_release_resource),
1912 	DEVMETHOD(bus_activate_resource,	pccbb_activate_resource),
1913 	DEVMETHOD(bus_deactivate_resource,	pccbb_deactivate_resource),
1914 	DEVMETHOD(bus_driver_added,		pccbb_driver_added),
1915 	DEVMETHOD(bus_child_detached,		pccbb_child_detached),
1916 	DEVMETHOD(bus_setup_intr,		bus_generic_setup_intr),
1917 	DEVMETHOD(bus_teardown_intr,		bus_generic_teardown_intr),
1918 
1919         /* 16-bit card interface */
1920 	DEVMETHOD(card_set_res_flags,		pccbb_pcic_set_res_flags),
1921 	DEVMETHOD(card_set_memory_offset,	pccbb_pcic_set_memory_offset),
1922 
1923         /* power interface */
1924 	DEVMETHOD(power_enable_socket,		pccbb_power_enable_socket),
1925 	DEVMETHOD(power_disable_socket,		pccbb_power_disable_socket),
1926 
1927         /* pcib compatibility interface */
1928 	DEVMETHOD(pcib_maxslots,		pccbb_maxslots),
1929 	DEVMETHOD(pcib_read_config,		pccbb_read_config),
1930 	DEVMETHOD(pcib_write_config,		pccbb_write_config),
1931 	{0,0}
1932 };
1933 
1934 static driver_t pccbb_driver = {
1935 	"pccbb",
1936 	pccbb_methods,
1937 	sizeof(struct pccbb_softc)
1938 };
1939 
1940 static devclass_t pccbb_devclass;
1941 
1942 DRIVER_MODULE(pccbb, pci, pccbb_driver, pccbb_devclass, 0, 0);
1943 
1944 SYSINIT(pccbb, SI_SUB_KTHREAD_IDLE, SI_ORDER_ANY, pccbb_start_threads, 0);
1945