diff options
author | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
---|---|---|
committer | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
commit | c71c03bda1e86c9d5198c5d83f712e695c4f2a1e (patch) | |
tree | ecb166cb3e2b7e2adb3b5e292245fefd23381ac8 /sound/soc/soc-jack.c | |
parent | ea53c912f8a86a8567697115b6a0d8152beee5c8 (diff) | |
parent | 6a00f206debf8a5c8899055726ad127dbeeed098 (diff) |
Merge branch 'mpi-master' into wip-k-fmlpwip-k-fmlp
Conflicts:
litmus/sched_cedf.c
Diffstat (limited to 'sound/soc/soc-jack.c')
-rw-r--r-- | sound/soc/soc-jack.c | 108 |
1 files changed, 87 insertions, 21 deletions
diff --git a/sound/soc/soc-jack.c b/sound/soc/soc-jack.c index 29159e1781d0..7c17b98d5846 100644 --- a/sound/soc/soc-jack.c +++ b/sound/soc/soc-jack.c | |||
@@ -13,11 +13,11 @@ | |||
13 | 13 | ||
14 | #include <sound/jack.h> | 14 | #include <sound/jack.h> |
15 | #include <sound/soc.h> | 15 | #include <sound/soc.h> |
16 | #include <sound/soc-dapm.h> | ||
17 | #include <linux/gpio.h> | 16 | #include <linux/gpio.h> |
18 | #include <linux/interrupt.h> | 17 | #include <linux/interrupt.h> |
19 | #include <linux/workqueue.h> | 18 | #include <linux/workqueue.h> |
20 | #include <linux/delay.h> | 19 | #include <linux/delay.h> |
20 | #include <trace/events/asoc.h> | ||
21 | 21 | ||
22 | /** | 22 | /** |
23 | * snd_soc_jack_new - Create a new jack | 23 | * snd_soc_jack_new - Create a new jack |
@@ -32,14 +32,15 @@ | |||
32 | * Returns zero if successful, or a negative error code on failure. | 32 | * Returns zero if successful, or a negative error code on failure. |
33 | * On success jack will be initialised. | 33 | * On success jack will be initialised. |
34 | */ | 34 | */ |
35 | int snd_soc_jack_new(struct snd_soc_card *card, const char *id, int type, | 35 | int snd_soc_jack_new(struct snd_soc_codec *codec, const char *id, int type, |
36 | struct snd_soc_jack *jack) | 36 | struct snd_soc_jack *jack) |
37 | { | 37 | { |
38 | jack->card = card; | 38 | jack->codec = codec; |
39 | INIT_LIST_HEAD(&jack->pins); | 39 | INIT_LIST_HEAD(&jack->pins); |
40 | INIT_LIST_HEAD(&jack->jack_zones); | ||
40 | BLOCKING_INIT_NOTIFIER_HEAD(&jack->notifier); | 41 | BLOCKING_INIT_NOTIFIER_HEAD(&jack->notifier); |
41 | 42 | ||
42 | return snd_jack_new(card->codec->card, id, type, &jack->jack); | 43 | return snd_jack_new(codec->card->snd_card, id, type, &jack->jack); |
43 | } | 44 | } |
44 | EXPORT_SYMBOL_GPL(snd_soc_jack_new); | 45 | EXPORT_SYMBOL_GPL(snd_soc_jack_new); |
45 | 46 | ||
@@ -60,14 +61,18 @@ EXPORT_SYMBOL_GPL(snd_soc_jack_new); | |||
60 | void snd_soc_jack_report(struct snd_soc_jack *jack, int status, int mask) | 61 | void snd_soc_jack_report(struct snd_soc_jack *jack, int status, int mask) |
61 | { | 62 | { |
62 | struct snd_soc_codec *codec; | 63 | struct snd_soc_codec *codec; |
64 | struct snd_soc_dapm_context *dapm; | ||
63 | struct snd_soc_jack_pin *pin; | 65 | struct snd_soc_jack_pin *pin; |
64 | int enable; | 66 | int enable; |
65 | int oldstatus; | 67 | int oldstatus; |
66 | 68 | ||
69 | trace_snd_soc_jack_report(jack, mask, status); | ||
70 | |||
67 | if (!jack) | 71 | if (!jack) |
68 | return; | 72 | return; |
69 | 73 | ||
70 | codec = jack->card->codec; | 74 | codec = jack->codec; |
75 | dapm = &codec->dapm; | ||
71 | 76 | ||
72 | mutex_lock(&codec->mutex); | 77 | mutex_lock(&codec->mutex); |
73 | 78 | ||
@@ -81,6 +86,8 @@ void snd_soc_jack_report(struct snd_soc_jack *jack, int status, int mask) | |||
81 | if (mask && (jack->status == oldstatus)) | 86 | if (mask && (jack->status == oldstatus)) |
82 | goto out; | 87 | goto out; |
83 | 88 | ||
89 | trace_snd_soc_jack_notify(jack, status); | ||
90 | |||
84 | list_for_each_entry(pin, &jack->pins, list) { | 91 | list_for_each_entry(pin, &jack->pins, list) { |
85 | enable = pin->mask & jack->status; | 92 | enable = pin->mask & jack->status; |
86 | 93 | ||
@@ -88,15 +95,15 @@ void snd_soc_jack_report(struct snd_soc_jack *jack, int status, int mask) | |||
88 | enable = !enable; | 95 | enable = !enable; |
89 | 96 | ||
90 | if (enable) | 97 | if (enable) |
91 | snd_soc_dapm_enable_pin(codec, pin->pin); | 98 | snd_soc_dapm_enable_pin(dapm, pin->pin); |
92 | else | 99 | else |
93 | snd_soc_dapm_disable_pin(codec, pin->pin); | 100 | snd_soc_dapm_disable_pin(dapm, pin->pin); |
94 | } | 101 | } |
95 | 102 | ||
96 | /* Report before the DAPM sync to help users updating micbias status */ | 103 | /* Report before the DAPM sync to help users updating micbias status */ |
97 | blocking_notifier_call_chain(&jack->notifier, status, NULL); | 104 | blocking_notifier_call_chain(&jack->notifier, status, jack); |
98 | 105 | ||
99 | snd_soc_dapm_sync(codec); | 106 | snd_soc_dapm_sync(dapm); |
100 | 107 | ||
101 | snd_jack_report(jack->jack, status); | 108 | snd_jack_report(jack->jack, status); |
102 | 109 | ||
@@ -106,6 +113,51 @@ out: | |||
106 | EXPORT_SYMBOL_GPL(snd_soc_jack_report); | 113 | EXPORT_SYMBOL_GPL(snd_soc_jack_report); |
107 | 114 | ||
108 | /** | 115 | /** |
116 | * snd_soc_jack_add_zones - Associate voltage zones with jack | ||
117 | * | ||
118 | * @jack: ASoC jack | ||
119 | * @count: Number of zones | ||
120 | * @zone: Array of zones | ||
121 | * | ||
122 | * After this function has been called the zones specified in the | ||
123 | * array will be associated with the jack. | ||
124 | */ | ||
125 | int snd_soc_jack_add_zones(struct snd_soc_jack *jack, int count, | ||
126 | struct snd_soc_jack_zone *zones) | ||
127 | { | ||
128 | int i; | ||
129 | |||
130 | for (i = 0; i < count; i++) { | ||
131 | INIT_LIST_HEAD(&zones[i].list); | ||
132 | list_add(&(zones[i].list), &jack->jack_zones); | ||
133 | } | ||
134 | return 0; | ||
135 | } | ||
136 | EXPORT_SYMBOL_GPL(snd_soc_jack_add_zones); | ||
137 | |||
138 | /** | ||
139 | * snd_soc_jack_get_type - Based on the mic bias value, this function returns | ||
140 | * the type of jack from the zones delcared in the jack type | ||
141 | * | ||
142 | * @micbias_voltage: mic bias voltage at adc channel when jack is plugged in | ||
143 | * | ||
144 | * Based on the mic bias value passed, this function helps identify | ||
145 | * the type of jack from the already delcared jack zones | ||
146 | */ | ||
147 | int snd_soc_jack_get_type(struct snd_soc_jack *jack, int micbias_voltage) | ||
148 | { | ||
149 | struct snd_soc_jack_zone *zone; | ||
150 | |||
151 | list_for_each_entry(zone, &jack->jack_zones, list) { | ||
152 | if (micbias_voltage >= zone->min_mv && | ||
153 | micbias_voltage < zone->max_mv) | ||
154 | return zone->jack_type; | ||
155 | } | ||
156 | return 0; | ||
157 | } | ||
158 | EXPORT_SYMBOL_GPL(snd_soc_jack_get_type); | ||
159 | |||
160 | /** | ||
109 | * snd_soc_jack_add_pins - Associate DAPM pins with an ASoC jack | 161 | * snd_soc_jack_add_pins - Associate DAPM pins with an ASoC jack |
110 | * | 162 | * |
111 | * @jack: ASoC jack | 163 | * @jack: ASoC jack |
@@ -188,10 +240,7 @@ static void snd_soc_jack_gpio_detect(struct snd_soc_jack_gpio *gpio) | |||
188 | int enable; | 240 | int enable; |
189 | int report; | 241 | int report; |
190 | 242 | ||
191 | if (gpio->debounce_time > 0) | 243 | enable = gpio_get_value_cansleep(gpio->gpio); |
192 | mdelay(gpio->debounce_time); | ||
193 | |||
194 | enable = gpio_get_value(gpio->gpio); | ||
195 | if (gpio->invert) | 244 | if (gpio->invert) |
196 | enable = !enable; | 245 | enable = !enable; |
197 | 246 | ||
@@ -210,8 +259,15 @@ static void snd_soc_jack_gpio_detect(struct snd_soc_jack_gpio *gpio) | |||
210 | static irqreturn_t gpio_handler(int irq, void *data) | 259 | static irqreturn_t gpio_handler(int irq, void *data) |
211 | { | 260 | { |
212 | struct snd_soc_jack_gpio *gpio = data; | 261 | struct snd_soc_jack_gpio *gpio = data; |
262 | struct device *dev = gpio->jack->codec->card->dev; | ||
263 | |||
264 | trace_snd_soc_jack_irq(gpio->name); | ||
213 | 265 | ||
214 | schedule_work(&gpio->work); | 266 | if (device_may_wakeup(dev)) |
267 | pm_wakeup_event(dev, gpio->debounce_time + 50); | ||
268 | |||
269 | schedule_delayed_work(&gpio->work, | ||
270 | msecs_to_jiffies(gpio->debounce_time)); | ||
215 | 271 | ||
216 | return IRQ_HANDLED; | 272 | return IRQ_HANDLED; |
217 | } | 273 | } |
@@ -221,7 +277,7 @@ static void gpio_work(struct work_struct *work) | |||
221 | { | 277 | { |
222 | struct snd_soc_jack_gpio *gpio; | 278 | struct snd_soc_jack_gpio *gpio; |
223 | 279 | ||
224 | gpio = container_of(work, struct snd_soc_jack_gpio, work); | 280 | gpio = container_of(work, struct snd_soc_jack_gpio, work.work); |
225 | snd_soc_jack_gpio_detect(gpio); | 281 | snd_soc_jack_gpio_detect(gpio); |
226 | } | 282 | } |
227 | 283 | ||
@@ -262,17 +318,26 @@ int snd_soc_jack_add_gpios(struct snd_soc_jack *jack, int count, | |||
262 | if (ret) | 318 | if (ret) |
263 | goto err; | 319 | goto err; |
264 | 320 | ||
265 | INIT_WORK(&gpios[i].work, gpio_work); | 321 | INIT_DELAYED_WORK(&gpios[i].work, gpio_work); |
266 | gpios[i].jack = jack; | 322 | gpios[i].jack = jack; |
267 | 323 | ||
268 | ret = request_irq(gpio_to_irq(gpios[i].gpio), | 324 | ret = request_any_context_irq(gpio_to_irq(gpios[i].gpio), |
269 | gpio_handler, | 325 | gpio_handler, |
270 | IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, | 326 | IRQF_TRIGGER_RISING | |
271 | jack->card->dev->driver->name, | 327 | IRQF_TRIGGER_FALLING, |
272 | &gpios[i]); | 328 | gpios[i].name, |
329 | &gpios[i]); | ||
273 | if (ret) | 330 | if (ret) |
274 | goto err; | 331 | goto err; |
275 | 332 | ||
333 | if (gpios[i].wake) { | ||
334 | ret = irq_set_irq_wake(gpio_to_irq(gpios[i].gpio), 1); | ||
335 | if (ret != 0) | ||
336 | printk(KERN_ERR | ||
337 | "Failed to mark GPIO %d as wake source: %d\n", | ||
338 | gpios[i].gpio, ret); | ||
339 | } | ||
340 | |||
276 | #ifdef CONFIG_GPIO_SYSFS | 341 | #ifdef CONFIG_GPIO_SYSFS |
277 | /* Expose GPIO value over sysfs for diagnostic purposes */ | 342 | /* Expose GPIO value over sysfs for diagnostic purposes */ |
278 | gpio_export(gpios[i].gpio, false); | 343 | gpio_export(gpios[i].gpio, false); |
@@ -312,6 +377,7 @@ void snd_soc_jack_free_gpios(struct snd_soc_jack *jack, int count, | |||
312 | gpio_unexport(gpios[i].gpio); | 377 | gpio_unexport(gpios[i].gpio); |
313 | #endif | 378 | #endif |
314 | free_irq(gpio_to_irq(gpios[i].gpio), &gpios[i]); | 379 | free_irq(gpio_to_irq(gpios[i].gpio), &gpios[i]); |
380 | cancel_delayed_work_sync(&gpios[i].work); | ||
315 | gpio_free(gpios[i].gpio); | 381 | gpio_free(gpios[i].gpio); |
316 | gpios[i].jack = NULL; | 382 | gpios[i].jack = NULL; |
317 | } | 383 | } |