xref: /freebsd/sys/dev/sound/pcm/ac97.c (revision 2b743a9e9ddc6736208dc8ca1ce06ce64ad20a19)
1 /*-
2  * Copyright (c) 1999 Cameron Grant <cg@freebsd.org>
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  */
26 
27 #include <dev/sound/pcm/sound.h>
28 #include <dev/sound/pcm/ac97.h>
29 #include <dev/sound/pcm/ac97_patch.h>
30 
31 #include <dev/pci/pcivar.h>
32 
33 #include "mixer_if.h"
34 
35 SND_DECLARE_FILE("$FreeBSD$");
36 
37 MALLOC_DEFINE(M_AC97, "ac97", "ac97 codec");
38 
39 struct ac97mixtable_entry {
40 	int	 reg:8;		/* register index		*/
41 				/* reg < 0 if inverted polarity	*/
42 	unsigned bits:4;	/* width of control field	*/
43 	unsigned ofs:4;		/* offset (only if stereo=0)	*/
44 	unsigned stereo:1;	/* set for stereo controls	*/
45 	unsigned mute:1;	/* bit15 is MUTE		*/
46 	unsigned recidx:4;	/* index in rec mux		*/
47 	unsigned mask:1;	/* use only masked bits		*/
48 	unsigned enable:1;	/* entry is enabled		*/
49 };
50 
51 #define AC97_NAMELEN	16
52 struct ac97_info {
53 	kobj_t methods;
54 	device_t dev;
55 	void *devinfo;
56 	u_int32_t id;
57 	unsigned count, caps, se, extcaps, extid, extstat, noext:1;
58 	u_int32_t flags;
59 	struct ac97mixtable_entry mix[32];
60 	char name[AC97_NAMELEN];
61 	struct mtx *lock;
62 };
63 
64 struct ac97_vendorid {
65 	u_int32_t   id;
66 	const char *name;
67 };
68 
69 struct ac97_codecid {
70 	u_int32_t  id;
71 	u_int8_t   stepmask;
72 	u_int8_t   noext:1;
73 	char 	  *name;
74 	ac97_patch patch;
75 };
76 
77 static const struct ac97mixtable_entry ac97mixtable_default[32] = {
78     /*	[offset]			reg	     bits of st mu re mk en */
79 	[SOUND_MIXER_VOLUME]	= { AC97_MIX_MASTER, 	5, 0, 1, 1, 6, 0, 1 },
80 	[SOUND_MIXER_OGAIN]	= { AC97_MIX_AUXOUT, 	5, 0, 1, 1, 0, 0, 0 },
81 	[SOUND_MIXER_PHONEOUT]	= { AC97_MIX_MONO, 	5, 0, 0, 1, 7, 0, 0 },
82 	[SOUND_MIXER_BASS]	= { AC97_MIX_TONE, 	4, 8, 0, 0, 0, 1, 0 },
83 	[SOUND_MIXER_TREBLE]	= { AC97_MIX_TONE, 	4, 0, 0, 0, 0, 1, 0 },
84 	[SOUND_MIXER_PCM]	= { AC97_MIX_PCM, 	5, 0, 1, 1, 0, 0, 1 },
85 	[SOUND_MIXER_SPEAKER]	= { AC97_MIX_BEEP, 	4, 1, 0, 1, 0, 0, 0 },
86 	[SOUND_MIXER_LINE]	= { AC97_MIX_LINE, 	5, 0, 1, 1, 5, 0, 1 },
87 	[SOUND_MIXER_PHONEIN]	= { AC97_MIX_PHONE, 	5, 0, 0, 1, 8, 0, 0 },
88 	[SOUND_MIXER_MIC] 	= { AC97_MIX_MIC, 	5, 0, 0, 1, 1, 1, 1 },
89 	/* use igain for the mic 20dB boost */
90 	[SOUND_MIXER_IGAIN] 	= { -AC97_MIX_MIC, 	1, 6, 0, 0, 0, 1, 1 },
91 	[SOUND_MIXER_CD]	= { AC97_MIX_CD, 	5, 0, 1, 1, 2, 0, 1 },
92 	[SOUND_MIXER_LINE1]	= { AC97_MIX_AUX, 	5, 0, 1, 1, 4, 0, 0 },
93 	[SOUND_MIXER_VIDEO]	= { AC97_MIX_VIDEO, 	5, 0, 1, 1, 3, 0, 0 },
94 	[SOUND_MIXER_RECLEV]	= { -AC97_MIX_RGAIN, 	4, 0, 1, 1, 0, 0, 1 }
95 };
96 
97 static const struct ac97_vendorid ac97vendorid[] = {
98 	{ 0x41445300, "Analog Devices" },
99 	{ 0x414b4d00, "Asahi Kasei" },
100 	{ 0x414c4300, "Realtek" },
101 	{ 0x414c4700, "Avance Logic" },
102 	{ 0x43525900, "Cirrus Logic" },
103 	{ 0x434d4900, "C-Media Electronics" },
104 	{ 0x43585400, "Conexant" },
105 	{ 0x44543000, "Diamond Technology" },
106 	{ 0x454d4300, "eMicro" },
107 	{ 0x45838300, "ESS Technology" },
108 	{ 0x48525300, "Intersil" },
109 	{ 0x49434500, "ICEnsemble" },
110 	{ 0x49544500, "ITE, Inc." },
111 	{ 0x4e534300, "National Semiconductor" },
112 	{ 0x50534300, "Philips Semiconductor" },
113 	{ 0x83847600, "SigmaTel" },
114 	{ 0x53494c00, "Silicon Laboratories" },
115 	{ 0x54524100, "TriTech" },
116 	{ 0x54584e00, "Texas Instruments" },
117 	{ 0x56494100, "VIA Technologies" },
118 	{ 0x57454300, "Winbond" },
119 	{ 0x574d4c00, "Wolfson" },
120 	{ 0x594d4800, "Yamaha" },
121 	/*
122 	 * XXX This is a fluke, really! The real vendor
123 	 * should be SigmaTel, not this! This should be
124 	 * removed someday!
125 	 */
126 	{ 0x01408300, "Creative" },
127 	{ 0x00000000, NULL }
128 };
129 
130 static struct ac97_codecid ac97codecid[] = {
131 	{ 0x41445303, 0x00, 0, "AD1819",	0 },
132 	{ 0x41445340, 0x00, 0, "AD1881",	0 },
133 	{ 0x41445348, 0x00, 0, "AD1881A",	0 },
134 	{ 0x41445360, 0x00, 0, "AD1885",	0 },
135 	{ 0x41445361, 0x00, 0, "AD1886", 	ad1886_patch },
136 	{ 0x41445362, 0x00, 0, "AD1887", 	0 },
137 	{ 0x41445363, 0x00, 0, "AD1886A", 	0 },
138 	{ 0x41445368, 0x00, 0, "AD1888", 	ad198x_patch },
139 	{ 0x41445370, 0x00, 0, "AD1980",	ad198x_patch },
140 	{ 0x41445372, 0x00, 0, "AD1981A",	0 },
141 	{ 0x41445374, 0x00, 0, "AD1981B",	ad1981b_patch },
142 	{ 0x41445375, 0x00, 0, "AD1985",	ad198x_patch },
143 	{ 0x41445378, 0x00, 0, "AD1986",	ad198x_patch },
144 	{ 0x414b4d00, 0x00, 1, "AK4540", 	0 },
145 	{ 0x414b4d01, 0x00, 1, "AK4542", 	0 },
146 	{ 0x414b4d02, 0x00, 1, "AK4543", 	0 },
147 	{ 0x414b4d06, 0x00, 0, "AK4544A",	0 },
148 	{ 0x454b4d07, 0x00, 0, "AK4545",	0 },
149 	{ 0x414c4320, 0x0f, 0, "ALC100",	0 },
150 	{ 0x414c4730, 0x0f, 0, "ALC101",	0 },
151 	{ 0x414c4710, 0x0f, 0, "ALC200", 	0 },
152 	{ 0x414c4740, 0x0f, 0, "ALC202", 	0 },
153 	{ 0x414c4720, 0x0f, 0, "ALC650", 	0 },
154 	{ 0x414c4752, 0x0f, 0, "ALC250",	0 },
155 	{ 0x414c4760, 0x0f, 0, "ALC655",	0 },
156 	{ 0x414c4770, 0x0f, 0, "ALC203",	0 },
157 	{ 0x414c4780, 0x0f, 0, "ALC658",	0 },
158 	{ 0x414c4790, 0x0f, 0, "ALC850",	0 },
159 	{ 0x43525900, 0x07, 0, "CS4297", 	0 },
160 	{ 0x43525910, 0x07, 0, "CS4297A", 	0 },
161 	{ 0x43525920, 0x07, 0, "CS4294/98",	0 },
162 	{ 0x4352592d, 0x07, 0, "CS4294",	0 },
163 	{ 0x43525930, 0x07, 0, "CS4299",	0 },
164 	{ 0x43525940, 0x07, 0, "CS4201",	0 },
165 	{ 0x43525958, 0x07, 0, "CS4205",	0 },
166 	{ 0x43525960, 0x07, 0, "CS4291A",	0 },
167 	{ 0x434d4961, 0x00, 0, "CMI9739",	cmi9739_patch },
168 	{ 0x434d4941, 0x00, 0, "CMI9738",	0 },
169 	{ 0x434d4978, 0x00, 0, "CMI9761",	0 },
170 	{ 0x434d4982, 0x00, 0, "CMI9761",	0 },
171 	{ 0x434d4983, 0x00, 0, "CMI9761",	0 },
172 	{ 0x43585421, 0x00, 0, "HSD11246",	0 },
173 	{ 0x43585428, 0x07, 0, "CX20468",	0 },
174 	{ 0x43585430, 0x00, 0, "CX20468-21",	0 },
175 	{ 0x44543000, 0x00, 0, "DT0398",	0 },
176 	{ 0x454d4323, 0x00, 0, "EM28023",	0 },
177 	{ 0x454d4328, 0x00, 0, "EM28028",	0 },
178 	{ 0x45838308, 0x00, 0, "ES1988",	0 }, /* Formerly ES1921(?) */
179 	{ 0x48525300, 0x00, 0, "HMP9701",	0 },
180 	{ 0x49434501, 0x00, 0, "ICE1230",	0 },
181 	{ 0x49434511, 0x00, 0, "ICE1232",	0 },
182 	{ 0x49434514, 0x00, 0, "ICE1232A",	0 },
183 	{ 0x49434551, 0x03, 0, "VT1616",	0 }, /* Via badged ICE */
184 	{ 0x49544520, 0x00, 0, "ITE2226E",	0 },
185 	{ 0x49544560, 0x07, 0, "ITE2646E",	0 }, /* XXX: patch needed */
186 	{ 0x4e534340, 0x00, 0, "LM4540",	0 }, /* Spec blank on revid */
187 	{ 0x4e534343, 0x00, 0, "LM4543",	0 }, /* Ditto */
188 	{ 0x4e534346, 0x00, 0, "LM4546A",	0 },
189 	{ 0x4e534348, 0x00, 0, "LM4548A",	0 },
190 	{ 0x4e534331, 0x00, 0, "LM4549",	0 },
191 	{ 0x4e534349, 0x00, 0, "LM4549A",	0 },
192 	{ 0x4e534350, 0x00, 0, "LM4550",	0 },
193 	{ 0x50534301, 0x00, 0, "UCB1510",	0 },
194 	{ 0x50534304, 0x00, 0, "UCB1400",	0 },
195 	{ 0x83847600, 0x00, 0, "STAC9700/83/84",	0 },
196 	{ 0x83847604, 0x00, 0, "STAC9701/03/04/05", 0 },
197 	{ 0x83847605, 0x00, 0, "STAC9704",	0 },
198 	{ 0x83847608, 0x00, 0, "STAC9708/11",	0 },
199 	{ 0x83847609, 0x00, 0, "STAC9721/23",	0 },
200 	{ 0x83847644, 0x00, 0, "STAC9744/45",	0 },
201 	{ 0x83847650, 0x00, 0, "STAC9750/51",	0 },
202 	{ 0x83847652, 0x00, 0, "STAC9752/53",	0 },
203 	{ 0x83847656, 0x00, 0, "STAC9756/57",	0 },
204 	{ 0x83847658, 0x00, 0, "STAC9758/59",	0 },
205 	{ 0x83847660, 0x00, 0, "STAC9760/61",	0 }, /* Extrapolated */
206 	{ 0x83847662, 0x00, 0, "STAC9762/63",	0 }, /* Extrapolated */
207 	{ 0x83847666, 0x00, 0, "STAC9766/67",	0 },
208 	{ 0x53494c22, 0x00, 0, "Si3036",	0 },
209 	{ 0x53494c23, 0x00, 0, "Si3038",	0 },
210 	{ 0x54524103, 0x00, 0, "TR28023",	0 }, /* Extrapolated */
211 	{ 0x54524106, 0x00, 0, "TR28026",	0 },
212 	{ 0x54524108, 0x00, 0, "TR28028",	0 },
213 	{ 0x54524123, 0x00, 0, "TR28602",	0 },
214 	{ 0x54524e03, 0x07, 0, "TLV320AIC27",	0 },
215 	{ 0x54584e20, 0x00, 0, "TLC320AD90",	0 },
216 	{ 0x56494161, 0x00, 0, "VIA1612A",      0 },
217 	{ 0x56494170, 0x00, 0, "VIA1617A",      0 },
218 	{ 0x574d4c00, 0x00, 0, "WM9701A",	0 },
219 	{ 0x574d4c03, 0x00, 0, "WM9703/4/7/8",	0 },
220 	{ 0x574d4c04, 0x00, 0, "WM9704Q",	0 },
221 	{ 0x574d4c05, 0x00, 0, "WM9705/10",	0 },
222 	{ 0x574d4d09, 0x00, 0, "WM9709",	0 },
223 	{ 0x574d4c12, 0x00, 0, "WM9711/12",	0 }, /* XXX: patch needed */
224 	{ 0x57454301, 0x00, 0, "W83971D",	0 },
225 	{ 0x594d4800, 0x00, 0, "YMF743",	0 },
226 	{ 0x594d4802, 0x00, 0, "YMF752",	0 },
227 	{ 0x594d4803, 0x00, 0, "YMF753",	0 },
228 	/*
229 	 * XXX This is a fluke, really! The real codec
230 	 * should be STAC9704, not this! This should be
231 	 * removed someday!
232 	 */
233 	{ 0x01408384, 0x00, 0, "EV1938",	0 },
234 	{ 0, 0, 0, NULL, 0 }
235 };
236 
237 static char *ac97enhancement[] = {
238 	"no 3D Stereo Enhancement",
239 	"Analog Devices Phat Stereo",
240 	"Creative Stereo Enhancement",
241 	"National Semi 3D Stereo Enhancement",
242 	"Yamaha Ymersion",
243 	"BBE 3D Stereo Enhancement",
244 	"Crystal Semi 3D Stereo Enhancement",
245 	"Qsound QXpander",
246 	"Spatializer 3D Stereo Enhancement",
247 	"SRS 3D Stereo Enhancement",
248 	"Platform Tech 3D Stereo Enhancement",
249 	"AKM 3D Audio",
250 	"Aureal Stereo Enhancement",
251 	"Aztech 3D Enhancement",
252 	"Binaura 3D Audio Enhancement",
253 	"ESS Technology Stereo Enhancement",
254 	"Harman International VMAx",
255 	"Nvidea 3D Stereo Enhancement",
256 	"Philips Incredible Sound",
257 	"Texas Instruments 3D Stereo Enhancement",
258 	"VLSI Technology 3D Stereo Enhancement",
259 	"TriTech 3D Stereo Enhancement",
260 	"Realtek 3D Stereo Enhancement",
261 	"Samsung 3D Stereo Enhancement",
262 	"Wolfson Microelectronics 3D Enhancement",
263 	"Delta Integration 3D Enhancement",
264 	"SigmaTel 3D Enhancement",
265 	"Reserved 27",
266 	"Rockwell 3D Stereo Enhancement",
267 	"Reserved 29",
268 	"Reserved 30",
269 	"Reserved 31"
270 };
271 
272 static char *ac97feature[] = {
273 	"mic channel",
274 	"reserved",
275 	"tone",
276 	"simulated stereo",
277 	"headphone",
278 	"bass boost",
279 	"18 bit DAC",
280 	"20 bit DAC",
281 	"18 bit ADC",
282 	"20 bit ADC"
283 };
284 
285 static char *ac97extfeature[] = {
286 	"variable rate PCM",
287 	"double rate PCM",
288 	"reserved 1",
289 	"variable rate mic",
290 	"reserved 2",
291 	"reserved 3",
292 	"center DAC",
293 	"surround DAC",
294 	"LFE DAC",
295 	"AMAP",
296 	"reserved 4",
297 	"reserved 5",
298 	"reserved 6",
299 	"reserved 7",
300 };
301 
302 u_int16_t
303 ac97_rdcd(struct ac97_info *codec, int reg)
304 {
305 	if (codec->flags & AC97_F_RDCD_BUG) {
306 		u_int16_t i[2], j = 100;
307 
308 		i[0] = AC97_READ(codec->methods, codec->devinfo, reg);
309 		i[1] = AC97_READ(codec->methods, codec->devinfo, reg);
310 		while (i[0] != i[1] && j)
311 			i[j-- & 1] = AC97_READ(codec->methods, codec->devinfo, reg);
312 #if 0
313 		if (j < 100) {
314 			device_printf(codec->dev, "%s(): Inconsistent register value at"
315 					" 0x%08x (retry: %d)\n", __func__, reg, 100 - j);
316 		}
317 #endif
318 		return i[!(j & 1)];
319 	}
320 	return AC97_READ(codec->methods, codec->devinfo, reg);
321 }
322 
323 void
324 ac97_wrcd(struct ac97_info *codec, int reg, u_int16_t val)
325 {
326 	AC97_WRITE(codec->methods, codec->devinfo, reg, val);
327 }
328 
329 static void
330 ac97_reset(struct ac97_info *codec)
331 {
332 	u_int32_t i, ps;
333 	ac97_wrcd(codec, AC97_REG_RESET, 0);
334 	for (i = 0; i < 500; i++) {
335 		ps = ac97_rdcd(codec, AC97_REG_POWER) & AC97_POWER_STATUS;
336 		if (ps == AC97_POWER_STATUS)
337 			return;
338 		DELAY(1000);
339 	}
340 	device_printf(codec->dev, "AC97 reset timed out.\n");
341 }
342 
343 int
344 ac97_setrate(struct ac97_info *codec, int which, int rate)
345 {
346 	u_int16_t v;
347 
348 	switch(which) {
349 	case AC97_REGEXT_FDACRATE:
350 	case AC97_REGEXT_SDACRATE:
351 	case AC97_REGEXT_LDACRATE:
352 	case AC97_REGEXT_LADCRATE:
353 	case AC97_REGEXT_MADCRATE:
354 		break;
355 
356 	default:
357 		return -1;
358 	}
359 
360 	snd_mtxlock(codec->lock);
361 	if (rate != 0) {
362 		v = rate;
363 		if (codec->extstat & AC97_EXTCAP_DRA)
364 			v >>= 1;
365 		ac97_wrcd(codec, which, v);
366 	}
367 	v = ac97_rdcd(codec, which);
368 	if (codec->extstat & AC97_EXTCAP_DRA)
369 		v <<= 1;
370 	snd_mtxunlock(codec->lock);
371 	return v;
372 }
373 
374 int
375 ac97_setextmode(struct ac97_info *codec, u_int16_t mode)
376 {
377 	mode &= AC97_EXTCAPS;
378 	if ((mode & ~codec->extcaps) != 0) {
379 		device_printf(codec->dev, "ac97 invalid mode set 0x%04x\n",
380 			      mode);
381 		return -1;
382 	}
383 	snd_mtxlock(codec->lock);
384 	ac97_wrcd(codec, AC97_REGEXT_STAT, mode);
385 	codec->extstat = ac97_rdcd(codec, AC97_REGEXT_STAT) & AC97_EXTCAPS;
386 	snd_mtxunlock(codec->lock);
387 	return (mode == codec->extstat)? 0 : -1;
388 }
389 
390 u_int16_t
391 ac97_getextmode(struct ac97_info *codec)
392 {
393 	return codec->extstat;
394 }
395 
396 u_int16_t
397 ac97_getextcaps(struct ac97_info *codec)
398 {
399 	return codec->extcaps;
400 }
401 
402 u_int16_t
403 ac97_getcaps(struct ac97_info *codec)
404 {
405 	return codec->caps;
406 }
407 
408 static int
409 ac97_setrecsrc(struct ac97_info *codec, int channel)
410 {
411 	struct ac97mixtable_entry *e = &codec->mix[channel];
412 
413 	if (e->recidx > 0) {
414 		int val = e->recidx - 1;
415 		val |= val << 8;
416 		snd_mtxlock(codec->lock);
417 		ac97_wrcd(codec, AC97_REG_RECSEL, val);
418 		snd_mtxunlock(codec->lock);
419 		return 0;
420 	} else
421 		return -1;
422 }
423 
424 static int
425 ac97_setmixer(struct ac97_info *codec, unsigned channel, unsigned left, unsigned right)
426 {
427 	struct ac97mixtable_entry *e = &codec->mix[channel];
428 
429 	if (e->reg && e->enable && e->bits) {
430 		int mask, max, val, reg;
431 
432 		reg = (e->reg >= 0) ? e->reg : -e->reg;	/* AC97 register    */
433 		max = (1 << e->bits) - 1;		/* actual range	    */
434 		mask = (max << 8) | max;		/* bits of interest */
435 
436 		if (!e->stereo)
437 			right = left;
438 
439 		/*
440 		 * Invert the range if the polarity requires so,
441 		 * then scale to 0..max-1 to compute the value to
442 		 * write into the codec, and scale back to 0..100
443 		 * for the return value.
444 		 */
445 		if (e->reg > 0) {
446 			left = 100 - left;
447 			right = 100 - right;
448 		}
449 
450 		left = (left * max) / 100;
451 		right = (right * max) / 100;
452 
453 		val = (left << 8) | right;
454 
455 		left = (left * 100) / max;
456 		right = (right * 100) / max;
457 
458 		if (e->reg > 0) {
459 			left = 100 - left;
460 			right = 100 - right;
461 		}
462 
463 		/*
464 		 * For mono controls, trim val and mask, also taking
465 		 * care of e->ofs (offset of control field).
466 		 */
467 		if (e->ofs) {
468 			val &= max;
469 			val <<= e->ofs;
470 			mask = (max << e->ofs);
471 		}
472 
473 		/*
474 		 * If we have a mute bit, add it to the mask and
475 		 * update val and set mute if both channels require a
476 		 * zero volume.
477 		 */
478 		if (e->mute == 1) {
479 			mask |= AC97_MUTE;
480 			if (left == 0 && right == 0)
481 				val = AC97_MUTE;
482 		}
483 
484 		/*
485 		 * If the mask bit is set, do not alter the other bits.
486 		 */
487 		snd_mtxlock(codec->lock);
488 		if (e->mask) {
489 			int cur = ac97_rdcd(codec, reg);
490 			val |= cur & ~(mask);
491 		}
492 		ac97_wrcd(codec, reg, val);
493 		snd_mtxunlock(codec->lock);
494 		return left | (right << 8);
495 	} else {
496 #if 0
497 		printf("ac97_setmixer: reg=%d, bits=%d, enable=%d\n", e->reg, e->bits, e->enable);
498 #endif
499 		return -1;
500 	}
501 }
502 
503 static void
504 ac97_fix_auxout(struct ac97_info *codec)
505 {
506 	int keep_ogain;
507 
508 	/*
509 	 * By default, The ac97 aux_out register (0x04) corresponds to OSS's
510 	 * OGAIN setting.
511 	 *
512 	 * We first check whether aux_out is a valid register.  If not
513 	 * we may not want to keep ogain.
514 	 */
515 	keep_ogain = ac97_rdcd(codec, AC97_MIX_AUXOUT) & 0x8000;
516 
517 	/*
518 	 * Determine what AUX_OUT really means, it can be:
519 	 *
520 	 * 1. Headphone out.
521 	 * 2. 4-Channel Out
522 	 * 3. True line level out (effectively master volume).
523 	 *
524 	 * See Sections 5.2.1 and 5.27 for AUX_OUT Options in AC97r2.{2,3}.
525 	 */
526 	if (codec->extcaps & AC97_EXTCAP_SDAC &&
527 	    ac97_rdcd(codec, AC97_MIXEXT_SURROUND) == 0x8080) {
528 		codec->mix[SOUND_MIXER_OGAIN].reg = AC97_MIXEXT_SURROUND;
529 		keep_ogain = 1;
530 	}
531 
532 	if (keep_ogain == 0) {
533 		bzero(&codec->mix[SOUND_MIXER_OGAIN],
534 		      sizeof(codec->mix[SOUND_MIXER_OGAIN]));
535 	}
536 }
537 
538 static void
539 ac97_fix_tone(struct ac97_info *codec)
540 {
541 	/* Hide treble and bass if they don't exist */
542 	if ((codec->caps & AC97_CAP_TONE) == 0) {
543 		bzero(&codec->mix[SOUND_MIXER_BASS],
544 		      sizeof(codec->mix[SOUND_MIXER_BASS]));
545 		bzero(&codec->mix[SOUND_MIXER_TREBLE],
546 		      sizeof(codec->mix[SOUND_MIXER_TREBLE]));
547 	}
548 }
549 
550 static const char*
551 ac97_hw_desc(u_int32_t id, const char* vname, const char* cname, char* buf)
552 {
553 	if (cname == NULL) {
554 		sprintf(buf, "Unknown AC97 Codec (id = 0x%08x)", id);
555 		return buf;
556 	}
557 
558 	if (vname == NULL) vname = "Unknown";
559 
560 	if (bootverbose) {
561 		sprintf(buf, "%s %s AC97 Codec (id = 0x%08x)", vname, cname, id);
562 	} else {
563 		sprintf(buf, "%s %s AC97 Codec", vname, cname);
564 	}
565 	return buf;
566 }
567 
568 static unsigned
569 ac97_initmixer(struct ac97_info *codec)
570 {
571 	ac97_patch codec_patch;
572 	const char *cname, *vname;
573 	char desc[80];
574 	u_int8_t model, step;
575 	unsigned i, j, k, bit, old;
576 	u_int32_t id;
577 	int reg;
578 
579 	snd_mtxlock(codec->lock);
580 	codec->count = AC97_INIT(codec->methods, codec->devinfo);
581 	if (codec->count == 0) {
582 		device_printf(codec->dev, "ac97 codec init failed\n");
583 		snd_mtxunlock(codec->lock);
584 		return ENODEV;
585 	}
586 
587 	ac97_wrcd(codec, AC97_REG_POWER, (codec->flags & AC97_F_EAPD_INV)? 0x8000 : 0x0000);
588 	ac97_reset(codec);
589 	ac97_wrcd(codec, AC97_REG_POWER, (codec->flags & AC97_F_EAPD_INV)? 0x8000 : 0x0000);
590 
591 	i = ac97_rdcd(codec, AC97_REG_RESET);
592 	j = ac97_rdcd(codec, AC97_REG_RESET);
593 	/*
594 	 * Let see if this codec can return consistent value.
595 	 * If not, turn on aggressive read workaround
596 	 * (STAC9704 comes in mind).
597 	 */
598 	if (i != j) {
599 		codec->flags |= AC97_F_RDCD_BUG;
600 		i = ac97_rdcd(codec, AC97_REG_RESET);
601 	}
602 	codec->caps = i & 0x03ff;
603 	codec->se =  (i & 0x7c00) >> 10;
604 
605 	id = (ac97_rdcd(codec, AC97_REG_ID1) << 16) | ac97_rdcd(codec, AC97_REG_ID2);
606 	if (id == 0 || id == 0xffffffff) {
607 		device_printf(codec->dev, "ac97 codec invalid or not present (id == %x)\n", id);
608 		snd_mtxunlock(codec->lock);
609 		return ENODEV;
610 	}
611 
612 	codec->id = id;
613 	codec->noext = 0;
614 	codec_patch = NULL;
615 
616 	cname = NULL;
617 	model = step = 0;
618 	for (i = 0; ac97codecid[i].id; i++) {
619 		u_int32_t modelmask = 0xffffffff ^ ac97codecid[i].stepmask;
620 		if ((ac97codecid[i].id & modelmask) == (id & modelmask)) {
621 			codec->noext = ac97codecid[i].noext;
622 			codec_patch = ac97codecid[i].patch;
623 			cname = ac97codecid[i].name;
624 			model = (id & modelmask) & 0xff;
625 			step = (id & ~modelmask) & 0xff;
626 			break;
627 		}
628 	}
629 
630 	vname = NULL;
631 	for (i = 0; ac97vendorid[i].id; i++) {
632 		if (ac97vendorid[i].id == (id & 0xffffff00)) {
633 			vname = ac97vendorid[i].name;
634 			break;
635 		}
636 	}
637 
638 	codec->extcaps = 0;
639 	codec->extid = 0;
640 	codec->extstat = 0;
641 	if (!codec->noext) {
642 		i = ac97_rdcd(codec, AC97_REGEXT_ID);
643 		if (i != 0xffff) {
644 			codec->extcaps = i & 0x3fff;
645 			codec->extid =  (i & 0xc000) >> 14;
646 			codec->extstat = ac97_rdcd(codec, AC97_REGEXT_STAT) & AC97_EXTCAPS;
647 		}
648 	}
649 
650 	for (i = 0; i < 32; i++) {
651 		codec->mix[i] = ac97mixtable_default[i];
652 	}
653 	ac97_fix_auxout(codec);
654 	ac97_fix_tone(codec);
655 	if (codec_patch)
656 		codec_patch(codec);
657 
658 	for (i = 0; i < 32; i++) {
659 		k = codec->noext? codec->mix[i].enable : 1;
660 		reg = codec->mix[i].reg;
661 		if (reg < 0)
662 			reg = -reg;
663 		if (k && reg) {
664 			j = old = ac97_rdcd(codec, reg);
665 			/*
666 			 * Test for mute bit (except for AC97_MIX_TONE,
667 			 * where we simply assume it as available).
668 			 */
669 			if (codec->mix[i].mute) {
670 				ac97_wrcd(codec, reg, j | 0x8000);
671 				j = ac97_rdcd(codec, reg);
672 			} else
673 				j |= 0x8000;
674 			if ((j & 0x8000)) {
675 				/*
676 				 * Test whether the control width should be
677 				 * 4, 5 or 6 bit. For 5bit register, we should
678 				 * test it whether it's really 5 or 6bit. Leave
679 				 * 4bit register alone, because sometimes an
680 				 * attempt to write past 4th bit may cause
681 				 * incorrect result especially for AC97_MIX_BEEP
682 				 * (ac97 2.3).
683 				 */
684 				bit = codec->mix[i].bits;
685 				if (bit == 5)
686 					bit++;
687 				j = ((1 << bit) - 1) << codec->mix[i].ofs;
688 				ac97_wrcd(codec, reg,
689 					j | (codec->mix[i].mute ? 0x8000 : 0));
690 				k = ac97_rdcd(codec, reg) & j;
691 				k >>= codec->mix[i].ofs;
692 				if (reg == AC97_MIX_TONE &&
693 							((k & 0x0001) == 0x0000))
694 					k >>= 1;
695 				for (j = 0; k >> j; j++)
696 					;
697 				if (j != 0) {
698 #if 0
699 					device_printf(codec->dev, "%2d: [ac97_rdcd() = %d] [Testbit = %d] %d -> %d\n",
700 						i, k, bit, codec->mix[i].bits, j);
701 #endif
702 					codec->mix[i].enable = 1;
703 					codec->mix[i].bits = j;
704 				} else if (reg == AC97_MIX_BEEP) {
705 					/*
706 					 * Few codec such as CX20468-21 does
707 					 * have this control register, although
708 					 * the only usable part is the mute bit.
709 					 */
710 					codec->mix[i].enable = 1;
711 				} else
712 					codec->mix[i].enable = 0;
713 			} else
714 				codec->mix[i].enable = 0;
715 			ac97_wrcd(codec, reg, old);
716 		}
717 #if 0
718 		printf("mixch %d, en=%d, b=%d\n", i, codec->mix[i].enable, codec->mix[i].bits);
719 #endif
720 	}
721 
722 	device_printf(codec->dev, "<%s>\n",
723 		      ac97_hw_desc(codec->id, vname, cname, desc));
724 
725 	if (bootverbose) {
726 		if (codec->flags & AC97_F_RDCD_BUG)
727 			device_printf(codec->dev, "Buggy AC97 Codec: aggressive ac97_rdcd() workaround enabled\n");
728 		device_printf(codec->dev, "Codec features ");
729 		for (i = j = 0; i < 10; i++)
730 			if (codec->caps & (1 << i))
731 				printf("%s%s", j++? ", " : "", ac97feature[i]);
732 		printf("%s%d bit master volume", j++? ", " : "", codec->mix[SOUND_MIXER_VOLUME].bits);
733 		printf("%s%s\n", j? ", " : "", ac97enhancement[codec->se]);
734 
735 		if (codec->extcaps != 0 || codec->extid) {
736 			device_printf(codec->dev, "%s codec",
737 				      codec->extid? "Secondary" : "Primary");
738 			if (codec->extcaps)
739 				printf(" extended features ");
740 			for (i = j = 0; i < 14; i++)
741 				if (codec->extcaps & (1 << i))
742 					printf("%s%s", j++? ", " : "", ac97extfeature[i]);
743 			printf("\n");
744 		}
745 	}
746 
747 	i = 0;
748 	while ((ac97_rdcd(codec, AC97_REG_POWER) & 2) == 0) {
749 		if (++i == 100) {
750 			device_printf(codec->dev, "ac97 codec reports dac not ready\n");
751 			break;
752 		}
753 		DELAY(1000);
754 	}
755 	if (bootverbose)
756 		device_printf(codec->dev, "ac97 codec dac ready count: %d\n", i);
757 	snd_mtxunlock(codec->lock);
758 	return 0;
759 }
760 
761 static unsigned
762 ac97_reinitmixer(struct ac97_info *codec)
763 {
764 	snd_mtxlock(codec->lock);
765 	codec->count = AC97_INIT(codec->methods, codec->devinfo);
766 	if (codec->count == 0) {
767 		device_printf(codec->dev, "ac97 codec init failed\n");
768 		snd_mtxunlock(codec->lock);
769 		return ENODEV;
770 	}
771 
772 	ac97_wrcd(codec, AC97_REG_POWER, (codec->flags & AC97_F_EAPD_INV)? 0x8000 : 0x0000);
773 	ac97_reset(codec);
774 	ac97_wrcd(codec, AC97_REG_POWER, (codec->flags & AC97_F_EAPD_INV)? 0x8000 : 0x0000);
775 
776 	if (!codec->noext) {
777 		ac97_wrcd(codec, AC97_REGEXT_STAT, codec->extstat);
778 		if ((ac97_rdcd(codec, AC97_REGEXT_STAT) & AC97_EXTCAPS)
779 		    != codec->extstat)
780 			device_printf(codec->dev, "ac97 codec failed to reset extended mode (%x, got %x)\n",
781 				      codec->extstat,
782 				      ac97_rdcd(codec, AC97_REGEXT_STAT) &
783 				      AC97_EXTCAPS);
784 	}
785 
786 	if ((ac97_rdcd(codec, AC97_REG_POWER) & 2) == 0)
787 		device_printf(codec->dev, "ac97 codec reports dac not ready\n");
788 	snd_mtxunlock(codec->lock);
789 	return 0;
790 }
791 
792 struct ac97_info *
793 ac97_create(device_t dev, void *devinfo, kobj_class_t cls)
794 {
795 	struct ac97_info *codec;
796 	int eapdinv;
797 
798 	codec = (struct ac97_info *)malloc(sizeof *codec, M_AC97, M_NOWAIT | M_ZERO);
799 	if (codec == NULL)
800 		return NULL;
801 
802 	snprintf(codec->name, AC97_NAMELEN, "%s:ac97", device_get_nameunit(dev));
803 	codec->lock = snd_mtxcreate(codec->name, "ac97 codec");
804 	codec->methods = kobj_create(cls, M_AC97, M_WAITOK | M_ZERO);
805 	if (codec->methods == NULL) {
806 		snd_mtxlock(codec->lock);
807 		snd_mtxfree(codec->lock);
808 		free(codec, M_AC97);
809 		return NULL;
810 	}
811 
812 	codec->dev = dev;
813 	codec->devinfo = devinfo;
814 	codec->flags = 0;
815 	if (resource_int_value(device_get_name(dev), device_get_unit(dev),
816 		    "eapdinv", &eapdinv) == 0) {
817 		if (eapdinv != 0)
818 			codec->flags |= AC97_F_EAPD_INV;
819 	}
820 	return codec;
821 }
822 
823 void
824 ac97_destroy(struct ac97_info *codec)
825 {
826 	snd_mtxlock(codec->lock);
827 	if (codec->methods != NULL)
828 		kobj_delete(codec->methods, M_AC97);
829 	snd_mtxfree(codec->lock);
830 	free(codec, M_AC97);
831 }
832 
833 void
834 ac97_setflags(struct ac97_info *codec, u_int32_t val)
835 {
836 	codec->flags = val;
837 }
838 
839 u_int32_t
840 ac97_getflags(struct ac97_info *codec)
841 {
842 	return codec->flags;
843 }
844 
845 /* -------------------------------------------------------------------- */
846 
847 #ifdef SND_DYNSYSCTL
848 static int
849 sysctl_hw_snd_ac97_eapd(SYSCTL_HANDLER_ARGS)
850 {
851 	struct ac97_info *codec;
852 	int ea, inv, err = 0;
853 	u_int16_t val;
854 
855 	codec = oidp->oid_arg1;
856 	if (codec == NULL || codec->id == 0 || codec->lock == NULL)
857 		return EINVAL;
858 	snd_mtxlock(codec->lock);
859 	val = ac97_rdcd(codec, AC97_REG_POWER);
860 	inv = (codec->flags & AC97_F_EAPD_INV) ? 0 : 1;
861 	ea = (val >> 15) ^ inv;
862 	snd_mtxunlock(codec->lock);
863 	err = sysctl_handle_int(oidp, &ea, sizeof(ea), req);
864 	if (err == 0 && req->newptr != NULL) {
865 		if (ea != 0 && ea != 1)
866 			return EINVAL;
867 		if (ea != ((val >> 15) ^ inv)) {
868 			snd_mtxlock(codec->lock);
869 			ac97_wrcd(codec, AC97_REG_POWER, val ^ 0x8000);
870 			snd_mtxunlock(codec->lock);
871 		}
872 	}
873 	return err;
874 }
875 #endif
876 
877 static void
878 ac97_init_sysctl(struct ac97_info *codec)
879 {
880 #ifdef SND_DYNSYSCTL
881 	u_int16_t orig, val;
882 
883 	if (codec == NULL || codec->dev == NULL)
884 		return;
885 	snd_mtxlock(codec->lock);
886 	orig = ac97_rdcd(codec, AC97_REG_POWER);
887 	ac97_wrcd(codec, AC97_REG_POWER, orig ^ 0x8000);
888 	val = ac97_rdcd(codec, AC97_REG_POWER);
889 	ac97_wrcd(codec, AC97_REG_POWER, orig);
890 	snd_mtxunlock(codec->lock);
891 	if ((val & 0x8000) == (orig & 0x8000))
892 		return;
893 	SYSCTL_ADD_PROC(device_get_sysctl_ctx(codec->dev),
894 	    SYSCTL_CHILDREN(device_get_sysctl_tree(codec->dev)),
895             OID_AUTO, "eapd", CTLTYPE_INT | CTLFLAG_RW,
896 	    codec, sizeof(codec), sysctl_hw_snd_ac97_eapd,
897 	    "I", "AC97 External Amplifier");
898 #endif
899 }
900 
901 static int
902 ac97mix_init(struct snd_mixer *m)
903 {
904 	struct ac97_info *codec = mix_getdevinfo(m);
905 	struct snddev_info *d;
906 	u_int32_t subvendor;
907 	u_int32_t i, mask;
908 
909 	if (codec == NULL)
910 		return -1;
911 
912 	if (ac97_initmixer(codec))
913 		return -1;
914 
915 	switch (codec->id) {
916 	case 0x41445374:	/* AD1981B */
917 		subvendor = (u_int32_t)pci_get_subdevice(codec->dev) << 16;
918 		subvendor |= (u_int32_t)pci_get_subvendor(codec->dev) &
919 		    0x0000ffff;
920 		/* IBM Thinkcentre */
921 		if (subvendor == 0x02d91014) {
922 			/* Enable headphone jack sensing */
923 			ac97_wrcd(codec, 0x72, ac97_rdcd(codec, 0x72) |
924 			    0x0800);
925 			mask = 0;
926 			if (codec->mix[SOUND_MIXER_OGAIN].enable)
927 				mask |= SOUND_MASK_OGAIN;
928 			if (codec->mix[SOUND_MIXER_PHONEOUT].enable)
929 				mask |= SOUND_MASK_PHONEOUT;
930 			/* Tie ogain/phone to master volume */
931 			if (codec->mix[SOUND_MIXER_VOLUME].enable)
932 				mix_setparentchild(m, SOUND_MIXER_VOLUME,
933 				    mask);
934 			else {
935 				mix_setparentchild(m, SOUND_MIXER_VOLUME,
936 				    mask);
937 				mix_setrealdev(m, SOUND_MIXER_VOLUME,
938 				    SOUND_MIXER_NONE);
939 			}
940 		}
941 		break;
942 	case 0x434d4941:	/* CMI9738 */
943 	case 0x434d4961:	/* CMI9739 */
944 	case 0x434d4978:	/* CMI9761 */
945 	case 0x434d4982:	/* CMI9761 */
946 	case 0x434d4983:	/* CMI9761 */
947 		ac97_wrcd(codec, AC97_MIX_PCM, 0);
948 		bzero(&codec->mix[SOUND_MIXER_PCM],
949 		    sizeof(codec->mix[SOUND_MIXER_PCM]));
950 		d = device_get_softc(codec->dev);
951 		if (d != NULL)
952 			d->flags |= SD_F_SOFTPCMVOL;
953 		/* XXX How about master volume ? */
954 		break;
955 	default:
956 		break;
957 	}
958 
959 #if 0
960 	/* XXX For the sake of debugging purposes */
961 	mix_setparentchild(m, SOUND_MIXER_VOLUME,
962 	    SOUND_MASK_PCM | SOUND_MASK_CD);
963 	mix_setrealdev(m, SOUND_MIXER_VOLUME, SOUND_MIXER_NONE);
964 	ac97_wrcd(codec, AC97_MIX_MASTER, 0);
965 #endif
966 
967 	mask = 0;
968 	for (i = 0; i < 32; i++)
969 		mask |= codec->mix[i].enable? 1 << i : 0;
970 	mix_setdevs(m, mask);
971 
972 	mask = 0;
973 	for (i = 0; i < 32; i++)
974 		mask |= codec->mix[i].recidx? 1 << i : 0;
975 	mix_setrecdevs(m, mask);
976 
977 	ac97_init_sysctl(codec);
978 
979 	return 0;
980 }
981 
982 static int
983 ac97mix_uninit(struct snd_mixer *m)
984 {
985 	struct ac97_info *codec = mix_getdevinfo(m);
986 
987 	if (codec == NULL)
988 		return -1;
989 	/*
990 	if (ac97_uninitmixer(codec))
991 		return -1;
992 	*/
993 	ac97_destroy(codec);
994 	return 0;
995 }
996 
997 static int
998 ac97mix_reinit(struct snd_mixer *m)
999 {
1000 	struct ac97_info *codec = mix_getdevinfo(m);
1001 
1002 	if (codec == NULL)
1003 		return -1;
1004 	return ac97_reinitmixer(codec);
1005 }
1006 
1007 static int
1008 ac97mix_set(struct snd_mixer *m, unsigned dev, unsigned left, unsigned right)
1009 {
1010 	struct ac97_info *codec = mix_getdevinfo(m);
1011 
1012 	if (codec == NULL)
1013 		return -1;
1014 	return ac97_setmixer(codec, dev, left, right);
1015 }
1016 
1017 static int
1018 ac97mix_setrecsrc(struct snd_mixer *m, u_int32_t src)
1019 {
1020 	int i;
1021 	struct ac97_info *codec = mix_getdevinfo(m);
1022 
1023 	if (codec == NULL)
1024 		return -1;
1025 	for (i = 0; i < SOUND_MIXER_NRDEVICES; i++)
1026 		if ((src & (1 << i)) != 0)
1027 			break;
1028 	return (ac97_setrecsrc(codec, i) == 0)? 1 << i : -1;
1029 }
1030 
1031 static kobj_method_t ac97mixer_methods[] = {
1032     	KOBJMETHOD(mixer_init,		ac97mix_init),
1033     	KOBJMETHOD(mixer_uninit,	ac97mix_uninit),
1034     	KOBJMETHOD(mixer_reinit,	ac97mix_reinit),
1035     	KOBJMETHOD(mixer_set,		ac97mix_set),
1036     	KOBJMETHOD(mixer_setrecsrc,	ac97mix_setrecsrc),
1037 	{ 0, 0 }
1038 };
1039 MIXER_DECLARE(ac97mixer);
1040 
1041 /* -------------------------------------------------------------------- */
1042 
1043 kobj_class_t
1044 ac97_getmixerclass(void)
1045 {
1046 	return &ac97mixer_class;
1047 }
1048