b09219dff2
commit. 1) sys/dev/sound/pcm/sound.h sys/dev/sound/pcm/channel.c * Be more specific: SD_F_SOFTVOL -> SD_F_SOFTPCMVOL 2) sys/dev/sound/pcm/mixer.[ch] * Implement mix_setparentchild() mix_setrealdev() mix_getparent() mix_getchild() The purpose of these functions is implement relative volume adjustment, such as to tie two or more mixer device into a single logical device. Usefull for the upcoming HDA driver and few AC97 codec (such as AD1981B) where the master volume "vol" need to be implemented using this logical manner. 3) sys/dev/sound/pcm/ac97_patch.[ch] * Patch for AD1981B codec to enable (automuting) headphone jack sense. 4) sys/dev/sound/pcm/ac97.c * Implement proper logical master volume for AD9181B codec through various mix_set{parentchild,realdev}(). Tie both "ogain" (headphone volume) and "phone" (speaker/lineout) to a logical "vol". 5) sys/dev/sound/pcm/usb/uaudio_pcm.c * ditto, for "vol" -> { "pcm" }. MFC after: 1 month
422 lines
9.5 KiB
C
422 lines
9.5 KiB
C
/* $FreeBSD$ */
|
|
|
|
/*-
|
|
* Copyright (c) 2000-2002 Hiroyuki Aizu <aizu@navi.org>
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
|
|
#include <sys/soundcard.h>
|
|
#include <dev/sound/pcm/sound.h>
|
|
#include <dev/sound/chip.h>
|
|
|
|
#include <dev/sound/usb/uaudio.h>
|
|
|
|
#include "mixer_if.h"
|
|
|
|
struct ua_info;
|
|
|
|
struct ua_chinfo {
|
|
struct ua_info *parent;
|
|
struct pcm_channel *channel;
|
|
struct snd_dbuf *buffer;
|
|
u_char *buf;
|
|
int dir, hwch;
|
|
u_int32_t fmt, spd, blksz; /* XXXXX */
|
|
};
|
|
|
|
struct ua_info {
|
|
device_t sc_dev;
|
|
u_int32_t bufsz;
|
|
struct ua_chinfo pch, rch;
|
|
#define FORMAT_NUM 32
|
|
u_int32_t ua_playfmt[FORMAT_NUM*2+1]; /* FORMAT_NUM format * (stereo or mono) + endptr */
|
|
u_int32_t ua_recfmt[FORMAT_NUM*2+1]; /* FORMAT_NUM format * (stereo or mono) + endptr */
|
|
struct pcmchan_caps ua_playcaps;
|
|
struct pcmchan_caps ua_reccaps;
|
|
};
|
|
|
|
#define UAUDIO_DEFAULT_BUFSZ 16*1024
|
|
|
|
/************************************************************/
|
|
static void *
|
|
ua_chan_init(kobj_t obj, void *devinfo, struct snd_dbuf *b, struct pcm_channel *c, int dir)
|
|
{
|
|
device_t pa_dev;
|
|
|
|
struct ua_info *sc = devinfo;
|
|
struct ua_chinfo *ch = (dir == PCMDIR_PLAY)? &sc->pch : &sc->rch;
|
|
|
|
ch->parent = sc;
|
|
ch->channel = c;
|
|
ch->buffer = b;
|
|
ch->dir = dir;
|
|
|
|
pa_dev = device_get_parent(sc->sc_dev);
|
|
|
|
ch->buf = malloc(sc->bufsz, M_DEVBUF, M_NOWAIT);
|
|
if (ch->buf == NULL)
|
|
return NULL;
|
|
if (sndbuf_setup(b, ch->buf, sc->bufsz) != 0) {
|
|
free(ch->buf, M_DEVBUF);
|
|
return NULL;
|
|
}
|
|
uaudio_chan_set_param_pcm_dma_buff(pa_dev, ch->buf, ch->buf+sc->bufsz, ch->channel, dir);
|
|
if (bootverbose)
|
|
device_printf(pa_dev, "%s buf %p\n", (dir == PCMDIR_PLAY)?
|
|
"play" : "rec", sndbuf_getbuf(ch->buffer));
|
|
|
|
ch->dir = dir;
|
|
#ifndef NO_RECORDING
|
|
ch->hwch = 1;
|
|
if (dir == PCMDIR_PLAY)
|
|
ch->hwch = 2;
|
|
#else
|
|
ch->hwch = 2;
|
|
#endif
|
|
|
|
return ch;
|
|
}
|
|
|
|
static int
|
|
ua_chan_free(kobj_t obj, void *data)
|
|
{
|
|
struct ua_chinfo *ua = data;
|
|
|
|
if (ua->buf != NULL)
|
|
free(ua->buf, M_DEVBUF);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ua_chan_setformat(kobj_t obj, void *data, u_int32_t format)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_info *ua;
|
|
|
|
struct ua_chinfo *ch = data;
|
|
|
|
/*
|
|
* At this point, no need to query as we shouldn't select an unsorted format
|
|
*/
|
|
ua = ch->parent;
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
uaudio_chan_set_param_format(pa_dev, format, ch->dir);
|
|
|
|
ch->fmt = format;
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ua_chan_setspeed(kobj_t obj, void *data, u_int32_t speed)
|
|
{
|
|
struct ua_chinfo *ch;
|
|
device_t pa_dev;
|
|
int bestspeed;
|
|
|
|
ch = data;
|
|
pa_dev = device_get_parent(ch->parent->sc_dev);
|
|
|
|
if ((bestspeed = uaudio_chan_set_param_speed(pa_dev, speed, ch->dir)))
|
|
ch->spd = bestspeed;
|
|
|
|
return ch->spd;
|
|
}
|
|
|
|
static int
|
|
ua_chan_setblocksize(kobj_t obj, void *data, u_int32_t blocksize)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_chinfo *ch = data;
|
|
struct ua_info *ua = ch->parent;
|
|
|
|
if (blocksize) {
|
|
RANGE(blocksize, 128, ua->bufsz / 2);
|
|
if (sndbuf_resize(ch->buffer, ua->bufsz/blocksize, blocksize) != 0) {
|
|
ch->blksz = blocksize;
|
|
}
|
|
}
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
uaudio_chan_set_param_blocksize(pa_dev, blocksize, ch->dir);
|
|
|
|
return ch->blksz;
|
|
}
|
|
|
|
static int
|
|
ua_chan_trigger(kobj_t obj, void *data, int go)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_info *ua;
|
|
struct ua_chinfo *ch = data;
|
|
|
|
if (go == PCMTRIG_EMLDMAWR || go == PCMTRIG_EMLDMARD)
|
|
return 0;
|
|
|
|
ua = ch->parent;
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
|
|
/* XXXXX */
|
|
if (ch->dir == PCMDIR_PLAY) {
|
|
if (go == PCMTRIG_START) {
|
|
uaudio_trigger_output(pa_dev);
|
|
} else {
|
|
uaudio_halt_out_dma(pa_dev);
|
|
}
|
|
} else {
|
|
#ifndef NO_RECORDING
|
|
if (go == PCMTRIG_START)
|
|
uaudio_trigger_input(pa_dev);
|
|
else
|
|
uaudio_halt_in_dma(pa_dev);
|
|
#endif
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ua_chan_getptr(kobj_t obj, void *data)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_info *ua;
|
|
struct ua_chinfo *ch = data;
|
|
|
|
ua = ch->parent;
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
|
|
return uaudio_chan_getptr(pa_dev, ch->dir);
|
|
}
|
|
|
|
static struct pcmchan_caps *
|
|
ua_chan_getcaps(kobj_t obj, void *data)
|
|
{
|
|
struct ua_chinfo *ch;
|
|
|
|
ch = data;
|
|
return (ch->dir == PCMDIR_PLAY) ? &(ch->parent->ua_playcaps) : &(ch->parent->ua_reccaps);
|
|
}
|
|
|
|
static kobj_method_t ua_chan_methods[] = {
|
|
KOBJMETHOD(channel_init, ua_chan_init),
|
|
KOBJMETHOD(channel_free, ua_chan_free),
|
|
KOBJMETHOD(channel_setformat, ua_chan_setformat),
|
|
KOBJMETHOD(channel_setspeed, ua_chan_setspeed),
|
|
KOBJMETHOD(channel_setblocksize, ua_chan_setblocksize),
|
|
KOBJMETHOD(channel_trigger, ua_chan_trigger),
|
|
KOBJMETHOD(channel_getptr, ua_chan_getptr),
|
|
KOBJMETHOD(channel_getcaps, ua_chan_getcaps),
|
|
{ 0, 0 }
|
|
};
|
|
|
|
CHANNEL_DECLARE(ua_chan);
|
|
|
|
/************************************************************/
|
|
static int
|
|
ua_mixer_init(struct snd_mixer *m)
|
|
{
|
|
u_int32_t mask;
|
|
device_t pa_dev;
|
|
struct snddev_info *d;
|
|
struct ua_info *ua = mix_getdevinfo(m);
|
|
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
d = device_get_softc(ua->sc_dev);
|
|
|
|
mask = uaudio_query_mix_info(pa_dev);
|
|
if (d != NULL) {
|
|
if (!(mask & SOUND_MASK_PCM)) {
|
|
/*
|
|
* Emulate missing pcm mixer controller
|
|
* through FEEDER_VOLUME
|
|
*/
|
|
d->flags |= SD_F_SOFTPCMVOL;
|
|
}
|
|
if (!(mask & SOUND_MASK_VOLUME)) {
|
|
mix_setparentchild(m, SOUND_MIXER_VOLUME,
|
|
SOUND_MASK_PCM);
|
|
mix_setrealdev(m, SOUND_MIXER_VOLUME,
|
|
SOUND_MIXER_NONE);
|
|
}
|
|
}
|
|
mix_setdevs(m, mask);
|
|
|
|
mask = uaudio_query_recsrc_info(pa_dev);
|
|
mix_setrecdevs(m, mask);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ua_mixer_set(struct snd_mixer *m, unsigned type, unsigned left, unsigned right)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_info *ua = mix_getdevinfo(m);
|
|
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
uaudio_mixer_set(pa_dev, type, left, right);
|
|
|
|
return left | (right << 8);
|
|
}
|
|
|
|
static int
|
|
ua_mixer_setrecsrc(struct snd_mixer *m, u_int32_t src)
|
|
{
|
|
device_t pa_dev;
|
|
struct ua_info *ua = mix_getdevinfo(m);
|
|
|
|
pa_dev = device_get_parent(ua->sc_dev);
|
|
return uaudio_mixer_setrecsrc(pa_dev, src);
|
|
}
|
|
|
|
static kobj_method_t ua_mixer_methods[] = {
|
|
KOBJMETHOD(mixer_init, ua_mixer_init),
|
|
KOBJMETHOD(mixer_set, ua_mixer_set),
|
|
KOBJMETHOD(mixer_setrecsrc, ua_mixer_setrecsrc),
|
|
|
|
{ 0, 0 }
|
|
};
|
|
MIXER_DECLARE(ua_mixer);
|
|
/************************************************************/
|
|
|
|
|
|
static int
|
|
ua_probe(device_t dev)
|
|
{
|
|
char *s;
|
|
struct sndcard_func *func;
|
|
|
|
/* The parent device has already been probed. */
|
|
|
|
func = device_get_ivars(dev);
|
|
if (func == NULL || func->func != SCF_PCM)
|
|
return (ENXIO);
|
|
|
|
s = "USB Audio";
|
|
|
|
device_set_desc(dev, s);
|
|
return BUS_PROBE_DEFAULT;
|
|
}
|
|
|
|
static int
|
|
ua_attach(device_t dev)
|
|
{
|
|
struct ua_info *ua;
|
|
char status[SND_STATUSLEN];
|
|
device_t pa_dev;
|
|
u_int32_t nplay, nrec;
|
|
int i;
|
|
|
|
ua = (struct ua_info *)malloc(sizeof *ua, M_DEVBUF, M_ZERO | M_NOWAIT);
|
|
if (ua == NULL)
|
|
return ENXIO;
|
|
|
|
ua->sc_dev = dev;
|
|
|
|
pa_dev = device_get_parent(dev);
|
|
|
|
ua->bufsz = pcm_getbuffersize(dev, 4096, UAUDIO_DEFAULT_BUFSZ, 65536);
|
|
if (bootverbose)
|
|
device_printf(dev, "using a default buffer size of %jd\n", (intmax_t)ua->bufsz);
|
|
|
|
if (mixer_init(dev, &ua_mixer_class, ua)) {
|
|
goto bad;
|
|
}
|
|
|
|
snprintf(status, SND_STATUSLEN, "at ? %s", PCM_KLDSTRING(snd_uaudio));
|
|
|
|
ua->ua_playcaps.fmtlist = ua->ua_playfmt;
|
|
ua->ua_reccaps.fmtlist = ua->ua_recfmt;
|
|
nplay = uaudio_query_formats(pa_dev, PCMDIR_PLAY, FORMAT_NUM * 2, &ua->ua_playcaps);
|
|
nrec = uaudio_query_formats(pa_dev, PCMDIR_REC, FORMAT_NUM * 2, &ua->ua_reccaps);
|
|
|
|
if (nplay > 1)
|
|
nplay = 1;
|
|
if (nrec > 1)
|
|
nrec = 1;
|
|
|
|
#ifndef NO_RECORDING
|
|
if (pcm_register(dev, ua, nplay, nrec)) {
|
|
#else
|
|
if (pcm_register(dev, ua, nplay, 0)) {
|
|
#endif
|
|
goto bad;
|
|
}
|
|
|
|
sndstat_unregister(dev);
|
|
uaudio_sndstat_register(dev);
|
|
|
|
for (i = 0; i < nplay; i++) {
|
|
pcm_addchan(dev, PCMDIR_PLAY, &ua_chan_class, ua);
|
|
}
|
|
#ifndef NO_RECORDING
|
|
for (i = 0; i < nrec; i++) {
|
|
pcm_addchan(dev, PCMDIR_REC, &ua_chan_class, ua);
|
|
}
|
|
#endif
|
|
pcm_setstatus(dev, status);
|
|
|
|
return 0;
|
|
|
|
bad: free(ua, M_DEVBUF);
|
|
return ENXIO;
|
|
}
|
|
|
|
static int
|
|
ua_detach(device_t dev)
|
|
{
|
|
int r;
|
|
struct ua_info *sc;
|
|
|
|
r = pcm_unregister(dev);
|
|
if (r)
|
|
return r;
|
|
|
|
sc = pcm_getdevinfo(dev);
|
|
free(sc, M_DEVBUF);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/************************************************************/
|
|
|
|
static device_method_t ua_pcm_methods[] = {
|
|
/* Device interface */
|
|
DEVMETHOD(device_probe, ua_probe),
|
|
DEVMETHOD(device_attach, ua_attach),
|
|
DEVMETHOD(device_detach, ua_detach),
|
|
|
|
{ 0, 0 }
|
|
};
|
|
|
|
static driver_t ua_pcm_driver = {
|
|
"pcm",
|
|
ua_pcm_methods,
|
|
PCM_SOFTC_SIZE,
|
|
};
|
|
|
|
|
|
DRIVER_MODULE(ua_pcm, uaudio, ua_pcm_driver, pcm_devclass, 0, 0);
|
|
MODULE_DEPEND(ua_pcm, uaudio, 1, 1, 1);
|
|
MODULE_DEPEND(ua_pcm, sound, SOUND_MINVER, SOUND_PREFVER, SOUND_MAXVER);
|
|
MODULE_VERSION(ua_pcm, 1);
|