xref: /freebsd/sys/dev/bhnd/bhndb/bhndb_subr.c (revision 46c1105fbb6fbff6d6ccd0a18571342eb992d637)
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/kernel.h>
35 
36 #include "bhndb_private.h"
37 #include "bhndbvar.h"
38 
39 /**
40  * Attach a BHND bridge device to @p parent.
41  *
42  * @param parent A parent PCI device.
43  * @param[out] bhndb On success, the probed and attached bhndb bridge device.
44  * @param unit The device unit number, or -1 to select the next available unit
45  * number.
46  *
47  * @retval 0 success
48  * @retval non-zero Failed to attach the bhndb device.
49  */
50 int
51 bhndb_attach_bridge(device_t parent, device_t *bhndb, int unit)
52 {
53 	int error;
54 
55 	*bhndb = device_add_child(parent, "bhndb", unit);
56 	if (*bhndb == NULL)
57 		return (ENXIO);
58 
59 	if (!(error = device_probe_and_attach(*bhndb)))
60 		return (0);
61 
62 	if ((device_delete_child(parent, *bhndb)))
63 		device_printf(parent, "failed to detach bhndb child\n");
64 
65 	return (error);
66 }
67 
68 /*
69  * Call BHNDB_SUSPEND_RESOURCE() for all resources in @p rl.
70  */
71 static void
72 bhndb_do_suspend_resources(device_t dev, struct resource_list *rl)
73 {
74 	struct resource_list_entry *rle;
75 
76 	/* Suspend all child resources. */
77 	STAILQ_FOREACH(rle, rl, link) {
78 		/* Skip non-allocated resources */
79 		if (rle->res == NULL)
80 			continue;
81 
82 		BHNDB_SUSPEND_RESOURCE(device_get_parent(dev), dev, rle->type,
83 		    rle->res);
84 	}
85 }
86 
87 /**
88  * Helper function for implementing BUS_RESUME_CHILD() on bridged
89  * bhnd(4) buses.
90  *
91  * This implementation of BUS_RESUME_CHILD() uses BUS_GET_RESOURCE_LIST()
92  * to find the child's resources and call BHNDB_SUSPEND_RESOURCE() for all
93  * child resources, ensuring that the device's allocated bridge resources
94  * will be available to other devices during bus resumption.
95  *
96  * Before suspending any resources, @p child is suspended by
97  * calling bhnd_generic_suspend_child().
98  *
99  * If @p child is not a direct child of @p dev, suspension is delegated to
100  * the @p dev parent.
101  */
102 int
103 bhnd_generic_br_suspend_child(device_t dev, device_t child)
104 {
105 	struct resource_list		*rl;
106 	int				 error;
107 
108 	if (device_get_parent(child) != dev)
109 		BUS_SUSPEND_CHILD(device_get_parent(dev), child);
110 
111 	if (device_is_suspended(child))
112 		return (EBUSY);
113 
114 	/* Suspend the child device */
115 	if ((error = bhnd_generic_suspend_child(dev, child)))
116 		return (error);
117 
118 	/* Fetch the resource list. If none, there's nothing else to do */
119 	rl = BUS_GET_RESOURCE_LIST(device_get_parent(child), child);
120 	if (rl == NULL)
121 		return (0);
122 
123 	/* Suspend all child resources. */
124 	bhndb_do_suspend_resources(dev, rl);
125 
126 	return (0);
127 }
128 
129 /**
130  * Helper function for implementing BUS_RESUME_CHILD() on bridged
131  * bhnd(4) bus devices.
132  *
133  * This implementation of BUS_RESUME_CHILD() uses BUS_GET_RESOURCE_LIST()
134  * to find the child's resources and call BHNDB_RESUME_RESOURCE() for all
135  * child resources, before delegating to bhnd_generic_resume_child().
136  *
137  * If resource resumption fails, @p child will not be resumed.
138  *
139  * If @p child is not a direct child of @p dev, suspension is delegated to
140  * the @p dev parent.
141  */
142 int
143 bhnd_generic_br_resume_child(device_t dev, device_t child)
144 {
145 	struct resource_list		*rl;
146 	struct resource_list_entry	*rle;
147 	int				 error;
148 
149 	if (device_get_parent(child) != dev)
150 		BUS_RESUME_CHILD(device_get_parent(dev), child);
151 
152 	if (!device_is_suspended(child))
153 		return (EBUSY);
154 
155 	/* Fetch the resource list. If none, there's nothing else to do */
156 	rl = BUS_GET_RESOURCE_LIST(device_get_parent(child), child);
157 	if (rl == NULL)
158 		return (bhnd_generic_resume_child(dev, child));
159 
160 	/* Resume all resources */
161 	STAILQ_FOREACH(rle, rl, link) {
162 		/* Skip non-allocated resources */
163 		if (rle->res == NULL)
164 			continue;
165 
166 		error = BHNDB_RESUME_RESOURCE(device_get_parent(dev), dev,
167 		    rle->type, rle->res);
168 		if (error) {
169 			/* Put all resources back into a suspend state */
170 			bhndb_do_suspend_resources(dev, rl);
171 			return (error);
172 		}
173 	}
174 
175 	/* Now that all resources are resumed, resume child */
176 	if ((error = bhnd_generic_resume_child(dev, child))) {
177 		/* Put all resources back into a suspend state */
178 		bhndb_do_suspend_resources(dev, rl);
179 	}
180 
181 	return (error);
182 }
183 
184 /**
185  * Find a SYS_RES_MEMORY resource containing the given address range.
186  *
187  * @param br The bhndb resource state to search.
188  * @param start The start address of the range to search for.
189  * @param count The size of the range to search for.
190  *
191  * @retval resource the host resource containing the requested range.
192  * @retval NULL if no resource containing the requested range can be found.
193  */
194 struct resource *
195 bhndb_find_resource_range(struct bhndb_resources *br, rman_res_t start,
196      rman_res_t count)
197 {
198 	for (u_int i = 0; br->res_spec[i].type != -1; i++) {
199 		struct resource *r = br->res[i];
200 
201 		if (br->res_spec->type != SYS_RES_MEMORY)
202 			continue;
203 
204 		/* Verify range */
205 		if (rman_get_start(r) > start)
206 			continue;
207 
208 		if (rman_get_end(r) < (start + count - 1))
209 			continue;
210 
211 		return (r);
212 	}
213 
214 	return (NULL);
215 }
216 
217 /**
218  * Find the resource containing @p win.
219  *
220  * @param br The bhndb resource state to search.
221  * @param win A register window.
222  *
223  * @retval resource the resource containing @p win.
224  * @retval NULL if no resource containing @p win can be found.
225  */
226 struct resource *
227 bhndb_find_regwin_resource(struct bhndb_resources *br,
228     const struct bhndb_regwin *win)
229 {
230 	const struct resource_spec *rspecs;
231 
232 	rspecs = br->cfg->resource_specs;
233 	for (u_int i = 0; rspecs[i].type != -1; i++) {
234 		if (win->res.type != rspecs[i].type)
235 			continue;
236 
237 		if (win->res.rid != rspecs[i].rid)
238 			continue;
239 
240 		/* Found declared resource */
241 		return (br->res[i]);
242 	}
243 
244 	device_printf(br->dev,
245 	    "missing regwin resource spec (type=%d, rid=%d)\n",
246 	    win->res.type, win->res.rid);
247 
248 	return (NULL);
249 }
250 
251 /**
252  * Allocate and initialize a new resource state structure, allocating
253  * bus resources from @p parent_dev according to @p cfg.
254  *
255  * @param dev The bridge device.
256  * @param parent_dev The parent device from which resources will be allocated.
257  * @param cfg The hardware configuration to be used.
258  */
259 struct bhndb_resources *
260 bhndb_alloc_resources(device_t dev, device_t parent_dev,
261     const struct bhndb_hwcfg *cfg)
262 {
263 	struct bhndb_resources		*r;
264 	const struct bhndb_regwin	*win;
265 	bus_size_t			 last_window_size;
266 	size_t				 res_num;
267 	u_int				 rnid;
268 	int				 error;
269 	bool				 free_parent_res;
270 	bool				 free_ht_mem, free_br_mem;
271 
272 	free_parent_res = false;
273 	free_ht_mem = false;
274 	free_br_mem = false;
275 
276 	r = malloc(sizeof(*r), M_BHND, M_NOWAIT|M_ZERO);
277 	if (r == NULL)
278 		return (NULL);
279 
280 	/* Basic initialization */
281 	r->dev = dev;
282 	r->parent_dev = parent_dev;
283 	r->cfg = cfg;
284 	r->min_prio = BHNDB_PRIORITY_NONE;
285 	STAILQ_INIT(&r->bus_regions);
286 
287 	/* Initialize host address space resource manager. */
288 	r->ht_mem_rman.rm_start = 0;
289 	r->ht_mem_rman.rm_end = ~0;
290 	r->ht_mem_rman.rm_type = RMAN_ARRAY;
291 	r->ht_mem_rman.rm_descr = "BHNDB host memory";
292 	if ((error = rman_init(&r->ht_mem_rman))) {
293 		device_printf(r->dev, "could not initialize ht_mem_rman\n");
294 		goto failed;
295 	}
296 	free_ht_mem = true;
297 
298 
299 	/* Initialize resource manager for the bridged address space. */
300 	r->br_mem_rman.rm_start = 0;
301 	r->br_mem_rman.rm_end = BUS_SPACE_MAXADDR_32BIT;
302 	r->br_mem_rman.rm_type = RMAN_ARRAY;
303 	r->br_mem_rman.rm_descr = "BHNDB bridged memory";
304 
305 	if ((error = rman_init(&r->br_mem_rman))) {
306 		device_printf(r->dev, "could not initialize br_mem_rman\n");
307 		goto failed;
308 	}
309 	free_br_mem = true;
310 
311 	error = rman_manage_region(&r->br_mem_rman, 0, BUS_SPACE_MAXADDR_32BIT);
312 	if (error) {
313 		device_printf(r->dev, "could not configure br_mem_rman\n");
314 		goto failed;
315 	}
316 
317 
318 	/* Determine our bridge resource count from the hardware config. */
319 	res_num = 0;
320 	for (size_t i = 0; cfg->resource_specs[i].type != -1; i++)
321 		res_num++;
322 
323 	/* Allocate space for a non-const copy of our resource_spec
324 	 * table; this will be updated with the RIDs assigned by
325 	 * bus_alloc_resources. */
326 	r->res_spec = malloc(sizeof(r->res_spec[0]) * (res_num + 1), M_BHND,
327 	    M_NOWAIT);
328 	if (r->res_spec == NULL)
329 		goto failed;
330 
331 	/* Initialize and terminate the table */
332 	for (size_t i = 0; i < res_num; i++)
333 		r->res_spec[i] = cfg->resource_specs[i];
334 
335 	r->res_spec[res_num].type = -1;
336 
337 	/* Allocate space for our resource references */
338 	r->res = malloc(sizeof(r->res[0]) * res_num, M_BHND, M_NOWAIT);
339 	if (r->res == NULL)
340 		goto failed;
341 
342 	/* Allocate resources */
343 	error = bus_alloc_resources(r->parent_dev, r->res_spec, r->res);
344 	if (error) {
345 		device_printf(r->dev,
346 		    "could not allocate bridge resources on %s: %d\n",
347 		    device_get_nameunit(r->parent_dev), error);
348 		goto failed;
349 	} else {
350 		free_parent_res = true;
351 	}
352 
353 	/* Add allocated memory resources to our host memory resource manager */
354 	for (u_int i = 0; r->res_spec[i].type != -1; i++) {
355 		struct resource *res;
356 
357 		/* skip non-memory resources */
358 		if (r->res_spec[i].type != SYS_RES_MEMORY)
359 			continue;
360 
361 		/* add host resource to set of managed regions */
362 		res = r->res[i];
363 		error = rman_manage_region(&r->ht_mem_rman, rman_get_start(res),
364 		    rman_get_end(res));
365 		if (error) {
366 			device_printf(r->dev,
367 			    "could not register host memory region with "
368 			    "ht_mem_rman: %d\n", error);
369 			goto failed;
370 		}
371 	}
372 
373 	/* Fetch the dynamic regwin count and verify that it does not exceed
374 	 * what is representable via our freelist bitmask. */
375 	r->dwa_count = bhndb_regwin_count(cfg->register_windows,
376 	    BHNDB_REGWIN_T_DYN);
377 	if (r->dwa_count >= (8 * sizeof(r->dwa_freelist))) {
378 		device_printf(r->dev, "max dynamic regwin count exceeded\n");
379 		goto failed;
380 	}
381 
382 	/* Allocate the dynamic window allocation table. */
383 	r->dw_alloc = malloc(sizeof(r->dw_alloc[0]) * r->dwa_count, M_BHND,
384 	    M_NOWAIT);
385 	if (r->dw_alloc == NULL)
386 		goto failed;
387 
388 	/* Initialize the dynamic window table and freelist. */
389 	r->dwa_freelist = 0;
390 	rnid = 0;
391 	last_window_size = 0;
392 	for (win = cfg->register_windows;
393 	    win->win_type != BHNDB_REGWIN_T_INVALID; win++)
394 	{
395 		struct bhndb_dw_alloc *dwa;
396 
397 		/* Skip non-DYN windows */
398 		if (win->win_type != BHNDB_REGWIN_T_DYN)
399 			continue;
400 
401 		/* Validate the window size */
402 		if (win->win_size == 0) {
403 			device_printf(r->dev, "ignoring zero-length dynamic "
404 			    "register window\n");
405 			continue;
406 		} else if (last_window_size == 0) {
407 			last_window_size = win->win_size;
408 		} else if (last_window_size != win->win_size) {
409 			/*
410 			 * No existing hardware should trigger this.
411 			 *
412 			 * If you run into this in the future, the dynamic
413 			 * window allocator and the resource priority system
414 			 * will need to be extended to support multiple register
415 			 * window allocation pools.
416 			 */
417 			device_printf(r->dev, "devices that vend multiple "
418 			    "dynamic register window sizes are not currently "
419 			    "supported\n");
420 			goto failed;
421 		}
422 
423 		dwa = &r->dw_alloc[rnid];
424 		dwa->win = win;
425 		dwa->parent_res = NULL;
426 		dwa->rnid = rnid;
427 		dwa->target = 0x0;
428 
429 		LIST_INIT(&dwa->refs);
430 
431 		/* Find and validate corresponding resource. */
432 		dwa->parent_res = bhndb_find_regwin_resource(r, win);
433 		if (dwa->parent_res == NULL)
434 			goto failed;
435 
436 		if (rman_get_size(dwa->parent_res) < win->win_offset +
437 		    win->win_size)
438 		{
439 			device_printf(r->dev, "resource %d too small for "
440 			    "register window with offset %llx and size %llx\n",
441 			    rman_get_rid(dwa->parent_res),
442 			    (unsigned long long) win->win_offset,
443 			    (unsigned long long) win->win_size);
444 
445 			error = EINVAL;
446 			goto failed;
447 		}
448 
449 		/* Add to freelist */
450 		r->dwa_freelist |= (1 << rnid);
451 
452 		rnid++;
453 	}
454 
455 	return (r);
456 
457 failed:
458 	if (free_parent_res)
459 		bus_release_resources(r->parent_dev, r->res_spec, r->res);
460 
461 	if (free_ht_mem)
462 		rman_fini(&r->ht_mem_rman);
463 
464 	if (free_br_mem)
465 		rman_fini(&r->br_mem_rman);
466 
467 	if (r->res != NULL)
468 		free(r->res, M_BHND);
469 
470 	if (r->res_spec != NULL)
471 		free(r->res_spec, M_BHND);
472 
473 	if (r->dw_alloc != NULL)
474 		free(r->dw_alloc, M_BHND);
475 
476 	free (r, M_BHND);
477 
478 	return (NULL);
479 }
480 
481 /**
482  * Deallocate the given bridge resource structure and any associated resources.
483  *
484  * @param br Resource state to be deallocated.
485  */
486 void
487 bhndb_free_resources(struct bhndb_resources *br)
488 {
489 	struct bhndb_region	*region, *r_next;
490 	struct bhndb_dw_alloc	*dwa;
491 	struct bhndb_dw_rentry	*dwr, *dwr_next;
492 
493 	/* No window regions may still be held */
494 	if (__builtin_popcount(br->dwa_freelist) != br->dwa_count) {
495 		device_printf(br->dev, "leaked %llu dynamic register regions\n",
496 		    (unsigned long long) br->dwa_count - br->dwa_freelist);
497 	}
498 
499 	/* Release resources allocated through our parent. */
500 	bus_release_resources(br->parent_dev, br->res_spec, br->res);
501 
502 	/* Clean up resource reservations */
503 	for (size_t i = 0; i < br->dwa_count; i++) {
504 		dwa = &br->dw_alloc[i];
505 
506 		LIST_FOREACH_SAFE(dwr, &dwa->refs, dw_link, dwr_next) {
507 			LIST_REMOVE(dwr, dw_link);
508 			free(dwr, M_BHND);
509 		}
510 	}
511 
512 	/* Release bus regions */
513 	STAILQ_FOREACH_SAFE(region, &br->bus_regions, link, r_next) {
514 		STAILQ_REMOVE(&br->bus_regions, region, bhndb_region, link);
515 		free(region, M_BHND);
516 	}
517 
518 	/* Release our resource managers */
519 	rman_fini(&br->ht_mem_rman);
520 	rman_fini(&br->br_mem_rman);
521 
522 	/* Free backing resource state structures */
523 	free(br->res, M_BHND);
524 	free(br->res_spec, M_BHND);
525 	free(br->dw_alloc, M_BHND);
526 }
527 
528 /**
529  * Add a bus region entry to @p r for the given base @p addr and @p size.
530  *
531  * @param br The resource state to which the bus region entry will be added.
532  * @param addr The base address of this region.
533  * @param size The size of this region.
534  * @param priority The resource priority to be assigned to allocations
535  * made within this bus region.
536  * @param static_regwin If available, a static register window mapping this
537  * bus region entry. If not available, NULL.
538  *
539  * @retval 0 success
540  * @retval non-zero if adding the bus region fails.
541  */
542 int
543 bhndb_add_resource_region(struct bhndb_resources *br, bhnd_addr_t addr,
544     bhnd_size_t size, bhndb_priority_t priority,
545     const struct bhndb_regwin *static_regwin)
546 {
547 	struct bhndb_region	*reg;
548 
549 	/* Insert in the bus resource list */
550 	reg = malloc(sizeof(*reg), M_BHND, M_NOWAIT);
551 	if (reg == NULL)
552 		return (ENOMEM);
553 
554 	*reg = (struct bhndb_region) {
555 		.addr = addr,
556 		.size = size,
557 		.priority = priority,
558 		.static_regwin = static_regwin
559 	};
560 
561 	STAILQ_INSERT_HEAD(&br->bus_regions, reg, link);
562 
563 	return (0);
564 }
565 
566 
567 /**
568  * Find the maximum start and end limits of the register window mapping
569  * resource @p r.
570  *
571  * If the memory range is not mapped by an existing dynamic or static register
572  * window, ENOENT will be returned.
573  *
574  * @param br The resource state to search.
575  * @param r The resource to search for in @p br.
576  * @param addr The requested starting address.
577  * @param size The requested size.
578  *
579  * @retval bhndb_region A region that fully contains the requested range.
580  * @retval NULL If no mapping region can be found.
581  */
582 int
583 bhndb_find_resource_limits(struct bhndb_resources *br, struct resource *r,
584     rman_res_t *start, rman_res_t *end)
585 {
586 	struct bhndb_dw_alloc	*dynamic;
587 	struct bhndb_region	*sregion;
588 
589 	/* Check for an enclosing dynamic register window */
590 	if ((dynamic = bhndb_dw_find_resource(br, r))) {
591 		*start = dynamic->target;
592 		*end = dynamic->target + dynamic->win->win_size - 1;
593 		return (0);
594 	}
595 
596 	/* Check for a static region */
597 	sregion = bhndb_find_resource_region(br, rman_get_start(r),
598 	    rman_get_size(r));
599 	if (sregion != NULL && sregion->static_regwin != NULL) {
600 		*start = sregion->addr;
601 		*end = sregion->addr + sregion->size - 1;
602 
603 		return (0);
604 	}
605 
606 	/* Not found */
607 	return (ENOENT);
608 }
609 
610 /**
611  * Find the bus region that maps @p size bytes at @p addr.
612  *
613  * @param br The resource state to search.
614  * @param addr The requested starting address.
615  * @param size The requested size.
616  *
617  * @retval bhndb_region A region that fully contains the requested range.
618  * @retval NULL If no mapping region can be found.
619  */
620 struct bhndb_region *
621 bhndb_find_resource_region(struct bhndb_resources *br, bhnd_addr_t addr,
622     bhnd_size_t size)
623 {
624 	struct bhndb_region *region;
625 
626 	STAILQ_FOREACH(region, &br->bus_regions, link) {
627 		/* Request must fit within the region's mapping  */
628 		if (addr < region->addr)
629 			continue;
630 
631 		if (addr + size > region->addr + region->size)
632 			continue;
633 
634 		return (region);
635 	}
636 
637 	/* Not found */
638 	return (NULL);
639 }
640 
641 /**
642  * Find the entry matching @p r in @p dwa's references, if any.
643  *
644  * @param dwa The dynamic window allocation to search
645  * @param r The resource to search for in @p dwa.
646  */
647 static struct bhndb_dw_rentry *
648 bhndb_dw_find_resource_entry(struct bhndb_dw_alloc *dwa, struct resource *r)
649 {
650 	struct bhndb_dw_rentry	*rentry;
651 
652 	LIST_FOREACH(rentry, &dwa->refs, dw_link) {
653 		struct resource *dw_res = rentry->dw_res;
654 
655 		/* Match dev/rid/addr/size */
656 		if (rman_get_device(dw_res)	!= rman_get_device(r) ||
657 			rman_get_rid(dw_res)	!= rman_get_rid(r) ||
658 			rman_get_start(dw_res)	!= rman_get_start(r) ||
659 			rman_get_size(dw_res)	!= rman_get_size(r))
660 		{
661 			continue;
662 		}
663 
664 		/* Matching allocation found */
665 		return (rentry);
666 	}
667 
668 	return (NULL);
669 }
670 
671 /**
672  * Find the dynamic region allocated for @p r, if any.
673  *
674  * @param br The resource state to search.
675  * @param r The resource to search for.
676  *
677  * @retval bhndb_dw_alloc The allocation record for @p r.
678  * @retval NULL if no dynamic window is allocated for @p r.
679  */
680 struct bhndb_dw_alloc *
681 bhndb_dw_find_resource(struct bhndb_resources *br, struct resource *r)
682 {
683 	struct bhndb_dw_alloc	*dwa;
684 
685 	for (size_t i = 0; i < br->dwa_count; i++) {
686 		dwa = &br->dw_alloc[i];
687 
688 		/* Skip free dynamic windows */
689 		if (bhndb_dw_is_free(br, dwa))
690 			continue;
691 
692 		/* Matching allocation found? */
693 		if (bhndb_dw_find_resource_entry(dwa, r) != NULL)
694 			return (dwa);
695 	}
696 
697 	return (NULL);
698 }
699 
700 /**
701  * Find an existing dynamic window mapping @p size bytes
702  * at @p addr. The window may or may not be free.
703  *
704  * @param br The resource state to search.
705  * @param addr The requested starting address.
706  * @param size The requested size.
707  *
708  * @retval bhndb_dw_alloc A window allocation that fully contains the requested
709  * range.
710  * @retval NULL If no mapping region can be found.
711  */
712 struct bhndb_dw_alloc *
713 bhndb_dw_find_mapping(struct bhndb_resources *br, bhnd_addr_t addr,
714     bhnd_size_t size)
715 {
716 	struct bhndb_dw_alloc		*dwr;
717 	const struct bhndb_regwin	*win;
718 
719 	/* Search for an existing dynamic mapping of this address range. */
720 	for (size_t i = 0; i < br->dwa_count; i++) {
721 		dwr = &br->dw_alloc[i];
722 		win = dwr->win;
723 
724 		/* Verify the range */
725 		if (addr < dwr->target)
726 			continue;
727 
728 		if (addr + size > dwr->target + win->win_size)
729 			continue;
730 
731 		/* Found a usable mapping */
732 		return (dwr);
733 	}
734 
735 	/* not found */
736 	return (NULL);
737 }
738 
739 /**
740  * Retain a reference to @p dwa for use by @p res.
741  *
742  * @param br The resource state owning @p dwa.
743  * @param dwa The allocation record to be retained.
744  * @param res The resource that will own a reference to @p dwa.
745  *
746  * @retval 0 success
747  * @retval ENOMEM Failed to allocate a new reference structure.
748  */
749 int
750 bhndb_dw_retain(struct bhndb_resources *br, struct bhndb_dw_alloc *dwa,
751     struct resource *res)
752 {
753 	struct bhndb_dw_rentry *rentry;
754 
755 	KASSERT(bhndb_dw_find_resource_entry(dwa, res) == NULL,
756 	    ("double-retain of dynamic window for same resource"));
757 
758 	/* Insert a reference entry; we use M_NOWAIT to allow use from
759 	 * within a non-sleepable lock */
760 	rentry = malloc(sizeof(*rentry), M_BHND, M_NOWAIT);
761 	if (rentry == NULL)
762 		return (ENOMEM);
763 
764 	rentry->dw_res = res;
765 	LIST_INSERT_HEAD(&dwa->refs, rentry, dw_link);
766 
767 	/* Update the free list */
768 	br->dwa_freelist &= ~(1 << (dwa->rnid));
769 
770 	return (0);
771 }
772 
773 /**
774  * Release a reference to @p dwa previously retained by @p res. If the
775  * reference count of @p dwa reaches zero, it will be added to the
776  * free list.
777  *
778  * @param br The resource state owning @p dwa.
779  * @param dwa The allocation record to be released.
780  * @param res The resource that currently owns a reference to @p dwa.
781  */
782 void
783 bhndb_dw_release(struct bhndb_resources *br, struct bhndb_dw_alloc *dwa,
784     struct resource *r)
785 {
786 	struct bhndb_dw_rentry	*rentry;
787 
788 	/* Find the rentry */
789 	rentry = bhndb_dw_find_resource_entry(dwa, r);
790 	KASSERT(rentry != NULL, ("over release of resource entry"));
791 
792 	LIST_REMOVE(rentry, dw_link);
793 	free(rentry, M_BHND);
794 
795 	/* If this was the last reference, update the free list */
796 	if (LIST_EMPTY(&dwa->refs))
797 		br->dwa_freelist |= (1 << (dwa->rnid));
798 }
799 
800 /**
801  * Attempt to set (or reset) the target address of @p dwa to map @p size bytes
802  * at @p addr.
803  *
804  * This will apply any necessary window alignment and verify that
805  * the window is capable of mapping the requested range prior to modifying
806  * therecord.
807  *
808  * @param dev The device on which to issue the BHNDB_SET_WINDOW_ADDR() request.
809  * @param br The resource state owning @p dwa.
810  * @param dwa The allocation record to be configured.
811  * @param addr The address to be mapped via @p dwa.
812  * @param size The number of bytes to be mapped at @p addr.
813  *
814  * @retval 0 success
815  * @retval non-zero no usable register window available.
816  */
817 int
818 bhndb_dw_set_addr(device_t dev, struct bhndb_resources *br,
819     struct bhndb_dw_alloc *dwa, bus_addr_t addr, bus_size_t size)
820 {
821 	const struct bhndb_regwin	*rw;
822 	bus_addr_t			 offset;
823 	int				 error;
824 
825 	rw = dwa->win;
826 
827 	KASSERT(bhndb_dw_is_free(br, dwa),
828 	    ("attempting to set the target address on an in-use window"));
829 
830 	/* Page-align the target address */
831 	offset = addr % rw->win_size;
832 	dwa->target = addr - offset;
833 
834 	/* Verify that the window is large enough for the full target */
835 	if (rw->win_size - offset < size)
836 		return (ENOMEM);
837 
838 	/* Update the window target */
839 	error = BHNDB_SET_WINDOW_ADDR(dev, dwa->win, dwa->target);
840 	if (error) {
841 		dwa->target = 0x0;
842 		return (error);
843 	}
844 
845 	return (0);
846 }
847 
848 /**
849  * Return the count of @p type register windows in @p table.
850  *
851  * @param table The table to search.
852  * @param type The required window type, or BHNDB_REGWIN_T_INVALID to
853  * count all register window types.
854  */
855 size_t
856 bhndb_regwin_count(const struct bhndb_regwin *table,
857     bhndb_regwin_type_t type)
858 {
859 	const struct bhndb_regwin	*rw;
860 	size_t				 count;
861 
862 	count = 0;
863 	for (rw = table; rw->win_type != BHNDB_REGWIN_T_INVALID; rw++) {
864 		if (type == BHNDB_REGWIN_T_INVALID || rw->win_type == type)
865 			count++;
866 	}
867 
868 	return (count);
869 }
870 
871 /**
872  * Search @p table for the first window with the given @p type.
873  *
874  * @param table The table to search.
875  * @param type The required window type.
876  * @param min_size The minimum window size.
877  *
878  * @retval bhndb_regwin The first matching window.
879  * @retval NULL If no window of the requested type could be found.
880  */
881 const struct bhndb_regwin *
882 bhndb_regwin_find_type(const struct bhndb_regwin *table,
883     bhndb_regwin_type_t type, bus_size_t min_size)
884 {
885 	const struct bhndb_regwin *rw;
886 
887 	for (rw = table; rw->win_type != BHNDB_REGWIN_T_INVALID; rw++)
888 	{
889 		if (rw->win_type == type && rw->win_size >= min_size)
890 			return (rw);
891 	}
892 
893 	return (NULL);
894 }
895 
896 /**
897  * Search @p windows for the first matching core window.
898  *
899  * @param table The table to search.
900  * @param class The required core class.
901  * @param unit The required core unit, or -1.
902  * @param port_type The required port type.
903  * @param port The required port.
904  * @param region The required region.
905  *
906  * @retval bhndb_regwin The first matching window.
907  * @retval NULL If no matching window was found.
908  */
909 const struct bhndb_regwin *
910 bhndb_regwin_find_core(const struct bhndb_regwin *table, bhnd_devclass_t class,
911     int unit, bhnd_port_type port_type, u_int port, u_int region)
912 {
913 	const struct bhndb_regwin *rw;
914 
915 	for (rw = table; rw->win_type != BHNDB_REGWIN_T_INVALID; rw++)
916 	{
917 		if (rw->win_type != BHNDB_REGWIN_T_CORE)
918 			continue;
919 
920 		if (rw->d.core.class != class)
921 			continue;
922 
923 		if (unit != -1 && rw->d.core.unit != unit)
924 			continue;
925 
926 		if (rw->d.core.port_type != port_type)
927 			continue;
928 
929 		if (rw->d.core.port != port)
930 			continue;
931 
932 		if (rw->d.core.region != region)
933 			continue;
934 
935 		return (rw);
936 	}
937 
938 	return (NULL);
939 }
940 
941 /**
942  * Search @p windows for the best available window of at least @p min_size.
943  *
944  * Search order:
945  * - BHND_REGWIN_T_CORE
946  * - BHND_REGWIN_T_DYN
947  *
948  * @param table The table to search.
949  * @param class The required core class.
950  * @param unit The required core unit, or -1.
951  * @param port_type The required port type.
952  * @param port The required port.
953  * @param region The required region.
954  * @param min_size The minimum window size.
955  *
956  * @retval bhndb_regwin The first matching window.
957  * @retval NULL If no matching window was found.
958  */
959 const struct bhndb_regwin *
960 bhndb_regwin_find_best(const struct bhndb_regwin *table,
961     bhnd_devclass_t class, int unit, bhnd_port_type port_type, u_int port,
962     u_int region, bus_size_t min_size)
963 {
964 	const struct bhndb_regwin *rw;
965 
966 	/* Prefer a fixed core mapping */
967 	rw = bhndb_regwin_find_core(table, class, unit, port_type,
968 	    port, region);
969 	if (rw != NULL)
970 		return (rw);
971 
972 	/* Fall back on a generic dynamic window */
973 	return (bhndb_regwin_find_type(table, BHNDB_REGWIN_T_DYN, min_size));
974 }
975 
976 /**
977  * Return true if @p regw defines a static port register window, and
978  * the mapped port is actually defined on @p dev.
979  *
980  * @param regw A register window to match against.
981  * @param dev A bhnd(4) bus device.
982  */
983 bool
984 bhndb_regwin_matches_device(const struct bhndb_regwin *regw, device_t dev)
985 {
986 	/* Only core windows are supported */
987 	if (regw->win_type != BHNDB_REGWIN_T_CORE)
988 		return (false);
989 
990 	/* Device class must match */
991 	if (bhnd_get_class(dev) != regw->d.core.class)
992 		return (false);
993 
994 	/* Device unit must match */
995 	if (bhnd_get_core_unit(dev) != regw->d.core.unit)
996 		return (false);
997 
998 	/* The regwin port/region must be defined. */
999 	if (!bhnd_is_region_valid(dev, regw->d.core.port_type, regw->d.core.port,
1000 	    regw->d.core.region))
1001 	{
1002 		return (false);
1003 	}
1004 
1005 	/* Matches */
1006 	return (true);
1007 }
1008 
1009 /**
1010  * Search for a core resource priority descriptor in @p table that matches
1011  * @p device.
1012  *
1013  * @param table The table to search.
1014  * @param device A bhnd(4) bus device.
1015  */
1016 const struct bhndb_hw_priority *
1017 bhndb_hw_priority_find_device(const struct bhndb_hw_priority *table,
1018     device_t device)
1019 {
1020 	const struct bhndb_hw_priority *hp;
1021 
1022 	for (hp = table; hp->ports != NULL; hp++) {
1023 		if (bhnd_device_matches(device, &hp->match))
1024 			return (hp);
1025 	}
1026 
1027 	/* not found */
1028 	return (NULL);
1029 }
1030