#include <sys/types.h>
#include <sys/ddi.h>
#include <sys/sunddi.h>
#include <sys/audio/audio_driver.h>
#include <sys/audio/ac97.h>
#include <sys/note.h>
#include "ac97_impl.h"
#define ALC_DATA_FLOW_CTRL_REGISTER 0x6a
#define ADFC_SPDIFIN_EN 0x8000
#define ADFC_SPDIFIN_MON_EN 0x4000
#define ADFC_SPDIF_OUT_MASK 0x3000
#define ADFC_SPDIF_OUT_ACLINK 0x0000
#define ADFC_SPDIF_OUT_ADC 0x1000
#define ADFC_SPDIF_OUT_BYPASS 0x2000
#define ADFC_PCM_SPDIFIN 0x0800
#define ADFC_BACK_SURROUND 0x0400
#define ADFC_CENTER_LFE 0x0400
#define ADFC_MIC 0x0000
#define ADFC_SURROUND 0x0200
#define ADFC_LINEIN 0x0000
#define ADFC_FRONT_MIC_MONO_OUT 0x0100
#define ADFC_ANALOG_INPUT_PASS_CLFE 0x0020
#define ADFC_ANALOG_INPUT_PASS_SURROUND 0x0010
#define ADFC_SURROUND_MIRROR 0x0001
#define ALC_SURROUND_DAC_REGISTER 0x64
#define ASD_SURROUND_MUTE 0x8000
#define ASD_SURR_LEFT_VOL 0x1f00
#define ASD_SURR_RIGHT_VOL 0x001f
#define ALC_CEN_LFE_DAC_REGISTER 0x66
#define ACLD_CEN_LFE_MUTE 0x8000
#define ACLD_LFE_VOL 0x1f00
#define ACLD_CEN_VOL 0x001f
#define ALC_MISC_CTRL_REGISTER 0x7a
#define AMC_XTLSEL 0x8000
#define AMC_VREFOUT_DIS 0x1000
#define AMC_INDEP_MUTE_CTRL 0x0800
#define AMC_JD2_SURR_CEN_LFE 0x0008
#define AMC_JD1_SURR_CEN_LFE 0x0004
#define AMC_PIN47_SPDIF 0x0002
#define AMC_PIN47_EAPD 0x0000
#define AMC_JD0_SURR_CEN_LFE 0x0001
static void
alc650_set_linein_func(ac97_ctrl_t *actrl, uint64_t value)
{
ac97_t *ac = actrl->actrl_ac97;
ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);
if (value & 2) {
ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_SURROUND);
} else {
ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_SURROUND);
}
}
static void
alc650_set_mic_func(ac97_ctrl_t *actrl, uint64_t value)
{
ac97_t *ac = actrl->actrl_ac97;
ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);
if (value & 2) {
ac_set(ac, ALC_MISC_CTRL_REGISTER, AMC_VREFOUT_DIS);
ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_CENTER_LFE);
} else {
ac_clr(ac, ALC_MISC_CTRL_REGISTER, AMC_VREFOUT_DIS);
ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_CENTER_LFE);
}
}
#if 0
static void
alc850_set_auxin_func(ac97_ctrl_t *actrl, uint64_t value)
{
ac97_t *ac = actrl->actrl_ac97;
ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);
if (value & 2) {
ac_set(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_BACK_SURROUND);
} else {
ac_clr(ac, ALC_DATA_FLOW_CTRL_REGISTER, ADFC_BACK_SURROUND);
}
}
#endif
static void
alc650_set_pcm(ac97_ctrl_t *actrl, uint64_t value)
{
ac97_t *ac = actrl->actrl_ac97;
uint16_t adj_value;
uint16_t mute;
uint8_t vol;
vol = value & 0xff;
mute = vol ? 0 : ASD_SURROUND_MUTE;
adj_value = ac_val_scale(vol, vol, 5) | mute;
ac_wr(ac, AC97_INTERRUPT_PAGING_REGISTER, 0);
ac_wr(ac, AC97_PCM_OUT_VOLUME_REGISTER, adj_value);
ac_wr(ac, ALC_SURROUND_DAC_REGISTER, adj_value);
ac_wr(ac, ALC_CEN_LFE_DAC_REGISTER, adj_value);
}
static const char *alc_linein_funcs[] = {
AUDIO_PORT_LINEIN,
AUDIO_PORT_SURROUND,
NULL
};
static const char *alc_mic_funcs[] = {
AUDIO_PORT_MIC,
AUDIO_PORT_CENLFE,
NULL
};
static ac97_ctrl_probe_t alc650_linein_func_cpt = {
AUDIO_CTRL_ID_JACK1, 1, 3, 3, AUDIO_CTRL_TYPE_ENUM, AC97_FLAGS,
0, alc650_set_linein_func, NULL, 0, alc_linein_funcs
};
static ac97_ctrl_probe_t alc650_mic_func_cpt = {
AUDIO_CTRL_ID_JACK2, 1, 3, 3, AUDIO_CTRL_TYPE_ENUM, AC97_FLAGS,
0, alc650_set_mic_func, NULL, 0, alc_mic_funcs
};
static void
alc_pcm_override(ac97_t *ac)
{
ac97_ctrl_t *ctrl;
ctrl = ac97_control_find(ac, AUDIO_CTRL_ID_VOLUME);
if (ctrl != NULL) {
ctrl->actrl_write_fn = alc650_set_pcm;
}
}
void
alc650_init(ac97_t *ac)
{
ac97_ctrl_probe_t cp;
int ival;
bcopy(&alc650_linein_func_cpt, &cp, sizeof (cp));
ival = ac_get_prop(ac, AC97_PROP_LINEIN_FUNC, 0);
if ((ival >= 1) && (ival <= 2)) {
cp.cp_initval = ival;
}
ac_add_control(ac, &cp);
bcopy(&alc650_mic_func_cpt, &cp, sizeof (cp));
ival = ac_get_prop(ac, AC97_PROP_MIC_FUNC, 0);
if ((ival >= 1) && (ival <= 2)) {
cp.cp_initval = ival;
}
ac_add_control(ac, &cp);
alc_pcm_override(ac);
}
void
alc850_init(ac97_t *ac)
{
alc_pcm_override(ac);
}