1 /* $NetBSD: dvma.c,v 1.39 2009/11/10 17:37:15 he Exp $ */
4 * Copyright (c) 1996 The NetBSD Foundation, Inc.
7 * This code is derived from software contributed to The NetBSD Foundation
8 * by Gordon W. Ross and Jeremy Cooper.
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
33 * DVMA (Direct Virtual Memory Access - like DMA)
35 * In the Sun3 architecture, memory cycles initiated by secondary bus
36 * masters (DVMA devices) passed through the same MMU that governed CPU
37 * accesses. All DVMA devices were wired in such a way so that an offset
38 * was added to the addresses they issued, causing them to access virtual
39 * memory starting at address 0x0FF00000 - the offset. The task of
40 * enabling a DVMA device to access main memory only involved creating
41 * valid mapping in the MMU that translated these high addresses into the
42 * appropriate physical addresses.
44 * The Sun3x presents a challenge to programming DVMA because the MMU is no
45 * longer shared by both secondary bus masters and the CPU. The MC68030's
46 * built-in MMU serves only to manage virtual memory accesses initiated by
47 * the CPU. Secondary bus master bus accesses pass through a different MMU,
48 * aptly named the 'I/O Mapper'. To enable every device driver that uses
49 * DVMA to understand that these two address spaces are disconnected would
50 * require a tremendous amount of code re-writing. To avoid this, we will
51 * ensure that the I/O Mapper and the MC68030 MMU are programmed together,
52 * so that DVMA mappings are consistent in both the CPU virtual address
53 * space and secondary bus master address space - creating an environment
54 * just like the Sun3 system.
56 * The maximum address space that any DVMA device in the Sun3x architecture
57 * is capable of addressing is 24 bits wide (16 Megabytes.) We can alias
58 * all of the mappings that exist in the I/O mapper by duplicating them in
59 * a specially reserved section of the CPU's virtual address space, 16
60 * Megabytes in size. Whenever a DVMA buffer is allocated, the allocation
61 * code will enter in a mapping both in the MC68030 MMU page tables and the
64 * The address returned by the allocation routine is a virtual address that
65 * the requesting driver must use to access the buffer. It is up to the
66 * device driver to convert this virtual address into the appropriate slave
67 * address that its device should issue to access the buffer. (There will be
68 * routines that assist the driver in doing so.)
71 #include <sys/cdefs.h>
72 __KERNEL_RCSID(0, "$NetBSD: dvma.c,v 1.39 2009/11/10 17:37:15 he Exp $");
74 #include <sys/param.h>
75 #include <sys/systm.h>
76 #include <sys/device.h>
78 #include <sys/malloc.h>
79 #include <sys/extent.h>
81 #include <sys/vnode.h>
85 #include <uvm/uvm_extern.h>
87 #define _SUN68K_BUS_DMA_PRIVATE
88 #include <machine/autoconf.h>
89 #include <machine/bus.h>
90 #include <machine/cpu.h>
91 #include <machine/dvma.h>
92 #include <machine/pmap.h>
94 #include <sun3/sun3/machdep.h>
96 #include <sun3/sun3x/enable.h>
97 #include <sun3/sun3x/iommu.h>
100 * Use an extent map to manage DVMA scratch-memory pages.
101 * Note: SunOS says last three pages are reserved (PROM?)
102 * Note: need a separate map (sub-map?) for last 1MB for
103 * use by VME slave interface.
106 /* Number of slots in dvmamap. */
107 struct extent
*dvma_extent
;
114 * Create the extent map for DVMA pages.
116 dvma_extent
= extent_create("dvma", DVMA_MAP_BASE
,
117 DVMA_MAP_BASE
+ (DVMA_MAP_AVAIL
- 1), M_DEVBUF
,
118 NULL
, 0, EX_NOCOALESCE
|EX_NOWAIT
);
121 * Enable DVMA in the System Enable register.
122 * Note: This is only necessary for VME slave accesses.
123 * On-board devices are always capable of DVMA.
125 *enable_reg
|= ENA_SDVMA
;
130 * Given a DVMA address, return the physical address that
131 * would be used by some OTHER bus-master besides the CPU.
132 * (Examples: on-board ie/le, VME xy board).
135 dvma_kvtopa(void *kva
, int bustype
)
140 if ((addr
& DVMA_MAP_BASE
) != DVMA_MAP_BASE
)
141 panic("dvma_kvtopa: bad dmva addr=0x%lx", addr
);
146 mask
= DVMA_OBIO_SLAVE_MASK
;
148 default: /* VME bus device. */
149 mask
= DVMA_VME_SLAVE_MASK
;
158 * Map a range [va, va+len] of wired virtual addresses in the given map
159 * to a kernel address in DVMA space.
162 dvma_mapin(void *kmem_va
, int len
, int canwait
)
171 kva
= (vaddr_t
)kmem_va
;
174 * Addresses below VM_MIN_KERNEL_ADDRESS are not part of the kernel
175 * map and should not participate in DVMA.
177 if (kva
< VM_MIN_KERNEL_ADDRESS
)
178 panic("dvma_mapin: bad kva");
182 * Calculate the offset of the data buffer from a page boundary.
185 kva
-= off
; /* Truncate starting address to nearest page. */
186 len
= round_page(len
+ off
); /* Round the buffer length to pages. */
187 npf
= btoc(len
); /* Determine the number of pages to be mapped. */
190 * Try to allocate DVMA space of the appropriate size
191 * in which to do a transfer.
194 error
= extent_alloc(dvma_extent
, len
, PAGE_SIZE
, 0,
195 EX_FAST
| EX_NOWAIT
| (canwait
? EX_WAITSPACE
: 0), &tva
);
201 * Tva is the starting page to which the data buffer will be double
202 * mapped. Dvma_addr is the starting address of the buffer within
203 * that page and is the return value of the function.
205 dvma_addr
= (void *)(tva
+ off
);
207 for (; npf
--; kva
+= PAGE_SIZE
, tva
+= PAGE_SIZE
) {
209 * Retrieve the physical address of each page in the buffer
210 * and enter mappings into the I/O MMU so they may be seen
211 * by external bus masters and into the special DVMA space
212 * in the MC68030 MMU so they may be seen by the CPU.
214 rv
= pmap_extract(pmap_kernel(), kva
, &pa
);
217 panic("dvma_mapin: null page frame");
220 iommu_enter((tva
& IOMMU_VA_MASK
), pa
);
222 pa
| PMAP_NC
, VM_PROT_READ
| VM_PROT_WRITE
, 0);
224 pmap_update(pmap_kernel());
230 * Remove double map of `va' in DVMA space at `kva'.
232 * TODO - This function might be the perfect place to handle the
233 * synchronization between the DVMA cache and central RAM
237 dvma_mapout(void *dvma_addr
, int len
)
242 kva
= (u_long
)dvma_addr
;
243 off
= (int)kva
& PGOFSET
;
245 len
= round_page(len
+ off
);
247 iommu_remove((kva
& IOMMU_VA_MASK
), len
);
248 pmap_kremove(kva
, len
);
249 pmap_update(pmap_kernel());
252 if (extent_free(dvma_extent
, kva
, len
, EX_NOWAIT
| EX_MALLOCOK
))
253 panic("dvma_mapout: unable to free region: 0x%lx,0x%x",
259 * Allocate actual memory pages in DVMA space.
260 * (For sun3 compatibility - the ie driver.)
263 dvma_malloc(size_t bytes
)
265 void *new_mem
, *dvma_mem
;
270 new_size
= m68k_round_page(bytes
);
271 new_mem
= (void *)uvm_km_alloc(kernel_map
, new_size
, 0, UVM_KMF_WIRED
);
274 dvma_mem
= dvma_mapin(new_mem
, new_size
, 1);
279 * Free pages from dvma_malloc()
282 dvma_free(void *addr
, size_t size
)
284 vsize_t sz
= m68k_round_page(size
);
286 dvma_mapout(addr
, sz
);
287 /* XXX: need kmem address to free it...
288 Oh well, we never call this anyway. */
292 _bus_dmamap_load_raw(bus_dma_tag_t t
, bus_dmamap_t map
, bus_dma_segment_t
*segs
,
293 int nsegs
, bus_size_t size
, int flags
)
296 panic("_bus_dmamap_load_raw(): not implemented yet.");
300 _bus_dmamap_load(bus_dma_tag_t t
, bus_dmamap_t map
, void *buf
,
301 bus_size_t buflen
, struct proc
*p
, int flags
)
310 * Make sure that on error condition we return "no valid mappings".
315 if (buflen
> map
->_dm_size
)
320 sgsize
= round_page(off
+ buflen
);
322 /* Try to allocate DVMA space. */
324 error
= extent_alloc(dvma_extent
, sgsize
, PAGE_SIZE
, 0,
325 EX_FAST
| ((flags
& BUS_DMA_NOWAIT
) == 0 ? EX_WAITOK
: EX_NOWAIT
),
331 /* Fill in the segment. */
332 map
->dm_segs
[0].ds_addr
= dva
+ off
;
333 map
->dm_segs
[0].ds_len
= buflen
;
334 map
->dm_segs
[0]._ds_va
= dva
;
335 map
->dm_segs
[0]._ds_sgsize
= sgsize
;
338 * Now map the DVMA addresses we allocated to point to the
339 * pages of the caller's buffer.
342 pmap
= p
->p_vmspace
->vm_map
.pmap
;
344 pmap
= pmap_kernel();
347 rv
= pmap_extract(pmap
, kva
, &pa
);
350 panic("%s: unmapped VA", __func__
);
352 iommu_enter((dva
& IOMMU_VA_MASK
), pa
);
354 pa
| PMAP_NC
, VM_PROT_READ
| VM_PROT_WRITE
, 0);
361 map
->dm_mapsize
= map
->dm_segs
[0].ds_len
;
367 _bus_dmamap_unload(bus_dma_tag_t t
, bus_dmamap_t map
)
369 bus_dma_segment_t
*segs
;
375 if (map
->dm_nsegs
!= 1)
376 panic("%s: invalid nsegs = %d", __func__
, map
->dm_nsegs
);
380 dva
= segs
[0]._ds_va
& ~PGOFSET
;
381 sgsize
= segs
[0]._ds_sgsize
;
383 /* Unmap the DVMA addresses. */
384 iommu_remove((dva
& IOMMU_VA_MASK
), sgsize
);
385 pmap_kremove(dva
, sgsize
);
386 pmap_update(pmap_kernel());
388 /* Free the DVMA addresses. */
390 error
= extent_free(dvma_extent
, dva
, sgsize
, EX_NOWAIT
);
394 panic("%s: unable to free DVMA region", __func__
);
397 /* Mark the mappings as invalid. */