188447a05SGarrett D'Amore /*
288447a05SGarrett D'Amore  * CDDL HEADER START
388447a05SGarrett D'Amore  *
488447a05SGarrett D'Amore  * The contents of this file are subject to the terms of the
588447a05SGarrett D'Amore  * Common Development and Distribution License (the "License").
688447a05SGarrett D'Amore  * You may not use this file except in compliance with the License.
788447a05SGarrett D'Amore  *
888447a05SGarrett D'Amore  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
988447a05SGarrett D'Amore  * or http://www.opensolaris.org/os/licensing.
1088447a05SGarrett D'Amore  * See the License for the specific language governing permissions
1188447a05SGarrett D'Amore  * and limitations under the License.
1288447a05SGarrett D'Amore  *
1388447a05SGarrett D'Amore  * When distributing Covered Code, include this CDDL HEADER in each
1488447a05SGarrett D'Amore  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
1588447a05SGarrett D'Amore  * If applicable, add the following below this CDDL HEADER, with the
1688447a05SGarrett D'Amore  * fields enclosed by brackets "[]" replaced with your own identifying
1788447a05SGarrett D'Amore  * information: Portions Copyright [yyyy] [name of copyright owner]
1888447a05SGarrett D'Amore  *
1988447a05SGarrett D'Amore  * CDDL HEADER END
2088447a05SGarrett D'Amore  */
2188447a05SGarrett D'Amore /*
2288447a05SGarrett D'Amore  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
2388447a05SGarrett D'Amore  * Use is subject to license terms.
2488447a05SGarrett D'Amore  */
2588447a05SGarrett D'Amore 
2688447a05SGarrett D'Amore /*
2788447a05SGarrett D'Amore  * ALC (Realtek/Advance Logic) codec extensions.
2888447a05SGarrett D'Amore  */
2988447a05SGarrett D'Amore 
3088447a05SGarrett D'Amore #include <sys/types.h>
3188447a05SGarrett D'Amore #include <sys/ddi.h>
3288447a05SGarrett D'Amore #include <sys/sunddi.h>
3388447a05SGarrett D'Amore #include <sys/audio/audio_driver.h>
3488447a05SGarrett D'Amore #include <sys/audio/ac97.h>
3588447a05SGarrett D'Amore #include <sys/note.h>
3688447a05SGarrett D'Amore #include "ac97_impl.h"
3788447a05SGarrett D'Amore 
3888447a05SGarrett D'Amore #define	ALC_DATA_FLOW_CTRL_REGISTER	0x6a
3988447a05SGarrett D'Amore #define	ADFC_SPDIFIN_EN			0x8000
4088447a05SGarrett D'Amore #define	ADFC_SPDIFIN_MON_EN		0x4000
4188447a05SGarrett D'Amore #define	ADFC_SPDIF_OUT_MASK		0x3000
4288447a05SGarrett D'Amore #define	ADFC_SPDIF_OUT_ACLINK		0x0000
4388447a05SGarrett D'Amore #define	ADFC_SPDIF_OUT_ADC		0x1000
4488447a05SGarrett D'Amore #define	ADFC_SPDIF_OUT_BYPASS		0x2000
4588447a05SGarrett D'Amore #define	ADFC_PCM_SPDIFIN		0x0800
4688447a05SGarrett D'Amore #define	ADFC_BACK_SURROUND		0x0400	/* ALC850 only */
4788447a05SGarrett D'Amore #define	ADFC_CENTER_LFE			0x0400	/* ALC650 series */
4888447a05SGarrett D'Amore #define	ADFC_MIC			0x0000
4988447a05SGarrett D'Amore #define	ADFC_SURROUND			0x0200
5088447a05SGarrett D'Amore #define	ADFC_LINEIN			0x0000
5188447a05SGarrett D'Amore #define	ADFC_FRONT_MIC_MONO_OUT		0x0100	/* ALC850 */
5288447a05SGarrett D'Amore #define	ADFC_ANALOG_INPUT_PASS_CLFE	0x0020
5388447a05SGarrett D'Amore #define	ADFC_ANALOG_INPUT_PASS_SURROUND	0x0010
5488447a05SGarrett D'Amore #define	ADFC_SURROUND_MIRROR		0x0001
5588447a05SGarrett D'Amore 
5688447a05SGarrett D'Amore #define	ALC_SURROUND_DAC_REGISTER	0x64
5788447a05SGarrett D'Amore #define	ASD_SURROUND_MUTE		0x8000
5888447a05SGarrett D'Amore #define	ASD_SURR_LEFT_VOL		0x1f00
5988447a05SGarrett D'Amore #define	ASD_SURR_RIGHT_VOL		0x001f
6088447a05SGarrett D'Amore 
6188447a05SGarrett D'Amore #define	ALC_CEN_LFE_DAC_REGISTER	0x66
6288447a05SGarrett D'Amore #define	ACLD_CEN_LFE_MUTE		0x8000
6388447a05SGarrett D'Amore #define	ACLD_LFE_VOL			0x1f00
6488447a05SGarrett D'Amore #define	ACLD_CEN_VOL			0x001f
6588447a05SGarrett D'Amore 
6688447a05SGarrett D'Amore #define	ALC_MISC_CTRL_REGISTER		0x7a
6788447a05SGarrett D'Amore #define	AMC_XTLSEL			0x8000
6888447a05SGarrett D'Amore #define	AMC_VREFOUT_DIS			0x1000
6988447a05SGarrett D'Amore #define	AMC_INDEP_MUTE_CTRL		0x0800
7088447a05SGarrett D'Amore #define	AMC_JD2_SURR_CEN_LFE		0x0008
7188447a05SGarrett D'Amore #define	AMC_JD1_SURR_CEN_LFE		0x0004
7288447a05SGarrett D'Amore #define	AMC_PIN47_SPDIF			0x0002
7388447a05SGarrett D'Amore #define	AMC_PIN47_EAPD			0x0000
7488447a05SGarrett D'Amore #define	AMC_JD0_SURR_CEN_LFE		0x0001
7588447a05SGarrett D'Amore 
7688447a05SGarrett D'Amore static void
alc650_set_linein_func(ac97_ctrl_t * actrl,uint64_t value)7788447a05SGarrett D'Amore alc650_set_linein_func(ac97_ctrl_t *actrl, uint64_t value)
7888447a05SGarrett D'Amore {
7988447a05SGarrett D'Amore 	ac97_t		*ac = actrl->actrl_ac97;
8088447a05SGarrett D'Amore 
81*33ab04abSGarrett D'Amore 	ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);	/* select page 0 */
8288447a05SGarrett D'Amore 	if (value & 2) {
83*33ab04abSGarrett D'Amore 		ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_SURROUND);
8488447a05SGarrett D'Amore 	} else {
85*33ab04abSGarrett D'Amore 		ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_SURROUND);
8688447a05SGarrett D'Amore 	}
8788447a05SGarrett D'Amore }
8888447a05SGarrett D'Amore 
8988447a05SGarrett D'Amore static void
alc650_set_mic_func(ac97_ctrl_t * actrl,uint64_t value)9088447a05SGarrett D'Amore alc650_set_mic_func(ac97_ctrl_t *actrl, uint64_t value)
9188447a05SGarrett D'Amore {
9288447a05SGarrett D'Amore 	ac97_t		*ac = actrl->actrl_ac97;
9388447a05SGarrett D'Amore 
94*33ab04abSGarrett D'Amore 	ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);	/* select page 0 */
9588447a05SGarrett D'Amore 	if (value & 2) {
96*33ab04abSGarrett D'Amore 		ac_set(ac, ALC_MISC_CTRL_REGISTER, AMC_VREFOUT_DIS);
97*33ab04abSGarrett D'Amore 		ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_CENTER_LFE);
9888447a05SGarrett D'Amore 	} else {
99*33ab04abSGarrett D'Amore 		ac_clr(ac, ALC_MISC_CTRL_REGISTER, AMC_VREFOUT_DIS);
100*33ab04abSGarrett D'Amore 		ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_CENTER_LFE);
10188447a05SGarrett D'Amore 	}
10288447a05SGarrett D'Amore }
10388447a05SGarrett D'Amore 
10488447a05SGarrett D'Amore #if 0
10588447a05SGarrett D'Amore static void
10688447a05SGarrett D'Amore alc850_set_auxin_func(ac97_ctrl_t *actrl, uint64_t value)
10788447a05SGarrett D'Amore {
10888447a05SGarrett D'Amore 	ac97_t		*ac = actrl->actrl_ac97;
10988447a05SGarrett D'Amore 
110*33ab04abSGarrett D'Amore 	ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);	/* select page 0 */
11188447a05SGarrett D'Amore 	if (value & 2) {
112*33ab04abSGarrett D'Amore 		ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_BACK_SURROUND);
11388447a05SGarrett D'Amore 	} else {
114*33ab04abSGarrett D'Amore 		ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_BACK_SURROUND);
11588447a05SGarrett D'Amore 	}
11688447a05SGarrett D'Amore }
11788447a05SGarrett D'Amore #endif
11888447a05SGarrett D'Amore 
11988447a05SGarrett D'Amore static void
alc650_set_pcm(ac97_ctrl_t * actrl,uint64_t value)12088447a05SGarrett D'Amore alc650_set_pcm(ac97_ctrl_t *actrl, uint64_t value)
12188447a05SGarrett D'Amore {
12288447a05SGarrett D'Amore 	ac97_t		*ac = actrl->actrl_ac97;
12388447a05SGarrett D'Amore 	uint16_t	adj_value;
12488447a05SGarrett D'Amore 	uint16_t	mute;
12588447a05SGarrett D'Amore 	uint8_t		vol;
12688447a05SGarrett D'Amore 
12788447a05SGarrett D'Amore 	/* limit input values to 16 bits and split to right and left */
12888447a05SGarrett D'Amore 	vol = value & 0xff;
12988447a05SGarrett D'Amore 
13088447a05SGarrett D'Amore 	/* If this control is mute-able than set as muted if needed */
13188447a05SGarrett D'Amore 	mute = vol ? 0 : ASD_SURROUND_MUTE;
132*33ab04abSGarrett D'Amore 	adj_value = ac_val_scale(vol, vol, 5) | mute;
13388447a05SGarrett D'Amore 
13488447a05SGarrett D'Amore 	/* select page 0 */
135*33ab04abSGarrett D'Amore 	ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);
13688447a05SGarrett D'Amore 	/* adjust all three PCM volumes */
137*33ab04abSGarrett D'Amore 	ac_wr(ac, AC97_PCM_OUT_VOLUME_REGISTER, adj_value);
138*33ab04abSGarrett D'Amore 	ac_wr(ac, ALC_SURROUND_DAC_REGISTER, adj_value);
139*33ab04abSGarrett D'Amore 	ac_wr(ac, ALC_CEN_LFE_DAC_REGISTER, adj_value);
14088447a05SGarrett D'Amore }
14188447a05SGarrett D'Amore 
14288447a05SGarrett D'Amore static const char *alc_linein_funcs[] = {
14388447a05SGarrett D'Amore 	AUDIO_PORT_LINEIN,
14488447a05SGarrett D'Amore 	AUDIO_PORT_SURROUND,
14588447a05SGarrett D'Amore 	NULL
14688447a05SGarrett D'Amore };
14788447a05SGarrett D'Amore 
14888447a05SGarrett D'Amore static const char *alc_mic_funcs[] = {
14988447a05SGarrett D'Amore 	AUDIO_PORT_MIC,
15088447a05SGarrett D'Amore 	AUDIO_PORT_CENLFE,
15188447a05SGarrett D'Amore 	NULL
15288447a05SGarrett D'Amore };
15388447a05SGarrett D'Amore 
15488447a05SGarrett D'Amore static ac97_ctrl_probe_t alc650_linein_func_cpt = {
15588447a05SGarrett D'Amore 	AUDIO_CTRL_ID_JACK1, 1, 3, 3, AUDIO_CTRL_TYPE_ENUM, AC97_FLAGS,
15688447a05SGarrett D'Amore 	0, alc650_set_linein_func, NULL, 0, alc_linein_funcs
15788447a05SGarrett D'Amore };
15888447a05SGarrett D'Amore static ac97_ctrl_probe_t alc650_mic_func_cpt = {
15988447a05SGarrett D'Amore 	AUDIO_CTRL_ID_JACK2, 1, 3, 3, AUDIO_CTRL_TYPE_ENUM, AC97_FLAGS,
16088447a05SGarrett D'Amore 	0, alc650_set_mic_func, NULL, 0, alc_mic_funcs
16188447a05SGarrett D'Amore };
16288447a05SGarrett D'Amore 
16388447a05SGarrett D'Amore static void
alc_pcm_override(ac97_t * ac)16488447a05SGarrett D'Amore alc_pcm_override(ac97_t *ac)
16588447a05SGarrett D'Amore {
16688447a05SGarrett D'Amore 	ac97_ctrl_t	*ctrl;
16788447a05SGarrett D'Amore 
16888447a05SGarrett D'Amore 	/* override master PCM volume function */
16988447a05SGarrett D'Amore 	ctrl = ac97_control_find(ac, AUDIO_CTRL_ID_VOLUME);
17088447a05SGarrett D'Amore 	if (ctrl != NULL) {
17188447a05SGarrett D'Amore 		ctrl->actrl_write_fn = alc650_set_pcm;
17288447a05SGarrett D'Amore 	}
17388447a05SGarrett D'Amore }
17488447a05SGarrett D'Amore 
17588447a05SGarrett D'Amore void
alc650_init(ac97_t * ac)17688447a05SGarrett D'Amore alc650_init(ac97_t *ac)
17788447a05SGarrett D'Amore {
17888447a05SGarrett D'Amore 	ac97_ctrl_probe_t	cp;
17988447a05SGarrett D'Amore 	int			ival;
18088447a05SGarrett D'Amore 
18188447a05SGarrett D'Amore 	bcopy(&alc650_linein_func_cpt, &cp, sizeof (cp));
182*33ab04abSGarrett D'Amore 	ival = ac_get_prop(ac, AC97_PROP_LINEIN_FUNC, 0);
18388447a05SGarrett D'Amore 	if ((ival >= 1) && (ival <= 2)) {
18488447a05SGarrett D'Amore 		cp.cp_initval = ival;
18588447a05SGarrett D'Amore 	}
186*33ab04abSGarrett D'Amore 	ac_add_control(ac, &cp);
18788447a05SGarrett D'Amore 
18888447a05SGarrett D'Amore 	bcopy(&alc650_mic_func_cpt, &cp, sizeof (cp));
189*33ab04abSGarrett D'Amore 	ival = ac_get_prop(ac, AC97_PROP_MIC_FUNC, 0);
19088447a05SGarrett D'Amore 	if ((ival >= 1) && (ival <= 2)) {
19188447a05SGarrett D'Amore 		cp.cp_initval = ival;
19288447a05SGarrett D'Amore 	}
193*33ab04abSGarrett D'Amore 	ac_add_control(ac, &cp);
19488447a05SGarrett D'Amore 
19588447a05SGarrett D'Amore 	alc_pcm_override(ac);
19688447a05SGarrett D'Amore }
19788447a05SGarrett D'Amore 
19888447a05SGarrett D'Amore void
alc850_init(ac97_t * ac)19988447a05SGarrett D'Amore alc850_init(ac97_t *ac)
20088447a05SGarrett D'Amore {
20188447a05SGarrett D'Amore 	/*
20288447a05SGarrett D'Amore 	 * NB: We could probably enable 7.1 here using the AUXIN source,
20388447a05SGarrett D'Amore 	 * but there are a few details still missing from the data sheet.
20488447a05SGarrett D'Amore 	 * (Such as, how is volume from the back-surround DAC managed?,
20588447a05SGarrett D'Amore 	 * and what SDATA slots are the back surround delivered on?)
20688447a05SGarrett D'Amore 	 *
20788447a05SGarrett D'Amore 	 * Also, the AC'97 controllers themselves don't necessarily support
20888447a05SGarrett D'Amore 	 * 7.1, so we'd have to figure out how to coordinate detection
20988447a05SGarrett D'Amore 	 * with the controller.  5.1 should be good enough for now.
21088447a05SGarrett D'Amore 	 *
21188447a05SGarrett D'Amore 	 * Unlike other products, ALC850 has separate pins for 5.1 data,
21288447a05SGarrett D'Amore 	 * so jack retasking isn't needed.  However, it can retask
21388447a05SGarrett D'Amore 	 * some jacks, but we don't have full details for that right
21488447a05SGarrett D'Amore 	 * now.  We've not seen it on any systems (yet) where this was
21588447a05SGarrett D'Amore 	 * necessary, though.
21688447a05SGarrett D'Amore 	 */
21788447a05SGarrett D'Amore 
21888447a05SGarrett D'Amore 	alc_pcm_override(ac);
21988447a05SGarrett D'Amore }
220