Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/arm/acpi If a device node is a child of a module de...
details: https://anonhg.NetBSD.org/src/rev/b786ecc47592
branches: trunk
changeset: 967989:b786ecc47592
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Mon Dec 30 19:50:29 2019 +0000
description:
If a device node is a child of a module device, and the module device declares DMA range restrictions, use them.
diffstat:
sys/arch/arm/acpi/acpi_machdep.c | 64 +++++++++++++++++++++++++++++++++++++--
1 files changed, 60 insertions(+), 4 deletions(-)
diffs (95 lines):
diff -r 90ca6d15f535 -r b786ecc47592 sys/arch/arm/acpi/acpi_machdep.c
--- a/sys/arch/arm/acpi/acpi_machdep.c Mon Dec 30 19:49:38 2019 +0000
+++ b/sys/arch/arm/acpi/acpi_machdep.c Mon Dec 30 19:50:29 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: acpi_machdep.c,v 1.14 2019/12/29 23:47:56 jmcneill Exp $ */
+/* $NetBSD: acpi_machdep.c,v 1.15 2019/12/30 19:50:29 jmcneill Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -32,7 +32,7 @@
#include "pci.h"
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: acpi_machdep.c,v 1.14 2019/12/29 23:47:56 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: acpi_machdep.c,v 1.15 2019/12/30 19:50:29 jmcneill Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -367,18 +367,74 @@
config_found_ia(sc->sc_dev, "acpisdtbus", hdrp, NULL);
}
+static const char * const module_hid[] = {
+ "ACPI0004", /* Module device */
+ NULL
+};
+
+static bus_dma_tag_t
+arm_acpi_dma_tag_subregion(struct acpi_softc *sc, bus_dma_tag_t dmat,
+ ACPI_HANDLE handle)
+{
+ struct acpi_resources res;
+ struct acpi_mem *mem;
+ bus_dma_tag_t newtag;
+ ACPI_STATUS rv;
+ int error;
+
+ rv = acpi_resource_parse(sc->sc_dev, handle, "_DMA", &res,
+ &acpi_resource_parse_ops_quiet);
+ if (ACPI_FAILURE(rv))
+ return dmat; /* no translation required */
+
+ mem = acpi_res_mem(&res, 0);
+ if (mem == NULL)
+ goto done;
+
+ aprint_debug_dev(sc->sc_dev, "_DMA range %#lx-%#lx\n",
+ mem->ar_base, mem->ar_base + mem->ar_length - 1);
+
+ error = bus_dmatag_subregion(dmat,
+ mem->ar_base, mem->ar_base + mem->ar_length - 1,
+ &newtag, BUS_DMA_WAITOK);
+ if (error != 0) {
+ aprint_error_dev(sc->sc_dev,
+ "_DMA subregion failed: %d\n", error);
+ goto done;
+ }
+ dmat = newtag;
+
+done:
+ acpi_resource_cleanup(&res);
+
+ return dmat;
+}
+
bus_dma_tag_t
arm_acpi_dma_tag(struct acpi_softc *sc, struct acpi_devnode *ad)
{
ACPI_INTEGER cca;
+ bus_dma_tag_t dmat;
if (ACPI_FAILURE(acpi_eval_integer(ad->ad_handle, "_CCA", &cca)))
cca = 1;
if (cca)
- return &acpi_coherent_dma_tag;
+ dmat = &acpi_coherent_dma_tag;
else
- return &arm_generic_dma_tag;
+ dmat = &arm_generic_dma_tag;
+
+ /*
+ * If the parent device is a bus, it may define valid DMA ranges
+ * and translations for child nodes.
+ */
+ if (ad->ad_parent != NULL &&
+ acpi_match_hid(ad->ad_parent->ad_devinfo, module_hid)) {
+ dmat = arm_acpi_dma_tag_subregion(sc, dmat,
+ ad->ad_parent->ad_handle);
+ }
+
+ return dmat;
}
__strong_alias(acpi_get_dma_tag,arm_acpi_dma_tag);
__strong_alias(acpi_get_dma64_tag,arm_acpi_dma_tag);
Home |
Main Index |
Thread Index |
Old Index