Index: sys/arch/arm/arm32/bus_dma.c =================================================================== RCS file: /home/netbsd/src/sys/arch/arm/arm32/bus_dma.c,v retrieving revision 1.113 diff -p -u -r1.113 bus_dma.c --- sys/arch/arm/arm32/bus_dma.c 14 Sep 2018 10:13:02 -0000 1.113 +++ sys/arch/arm/arm32/bus_dma.c 8 Jan 2019 15:10:01 -0000 @@ -1421,26 +1421,44 @@ _bus_dmamem_map(bus_dma_tag_t t, bus_dma for (pa = segs[curseg].ds_addr; pa < (segs[curseg].ds_addr + segs[curseg].ds_len); pa += PAGE_SIZE, va += PAGE_SIZE, size -= PAGE_SIZE) { - bool uncached = (flags & BUS_DMA_COHERENT); + int pmap_flags = 0; + #ifdef DEBUG_DMA printf("wiring p%lx to v%lx", pa, va); #endif /* DEBUG_DMA */ + if (size == 0) panic("_bus_dmamem_map: size botch"); - const struct arm32_dma_range * const dr = - _bus_dma_paddr_inrange(t->_ranges, t->_nranges, pa); +#if defined(ARMV6_CACHED_DMA_MEMORY) && !ARM_MMU_V6C /* - * If this dma region is coherent then there is - * no need for an uncached mapping. + * Force cached mapping for ARMv6+, which allows us + * unaligned access to DMA buffer. For ARMv6 MMU in + * backward compatible mode, unaligned access is + * permitted for non-cached memory. */ - if (dr != NULL - && (dr->dr_flags & _BUS_DMAMAP_COHERENT)) { - uncached = false; + if (!CPU_IS_ARMV6_P() && !CPU_IS_ARMV7_P()) +#endif + { + bool uncached = (flags & BUS_DMA_COHERENT); + + const struct arm32_dma_range * const dr = + _bus_dma_paddr_inrange(t->_ranges, + t->_nranges, pa); + /* + * If this DMA region is coherent then there is + * no need for an uncached mapping. + */ + if (dr != NULL + && (dr->dr_flags & _BUS_DMAMAP_COHERENT)) + uncached = false; + + if (uncached) + pmap_flags = PMAP_NOCACHE; } pmap_kenter_pa(va, pa, VM_PROT_READ | VM_PROT_WRITE, - PMAP_WIRED | (uncached ? PMAP_NOCACHE : 0)); + PMAP_WIRED | pmap_flags); } } pmap_update(pmap_kernel()); Index: sys/arch/arm/conf/Makefile.arm =================================================================== RCS file: /home/netbsd/src/sys/arch/arm/conf/Makefile.arm,v retrieving revision 1.49 diff -p -u -r1.49 Makefile.arm --- sys/arch/arm/conf/Makefile.arm 22 Sep 2018 12:24:01 -0000 1.49 +++ sys/arch/arm/conf/Makefile.arm 8 Jan 2019 13:22:58 -0000 @@ -53,6 +53,20 @@ CPPFLAGS.cpufunc_asm_armv6.S+= -mcpu=arm CPPFLAGS.cpufunc_asm_arm11.S+= -mcpu=arm1136j-s CPPFLAGS.cpufunc_asm_xscale.S+= -mcpu=xscale +.if !empty(MACHINE_ARCH:Mearmv6*) || !empty(MACHINE_ARCH:Mearmv7*) +. if ARMV6_CACHED_DMA_MEMORY +# Use cached memory for DMA on ARMv6+, which allows us unaligned +# access to DMA buffer. +CPPFLAGS+= -DARMV6_CACHED_DMA_MEMORY +. elif ARM11_COMPAT_MMU +# For ARMv6 MMU in backward compatible mode, unaligned access is +# permitted for non-cached memory. +. else +# Otherwise, we need strictly-aligned access to DMA buffer. +CFLAGS+= -mno-unaligned-access +. endif +.endif + ## ## (3) libkern and compat ## Index: sys/arch/arm/conf/std.arm =================================================================== RCS file: /home/netbsd/src/sys/arch/arm/conf/std.arm,v retrieving revision 1.5 diff -p -u -r1.5 std.arm --- sys/arch/arm/conf/std.arm 18 Dec 2017 15:53:38 -0000 1.5 +++ sys/arch/arm/conf/std.arm 8 Jan 2019 09:17:58 -0000 @@ -3,3 +3,8 @@ # standard NetBSD/arm options options CPU_IN_CKSUM + +# Use cached memory for bus_dma(9) on ARMv6+, which allows us unaligned +# access to DMA buffer. +# XXX This breaks drivers that do not invalidate cache appropriately. +#makeoptions ARMV6_CACHED_DMA_MEMORY=1 Index: sys/arch/evbarm/conf/RPI =================================================================== RCS file: /home/netbsd/src/sys/arch/evbarm/conf/RPI,v retrieving revision 1.85 diff -p -u -r1.85 RPI --- sys/arch/evbarm/conf/RPI 18 Oct 2018 09:01:54 -0000 1.85 +++ sys/arch/evbarm/conf/RPI 8 Jan 2019 13:14:35 -0000 @@ -24,6 +24,7 @@ options CPU_ARM1176 options SOC_BCM2835 options TPIDRPRW_IS_CURLWP options ARM11_COMPAT_MMU +makeoptions ARM11_COMPAT_MMU=1 options __HAVE_MM_MD_CACHE_ALIASING makeoptions CPUFLAGS="-march=armv6z -mtune=arm1176jzf-s -mfpu=vfp" Index: sys/arch/evbarm/conf/RPI2 =================================================================== RCS file: /home/netbsd/src/sys/arch/evbarm/conf/RPI2,v retrieving revision 1.7 diff -p -u -r1.7 RPI2 --- sys/arch/evbarm/conf/RPI2 18 Oct 2018 09:01:54 -0000 1.7 +++ sys/arch/evbarm/conf/RPI2 8 Jan 2019 13:14:55 -0000 @@ -10,6 +10,7 @@ no options CPU_ARM1176 no options SOC_BCM2835 no options TPIDRPRW_IS_CURLWP no options ARM11_COMPAT_MMU +no makeoptions ARM11_COMPAT_MMU no options __HAVE_MM_MD_CACHE_ALIASING no makeoptions CPUFLAGS no makeoptions DTS