1180 lines
32 KiB
C
1180 lines
32 KiB
C
/*
|
|
* Copyright (c) 1999 Cameron Grant <gandalf@vilnya.demon.co.uk>
|
|
* Portions Copyright by Luigi Rizzo - 1997-99
|
|
* 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 AND CONTRIBUTORS ``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 OR CONTRIBUTORS 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.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include <dev/sound/pcm/sound.h>
|
|
|
|
#define MIN_CHUNK_SIZE 256 /* for uiomove etc. */
|
|
#define DMA_ALIGN_THRESHOLD 4
|
|
#define DMA_ALIGN_MASK (~(DMA_ALIGN_THRESHOLD - 1))
|
|
|
|
#define ISA_DMA(b) (((b)->chan >= 0 && (b)->chan != 4 && (b)->chan < 8))
|
|
#define CANCHANGE(c) (!(c)->buffer.dl)
|
|
/*
|
|
#define DEB(x) x
|
|
*/
|
|
static void chn_clearbuf(pcm_channel *c, snd_dbuf *b, int length);
|
|
static void chn_dmaupdate(pcm_channel *c);
|
|
static void chn_wrintr(pcm_channel *c);
|
|
static void chn_rdintr(pcm_channel *c);
|
|
/*
|
|
* SOUND OUTPUT
|
|
|
|
We use a circular buffer to store samples directed to the DAC.
|
|
The buffer is split into two variable-size regions, each identified
|
|
by an offset in the buffer (rp,fp) and a length (rl,fl):
|
|
|
|
0 rp,rl fp,fl bufsize
|
|
|__________>____________>________|
|
|
FREE d READY w FREE
|
|
|
|
READY: data written from the process and ready to be sent to the DAC;
|
|
FREE: free part of the buffer.
|
|
|
|
Both regions can wrap around the end of the buffer. At initialization,
|
|
READY is empty, FREE takes all the available space, and dma is
|
|
idle. dl contains the length of the current DMA transfer, dl=0
|
|
means that the dma is idle.
|
|
|
|
The two boundaries (rp,fp) in the buffers are advanced by DMA [d]
|
|
and write() [w] operations. The first portion of the READY region
|
|
is used for DMA transfers. The transfer is started at rp and with
|
|
chunks of length dl. During DMA operations, dsp_wr_dmaupdate()
|
|
updates rp, rl and fl tracking the ISA DMA engine as the transfer
|
|
makes progress.
|
|
When a new block is written, fp advances and rl,fl are updated
|
|
accordingly.
|
|
|
|
The code works as follows: the user write routine dsp_write_body()
|
|
fills up the READY region with new data (reclaiming space from the
|
|
FREE region) and starts the write DMA engine if inactive. When a
|
|
DMA transfer is complete, an interrupt causes dsp_wrintr() to be
|
|
called which extends the FREE region and possibly starts the next
|
|
transfer.
|
|
|
|
In some cases, the code tries to track the current status of DMA
|
|
operations by calling dsp_wr_dmaupdate() which changes rp, rl and fl.
|
|
|
|
The system tries to make all DMA transfers use the same size,
|
|
play_blocksize or rec_blocksize. The size is either selected by
|
|
the user, or computed by the system to correspond to about .25s of
|
|
audio. The blocksize must be within a range which is currently:
|
|
|
|
min(5ms, 40 bytes) ... 1/2 buffer size.
|
|
|
|
When there aren't enough data (write) or space (read), a transfer
|
|
is started with a reduced size.
|
|
|
|
To reduce problems in case of overruns, the routine which fills up
|
|
the buffer should initialize (e.g. by repeating the last value) a
|
|
reasonably long area after the last block so that no noise is
|
|
produced on overruns.
|
|
|
|
*
|
|
*/
|
|
|
|
|
|
/* XXX this is broken: in the event a bounce buffer is used, data never
|
|
* gets copied in or out of the real buffer. fix requires mods to isa_dma.c
|
|
* and possibly fixes to other autodma mode clients
|
|
*/
|
|
static void
|
|
chn_isadmabounce(pcm_channel *c)
|
|
{
|
|
if (ISA_DMA(&c->buffer)) {
|
|
/* tell isa_dma to bounce data in/out */
|
|
} else KASSERT(1, ("chn_isadmabounce called on invalid channel"));
|
|
}
|
|
|
|
static int
|
|
chn_polltrigger(pcm_channel *c)
|
|
{
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
unsigned lim = (c->flags & CHN_F_HAS_SIZE)? c->blocksize2nd : 1;
|
|
int trig = 0;
|
|
|
|
if (c->flags & CHN_F_MAPPED)
|
|
trig = ((bs->int_count > bs->prev_int_count) || bs->first_poll);
|
|
else trig = (((c->direction == PCMDIR_PLAY)? bs->rl : bs->fl) < lim);
|
|
return trig;
|
|
}
|
|
|
|
static int
|
|
chn_pollreset(pcm_channel *c)
|
|
{
|
|
snd_dbuf *bs = &c->buffer;
|
|
|
|
if (c->flags & CHN_F_MAPPED) bs->prev_int_count = bs->int_count;
|
|
bs->first_poll = 0;
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* chn_dmadone() updates pointers and wakes up any process waiting
|
|
* on a select(). Must be called at spltty().
|
|
*/
|
|
static void
|
|
chn_dmadone(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
|
|
if (c->direction == PCMDIR_PLAY)
|
|
chn_checkunderflow(c);
|
|
else
|
|
chn_dmaupdate(c);
|
|
if (ISA_DMA(b)) chn_isadmabounce(c); /* sync bounce buffer */
|
|
b->int_count++;
|
|
}
|
|
|
|
/*
|
|
* chn_dmawakeup() wakes up any process sleeping. Separated from
|
|
* chn_dmadone() so that wakeup occurs only when feed from a
|
|
* secondary buffer to a DMA buffer takes place. Must be called
|
|
* at spltty().
|
|
*/
|
|
static void
|
|
chn_dmawakeup(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
|
|
wakeup(b);
|
|
}
|
|
|
|
/*
|
|
* chn_dmaupdate() tracks the status of a dma transfer,
|
|
* updating pointers. It must be called at spltty().
|
|
*
|
|
* NOTE: when we are using auto dma in the device, rl might become
|
|
* negative.
|
|
*/
|
|
DEB (static int chn_updatecount=0);
|
|
|
|
static void
|
|
chn_dmaupdate(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
int delta, hwptr;
|
|
DEB (int b_rl=b->rl; int b_fl=b->fl; int b_rp=b->rp; int b_fp=b->fp);
|
|
|
|
hwptr = chn_getptr(c);
|
|
if (c->direction == PCMDIR_PLAY) {
|
|
delta = (b->bufsize + hwptr - b->rp) % b->bufsize;
|
|
b->rp = hwptr;
|
|
b->rl -= delta;
|
|
b->fl += delta;
|
|
|
|
if (b->rl < 0) {
|
|
DEB(printf("OUCH!(%d) rl %d(%d) delta %d bufsize %d hwptr %d rp %d(%d)\n", chn_updatecount++, b->rl, b_rl, delta, b->bufsize, hwptr, b->rp, b_rp));
|
|
}
|
|
} else {
|
|
delta = (b->bufsize + hwptr - b->fp) % b->bufsize;
|
|
b->fp = hwptr;
|
|
b->rl += delta;
|
|
b->fl -= delta;
|
|
if (b->fl < 0) {
|
|
DEB(printf("OUCH!(%d) fl %d(%d) delta %d bufsize %d hwptr %d fp %d(%d)\n", chn_updatecount++, b->fl, b_fl, delta, b->bufsize, hwptr, b->fp, b_fp));
|
|
}
|
|
}
|
|
b->total += delta;
|
|
}
|
|
|
|
/*
|
|
* Check channel for underflow occured. Reset DMA buffer in case of
|
|
* underflow, so that new data can go into the buffer. It must be
|
|
* called at spltty().
|
|
*/
|
|
void
|
|
chn_checkunderflow(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
|
|
if (b->underflow) {
|
|
DEB(printf("Clear underflow condition\n"));
|
|
/*
|
|
* The DMA keeps running even after underflow occurs.
|
|
* Hence the value returned by chn_getptr() here soon
|
|
* gets a lag when we get back to chn_write(). Although
|
|
* there are no easy and precise methods to figure out
|
|
* the lag, a quarter of b->bufsize would be a fair
|
|
* choice, provided that a DMA interrupt generates upon
|
|
* each transfer of a half b->bufsize.
|
|
*/
|
|
b->rp = chn_getptr(c);
|
|
b->fp = (b->rp + b->bufsize / 4) % b->bufsize;
|
|
b->rl = b->bufsize / 4;
|
|
b->fl = b->bufsize - b->rl;
|
|
b->underflow = 0;
|
|
} else {
|
|
chn_dmaupdate(c);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Feeds new data to the write dma buffer. Can be called in the bottom half.
|
|
* Hence must be called at spltty.
|
|
*/
|
|
int
|
|
chn_wrfeed(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
int a, l, lacc;
|
|
|
|
/* ensure we always have a whole number of samples */
|
|
a = (1 << c->align) - 1;
|
|
lacc = 0;
|
|
/* Don't allow write unaligned data */
|
|
while (bs->rl > a && b->fl > a) {
|
|
/* ensure we always have a whole number of samples */
|
|
l = min(min(bs->rl, bs->bufsize - bs->rp), min(b->fl, b->bufsize - b->fp)) & ~a;
|
|
if (l == 0)
|
|
return lacc;
|
|
/* Move the samples, update the markers and pointers. */
|
|
bcopy(bs->buf + bs->rp, b->buf + b->fp, l);
|
|
bs->fl += l;
|
|
bs->rl -= l;
|
|
bs->rp = (bs->rp + l) % bs->bufsize;
|
|
b->rl += l;
|
|
b->fl -= l;
|
|
b->fp = (b->fp + l) % b->bufsize;
|
|
/* Clear the new space in the secondary buffer. */
|
|
chn_clearbuf(c, bs, l);
|
|
/* Accumulate the total bytes of the moved samples. */
|
|
lacc += l;
|
|
/* A feed to the DMA buffer is equivalent to an interrupt. */
|
|
bs->int_count++;
|
|
if (bs->sel.si_pid && chn_polltrigger(c)) selwakeup(&bs->sel);
|
|
}
|
|
|
|
return lacc;
|
|
}
|
|
|
|
/* Feeds new data to the secondary write buffer. */
|
|
static int
|
|
chn_wrfeed2nd(pcm_channel *c, struct uio *buf)
|
|
{
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
int l, w, wacc;
|
|
|
|
/* The DMA buffer may have some space. */
|
|
while (chn_wrfeed(c) > 0);
|
|
|
|
/* ensure we always have a whole number of samples */
|
|
wacc = 0;
|
|
while (buf->uio_resid > 0 && bs->fl > 0) {
|
|
/*
|
|
* The size of the data to move here does not have to be
|
|
* aligned. We take care of it upon moving the data to a
|
|
* DMA buffer.
|
|
*/
|
|
l = min(bs->fl, bs->bufsize - bs->fp);
|
|
/* Move the samples, update the markers and pointers. */
|
|
w = c->feeder->feed(c->feeder, c, bs->buf + bs->fp, l, buf);
|
|
if (w == 0) panic("no feed");
|
|
bs->rl += w;
|
|
bs->fl -= w;
|
|
bs->fp = (bs->fp + w) % bs->bufsize;
|
|
/* Accumulate the total bytes of the moved samples. */
|
|
bs->total += w;
|
|
wacc += w;
|
|
|
|
/* If any pcm data gets moved, push it to the DMA buffer. */
|
|
if (w > 0)
|
|
while (chn_wrfeed(c) > 0);
|
|
}
|
|
|
|
return wacc;
|
|
}
|
|
|
|
/*
|
|
* Write interrupt routine. Can be called from other places (e.g.
|
|
* to start a paused transfer), but with interrupts disabled.
|
|
*/
|
|
static void
|
|
chn_wrintr(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
int start, dl, l;
|
|
|
|
if (b->underflow && !(c->flags & CHN_F_MAPPED)) {
|
|
/* printf("underflow return\n");
|
|
*/ return; /* nothing new happened */
|
|
}
|
|
if (b->dl) chn_dmadone(c);
|
|
|
|
/*
|
|
* start another dma operation only if have ready data in the buffer,
|
|
* there is no pending abort, have a full-duplex device, or have a
|
|
* half duplex device and there is no pending op on the other side.
|
|
*
|
|
* Force transfers to be aligned to a boundary of 4, which is
|
|
* needed when doing stereo and 16-bit.
|
|
*/
|
|
|
|
/*
|
|
* Prepare new space of at least c->blocksize in the DMA
|
|
* buffer for mmap.
|
|
*/
|
|
if (c->flags & CHN_F_MAPPED && b->fl < c->blocksize) {
|
|
dl = c->blocksize - b->fl;
|
|
b->fl += dl;
|
|
b->rl -= dl;
|
|
b->rp = (b->rp + dl) % b->bufsize;
|
|
chn_clearbuf(c, b, dl);
|
|
}
|
|
|
|
/* Check underflow and update the pointers. */
|
|
chn_checkunderflow(c);
|
|
|
|
/*
|
|
* Fill up the DMA buffer, followed by waking up the top half.
|
|
* If some of the pcm data in uio are still left, the top half
|
|
* goes to sleep by itself.
|
|
*/
|
|
while (chn_wrfeed(c) > 0);
|
|
chn_clearbuf(c, b, b->fl);
|
|
chn_dmawakeup(c);
|
|
if (c->flags & CHN_F_MAPPED)
|
|
start = c->flags & CHN_F_TRIGGERED;
|
|
else {
|
|
/* printf("%d >= %d && !(%x & %x)\n", b->rl, DMA_ALIGN_THRESHOLD, c->flags, CHN_F_ABORTING | CHN_F_CLOSING);
|
|
*/ start = (b->rl >= DMA_ALIGN_THRESHOLD && !(c->flags & CHN_F_ABORTING));
|
|
}
|
|
if (start) {
|
|
chn_dmaupdate(c);
|
|
if (c->flags & CHN_F_MAPPED) l = c->blocksize;
|
|
else l = min(b->rl, c->blocksize) & DMA_ALIGN_MASK;
|
|
/*
|
|
* check if we need to reprogram the DMA on the sound card.
|
|
* This happens if the size has changed from zero
|
|
*/
|
|
if (b->dl == 0) {
|
|
/* Start DMA operation */
|
|
b->dl = c->blocksize; /* record new transfer size */
|
|
chn_trigger(c, PCMTRIG_START);
|
|
}
|
|
/*
|
|
* Emulate writing by DMA, i.e. transfer the pcm data from
|
|
* the emulated-DMA buffer to the device itself.
|
|
*/
|
|
chn_trigger(c, PCMTRIG_EMLDMAWR);
|
|
if (b->dl != l) {
|
|
DEB(printf("near underflow %d, %d, %d\n", l, b->dl, b->fl));
|
|
/*
|
|
* we are near to underflow condition, so to prevent
|
|
* audio 'clicks' clear next b->fl bytes
|
|
*/
|
|
chn_clearbuf(c, b, b->fl);
|
|
}
|
|
} else {
|
|
/* cannot start a new dma transfer */
|
|
DEB(printf("underflow, flags 0x%08x rp %d rl %d\n", c->flags, b->rp, b->rl));
|
|
if (b->dl) { /* DMA was active */
|
|
b->underflow = 1; /* set underflow flag */
|
|
chn_clearbuf(c, b, b->bufsize); /* and clear all DMA buffer */
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* user write routine
|
|
*
|
|
* advance the boundary between READY and FREE, fill the space with
|
|
* uiomove(), and possibly start DMA. Do the above until the transfer
|
|
* is complete.
|
|
*
|
|
* To minimize latency in case a pending DMA transfer is about to end,
|
|
* we do the transfer in pieces of increasing sizes, extending the
|
|
* READY area at every checkpoint. In the (necessary) assumption that
|
|
* memory bandwidth is larger than the rate at which the dma consumes
|
|
* data, we reduce the latency to something proportional to the length
|
|
* of the first piece, while keeping the overhead low and being able
|
|
* to feed the DMA with large blocks.
|
|
*/
|
|
|
|
int
|
|
chn_write(pcm_channel *c, struct uio *buf)
|
|
{
|
|
int ret = 0, timeout, res, newsize;
|
|
long s;
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
if (c->flags & CHN_F_WRITING) {
|
|
/* This shouldn't happen and is actually silly
|
|
* - will never wake up, just timeout; why not sleep on b?
|
|
*/
|
|
tsleep(&s, PZERO, "pcmwrW", hz);
|
|
return EBUSY;
|
|
}
|
|
c->flags |= CHN_F_WRITING;
|
|
c->flags &= ~CHN_F_ABORTING;
|
|
s = spltty();
|
|
|
|
/*
|
|
* XXX Certain applications attempt to write larger size
|
|
* of pcm data than c->blocksize2nd without blocking,
|
|
* resulting partial write. Expand the block size so that
|
|
* the write operation avoids blocking.
|
|
*/
|
|
if ((c->flags & CHN_F_NBIO) && buf->uio_resid > c->blocksize2nd) {
|
|
for (newsize = 1 ; newsize < min(buf->uio_resid, CHN_2NDBUFWHOLESIZE) ; newsize <<= 1);
|
|
chn_setblocksize(c, newsize * c->fragments);
|
|
c->blocksize2nd = newsize;
|
|
c->fragments = bs->bufsize / c->blocksize2nd;
|
|
}
|
|
|
|
/* Store the initial size in the uio. */
|
|
res = buf->uio_resid;
|
|
|
|
/*
|
|
* Fill up the secondary and DMA buffer.
|
|
* chn_wrfeed*() takes care of the alignment.
|
|
*/
|
|
|
|
/* Check for underflow before writing into the buffers. */
|
|
chn_checkunderflow(c);
|
|
while (chn_wrfeed2nd(c, buf) > 0);
|
|
|
|
/* Start playing if not yet. */
|
|
if ((bs->rl || b->rl) && !b->dl) {
|
|
chn_wrintr(c);
|
|
}
|
|
|
|
if (c->flags & CHN_F_NBIO) {
|
|
/* If no pcm data was written on nonblocking, return EAGAIN. */
|
|
if (buf->uio_resid == res)
|
|
ret = EAGAIN;
|
|
} else {
|
|
/* Wait until all samples are played in blocking mode. */
|
|
while (buf->uio_resid > 0) {
|
|
/* Check for underflow before writing into the buffers. */
|
|
chn_checkunderflow(c);
|
|
/* Fill up the buffers with new pcm data. */
|
|
while (chn_wrfeed2nd(c, buf) > 0);
|
|
|
|
/* Start playing if necessary. */
|
|
if ((bs->rl || b->rl) && !b->dl) chn_wrintr(c);
|
|
|
|
/* Have we finished to feed the secondary buffer? */
|
|
if (buf->uio_resid == 0)
|
|
break;
|
|
|
|
/* Wait for new free space to write new pcm samples. */
|
|
splx(s);
|
|
timeout = (buf->uio_resid >= b->dl)? hz / 20 : 1;
|
|
ret = tsleep(b, PRIBIO | PCATCH, "pcmwr", timeout);
|
|
s = spltty();
|
|
/* if (ret == EINTR) chn_abort(c); */
|
|
if (ret == EINTR || ret == ERESTART) break;
|
|
}
|
|
}
|
|
c->flags &= ~CHN_F_WRITING;
|
|
splx(s);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* SOUND INPUT
|
|
*
|
|
|
|
The input part is similar to the output one, with a circular buffer
|
|
split in two regions, and boundaries advancing because of read() calls
|
|
[r] or dma operation [d]. At initialization, as for the write
|
|
routine, READY is empty, and FREE takes all the space.
|
|
|
|
0 rp,rl fp,fl bufsize
|
|
|__________>____________>________|
|
|
FREE r READY d FREE
|
|
|
|
Operation is as follows: upon user read (dsp_read_body()) a DMA read
|
|
is started if not already active (marked by b->dl > 0),
|
|
then as soon as data are available in the READY region they are
|
|
transferred to the user buffer, thus advancing the boundary between FREE
|
|
and READY. Upon interrupts, caused by a completion of a DMA transfer,
|
|
the READY region is extended and possibly a new transfer is started.
|
|
|
|
When necessary, dsp_rd_dmaupdate() is called to advance fp (and update
|
|
rl,fl accordingly). Upon user reads, rp is advanced and rl,fl are
|
|
updated accordingly.
|
|
|
|
The rules to choose the size of the new DMA area are similar to
|
|
the other case, with a preferred constant transfer size equal to
|
|
rec_blocksize, and fallback to smaller sizes if no space is available.
|
|
|
|
*/
|
|
|
|
/*
|
|
* Feed new data from the read buffer. Can be called in the bottom half.
|
|
* Hence must be called at spltty.
|
|
*/
|
|
int
|
|
chn_rdfeed(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
int l, lacc;
|
|
|
|
/* ensure we always have a whole number of samples */
|
|
lacc = 0;
|
|
while (bs->fl >= DMA_ALIGN_THRESHOLD && b->rl >= DMA_ALIGN_THRESHOLD) {
|
|
l = min(min(bs->fl, bs->bufsize - bs->fp), min(b->rl, b->bufsize - b->rp)) & DMA_ALIGN_MASK;
|
|
/* Move the samples, update the markers and pointers. */
|
|
bcopy(b->buf + b->rp, bs->buf + bs->fp, l);
|
|
bs->fl -= l;
|
|
bs->rl += l;
|
|
bs->fp = (bs->fp + l) % bs->bufsize;
|
|
b->rl -= l;
|
|
b->fl += l;
|
|
b->rp = (b->rp + l) % b->bufsize;
|
|
/* Clear the new space in the DMA buffer. */
|
|
chn_clearbuf(c, b, l);
|
|
/* Accumulate the total bytes of the moved samples. */
|
|
lacc += l;
|
|
/* A feed from the DMA buffer is equivalent to an interrupt. */
|
|
bs->int_count++;
|
|
if (bs->sel.si_pid && chn_polltrigger(c)) selwakeup(&bs->sel);
|
|
}
|
|
|
|
return lacc;
|
|
}
|
|
|
|
/* Feeds new data from the secondary read buffer. */
|
|
static int
|
|
chn_rdfeed2nd(pcm_channel *c, struct uio *buf)
|
|
{
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
int l, w, wacc;
|
|
|
|
/* The DMA buffer may have pcm data. */
|
|
while(chn_rdfeed(c) > 0);
|
|
|
|
/* ensure we always have a whole number of samples */
|
|
wacc = 0;
|
|
while (buf->uio_resid > 0 && bs->rl > 0) {
|
|
/*
|
|
* The size of the data to move here does not have to be
|
|
* aligned. We take care of it upon moving the data to a
|
|
* DMA buffer.
|
|
*/
|
|
l = min(bs->rl, bs->bufsize - bs->rp);
|
|
/* Move the samples, update the markers and pointers. */
|
|
w = c->feeder->feed(c->feeder, c, bs->buf + bs->rp, l, buf);
|
|
if (w == 0) panic("no feed");
|
|
bs->fl += w;
|
|
bs->rl -= w;
|
|
bs->rp = (bs->rp + w) % bs->bufsize;
|
|
/* Clear the new space in the secondary buffer. */
|
|
chn_clearbuf(c, bs, l);
|
|
/* Accumulate the total bytes of the moved samples. */
|
|
bs->total += w;
|
|
wacc += w;
|
|
|
|
/* If any pcm data gets moved, suck up the DMA buffer. */
|
|
if (w > 0)
|
|
while (chn_rdfeed(c) > 0);
|
|
}
|
|
|
|
return wacc;
|
|
}
|
|
|
|
/* read interrupt routine. Must be called with interrupts blocked. */
|
|
static void
|
|
chn_rdintr(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
int start, dl;
|
|
|
|
if (b->dl) chn_dmadone(c);
|
|
|
|
DEB(printf("rdintr: start dl %d, rp:rl %d:%d, fp:fl %d:%d\n",
|
|
b->dl, b->rp, b->rl, b->fp, b->fl));
|
|
/* Restart if have enough free space to absorb overruns */
|
|
|
|
/*
|
|
* Prepare new space of at least c->blocksize in the secondary
|
|
* buffer for mmap.
|
|
*/
|
|
if (c->flags & CHN_F_MAPPED && bs->fl < c->blocksize) {
|
|
dl = c->blocksize - bs->fl;
|
|
bs->fl += dl;
|
|
bs->rl -= dl;
|
|
bs->rp = (bs->rp + dl) % bs->bufsize;
|
|
chn_clearbuf(c, bs, dl);
|
|
}
|
|
|
|
/* Update the pointers. */
|
|
chn_dmaupdate(c);
|
|
|
|
/*
|
|
* Suck up the DMA buffer, followed by waking up the top half.
|
|
* If some of the pcm data in the secondary buffer are still left,
|
|
* the top half goes to sleep by itself.
|
|
*/
|
|
while(chn_rdfeed(c) > 0);
|
|
chn_dmawakeup(c);
|
|
if (c->flags & CHN_F_MAPPED)
|
|
start = c->flags & CHN_F_TRIGGERED;
|
|
else
|
|
start = (b->fl > 0x200 && !(c->flags & CHN_F_ABORTING));
|
|
if (start) {
|
|
int l = min(b->fl - 0x100, c->blocksize);
|
|
if (c->flags & CHN_F_MAPPED) l = c->blocksize;
|
|
l &= DMA_ALIGN_MASK ; /* realign sizes */
|
|
|
|
DEB(printf("rdintr: dl %d -> %d\n", b->dl, l);)
|
|
if (l != b->dl) {
|
|
/* size has changed. Stop and restart */
|
|
if (b->dl) {
|
|
chn_trigger(c, PCMTRIG_STOP);
|
|
chn_dmaupdate(c);
|
|
l = min(b->fl - 0x100, c->blocksize);
|
|
l &= DMA_ALIGN_MASK ; /* realign sizes */
|
|
}
|
|
b->dl = l;
|
|
chn_trigger(c, PCMTRIG_START);
|
|
}
|
|
} else {
|
|
if (b->dl) { /* was active */
|
|
b->dl = 0;
|
|
chn_trigger(c, PCMTRIG_STOP);
|
|
chn_dmaupdate(c);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* body of user-read routine
|
|
*
|
|
* Start DMA if not active; wait for READY not empty.
|
|
* Transfer data from READY region using uiomove(), advance boundary
|
|
* between FREE and READY. Repeat until transfer is complete.
|
|
*
|
|
* To avoid excessive latency in freeing up space for the DMA
|
|
* engine, transfers are done in blocks of increasing size, so that
|
|
* the latency is proportional to the size of the smallest block, but
|
|
* we have a low overhead and are able to feed the dma engine with
|
|
* large blocks.
|
|
*
|
|
* NOTE: in the current version, read will not return more than
|
|
* blocksize bytes at once (unless more are already available), to
|
|
* avoid that requests using very large buffers block for too long.
|
|
*/
|
|
|
|
int
|
|
chn_read(pcm_channel *c, struct uio *buf)
|
|
{
|
|
int ret = 0, timeout, limit, res;
|
|
long s;
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
if (c->flags & CHN_F_READING) {
|
|
/* This shouldn't happen and is actually silly */
|
|
tsleep(&s, PZERO, "pcmrdR", hz);
|
|
return (EBUSY);
|
|
}
|
|
|
|
s = spltty();
|
|
|
|
/* Store the initial size in the uio. */
|
|
res = buf->uio_resid;
|
|
|
|
c->flags |= CHN_F_READING;
|
|
c->flags &= ~CHN_F_ABORTING;
|
|
limit = buf->uio_resid - c->blocksize;
|
|
if (limit < 0) limit = 0;
|
|
|
|
/* Update the pointers and suck up the DMA and secondary buffers. */
|
|
chn_dmaupdate(c);
|
|
while (chn_rdfeed2nd(c, buf) > 0);
|
|
|
|
/* Start capturing if not yet. */
|
|
if ((!bs->rl || !b->rl) && !b->dl) chn_rdintr(c);
|
|
|
|
if (!(c->flags & CHN_F_NBIO)) {
|
|
/* Wait until all samples are captured. */
|
|
while (buf->uio_resid > 0) {
|
|
/* Suck up the DMA and secondary buffers. */
|
|
chn_dmaupdate(c);
|
|
while (chn_rdfeed2nd(c, buf) > 0);
|
|
|
|
/* Start capturing if necessary. */
|
|
if ((!bs->rl || !b->rl) && !b->dl) chn_rdintr(c);
|
|
|
|
/* Have we finished to feed the uio? */
|
|
if (buf->uio_resid == 0)
|
|
break;
|
|
|
|
/* Wait for new pcm samples. */
|
|
splx(s);
|
|
timeout = (buf->uio_resid - limit >= b->dl)? hz / 20 : 1;
|
|
ret = tsleep(b, PRIBIO | PCATCH, "pcmrd", timeout);
|
|
s = spltty();
|
|
if (ret == EINTR) chn_abort(c);
|
|
if (ret == EINTR || ret == ERESTART) break;
|
|
}
|
|
} else {
|
|
/* If no pcm data was read on nonblocking, return EAGAIN. */
|
|
if (buf->uio_resid == res)
|
|
ret = EAGAIN;
|
|
}
|
|
c->flags &= ~CHN_F_READING;
|
|
splx(s);
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
chn_intr(pcm_channel *c)
|
|
{
|
|
if (c->direction == PCMDIR_PLAY) chn_wrintr(c); else chn_rdintr(c);
|
|
}
|
|
|
|
static void
|
|
chn_dma_setmap(void *arg, bus_dma_segment_t *segs, int nseg, int error)
|
|
{
|
|
snd_dbuf *b = (snd_dbuf *)arg;
|
|
|
|
if (bootverbose) {
|
|
printf("pcm: setmap %lx, %lx; ", (unsigned long)segs->ds_addr,
|
|
(unsigned long)segs->ds_len);
|
|
printf("%p -> %lx\n", b->buf, (unsigned long)vtophys(b->buf));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Allocate memory for DMA buffer. If the device do not perform DMA transfer,
|
|
* the drvier can call malloc(9) by its own.
|
|
*/
|
|
int
|
|
chn_allocbuf(snd_dbuf *b, bus_dma_tag_t parent_dmat)
|
|
{
|
|
if (bus_dmamem_alloc(parent_dmat, (void **)&b->buf,
|
|
BUS_DMA_NOWAIT, &b->dmamap)) return -1;
|
|
if (bus_dmamap_load(parent_dmat, b->dmamap, b->buf,
|
|
b->bufsize, chn_dma_setmap, b, 0)) return -1;
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
chn_clearbuf(pcm_channel *c, snd_dbuf *b, int length)
|
|
{
|
|
int i;
|
|
u_int16_t data, *p;
|
|
|
|
/* rely on length & DMA_ALIGN_MASK == 0 */
|
|
length&=DMA_ALIGN_MASK;
|
|
if (c->hwfmt & AFMT_SIGNED) data = 0x00; else data = 0x80;
|
|
if (c->hwfmt & AFMT_16BIT) data <<= 8; else data |= data << 8;
|
|
if (c->hwfmt & AFMT_BIGENDIAN)
|
|
data = ((data >> 8) & 0x00ff) | ((data << 8) & 0xff00);
|
|
for (i = b->fp, p=(u_int16_t*)(b->buf+b->fp) ; i < b->bufsize && length; i += 2, length-=2)
|
|
*p++ = data;
|
|
for (i = 0, p=(u_int16_t*)b->buf; i < b->bufsize && length; i += 2, length-=2)
|
|
*p++ = data;
|
|
|
|
return;
|
|
}
|
|
|
|
void
|
|
chn_resetbuf(pcm_channel *c)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
c->smegcnt = 0;
|
|
|
|
b->sample_size = 1;
|
|
b->sample_size <<= (c->hwfmt & AFMT_STEREO)? 1 : 0;
|
|
b->sample_size <<= (c->hwfmt & AFMT_16BIT)? 1 : 0;
|
|
|
|
b->rp = b->fp = 0;
|
|
b->dl = b->rl = 0;
|
|
b->fl = b->bufsize;
|
|
b->prev_total = b->total = 0;
|
|
b->prev_int_count = b->int_count = 0;
|
|
b->first_poll = 1;
|
|
b->underflow = 0;
|
|
chn_clearbuf(c, b, b->bufsize);
|
|
|
|
bs->rp = bs->fp = 0;
|
|
bs->dl = bs->rl = 0;
|
|
bs->fl = bs->bufsize;
|
|
bs->prev_total = bs->total = 0;
|
|
b->prev_int_count = b->int_count = 0;
|
|
b->first_poll = 1;
|
|
b->underflow = 0;
|
|
chn_clearbuf(c, bs, bs->bufsize);
|
|
}
|
|
|
|
void
|
|
buf_isadma(snd_dbuf *b, int go)
|
|
{
|
|
if (ISA_DMA(b)) {
|
|
switch (go) {
|
|
case PCMTRIG_START:
|
|
DEB(printf("buf 0x%p ISA DMA started\n", b));
|
|
isa_dmastart(b->dir | ISADMA_RAW, b->buf,
|
|
b->bufsize, b->chan);
|
|
break;
|
|
case PCMTRIG_STOP:
|
|
case PCMTRIG_ABORT:
|
|
DEB(printf("buf 0x%p ISA DMA stopped\n", b));
|
|
isa_dmastop(b->chan);
|
|
isa_dmadone(b->dir | ISADMA_RAW, b->buf, b->bufsize,
|
|
b->chan);
|
|
break;
|
|
}
|
|
} else KASSERT(1, ("buf_isadma called on invalid channel"));
|
|
}
|
|
|
|
int
|
|
buf_isadmaptr(snd_dbuf *b)
|
|
{
|
|
if (ISA_DMA(b)) {
|
|
int i = b->dl? isa_dmastatus(b->chan) : b->bufsize;
|
|
if (i < 0) i = 0;
|
|
return b->bufsize - i;
|
|
} else KASSERT(1, ("buf_isadmaptr called on invalid channel"));
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
* chn_sync waits until the space in the given channel goes above
|
|
* a threshold. The threshold is checked against fl or rl respectively.
|
|
* Assume that the condition can become true, do not check here...
|
|
*/
|
|
int
|
|
chn_sync(pcm_channel *c, int threshold)
|
|
{
|
|
u_long s, rdy;
|
|
int ret;
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
for (;;) {
|
|
s = spltty();
|
|
chn_checkunderflow(c);
|
|
while (chn_wrfeed(c) > 0);
|
|
rdy = (c->direction == PCMDIR_PLAY)? bs->fl : bs->rl;
|
|
if (rdy <= threshold) {
|
|
ret = tsleep((caddr_t)b, PRIBIO | PCATCH, "pcmsyn", 1);
|
|
splx(s);
|
|
if (ret == ERESTART || ret == EINTR) {
|
|
DEB(printf("chn_sync: tsleep returns %d\n", ret));
|
|
return -1;
|
|
}
|
|
} else break;
|
|
}
|
|
splx(s);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_poll(pcm_channel *c, int ev, struct proc *p)
|
|
{
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
u_long s;
|
|
int ret;
|
|
|
|
s = spltty();
|
|
if (c->direction == PCMDIR_PLAY) {
|
|
/* Fill up the DMA buffer. */
|
|
chn_checkunderflow(c);
|
|
while(chn_wrfeed(c) > 0);
|
|
if (!b->dl) chn_wrintr(c);
|
|
} else {
|
|
/* Suck up the DMA buffer. */
|
|
chn_dmaupdate(c);
|
|
while(chn_rdfeed(c) > 0);
|
|
if (!b->dl) chn_rdintr(c);
|
|
}
|
|
ret = 0;
|
|
if (chn_polltrigger(c) && chn_pollreset(c))
|
|
ret = ev;
|
|
else {
|
|
selrecord(p, &bs->sel);
|
|
ret = 0;
|
|
}
|
|
splx(s);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* chn_abort is a non-blocking function which aborts a pending
|
|
* DMA transfer and flushes the buffers.
|
|
* It returns the number of bytes that have not been transferred.
|
|
*/
|
|
int
|
|
chn_abort(pcm_channel *c)
|
|
{
|
|
int missing = 0, cnt = 0;
|
|
snd_dbuf *b = &c->buffer;
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
if (!b->dl) return 0;
|
|
c->flags |= CHN_F_ABORTING;
|
|
while (!b->underflow && (b->dl > 0) && (cnt < 20)) {
|
|
tsleep((caddr_t)b, PRIBIO, "pcmabr", hz / 20);
|
|
cnt++;
|
|
}
|
|
chn_trigger(c, PCMTRIG_ABORT);
|
|
b->dl = 0;
|
|
if (c->direction == PCMDIR_PLAY)
|
|
chn_checkunderflow(c);
|
|
else
|
|
chn_dmaupdate(c);
|
|
missing = bs->rl;
|
|
return missing;
|
|
}
|
|
|
|
/*
|
|
* this routine tries to flush the dma transfer. It is called
|
|
* on a close. We immediately abort any read DMA
|
|
* operation, and then wait for the play buffer to drain.
|
|
*/
|
|
|
|
int
|
|
chn_flush(pcm_channel *c)
|
|
{
|
|
int ret, count = 50, s;
|
|
snd_dbuf *b = &c->buffer;
|
|
|
|
DEB(printf("chn_flush c->flags 0x%08x\n", c->flags));
|
|
c->flags |= CHN_F_CLOSING;
|
|
if (c->direction == PCMDIR_REC)
|
|
chn_abort(c);
|
|
else if (b->dl) {
|
|
while ((b->rl > 0) && !b->underflow && (count-- > 0)) {
|
|
/* still pending output data. */
|
|
ret = tsleep((caddr_t)b, PRIBIO | PCATCH, "pcmflu", hz / 10);
|
|
s = spltty();
|
|
chn_dmaupdate(c);
|
|
splx(s);
|
|
DEB(printf("chn_flush: now rl = %d, fl = %d\n", b->rl, b->fl));
|
|
if (ret == EINTR || ret == ERESTART) {
|
|
DEB(printf("chn_flush: tsleep returns %d\n", ret));
|
|
return ret;
|
|
}
|
|
}
|
|
}
|
|
if (count == 0)
|
|
DEB(printf("chn_flush: timeout flushing dbuf_out, cnt 0x%x flags 0x%x\n", b->rl, c->flags));
|
|
c->flags &= ~CHN_F_CLOSING;
|
|
if (c->direction == PCMDIR_PLAY) chn_abort(c);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_reset(pcm_channel *c)
|
|
{
|
|
chn_abort(c);
|
|
c->flags &= CHN_F_RESET;
|
|
chn_resetbuf(c);
|
|
c->flags |= CHN_F_INIT;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_reinit(pcm_channel *c)
|
|
{
|
|
if ((c->flags & CHN_F_INIT) && CANCHANGE(c)) {
|
|
chn_setformat(c, c->format);
|
|
chn_setspeed(c, c->speed);
|
|
chn_setvolume(c, (c->volume >> 8) & 0xff, c->volume & 0xff);
|
|
c->flags &= ~CHN_F_INIT;
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_init(pcm_channel *c, void *devinfo, int dir)
|
|
{
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
|
|
/* Initialize the hardware and DMA buffer first. */
|
|
c->flags = 0;
|
|
c->feeder = &feeder_root;
|
|
c->buffer.chan = -1;
|
|
c->devinfo = c->init(devinfo, &c->buffer, c, dir);
|
|
if (c->devinfo == NULL)
|
|
return 1;
|
|
chn_setdir(c, dir);
|
|
|
|
/* And the secondary buffer. */
|
|
c->blocksize2nd = CHN_2NDBUFBLKSIZE;
|
|
c->fragments = CHN_2NDBUFBLKNUM;
|
|
bs->bufsize = c->blocksize2nd * c->fragments;
|
|
bs->buf = malloc(bs->bufsize, M_DEVBUF, M_NOWAIT);
|
|
if (bs->buf == NULL)
|
|
return 1;
|
|
bzero(bs->buf, bs->bufsize);
|
|
bs->rl = bs->rp = bs->fp = 0;
|
|
bs->fl = bs->bufsize;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_setdir(pcm_channel *c, int dir)
|
|
{
|
|
int r;
|
|
|
|
c->direction = dir;
|
|
r = c->setdir(c->devinfo, c->direction);
|
|
if (!r && ISA_DMA(&c->buffer))
|
|
c->buffer.dir = (dir == PCMDIR_PLAY)? ISADMA_WRITE : ISADMA_READ;
|
|
return r;
|
|
}
|
|
|
|
int
|
|
chn_setvolume(pcm_channel *c, int left, int right)
|
|
{
|
|
/* could add a feeder for volume changing if channel returns -1 */
|
|
if (CANCHANGE(c)) {
|
|
return -1;
|
|
}
|
|
c->volume = (left << 8) | right;
|
|
c->flags |= CHN_F_INIT;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_setspeed(pcm_channel *c, int speed)
|
|
{
|
|
/* could add a feeder for rate conversion */
|
|
if (CANCHANGE(c)) {
|
|
c->speed = c->setspeed(c->devinfo, speed);
|
|
return c->speed;
|
|
}
|
|
c->speed = speed;
|
|
c->flags |= CHN_F_INIT;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
chn_setformat(pcm_channel *c, u_int32_t fmt)
|
|
{
|
|
if (CANCHANGE(c)) {
|
|
c->hwfmt = c->format = fmt;
|
|
c->hwfmt = chn_feedchain(c);
|
|
chn_resetbuf(c);
|
|
c->setformat(c->devinfo, c->hwfmt);
|
|
return fmt;
|
|
}
|
|
c->format = fmt;
|
|
c->flags |= CHN_F_INIT;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* The seconday buffer is modified only during interrupt.
|
|
* Hence the size of the secondary buffer can be changed
|
|
* at any time as long as an interrupt is disabled.
|
|
*/
|
|
int
|
|
chn_setblocksize(pcm_channel *c, int blksz)
|
|
{
|
|
snd_dbuf *bs = &c->buffer2nd;
|
|
u_int8_t *tmpbuf;
|
|
int s, tmpbuf_fl, tmpbuf_fp, l;
|
|
|
|
c->flags &= ~CHN_F_HAS_SIZE;
|
|
if (blksz >= 2) c->flags |= CHN_F_HAS_SIZE;
|
|
if (blksz < 0) blksz = -blksz;
|
|
if (blksz < 2) blksz = c->buffer.sample_size * (c->speed >> 2);
|
|
/* XXX How small can the lower bound be? */
|
|
RANGE(blksz, 64, CHN_2NDBUFWHOLESIZE);
|
|
|
|
/*
|
|
* Allocate a temporary buffer. It holds the pcm data
|
|
* until the size of the secondary buffer gets changed.
|
|
* bs->buf is not affected, so mmap should work fine.
|
|
*/
|
|
tmpbuf = malloc(blksz, M_TEMP, M_NOWAIT);
|
|
if (tmpbuf == NULL) {
|
|
DEB(printf("chn_setblocksize: out of memory."));
|
|
return 1;
|
|
}
|
|
bzero(tmpbuf, blksz);
|
|
tmpbuf_fl = blksz;
|
|
tmpbuf_fp = 0;
|
|
s = spltty();
|
|
while (bs->rl > 0 && tmpbuf_fl > 0) {
|
|
l = min(min(bs->rl, bs->bufsize - bs->rp), tmpbuf_fl);
|
|
bcopy(bs->buf + bs->rp, tmpbuf + tmpbuf_fp, l);
|
|
tmpbuf_fl -= l;
|
|
tmpbuf_fp = (tmpbuf_fp + l) % blksz;
|
|
bs->rl -= l;
|
|
bs->fl += l;
|
|
bs->rp = (bs->rp + l) % bs->bufsize;
|
|
}
|
|
/* Change the size of the seconary buffer. */
|
|
bs->bufsize = blksz;
|
|
c->fragments = CHN_2NDBUFBLKNUM;
|
|
c->blocksize2nd = bs->bufsize / c->fragments;
|
|
/* Clear the secondary buffer and restore the pcm data. */
|
|
bzero(bs->buf, bs->bufsize);
|
|
bs->rl = bs->bufsize - tmpbuf_fl;
|
|
bs->rp = 0;
|
|
bs->fl = tmpbuf_fl;
|
|
bs->fp = tmpbuf_fp;
|
|
bcopy(tmpbuf, bs->buf, bs->rl);
|
|
|
|
free(tmpbuf, M_TEMP);
|
|
splx(s);
|
|
return c->blocksize2nd;
|
|
}
|
|
|
|
int
|
|
chn_trigger(pcm_channel *c, int go)
|
|
{
|
|
return c->trigger(c->devinfo, go);
|
|
}
|
|
|
|
int
|
|
chn_getptr(pcm_channel *c)
|
|
{
|
|
int hwptr;
|
|
int a = (1 << c->align) - 1;
|
|
|
|
hwptr=c->getptr(c->devinfo);
|
|
/* don't allow unaligned values in the hwa ptr */
|
|
hwptr &= ~a ; /* Apply channel align mask */
|
|
hwptr &= DMA_ALIGN_MASK; /* Apply DMA align mask */
|
|
return hwptr;
|
|
}
|
|
|
|
pcmchan_caps *
|
|
chn_getcaps(pcm_channel *c)
|
|
{
|
|
return c->getcaps(c->devinfo);
|
|
}
|