NetBSD/sys/arch/x68k/dev/vs.c
jmcneill 8a962f23f2 Merge jmcneill-audiomp3 branch, which is derived from ad-audiomp2. From
the original ad-audiomp branch notes:

  Add MP locking to the audio drivers.

  Making the audio drivers MP safe is necessary before efforts
  can be made to make the VM system MP safe.

  The are two locks per device instance, an ISR lock and
  a character device lock. The ISR lock replaces calls to
  splaudio()/splx(), and will be held across calls to device
  methods which were called at splaudio() before (e.g.
  trigger_output). The character device lock is held across
  calls to nearly all of the methods, excluding some only
  used for initialization, e.g. get_locks.

Welcome to 5.99.57.
2011-11-23 23:07:28 +00:00

816 lines
19 KiB
C

/* $NetBSD: vs.c,v 1.35 2011/11/23 23:07:30 jmcneill Exp $ */
/*
* Copyright (c) 2001 Tetsuya Isaki. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
/*
* VS - OKI MSM6258 ADPCM voice synthesizer device driver.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: vs.c,v 1.35 2011/11/23 23:07:30 jmcneill Exp $");
#include "audio.h"
#include "vs.h"
#if NAUDIO > 0 && NVS > 0
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/kmem.h>
#include <sys/audioio.h>
#include <dev/audio_if.h>
#include <dev/mulaw.h>
#include <machine/bus.h>
#include <machine/cpu.h>
#include <dev/ic/msm6258var.h>
#include <arch/x68k/dev/dmacvar.h>
#include <arch/x68k/dev/intiovar.h>
#include <arch/x68k/dev/opmvar.h>
#include <arch/x68k/dev/vsvar.h>
#ifdef VS_DEBUG
#define DPRINTF(y,x) if (vs_debug >= (y)) printf x
static int vs_debug;
#ifdef AUDIO_DEBUG
extern int audiodebug;
#endif
#else
#define DPRINTF(y,x)
#endif
static int vs_match(device_t, cfdata_t, void *);
static void vs_attach(device_t, device_t, void *);
static int vs_dmaintr(void *);
static int vs_dmaerrintr(void *);
/* MI audio layer interface */
static int vs_open(void *, int);
static void vs_close(void *);
static int vs_query_encoding(void *, struct audio_encoding *);
static int vs_set_params(void *, int, int, audio_params_t *,
audio_params_t *, stream_filter_list_t *, stream_filter_list_t *);
static int vs_trigger_output(void *, void *, void *, int,
void (*)(void *), void *, const audio_params_t *);
static int vs_trigger_input(void *, void *, void *, int,
void (*)(void *), void *, const audio_params_t *);
static int vs_halt_output(void *);
static int vs_halt_input(void *);
static int vs_allocmem(struct vs_softc *, size_t, size_t, size_t,
struct vs_dma *);
static void vs_freemem(struct vs_dma *);
static int vs_getdev(void *, struct audio_device *);
static int vs_set_port(void *, mixer_ctrl_t *);
static int vs_get_port(void *, mixer_ctrl_t *);
static int vs_query_devinfo(void *, mixer_devinfo_t *);
static void *vs_allocm(void *, int, size_t);
static void vs_freem(void *, void *, size_t);
static size_t vs_round_buffersize(void *, int, size_t);
static int vs_get_props(void *);
static void vs_get_locks(void *, kmutex_t **, kmutex_t **);
/* lower functions */
static int vs_round_sr(u_long);
static void vs_set_sr(struct vs_softc *, int);
static inline void vs_set_po(struct vs_softc *, u_long);
extern struct cfdriver vs_cd;
CFATTACH_DECL_NEW(vs, sizeof(struct vs_softc),
vs_match, vs_attach, NULL, NULL);
static int vs_attached;
static const struct audio_hw_if vs_hw_if = {
vs_open,
vs_close,
NULL, /* drain */
vs_query_encoding,
vs_set_params,
NULL, /* round_blocksize */
NULL, /* commit_settings */
NULL, /* init_output */
NULL, /* init_input */
NULL, /* start_output */
NULL, /* start_input */
vs_halt_output,
vs_halt_input,
NULL, /* speaker_ctl */
vs_getdev,
NULL, /* setfd */
vs_set_port,
vs_get_port,
vs_query_devinfo,
vs_allocm,
vs_freem,
vs_round_buffersize,
NULL, /* mappage */
vs_get_props,
vs_trigger_output,
vs_trigger_input,
NULL,
vs_get_locks,
};
static struct audio_device vs_device = {
"OKI MSM6258",
"",
"vs"
};
struct {
u_long rate;
u_char clk;
u_char den;
} vs_l2r[] = {
{ VS_RATE_15K, VS_CLK_8MHZ, VS_SRATE_512 },
{ VS_RATE_10K, VS_CLK_8MHZ, VS_SRATE_768 },
{ VS_RATE_7K, VS_CLK_8MHZ, VS_SRATE_1024},
{ VS_RATE_5K, VS_CLK_4MHZ, VS_SRATE_768 },
{ VS_RATE_3K, VS_CLK_4MHZ, VS_SRATE_1024}
};
#define NUM_RATE (sizeof(vs_l2r)/sizeof(vs_l2r[0]))
struct {
const char *name;
int encoding;
int precision;
} vs_encodings[] = {
{AudioEadpcm, AUDIO_ENCODING_ADPCM, 4},
{AudioEslinear, AUDIO_ENCODING_SLINEAR, 8},
{AudioEulinear, AUDIO_ENCODING_ULINEAR, 8},
{AudioEmulaw, AUDIO_ENCODING_ULAW, 8},
{AudioEslinear_be, AUDIO_ENCODING_SLINEAR_BE, 16},
{AudioEslinear_le, AUDIO_ENCODING_SLINEAR_LE, 16},
};
static int
vs_match(device_t parent, cfdata_t cf, void *aux)
{
struct intio_attach_args *ia;
ia = aux;
if (strcmp(ia->ia_name, "vs") || vs_attached)
return 0;
if (ia->ia_addr == INTIOCF_ADDR_DEFAULT)
ia->ia_addr = VS_ADDR;
if (ia->ia_dma == INTIOCF_DMA_DEFAULT)
ia->ia_dma = VS_DMA;
if (ia->ia_dmaintr == INTIOCF_DMAINTR_DEFAULT)
ia->ia_dmaintr = VS_DMAINTR;
/* fixed parameters */
if (ia->ia_addr != VS_ADDR)
return 0;
if (ia->ia_dma != VS_DMA)
return 0;
if (ia->ia_dmaintr != VS_DMAINTR)
return 0;
#ifdef VS_DEBUG
vs_debug = 1;
#ifdef AUDIO_DEBUG
audiodebug = 2;
#endif
#endif
return 1;
}
static void
vs_attach(device_t parent, device_t self, void *aux)
{
struct vs_softc *sc;
bus_space_tag_t iot;
bus_space_handle_t ioh;
struct intio_attach_args *ia;
sc = device_private(self);
sc->sc_dev = self;
ia = aux;
vs_attached = 1;
printf("\n");
/* Re-map the I/O space */
iot = ia->ia_bst;
bus_space_map(iot, ia->ia_addr, 0x2000, BUS_SPACE_MAP_SHIFTED, &ioh);
/* Initialize sc */
sc->sc_iot = iot;
sc->sc_ioh = ioh;
sc->sc_hw_if = &vs_hw_if;
sc->sc_addr = (void *) ia->ia_addr;
sc->sc_dmas = NULL;
mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_NONE);
mutex_init(&sc->sc_intr_lock, MUTEX_DEFAULT, IPL_SCHED);
/* XXX */
bus_space_map(iot, PPI_ADDR, PPI_MAPSIZE, BUS_SPACE_MAP_SHIFTED,
&sc->sc_ppi);
/* Initialize DMAC */
sc->sc_dmat = ia->ia_dmat;
sc->sc_dma_ch = dmac_alloc_channel(parent, ia->ia_dma, "vs",
ia->ia_dmaintr, vs_dmaintr, sc,
ia->ia_dmaintr+1, vs_dmaerrintr, sc);
aprint_normal_dev(self, "MSM6258V ADPCM voice synthesizer\n");
audio_attach_mi(&vs_hw_if, sc, sc->sc_dev);
}
/*
* vs interrupt handler
*/
static int
vs_dmaintr(void *hdl)
{
struct vs_softc *sc;
DPRINTF(2, ("vs_dmaintr\n"));
sc = hdl;
mutex_spin_enter(&sc->sc_intr_lock);
if (sc->sc_pintr) {
/* start next transfer */
sc->sc_current.dmap += sc->sc_current.blksize;
if (sc->sc_current.dmap + sc->sc_current.blksize
> sc->sc_current.bufsize)
sc->sc_current.dmap -= sc->sc_current.bufsize;
dmac_start_xfer_offset(sc->sc_dma_ch->ch_softc,
sc->sc_current.xfer,
sc->sc_current.dmap,
sc->sc_current.blksize);
sc->sc_pintr(sc->sc_parg);
} else if (sc->sc_rintr) {
/* start next transfer */
sc->sc_current.dmap += sc->sc_current.blksize;
if (sc->sc_current.dmap + sc->sc_current.blksize
> sc->sc_current.bufsize)
sc->sc_current.dmap -= sc->sc_current.bufsize;
dmac_start_xfer_offset(sc->sc_dma_ch->ch_softc,
sc->sc_current.xfer,
sc->sc_current.dmap,
sc->sc_current.blksize);
sc->sc_rintr(sc->sc_rarg);
} else {
printf("vs_dmaintr: spurious interrupt\n");
}
mutex_spin_exit(&sc->sc_intr_lock);
return 1;
}
static int
vs_dmaerrintr(void *hdl)
{
struct vs_softc *sc;
sc = hdl;
DPRINTF(1, ("%s: DMA transfer error.\n", device_xname(sc->sc_dev)));
/* XXX */
vs_dmaintr(sc);
return 1;
}
/*
* audio MD layer interfaces
*/
static int
vs_open(void *hdl, int flags)
{
struct vs_softc *sc;
DPRINTF(1, ("vs_open: flags=%d\n", flags));
sc = hdl;
sc->sc_pintr = NULL;
sc->sc_rintr = NULL;
return 0;
}
static void
vs_close(void *hdl)
{
DPRINTF(1, ("vs_close\n"));
}
static int
vs_query_encoding(void *hdl, struct audio_encoding *fp)
{
DPRINTF(1, ("vs_query_encoding\n"));
if (fp->index >= sizeof(vs_encodings) / sizeof(vs_encodings[0]))
return EINVAL;
strcpy(fp->name, vs_encodings[fp->index].name);
fp->encoding = vs_encodings[fp->index].encoding;
fp->precision = vs_encodings[fp->index].precision;
if (fp->encoding == AUDIO_ENCODING_ADPCM)
fp->flags = 0;
else
fp->flags = AUDIO_ENCODINGFLAG_EMULATED;
return 0;
}
static int
vs_round_sr(u_long rate)
{
int i;
int diff;
int nearest;
diff = rate;
nearest = 0;
for (i = 0; i < NUM_RATE; i++) {
if (rate >= vs_l2r[i].rate) {
if (rate - vs_l2r[i].rate < diff) {
diff = rate - vs_l2r[i].rate;
nearest = i;
}
} else {
if (vs_l2r[i].rate - rate < diff) {
diff = vs_l2r[i].rate - rate;
nearest = i;
}
}
}
if (diff * 100 / rate > 15)
return -1;
else
return nearest;
}
static int
vs_set_params(void *hdl, int setmode, int usemode,
audio_params_t *play, audio_params_t *rec,
stream_filter_list_t *pfil, stream_filter_list_t *rfil)
{
struct vs_softc *sc;
struct audio_params *p;
int mode;
int rate;
stream_filter_factory_t *pswcode;
stream_filter_factory_t *rswcode;
audio_params_t hw;
int matched;
DPRINTF(1, ("vs_set_params: setmode=%d, usemode=%d\n",
setmode, usemode));
sc = hdl;
/* set first record info, then play info */
for (mode = AUMODE_RECORD; mode != -1;
mode = (mode == AUMODE_RECORD) ? AUMODE_PLAY : -1) {
if ((setmode & mode) == 0)
continue;
p = (mode == AUMODE_PLAY) ? play : rec;
if (p->channels != 1)
return EINVAL;
rate = p->sample_rate;
pswcode = NULL;
rswcode = NULL;
hw = *p;
hw.encoding = AUDIO_ENCODING_ADPCM;
hw.precision = hw.validbits = 4;
DPRINTF(1, ("vs_set_params: encoding=%d, precision=%d\n",
p->encoding, p->precision));
matched = 0;
switch (p->precision) {
case 4:
if (p->encoding == AUDIO_ENCODING_ADPCM)
matched = 1;
break;
case 8:
switch (p->encoding) {
case AUDIO_ENCODING_ULAW:
matched = 1;
hw.encoding = AUDIO_ENCODING_ULINEAR_LE;
hw.precision = hw.validbits = 8;
pfil->prepend(pfil, mulaw_to_linear8, &hw);
hw.encoding = AUDIO_ENCODING_ADPCM;
hw.precision = hw.validbits = 4;
pfil->prepend(pfil, msm6258_linear8_to_adpcm, &hw);
rfil->append(rfil, msm6258_adpcm_to_linear8, &hw);
hw.encoding = AUDIO_ENCODING_ULINEAR_LE;
hw.precision = hw.validbits = 8;
rfil->append(rfil, linear8_to_mulaw, &hw);
break;
case AUDIO_ENCODING_SLINEAR:
case AUDIO_ENCODING_SLINEAR_LE:
case AUDIO_ENCODING_SLINEAR_BE:
case AUDIO_ENCODING_ULINEAR:
case AUDIO_ENCODING_ULINEAR_LE:
case AUDIO_ENCODING_ULINEAR_BE:
matched = 1;
pfil->append(pfil, msm6258_linear8_to_adpcm, &hw);
rfil->append(rfil, msm6258_adpcm_to_linear8, &hw);
break;
}
break;
case 16:
switch (p->encoding) {
case AUDIO_ENCODING_SLINEAR_LE:
case AUDIO_ENCODING_SLINEAR_BE:
matched = 1;
pfil->append(pfil, msm6258_slinear16_to_adpcm, &hw);
rfil->append(rfil, msm6258_adpcm_to_slinear16, &hw);
break;
}
break;
}
if (matched == 0) {
DPRINTF(1, ("vs_set_params: mode=%d, encoding=%d\n",
mode, p->encoding));
return EINVAL;
}
DPRINTF(1, ("vs_set_params: rate=%d -> ", rate));
rate = vs_round_sr(rate);
DPRINTF(1, ("%d\n", rate));
if (rate < 0)
return EINVAL;
if (mode == AUMODE_PLAY) {
sc->sc_current.prate = rate;
} else {
sc->sc_current.rrate = rate;
}
}
return 0;
}
static void
vs_set_sr(struct vs_softc *sc, int rate)
{
DPRINTF(1, ("setting sample rate to %d, %d\n",
rate, (int)vs_l2r[rate].rate));
bus_space_write_1(sc->sc_iot, sc->sc_ppi, PPI_PORTC,
(bus_space_read_1 (sc->sc_iot, sc->sc_ppi,
PPI_PORTC) & 0xf0)
| vs_l2r[rate].den);
adpcm_chgclk(vs_l2r[rate].clk);
}
static inline void
vs_set_po(struct vs_softc *sc, u_long po)
{
bus_space_write_1(sc->sc_iot, sc->sc_ppi, PPI_PORTC,
(bus_space_read_1(sc->sc_iot, sc->sc_ppi, PPI_PORTC)
& 0xfc) | po);
}
static int
vs_trigger_output(void *hdl, void *start, void *end, int bsize,
void (*intr)(void *), void *arg,
const audio_params_t *p)
{
struct vs_softc *sc;
struct vs_dma *vd;
struct dmac_dma_xfer *xf;
struct dmac_channel_stat *chan;
DPRINTF(2, ("vs_trigger_output: start=%p, bsize=%d, intr=%p, arg=%p\n",
start, bsize, intr, arg));
sc = hdl;
chan = sc->sc_dma_ch;
sc->sc_pintr = intr;
sc->sc_parg = arg;
sc->sc_current.blksize = bsize;
sc->sc_current.bufsize = (char *)end - (char *)start;
sc->sc_current.dmap = 0;
/* Find DMA buffer. */
for (vd = sc->sc_dmas; vd != NULL && KVADDR(vd) != start;
vd = vd->vd_next)
continue;
if (vd == NULL) {
printf("%s: trigger_output: bad addr %p\n",
device_xname(sc->sc_dev), start);
return EINVAL;
}
vs_set_sr(sc, sc->sc_current.prate);
vs_set_po(sc, VS_PANOUT_LR);
xf = dmac_alloc_xfer(chan, sc->sc_dmat, vd->vd_map);
sc->sc_current.xfer = xf;
chan->ch_dcr = (DMAC_DCR_XRM_CSWOH | DMAC_DCR_OTYP_EASYNC |
DMAC_DCR_OPS_8BIT);
chan->ch_ocr = DMAC_OCR_REQG_EXTERNAL;
xf->dx_ocr = DMAC_OCR_DIR_MTD;
xf->dx_scr = DMAC_SCR_MAC_COUNT_UP | DMAC_SCR_DAC_NO_COUNT;
xf->dx_device = sc->sc_addr + MSM6258_DATA*2 + 1;
dmac_load_xfer(chan->ch_softc, xf);
dmac_start_xfer_offset(chan->ch_softc, xf, 0, sc->sc_current.blksize);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, MSM6258_STAT, 2);
return 0;
}
static int
vs_trigger_input(void *hdl, void *start, void *end, int bsize,
void (*intr)(void *), void *arg,
const audio_params_t *p)
{
struct vs_softc *sc;
struct vs_dma *vd;
struct dmac_dma_xfer *xf;
struct dmac_channel_stat *chan;
DPRINTF(2, ("vs_trigger_input: start=%p, bsize=%d, intr=%p, arg=%p\n",
start, bsize, intr, arg));
sc = hdl;
chan = sc->sc_dma_ch;
sc->sc_rintr = intr;
sc->sc_rarg = arg;
sc->sc_current.blksize = bsize;
sc->sc_current.bufsize = (char *)end - (char *)start;
sc->sc_current.dmap = 0;
/* Find DMA buffer. */
for (vd = sc->sc_dmas; vd != NULL && KVADDR(vd) != start;
vd = vd->vd_next)
continue;
if (vd == NULL) {
printf("%s: trigger_output: bad addr %p\n",
device_xname(sc->sc_dev), start);
return EINVAL;
}
vs_set_sr(sc, sc->sc_current.rrate);
xf = dmac_alloc_xfer(chan, sc->sc_dmat, vd->vd_map);
sc->sc_current.xfer = xf;
chan->ch_dcr = (DMAC_DCR_XRM_CSWOH | DMAC_DCR_OTYP_EASYNC |
DMAC_DCR_OPS_8BIT);
chan->ch_ocr = DMAC_OCR_REQG_EXTERNAL;
xf->dx_ocr = DMAC_OCR_DIR_DTM;
xf->dx_scr = DMAC_SCR_MAC_COUNT_UP | DMAC_SCR_DAC_NO_COUNT;
xf->dx_device = sc->sc_addr + MSM6258_DATA*2 + 1;
dmac_load_xfer(chan->ch_softc, xf);
dmac_start_xfer_offset(chan->ch_softc, xf, 0, sc->sc_current.blksize);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, MSM6258_STAT, 4);
return 0;
}
static int
vs_halt_output(void *hdl)
{
struct vs_softc *sc;
DPRINTF(1, ("vs_halt_output\n"));
sc = hdl;
/* stop ADPCM play */
dmac_abort_xfer(sc->sc_dma_ch->ch_softc, sc->sc_current.xfer);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, MSM6258_STAT, 1);
return 0;
}
static int
vs_halt_input(void *hdl)
{
struct vs_softc *sc;
DPRINTF(1, ("vs_halt_input\n"));
sc = hdl;
/* stop ADPCM recoding */
dmac_abort_xfer(sc->sc_dma_ch->ch_softc, sc->sc_current.xfer);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, MSM6258_STAT, 1);
return 0;
}
static int
vs_allocmem(struct vs_softc *sc, size_t size, size_t align, size_t boundary,
struct vs_dma *vd)
{
int error;
#ifdef DIAGNOSTIC
if (size > DMAC_MAXSEGSZ)
panic ("vs_allocmem: maximum size exceeded, %d", (int) size);
#endif
vd->vd_size = size;
error = bus_dmamem_alloc(vd->vd_dmat, vd->vd_size, align, boundary,
vd->vd_segs,
sizeof (vd->vd_segs) / sizeof (vd->vd_segs[0]),
&vd->vd_nsegs, BUS_DMA_WAITOK);
if (error)
goto out;
error = bus_dmamem_map(vd->vd_dmat, vd->vd_segs, vd->vd_nsegs,
vd->vd_size, &vd->vd_addr,
BUS_DMA_WAITOK | BUS_DMA_COHERENT);
if (error)
goto free;
error = bus_dmamap_create(vd->vd_dmat, vd->vd_size, 1, DMAC_MAXSEGSZ,
0, BUS_DMA_WAITOK, &vd->vd_map);
if (error)
goto unmap;
error = bus_dmamap_load(vd->vd_dmat, vd->vd_map, vd->vd_addr,
vd->vd_size, NULL, BUS_DMA_WAITOK);
if (error)
goto destroy;
return 0;
destroy:
bus_dmamap_destroy(vd->vd_dmat, vd->vd_map);
unmap:
bus_dmamem_unmap(vd->vd_dmat, vd->vd_addr, vd->vd_size);
free:
bus_dmamem_free(vd->vd_dmat, vd->vd_segs, vd->vd_nsegs);
out:
return error;
}
static void
vs_freemem(struct vs_dma *vd)
{
bus_dmamap_unload(vd->vd_dmat, vd->vd_map);
bus_dmamap_destroy(vd->vd_dmat, vd->vd_map);
bus_dmamem_unmap(vd->vd_dmat, vd->vd_addr, vd->vd_size);
bus_dmamem_free(vd->vd_dmat, vd->vd_segs, vd->vd_nsegs);
}
static int
vs_getdev(void *hdl, struct audio_device *retp)
{
DPRINTF(1, ("vs_getdev\n"));
*retp = vs_device;
return 0;
}
static int
vs_set_port(void *hdl, mixer_ctrl_t *cp)
{
DPRINTF(1, ("vs_set_port\n"));
return 0;
}
static int
vs_get_port(void *hdl, mixer_ctrl_t *cp)
{
DPRINTF(1, ("vs_get_port\n"));
return 0;
}
static int
vs_query_devinfo(void *hdl, mixer_devinfo_t *mi)
{
DPRINTF(1, ("vs_query_devinfo\n"));
switch (mi->index) {
default:
return EINVAL;
}
return 0;
}
static void *
vs_allocm(void *hdl, int direction, size_t size)
{
struct vs_softc *sc;
struct vs_dma *vd;
int error;
if ((vd = kmem_alloc(sizeof(*vd), KM_SLEEP)) == NULL)
return NULL;
sc = hdl;
vd->vd_dmat = sc->sc_dmat;
error = vs_allocmem(sc, size, 32, 0, vd);
if (error) {
kmem_free(vd, sizeof(*vd));
return NULL;
}
vd->vd_next = sc->sc_dmas;
sc->sc_dmas = vd;
return KVADDR(vd);
}
static void
vs_freem(void *hdl, void *addr, size_t size)
{
struct vs_softc *sc;
struct vs_dma *p, **pp;
sc = hdl;
for (pp = &sc->sc_dmas; (p = *pp) != NULL; pp = &p->vd_next) {
if (KVADDR(p) == addr) {
vs_freemem(p);
*pp = p->vd_next;
kmem_free(p, sizeof(*p));
return;
}
}
}
static size_t
vs_round_buffersize(void *hdl, int direction, size_t bufsize)
{
if (bufsize > DMAC_MAXSEGSZ)
bufsize = DMAC_MAXSEGSZ;
return bufsize;
}
#if 0
paddr_t
vs_mappage(void *addr, void *mem, off_t off, int prot)
{
struct vs_softc *sc;
struct vs_dma *p;
if (off < 0)
return -1;
sc = addr;
for (p = sc->sc_dmas; p != NULL && KVADDR(p) != mem;
p = p->vd_next)
continue;
if (p == NULL) {
printf("%s: mappage: bad addr %p\n",
device_xname(sc->sc_dev), start);
return -1;
}
return bus_dmamem_mmap(sc->sc_dmat, p->vd_segs, p->vd_nsegs,
off, prot, BUS_DMA_WAITOK);
}
#endif
static int
vs_get_props(void *hdl)
{
DPRINTF(1, ("vs_get_props\n"));
return 0 /* | dependent | half duplex | no mmap */;
}
static void
vs_get_locks(void *hdl, kmutex_t **intr, kmutex_t **thread)
{
struct vs_softc *sc;
DPRINTF(1, ("vs_get_locks\n"));
sc = hdl;
*intr = &sc->sc_intr_lock;
*thread = &sc->sc_lock;
}
#endif /* NAUDIO > 0 && NVS > 0*/