diff options
-rw-r--r-- | include/sound/ac97_codec.h | 2 | ||||
-rw-r--r-- | sound/pci/ac97/ac97_patch.c | 5 | ||||
-rw-r--r-- | sound/pci/cs5535audio/Makefile | 3 | ||||
-rw-r--r-- | sound/pci/cs5535audio/cs5535audio.c | 12 | ||||
-rw-r--r-- | sound/pci/cs5535audio/cs5535audio.h | 39 | ||||
-rw-r--r-- | sound/pci/cs5535audio/cs5535audio_olpc.c | 179 | ||||
-rw-r--r-- | sound/pci/cs5535audio/cs5535audio_pcm.c | 15 |
7 files changed, 252 insertions, 3 deletions
diff --git a/include/sound/ac97_codec.h b/include/sound/ac97_codec.h index 9c309daf492b..251fc1cd5002 100644 --- a/include/sound/ac97_codec.h +++ b/include/sound/ac97_codec.h | |||
@@ -281,10 +281,12 @@ | |||
281 | /* specific - Analog Devices */ | 281 | /* specific - Analog Devices */ |
282 | #define AC97_AD_TEST 0x5a /* test register */ | 282 | #define AC97_AD_TEST 0x5a /* test register */ |
283 | #define AC97_AD_TEST2 0x5c /* undocumented test register 2 */ | 283 | #define AC97_AD_TEST2 0x5c /* undocumented test register 2 */ |
284 | #define AC97_AD_HPFD_SHIFT 12 /* High Pass Filter Disable */ | ||
284 | #define AC97_AD_CODEC_CFG 0x70 /* codec configuration */ | 285 | #define AC97_AD_CODEC_CFG 0x70 /* codec configuration */ |
285 | #define AC97_AD_JACK_SPDIF 0x72 /* Jack Sense & S/PDIF */ | 286 | #define AC97_AD_JACK_SPDIF 0x72 /* Jack Sense & S/PDIF */ |
286 | #define AC97_AD_SERIAL_CFG 0x74 /* Serial Configuration */ | 287 | #define AC97_AD_SERIAL_CFG 0x74 /* Serial Configuration */ |
287 | #define AC97_AD_MISC 0x76 /* Misc Control Bits */ | 288 | #define AC97_AD_MISC 0x76 /* Misc Control Bits */ |
289 | #define AC97_AD_VREFD_SHIFT 2 /* V_REFOUT Disable (AD1888) */ | ||
288 | 290 | ||
289 | /* specific - Cirrus Logic */ | 291 | /* specific - Cirrus Logic */ |
290 | #define AC97_CSR_ACMODE 0x5e /* AC Mode Register */ | 292 | #define AC97_CSR_ACMODE 0x5e /* AC Mode Register */ |
diff --git a/sound/pci/ac97/ac97_patch.c b/sound/pci/ac97/ac97_patch.c index 2c7cd97d2234..81bc93e5f1e3 100644 --- a/sound/pci/ac97/ac97_patch.c +++ b/sound/pci/ac97/ac97_patch.c | |||
@@ -2054,8 +2054,9 @@ static const struct snd_kcontrol_new snd_ac97_ad1888_controls[] = { | |||
2054 | .get = snd_ac97_ad1888_lohpsel_get, | 2054 | .get = snd_ac97_ad1888_lohpsel_get, |
2055 | .put = snd_ac97_ad1888_lohpsel_put | 2055 | .put = snd_ac97_ad1888_lohpsel_put |
2056 | }, | 2056 | }, |
2057 | AC97_SINGLE("V_REFOUT Enable", AC97_AD_MISC, 2, 1, 1), | 2057 | AC97_SINGLE("V_REFOUT Enable", AC97_AD_MISC, AC97_AD_VREFD_SHIFT, 1, 1), |
2058 | AC97_SINGLE("High Pass Filter Enable", AC97_AD_TEST2, 12, 1, 1), | 2058 | AC97_SINGLE("High Pass Filter Enable", AC97_AD_TEST2, |
2059 | AC97_AD_HPFD_SHIFT, 1, 1), | ||
2059 | AC97_SINGLE("Spread Front to Surround and Center/LFE", AC97_AD_MISC, 7, 1, 0), | 2060 | AC97_SINGLE("Spread Front to Surround and Center/LFE", AC97_AD_MISC, 7, 1, 0), |
2060 | { | 2061 | { |
2061 | .iface = SNDRV_CTL_ELEM_IFACE_MIXER, | 2062 | .iface = SNDRV_CTL_ELEM_IFACE_MIXER, |
diff --git a/sound/pci/cs5535audio/Makefile b/sound/pci/cs5535audio/Makefile index bb3d57e6a3cb..fda7a94c992f 100644 --- a/sound/pci/cs5535audio/Makefile +++ b/sound/pci/cs5535audio/Makefile | |||
@@ -4,6 +4,9 @@ | |||
4 | 4 | ||
5 | snd-cs5535audio-y := cs5535audio.o cs5535audio_pcm.o | 5 | snd-cs5535audio-y := cs5535audio.o cs5535audio_pcm.o |
6 | snd-cs5535audio-$(CONFIG_PM) += cs5535audio_pm.o | 6 | snd-cs5535audio-$(CONFIG_PM) += cs5535audio_pm.o |
7 | ifdef CONFIG_MGEODE_LX | ||
8 | snd-cs5535audio-$(CONFIG_OLPC) += cs5535audio_olpc.o | ||
9 | endif | ||
7 | 10 | ||
8 | # Toplevel Module Dependency | 11 | # Toplevel Module Dependency |
9 | obj-$(CONFIG_SND_CS5535AUDIO) += snd-cs5535audio.o | 12 | obj-$(CONFIG_SND_CS5535AUDIO) += snd-cs5535audio.o |
diff --git a/sound/pci/cs5535audio/cs5535audio.c b/sound/pci/cs5535audio/cs5535audio.c index 1d8b16052535..826e6dec2e97 100644 --- a/sound/pci/cs5535audio/cs5535audio.c +++ b/sound/pci/cs5535audio/cs5535audio.c | |||
@@ -159,10 +159,14 @@ static int __devinit snd_cs5535audio_mixer(struct cs5535audio *cs5535au) | |||
159 | return err; | 159 | return err; |
160 | 160 | ||
161 | memset(&ac97, 0, sizeof(ac97)); | 161 | memset(&ac97, 0, sizeof(ac97)); |
162 | ac97.scaps = AC97_SCAP_AUDIO|AC97_SCAP_SKIP_MODEM; | 162 | ac97.scaps = AC97_SCAP_AUDIO | AC97_SCAP_SKIP_MODEM |
163 | | AC97_SCAP_POWER_SAVE; | ||
163 | ac97.private_data = cs5535au; | 164 | ac97.private_data = cs5535au; |
164 | ac97.pci = cs5535au->pci; | 165 | ac97.pci = cs5535au->pci; |
165 | 166 | ||
167 | /* set any OLPC-specific scaps */ | ||
168 | olpc_prequirks(card, &ac97); | ||
169 | |||
166 | if ((err = snd_ac97_mixer(pbus, &ac97, &cs5535au->ac97)) < 0) { | 170 | if ((err = snd_ac97_mixer(pbus, &ac97, &cs5535au->ac97)) < 0) { |
167 | snd_printk(KERN_ERR "mixer failed\n"); | 171 | snd_printk(KERN_ERR "mixer failed\n"); |
168 | return err; | 172 | return err; |
@@ -170,6 +174,12 @@ static int __devinit snd_cs5535audio_mixer(struct cs5535audio *cs5535au) | |||
170 | 174 | ||
171 | snd_ac97_tune_hardware(cs5535au->ac97, ac97_quirks, ac97_quirk); | 175 | snd_ac97_tune_hardware(cs5535au->ac97, ac97_quirks, ac97_quirk); |
172 | 176 | ||
177 | err = olpc_quirks(card, cs5535au->ac97); | ||
178 | if (err < 0) { | ||
179 | snd_printk(KERN_ERR "olpc quirks failed\n"); | ||
180 | return err; | ||
181 | } | ||
182 | |||
173 | return 0; | 183 | return 0; |
174 | } | 184 | } |
175 | 185 | ||
diff --git a/sound/pci/cs5535audio/cs5535audio.h b/sound/pci/cs5535audio/cs5535audio.h index 66bae7664193..7a298ac662e3 100644 --- a/sound/pci/cs5535audio/cs5535audio.h +++ b/sound/pci/cs5535audio/cs5535audio.h | |||
@@ -78,6 +78,7 @@ struct cs5535audio_dma { | |||
78 | unsigned int buf_addr, buf_bytes; | 78 | unsigned int buf_addr, buf_bytes; |
79 | unsigned int period_bytes, periods; | 79 | unsigned int period_bytes, periods; |
80 | u32 saved_prd; | 80 | u32 saved_prd; |
81 | int pcm_open_flag; | ||
81 | }; | 82 | }; |
82 | 83 | ||
83 | struct cs5535audio { | 84 | struct cs5535audio { |
@@ -93,8 +94,46 @@ struct cs5535audio { | |||
93 | struct cs5535audio_dma dmas[NUM_CS5535AUDIO_DMAS]; | 94 | struct cs5535audio_dma dmas[NUM_CS5535AUDIO_DMAS]; |
94 | }; | 95 | }; |
95 | 96 | ||
97 | #ifdef CONFIG_PM | ||
96 | int snd_cs5535audio_suspend(struct pci_dev *pci, pm_message_t state); | 98 | int snd_cs5535audio_suspend(struct pci_dev *pci, pm_message_t state); |
97 | int snd_cs5535audio_resume(struct pci_dev *pci); | 99 | int snd_cs5535audio_resume(struct pci_dev *pci); |
100 | #endif | ||
101 | |||
102 | #if defined(CONFIG_OLPC) && defined(CONFIG_MGEODE_LX) | ||
103 | void __devinit olpc_prequirks(struct snd_card *card, | ||
104 | struct snd_ac97_template *ac97); | ||
105 | int __devinit olpc_quirks(struct snd_card *card, struct snd_ac97 *ac97); | ||
106 | void olpc_analog_input(struct snd_ac97 *ac97, int on); | ||
107 | void olpc_mic_bias(struct snd_ac97 *ac97, int on); | ||
108 | |||
109 | static inline void olpc_capture_open(struct snd_ac97 *ac97) | ||
110 | { | ||
111 | /* default to Analog Input off */ | ||
112 | olpc_analog_input(ac97, 0); | ||
113 | /* enable MIC Bias for recording */ | ||
114 | olpc_mic_bias(ac97, 1); | ||
115 | } | ||
116 | |||
117 | static inline void olpc_capture_close(struct snd_ac97 *ac97) | ||
118 | { | ||
119 | /* disable Analog Input */ | ||
120 | olpc_analog_input(ac97, 0); | ||
121 | /* disable the MIC Bias (so the recording LED turns off) */ | ||
122 | olpc_mic_bias(ac97, 0); | ||
123 | } | ||
124 | #else | ||
125 | static inline void olpc_prequirks(struct snd_card *card, | ||
126 | struct snd_ac97_template *ac97) { } | ||
127 | static inline int olpc_quirks(struct snd_card *card, struct snd_ac97 *ac97) | ||
128 | { | ||
129 | return 0; | ||
130 | } | ||
131 | static inline void olpc_analog_input(struct snd_ac97 *ac97, int on) { } | ||
132 | static inline void olpc_mic_bias(struct snd_ac97 *ac97, int on) { } | ||
133 | static inline void olpc_capture_open(struct snd_ac97 *ac97) { } | ||
134 | static inline void olpc_capture_close(struct snd_ac97 *ac97) { } | ||
135 | #endif | ||
136 | |||
98 | int __devinit snd_cs5535audio_pcm(struct cs5535audio *cs5535audio); | 137 | int __devinit snd_cs5535audio_pcm(struct cs5535audio *cs5535audio); |
99 | 138 | ||
100 | #endif /* __SOUND_CS5535AUDIO_H */ | 139 | #endif /* __SOUND_CS5535AUDIO_H */ |
diff --git a/sound/pci/cs5535audio/cs5535audio_olpc.c b/sound/pci/cs5535audio/cs5535audio_olpc.c new file mode 100644 index 000000000000..5c6814335cd7 --- /dev/null +++ b/sound/pci/cs5535audio/cs5535audio_olpc.c | |||
@@ -0,0 +1,179 @@ | |||
1 | /* | ||
2 | * OLPC XO-1 additional sound features | ||
3 | * | ||
4 | * Copyright © 2006 Jaya Kumar <jayakumar.lkml@gmail.com> | ||
5 | * Copyright © 2007-2008 Andres Salomon <dilinger@debian.org> | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License as published by | ||
9 | * the Free Software Foundation; either version 2 of the License, or | ||
10 | * (at your option) any later version. | ||
11 | */ | ||
12 | #include <sound/core.h> | ||
13 | #include <sound/info.h> | ||
14 | #include <sound/control.h> | ||
15 | #include <sound/ac97_codec.h> | ||
16 | |||
17 | #include <asm/olpc.h> | ||
18 | #include "cs5535audio.h" | ||
19 | |||
20 | /* | ||
21 | * OLPC has an additional feature on top of the regular AD1888 codec features. | ||
22 | * It has an Analog Input mode that is switched into (after disabling the | ||
23 | * High Pass Filter) via GPIO. It is supported on B2 and later models. | ||
24 | */ | ||
25 | void olpc_analog_input(struct snd_ac97 *ac97, int on) | ||
26 | { | ||
27 | int err; | ||
28 | |||
29 | if (!machine_is_olpc()) | ||
30 | return; | ||
31 | |||
32 | /* update the High Pass Filter (via AC97_AD_TEST2) */ | ||
33 | err = snd_ac97_update_bits(ac97, AC97_AD_TEST2, | ||
34 | 1 << AC97_AD_HPFD_SHIFT, on << AC97_AD_HPFD_SHIFT); | ||
35 | if (err < 0) { | ||
36 | snd_printk(KERN_ERR "setting High Pass Filter - %d\n", err); | ||
37 | return; | ||
38 | } | ||
39 | |||
40 | /* set Analog Input through GPIO */ | ||
41 | if (on) | ||
42 | geode_gpio_set(OLPC_GPIO_MIC_AC, GPIO_OUTPUT_VAL); | ||
43 | else | ||
44 | geode_gpio_clear(OLPC_GPIO_MIC_AC, GPIO_OUTPUT_VAL); | ||
45 | } | ||
46 | |||
47 | /* | ||
48 | * OLPC XO-1's V_REFOUT is a mic bias enable. | ||
49 | */ | ||
50 | void olpc_mic_bias(struct snd_ac97 *ac97, int on) | ||
51 | { | ||
52 | int err; | ||
53 | |||
54 | if (!machine_is_olpc()) | ||
55 | return; | ||
56 | |||
57 | on = on ? 0 : 1; | ||
58 | err = snd_ac97_update_bits(ac97, AC97_AD_MISC, | ||
59 | 1 << AC97_AD_VREFD_SHIFT, on << AC97_AD_VREFD_SHIFT); | ||
60 | if (err < 0) | ||
61 | snd_printk(KERN_ERR "setting MIC Bias - %d\n", err); | ||
62 | } | ||
63 | |||
64 | static int olpc_dc_info(struct snd_kcontrol *kctl, | ||
65 | struct snd_ctl_elem_info *uinfo) | ||
66 | { | ||
67 | uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN; | ||
68 | uinfo->count = 1; | ||
69 | uinfo->value.integer.min = 0; | ||
70 | uinfo->value.integer.max = 1; | ||
71 | return 0; | ||
72 | } | ||
73 | |||
74 | static int olpc_dc_get(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *v) | ||
75 | { | ||
76 | v->value.integer.value[0] = geode_gpio_isset(OLPC_GPIO_MIC_AC, | ||
77 | GPIO_OUTPUT_VAL); | ||
78 | return 0; | ||
79 | } | ||
80 | |||
81 | static int olpc_dc_put(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *v) | ||
82 | { | ||
83 | struct cs5535audio *cs5535au = snd_kcontrol_chip(kctl); | ||
84 | |||
85 | olpc_analog_input(cs5535au->ac97, v->value.integer.value[0]); | ||
86 | return 1; | ||
87 | } | ||
88 | |||
89 | static int olpc_mic_info(struct snd_kcontrol *kctl, | ||
90 | struct snd_ctl_elem_info *uinfo) | ||
91 | { | ||
92 | uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN; | ||
93 | uinfo->count = 1; | ||
94 | uinfo->value.integer.min = 0; | ||
95 | uinfo->value.integer.max = 1; | ||
96 | return 0; | ||
97 | } | ||
98 | |||
99 | static int olpc_mic_get(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *v) | ||
100 | { | ||
101 | struct cs5535audio *cs5535au = snd_kcontrol_chip(kctl); | ||
102 | struct snd_ac97 *ac97 = cs5535au->ac97; | ||
103 | int i; | ||
104 | |||
105 | i = (snd_ac97_read(ac97, AC97_AD_MISC) >> AC97_AD_VREFD_SHIFT) & 0x1; | ||
106 | v->value.integer.value[0] = i ? 0 : 1; | ||
107 | return 0; | ||
108 | } | ||
109 | |||
110 | static int olpc_mic_put(struct snd_kcontrol *kctl, struct snd_ctl_elem_value *v) | ||
111 | { | ||
112 | struct cs5535audio *cs5535au = snd_kcontrol_chip(kctl); | ||
113 | |||
114 | olpc_mic_bias(cs5535au->ac97, v->value.integer.value[0]); | ||
115 | return 1; | ||
116 | } | ||
117 | |||
118 | static struct snd_kcontrol_new olpc_cs5535audio_ctls[] __devinitdata = { | ||
119 | { | ||
120 | .iface = SNDRV_CTL_ELEM_IFACE_MIXER, | ||
121 | .name = "DC Mode Enable", | ||
122 | .info = olpc_dc_info, | ||
123 | .get = olpc_dc_get, | ||
124 | .put = olpc_dc_put, | ||
125 | .private_value = 0, | ||
126 | }, | ||
127 | { | ||
128 | .iface = SNDRV_CTL_ELEM_IFACE_MIXER, | ||
129 | .name = "MIC Bias Enable", | ||
130 | .info = olpc_mic_info, | ||
131 | .get = olpc_mic_get, | ||
132 | .put = olpc_mic_put, | ||
133 | .private_value = 0, | ||
134 | }, | ||
135 | }; | ||
136 | |||
137 | void __devinit olpc_prequirks(struct snd_card *card, | ||
138 | struct snd_ac97_template *ac97) | ||
139 | { | ||
140 | if (!machine_is_olpc()) | ||
141 | return; | ||
142 | |||
143 | /* invert EAPD if on an OLPC B3 or higher */ | ||
144 | if (olpc_board_at_least(olpc_board_pre(0xb3))) | ||
145 | ac97->scaps |= AC97_SCAP_INV_EAPD; | ||
146 | } | ||
147 | |||
148 | int __devinit olpc_quirks(struct snd_card *card, struct snd_ac97 *ac97) | ||
149 | { | ||
150 | struct snd_ctl_elem_id elem; | ||
151 | int i, err; | ||
152 | |||
153 | if (!machine_is_olpc()) | ||
154 | return 0; | ||
155 | |||
156 | /* drop the original AD1888 HPF control */ | ||
157 | memset(&elem, 0, sizeof(elem)); | ||
158 | elem.iface = SNDRV_CTL_ELEM_IFACE_MIXER; | ||
159 | strncpy(elem.name, "High Pass Filter Enable", sizeof(elem.name)); | ||
160 | snd_ctl_remove_id(card, &elem); | ||
161 | |||
162 | /* drop the original V_REFOUT control */ | ||
163 | memset(&elem, 0, sizeof(elem)); | ||
164 | elem.iface = SNDRV_CTL_ELEM_IFACE_MIXER; | ||
165 | strncpy(elem.name, "V_REFOUT Enable", sizeof(elem.name)); | ||
166 | snd_ctl_remove_id(card, &elem); | ||
167 | |||
168 | /* add the OLPC-specific controls */ | ||
169 | for (i = 0; i < ARRAY_SIZE(olpc_cs5535audio_ctls); i++) { | ||
170 | err = snd_ctl_add(card, snd_ctl_new1(&olpc_cs5535audio_ctls[i], | ||
171 | ac97->private_data)); | ||
172 | if (err < 0) | ||
173 | return err; | ||
174 | } | ||
175 | |||
176 | /* turn off the mic by default */ | ||
177 | olpc_mic_bias(ac97, 0); | ||
178 | return 0; | ||
179 | } | ||
diff --git a/sound/pci/cs5535audio/cs5535audio_pcm.c b/sound/pci/cs5535audio/cs5535audio_pcm.c index cdcda87116c3..0f48a871f17b 100644 --- a/sound/pci/cs5535audio/cs5535audio_pcm.c +++ b/sound/pci/cs5535audio/cs5535audio_pcm.c | |||
@@ -260,6 +260,9 @@ static int snd_cs5535audio_hw_params(struct snd_pcm_substream *substream, | |||
260 | err = cs5535audio_build_dma_packets(cs5535au, dma, substream, | 260 | err = cs5535audio_build_dma_packets(cs5535au, dma, substream, |
261 | params_periods(hw_params), | 261 | params_periods(hw_params), |
262 | params_period_bytes(hw_params)); | 262 | params_period_bytes(hw_params)); |
263 | if (!err) | ||
264 | dma->pcm_open_flag = 1; | ||
265 | |||
263 | return err; | 266 | return err; |
264 | } | 267 | } |
265 | 268 | ||
@@ -268,6 +271,15 @@ static int snd_cs5535audio_hw_free(struct snd_pcm_substream *substream) | |||
268 | struct cs5535audio *cs5535au = snd_pcm_substream_chip(substream); | 271 | struct cs5535audio *cs5535au = snd_pcm_substream_chip(substream); |
269 | struct cs5535audio_dma *dma = substream->runtime->private_data; | 272 | struct cs5535audio_dma *dma = substream->runtime->private_data; |
270 | 273 | ||
274 | if (dma->pcm_open_flag) { | ||
275 | if (substream == cs5535au->playback_substream) | ||
276 | snd_ac97_update_power(cs5535au->ac97, | ||
277 | AC97_PCM_FRONT_DAC_RATE, 0); | ||
278 | else | ||
279 | snd_ac97_update_power(cs5535au->ac97, | ||
280 | AC97_PCM_LR_ADC_RATE, 0); | ||
281 | dma->pcm_open_flag = 0; | ||
282 | } | ||
271 | cs5535audio_clear_dma_packets(cs5535au, dma, substream); | 283 | cs5535audio_clear_dma_packets(cs5535au, dma, substream); |
272 | return snd_pcm_lib_free_pages(substream); | 284 | return snd_pcm_lib_free_pages(substream); |
273 | } | 285 | } |
@@ -351,11 +363,14 @@ static int snd_cs5535audio_capture_open(struct snd_pcm_substream *substream) | |||
351 | if ((err = snd_pcm_hw_constraint_integer(runtime, | 363 | if ((err = snd_pcm_hw_constraint_integer(runtime, |
352 | SNDRV_PCM_HW_PARAM_PERIODS)) < 0) | 364 | SNDRV_PCM_HW_PARAM_PERIODS)) < 0) |
353 | return err; | 365 | return err; |
366 | olpc_capture_open(cs5535au->ac97); | ||
354 | return 0; | 367 | return 0; |
355 | } | 368 | } |
356 | 369 | ||
357 | static int snd_cs5535audio_capture_close(struct snd_pcm_substream *substream) | 370 | static int snd_cs5535audio_capture_close(struct snd_pcm_substream *substream) |
358 | { | 371 | { |
372 | struct cs5535audio *cs5535au = snd_pcm_substream_chip(substream); | ||
373 | olpc_capture_close(cs5535au->ac97); | ||
359 | return 0; | 374 | return 0; |
360 | } | 375 | } |
361 | 376 | ||