summaryrefslogtreecommitdiff
path: root/sys/dev/ic/ac97.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/ac97.c')
-rw-r--r--sys/dev/ic/ac97.c278
1 files changed, 155 insertions, 123 deletions
diff --git a/sys/dev/ic/ac97.c b/sys/dev/ic/ac97.c
index 01baff36dbb..9dba6ff8fbc 100644
--- a/sys/dev/ic/ac97.c
+++ b/sys/dev/ic/ac97.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ac97.c,v 1.14 2001/05/15 22:43:20 deraadt Exp $ */
+/* $OpenBSD: ac97.c,v 1.15 2001/05/16 05:32:42 mickey Exp $ */
/*
* Copyright (c) 1999, 2000 Constantine Sapuntzakis
@@ -70,25 +70,25 @@
#include <dev/audio_if.h>
#include <dev/ic/ac97.h>
-static const struct audio_mixer_enum ac97_on_off = {
+const struct audio_mixer_enum ac97_on_off = {
2,
{ { { AudioNoff } , 0 },
{ { AudioNon } , 1 } }
};
-static const struct audio_mixer_enum ac97_mic_select = {
+const struct audio_mixer_enum ac97_mic_select = {
2,
{ { { AudioNmicrophone "0" }, 0 },
{ { AudioNmicrophone "1" }, 1 } }
};
-static const struct audio_mixer_enum ac97_mono_select = {
+const struct audio_mixer_enum ac97_mono_select = {
2,
{ { { AudioNmixerout }, 0 },
{ { AudioNmicrophone }, 1 } }
};
-static const struct audio_mixer_enum ac97_source = {
+const struct audio_mixer_enum ac97_source = {
8,
{ { { AudioNmicrophone } , 0 },
{ { AudioNcd }, 1 },
@@ -100,12 +100,12 @@ static const struct audio_mixer_enum ac97_source = {
{ { "phone" }, 7 }}
};
-static const struct audio_mixer_value ac97_volume_stereo = {
+const struct audio_mixer_value ac97_volume_stereo = {
{ AudioNvolume },
2
};
-static const struct audio_mixer_value ac97_volume_mono = {
+const struct audio_mixer_value ac97_volume_mono = {
{ AudioNvolume },
1
};
@@ -113,24 +113,24 @@ static const struct audio_mixer_value ac97_volume_mono = {
#define WRAP(a) &a, sizeof(a)
const struct ac97_source_info {
- const char *class;
- const char *device;
- const char *qualifier;
+ char *class;
+ char *device;
+ char *qualifier;
int type;
const void *info;
- int info_size;
+ int16_t info_size;
u_int8_t reg;
- u_int16_t default_value;
u_int8_t bits:3;
u_int8_t ofs:4;
u_int8_t mute:1;
u_int8_t polarity:1; /* Does 0 == MAX or MIN */
+ u_int16_t default_value;
- int prev;
- int next;
- int mixer_class;
+ int16_t prev;
+ int16_t next;
+ int16_t mixer_class;
} source_info[] = {
{
AudioCinputs, NULL, NULL, AUDIO_MIXER_CLASS,
@@ -142,106 +142,106 @@ const struct ac97_source_info {
/* Stereo master volume*/
AudioCoutputs, AudioNmaster, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_MASTER_VOLUME, 0x8000, 5, 0, 1,
+ AC97_REG_MASTER_VOLUME, 5, 0, 1, 0, 0x8000
}, {
/* Mono volume */
AudioCoutputs, AudioNmono, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_MASTER_VOLUME_MONO, 0x8000, 6, 0, 1,
+ AC97_REG_MASTER_VOLUME_MONO, 6, 0, 1, 0, 0x8000
}, {
AudioCoutputs, AudioNmono, AudioNsource, AUDIO_MIXER_ENUM,
WRAP(ac97_mono_select),
- AC97_REG_GP, 0x0000, 1, 9, 0,
+ AC97_REG_GP, 1, 9, 0, 0, 0x0000
}, {
/* Headphone volume */
AudioCoutputs, AudioNheadphone, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_HEADPHONE_VOLUME, 0x8000, 6, 0, 1,
+ AC97_REG_HEADPHONE_VOLUME, 6, 0, 1, 0, 0x8000
}, {
/* Tone */
AudioCoutputs, "tone", NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_MASTER_TONE, 0x0f0f, 4, 0, 0,
+ AC97_REG_MASTER_TONE, 4, 0, 0, 0, 0x0f0f
}, {
/* PC Beep Volume */
AudioCinputs, AudioNspeaker, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_PCBEEP_VOLUME, 0x0000, 4, 1, 1,
+ AC97_REG_PCBEEP_VOLUME, 4, 1, 1, 0x0000
}, {
/* Phone */
AudioCinputs, "phone", NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_PHONE_VOLUME, 0x8008, 5, 0, 1,
+ AC97_REG_PHONE_VOLUME, 5, 0, 1, 0, 0x8008
}, {
/* Mic Volume */
AudioCinputs, AudioNmicrophone, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_MIC_VOLUME, 0x8008, 5, 0, 1,
+ AC97_REG_MIC_VOLUME, 5, 0, 1, 0, 0x8008
}, {
AudioCinputs, AudioNmicrophone, AudioNpreamp, AUDIO_MIXER_ENUM,
WRAP(ac97_on_off),
- AC97_REG_MIC_VOLUME, 0x8008, 1, 6, 0,
+ AC97_REG_MIC_VOLUME, 1, 6, 0, 0, 0x8008
}, {
AudioCinputs, AudioNmicrophone, AudioNsource, AUDIO_MIXER_ENUM,
WRAP(ac97_mic_select),
- AC97_REG_GP, 0x0000, 1, 8, 0,
+ AC97_REG_GP, 1, 8, 0, 0x0000
}, {
/* Line in Volume */
AudioCinputs, AudioNline, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_LINEIN_VOLUME, 0x8808, 5, 0, 1,
+ AC97_REG_LINEIN_VOLUME, 5, 0, 1, 0, 0x8808
}, {
/* CD Volume */
AudioCinputs, AudioNcd, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_CD_VOLUME, 0x8808, 5, 0, 1,
+ AC97_REG_CD_VOLUME, 5, 0, 1, 0, 0x8808
}, {
/* Video Volume */
AudioCinputs, "video", NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_VIDEO_VOLUME, 0x8808, 5, 0, 1,
+ AC97_REG_VIDEO_VOLUME, 5, 0, 1, 0, 0x8808
}, {
/* AUX volume */
AudioCinputs, AudioNaux, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_AUX_VOLUME, 0x8808, 5, 0, 1,
+ AC97_REG_AUX_VOLUME, 5, 0, 1, 0, 0x8808
}, {
/* PCM out volume */
AudioCinputs, AudioNdac, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_PCMOUT_VOLUME, 0x8808, 5, 0, 1,
+ AC97_REG_PCMOUT_VOLUME, 5, 0, 1, 0, 0x8808
}, {
/* Record Source - some logic for this is hard coded - see below */
AudioCrecord, AudioNsource, NULL, AUDIO_MIXER_ENUM,
WRAP(ac97_source),
- AC97_REG_RECORD_SELECT, 0x0000, 3, 0, 0,
+ AC97_REG_RECORD_SELECT, 3, 0, 0, 0, 0x0000
}, {
/* Record Gain */
AudioCrecord, AudioNvolume, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_stereo),
- AC97_REG_RECORD_GAIN, 0x8000, 4, 0, 1,
+ AC97_REG_RECORD_GAIN, 4, 0, 1, 0, 0x8000
}, {
/* Record Gain mic */
AudioCrecord, AudioNmicrophone, NULL, AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_RECORD_GAIN_MIC, 0x8000, 4, 0, 1, 1,
+ AC97_REG_RECORD_GAIN_MIC, 4, 0, 1, 1, 0x8000
}, {
/* */
AudioCoutputs, AudioNloudness, NULL, AUDIO_MIXER_ENUM,
WRAP(ac97_on_off),
- AC97_REG_GP, 0x0000, 1, 12, 0,
+ AC97_REG_GP, 1, 12, 0, 0, 0x0000
}, {
AudioCoutputs, AudioNspatial, NULL, AUDIO_MIXER_ENUM,
WRAP(ac97_on_off),
- AC97_REG_GP, 0x0000, 1, 13, 0,
+ AC97_REG_GP, 1, 13, 0, 0, 0x0000
}, {
AudioCoutputs, AudioNspatial, "center", AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_3D_CONTROL, 0x0000, 4, 8, 0, 1,
+ AC97_REG_3D_CONTROL, 4, 8, 0, 1, 0x0000
}, {
AudioCoutputs, AudioNspatial, "depth", AUDIO_MIXER_VALUE,
WRAP(ac97_volume_mono),
- AC97_REG_3D_CONTROL, 0x0000, 4, 0, 0, 1,
+ AC97_REG_3D_CONTROL, 4, 0, 0, 1, 0x0000
},
/* Missing features: Simulated Stereo, POP, Loopback mode */
@@ -278,45 +278,72 @@ struct ac97_codec_if_vtbl ac97civ = {
ac97_restore_shadow
};
-static const struct ac97_codecid {
+const struct ac97_codecid {
+ u_int8_t id;
+ u_int8_t mask;
+ u_int8_t rev;
+ u_int8_t shift; /* no use yet */
+ char *const name;
+} ac97_ad[] = {
+ { 0x40, 0xff, 0, 0, "AD1881" },
+ { 0x60, 0xff, 0, 0, "AD1885" },
+}, ac97_ak[] = {
+ { 0x00, 0xfe, 1, 0, "AK4540" },
+ { 0x01, 0xfe, 1, 0, "AK4540" },
+ { 0x02, 0xff, 0, 0, "AK4543" },
+}, ac97_av[] = {
+ { 0x10, 0xff, 0, 0, "ALC200" },
+}, ac97_cs[] = {
+ { 0x00, 0xf8, 7, 0, "CS4297" },
+ { 0x10, 0xf8, 7, 0, "CS4297A" },
+ { 0x20, 0xf8, 7, 0, "CS4298" },
+ { 0x28, 0xf8, 7, 0, "CS4294" },
+ { 0x30, 0xf8, 7, 0, "CS4299" },
+}, ac97_ns[] = {
+ { 0x31, 0xff, 0, 0, "LM4549" },
+ { 0x00, 0x00 }
+}, ac97_sl[] = {
+ { 0x22, 0xff, 0, 0, "Si3036" },
+ { 0x23, 0xff, 0, 0, "Si3038" },
+}, ac97_st[] = {
+ { 0x00, 0xff, 0, 0, "STAC9700" },
+ { 0x04, 0xff, 0, 0, "STAC970[135]" },
+ { 0x05, 0xff, 0, 0, "STAC9704" },
+ { 0x08, 0xff, 0, 0, "STAC9708" },
+ { 0x09, 0xff, 0, 0, "STAC9721/23" },
+ { 0x44, 0xff, 0, 0, "STAC9744/45" },
+ { 0x56, 0xff, 0, 0, "STAC9756/57" },
+ { 0x84, 0xff, 0, 0, "STAC9784/85" },
+}, ac97_tt[] = {
+ { 0x02, 0xff, 0, 0, "TR28022" },
+ { 0x03, 0xff, 0, 0, "TR28023" },
+ { 0x08, 0xff, 0, 0, "TR28028" },
+ { 0x23, 0xff, 0, 0, "unknown" },
+}, ac97_wo[] = {
+ { 0x00, 0xff, 0, 0, "WM9701A" },
+ { 0x03, 0xff, 0, 0, "WM9704M/Q-0" }, /* also WM9703 */
+ { 0x04, 0xff, 0, 0, "WM9704M/Q-1" },
+};
+
+#define cl(n) n, sizeof(n)/sizeof(n[0])
+const struct ac97_vendorid {
u_int32_t id;
char * const name;
-} ac97codecid[] = {
- { 0x41445340, "Analog Devices AD1881" },
- { 0x41445360, "Analog Devices AD1885" },
- { 0x414B4D00, "Asahi Kasei AK4540 rev 0" },
- { 0x414B4D01, "Asahi Kasei AK4540 rev 1" },
- { 0x414B4D02, "Asahi Kasei AK4543" },
- { 0x414c4710, "Avance ALC200" },
- { 0x43525900, "Cirrus Logic CS4297" },
- { 0x43525903, "Cirrus Logic CS4297" },
- { 0x43525913, "Cirrus Logic CS4297A" },
- { 0x43525914, "Cirrus Logic CS4297A?" },
- { 0x43525923, "Cirrus Logic CS4298" },
- { 0x4352592b, "Cirrus Logic CS4294" },
- { 0x43525931, "Cirrus Logic CS4299" },
- { 0x43525933, "Cirrus Logic CS4298A?" },
- { 0x43525934, "Cirrus Logic CS4299" },
- { 0x4e534331, "National Semiconductor LM4549" },
- { 0x53494c22, "Silicon Laboratory Si3036" },
- { 0x53494c23, "Silicon Laboratory Si3038" },
- { 0x54524102, "TriTech TR28022" },
- { 0x54524103, "TriTech TR28023" },
- { 0x54524108, "TriTech TR28028" },
- { 0x54524123, "TriTech unknown" },
- { 0x574d4c00, "Wolfson WM9704" },
- { 0x574d4c03, "Wolfson WM9707" },
- { 0x83847600, "SigmaTel STAC9700" },
- { 0x83847604, "SigmaTel STAC9701/3/4/5" },
- { 0x83847605, "SigmaTel STAC9704" },
- { 0x83847608, "SigmaTel STAC9708" },
- { 0x83847609, "SigmaTel STAC9721/23" },
- { 0x83847644, "SigmaTel STAC9744/45" },
- { 0x83847684, "SigmaTel STAC9783/84?" },
- { 0, NULL }
+ const struct ac97_codecid *const codecs;
+ u_int8_t num;
+} ac97_vendors[] = {
+ { 0x41445300, "Analog Devices", cl(ac97_ad) },
+ { 0x414B4D00, "Asahi Kasei", cl(ac97_ak) },
+ { 0x414c4700, "Avance", cl(ac97_av) },
+ { 0x43525900, "Cirrus Logic", cl(ac97_cs) },
+ { 0x4e534300, "National Semiconductor", cl(ac97_ns) },
+ { 0x53494c00, "Silicon Laboratory", cl(ac97_sl) },
+ { 0x54524100, "TriTech Microelectronics", cl(ac97_tt) },
+ { 0x574d4c00, "Wolfson", cl(ac97_wo) },
+ { 0x83847600, "SigmaTel", cl(ac97_st) },
};
-static const char * const ac97enhancement[] = {
+const char * const ac97enhancement[] = {
"No 3D Stereo",
"Analog Devices Phat Stereo",
"Creative"
@@ -351,7 +378,7 @@ static const char * const ac97enhancement[] = {
"Unknown 3D",
};
-static const char * const ac97feature[] = {
+const char * const ac97feature[] = {
"mic channel",
"reserved",
"tone",
@@ -422,13 +449,13 @@ void
ac97_setup_defaults(as)
struct ac97_softc *as;
{
- const struct ac97_source_info *si;
int idx;
bzero(as->shadow_reg, sizeof(as->shadow_reg));
for (idx = 0; idx < SOURCE_INFO_SIZE; idx++) {
- si = &source_info[idx];
+ const struct ac97_source_info *si = &source_info[idx];
+
ac97_write(as, si->reg, si->default_value);
}
}
@@ -438,11 +465,10 @@ ac97_restore_shadow(self)
struct ac97_codec_if *self;
{
struct ac97_softc *as = (struct ac97_softc *)self;
- const struct ac97_source_info *si;
int idx;
for (idx = 0; idx < SOURCE_INFO_SIZE; idx++) {
- si = &source_info[idx];
+ const struct ac97_source_info *si = &source_info[idx];
ac97_write(as, si->reg, as->shadow_reg[si->reg >> 1]);
}
@@ -459,8 +485,8 @@ void
ac97_setup_source_info(as)
struct ac97_softc *as;
{
- int idx, ouridx;
struct ac97_source_info *si, *si2;
+ int idx, ouridx;
for (idx = 0, ouridx = 0; idx < SOURCE_INFO_SIZE; idx++) {
si = &as->source_info[ouridx];
@@ -520,10 +546,7 @@ ac97_setup_source_info(as)
/* Setup prev and next pointers */
- if (si->prev != 0)
- continue;
-
- if (si->qualifier)
+ if (si->prev != 0 || si->qualifier)
continue;
si->prev = AUDIO_MIXER_LAST;
@@ -554,22 +577,22 @@ ac97_attach(host_if)
struct ac97_host_if *host_if;
{
struct ac97_softc *as;
- int error, i, j;
u_int16_t id1, id2, caps;
u_int32_t id;
mixer_ctrl_t ctl;
+ int error, i;
as = malloc(sizeof(struct ac97_softc), M_DEVBUF, M_WAITOK);
+ if (!as)
+ return (ENOMEM);
- if (!as) return (ENOMEM);
-
- bzero (as, sizeof(*as));
+ bzero(as, sizeof(*as));
as->codec_if.vtbl = &ac97civ;
as->host_if = host_if;
if ((error = host_if->attach(host_if->arg, &as->codec_if))) {
- free (as, M_DEVBUF);
+ free(as, M_DEVBUF);
return (error);
}
@@ -589,25 +612,44 @@ ac97_attach(host_if)
ac97_read(as, AC97_REG_RESET, &caps);
id = (id1 << 16) | id2;
-
if (id) {
+ register const struct ac97_vendorid *vendor;
+ register const struct ac97_codecid *codec;
+
printf("ac97: codec id 0x%08x", id);
- for (i = 0; ac97codecid[i].id; i++) {
- if (ac97codecid[i].id == id)
- printf(" (%s)", ac97codecid[i].name);
- }
- printf("\nac97: codec features ");
- for (i = j = 0; i < 10; i++) {
- if (caps & (1 << i)) {
- printf("%s%s", j? ", " : "", ac97feature[i]);
- j++;
+ for (vendor = &ac97_vendors[sizeof(ac97_vendors) /
+ sizeof(ac97_vendors[0]) - 1];
+ vendor >= ac97_vendors; vendor--) {
+ if (vendor->id == (id & AC97_VENDOR_ID_MASK)) {
+ printf(" (%s", vendor->name);
+ for (codec = &vendor->codecs[vendor->num-1];
+ codec >= vendor->codecs; codec--) {
+ if (codec->id == (id & codec->mask))
+ break;
+ }
+ if (codec->mask)
+ printf(" %s", codec->name);
+ else
+ printf(" <%2x>", id & codec->mask);
+ if (codec->rev)
+ printf(" rev %d", id & codec->rev);
+ printf(")");
+ break;
}
}
- printf("%s%s\n", j? ", " : "",
- ac97enhancement[(caps >> 10) & 0x1f]);
+ printf("\n");
} else
printf("ac97: codec id not read\n");
+ if (caps) {
+ printf("ac97: codec features ");
+ for (i = 0; i < 10; i++) {
+ if (caps & (1 << i))
+ printf("%s, ", ac97feature[i]);
+ }
+ printf("%s\n", ac97enhancement[AC97_SOUND_ENHANCEMENT(caps)]);
+ }
+
ac97_setup_source_info(as);
/* Just enable the DAC and master volumes by default */
@@ -618,19 +660,19 @@ ac97_attach(host_if)
ctl.dev = ac97_get_portnum_by_name(&as->codec_if, AudioCoutputs,
AudioNmaster, AudioNmute);
ac97_mixer_set_port(&as->codec_if, &ctl);
+
ctl.dev = ac97_get_portnum_by_name(&as->codec_if, AudioCinputs,
AudioNdac, AudioNmute);
-
ac97_mixer_set_port(&as->codec_if, &ctl);
+
ctl.dev = ac97_get_portnum_by_name(&as->codec_if, AudioCrecord,
AudioNvolume, AudioNmute);
ac97_mixer_set_port(&as->codec_if, &ctl);
-
- ctl.dev = ac97_get_portnum_by_name(&as->codec_if, AudioCrecord,
- AudioNsource, NULL);
ctl.type = AUDIO_MIXER_ENUM;
ctl.un.ord = 0;
+ ctl.dev = ac97_get_portnum_by_name(&as->codec_if, AudioCrecord,
+ AudioNsource, NULL);
ac97_mixer_set_port(&as->codec_if, &ctl);
return (0);
@@ -670,8 +712,6 @@ ac97_query_devinfo(codec_if, dip)
return (ENXIO);
}
-
-
int
ac97_mixer_set_port(codec_if, cp)
struct ac97_codec_if *codec_if;
@@ -683,10 +723,8 @@ ac97_mixer_set_port(codec_if, cp)
u_int16_t val, newval;
int error;
- if (cp->dev < 0 || cp->dev >= as->num_source_info)
- return (EINVAL);
-
- if (cp->type != si->type)
+ if (cp->dev < 0 || cp->dev >= as->num_source_info ||
+ cp->type != si->type)
return (EINVAL);
ac97_read(as, si->reg, &val);
@@ -727,8 +765,8 @@ ac97_mixer_set_port(codec_if, cp)
r = 255 - r;
}
- l = l >> (8 - si->bits);
- r = r >> (8 - si->bits);
+ l >>= 8 - si->bits;
+ r >>= 8 - si->bits;
newval = ((l & mask) << si->ofs);
if (value->num_channels == 2) {
@@ -779,10 +817,8 @@ ac97_mixer_get_port(codec_if, cp)
u_int16_t mask;
u_int16_t val;
- if (cp->dev < 0 || cp->dev >= as->num_source_info)
- return (EINVAL);
-
- if (cp->type != si->type)
+ if (cp->dev < 0 || cp->dev >= as->num_source_info ||
+ cp->type != si->type)
return (EINVAL);
ac97_read(as, si->reg, &val);
@@ -806,15 +842,12 @@ ac97_mixer_get_port(codec_if, cp)
(cp->un.value.num_channels > value->num_channels))
return (EINVAL);
- if (value->num_channels == 1) {
- l = r = (val >> si->ofs) & mask;
- } else {
- l = (val >> si->ofs) & mask;
+ l = r = (val >> si->ofs) & mask;
+ if (value->num_channels > 1)
r = (val >> (si->ofs + 8)) & mask;
- }
- l = (l << (8 - si->bits));
- r = (r << (8 - si->bits));
+ l <<= 8 - si->bits;
+ r <<= 8 - si->bits;
if (!si->polarity) {
l = 255 - l;
r = 255 - r;
@@ -840,4 +873,3 @@ ac97_mixer_get_port(codec_if, cp)
return (0);
}
-