#define _ALPHA_BUS_DMA_PRIVATE
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/malloc.h>
#include <sys/proc.h>
#include <uvm/uvm_extern.h>
#include <machine/bus.h>
#include <alpha/dev/sgmapvar.h>
vaddr_t alpha_sgmap_prefetch_spill_page_va;
bus_addr_t alpha_sgmap_prefetch_spill_page_pa;
void
alpha_sgmap_init(bus_dma_tag_t t, struct alpha_sgmap *sgmap, const char *name,
bus_addr_t wbase, bus_addr_t sgvabase, bus_size_t sgvasize, size_t ptesize,
void *ptva, bus_size_t minptalign)
{
bus_dma_segment_t seg;
size_t ptsize;
int rseg;
if (sgvasize & PGOFSET) {
printf("size botch for sgmap `%s'\n", name);
goto die;
}
sgmap->aps_wbase = wbase;
sgmap->aps_sgvabase = sgvabase;
sgmap->aps_sgvasize = sgvasize;
if (ptva != NULL) {
sgmap->aps_pt = ptva;
sgmap->aps_ptpa = 0;
} else {
ptsize = (sgvasize / PAGE_SIZE) * ptesize;
if (minptalign != 0) {
if (minptalign < ptsize)
minptalign = ptsize;
} else
minptalign = ptsize;
if (bus_dmamem_alloc(t, ptsize, minptalign, 0, &seg, 1, &rseg,
BUS_DMA_NOWAIT)) {
panic("unable to allocate page table for sgmap `%s'",
name);
goto die;
}
sgmap->aps_ptpa = seg.ds_addr;
sgmap->aps_pt = (caddr_t)ALPHA_PHYS_TO_K0SEG(sgmap->aps_ptpa);
}
sgmap->aps_ex = extent_create((char *)name, sgvabase, sgvasize - 1,
M_DEVBUF, NULL, 0, EX_NOWAIT|EX_NOCOALESCE);
if (sgmap->aps_ex == NULL) {
printf("unable to create extent map for sgmap `%s'\n",
name);
goto die;
}
mtx_init(&sgmap->aps_mtx, IPL_HIGH);
if (alpha_sgmap_prefetch_spill_page_va == 0) {
if (bus_dmamem_alloc(t, PAGE_SIZE, 0, 0, &seg, 1, &rseg,
BUS_DMA_NOWAIT)) {
printf("unable to allocate spill page for sgmap `%s'\n",
name);
goto die;
}
alpha_sgmap_prefetch_spill_page_pa = seg.ds_addr;
alpha_sgmap_prefetch_spill_page_va =
ALPHA_PHYS_TO_K0SEG(alpha_sgmap_prefetch_spill_page_pa);
bzero((caddr_t)alpha_sgmap_prefetch_spill_page_va, PAGE_SIZE);
}
return;
die:
panic("alpha_sgmap_init");
}
int
alpha_sgmap_dmamap_setup(bus_dmamap_t map, int nsegments, int flags)
{
map->_dm_cookie = mallocarray(nsegments, sizeof(struct extent_region),
M_DEVBUF, (flags & BUS_DMA_NOWAIT) ? M_NOWAIT : M_WAITOK);
if (map->_dm_cookie != NULL)
map->_dm_cookiesize = nsegments * sizeof(struct extent_region);
return (map->_dm_cookie == NULL);
}
int
alpha_sgmap_dmamap_create(bus_dma_tag_t t, bus_size_t size, int nsegments,
bus_size_t maxsegsz, bus_size_t boundary, int flags, bus_dmamap_t *dmamp)
{
bus_dmamap_t map;
int error;
error = _bus_dmamap_create(t, size, nsegments, maxsegsz,
boundary, flags, dmamp);
if (error)
return (error);
map = *dmamp;
if (alpha_sgmap_dmamap_setup(map, nsegments, flags)) {
_bus_dmamap_destroy(t, map);
return (ENOMEM);
}
return (0);
}
void
alpha_sgmap_dmamap_teardown(bus_dmamap_t map)
{
free(map->_dm_cookie, M_DEVBUF, map->_dm_cookiesize);
}
void
alpha_sgmap_dmamap_destroy(bus_dma_tag_t t, bus_dmamap_t map)
{
KASSERT(map->dm_mapsize == 0);
alpha_sgmap_dmamap_teardown(map);
_bus_dmamap_destroy(t, map);
}