Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/atari/atari Replace bus_dmamap_sync(9) op with a ne...



details:   https://anonhg.NetBSD.org/src/rev/41db00dfcd77
branches:  trunk
changeset: 360417:41db00dfcd77
user:      tsutsui <tsutsui%NetBSD.org@localhost>
date:      Sat Mar 10 03:44:43 2018 +0000

description:
Replace bus_dmamap_sync(9) op with a newer one taken from m68k/bus_dma.c.

This could fix memory corruption issue caused by PREREAD ops with regions
whose boundaries are not aligned at cacheline size.

diffstat:

 sys/arch/atari/atari/bus.c |  209 +++++++++++++++++++++++++++++++++++++++-----
 1 files changed, 183 insertions(+), 26 deletions(-)

diffs (246 lines):

diff -r 9cc55b52d796 -r 41db00dfcd77 sys/arch/atari/atari/bus.c
--- a/sys/arch/atari/atari/bus.c        Sat Mar 10 02:48:51 2018 +0000
+++ b/sys/arch/atari/atari/bus.c        Sat Mar 10 03:44:43 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: bus.c,v 1.59 2018/01/20 17:37:15 tsutsui Exp $ */
+/*     $NetBSD: bus.c,v 1.60 2018/03/10 03:44:43 tsutsui Exp $ */
 
 /*-
  * Copyright (c) 1998 The NetBSD Foundation, Inc.
@@ -33,7 +33,7 @@
 #include "opt_m68k_arch.h"
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: bus.c,v 1.59 2018/01/20 17:37:15 tsutsui Exp $");
+__KERNEL_RCSID(0, "$NetBSD: bus.c,v 1.60 2018/03/10 03:44:43 tsutsui Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -548,46 +548,203 @@
  * by bus-specific DMA map synchronization functions.
  */
 void
-_bus_dmamap_sync(bus_dma_tag_t t, bus_dmamap_t map, bus_addr_t off,
+_bus_dmamap_sync(bus_dma_tag_t t, bus_dmamap_t map, bus_addr_t offset,
     bus_size_t len, int ops)
 {
 #if defined(M68040) || defined(M68060)
-       int     i, pa_off, inc, seglen;
-       u_long  pa, end_pa;
+       bus_addr_t p, e, ps, pe;
+       bus_size_t seglen;
+       bus_dma_segment_t *seg;
+       int i;
+#endif
 
-       pa_off = t->_displacement;
+#if defined(M68020) || defined(M68030)
+#if defined(M68040) || defined(M68060)
+       if (cputype == CPU_68020 || cputype == CPU_68030)
+#endif
+               /* assume no L2 physical cache */
+               return;
+#endif
 
-       /* Flush granularity */
-       inc = (len > 1024) ? PAGE_SIZE : 16;
+#if defined(M68040) || defined(M68060)
+       /* If the whole DMA map is uncached, do nothing. */
+       if ((map->_dm_flags & BUS_DMA_COHERENT) != 0)
+               return;
+
+       /* Short-circuit for unsupported `ops' */
+       if ((ops & (BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE)) == 0)
+               return;
 
-       for (i = 0; i < map->dm_nsegs && len > 0; i++) {
-               if (map->dm_segs[i].ds_len <= off) {
+       /*
+        * flush/purge the cache.
+        */
+       for (i = 0; i < map->dm_nsegs && len != 0; i++) {
+               seg = &map->dm_segs[i];
+               if (seg->ds_len <= offset) {
                        /* Segment irrelevant - before requested offset */
-                       off -= map->dm_segs[i].ds_len;
+                       offset -= seg->ds_len;
                        continue;
                }
-               seglen = map->dm_segs[i].ds_len - off;
+
+               /*
+                * Now at the first segment to sync; nail
+                * each segment until we have exhausted the
+                * length.
+                */
+               seglen = seg->ds_len - offset;
                if (seglen > len)
                        seglen = len;
-               len -= seglen;
-               pa = map->dm_segs[i].ds_addr + off - pa_off;
-               end_pa = pa + seglen;
+
+               ps = seg->ds_addr + offset;
+               pe = ps + seglen;
+
+               if (ops & BUS_DMASYNC_PREWRITE) {
+                       p = ps & ~CACHELINE_MASK;
+                       e = (pe + CACHELINE_MASK) & ~CACHELINE_MASK;
+
+                       /* flush cacheline */
+                       while ((p < e) && (p & (CACHELINE_SIZE * 8 - 1)) != 0) {
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                       }
 
-               if (inc == 16) {
-                       pa &= ~15;
-                       while (pa < end_pa) {
-                               DCFL(pa);
-                               pa += 16;
+                       /* flush cachelines per 128bytes */
+                       while ((p < e) && (p & PAGE_MASK) != 0) {
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* flush page */
+                       while (p + PAGE_SIZE <= e) {
+                               DCFP(p);
+                               p += PAGE_SIZE;
                        }
-               } else {
-                       pa &= ~PGOFSET;
-                       while (pa < end_pa) {
-                               DCFP(pa);
-                               pa += PAGE_SIZE;
+
+                       /* flush cachelines per 128bytes */
+                       while (p + CACHELINE_SIZE * 8 <= e) {
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* flush cacheline */
+                       while (p < e) {
+                               DCFL(p);
+                               p += CACHELINE_SIZE;
                        }
                }
+
+               /*
+                * Normally, the `PREREAD' flag instructs us to purge the
+                * cache for the specified offset and length. However, if
+                * the offset/length is not aligned to a cacheline boundary,
+                * we may end up purging some legitimate data from the
+                * start/end of the cache. In such a case, *flush* the
+                * cachelines at the start and end of the required region.
+                */
+               else if (ops & BUS_DMASYNC_PREREAD) {
+                       /* flush cacheline on start boundary */
+                       if (ps & CACHELINE_MASK) {
+                               DCFL(ps & ~CACHELINE_MASK);
+                       }
+
+                       p = (ps + CACHELINE_MASK) & ~CACHELINE_MASK;
+                       e = pe & ~CACHELINE_MASK;
+
+                       /* purge cacheline */
+                       while ((p < e) && (p & (CACHELINE_SIZE * 8 - 1)) != 0) {
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* purge cachelines per 128bytes */
+                       while ((p < e) && (p & PAGE_MASK) != 0) {
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* purge page */
+                       while (p + PAGE_SIZE <= e) {
+                               DCPP(p);
+                               p += PAGE_SIZE;
+                       }
+
+                       /* purge cachelines per 128bytes */
+                       while (p + CACHELINE_SIZE * 8 <= e) {
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* purge cacheline */
+                       while (p < e) {
+                               DCPL(p);
+                               p += CACHELINE_SIZE;
+                       }
+
+                       /* flush cacheline on end boundary */
+                       if (p < pe) {
+                               DCFL(p);
+                       }
+               }
+               offset = 0;
+               len -= seglen;
        }
-#endif
+#endif /* defined(M68040) || defined(M68060) */
 }
 
 /*



Home | Main Index | Thread Index | Old Index