mirror of
https://github.com/S3NEO/android_kernel_samsung_msm8226.git
synced 2024-11-07 03:47:13 +00:00
ALSA: ALS4000, slight mixer improvements
- add 8kHz / 20 kHz low-pass filter switch control - add ALS4000 Mono capture route control - add annotations to specs pages - improve ALS4000 PM saved regs selection (remove SB dummy register, add missing ones) - add some missing ALS4000 register defines - constify two variables Signed-off-by: Andreas Mohr <andi@lisas.de> Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
parent
4c9f1d3ed7
commit
ce71bfd1aa
2 changed files with 121 additions and 39 deletions
|
@ -249,6 +249,7 @@ struct snd_sb {
|
||||||
#define SB_ALS4000_3D_AUTO_MUTE 0x52
|
#define SB_ALS4000_3D_AUTO_MUTE 0x52
|
||||||
#define SB_ALS4000_ANALOG_BLOCK_CTRL 0x53
|
#define SB_ALS4000_ANALOG_BLOCK_CTRL 0x53
|
||||||
#define SB_ALS4000_3D_DELAYLINE_PATTERN 0x54
|
#define SB_ALS4000_3D_DELAYLINE_PATTERN 0x54
|
||||||
|
#define SB_ALS4000_CR3_CONFIGURATION 0xc3 /* bit 7 is Digital Loop Enable */
|
||||||
#define SB_ALS4000_QSOUND 0xdb
|
#define SB_ALS4000_QSOUND 0xdb
|
||||||
|
|
||||||
/* IRQ setting bitmap */
|
/* IRQ setting bitmap */
|
||||||
|
@ -330,7 +331,8 @@ enum {
|
||||||
SB_MIX_DOUBLE,
|
SB_MIX_DOUBLE,
|
||||||
SB_MIX_INPUT_SW,
|
SB_MIX_INPUT_SW,
|
||||||
SB_MIX_CAPTURE_PRO,
|
SB_MIX_CAPTURE_PRO,
|
||||||
SB_MIX_CAPTURE_DT019X
|
SB_MIX_CAPTURE_DT019X,
|
||||||
|
SB_MIX_MONO_CAPTURE_ALS4K
|
||||||
};
|
};
|
||||||
|
|
||||||
#define SB_MIXVAL_DOUBLE(left_reg, right_reg, left_shift, right_shift, mask) \
|
#define SB_MIXVAL_DOUBLE(left_reg, right_reg, left_shift, right_shift, mask) \
|
||||||
|
|
|
@ -182,7 +182,7 @@ static int snd_sbmixer_put_double(struct snd_kcontrol *kcontrol, struct snd_ctl_
|
||||||
|
|
||||||
static int snd_dt019x_input_sw_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
static int snd_dt019x_input_sw_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
||||||
{
|
{
|
||||||
static char *texts[5] = {
|
static const char *texts[5] = {
|
||||||
"CD", "Mic", "Line", "Synth", "Master"
|
"CD", "Mic", "Line", "Synth", "Master"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -268,13 +268,74 @@ static int snd_dt019x_input_sw_put(struct snd_kcontrol *kcontrol, struct snd_ctl
|
||||||
return change;
|
return change;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ALS4000 mono recording control switch
|
||||||
|
*/
|
||||||
|
|
||||||
|
static int snd_als4k_mono_capture_route_info(struct snd_kcontrol *kcontrol,
|
||||||
|
struct snd_ctl_elem_info *uinfo)
|
||||||
|
{
|
||||||
|
static const char *texts[3] = {
|
||||||
|
"L chan only", "R chan only", "L ch/2 + R ch/2"
|
||||||
|
};
|
||||||
|
|
||||||
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
|
||||||
|
uinfo->count = 1;
|
||||||
|
uinfo->value.enumerated.items = 3;
|
||||||
|
if (uinfo->value.enumerated.item > 2)
|
||||||
|
uinfo->value.enumerated.item = 2;
|
||||||
|
strcpy(uinfo->value.enumerated.name,
|
||||||
|
texts[uinfo->value.enumerated.item]);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int snd_als4k_mono_capture_route_get(struct snd_kcontrol *kcontrol,
|
||||||
|
struct snd_ctl_elem_value *ucontrol)
|
||||||
|
{
|
||||||
|
struct snd_sb *sb = snd_kcontrol_chip(kcontrol);
|
||||||
|
unsigned long flags;
|
||||||
|
unsigned char oval;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&sb->mixer_lock, flags);
|
||||||
|
oval = snd_sbmixer_read(sb, SB_ALS4000_MONO_IO_CTRL);
|
||||||
|
spin_unlock_irqrestore(&sb->mixer_lock, flags);
|
||||||
|
oval >>= 6;
|
||||||
|
if (oval > 2)
|
||||||
|
oval = 2;
|
||||||
|
|
||||||
|
ucontrol->value.enumerated.item[0] = oval;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int snd_als4k_mono_capture_route_put(struct snd_kcontrol *kcontrol,
|
||||||
|
struct snd_ctl_elem_value *ucontrol)
|
||||||
|
{
|
||||||
|
struct snd_sb *sb = snd_kcontrol_chip(kcontrol);
|
||||||
|
unsigned long flags;
|
||||||
|
int change;
|
||||||
|
unsigned char nval, oval;
|
||||||
|
|
||||||
|
if (ucontrol->value.enumerated.item[0] > 2)
|
||||||
|
return -EINVAL;
|
||||||
|
spin_lock_irqsave(&sb->mixer_lock, flags);
|
||||||
|
oval = snd_sbmixer_read(sb, SB_ALS4000_MONO_IO_CTRL);
|
||||||
|
|
||||||
|
nval = (oval & ~(3 << 6))
|
||||||
|
| (ucontrol->value.enumerated.item[0] << 6);
|
||||||
|
change = nval != oval;
|
||||||
|
if (change)
|
||||||
|
snd_sbmixer_write(sb, SB_ALS4000_MONO_IO_CTRL, nval);
|
||||||
|
spin_unlock_irqrestore(&sb->mixer_lock, flags);
|
||||||
|
return change;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* SBPRO input multiplexer
|
* SBPRO input multiplexer
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int snd_sb8mixer_info_mux(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
static int snd_sb8mixer_info_mux(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
||||||
{
|
{
|
||||||
static char *texts[3] = {
|
static const char *texts[3] = {
|
||||||
"Mic", "CD", "Line"
|
"Mic", "CD", "Line"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -442,6 +503,12 @@ int snd_sbmixer_add_ctl(struct snd_sb *chip, const char *name, int index, int ty
|
||||||
.get = snd_dt019x_input_sw_get,
|
.get = snd_dt019x_input_sw_get,
|
||||||
.put = snd_dt019x_input_sw_put,
|
.put = snd_dt019x_input_sw_put,
|
||||||
},
|
},
|
||||||
|
[SB_MIX_MONO_CAPTURE_ALS4K] = {
|
||||||
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
||||||
|
.info = snd_als4k_mono_capture_route_info,
|
||||||
|
.get = snd_als4k_mono_capture_route_get,
|
||||||
|
.put = snd_als4k_mono_capture_route_put,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
struct snd_kcontrol *ctl;
|
struct snd_kcontrol *ctl;
|
||||||
int err;
|
int err;
|
||||||
|
@ -636,6 +703,8 @@ static struct sbmix_elem snd_dt019x_ctl_capture_source =
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct sbmix_elem *snd_dt019x_controls[] = {
|
static struct sbmix_elem *snd_dt019x_controls[] = {
|
||||||
|
/* ALS4000 below has some parts which we might be lacking,
|
||||||
|
* e.g. snd_als4000_ctl_mono_playback_switch - check it! */
|
||||||
&snd_dt019x_ctl_master_play_vol,
|
&snd_dt019x_ctl_master_play_vol,
|
||||||
&snd_dt019x_ctl_pcm_play_vol,
|
&snd_dt019x_ctl_pcm_play_vol,
|
||||||
&snd_dt019x_ctl_synth_play_vol,
|
&snd_dt019x_ctl_synth_play_vol,
|
||||||
|
@ -666,18 +735,21 @@ static unsigned char snd_dt019x_init_values[][2] = {
|
||||||
/*
|
/*
|
||||||
* ALS4000 specific mixer elements
|
* ALS4000 specific mixer elements
|
||||||
*/
|
*/
|
||||||
/* FIXME: SB_ALS4000_MONO_IO_CTRL needs output select ctrl! */
|
|
||||||
static struct sbmix_elem snd_als4000_ctl_master_mono_playback_switch =
|
static struct sbmix_elem snd_als4000_ctl_master_mono_playback_switch =
|
||||||
SB_SINGLE("Master Mono Playback Switch", SB_ALS4000_MONO_IO_CTRL, 5, 1);
|
SB_SINGLE("Master Mono Playback Switch", SB_ALS4000_MONO_IO_CTRL, 5, 1);
|
||||||
static struct sbmix_elem snd_als4000_ctl_master_mono_capture_route =
|
static struct sbmix_elem snd_als4k_ctl_master_mono_capture_route = {
|
||||||
SB_SINGLE("Master Mono Capture Route", SB_ALS4000_MONO_IO_CTRL, 6, 0x03);
|
.name = "Master Mono Capture Route",
|
||||||
/* FIXME: mono playback switch also available on DT019X? */
|
.type = SB_MIX_MONO_CAPTURE_ALS4K
|
||||||
|
};
|
||||||
static struct sbmix_elem snd_als4000_ctl_mono_playback_switch =
|
static struct sbmix_elem snd_als4000_ctl_mono_playback_switch =
|
||||||
SB_SINGLE("Mono Playback Switch", SB_DT019X_OUTPUT_SW2, 0, 1);
|
SB_SINGLE("Mono Playback Switch", SB_DT019X_OUTPUT_SW2, 0, 1);
|
||||||
static struct sbmix_elem snd_als4000_ctl_mic_20db_boost =
|
static struct sbmix_elem snd_als4000_ctl_mic_20db_boost =
|
||||||
SB_SINGLE("Mic Boost (+20dB)", SB_ALS4000_MIC_IN_GAIN, 0, 0x03);
|
SB_SINGLE("Mic Boost (+20dB)", SB_ALS4000_MIC_IN_GAIN, 0, 0x03);
|
||||||
static struct sbmix_elem snd_als4000_ctl_mixer_loopback =
|
static struct sbmix_elem snd_als4000_ctl_mixer_analog_loopback =
|
||||||
SB_SINGLE("Analog Loopback", SB_ALS4000_MIC_IN_GAIN, 7, 0x01);
|
SB_SINGLE("Analog Loopback Switch", SB_ALS4000_MIC_IN_GAIN, 7, 0x01);
|
||||||
|
static struct sbmix_elem snd_als4000_ctl_mixer_digital_loopback =
|
||||||
|
SB_SINGLE("Digital Loopback Switch",
|
||||||
|
SB_ALS4000_CR3_CONFIGURATION, 7, 0x01);
|
||||||
/* FIXME: functionality of 3D controls might be swapped, I didn't find
|
/* FIXME: functionality of 3D controls might be swapped, I didn't find
|
||||||
* a description of how to identify what is supposed to be what */
|
* a description of how to identify what is supposed to be what */
|
||||||
static struct sbmix_elem snd_als4000_3d_control_switch =
|
static struct sbmix_elem snd_als4000_3d_control_switch =
|
||||||
|
@ -694,6 +766,9 @@ static struct sbmix_elem snd_als4000_3d_control_delay =
|
||||||
SB_SINGLE("3D Control - Wide", SB_ALS4000_3D_TIME_DELAY, 0, 0x0f);
|
SB_SINGLE("3D Control - Wide", SB_ALS4000_3D_TIME_DELAY, 0, 0x0f);
|
||||||
static struct sbmix_elem snd_als4000_3d_control_poweroff_switch =
|
static struct sbmix_elem snd_als4000_3d_control_poweroff_switch =
|
||||||
SB_SINGLE("3D PowerOff Switch", SB_ALS4000_3D_TIME_DELAY, 4, 0x01);
|
SB_SINGLE("3D PowerOff Switch", SB_ALS4000_3D_TIME_DELAY, 4, 0x01);
|
||||||
|
static struct sbmix_elem snd_als4000_ctl_3db_freq_control_switch =
|
||||||
|
SB_SINGLE("Master Playback 8kHz / 20kHz LPF Switch",
|
||||||
|
SB_ALS4000_FMDAC, 5, 0x01);
|
||||||
#ifdef NOT_AVAILABLE
|
#ifdef NOT_AVAILABLE
|
||||||
static struct sbmix_elem snd_als4000_ctl_fmdac =
|
static struct sbmix_elem snd_als4000_ctl_fmdac =
|
||||||
SB_SINGLE("FMDAC Switch (Option ?)", SB_ALS4000_FMDAC, 0, 0x01);
|
SB_SINGLE("FMDAC Switch (Option ?)", SB_ALS4000_FMDAC, 0, 0x01);
|
||||||
|
@ -702,35 +777,37 @@ static struct sbmix_elem snd_als4000_ctl_qsound =
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static struct sbmix_elem *snd_als4000_controls[] = {
|
static struct sbmix_elem *snd_als4000_controls[] = {
|
||||||
&snd_sb16_ctl_master_play_vol,
|
/* ALS4000a.PDF regs page */
|
||||||
&snd_dt019x_ctl_pcm_play_switch,
|
&snd_sb16_ctl_master_play_vol, /* MX30/31 12 */
|
||||||
&snd_sb16_ctl_pcm_play_vol,
|
&snd_dt019x_ctl_pcm_play_switch, /* MX4C 16 */
|
||||||
&snd_sb16_ctl_synth_capture_route,
|
&snd_sb16_ctl_pcm_play_vol, /* MX32/33 12 */
|
||||||
&snd_dt019x_ctl_synth_play_switch,
|
&snd_sb16_ctl_synth_capture_route, /* MX3D/3E 14 */
|
||||||
&snd_sb16_ctl_synth_play_vol,
|
&snd_dt019x_ctl_synth_play_switch, /* MX4C 16 */
|
||||||
&snd_sb16_ctl_cd_capture_route,
|
&snd_sb16_ctl_synth_play_vol, /* MX34/35 12/13 */
|
||||||
&snd_sb16_ctl_cd_play_switch,
|
&snd_sb16_ctl_cd_capture_route, /* MX3D/3E 14 */
|
||||||
&snd_sb16_ctl_cd_play_vol,
|
&snd_sb16_ctl_cd_play_switch, /* MX3C 14 */
|
||||||
&snd_sb16_ctl_line_capture_route,
|
&snd_sb16_ctl_cd_play_vol, /* MX36/37 13 */
|
||||||
&snd_sb16_ctl_line_play_switch,
|
&snd_sb16_ctl_line_capture_route, /* MX3D/3E 14 */
|
||||||
&snd_sb16_ctl_line_play_vol,
|
&snd_sb16_ctl_line_play_switch, /* MX3C 14 */
|
||||||
&snd_sb16_ctl_mic_capture_route,
|
&snd_sb16_ctl_line_play_vol, /* MX38/39 13 */
|
||||||
&snd_als4000_ctl_mic_20db_boost,
|
&snd_sb16_ctl_mic_capture_route, /* MX3D/3E 14 */
|
||||||
&snd_sb16_ctl_auto_mic_gain,
|
&snd_als4000_ctl_mic_20db_boost, /* MX4D 16 */
|
||||||
&snd_sb16_ctl_mic_play_switch,
|
&snd_sb16_ctl_mic_play_switch, /* MX3C 14 */
|
||||||
&snd_sb16_ctl_mic_play_vol,
|
&snd_sb16_ctl_mic_play_vol, /* MX3A 13 */
|
||||||
&snd_sb16_ctl_pc_speaker_vol,
|
&snd_sb16_ctl_pc_speaker_vol, /* MX3B 14 */
|
||||||
&snd_sb16_ctl_capture_vol,
|
&snd_sb16_ctl_capture_vol, /* MX3F/40 15 */
|
||||||
&snd_sb16_ctl_play_vol,
|
&snd_sb16_ctl_play_vol, /* MX41/42 15 */
|
||||||
&snd_als4000_ctl_master_mono_playback_switch,
|
&snd_als4000_ctl_master_mono_playback_switch, /* MX4C 16 */
|
||||||
&snd_als4000_ctl_master_mono_capture_route,
|
&snd_als4k_ctl_master_mono_capture_route, /* MX4B 16 */
|
||||||
&snd_als4000_ctl_mono_playback_switch,
|
&snd_als4000_ctl_mono_playback_switch, /* MX4C 16 */
|
||||||
&snd_als4000_ctl_mixer_loopback,
|
&snd_als4000_ctl_mixer_analog_loopback, /* MX4D 16 */
|
||||||
&snd_als4000_3d_control_switch,
|
&snd_als4000_ctl_mixer_digital_loopback, /* CR3 21 */
|
||||||
&snd_als4000_3d_control_ratio,
|
&snd_als4000_3d_control_switch, /* MX50 17 */
|
||||||
&snd_als4000_3d_control_freq,
|
&snd_als4000_3d_control_ratio, /* MX50 17 */
|
||||||
&snd_als4000_3d_control_delay,
|
&snd_als4000_3d_control_freq, /* MX50 17 */
|
||||||
&snd_als4000_3d_control_poweroff_switch,
|
&snd_als4000_3d_control_delay, /* MX51 18 */
|
||||||
|
&snd_als4000_3d_control_poweroff_switch, /* MX51 18 */
|
||||||
|
&snd_als4000_ctl_3db_freq_control_switch, /* MX4F 17 */
|
||||||
#ifdef NOT_AVAILABLE
|
#ifdef NOT_AVAILABLE
|
||||||
&snd_als4000_ctl_fmdac,
|
&snd_als4000_ctl_fmdac,
|
||||||
&snd_als4000_ctl_qsound,
|
&snd_als4000_ctl_qsound,
|
||||||
|
@ -905,13 +982,14 @@ static unsigned char dt019x_saved_regs[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
static unsigned char als4000_saved_regs[] = {
|
static unsigned char als4000_saved_regs[] = {
|
||||||
|
/* please verify in dsheet whether regs to be added
|
||||||
|
are actually real H/W or just dummy */
|
||||||
SB_DSP4_MASTER_DEV, SB_DSP4_MASTER_DEV + 1,
|
SB_DSP4_MASTER_DEV, SB_DSP4_MASTER_DEV + 1,
|
||||||
SB_DSP4_OUTPUT_SW,
|
SB_DSP4_OUTPUT_SW,
|
||||||
SB_DSP4_PCM_DEV, SB_DSP4_PCM_DEV + 1,
|
SB_DSP4_PCM_DEV, SB_DSP4_PCM_DEV + 1,
|
||||||
SB_DSP4_INPUT_LEFT, SB_DSP4_INPUT_RIGHT,
|
SB_DSP4_INPUT_LEFT, SB_DSP4_INPUT_RIGHT,
|
||||||
SB_DSP4_SYNTH_DEV, SB_DSP4_SYNTH_DEV + 1,
|
SB_DSP4_SYNTH_DEV, SB_DSP4_SYNTH_DEV + 1,
|
||||||
SB_DSP4_CD_DEV, SB_DSP4_CD_DEV + 1,
|
SB_DSP4_CD_DEV, SB_DSP4_CD_DEV + 1,
|
||||||
SB_DSP4_MIC_AGC,
|
|
||||||
SB_DSP4_MIC_DEV,
|
SB_DSP4_MIC_DEV,
|
||||||
SB_DSP4_SPEAKER_DEV,
|
SB_DSP4_SPEAKER_DEV,
|
||||||
SB_DSP4_IGAIN_DEV, SB_DSP4_IGAIN_DEV + 1,
|
SB_DSP4_IGAIN_DEV, SB_DSP4_IGAIN_DEV + 1,
|
||||||
|
@ -919,8 +997,10 @@ static unsigned char als4000_saved_regs[] = {
|
||||||
SB_DT019X_OUTPUT_SW2,
|
SB_DT019X_OUTPUT_SW2,
|
||||||
SB_ALS4000_MONO_IO_CTRL,
|
SB_ALS4000_MONO_IO_CTRL,
|
||||||
SB_ALS4000_MIC_IN_GAIN,
|
SB_ALS4000_MIC_IN_GAIN,
|
||||||
|
SB_ALS4000_FMDAC,
|
||||||
SB_ALS4000_3D_SND_FX,
|
SB_ALS4000_3D_SND_FX,
|
||||||
SB_ALS4000_3D_TIME_DELAY,
|
SB_ALS4000_3D_TIME_DELAY,
|
||||||
|
SB_ALS4000_CR3_CONFIGURATION,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void save_mixer(struct snd_sb *chip, unsigned char *regs, int num_regs)
|
static void save_mixer(struct snd_sb *chip, unsigned char *regs, int num_regs)
|
||||||
|
|
Loading…
Reference in a new issue