xref: /freebsd/sys/dev/bhnd/bcma/bcma.c (revision 094fc1ed0f2627525c7b0342efcbad5be7a8546a)
1 /*-
2  * Copyright (c) 2015 Landon Fuller <landon@landonf.org>
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.
11  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12  *    similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any
13  *    redistribution must be conditioned upon including a substantially
14  *    similar Disclaimer requirement for further binary redistribution.
15  *
16  * NO WARRANTY
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19  * LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY
20  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
21  * THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY,
22  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
27  * THE POSSIBILITY OF SUCH DAMAGES.
28  */
29 
30 #include <sys/cdefs.h>
31 __FBSDID("$FreeBSD$");
32 
33 #include <sys/param.h>
34 #include <sys/bus.h>
35 #include <sys/kernel.h>
36 #include <sys/malloc.h>
37 #include <sys/module.h>
38 #include <sys/systm.h>
39 
40 #include <machine/bus.h>
41 
42 #include <dev/bhnd/cores/pmu/bhnd_pmu.h>
43 
44 #include "bcma_dmp.h"
45 
46 #include "bcma_eromreg.h"
47 #include "bcma_eromvar.h"
48 
49 #include "bcmavar.h"
50 
51 /* RID used when allocating EROM table */
52 #define	BCMA_EROM_RID	0
53 
54 static bhnd_erom_class_t *
55 bcma_get_erom_class(driver_t *driver)
56 {
57 	return (&bcma_erom_parser);
58 }
59 
60 int
61 bcma_probe(device_t dev)
62 {
63 	device_set_desc(dev, "BCMA BHND bus");
64 	return (BUS_PROBE_DEFAULT);
65 }
66 
67 /**
68  * Default bcma(4) bus driver implementation of DEVICE_ATTACH().
69  *
70  * This implementation initializes internal bcma(4) state and performs
71  * bus enumeration, and must be called by subclassing drivers in
72  * DEVICE_ATTACH() before any other bus methods.
73  */
74 int
75 bcma_attach(device_t dev)
76 {
77 	int error;
78 
79 	/* Enumerate children */
80 	if ((error = bcma_add_children(dev))) {
81 		device_delete_children(dev);
82 		return (error);
83 	}
84 
85 	return (0);
86 }
87 
88 int
89 bcma_detach(device_t dev)
90 {
91 	return (bhnd_generic_detach(dev));
92 }
93 
94 static device_t
95 bcma_add_child(device_t dev, u_int order, const char *name, int unit)
96 {
97 	struct bcma_devinfo	*dinfo;
98 	device_t		 child;
99 
100 	child = device_add_child_ordered(dev, order, name, unit);
101 	if (child == NULL)
102 		return (NULL);
103 
104 	if ((dinfo = bcma_alloc_dinfo(dev)) == NULL) {
105 		device_delete_child(dev, child);
106 		return (NULL);
107 	}
108 
109 	device_set_ivars(child, dinfo);
110 
111 	return (child);
112 }
113 
114 static void
115 bcma_child_deleted(device_t dev, device_t child)
116 {
117 	struct bhnd_softc	*sc;
118 	struct bcma_devinfo	*dinfo;
119 
120 	sc = device_get_softc(dev);
121 
122 	/* Call required bhnd(4) implementation */
123 	bhnd_generic_child_deleted(dev, child);
124 
125 	/* Free bcma device info */
126 	if ((dinfo = device_get_ivars(child)) != NULL)
127 		bcma_free_dinfo(dev, dinfo);
128 
129 	device_set_ivars(child, NULL);
130 }
131 
132 static int
133 bcma_read_ivar(device_t dev, device_t child, int index, uintptr_t *result)
134 {
135 	const struct bcma_devinfo *dinfo;
136 	const struct bhnd_core_info *ci;
137 
138 	dinfo = device_get_ivars(child);
139 	ci = &dinfo->corecfg->core_info;
140 
141 	switch (index) {
142 	case BHND_IVAR_VENDOR:
143 		*result = ci->vendor;
144 		return (0);
145 	case BHND_IVAR_DEVICE:
146 		*result = ci->device;
147 		return (0);
148 	case BHND_IVAR_HWREV:
149 		*result = ci->hwrev;
150 		return (0);
151 	case BHND_IVAR_DEVICE_CLASS:
152 		*result = bhnd_core_class(ci);
153 		return (0);
154 	case BHND_IVAR_VENDOR_NAME:
155 		*result = (uintptr_t) bhnd_vendor_name(ci->vendor);
156 		return (0);
157 	case BHND_IVAR_DEVICE_NAME:
158 		*result = (uintptr_t) bhnd_core_name(ci);
159 		return (0);
160 	case BHND_IVAR_CORE_INDEX:
161 		*result = ci->core_idx;
162 		return (0);
163 	case BHND_IVAR_CORE_UNIT:
164 		*result = ci->unit;
165 		return (0);
166 	case BHND_IVAR_PMU_INFO:
167 		*result = (uintptr_t) dinfo->pmu_info;
168 		return (0);
169 	default:
170 		return (ENOENT);
171 	}
172 }
173 
174 static int
175 bcma_write_ivar(device_t dev, device_t child, int index, uintptr_t value)
176 {
177 	struct bcma_devinfo *dinfo;
178 
179 	dinfo = device_get_ivars(child);
180 
181 	switch (index) {
182 	case BHND_IVAR_VENDOR:
183 	case BHND_IVAR_DEVICE:
184 	case BHND_IVAR_HWREV:
185 	case BHND_IVAR_DEVICE_CLASS:
186 	case BHND_IVAR_VENDOR_NAME:
187 	case BHND_IVAR_DEVICE_NAME:
188 	case BHND_IVAR_CORE_INDEX:
189 	case BHND_IVAR_CORE_UNIT:
190 		return (EINVAL);
191 	case BHND_IVAR_PMU_INFO:
192 		dinfo->pmu_info = (struct bhnd_core_pmu_info *) value;
193 		return (0);
194 	default:
195 		return (ENOENT);
196 	}
197 }
198 
199 static struct resource_list *
200 bcma_get_resource_list(device_t dev, device_t child)
201 {
202 	struct bcma_devinfo *dinfo = device_get_ivars(child);
203 	return (&dinfo->resources);
204 }
205 
206 static int
207 bcma_read_iost(device_t dev, device_t child, uint16_t *iost)
208 {
209 	uint32_t	value;
210 	int		error;
211 
212 	if ((error = bhnd_read_config(child, BCMA_DMP_IOSTATUS, &value, 4)))
213 		return (error);
214 
215 	/* Return only the bottom 16 bits */
216 	*iost = (value & BCMA_DMP_IOST_MASK);
217 	return (0);
218 }
219 
220 static int
221 bcma_read_ioctl(device_t dev, device_t child, uint16_t *ioctl)
222 {
223 	uint32_t	value;
224 	int		error;
225 
226 	if ((error = bhnd_read_config(child, BCMA_DMP_IOCTRL, &value, 4)))
227 		return (error);
228 
229 	/* Return only the bottom 16 bits */
230 	*ioctl = (value & BCMA_DMP_IOCTRL_MASK);
231 	return (0);
232 }
233 
234 static int
235 bcma_write_ioctl(device_t dev, device_t child, uint16_t value, uint16_t mask)
236 {
237 	struct bcma_devinfo	*dinfo;
238 	struct bhnd_resource	*r;
239 	uint32_t		 ioctl;
240 
241 	if (device_get_parent(child) != dev)
242 		return (EINVAL);
243 
244 	dinfo = device_get_ivars(child);
245 	if ((r = dinfo->res_agent) == NULL)
246 		return (ENODEV);
247 
248 	/* Write new value */
249 	ioctl = bhnd_bus_read_4(r, BCMA_DMP_IOCTRL);
250 	ioctl &= ~(BCMA_DMP_IOCTRL_MASK & mask);
251 	ioctl |= (value & mask);
252 
253 	bhnd_bus_write_4(r, BCMA_DMP_IOCTRL, ioctl);
254 
255 	/* Perform read-back and wait for completion */
256 	bhnd_bus_read_4(r, BCMA_DMP_IOCTRL);
257 	DELAY(10);
258 
259 	return (0);
260 }
261 
262 static bool
263 bcma_is_hw_suspended(device_t dev, device_t child)
264 {
265 	uint32_t	rst;
266 	uint16_t	ioctl;
267 	int		error;
268 
269 	/* Is core held in RESET? */
270 	error = bhnd_read_config(child, BCMA_DMP_RESETCTRL, &rst, 4);
271 	if (error) {
272 		device_printf(child, "error reading HW reset state: %d\n",
273 		    error);
274 		return (true);
275 	}
276 
277 	if (rst & BCMA_DMP_RC_RESET)
278 		return (true);
279 
280 	/* Is core clocked? */
281 	error = bhnd_read_ioctl(child, &ioctl);
282 	if (error) {
283 		device_printf(child, "error reading HW ioctl register: %d\n",
284 		    error);
285 		return (true);
286 	}
287 
288 	if (!(ioctl & BHND_IOCTL_CLK_EN))
289 		return (true);
290 
291 	return (false);
292 }
293 
294 static int
295 bcma_reset_hw(device_t dev, device_t child, uint16_t ioctl)
296 {
297 	struct bcma_devinfo		*dinfo;
298 	struct bhnd_core_pmu_info	*pm;
299 	struct bhnd_resource		*r;
300 	int				 error;
301 
302 	if (device_get_parent(child) != dev)
303 		return (EINVAL);
304 
305 	dinfo = device_get_ivars(child);
306 	pm = dinfo->pmu_info;
307 
308 	/* We require exclusive control over BHND_IOCTL_CLK_EN and
309 	 * BHND_IOCTL_CLK_FORCE. */
310 	if (ioctl & (BHND_IOCTL_CLK_EN | BHND_IOCTL_CLK_FORCE))
311 		return (EINVAL);
312 
313 	/* Can't suspend the core without access to the agent registers */
314 	if ((r = dinfo->res_agent) == NULL)
315 		return (ENODEV);
316 
317 	/* Place core into known RESET state */
318 	if ((error = BHND_BUS_SUSPEND_HW(dev, child)))
319 		return (error);
320 
321 	/*
322 	 * Leaving the core in reset:
323 	 * - Set the caller's IOCTL flags
324 	 * - Enable clocks
325 	 * - Force clock distribution to ensure propagation throughout the
326 	 *   core.
327 	 */
328 	error = bhnd_write_ioctl(child,
329 	    ioctl | BHND_IOCTL_CLK_EN | BHND_IOCTL_CLK_FORCE, UINT16_MAX);
330 	if (error)
331 		return (error);
332 
333 	/* Bring the core out of reset */
334 	if ((error = bcma_dmp_write_reset(child, dinfo, 0x0)))
335 		return (error);
336 
337 	/* Disable forced clock gating (leaving clock enabled) */
338 	error = bhnd_write_ioctl(child, 0x0, BHND_IOCTL_CLK_FORCE);
339 	if (error)
340 		return (error);
341 
342 	return (0);
343 }
344 
345 static int
346 bcma_suspend_hw(device_t dev, device_t child)
347 {
348 	struct bcma_devinfo		*dinfo;
349 	struct bhnd_core_pmu_info	*pm;
350 	struct bhnd_resource		*r;
351 	uint32_t			 rst;
352 	int				 error;
353 
354 	if (device_get_parent(child) != dev)
355 		return (EINVAL);
356 
357 	dinfo = device_get_ivars(child);
358 	pm = dinfo->pmu_info;
359 
360 	/* Can't suspend the core without access to the agent registers */
361 	if ((r = dinfo->res_agent) == NULL)
362 		return (ENODEV);
363 
364 	/* Wait for any pending reset operations to clear */
365 	if ((error = bcma_dmp_wait_reset(child, dinfo)))
366 		return (error);
367 
368 	/* Already in reset? */
369 	rst = bhnd_bus_read_4(r, BCMA_DMP_RESETCTRL);
370 	if (rst & BCMA_DMP_RC_RESET)
371 		return (0);
372 
373 	/* Put core into reset */
374 	if ((error = bcma_dmp_write_reset(child, dinfo, BCMA_DMP_RC_RESET)))
375 		return (error);
376 
377 	/* Clear core flags */
378 	if ((error = bhnd_write_ioctl(child, 0x0, UINT16_MAX)))
379 		return (error);
380 
381 	/* Inform PMU that all outstanding request state should be discarded */
382 	if (pm != NULL) {
383 		if ((error = BHND_PMU_CORE_RELEASE(pm->pm_pmu, pm)))
384 			return (error);
385 	}
386 
387 	return (0);
388 }
389 
390 static int
391 bcma_read_config(device_t dev, device_t child, bus_size_t offset, void *value,
392     u_int width)
393 {
394 	struct bcma_devinfo	*dinfo;
395 	struct bhnd_resource	*r;
396 
397 	/* Must be a directly attached child core */
398 	if (device_get_parent(child) != dev)
399 		return (EINVAL);
400 
401 	/* Fetch the agent registers */
402 	dinfo = device_get_ivars(child);
403 	if ((r = dinfo->res_agent) == NULL)
404 		return (ENODEV);
405 
406 	/* Verify bounds */
407 	if (offset > rman_get_size(r->res))
408 		return (EFAULT);
409 
410 	if (rman_get_size(r->res) - offset < width)
411 		return (EFAULT);
412 
413 	switch (width) {
414 	case 1:
415 		*((uint8_t *)value) = bhnd_bus_read_1(r, offset);
416 		return (0);
417 	case 2:
418 		*((uint16_t *)value) = bhnd_bus_read_2(r, offset);
419 		return (0);
420 	case 4:
421 		*((uint32_t *)value) = bhnd_bus_read_4(r, offset);
422 		return (0);
423 	default:
424 		return (EINVAL);
425 	}
426 }
427 
428 static int
429 bcma_write_config(device_t dev, device_t child, bus_size_t offset,
430     const void *value, u_int width)
431 {
432 	struct bcma_devinfo	*dinfo;
433 	struct bhnd_resource	*r;
434 
435 	/* Must be a directly attached child core */
436 	if (device_get_parent(child) != dev)
437 		return (EINVAL);
438 
439 	/* Fetch the agent registers */
440 	dinfo = device_get_ivars(child);
441 	if ((r = dinfo->res_agent) == NULL)
442 		return (ENODEV);
443 
444 	/* Verify bounds */
445 	if (offset > rman_get_size(r->res))
446 		return (EFAULT);
447 
448 	if (rman_get_size(r->res) - offset < width)
449 		return (EFAULT);
450 
451 	switch (width) {
452 	case 1:
453 		bhnd_bus_write_1(r, offset, *(const uint8_t *)value);
454 		return (0);
455 	case 2:
456 		bhnd_bus_write_2(r, offset, *(const uint16_t *)value);
457 		return (0);
458 	case 4:
459 		bhnd_bus_write_4(r, offset, *(const uint32_t *)value);
460 		return (0);
461 	default:
462 		return (EINVAL);
463 	}
464 }
465 
466 static u_int
467 bcma_get_port_count(device_t dev, device_t child, bhnd_port_type type)
468 {
469 	struct bcma_devinfo *dinfo;
470 
471 	/* delegate non-bus-attached devices to our parent */
472 	if (device_get_parent(child) != dev)
473 		return (BHND_BUS_GET_PORT_COUNT(device_get_parent(dev), child,
474 		    type));
475 
476 	dinfo = device_get_ivars(child);
477 	switch (type) {
478 	case BHND_PORT_DEVICE:
479 		return (dinfo->corecfg->num_dev_ports);
480 	case BHND_PORT_BRIDGE:
481 		return (dinfo->corecfg->num_bridge_ports);
482 	case BHND_PORT_AGENT:
483 		return (dinfo->corecfg->num_wrapper_ports);
484 	default:
485 		device_printf(dev, "%s: unknown type (%d)\n",
486 		    __func__,
487 		    type);
488 		return (0);
489 	}
490 }
491 
492 static u_int
493 bcma_get_region_count(device_t dev, device_t child, bhnd_port_type type,
494     u_int port_num)
495 {
496 	struct bcma_devinfo	*dinfo;
497 	struct bcma_sport_list	*ports;
498 	struct bcma_sport	*port;
499 
500 	/* delegate non-bus-attached devices to our parent */
501 	if (device_get_parent(child) != dev)
502 		return (BHND_BUS_GET_REGION_COUNT(device_get_parent(dev), child,
503 		    type, port_num));
504 
505 	dinfo = device_get_ivars(child);
506 	ports = bcma_corecfg_get_port_list(dinfo->corecfg, type);
507 
508 	STAILQ_FOREACH(port, ports, sp_link) {
509 		if (port->sp_num == port_num)
510 			return (port->sp_num_maps);
511 	}
512 
513 	/* not found */
514 	return (0);
515 }
516 
517 static int
518 bcma_get_port_rid(device_t dev, device_t child, bhnd_port_type port_type,
519     u_int port_num, u_int region_num)
520 {
521 	struct bcma_devinfo	*dinfo;
522 	struct bcma_map		*map;
523 	struct bcma_sport_list	*ports;
524 	struct bcma_sport	*port;
525 
526 	dinfo = device_get_ivars(child);
527 	ports = bcma_corecfg_get_port_list(dinfo->corecfg, port_type);
528 
529 	STAILQ_FOREACH(port, ports, sp_link) {
530 		if (port->sp_num != port_num)
531 			continue;
532 
533 		STAILQ_FOREACH(map, &port->sp_maps, m_link)
534 			if (map->m_region_num == region_num)
535 				return map->m_rid;
536 	}
537 
538 	return -1;
539 }
540 
541 static int
542 bcma_decode_port_rid(device_t dev, device_t child, int type, int rid,
543     bhnd_port_type *port_type, u_int *port_num, u_int *region_num)
544 {
545 	struct bcma_devinfo	*dinfo;
546 	struct bcma_map		*map;
547 	struct bcma_sport_list	*ports;
548 	struct bcma_sport	*port;
549 
550 	dinfo = device_get_ivars(child);
551 
552 	/* Ports are always memory mapped */
553 	if (type != SYS_RES_MEMORY)
554 		return (EINVAL);
555 
556 	/* Starting with the most likely device list, search all three port
557 	 * lists */
558 	bhnd_port_type types[] = {
559 	    BHND_PORT_DEVICE,
560 	    BHND_PORT_AGENT,
561 	    BHND_PORT_BRIDGE
562 	};
563 
564 	for (int i = 0; i < nitems(types); i++) {
565 		ports = bcma_corecfg_get_port_list(dinfo->corecfg, types[i]);
566 
567 		STAILQ_FOREACH(port, ports, sp_link) {
568 			STAILQ_FOREACH(map, &port->sp_maps, m_link) {
569 				if (map->m_rid != rid)
570 					continue;
571 
572 				*port_type = port->sp_type;
573 				*port_num = port->sp_num;
574 				*region_num = map->m_region_num;
575 				return (0);
576 			}
577 		}
578 	}
579 
580 	return (ENOENT);
581 }
582 
583 static int
584 bcma_get_region_addr(device_t dev, device_t child, bhnd_port_type port_type,
585     u_int port_num, u_int region_num, bhnd_addr_t *addr, bhnd_size_t *size)
586 {
587 	struct bcma_devinfo	*dinfo;
588 	struct bcma_map		*map;
589 	struct bcma_sport_list	*ports;
590 	struct bcma_sport	*port;
591 
592 	dinfo = device_get_ivars(child);
593 	ports = bcma_corecfg_get_port_list(dinfo->corecfg, port_type);
594 
595 	/* Search the port list */
596 	STAILQ_FOREACH(port, ports, sp_link) {
597 		if (port->sp_num != port_num)
598 			continue;
599 
600 		STAILQ_FOREACH(map, &port->sp_maps, m_link) {
601 			if (map->m_region_num != region_num)
602 				continue;
603 
604 			/* Found! */
605 			*addr = map->m_base;
606 			*size = map->m_size;
607 			return (0);
608 		}
609 	}
610 
611 	return (ENOENT);
612 }
613 
614 /**
615  * Default bcma(4) bus driver implementation of BHND_BUS_GET_INTR_COUNT().
616  *
617  * This implementation consults @p child's agent register block,
618  * returning the number of interrupt output lines routed to @p child.
619  */
620 int
621 bcma_get_intr_count(device_t dev, device_t child)
622 {
623 	struct bcma_devinfo	*dinfo;
624 	uint32_t		 dmpcfg, oobw;
625 
626 	dinfo = device_get_ivars(child);
627 
628 	/* Agent block must be mapped */
629 	if (dinfo->res_agent == NULL)
630 		return (0);
631 
632 	/* Agent must support OOB */
633 	dmpcfg = bhnd_bus_read_4(dinfo->res_agent, BCMA_DMP_CONFIG);
634 	if (!BCMA_DMP_GET_FLAG(dmpcfg, BCMA_DMP_CFG_OOB))
635 		return (0);
636 
637 	/* Return OOB width as interrupt count */
638 	oobw = bhnd_bus_read_4(dinfo->res_agent,
639 	    BCMA_DMP_OOB_OUTWIDTH(BCMA_OOB_BANK_INTR));
640 	if (oobw > BCMA_OOB_NUM_SEL) {
641 		device_printf(dev, "ignoring invalid OOBOUTWIDTH for core %u: "
642 		    "%#x\n", BCMA_DINFO_COREIDX(dinfo), oobw);
643 		return (0);
644 	}
645 
646 	return (oobw);
647 }
648 
649 /**
650  * Default bcma(4) bus driver implementation of BHND_BUS_GET_CORE_IVEC().
651  *
652  * This implementation consults @p child's agent register block,
653  * returning the interrupt output line routed to @p child, at OOB selector
654  * @p intr.
655  */
656 int
657 bcma_get_core_ivec(device_t dev, device_t child, u_int intr, uint32_t *ivec)
658 {
659 	struct bcma_devinfo	*dinfo;
660 	uint32_t		 oobsel;
661 
662 	dinfo = device_get_ivars(child);
663 
664 	/* Interrupt ID must be valid. */
665 	if (intr >= bcma_get_intr_count(dev, child))
666 		return (ENXIO);
667 
668 	/* Fetch OOBSEL busline value */
669 	KASSERT(dinfo->res_agent != NULL, ("missing agent registers"));
670 	oobsel = bhnd_bus_read_4(dinfo->res_agent, BCMA_DMP_OOBSELOUT(
671 	    BCMA_OOB_BANK_INTR, intr));
672 	*ivec = (oobsel >> BCMA_DMP_OOBSEL_SHIFT(intr)) &
673 	    BCMA_DMP_OOBSEL_BUSLINE_MASK;
674 
675 	return (0);
676 }
677 
678 /**
679  * Scan the device enumeration ROM table, adding all valid discovered cores to
680  * the bus.
681  *
682  * @param bus The bcma bus.
683  */
684 int
685 bcma_add_children(device_t bus)
686 {
687 	bhnd_erom_t			*erom;
688 	struct bcma_erom		*bcma_erom;
689 	struct bhnd_erom_io		*eio;
690 	const struct bhnd_chipid	*cid;
691 	struct bcma_corecfg		*corecfg;
692 	struct bcma_devinfo		*dinfo;
693 	device_t			 child;
694 	int				 error;
695 
696 	cid = BHND_BUS_GET_CHIPID(bus, bus);
697 	corecfg = NULL;
698 
699 	/* Allocate our EROM parser */
700 	eio = bhnd_erom_iores_new(bus, BCMA_EROM_RID);
701 	erom = bhnd_erom_alloc(&bcma_erom_parser, cid, eio);
702 	if (erom == NULL) {
703 		bhnd_erom_io_fini(eio);
704 		return (ENODEV);
705 	}
706 
707 	/* Add all cores. */
708 	bcma_erom = (struct bcma_erom *)erom;
709 	while ((error = bcma_erom_next_corecfg(bcma_erom, &corecfg)) == 0) {
710 		int nintr;
711 
712 		/* Add the child device */
713 		child = BUS_ADD_CHILD(bus, 0, NULL, -1);
714 		if (child == NULL) {
715 			error = ENXIO;
716 			goto cleanup;
717 		}
718 
719 		/* Initialize device ivars */
720 		dinfo = device_get_ivars(child);
721 		if ((error = bcma_init_dinfo(bus, dinfo, corecfg)))
722 			goto cleanup;
723 
724 		/* The dinfo instance now owns the corecfg value */
725 		corecfg = NULL;
726 
727 		/* Allocate device's agent registers, if any */
728 		if ((error = bcma_dinfo_alloc_agent(bus, child, dinfo)))
729 			goto cleanup;
730 
731 		/* Assign interrupts */
732 		nintr = bhnd_get_intr_count(child);
733 		for (int rid = 0; rid < nintr; rid++) {
734 			error = BHND_BUS_ASSIGN_INTR(bus, child, rid);
735 			if (error) {
736 				device_printf(bus, "failed to assign interrupt "
737 				    "%d to core %u: %d\n", rid,
738 				    BCMA_DINFO_COREIDX(dinfo), error);
739 			}
740 		}
741 
742 		/* If pins are floating or the hardware is otherwise
743 		 * unpopulated, the device shouldn't be used. */
744 		if (bhnd_is_hw_disabled(child))
745 			device_disable(child);
746 
747 		/* Issue bus callback for fully initialized child. */
748 		BHND_BUS_CHILD_ADDED(bus, child);
749 	}
750 
751 	/* EOF while parsing cores is expected */
752 	if (error == ENOENT)
753 		error = 0;
754 
755 cleanup:
756 	bhnd_erom_free(erom);
757 
758 	if (corecfg != NULL)
759 		bcma_free_corecfg(corecfg);
760 
761 	if (error)
762 		device_delete_children(bus);
763 
764 	return (error);
765 }
766 
767 
768 static device_method_t bcma_methods[] = {
769 	/* Device interface */
770 	DEVMETHOD(device_probe,			bcma_probe),
771 	DEVMETHOD(device_attach,		bcma_attach),
772 	DEVMETHOD(device_detach,		bcma_detach),
773 
774 	/* Bus interface */
775 	DEVMETHOD(bus_add_child,		bcma_add_child),
776 	DEVMETHOD(bus_child_deleted,		bcma_child_deleted),
777 	DEVMETHOD(bus_read_ivar,		bcma_read_ivar),
778 	DEVMETHOD(bus_write_ivar,		bcma_write_ivar),
779 	DEVMETHOD(bus_get_resource_list,	bcma_get_resource_list),
780 
781 	/* BHND interface */
782 	DEVMETHOD(bhnd_bus_get_erom_class,	bcma_get_erom_class),
783 	DEVMETHOD(bhnd_bus_read_ioctl,		bcma_read_ioctl),
784 	DEVMETHOD(bhnd_bus_write_ioctl,		bcma_write_ioctl),
785 	DEVMETHOD(bhnd_bus_read_iost,		bcma_read_iost),
786 	DEVMETHOD(bhnd_bus_is_hw_suspended,	bcma_is_hw_suspended),
787 	DEVMETHOD(bhnd_bus_reset_hw,		bcma_reset_hw),
788 	DEVMETHOD(bhnd_bus_suspend_hw,		bcma_suspend_hw),
789 	DEVMETHOD(bhnd_bus_read_config,		bcma_read_config),
790 	DEVMETHOD(bhnd_bus_write_config,	bcma_write_config),
791 	DEVMETHOD(bhnd_bus_get_port_count,	bcma_get_port_count),
792 	DEVMETHOD(bhnd_bus_get_region_count,	bcma_get_region_count),
793 	DEVMETHOD(bhnd_bus_get_port_rid,	bcma_get_port_rid),
794 	DEVMETHOD(bhnd_bus_decode_port_rid,	bcma_decode_port_rid),
795 	DEVMETHOD(bhnd_bus_get_region_addr,	bcma_get_region_addr),
796 	DEVMETHOD(bhnd_bus_get_intr_count,	bcma_get_intr_count),
797 	DEVMETHOD(bhnd_bus_get_core_ivec,	bcma_get_core_ivec),
798 
799 	DEVMETHOD_END
800 };
801 
802 DEFINE_CLASS_1(bhnd, bcma_driver, bcma_methods, sizeof(struct bcma_softc), bhnd_driver);
803 MODULE_VERSION(bcma, 1);
804 MODULE_DEPEND(bcma, bhnd, 1, 1, 1);
805