summaryrefslogtreecommitdiff
path: root/sys/dev/isa
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/isa')
-rw-r--r--sys/dev/isa/ad1848.c533
-rw-r--r--sys/dev/isa/ad1848var.h90
-rw-r--r--sys/dev/isa/gus.c330
-rw-r--r--sys/dev/isa/pss.c207
-rw-r--r--sys/dev/isa/wss.c212
-rw-r--r--sys/dev/isa/ym.c470
-rw-r--r--sys/dev/isa/ym_isapnp.c126
-rw-r--r--sys/dev/isa/ymvar.h89
8 files changed, 1105 insertions, 952 deletions
diff --git a/sys/dev/isa/ad1848.c b/sys/dev/isa/ad1848.c
index 2e3e495d4cf..c421a510bee 100644
--- a/sys/dev/isa/ad1848.c
+++ b/sys/dev/isa/ad1848.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ad1848.c,v 1.10 1998/04/26 21:02:38 provos Exp $ */
+/* $OpenBSD: ad1848.c,v 1.11 1998/05/08 18:37:18 csapuntz Exp $ */
/* $NetBSD: ad1848.c,v 1.45 1998/01/30 02:02:38 augustss Exp $ */
/*
@@ -138,8 +138,8 @@ static int ad1848_init_values[] = {
0, /* unused */
0, /* IRQ status */
0, /* unused */
- /* Mono input (mic) Control */
- MONO_INPUT_MUTE|ATTEN_6, /* mute mic by default */
+ /* Mono input (a.k.a speaker) (mic) Control */
+ MONO_INPUT_MUTE|ATTEN_6, /* mute speaker by default */
0, /* unused */
0, /* record format */
0, /* Crystal Clock Select */
@@ -566,16 +566,16 @@ ad1848_attach(sc)
/* Set default gains */
(void) ad1848_set_rec_gain(sc, &vol_mid);
- (void) ad1848_set_out_gain(sc, &vol_mid);
- (void) ad1848_set_mon_gain(sc, &vol_0);
- (void) ad1848_set_aux1_gain(sc, &vol_mid); /* CD volume */
+ (void) ad1848_set_channel_gain(sc, AD1848_DAC_CHANNEL, &vol_mid);
+ (void) ad1848_set_channel_gain(sc, AD1848_MONITOR_CHANNEL, &vol_0);
+ (void) ad1848_set_channel_gain(sc, AD1848_AUX1_CHANNEL, &vol_mid); /* CD volume */
if (sc->mode == 2) {
- /* aux1 was really the DAC output */
- (void) ad1848_set_aux2_gain(sc, &vol_mid); /* CD volume */
- (void) cs4231_set_linein_gain(sc, &vol_mid);
- (void) cs4231_set_mono_gain(sc, &vol_0); /* mic */
+ (void) ad1848_set_channel_gain(sc, AD1848_AUX2_CHANNEL, &vol_mid); /* CD volume */
+ (void) ad1848_set_channel_gain(sc, AD1848_LINE_CHANNEL, &vol_mid);
+ (void) ad1848_set_channel_gain(sc, AD1848_MONO_CHANNEL, &vol_0);
+ sc->mute[AD1848_MONO_CHANNEL] = MUTE_ALL;
} else
- (void) ad1848_set_aux2_gain(sc, &vol_0);
+ (void) ad1848_set_channel_gain(sc, AD1848_AUX2_CHANNEL, &vol_0);
/* Set default port */
(void) ad1848_set_rec_port(sc, MIC_IN_PORT);
@@ -587,128 +587,161 @@ ad1848_attach(sc)
/*
* Various routines to interface to higher level audio driver
*/
-int
-ad1848_set_rec_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
+struct ad1848_mixerinfo {
+ int left_reg;
+ int right_reg;
+ int atten_bits;
+ int atten_mask;
+} mixer_channel_info[] =
+{ { SP_LEFT_AUX2_CONTROL, SP_RIGHT_AUX2_CONTROL, AUX_INPUT_ATTEN_BITS,
+ AUX_INPUT_ATTEN_MASK },
+ { SP_LEFT_AUX1_CONTROL, SP_RIGHT_AUX1_CONTROL, AUX_INPUT_ATTEN_BITS,
+ AUX_INPUT_ATTEN_MASK },
+ { SP_LEFT_OUTPUT_CONTROL, SP_RIGHT_OUTPUT_CONTROL,
+ OUTPUT_ATTEN_BITS, OUTPUT_ATTEN_MASK },
+ { CS_LEFT_LINE_CONTROL, CS_RIGHT_LINE_CONTROL, LINE_INPUT_ATTEN_BITS,
+ LINE_INPUT_ATTEN_MASK },
+ { CS_MONO_IO_CONTROL, 0, MONO_INPUT_ATTEN_BITS, MONO_INPUT_ATTEN_MASK },
+ { SP_DIGITAL_MIX, 0, OUTPUT_ATTEN_BITS, MIX_ATTEN_MASK }
+};
+
+/*
+ * This function doesn't set the mute flags but does use them.
+ * The mute flags reflect the mutes that have been applied by the user.
+ * However, the driver occasionally wants to mute devices (e.g. when chaing
+ * sampling rate). These operations should not affect the mute flags.
+ */
+
+void
+ad1848_mute_channel(sc, device, mute)
+ struct ad1848_softc *sc;
+ int device;
+ int mute;
{
- u_char reg, gain;
-
- DPRINTF(("ad1848_set_rec_gain: %d:%d\n", gp->left, gp->right));
+ u_char reg;
- sc->rec_gain = *gp;
+ reg = ad_read(sc, mixer_channel_info[device].left_reg);
- gain = (gp->left * GAIN_22_5)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_LEFT_INPUT_CONTROL);
- reg &= INPUT_GAIN_MASK;
- ad_write(sc, SP_LEFT_INPUT_CONTROL, (gain&0x0f)|reg);
+ if (mute & MUTE_LEFT) {
+ if (device == AD1848_MONITOR_CHANNEL)
+ ad_write(sc, mixer_channel_info[device].left_reg, reg & 0xFE);
+ else
+ ad_write(sc, mixer_channel_info[device].left_reg, reg | 0x80);
+ } else if (!(sc->mute[device] & MUTE_LEFT)) {
+ if (device == AD1848_MONITOR_CHANNEL)
+ ad_write(sc, mixer_channel_info[device].left_reg, reg | 0x01);
+ else
+ ad_write(sc, mixer_channel_info[device].left_reg, reg & ~0x80);
+ }
- gain = (gp->right * GAIN_22_5)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_RIGHT_INPUT_CONTROL);
- reg &= INPUT_GAIN_MASK;
- ad_write(sc, SP_RIGHT_INPUT_CONTROL, (gain&0x0f)|reg);
+ if (!mixer_channel_info[device].right_reg) {
+ return;
+ }
- return(0);
-}
+ reg = ad_read(sc, mixer_channel_info[device].right_reg);
-int
-ad1848_get_rec_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- *gp = sc->rec_gain;
- return(0);
+ if (mute & MUTE_RIGHT)
+ ad_write(sc, mixer_channel_info[device].right_reg, reg | 0x80);
+ else if (!(sc->mute[device] & MUTE_RIGHT)) {
+ ad_write(sc, mixer_channel_info[device].right_reg, reg & ~0x80);
+ }
}
+
int
-ad1848_set_out_gain(sc, gp)
+ad1848_set_channel_gain(sc, device, gp)
struct ad1848_softc *sc;
+ int device;
struct ad1848_volume *gp;
{
+ struct ad1848_mixerinfo *info = &mixer_channel_info[device];
u_char reg;
u_int atten;
-
- DPRINTF(("ad1848_set_out_gain: %d:%d\n", gp->left, gp->right));
- sc->out_gain = *gp;
+ sc->gains[device] = *gp;
- atten = ((AUDIO_MAX_GAIN - gp->left) * OUTPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_LEFT_OUTPUT_CONTROL);
- reg &= OUTPUT_ATTEN_MASK;
- ad_write(sc, SP_LEFT_OUTPUT_CONTROL, (atten&0x3f)|reg);
+ atten = ((AUDIO_MAX_GAIN - gp->left) * info->atten_bits)/AUDIO_MAX_GAIN;
- atten = ((AUDIO_MAX_GAIN - gp->right) * OUTPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_RIGHT_OUTPUT_CONTROL);
- reg &= OUTPUT_ATTEN_MASK;
- ad_write(sc, SP_RIGHT_OUTPUT_CONTROL, (atten&0x3f)|reg);
+ reg = ad_read(sc, info->left_reg) & (info->atten_mask);
+ if (device == AD1848_MONITOR_CHANNEL)
+ reg |= ((atten & info->atten_bits) << 2);
+ else
+ reg |= ((atten & info->atten_bits));
- return(0);
-}
+ ad_write(sc, info->left_reg, reg);
+
+ if (!info->right_reg)
+ return (0);
+
+ atten = ((AUDIO_MAX_GAIN - gp->right) * info->atten_bits)/AUDIO_MAX_GAIN;
+ reg = ad_read(sc, info->right_reg);
+ reg &= (info->atten_mask);
+ ad_write(sc, info->right_reg, (atten& info->atten_bits)|reg);
-int
-ad1848_get_out_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- *gp = sc->out_gain;
return(0);
}
+
int
-ad1848_set_mon_gain(sc, gp) /* monitor gain */
+ad1848_get_device_gain(sc, device, gp)
struct ad1848_softc *sc;
+ int device;
struct ad1848_volume *gp;
{
- u_char reg;
- u_int atten;
-
- DPRINTF(("ad1848_set_mon_gain: %d\n", gp->left));
-
- sc->mon_gain = *gp;
-
- atten = ((AUDIO_MAX_GAIN - gp->left) * OUTPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_DIGITAL_MIX);
- reg &= MIX_ATTEN_MASK;
- ad_write(sc, SP_DIGITAL_MIX, (atten&OUTPUT_ATTEN_BITS)|reg);
+ *gp = sc->gains[device];
return(0);
}
int
-ad1848_get_mon_gain(sc, gp)
+ad1848_get_rec_gain(sc, gp)
struct ad1848_softc *sc;
struct ad1848_volume *gp;
{
- *gp = sc->mon_gain;
+ *gp = sc->rec_gain;
return(0);
}
int
-cs4231_set_mono_gain(sc, gp)
+ad1848_set_rec_gain(sc, gp)
struct ad1848_softc *sc;
struct ad1848_volume *gp;
{
- u_char reg, oreg;
- u_int atten;
+ u_char reg, gain;
- DPRINTF(("cs4231_set_mono_gain: %d\n", gp->left));
+ DPRINTF(("ad1848_set_rec_gain: %d:%d\n", gp->left, gp->right));
+
+ sc->rec_gain = *gp;
- sc->mono_gain = *gp;
+ gain = (gp->left * GAIN_22_5)/AUDIO_MAX_GAIN;
+ reg = ad_read(sc, SP_LEFT_INPUT_CONTROL);
+ reg &= INPUT_GAIN_MASK;
+ ad_write(sc, SP_LEFT_INPUT_CONTROL, (gain&0x0f)|reg);
+
+ gain = (gp->right * GAIN_22_5)/AUDIO_MAX_GAIN;
+ reg = ad_read(sc, SP_RIGHT_INPUT_CONTROL);
+ reg &= INPUT_GAIN_MASK;
+ ad_write(sc, SP_RIGHT_INPUT_CONTROL, (gain&0x0f)|reg);
- atten = ((AUDIO_MAX_GAIN - gp->left) * MONO_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- oreg = reg = ad_read(sc, CS_MONO_IO_CONTROL);
- reg &= MONO_INPUT_ATTEN_MASK;
- ad_write(sc, CS_MONO_IO_CONTROL, (atten&MONO_INPUT_ATTEN_BITS)|reg);
- DPRINTF(("cs4231_set_mono_gain: was:%x\n", oreg));
return(0);
}
-int
-cs4231_get_mono_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
+
+void
+ad1848_mute_monitor(addr, mute)
+ void *addr;
+ int mute;
{
- *gp = sc->mono_gain;
- return(0);
+ struct ad1848_softc *sc = addr;
+
+ DPRINTF(("ad1848_mute_monitor: %smuting\n", mute ? "" : "un"));
+ if (sc->mode == 2) {
+ ad1848_mute_channel(sc, AD1848_DAC_CHANNEL, mute ? MUTE_ALL : 0);
+ ad1848_mute_channel(sc, AD1848_MONO_CHANNEL, mute ? MUTE_MONO : 0);
+ ad1848_mute_channel(sc, AD1848_LINE_CHANNEL, mute ? MUTE_ALL : 0);
+ }
+
+ ad1848_mute_channel(sc, AD1848_AUX2_CHANNEL, mute ? MUTE_ALL : 0);
+ ad1848_mute_channel(sc, AD1848_AUX1_CHANNEL, mute ? MUTE_ALL : 0);
}
int
@@ -745,228 +778,170 @@ ad1848_get_mic_gain(sc, gp)
return(0);
}
-void
-ad1848_mute_monitor(addr, mute)
- void *addr;
- int mute;
-{
- struct ad1848_softc *sc = addr;
- DPRINTF(("ad1848_mute_monitor: %smuting\n", mute ? "" : "un"));
- if (sc->mode == 2) {
- cs4231_mute_monitor(sc, mute);
- cs4231_mute_mono(sc, mute);
- cs4231_mute_line(sc, mute);
- }
-
- ad1848_mute_aux1(sc, mute);
- ad1848_mute_aux2(sc, mute);
-}
+static ad1848_devmap_t *ad1848_mixer_find_dev __P((ad1848_devmap_t *, int, mixer_ctrl_t *));
-void
-cs4231_mute_monitor(sc, mute)
- struct ad1848_softc *sc;
- int mute;
-{
- u_char reg;
- if (mute) {
- DPRINTF(("cs4231_mute_monitor: muting\n"));
- reg = ad_read(sc, SP_LEFT_OUTPUT_CONTROL);
- ad_write(sc, SP_LEFT_OUTPUT_CONTROL, OUTPUT_MUTE|reg);
- reg = ad_read(sc, SP_RIGHT_OUTPUT_CONTROL);
- ad_write(sc, SP_RIGHT_OUTPUT_CONTROL, OUTPUT_MUTE|reg);
- } else if (!sc->mon_mute) {
- DPRINTF(("cs4231_mute_monitor: unmuting\n"));
- reg = ad_read(sc, SP_LEFT_OUTPUT_CONTROL);
- ad_write(sc, SP_LEFT_OUTPUT_CONTROL, reg & ~OUTPUT_MUTE);
- reg = ad_read(sc, SP_RIGHT_OUTPUT_CONTROL);
- ad_write(sc, SP_RIGHT_OUTPUT_CONTROL, reg & ~OUTPUT_MUTE);
- }
-}
+static ad1848_devmap_t *
+ad1848_mixer_find_dev(map, cnt, cp)
+ ad1848_devmap_t *map;
+ int cnt;
+ mixer_ctrl_t *cp;
-void
-cs4231_mute_mono(sc, mute)
- struct ad1848_softc *sc;
- int mute;
{
- u_char reg;
- if (mute) {
- DPRINTF(("cs4231_mute_mono: muting\n"));
- reg = ad_read(sc, CS_MONO_IO_CONTROL);
- ad_write(sc, CS_MONO_IO_CONTROL, MONO_INPUT_MUTE|reg);
- } else if (!sc->mono_mute) {
- DPRINTF(("cs4231_mute_mono: unmuting\n"));
- reg = ad_read(sc, CS_MONO_IO_CONTROL);
- ad_write(sc, CS_MONO_IO_CONTROL, reg & ~MONO_INPUT_MUTE);
- }
-}
+ int idx;
-void
-cs4231_mute_line(sc, mute)
- struct ad1848_softc *sc;
- int mute;
-{
- u_char reg;
- if (mute) {
- DPRINTF(("cs4231_mute_line: muting\n"));
- reg = ad_read(sc, CS_LEFT_LINE_CONTROL);
- ad_write(sc, CS_LEFT_LINE_CONTROL, LINE_INPUT_MUTE|reg);
- reg = ad_read(sc, CS_RIGHT_LINE_CONTROL);
- ad_write(sc, CS_RIGHT_LINE_CONTROL, LINE_INPUT_MUTE|reg);
- } else if (!sc->line_mute) {
- DPRINTF(("cs4231_mute_line: unmuting\n"));
- reg = ad_read(sc, CS_LEFT_LINE_CONTROL);
- ad_write(sc, CS_LEFT_LINE_CONTROL, reg & ~LINE_INPUT_MUTE);
- reg = ad_read(sc, CS_RIGHT_LINE_CONTROL);
- ad_write(sc, CS_RIGHT_LINE_CONTROL, reg & ~LINE_INPUT_MUTE);
- }
+ for (idx = 0; idx < cnt; idx++) {
+ if (map[idx].id == cp->dev) {
+ return (&map[idx]);
+ }
+ }
+ return (NULL);
}
-void
-ad1848_mute_aux1(sc, mute)
- struct ad1848_softc *sc;
- int mute;
+int
+ad1848_mixer_get_port(ac, map, cnt, cp)
+ struct ad1848_softc *ac;
+ struct ad1848_devmap *map;
+ int cnt;
+ mixer_ctrl_t *cp;
{
- u_char reg;
- if (mute) {
- DPRINTF(("ad1848_mute_aux1: muting\n"));
- reg = ad_read(sc, SP_LEFT_AUX1_CONTROL);
- ad_write(sc, SP_LEFT_AUX1_CONTROL, AUX_INPUT_MUTE|reg);
- reg = ad_read(sc, SP_RIGHT_AUX1_CONTROL);
- ad_write(sc, SP_RIGHT_AUX1_CONTROL, AUX_INPUT_MUTE|reg);
- } else if (!sc->aux1_mute) {
- DPRINTF(("ad1848_mute_aux1: unmuting\n"));
- reg = ad_read(sc, SP_LEFT_AUX1_CONTROL);
- ad_write(sc, SP_LEFT_AUX1_CONTROL, reg & ~AUX_INPUT_MUTE);
- reg = ad_read(sc, SP_RIGHT_AUX1_CONTROL);
- ad_write(sc, SP_RIGHT_AUX1_CONTROL, reg & ~AUX_INPUT_MUTE);
- }
-}
+ ad1848_devmap_t *entry;
+ struct ad1848_volume vol;
+ int error = EINVAL;
+ int dev;
-void
-ad1848_mute_aux2(sc, mute)
- struct ad1848_softc *sc;
- int mute;
-{
- u_char reg;
- if (mute) {
- DPRINTF(("ad1848_mute_aux2: muting\n"));
- reg = ad_read(sc, SP_LEFT_AUX2_CONTROL);
- ad_write(sc, SP_LEFT_AUX2_CONTROL, AUX_INPUT_MUTE|reg);
- reg = ad_read(sc, SP_RIGHT_AUX2_CONTROL);
- ad_write(sc, SP_RIGHT_AUX2_CONTROL, AUX_INPUT_MUTE|reg);
- } else if (!sc->aux2_mute) {
- DPRINTF(("ad1848_mute_aux2: unmuting\n"));
- reg = ad_read(sc, SP_LEFT_AUX2_CONTROL);
- ad_write(sc, SP_LEFT_AUX2_CONTROL, reg & ~AUX_INPUT_MUTE);
- reg = ad_read(sc, SP_RIGHT_AUX2_CONTROL);
- ad_write(sc, SP_RIGHT_AUX2_CONTROL, reg & ~AUX_INPUT_MUTE);
- }
-}
+ if (!(entry = ad1848_mixer_find_dev(map, cnt, cp)))
+ return (ENXIO);
-int
-ad1848_set_aux1_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- u_char reg;
- u_int atten;
-
- DPRINTF(("ad1848_set_aux1_gain: %d:%d\n", gp->left, gp->right));
-
- sc->aux1_gain = *gp;
+ dev = entry->dev;
- atten = ((AUDIO_MAX_GAIN - gp->left) * AUX_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_LEFT_AUX1_CONTROL);
- reg &= ~(AUX_INPUT_ATTEN_BITS);
- ad_write(sc, SP_LEFT_AUX1_CONTROL, (atten&0x1f)|reg);
+ switch (entry->kind) {
+ case AD1848_KIND_LVL:
+ if (cp->type != AUDIO_MIXER_VALUE)
+ break;
- atten = ((AUDIO_MAX_GAIN - gp->right) * AUX_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_RIGHT_AUX1_CONTROL);
- reg &= ~(AUX_INPUT_ATTEN_BITS);
- ad_write(sc, SP_RIGHT_AUX1_CONTROL, (atten&0x1f)|reg);
+ if (dev < AD1848_AUX2_CHANNEL ||
+ dev > AD1848_MONITOR_CHANNEL)
+ break;
- return(0);
-}
+ if (cp->un.value.num_channels != 1 &&
+ mixer_channel_info[dev].right_reg == 0)
+ break;
-int
-ad1848_get_aux1_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- *gp = sc->aux1_gain;
- return(0);
-}
+ error = ad1848_get_device_gain(ac, dev, &vol);
+ if (!error)
+ ad1848_from_vol(cp, &vol);
-int
-cs4231_set_linein_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- u_char reg, oregl, oregr;
- u_int atten;
-
- DPRINTF(("ad1848_set_linein_gain: %d:%d\n", gp->left, gp->right));
-
- sc->line_gain = *gp;
+ break;
- atten = ((AUDIO_MAX_GAIN - gp->left) * LINE_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- oregl = reg = ad_read(sc, CS_LEFT_LINE_CONTROL);
- reg &= ~(LINE_INPUT_ATTEN_BITS);
- ad_write(sc, CS_LEFT_LINE_CONTROL, (atten&LINE_INPUT_ATTEN_BITS)|reg);
+ case AD1848_KIND_MUTE:
+ if (cp->type != AUDIO_MIXER_ENUM) break;
- atten = ((AUDIO_MAX_GAIN - gp->right) * LINE_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- oregr = reg = ad_read(sc, CS_RIGHT_LINE_CONTROL);
- reg &= ~(LINE_INPUT_ATTEN_BITS);
- ad_write(sc, CS_RIGHT_LINE_CONTROL, (atten&LINE_INPUT_ATTEN_BITS)|reg);
+ cp->un.ord = ac->mute[dev] ? 1 : 0;
+ error = 0;
+ break;
- DPRINTF(("ad1848_set_linein_gain: was %x:%x\n", oregl, oregr));
- return(0);
-}
+ case AD1848_KIND_RECORDGAIN:
+ if (cp->type != AUDIO_MIXER_VALUE) break;
-int
-cs4231_get_linein_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- *gp = sc->line_gain;
- return(0);
+ error = ad1848_get_rec_gain(ac, &vol);
+ if (!error)
+ ad1848_from_vol(cp, &vol);
+
+ break;
+
+ case AD1848_KIND_MICGAIN:
+ if (cp->type != AUDIO_MIXER_VALUE) break;
+
+ error = ad1848_get_mic_gain(ac, &vol);
+ if (!error)
+ ad1848_from_vol(cp, &vol);
+
+ break;
+
+ case AD1848_KIND_RECORDSOURCE:
+ if (cp->type != AUDIO_MIXER_ENUM) break;
+ cp->un.ord = ad1848_get_rec_port(ac);
+ error = 0;
+ break;
+ default:
+ printf ("Invalid kind\n");
+ break;
+ }
+
+ return (error);
}
-int
-ad1848_set_aux2_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
+int
+ad1848_mixer_set_port(ac, map, cnt, cp)
+ struct ad1848_softc *ac;
+ struct ad1848_devmap *map;
+ int cnt;
+ mixer_ctrl_t *cp;
{
- u_char reg;
- u_int atten;
+ ad1848_devmap_t *entry;
+ struct ad1848_volume vol;
+ int error = EINVAL;
+ int dev;
+
+ if (!(entry = ad1848_mixer_find_dev(map, cnt, cp)))
+ return (ENXIO);
+
+ dev = entry->dev;
+
+ switch (entry->kind) {
+ case AD1848_KIND_LVL:
+ if (cp->type != AUDIO_MIXER_VALUE)
+ break;
+
+ if (dev < AD1848_AUX2_CHANNEL ||
+ dev > AD1848_MONITOR_CHANNEL)
+ break;
+
+ if (cp->un.value.num_channels != 1 &&
+ mixer_channel_info[dev].right_reg == 0)
+ break;
- DPRINTF(("ad1848_set_aux2_gain: %d:%d\n", gp->left, gp->right));
-
- sc->aux2_gain = *gp;
+ ad1848_to_vol(cp, &vol);
+ error = ad1848_set_channel_gain(ac, dev, &vol);
+ break;
- atten = ((AUDIO_MAX_GAIN - gp->left) * AUX_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_LEFT_AUX2_CONTROL);
- reg &= ~(AUX_INPUT_ATTEN_BITS);
- ad_write(sc, SP_LEFT_AUX2_CONTROL, (atten&0x1f)|reg);
+ case AD1848_KIND_MUTE:
+ if (cp->type != AUDIO_MIXER_ENUM) break;
+
+ ac->mute[dev] = (cp->un.ord ? MUTE_ALL : 0);
+ ad1848_mute_channel(ac, dev, ac->mute[dev]);
+ error = 0;
+ break;
- atten = ((AUDIO_MAX_GAIN - gp->right) * AUX_INPUT_ATTEN_BITS)/AUDIO_MAX_GAIN;
- reg = ad_read(sc, SP_RIGHT_AUX2_CONTROL);
- reg &= ~(AUX_INPUT_ATTEN_BITS);
- ad_write(sc, SP_RIGHT_AUX2_CONTROL, (atten&0x1f)|reg);
+ case AD1848_KIND_RECORDGAIN:
+ if (cp->type != AUDIO_MIXER_VALUE) break;
- return(0);
-}
+ ad1848_to_vol(cp, &vol);
+ error = ad1848_set_rec_gain(ac, &vol);
+ break;
-int
-ad1848_get_aux2_gain(sc, gp)
- struct ad1848_softc *sc;
- struct ad1848_volume *gp;
-{
- *gp = sc->aux2_gain;
- return 0;
+ case AD1848_KIND_MICGAIN:
+ if (cp->type != AUDIO_MIXER_VALUE) break;
+
+ ad1848_to_vol(cp, &vol);
+ error = ad1848_set_mic_gain(ac, &vol);
+ break;
+
+ case AD1848_KIND_RECORDSOURCE:
+ if (cp->type != AUDIO_MIXER_ENUM) break;
+
+ error = ad1848_set_rec_port(ac, cp->un.ord);
+ break;
+ default:
+ printf ("Invalid kind\n");
+ break;
+ }
+
+ return (error);
}
+
int
ad1848_query_encoding(addr, fp)
void *addr;
diff --git a/sys/dev/isa/ad1848var.h b/sys/dev/isa/ad1848var.h
index b5fdaf67365..f7f02a0a5a0 100644
--- a/sys/dev/isa/ad1848var.h
+++ b/sys/dev/isa/ad1848var.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: ad1848var.h,v 1.6 1998/04/26 21:02:39 provos Exp $ */
+/* $OpenBSD: ad1848var.h,v 1.7 1998/05/08 18:37:20 csapuntz Exp $ */
/* $NetBSD: ad1848var.h,v 1.22 1998/01/19 22:18:26 augustss Exp $ */
/*
@@ -75,14 +75,17 @@ struct ad1848_softc {
int sc_recdrq; /* record/capture DMA */
/* We keep track of these */
- struct ad1848_volume rec_gain, aux1_gain, aux2_gain, out_gain, mon_gain, line_gain, mono_gain;
+ struct ad1848_volume gains[6];
+
+ struct ad1848_volume rec_gain;
int rec_port; /* recording port */
/* ad1848 */
u_char MCE_bit;
char mic_gain_on; /* CS4231 only */
- char mono_mute, aux1_mute, aux2_mute, line_mute, mon_mute;
+ char mute[6];
+
char *chip_name;
int mode;
@@ -101,6 +104,21 @@ struct ad1848_softc {
int sc_iobase;
};
+#define MUTE_LEFT 1
+#define MUTE_RIGHT 2
+#define MUTE_ALL (MUTE_LEFT | MUTE_RIGHT)
+#define MUTE_MONO MUTE_ALL
+
+/* Don't change this ordering without seriously looking around.
+ These are indexes into mute[] array and into a register information
+ array */
+#define AD1848_AUX2_CHANNEL 0
+#define AD1848_AUX1_CHANNEL 1
+#define AD1848_DAC_CHANNEL 2
+#define AD1848_LINE_CHANNEL 3
+#define AD1848_MONO_CHANNEL 4
+#define AD1848_MONITOR_CHANNEL 5 /* Doesn't seem to be on all later chips */
+
/*
* Ad1848 ports
*/
@@ -110,6 +128,59 @@ struct ad1848_softc {
#define DAC_IN_PORT 3
#ifdef _KERNEL
+
+#define AD1848_KIND_LVL 0
+#define AD1848_KIND_MUTE 1
+#define AD1848_KIND_RECORDGAIN 2
+#define AD1848_KIND_MICGAIN 3
+#define AD1848_KIND_RECORDSOURCE 4
+
+typedef struct ad1848_devmap {
+ int id;
+ int kind;
+ int dev;
+} ad1848_devmap_t;
+
+static __inline int ad1848_to_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
+static __inline int ad1848_from_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
+
+static __inline int
+ad1848_to_vol(cp, vol)
+ mixer_ctrl_t *cp;
+ struct ad1848_volume *vol;
+{
+ if (cp->un.value.num_channels == 1) {
+ vol->left = vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
+ return(1);
+ }
+ else if (cp->un.value.num_channels == 2) {
+ vol->left = cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT];
+ vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT];
+ return(1);
+ }
+ return(0);
+}
+
+static __inline int
+ad1848_from_vol(cp, vol)
+ mixer_ctrl_t *cp;
+ struct ad1848_volume *vol;
+{
+ if (cp->un.value.num_channels == 1) {
+ cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol->left;
+ return(1);
+ }
+ else if (cp->un.value.num_channels == 2) {
+ cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = vol->left;
+ cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = vol->right;
+ return(1);
+ }
+ return(0);
+}
+
+
+int ad1848_mixer_get_port __P((struct ad1848_softc *, ad1848_devmap_t *, int cnt, mixer_ctrl_t *));
+int ad1848_mixer_set_port __P((struct ad1848_softc *, ad1848_devmap_t *, int, mixer_ctrl_t *));
int ad1848_mapprobe __P((struct ad1848_softc *, int));
int ad1848_probe __P((struct ad1848_softc *));
void ad1848_unmap __P((struct ad1848_softc *));
@@ -140,21 +211,14 @@ int ad1848_intr __P((void *));
int ad1848_set_rec_port __P((struct ad1848_softc *, int));
int ad1848_get_rec_port __P((struct ad1848_softc *));
-int ad1848_set_aux1_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_get_aux1_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_set_aux2_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_get_aux2_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_set_out_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_get_out_gain __P((struct ad1848_softc *, struct ad1848_volume *));
+int ad1848_set_channel_gain __P((struct ad1848_softc *, int, struct ad1848_volume *));
+int ad1848_get_device_gain __P((struct ad1848_softc *, int, struct ad1848_volume *));
int ad1848_set_rec_gain __P((struct ad1848_softc *, struct ad1848_volume *));
int ad1848_get_rec_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_set_mon_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-int ad1848_get_mon_gain __P((struct ad1848_softc *, struct ad1848_volume *));
/* Note: The mic pre-MUX gain is not a variable gain, it's 20dB or 0dB */
int ad1848_set_mic_gain __P((struct ad1848_softc *, struct ad1848_volume *));
int ad1848_get_mic_gain __P((struct ad1848_softc *, struct ad1848_volume *));
-void ad1848_mute_aux1 __P((struct ad1848_softc *, int /* onoff */));
-void ad1848_mute_aux2 __P((struct ad1848_softc *, int /* onoff */));
+void ad1848_mute_channel __P((struct ad1848_softc *, int device, int mute));
void *ad1848_malloc __P((void *, unsigned long, int, int));
void ad1848_free __P((void *, void *, int));
diff --git a/sys/dev/isa/gus.c b/sys/dev/isa/gus.c
index eb01c44a1a8..6ba8eaa4046 100644
--- a/sys/dev/isa/gus.c
+++ b/sys/dev/isa/gus.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: gus.c,v 1.14 1998/04/26 21:02:41 provos Exp $ */
+/* $OpenBSD: gus.c,v 1.15 1998/05/08 18:37:21 csapuntz Exp $ */
/* $NetBSD: gus.c,v 1.51 1998/01/25 23:48:06 mycroft Exp $ */
/*-
@@ -428,9 +428,6 @@ STATIC void gusics_mic_mute __P((struct ics2101_softc *, int));
STATIC void gusics_linein_mute __P((struct ics2101_softc *, int));
STATIC void gusics_cd_mute __P((struct ics2101_softc *, int));
-STATIC __inline int gus_to_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-STATIC __inline int gus_from_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-
void stereo_dmaintr __P((void *));
/*
@@ -1067,11 +1064,11 @@ gusopen(addr, flags)
if (HAS_CODEC(sc)) {
ad1848_open(&sc->sc_codec, flags);
- sc->sc_codec.aux1_mute = 0;
- ad1848_mute_aux1(&sc->sc_codec, 0); /* turn on DAC output */
+ sc->sc_codec.mute[AD1848_AUX1_CHANNEL] = 0;
+ ad1848_mute_channel(&sc->sc_codec, AD1848_AUX1_CHANNEL, 0); /* turn on DAC output */
if (flags & FREAD) {
- sc->sc_codec.mono_mute = 0;
- cs4231_mute_mono(&sc->sc_codec, 0);
+ sc->sc_codec.mute[AD1848_MONO_CHANNEL] = 0;
+ ad1848_mute_channel(&sc->sc_codec, AD1848_MONO_CHANNEL, 0);
}
} else if (flags & FREAD) {
/* enable/unmute the microphone */
@@ -1299,8 +1296,8 @@ gusmax_close(addr)
struct ad1848_softc *ac = addr;
struct gus_softc *sc = ac->parent;
#if 0
- ac->aux1_mute = 1;
- ad1848_mute_aux1(ac, 1); /* turn off DAC output */
+ ac->mute[AD1848_AUX1_CHANNEL] = MUTE_ALL;
+ ad1848_mute_channel(ac, MUTE_ALL); /* turn off DAC output */
#endif
ad1848_close(ac);
gusclose(sc);
@@ -3131,39 +3128,24 @@ gus_halt_in_dma(addr)
return 0;
}
-STATIC __inline int
-gus_to_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- vol->left = vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- vol->left = cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT];
- vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT];
- return(1);
- }
- return(0);
-}
-STATIC __inline int
-gus_from_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol->left;
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = vol->left;
- cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = vol->right;
- return(1);
- }
- return(0);
-}
+static ad1848_devmap_t gusmapping[] = {
+ {GUSMAX_DAC_LVL, AD1848_KIND_LVL, AD1848_AUX1_CHANNEL},
+ {GUSMAX_LINE_IN_LVL, AD1848_KIND_LVL, AD1848_LINE_CHANNEL},
+ {GUSMAX_MONO_LVL, AD1848_KIND_LVL, AD1848_MONO_CHANNEL},
+ {GUSMAX_CD_LVL, AD1848_KIND_LVL, AD1848_AUX2_CHANNEL},
+ {GUSMAX_MONITOR_LVL, AD1848_KIND_LVL, AD1848_MONITOR_CHANNEL},
+ {GUSMAX_OUT_LVL, AD1848_KIND_LVL, AD1848_DAC_CHANNEL},
+ {GUSMAX_DAC_MUTE, AD1848_KIND_MUTE, AD1848_AUX1_CHANNEL},
+ {GUSMAX_LINE_IN_MUTE, AD1848_KIND_MUTE, AD1848_LINE_CHANNEL},
+ {GUSMAX_MONO_MUTE, AD1848_KIND_MUTE, AD1848_MONO_CHANNEL},
+ {GUSMAX_CD_MUTE, AD1848_KIND_MUTE, AD1848_AUX2_CHANNEL},
+ {GUSMAX_MONITOR_MUTE, AD1848_KIND_MUTE, AD1848_MONITOR_CHANNEL},
+ {GUSMAX_REC_LVL, AD1848_KIND_RECORDGAIN, -1},
+ {GUSMAX_RECORD_SOURCE, AD1848_KIND_RECORDSOURCE, -1}
+};
+
+static int nummap = sizeof(gusmapping) / sizeof(gusmapping[0]);
STATIC int
gusmax_mixer_get_port(addr, cp)
@@ -3173,72 +3155,14 @@ gusmax_mixer_get_port(addr, cp)
struct ad1848_softc *ac = addr;
struct gus_softc *sc = ac->parent;
struct ad1848_volume vol;
- int error = EINVAL;
+ int error = ad1848_mixer_get_port(ac, gusmapping, nummap, cp);
- DPRINTF(("gusmax_mixer_get_port: port=%d\n", cp->dev));
-
- switch (cp->dev) {
-#if 0 /* use mono level instead */
- case GUSMAX_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_mic_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
-#endif
-
- case GUSMAX_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux1_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
+ if (error != ENXIO)
+ return (error);
- case GUSMAX_LINE_IN_LVL: /* line in */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = cs4231_get_linein_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
-
- case GUSMAX_MONO_LVL: /* mono */
- if (cp->type == AUDIO_MIXER_VALUE &&
- cp->un.value.num_channels == 1) {
- error = cs4231_get_mono_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
-
- case GUSMAX_CD_LVL: /* CD */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux2_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
-
- case GUSMAX_MONITOR_LVL: /* monitor level */
- if (cp->type == AUDIO_MIXER_VALUE &&
- cp->un.value.num_channels == 1) {
- error = ad1848_get_mon_gain(ac, &vol);
- if (!error)
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] =
- vol.left;
- }
- break;
-
- case GUSMAX_OUT_LVL: /* output level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_out_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
+ error = EINVAL;
+ switch (cp->dev) {
case GUSMAX_SPEAKER_LVL: /* fake speaker for mute naming */
if (cp->type == AUDIO_MIXER_VALUE) {
if (sc->sc_mixcontrol & GUSMASK_LINE_OUT)
@@ -3246,43 +3170,7 @@ gusmax_mixer_get_port(addr, cp)
else
vol.left = vol.right = AUDIO_MIN_GAIN;
error = 0;
- gus_from_vol(cp, &vol);
- }
- break;
-
- case GUSMAX_LINE_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ac->line_mute;
- error = 0;
- }
- break;
-
-
- case GUSMAX_DAC_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ac->aux1_mute;
- error = 0;
- }
- break;
-
- case GUSMAX_CD_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ac->aux2_mute;
- error = 0;
- }
- break;
-
- case GUSMAX_MONO_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ac->mono_mute;
- error = 0;
- }
- break;
-
- case GUSMAX_MONITOR_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ac->mon_mute;
- error = 0;
+ ad1848_from_vol(cp, &vol);
}
break;
@@ -3292,22 +3180,6 @@ gusmax_mixer_get_port(addr, cp)
error = 0;
}
break;
-
- case GUSMAX_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_rec_gain(ac, &vol);
- if (!error)
- gus_from_vol(cp, &vol);
- }
- break;
-
- case GUSMAX_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ad1848_get_rec_port(ac);
- error = 0;
- }
- break;
-
default:
error = ENXIO;
break;
@@ -3384,7 +3256,7 @@ gus_mixer_get_port(addr, cp)
if (cp->type == AUDIO_MIXER_VALUE) {
vol.left = ic->sc_setting[GUSMIX_CHAN_MASTER][ICSMIX_LEFT];
vol.right = ic->sc_setting[GUSMIX_CHAN_MASTER][ICSMIX_RIGHT];
- if (gus_from_vol(cp, &vol))
+ if (ad1848_from_vol(cp, &vol))
error = 0;
}
break;
@@ -3393,7 +3265,7 @@ gus_mixer_get_port(addr, cp)
if (cp->type == AUDIO_MIXER_VALUE) {
vol.left = ic->sc_setting[GUSMIX_CHAN_MIC][ICSMIX_LEFT];
vol.right = ic->sc_setting[GUSMIX_CHAN_MIC][ICSMIX_RIGHT];
- if (gus_from_vol(cp, &vol))
+ if (ad1848_from_vol(cp, &vol))
error = 0;
}
break;
@@ -3402,7 +3274,7 @@ gus_mixer_get_port(addr, cp)
if (cp->type == AUDIO_MIXER_VALUE) {
vol.left = ic->sc_setting[GUSMIX_CHAN_LINE][ICSMIX_LEFT];
vol.right = ic->sc_setting[GUSMIX_CHAN_LINE][ICSMIX_RIGHT];
- if (gus_from_vol(cp, &vol))
+ if (ad1848_from_vol(cp, &vol))
error = 0;
}
break;
@@ -3412,7 +3284,7 @@ gus_mixer_get_port(addr, cp)
if (cp->type == AUDIO_MIXER_VALUE) {
vol.left = ic->sc_setting[GUSMIX_CHAN_CD][ICSMIX_LEFT];
vol.right = ic->sc_setting[GUSMIX_CHAN_CD][ICSMIX_RIGHT];
- if (gus_from_vol(cp, &vol))
+ if (ad1848_from_vol(cp, &vol))
error = 0;
}
break;
@@ -3421,7 +3293,7 @@ gus_mixer_get_port(addr, cp)
if (cp->type == AUDIO_MIXER_VALUE) {
vol.left = ic->sc_setting[GUSMIX_CHAN_DAC][ICSMIX_LEFT];
vol.right = ic->sc_setting[GUSMIX_CHAN_DAC][ICSMIX_RIGHT];
- if (gus_from_vol(cp, &vol))
+ if (ad1848_from_vol(cp, &vol))
error = 0;
}
break;
@@ -3494,70 +3366,18 @@ gusmax_mixer_set_port(addr, cp)
struct ad1848_softc *ac = addr;
struct gus_softc *sc = ac->parent;
struct ad1848_volume vol;
- int error = EINVAL;
+ int error = ad1848_mixer_set_port(ac, gusmapping, nummap, cp);
+ if (error != ENXIO)
+ return (error);
+
DPRINTF(("gusmax_mixer_set_port: dev=%d type=%d\n", cp->dev, cp->type));
switch (cp->dev) {
-#if 0
- case GUSMAX_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE &&
- cp->un.value.num_channels == 1) {
- /* XXX enable/disable pre-MUX fixed gain */
- if (gus_to_vol(cp, &vol))
- error = ad1848_set_mic_gain(ac, &vol);
- }
- break;
-#endif
-
- case GUSMAX_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol))
- error = ad1848_set_aux1_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_LINE_IN_LVL: /* line in */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol))
- error = cs4231_set_linein_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_MONO_LVL: /* mic/mono in */
- if (cp->type == AUDIO_MIXER_VALUE &&
- cp->un.value.num_channels == 1) {
- if (gus_to_vol(cp, &vol))
- error = cs4231_set_mono_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_CD_LVL: /* CD: AUX2 */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol))
- error = ad1848_set_aux2_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_MONITOR_LVL:
- if (cp->type == AUDIO_MIXER_VALUE &&
- cp->un.value.num_channels == 1) {
- vol.left = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- error = ad1848_set_mon_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_OUT_LVL: /* output volume */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol))
- error = ad1848_set_out_gain(ac, &vol);
- }
- break;
-
case GUSMAX_SPEAKER_LVL:
if (cp->type == AUDIO_MIXER_VALUE &&
cp->un.value.num_channels == 1) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
gus_speaker_ctl(sc, vol.left > AUDIO_MIN_GAIN ?
SPKR_ON : SPKR_OFF);
error = 0;
@@ -3565,53 +3385,6 @@ gusmax_mixer_set_port(addr, cp)
}
break;
- case GUSMAX_LINE_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- ac->line_mute = cp->un.ord ? 1 : 0;
- DPRINTF(("line mute %d\n", cp->un.ord));
- cs4231_mute_line(ac, ac->line_mute);
- gus_linein_ctl(sc, ac->line_mute ? SPKR_OFF : SPKR_ON);
- error = 0;
- }
- break;
-
- case GUSMAX_DAC_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- ac->aux1_mute = cp->un.ord ? 1 : 0;
- DPRINTF(("dac mute %d\n", cp->un.ord));
- ad1848_mute_aux1(ac, ac->aux1_mute);
- error = 0;
- }
- break;
-
- case GUSMAX_CD_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- ac->aux2_mute = cp->un.ord ? 1 : 0;
- DPRINTF(("cd mute %d\n", cp->un.ord));
- ad1848_mute_aux2(ac, ac->aux2_mute);
- error = 0;
- }
- break;
-
- case GUSMAX_MONO_MUTE: /* Microphone */
- if (cp->type == AUDIO_MIXER_ENUM) {
- ac->mono_mute = cp->un.ord ? 1 : 0;
- DPRINTF(("mono mute %d\n", cp->un.ord));
- cs4231_mute_mono(ac, ac->mono_mute);
- gus_mic_ctl(sc, ac->mono_mute ? SPKR_OFF : SPKR_ON);
- error = 0;
- }
- break;
-
- case GUSMAX_MONITOR_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- ac->mon_mute = cp->un.ord ? 1 : 0;
- DPRINTF(("mono mute %d\n", cp->un.ord));
- cs4231_mute_monitor(ac, ac->mon_mute);
- error = 0;
- }
- break;
-
case GUSMAX_SPEAKER_MUTE:
if (cp->type == AUDIO_MIXER_ENUM) {
gus_speaker_ctl(sc, cp->un.ord ? SPKR_OFF : SPKR_ON);
@@ -3619,19 +3392,6 @@ gusmax_mixer_set_port(addr, cp)
}
break;
- case GUSMAX_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol))
- error = ad1848_set_rec_gain(ac, &vol);
- }
- break;
-
- case GUSMAX_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- error = ad1848_set_rec_port(ac, cp->un.ord);
- }
- break;
-
default:
return ENXIO;
/*NOTREACHED*/
@@ -3705,7 +3465,7 @@ gus_mixer_set_port(addr, cp)
case GUSICS_MASTER_LVL:
if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
ics2101_mix_attenuate(ic,
GUSMIX_CHAN_MASTER,
ICSMIX_LEFT,
@@ -3721,7 +3481,7 @@ gus_mixer_set_port(addr, cp)
case GUSICS_MIC_IN_LVL: /* Microphone */
if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
ics2101_mix_attenuate(ic,
GUSMIX_CHAN_MIC,
ICSMIX_LEFT,
@@ -3737,7 +3497,7 @@ gus_mixer_set_port(addr, cp)
case GUSICS_LINE_IN_LVL: /* line in */
if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
ics2101_mix_attenuate(ic,
GUSMIX_CHAN_LINE,
ICSMIX_LEFT,
@@ -3754,7 +3514,7 @@ gus_mixer_set_port(addr, cp)
case GUSICS_CD_LVL:
if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
ics2101_mix_attenuate(ic,
GUSMIX_CHAN_CD,
ICSMIX_LEFT,
@@ -3770,7 +3530,7 @@ gus_mixer_set_port(addr, cp)
case GUSICS_DAC_LVL: /* dac out */
if (cp->type == AUDIO_MIXER_VALUE) {
- if (gus_to_vol(cp, &vol)) {
+ if (ad1848_to_vol(cp, &vol)) {
ics2101_mix_attenuate(ic,
GUSMIX_CHAN_DAC,
ICSMIX_LEFT,
diff --git a/sys/dev/isa/pss.c b/sys/dev/isa/pss.c
index c2bf800b640..8f907a9c456 100644
--- a/sys/dev/isa/pss.c
+++ b/sys/dev/isa/pss.c
@@ -1,12 +1,12 @@
-/* $OpenBSD: pss.c,v 1.13 1998/04/26 21:02:55 provos Exp $ */
+/* $OpenBSD: pss.c,v 1.14 1998/05/08 18:37:22 csapuntz Exp $ */
/* $NetBSD: pss.c,v 1.38 1998/01/12 09:43:44 thorpej Exp $ */
/*
* Copyright (c) 1994 John Brezak
* Copyright (c) 1991-1993 Regents of the University of California.
- * All rights reserved.
+ * All rightOAs reserved.
*
- * Redistribution and use in source and binary forms, with or without
+x * 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
@@ -213,9 +213,6 @@ int pss_get_master_mode __P((struct pss_softc *, u_int *));
int pss_get_treble __P((struct pss_softc *, u_char *));
int pss_get_bass __P((struct pss_softc *, u_char *));
-static int pss_to_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-static int pss_from_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-
#ifdef AUDIO_DEBUG
void wss_dump_regs __P((struct ad1848_softc *));
#endif
@@ -1134,39 +1131,6 @@ pcdattach(parent, self, aux)
}
#endif /* notyet */
-static int
-pss_to_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- vol->left = vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- vol->left = cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT];
- vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT];
- return(1);
- }
- return(0);
-}
-
-static int
-pss_from_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol->left;
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = vol->left;
- cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = vol->right;
- return(1);
- }
- return(0);
-}
int
pss_set_master_gain(sc, gp)
@@ -1362,6 +1326,20 @@ pss_getdev(addr, retp)
return 0;
}
+static ad1848_devmap_t mappings[] = {
+{ PSS_MIC_IN_LVL, AD1848_KIND_LVL, AD1848_AUX2_CHANNEL },
+{ PSS_LINE_IN_LVL, AD1848_KIND_LVL, AD1848_AUX1_CHANNEL },
+{ PSS_DAC_LVL, AD1848_KIND_LVL, AD1848_DAC_CHANNEL },
+{ PSS_MON_LVL, AD1848_KIND_LVL, AD1848_MONO_CHANNEL },
+{ PSS_MIC_IN_MUTE, AD1848_KIND_MUTE, AD1848_AUX2_CHANNEL },
+{ PSS_LINE_IN_MUTE, AD1848_KIND_MUTE, AD1848_AUX1_CHANNEL },
+{ PSS_DAC_MUTE, AD1848_KIND_MUTE, AD1848_DAC_CHANNEL },
+{ PSS_REC_LVL, AD1848_KIND_RECORDGAIN, -1 },
+{ PSS_RECORD_SOURCE, AD1848_KIND_RECORDSOURCE, -1}
+};
+
+static int nummap = sizeof(mappings) / sizeof(mappings[0]);
+
int
pss_mixer_set_port(addr, cp)
void *addr;
@@ -1370,81 +1348,15 @@ pss_mixer_set_port(addr, cp)
struct ad1848_softc *ac = addr;
struct pss_softc *sc = ac->parent;
struct ad1848_volume vol;
- int error = EINVAL;
+ int error = ad1848_mixer_set_port(ac, mappings, nummap, cp);
- DPRINTF(("pss_mixer_set_port: dev=%d type=%d\n", cp->dev, cp->type));
+ if (error != ENXIO)
+ return (error);
switch (cp->dev) {
- case PSS_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (pss_to_vol(cp, &vol))
- error = ad1848_set_aux2_gain(ac, &vol);
- }
- break;
-
- case PSS_MIC_IN_MUTE: /* Microphone */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->mic_mute = cp->un.ord;
- DPRINTF(("mic mute %d\n", cp->un.ord));
- ad1848_mute_aux2(ac, cp->un.ord);
- error = 0;
- }
- break;
-
- case PSS_LINE_IN_LVL: /* linein/CD */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (pss_to_vol(cp, &vol))
- error = ad1848_set_aux1_gain(ac, &vol);
- }
- break;
-
- case PSS_LINE_IN_MUTE: /* linein/CD */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->cd_mute = cp->un.ord;
- DPRINTF(("CD mute %d\n", cp->un.ord));
- ad1848_mute_aux1(ac, cp->un.ord);
- error = 0;
- }
- break;
-
- case PSS_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (pss_to_vol(cp, &vol))
- error = ad1848_set_out_gain(ac, &vol);
- }
- break;
-
- case PSS_DAC_MUTE: /* dac out */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->dac_mute = cp->un.ord;
- DPRINTF(("DAC mute %d\n", cp->un.ord));
- error = 0;
- }
- break;
-
- case PSS_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (pss_to_vol(cp, &vol))
- error = ad1848_set_rec_gain(ac, &vol);
- }
- break;
-
- case PSS_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- error = ad1848_set_rec_port(ac, cp->un.ord);
- }
- break;
-
- case PSS_MON_LVL:
- if (cp->type == AUDIO_MIXER_VALUE && cp->un.value.num_channels == 1) {
- vol.left = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- error = ad1848_set_mon_gain(ac, &vol);
- }
- break;
-
case PSS_MASTER_VOL: /* master volume */
if (cp->type == AUDIO_MIXER_VALUE) {
- if (pss_to_vol(cp, &vol))
+ if (ad1848_to_vol(cp, &vol))
error = pss_set_master_gain(sc, &vol);
}
break;
@@ -1481,84 +1393,19 @@ pss_mixer_get_port(addr, cp)
struct pss_softc *sc = ac->parent;
struct ad1848_volume vol;
u_char eq;
- int error = EINVAL;
-
- DPRINTF(("pss_mixer_get_port: port=%d\n", cp->dev));
-
- switch (cp->dev) {
- case PSS_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux2_gain(ac, &vol);
- if (!error)
- pss_from_vol(cp, &vol);
- }
- break;
+ int error = ad1848_mixer_get_port(ac, mappings, nummap, cp);
- case PSS_MIC_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->mic_mute;
- error = 0;
- }
- break;
-
- case PSS_LINE_IN_LVL: /* linein/CD */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux1_gain(ac, &vol);
- if (!error)
- pss_from_vol(cp, &vol);
- }
- break;
+ if (error != ENXIO)
+ return (error);
- case PSS_LINE_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->cd_mute;
- error = 0;
- }
- break;
-
- case PSS_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_out_gain(ac, &vol);
- if (!error)
- pss_from_vol(cp, &vol);
- }
- break;
-
- case PSS_DAC_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->dac_mute;
- error = 0;
- }
- break;
-
- case PSS_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_rec_gain(ac, &vol);
- if (!error)
- pss_from_vol(cp, &vol);
- }
- break;
-
- case PSS_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ad1848_get_rec_port(ac);
- error = 0;
- }
- break;
-
- case PSS_MON_LVL: /* monitor level */
- if (cp->type == AUDIO_MIXER_VALUE && cp->un.value.num_channels == 1) {
- error = ad1848_get_mon_gain(ac, &vol);
- if (!error)
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol.left;
- }
- break;
+ error = EINVAL;
+ switch (cp->dev) {
case PSS_MASTER_VOL: /* master volume */
if (cp->type == AUDIO_MIXER_VALUE) {
error = pss_get_master_gain(sc, &vol);
if (!error)
- pss_from_vol(cp, &vol);
+ ad1848_from_vol(cp, &vol);
}
break;
diff --git a/sys/dev/isa/wss.c b/sys/dev/isa/wss.c
index e14338bdd44..147298bf6a5 100644
--- a/sys/dev/isa/wss.c
+++ b/sys/dev/isa/wss.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: wss.c,v 1.15 1998/04/26 21:03:03 provos Exp $ */
+/* $OpenBSD: wss.c,v 1.16 1998/05/08 18:37:24 csapuntz Exp $ */
/* $NetBSD: wss.c,v 1.42 1998/01/19 22:18:23 augustss Exp $ */
/*
@@ -81,9 +81,6 @@ int wss_mixer_set_port __P((void *, mixer_ctrl_t *));
int wss_mixer_get_port __P((void *, mixer_ctrl_t *));
int wss_query_devinfo __P((void *, mixer_devinfo_t *));
-static int wss_to_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-static int wss_from_vol __P((mixer_ctrl_t *, struct ad1848_volume *));
-
/*
* Define our interface to the higher level audio driver.
*/
@@ -157,40 +154,6 @@ wssattach(sc)
audio_attach_mi(&wss_hw_if, 0, &sc->sc_ad1848, &sc->sc_dev);
}
-static int
-wss_to_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- vol->left = vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- vol->left = cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT];
- vol->right = cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT];
- return(1);
- }
- return(0);
-}
-
-static int
-wss_from_vol(cp, vol)
- mixer_ctrl_t *cp;
- struct ad1848_volume *vol;
-{
- if (cp->un.value.num_channels == 1) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol->left;
- return(1);
- }
- else if (cp->un.value.num_channels == 2) {
- cp->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = vol->left;
- cp->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = vol->right;
- return(1);
- }
- return(0);
-}
-
int
wss_getdev(addr, retp)
void *addr;
@@ -200,90 +163,28 @@ wss_getdev(addr, retp)
return 0;
}
+static ad1848_devmap_t mappings[] = {
+{ WSS_MIC_IN_LVL, AD1848_KIND_LVL, AD1848_AUX2_CHANNEL },
+{ WSS_LINE_IN_LVL, AD1848_KIND_LVL, AD1848_AUX1_CHANNEL },
+{ WSS_DAC_LVL, AD1848_KIND_LVL, AD1848_DAC_CHANNEL },
+{ WSS_MON_LVL, AD1848_KIND_LVL, AD1848_MONO_CHANNEL },
+{ WSS_MIC_IN_MUTE, AD1848_KIND_MUTE, AD1848_AUX2_CHANNEL },
+{ WSS_LINE_IN_MUTE, AD1848_KIND_MUTE, AD1848_AUX1_CHANNEL },
+{ WSS_DAC_MUTE, AD1848_KIND_MUTE, AD1848_DAC_CHANNEL },
+{ WSS_REC_LVL, AD1848_KIND_RECORDGAIN, -1 },
+{ WSS_RECORD_SOURCE, AD1848_KIND_RECORDSOURCE, -1}
+};
+
+static int nummap = sizeof(mappings) / sizeof(mappings[0]);
+
int
wss_mixer_set_port(addr, cp)
void *addr;
mixer_ctrl_t *cp;
{
struct ad1848_softc *ac = addr;
- struct wss_softc *sc = ac->parent;
- struct ad1848_volume vol;
- int error = EINVAL;
-
- DPRINTF(("wss_mixer_set_port: dev=%d type=%d\n", cp->dev, cp->type));
-
- switch (cp->dev) {
- case WSS_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (wss_to_vol(cp, &vol))
- error = ad1848_set_aux2_gain(ac, &vol);
- }
- break;
-
- case WSS_MIC_IN_MUTE: /* Microphone */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->mic_mute = cp->un.ord;
- DPRINTF(("mic mute %d\n", cp->un.ord));
- error = 0;
- }
- break;
-
- case WSS_LINE_IN_LVL: /* linein/CD */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (wss_to_vol(cp, &vol))
- error = ad1848_set_aux1_gain(ac, &vol);
- }
- break;
-
- case WSS_LINE_IN_MUTE: /* linein/CD */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->cd_mute = cp->un.ord;
- DPRINTF(("CD mute %d\n", cp->un.ord));
- error = 0;
- }
- break;
-
- case WSS_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (wss_to_vol(cp, &vol))
- error = ad1848_set_out_gain(ac, &vol);
- }
- break;
-
- case WSS_DAC_MUTE: /* dac out */
- if (cp->type == AUDIO_MIXER_ENUM) {
- sc->dac_mute = cp->un.ord;
- DPRINTF(("DAC mute %d\n", cp->un.ord));
- error = 0;
- }
- break;
-
- case WSS_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- if (wss_to_vol(cp, &vol))
- error = ad1848_set_rec_gain(ac, &vol);
- }
- break;
-
- case WSS_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- error = ad1848_set_rec_port(ac, cp->un.ord);
- }
- break;
- case WSS_MON_LVL:
- if (cp->type == AUDIO_MIXER_VALUE && cp->un.value.num_channels == 1) {
- vol.left = cp->un.value.level[AUDIO_MIXER_LEVEL_MONO];
- error = ad1848_set_mon_gain(ac, &vol);
- }
- break;
-
- default:
- return ENXIO;
- /*NOTREACHED*/
- }
-
- return 0;
+ return (ad1848_mixer_set_port(ac, mappings, nummap, cp));
}
int
@@ -292,87 +193,8 @@ wss_mixer_get_port(addr, cp)
mixer_ctrl_t *cp;
{
struct ad1848_softc *ac = addr;
- struct wss_softc *sc = ac->parent;
- struct ad1848_volume vol;
- int error = EINVAL;
-
- DPRINTF(("wss_mixer_get_port: port=%d\n", cp->dev));
-
- switch (cp->dev) {
- case WSS_MIC_IN_LVL: /* Microphone */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux2_gain(ac, &vol);
- if (!error)
- wss_from_vol(cp, &vol);
- }
- break;
-
- case WSS_MIC_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->mic_mute;
- error = 0;
- }
- break;
-
- case WSS_LINE_IN_LVL: /* linein/CD */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_aux1_gain(ac, &vol);
- if (!error)
- wss_from_vol(cp, &vol);
- }
- break;
-
- case WSS_LINE_IN_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->cd_mute;
- error = 0;
- }
- break;
-
- case WSS_DAC_LVL: /* dac out */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_out_gain(ac, &vol);
- if (!error)
- wss_from_vol(cp, &vol);
- }
- break;
-
- case WSS_DAC_MUTE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = sc->dac_mute;
- error = 0;
- }
- break;
-
- case WSS_REC_LVL: /* record level */
- if (cp->type == AUDIO_MIXER_VALUE) {
- error = ad1848_get_rec_gain(ac, &vol);
- if (!error)
- wss_from_vol(cp, &vol);
- }
- break;
-
- case WSS_RECORD_SOURCE:
- if (cp->type == AUDIO_MIXER_ENUM) {
- cp->un.ord = ad1848_get_rec_port(ac);
- error = 0;
- }
- break;
-
- case WSS_MON_LVL: /* monitor level */
- if (cp->type == AUDIO_MIXER_VALUE && cp->un.value.num_channels == 1) {
- error = ad1848_get_mon_gain(ac, &vol);
- if (!error)
- cp->un.value.level[AUDIO_MIXER_LEVEL_MONO] = vol.left;
- }
- break;
-
- default:
- error = ENXIO;
- break;
- }
- return(error);
+ return (ad1848_mixer_get_port(ac, mappings, nummap, cp));
}
int
diff --git a/sys/dev/isa/ym.c b/sys/dev/isa/ym.c
new file mode 100644
index 00000000000..e752bf3c902
--- /dev/null
+++ b/sys/dev/isa/ym.c
@@ -0,0 +1,470 @@
+/* $OpenBSD: ym.c,v 1.1 1998/05/08 18:37:25 csapuntz Exp $ */
+
+
+/*
+ * Copyright (c) 1998 Constantine Sapuntzakis. 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.
+ * 3. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * 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.
+ */
+
+
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/errno.h>
+#include <sys/ioctl.h>
+#include <sys/syslog.h>
+#include <sys/device.h>
+#include <sys/proc.h>
+#include <sys/buf.h>
+
+#include <machine/cpu.h>
+#include <machine/intr.h>
+#include <machine/bus.h>
+#include <machine/pio.h>
+
+#include <sys/audioio.h>
+#include <dev/audio_if.h>
+
+#include <dev/isa/isavar.h>
+#include <dev/isa/isadmavar.h>
+
+#include <dev/ic/ad1848reg.h>
+#include <dev/isa/ad1848var.h>
+#include <dev/ic/opl3sa3.h>
+#include <dev/isa/ymvar.h>
+
+
+int ym_getdev __P((void *, struct audio_device *));
+int ym_mixer_set_port __P((void *, mixer_ctrl_t *));
+int ym_mixer_get_port __P((void *, mixer_ctrl_t *));
+int ym_query_devinfo __P((void *, mixer_devinfo_t *));
+
+static void ym_mute __P((struct ym_softc *, int, int));
+static void ym_set_master_gain __P((struct ym_softc *, struct ad1848_volume *));
+static void ym_set_mic_gain __P((struct ym_softc *, struct ad1848_volume *));
+
+
+
+struct audio_hw_if ym_hw_if = {
+ ad1848_open,
+ ad1848_close,
+ NULL,
+ ad1848_query_encoding,
+ ad1848_set_params,
+ ad1848_round_blocksize,
+ ad1848_commit_settings,
+ ad1848_dma_init_output,
+ ad1848_dma_init_input,
+ ad1848_dma_output,
+ ad1848_dma_input,
+ ad1848_halt_out_dma,
+ ad1848_halt_in_dma,
+ NULL,
+ ym_getdev,
+ NULL,
+ ym_mixer_set_port,
+ ym_mixer_get_port,
+ ym_query_devinfo,
+ ad1848_malloc,
+ ad1848_free,
+ ad1848_round,
+ ad1848_mappage,
+ ad1848_get_props,
+};
+
+
+struct cfdriver ym_cd = {
+ NULL, "ym", DV_DULL
+};
+
+struct audio_device ym_device = {
+ "ym,ad1848",
+ "",
+ "ym"
+};
+
+static __inline int ym_read __P((struct ym_softc *, int));
+static __inline void ym_write __P((struct ym_softc *, int, int));
+
+void
+ym_attach(sc)
+ struct ym_softc *sc;
+
+{
+ struct ad1848_volume vol_mid = {220, 220};
+ struct ad1848_volume vol_0 = {0, 0};
+
+ sc->sc_ih = isa_intr_establish(sc->sc_ic, sc->ym_irq, IST_EDGE, IPL_AUDIO,
+ ad1848_intr, &sc->sc_ad1848, sc->sc_dev.dv_xname);
+
+ ad1848_attach(&sc->sc_ad1848);
+ sc->sc_ad1848.parent = sc;
+
+ /* Establish chip in well known mode */
+ ym_set_master_gain(sc, &vol_mid);
+ ym_set_mic_gain(sc, &vol_0);
+ sc->master_mute = 0;
+ ym_mute(sc, SA3_LCH, sc->master_mute);
+ ym_mute(sc, SA3_RCH, sc->master_mute);
+
+ sc->mic_mute = 1;
+ ym_mute(sc, SA3_MIC, sc->mic_mute);
+
+ audio_attach_mi(&ym_hw_if, 0, &sc->sc_ad1848, &sc->sc_dev);
+}
+
+static __inline int
+ym_read(sc, reg)
+ struct ym_softc *sc;
+ int reg;
+{
+ bus_space_write_1(sc->sc_iot, sc->sc_controlioh, 0, 0x1d);
+ bus_space_write_1(sc->sc_iot, sc->sc_controlioh, 0, (reg & 0xff));
+ return (bus_space_read_1(sc->sc_iot, sc->sc_controlioh, 1));
+}
+
+static __inline void
+ym_write(sc, reg, data)
+ struct ym_softc *sc;
+ int reg;
+ int data;
+{
+ bus_space_write_1(sc->sc_iot, sc->sc_controlioh, 0, 0x1d);
+ bus_space_write_1(sc->sc_iot, sc->sc_controlioh, 0, (reg & 0xff));
+ bus_space_write_1(sc->sc_iot, sc->sc_controlioh, 1, (data & 0xff));
+}
+
+
+
+int
+ym_getdev(addr, retp)
+ void *addr;
+ struct audio_device *retp;
+{
+ *retp = ym_device;
+ return 0;
+}
+
+
+static ad1848_devmap_t mappings[] = {
+{ YM_MIDI_LVL, AD1848_KIND_LVL, AD1848_AUX2_CHANNEL },
+{ YM_CD_LVL, AD1848_KIND_LVL, AD1848_AUX1_CHANNEL },
+{ YM_DAC_LVL, AD1848_KIND_LVL, AD1848_DAC_CHANNEL },
+{ YM_LINE_LVL, AD1848_KIND_LVL, AD1848_LINE_CHANNEL },
+{ YM_SPEAKER_LVL, AD1848_KIND_LVL, AD1848_MONO_CHANNEL },
+{ YM_MONITOR_LVL, AD1848_KIND_LVL, AD1848_MONITOR_CHANNEL },
+{ YM_MIDI_MUTE, AD1848_KIND_MUTE, AD1848_AUX2_CHANNEL },
+{ YM_CD_MUTE, AD1848_KIND_MUTE, AD1848_AUX1_CHANNEL },
+{ YM_DAC_MUTE, AD1848_KIND_MUTE, AD1848_DAC_CHANNEL },
+{ YM_LINE_MUTE, AD1848_KIND_MUTE, AD1848_LINE_CHANNEL },
+{ YM_SPEAKER_MUTE, AD1848_KIND_MUTE, AD1848_MONO_CHANNEL },
+{ YM_MONITOR_MUTE, AD1848_KIND_MUTE, AD1848_MONITOR_CHANNEL },
+{ YM_REC_LVL, AD1848_KIND_RECORDGAIN, -1 },
+{ YM_RECORD_SOURCE, AD1848_KIND_RECORDSOURCE, -1}
+};
+
+static int nummap = sizeof(mappings) / sizeof(mappings[0]);
+
+
+static void
+ym_mute(sc, left_reg, mute)
+ struct ym_softc *sc;
+ int left_reg;
+ int mute;
+
+{
+ u_char reg;
+
+ if (mute) {
+ reg = ym_read(sc, left_reg);
+ ym_write (sc, left_reg, reg | 0x80);
+ } else {
+ reg = ym_read(sc, left_reg);
+ ym_write (sc, left_reg, reg & ~0x80);
+ }
+}
+
+#define MIC_ATTEN_BITS 0x1f
+#define MASTER_ATTEN_BITS 0x0f
+
+
+static void
+ym_set_master_gain(sc, vol)
+ struct ym_softc *sc;
+ struct ad1848_volume *vol;
+{
+ u_char reg;
+ u_int atten;
+
+ sc->master_gain = *vol;
+
+ atten = ((AUDIO_MAX_GAIN - vol->left) * MASTER_ATTEN_BITS)/AUDIO_MAX_GAIN;
+
+ reg = ym_read(sc, SA3_LCH);
+
+ reg &= ~(MASTER_ATTEN_BITS);
+ reg |= atten;
+
+ ym_write (sc, SA3_LCH, reg);
+
+ atten = ((AUDIO_MAX_GAIN - vol->right) * MASTER_ATTEN_BITS)/AUDIO_MAX_GAIN;
+
+ reg = ym_read(sc, SA3_RCH) & ~(MASTER_ATTEN_BITS);
+ reg |= atten;
+
+ ym_write (sc, SA3_RCH, reg);
+}
+
+static void
+ym_set_mic_gain(sc, vol)
+ struct ym_softc *sc;
+ struct ad1848_volume *vol;
+{
+ u_char reg;
+ u_int atten;
+
+ sc->mic_gain = *vol;
+
+ atten = ((AUDIO_MAX_GAIN - vol->left) * MIC_ATTEN_BITS)/AUDIO_MAX_GAIN;
+
+ reg = ym_read(sc, SA3_MIC) & ~(MIC_ATTEN_BITS);
+ reg |= atten;
+
+ ym_write (sc, SA3_MIC, reg);
+}
+
+int
+ym_mixer_set_port(addr, cp)
+ void *addr;
+ mixer_ctrl_t *cp;
+{
+ struct ad1848_softc *ac = addr;
+ struct ym_softc *sc = ac->parent;
+ struct ad1848_volume vol;
+ int error = ad1848_mixer_set_port(ac, mappings, nummap, cp);
+
+ if (error != ENXIO)
+ return (error);
+
+ error = 0;
+
+ switch (cp->dev) {
+ case YM_OUTPUT_LVL:
+ ad1848_to_vol(cp, &vol);
+ ym_set_master_gain(sc, &vol);
+ break;
+
+ case YM_OUTPUT_MUTE:
+ sc->master_mute = (cp->un.ord != 0);
+ ym_mute(sc, SA3_LCH, sc->master_mute);
+ ym_mute(sc, SA3_RCH, sc->master_mute);
+ break;
+
+ case YM_MIC_LVL:
+ if (cp->un.value.num_channels != 1)
+ error = EINVAL;
+
+ ad1848_to_vol(cp, &vol);
+ ym_set_mic_gain(sc, &vol);
+ break;
+
+ case YM_MIC_MUTE:
+ sc->mic_mute = (cp->un.ord != 0);
+ ym_mute(sc, SA3_MIC, sc->mic_mute);
+ break;
+
+ default:
+ return ENXIO;
+ /*NOTREACHED*/
+ }
+
+ return (error);
+}
+
+int
+ym_mixer_get_port(addr, cp)
+ void *addr;
+ mixer_ctrl_t *cp;
+{
+ struct ad1848_softc *ac = addr;
+ struct ym_softc *sc = ac->parent;
+
+ int error = ad1848_mixer_get_port(ac, mappings, nummap, cp);
+
+ if (error != ENXIO)
+ return (error);
+
+ error = 0;
+
+ switch (cp->dev) {
+ case YM_OUTPUT_LVL:
+ ad1848_from_vol(cp, &sc->master_gain);
+ break;
+
+ case YM_OUTPUT_MUTE:
+ cp->un.ord = sc->master_mute;
+ break;
+
+ case YM_MIC_LVL:
+ if (cp->un.value.num_channels != 1)
+ error = EINVAL;
+
+ ad1848_from_vol(cp, &sc->mic_gain);
+ break;
+
+ case YM_MIC_MUTE:
+ cp->un.ord = sc->mic_mute;
+ break;
+
+ default:
+ error = ENXIO;
+ break;
+ }
+
+ return(error);
+}
+
+static char *mixer_classes[] = { AudioCinputs, AudioCrecord, AudioCoutputs,
+ AudioCmonitor };
+
+int
+ym_query_devinfo(addr, dip)
+ void *addr;
+ mixer_devinfo_t *dip;
+{
+ static char *mixer_port_names[] = { AudioNmidi, AudioNcd, AudioNdac,
+ AudioNline, AudioNspeaker,
+ AudioNmicrophone,
+ AudioNmonitor};
+
+ dip->next = dip->prev = AUDIO_MIXER_LAST;
+
+ switch(dip->index) {
+ case YM_INPUT_CLASS: /* input class descriptor */
+ case YM_OUTPUT_CLASS:
+ case YM_MONITOR_CLASS:
+ case YM_RECORD_CLASS:
+ dip->type = AUDIO_MIXER_CLASS;
+ dip->mixer_class = dip->index;
+ strcpy(dip->label.name,
+ mixer_classes[dip->index - YM_INPUT_CLASS]);
+ break;
+
+ case YM_MIDI_LVL:
+ case YM_CD_LVL:
+ case YM_DAC_LVL:
+ case YM_LINE_LVL:
+ case YM_SPEAKER_LVL:
+ case YM_MIC_LVL:
+ case YM_MONITOR_LVL:
+ dip->type = AUDIO_MIXER_VALUE;
+ if (dip->index == YM_MONITOR_LVL)
+ dip->mixer_class = YM_MONITOR_CLASS;
+ else
+ dip->mixer_class = YM_INPUT_CLASS;
+
+ dip->next = dip->index + 7;
+
+ strcpy(dip->label.name,mixer_port_names[dip->index - YM_MIDI_LVL]);
+
+ if (dip->index == YM_SPEAKER_LVL ||
+ dip->index == YM_MIC_LVL)
+ dip->un.v.num_channels = 1;
+ else
+ dip->un.v.num_channels = 2;
+
+ strcpy(dip->un.v.units.name, AudioNvolume);
+ break;
+
+ case YM_MIDI_MUTE:
+ case YM_CD_MUTE:
+ case YM_DAC_MUTE:
+ case YM_LINE_MUTE:
+ case YM_SPEAKER_MUTE:
+ case YM_MIC_MUTE:
+ case YM_MONITOR_MUTE:
+ if (dip->index == YM_MONITOR_MUTE)
+ dip->mixer_class = YM_MONITOR_CLASS;
+ else
+ dip->mixer_class = YM_INPUT_CLASS;
+ dip->type = AUDIO_MIXER_ENUM;
+ dip->prev = dip->index - 7;
+mute:
+ strcpy(dip->label.name, AudioNmute);
+ dip->un.e.num_mem = 2;
+ strcpy(dip->un.e.member[0].label.name, AudioNoff);
+ dip->un.e.member[0].ord = 0;
+ strcpy(dip->un.e.member[1].label.name, AudioNon);
+ dip->un.e.member[1].ord = 1;
+ break;
+
+
+ case YM_OUTPUT_LVL:
+ dip->type = AUDIO_MIXER_VALUE;
+ dip->mixer_class = YM_OUTPUT_CLASS;
+ dip->next = YM_OUTPUT_MUTE;
+ strcpy(dip->label.name, AudioNmaster);
+ dip->un.v.num_channels = 2;
+ strcpy(dip->un.v.units.name, AudioNvolume);
+ break;
+
+ case YM_OUTPUT_MUTE:
+ dip->mixer_class = YM_OUTPUT_CLASS;
+ dip->type = AUDIO_MIXER_ENUM;
+ dip->prev = YM_OUTPUT_LVL;
+ goto mute;
+
+ case YM_REC_LVL: /* record level */
+ dip->type = AUDIO_MIXER_VALUE;
+ dip->mixer_class = YM_RECORD_CLASS;
+ dip->next = YM_RECORD_SOURCE;
+ strcpy(dip->label.name, AudioNrecord);
+ dip->un.v.num_channels = 2;
+ strcpy(dip->un.v.units.name, AudioNvolume);
+ break;
+
+
+ case YM_RECORD_SOURCE:
+ dip->mixer_class = YM_RECORD_CLASS;
+ dip->type = AUDIO_MIXER_ENUM;
+ dip->prev = YM_REC_LVL;
+ strcpy(dip->label.name, AudioNsource);
+ dip->un.e.num_mem = 4;
+ strcpy(dip->un.e.member[0].label.name, AudioNmicrophone);
+ dip->un.e.member[0].ord = MIC_IN_PORT;
+ strcpy(dip->un.e.member[1].label.name, AudioNline);
+ dip->un.e.member[1].ord = LINE_IN_PORT;
+ strcpy(dip->un.e.member[2].label.name, AudioNdac);
+ dip->un.e.member[2].ord = DAC_IN_PORT;
+ strcpy(dip->un.e.member[3].label.name, AudioNcd);
+ dip->un.e.member[3].ord = AUX1_IN_PORT;
+ break;
+
+ default:
+ return ENXIO;
+ /*NOTREACHED*/
+ }
+
+ return 0;
+}
diff --git a/sys/dev/isa/ym_isapnp.c b/sys/dev/isa/ym_isapnp.c
new file mode 100644
index 00000000000..b0f41d1affb
--- /dev/null
+++ b/sys/dev/isa/ym_isapnp.c
@@ -0,0 +1,126 @@
+/* $OpenBSD: ym_isapnp.c,v 1.1 1998/05/08 18:37:25 csapuntz Exp $ */
+
+
+/*
+ * Copyright (c) 1991-1993 Regents of the University of California.
+ * 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.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the Computer Systems
+ * Engineering Group at Lawrence Berkeley Laboratory.
+ * 4. Neither the name of the University nor of the Laboratory may be used
+ * to endorse or promote products derived from this software without
+ * specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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.
+ *
+ */
+
+/*
+ * Driver for the Yamaha OPL3-SA3 chipset. This is found on many laptops
+ * and Pentium (II) motherboards.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/errno.h>
+#include <sys/ioctl.h>
+#include <sys/syslog.h>
+#include <sys/device.h>
+#include <sys/proc.h>
+
+#include <sys/audioio.h>
+#include <dev/audio_if.h>
+
+#include <dev/isa/isavar.h>
+#include <dev/isa/isadmavar.h>
+
+#include <dev/ic/ad1848reg.h>
+#include <dev/isa/ad1848var.h>
+
+#include <dev/ic/cs4231reg.h>
+#include <dev/isa/cs4231var.h>
+
+#include <dev/isa/wssreg.h>
+#include <dev/isa/ymvar.h>
+
+int ym_isapnp_match __P((struct device *, void *, void *));
+void ym_isapnp_attach __P((struct device *, struct device *, void *));
+
+struct cfattach ym_isapnp_ca = {
+ sizeof(struct ym_softc), ym_isapnp_match, ym_isapnp_attach
+};
+
+
+/*
+ * Probe / attach routines.
+ */
+
+/*
+ * Probe for the soundblaster hardware.
+ */
+int
+ym_isapnp_match(parent, match, aux)
+ struct device *parent;
+ void *match, *aux;
+{
+
+ return 1;
+}
+
+/*
+ * Attach hardware to driver, attach hardware driver to audio
+ * pseudo-device driver.
+ */
+void
+ym_isapnp_attach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ struct ym_softc *sc = (struct ym_softc *)self;
+ struct isa_attach_args *ia = aux;
+
+ if (ia->ipa_nio < 5) {
+ printf ("Insufficient I/O ports... not really attached\n");
+ return;
+ }
+
+ sc->sc_iot = ia->ia_iot;
+ sc->sc_ioh = ia->ipa_io[1].h;
+ sc->sc_ic = ia->ia_ic;
+
+ sc->ym_irq = ia->ipa_irq[0].num;
+ sc->ym_drq = ia->ipa_drq[0].num;
+ sc->ym_recdrq = ia->ipa_drq[1].num;
+
+ sc->sc_controlioh = ia->ipa_io[4].h;
+
+ sc->sc_ad1848.sc_iobase = ia->ipa_io[1].base + WSS_CODEC;
+ sc->sc_ad1848.sc_isa = parent->dv_parent;
+ sc->sc_ad1848.sc_iot = sc->sc_iot;
+ sc->sc_ad1848.sc_iooffs = WSS_CODEC;
+ sc->sc_ad1848.mode = 2;
+ sc->sc_ad1848.MCE_bit = MODE_CHANGE_ENABLE;
+
+ ym_attach(sc);
+}
+
diff --git a/sys/dev/isa/ymvar.h b/sys/dev/isa/ymvar.h
new file mode 100644
index 00000000000..6981dd46e8c
--- /dev/null
+++ b/sys/dev/isa/ymvar.h
@@ -0,0 +1,89 @@
+/* $OpenBSD: ymvar.h,v 1.1 1998/05/08 18:37:25 csapuntz Exp $ */
+/* $NetBSD: wssvar.h,v 1.1 1998/01/19 22:18:25 augustss Exp $ */
+
+/*
+ * Copyright (c) 1994 John Brezak
+ * Copyright (c) 1991-1993 Regents of the University of California.
+ * 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.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the Computer Systems
+ * Engineering Group at Lawrence Berkeley Laboratory.
+ * 4. Neither the name of the University nor of the Laboratory may be used
+ * to endorse or promote products derived from this software without
+ * specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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.
+ *
+ */
+
+/*
+ * Mixer devices
+ */
+#define YM_MIDI_LVL 0
+#define YM_CD_LVL 1
+#define YM_DAC_LVL 2
+#define YM_LINE_LVL 3
+#define YM_SPEAKER_LVL 4
+#define YM_MIC_LVL 5
+#define YM_MONITOR_LVL 6
+#define YM_MIDI_MUTE 7
+#define YM_CD_MUTE 8
+#define YM_DAC_MUTE 9
+#define YM_LINE_MUTE 10
+#define YM_SPEAKER_MUTE 11
+#define YM_MIC_MUTE 12
+#define YM_MONITOR_MUTE 13
+
+#define YM_REC_LVL 14
+#define YM_RECORD_SOURCE 15
+
+#define YM_OUTPUT_LVL 16
+#define YM_OUTPUT_MUTE 17
+
+/* Classes - don't change this without looking at mixer_classes array */
+#define YM_INPUT_CLASS 18
+#define YM_RECORD_CLASS 19
+#define YM_OUTPUT_CLASS 20
+#define YM_MONITOR_CLASS 21
+
+
+struct ym_softc {
+ struct device sc_dev; /* base device */
+ struct isadev sc_id; /* ISA device */
+ void *sc_ih; /* interrupt vectoring */
+ bus_space_tag_t sc_iot; /* tag */
+ bus_space_handle_t sc_ioh; /* handle */
+ isa_chipset_tag_t sc_ic;
+
+ bus_space_handle_t sc_controlioh;
+
+ struct ad1848_softc sc_ad1848;
+#define ym_irq sc_ad1848.sc_irq
+#define ym_drq sc_ad1848.sc_drq
+#define ym_recdrq sc_ad1848.sc_recdrq
+
+ int master_mute, mic_mute;
+ struct ad1848_volume mic_gain, master_gain;
+};
+
+void ym_attach __P((struct ym_softc *));