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 Create bus_dma tags for each device node b...
details: https://anonhg.NetBSD.org/src/rev/315223e999d3
branches: trunk
changeset: 1006042:315223e999d3
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Tue Dec 31 13:54:22 2019 +0000
description:
Create bus_dma tags for each device node based on _CCA and _DMA properties
found by walking up the device node tree. These tags encode range
restrictions, address translations, and whether or not the device is
cache coherent.
diffstat:
sys/arch/arm/acpi/acpi_machdep.c | 150 ++++++++++++++++++++++++--------------
1 files changed, 93 insertions(+), 57 deletions(-)
diffs (189 lines):
diff -r b7816fff858d -r 315223e999d3 sys/arch/arm/acpi/acpi_machdep.c
--- a/sys/arch/arm/acpi/acpi_machdep.c Tue Dec 31 13:39:15 2019 +0000
+++ b/sys/arch/arm/acpi/acpi_machdep.c Tue Dec 31 13:54:22 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: acpi_machdep.c,v 1.16 2019/12/31 11:42:46 jmcneill Exp $ */
+/* $NetBSD: acpi_machdep.c,v 1.17 2019/12/31 13:54:22 jmcneill Exp $ */
/*-
* Copyright (c) 2018 The NetBSD Foundation, Inc.
@@ -32,13 +32,14 @@
#include "pci.h"
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: acpi_machdep.c,v 1.16 2019/12/31 11:42:46 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: acpi_machdep.c,v 1.17 2019/12/31 13:54:22 jmcneill Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/bus.h>
#include <sys/cpu.h>
#include <sys/device.h>
+#include <sys/kmem.h>
#include <uvm/uvm_extern.h>
@@ -372,44 +373,6 @@
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;
-}
-
static ACPI_HANDLE
arm_acpi_dma_module(struct acpi_softc *sc, struct acpi_devnode *ad)
{
@@ -432,28 +395,101 @@
return NULL;
}
+static void
+arm_acpi_dma_init_ranges(struct acpi_softc *sc, struct acpi_devnode *ad,
+ struct arm32_bus_dma_tag *dmat, uint32_t flags)
+{
+ struct acpi_resources res;
+ struct acpi_mem *mem;
+ ACPI_HANDLE module;
+ ACPI_STATUS rv;
+ int n;
+
+ module = arm_acpi_dma_module(sc, ad->ad_parent);
+ if (module == NULL) {
+default_tag:
+ /* No translation required */
+ dmat->_nranges = 1;
+ dmat->_ranges = kmem_zalloc(sizeof(*dmat->_ranges), KM_SLEEP);
+ dmat->_ranges[0].dr_sysbase = 0;
+ dmat->_ranges[0].dr_busbase = 0;
+ dmat->_ranges[0].dr_len = UINTPTR_MAX;
+ dmat->_ranges[0].dr_flags = flags;
+ return;
+ }
+
+ rv = acpi_resource_parse(sc->sc_dev, module, "_DMA", &res,
+ &acpi_resource_parse_ops_quiet);
+ if (ACPI_FAILURE(rv)) {
+ aprint_error_dev(sc->sc_dev,
+ "failed to parse _DMA on %s: %s\n",
+ acpi_name(module), AcpiFormatException(rv));
+ goto default_tag;
+ }
+ if (res.ar_nmem == 0) {
+ acpi_resource_cleanup(&res);
+ goto default_tag;
+ }
+
+ dmat->_nranges = res.ar_nmem;
+ dmat->_ranges = kmem_zalloc(sizeof(*dmat->_ranges) * res.ar_nmem,
+ KM_SLEEP);
+
+ for (n = 0; n < res.ar_nmem; n++) {
+ mem = acpi_res_mem(&res, n);
+ dmat->_ranges[n].dr_busbase = mem->ar_base;
+ dmat->_ranges[n].dr_sysbase = mem->ar_base;
+ if (mem->ar_decode == ACPI_POS_DECODE)
+ dmat->_ranges[n].dr_sysbase += mem->ar_offset;
+ else
+ dmat->_ranges[n].dr_sysbase -= mem->ar_offset;
+ dmat->_ranges[n].dr_len = mem->ar_length;
+ dmat->_ranges[n].dr_flags = flags;
+
+ aprint_debug_dev(sc->sc_dev,
+ "%s: DMA sysbase %#lx busbase %#lx len %#lx%s\n",
+ acpi_name(ad->ad_handle),
+ dmat->_ranges[n].dr_sysbase,
+ dmat->_ranges[n].dr_busbase,
+ dmat->_ranges[n].dr_len,
+ flags ? " (coherent)" : "");
+ }
+
+ acpi_resource_cleanup(&res);
+}
+
+static uint32_t
+arm_acpi_dma_flags(struct acpi_softc *sc, struct acpi_devnode *ad)
+{
+ ACPI_INTEGER cca = 1; /* default cache coherent */
+ ACPI_STATUS rv;
+
+ for (; ad != NULL; ad = ad->ad_parent) {
+ if (ad->ad_devinfo->Type != ACPI_TYPE_DEVICE)
+ continue;
+
+ rv = acpi_eval_integer(ad->ad_handle, "_CCA", &cca);
+ if (ACPI_SUCCESS(rv))
+ break;
+ }
+
+ return cca ? _BUS_DMAMAP_COHERENT : 0;
+}
+
+
bus_dma_tag_t
arm_acpi_dma_tag(struct acpi_softc *sc, struct acpi_devnode *ad)
{
- ACPI_HANDLE module;
- ACPI_INTEGER cca;
- bus_dma_tag_t dmat;
-
- if (ACPI_FAILURE(acpi_eval_integer(ad->ad_handle, "_CCA", &cca)))
- cca = 1;
+ struct arm32_bus_dma_tag *dmat;
- if (cca)
- dmat = &acpi_coherent_dma_tag;
- else
- dmat = &arm_generic_dma_tag;
+ if (ad->ad_dmat != NULL)
+ return ad->ad_dmat;
+
+ dmat = kmem_alloc(sizeof(*dmat), KM_SLEEP);
+ *dmat = arm_generic_dma_tag;
- /*
- * If a parent device is a bus, it may define valid DMA ranges
- * and translations for child nodes.
- */
- module = arm_acpi_dma_module(sc, ad);
- if (module != NULL)
- dmat = arm_acpi_dma_tag_subregion(sc, dmat, module);
+ const uint32_t flags = arm_acpi_dma_flags(sc, ad);
+ arm_acpi_dma_init_ranges(sc, ad, dmat, flags);
return dmat;
}
Home |
Main Index |
Thread Index |
Old Index