diff options
Diffstat (limited to 'sound/soc/codecs/cx20442.c')
-rw-r--r-- | sound/soc/codecs/cx20442.c | 198 |
1 files changed, 58 insertions, 140 deletions
diff --git a/sound/soc/codecs/cx20442.c b/sound/soc/codecs/cx20442.c index f07a415c753f..d68ea532cc7f 100644 --- a/sound/soc/codecs/cx20442.c +++ b/sound/soc/codecs/cx20442.c | |||
@@ -18,14 +18,14 @@ | |||
18 | 18 | ||
19 | #include <sound/core.h> | 19 | #include <sound/core.h> |
20 | #include <sound/initval.h> | 20 | #include <sound/initval.h> |
21 | #include <sound/soc-dapm.h> | 21 | #include <sound/soc.h> |
22 | 22 | ||
23 | #include "cx20442.h" | 23 | #include "cx20442.h" |
24 | 24 | ||
25 | 25 | ||
26 | struct cx20442_priv { | 26 | struct cx20442_priv { |
27 | struct snd_soc_codec codec; | 27 | enum snd_soc_control_type control_type; |
28 | u8 reg_cache[1]; | 28 | void *control_data; |
29 | }; | 29 | }; |
30 | 30 | ||
31 | #define CX20442_PM 0x0 | 31 | #define CX20442_PM 0x0 |
@@ -86,23 +86,12 @@ static const struct snd_soc_dapm_route cx20442_audio_map[] = { | |||
86 | {"ADC", NULL, "Input Mixer"}, | 86 | {"ADC", NULL, "Input Mixer"}, |
87 | }; | 87 | }; |
88 | 88 | ||
89 | static int cx20442_add_widgets(struct snd_soc_codec *codec) | ||
90 | { | ||
91 | snd_soc_dapm_new_controls(codec, cx20442_dapm_widgets, | ||
92 | ARRAY_SIZE(cx20442_dapm_widgets)); | ||
93 | |||
94 | snd_soc_dapm_add_routes(codec, cx20442_audio_map, | ||
95 | ARRAY_SIZE(cx20442_audio_map)); | ||
96 | |||
97 | return 0; | ||
98 | } | ||
99 | |||
100 | static unsigned int cx20442_read_reg_cache(struct snd_soc_codec *codec, | 89 | static unsigned int cx20442_read_reg_cache(struct snd_soc_codec *codec, |
101 | unsigned int reg) | 90 | unsigned int reg) |
102 | { | 91 | { |
103 | u8 *reg_cache = codec->reg_cache; | 92 | u8 *reg_cache = codec->reg_cache; |
104 | 93 | ||
105 | if (reg >= codec->reg_cache_size) | 94 | if (reg >= codec->driver->reg_cache_size) |
106 | return -EINVAL; | 95 | return -EINVAL; |
107 | 96 | ||
108 | return reg_cache[reg]; | 97 | return reg_cache[reg]; |
@@ -164,16 +153,17 @@ static int cx20442_pm_to_v253_vsp(u8 value) | |||
164 | static int cx20442_write(struct snd_soc_codec *codec, unsigned int reg, | 153 | static int cx20442_write(struct snd_soc_codec *codec, unsigned int reg, |
165 | unsigned int value) | 154 | unsigned int value) |
166 | { | 155 | { |
156 | struct cx20442_priv *cx20442 = snd_soc_codec_get_drvdata(codec); | ||
167 | u8 *reg_cache = codec->reg_cache; | 157 | u8 *reg_cache = codec->reg_cache; |
168 | int vls, vsp, old, len; | 158 | int vls, vsp, old, len; |
169 | char buf[18]; | 159 | char buf[18]; |
170 | 160 | ||
171 | if (reg >= codec->reg_cache_size) | 161 | if (reg >= codec->driver->reg_cache_size) |
172 | return -EINVAL; | 162 | return -EINVAL; |
173 | 163 | ||
174 | /* hw_write and control_data pointers required for talking to the modem | 164 | /* hw_write and control_data pointers required for talking to the modem |
175 | * are expected to be set by the line discipline initialization code */ | 165 | * are expected to be set by the line discipline initialization code */ |
176 | if (!codec->hw_write || !codec->control_data) | 166 | if (!codec->hw_write || !cx20442->control_data) |
177 | return -EIO; | 167 | return -EIO; |
178 | 168 | ||
179 | old = reg_cache[reg]; | 169 | old = reg_cache[reg]; |
@@ -202,17 +192,13 @@ static int cx20442_write(struct snd_soc_codec *codec, unsigned int reg, | |||
202 | return -ENOMEM; | 192 | return -ENOMEM; |
203 | 193 | ||
204 | dev_dbg(codec->dev, "%s: %s\n", __func__, buf); | 194 | dev_dbg(codec->dev, "%s: %s\n", __func__, buf); |
205 | if (codec->hw_write(codec->control_data, buf, len) != len) | 195 | if (codec->hw_write(cx20442->control_data, buf, len) != len) |
206 | return -EIO; | 196 | return -EIO; |
207 | 197 | ||
208 | return 0; | 198 | return 0; |
209 | } | 199 | } |
210 | 200 | ||
211 | 201 | ||
212 | /* Moved up here as line discipline referres it during initialization */ | ||
213 | static struct snd_soc_codec *cx20442_codec; | ||
214 | |||
215 | |||
216 | /* | 202 | /* |
217 | * Line discpline related code | 203 | * Line discpline related code |
218 | * | 204 | * |
@@ -228,15 +214,15 @@ static const char *v253_init = "ate0m0q0+fclass=8\r"; | |||
228 | /* Line discipline .open() */ | 214 | /* Line discipline .open() */ |
229 | static int v253_open(struct tty_struct *tty) | 215 | static int v253_open(struct tty_struct *tty) |
230 | { | 216 | { |
231 | struct snd_soc_codec *codec = cx20442_codec; | ||
232 | int ret, len = strlen(v253_init); | 217 | int ret, len = strlen(v253_init); |
233 | 218 | ||
234 | /* Doesn't make sense without write callback */ | 219 | /* Doesn't make sense without write callback */ |
235 | if (!tty->ops->write) | 220 | if (!tty->ops->write) |
236 | return -EINVAL; | 221 | return -EINVAL; |
237 | 222 | ||
238 | /* Pass the codec structure address for use by other ldisc callbacks */ | 223 | /* Won't work if no codec pointer has been passed by a card driver */ |
239 | tty->disc_data = codec; | 224 | if (!tty->disc_data) |
225 | return -ENODEV; | ||
240 | 226 | ||
241 | if (tty->ops->write(tty, v253_init, len) != len) { | 227 | if (tty->ops->write(tty, v253_init, len) != len) { |
242 | ret = -EIO; | 228 | ret = -EIO; |
@@ -253,16 +239,19 @@ err: | |||
253 | static void v253_close(struct tty_struct *tty) | 239 | static void v253_close(struct tty_struct *tty) |
254 | { | 240 | { |
255 | struct snd_soc_codec *codec = tty->disc_data; | 241 | struct snd_soc_codec *codec = tty->disc_data; |
242 | struct cx20442_priv *cx20442; | ||
256 | 243 | ||
257 | tty->disc_data = NULL; | 244 | tty->disc_data = NULL; |
258 | 245 | ||
259 | if (!codec) | 246 | if (!codec) |
260 | return; | 247 | return; |
261 | 248 | ||
249 | cx20442 = snd_soc_codec_get_drvdata(codec); | ||
250 | |||
262 | /* Prevent the codec driver from further accessing the modem */ | 251 | /* Prevent the codec driver from further accessing the modem */ |
263 | codec->hw_write = NULL; | 252 | codec->hw_write = NULL; |
264 | codec->control_data = NULL; | 253 | cx20442->control_data = NULL; |
265 | codec->pop_time = 0; | 254 | codec->card->pop_time = 0; |
266 | } | 255 | } |
267 | 256 | ||
268 | /* Line discipline .hangup() */ | 257 | /* Line discipline .hangup() */ |
@@ -277,17 +266,20 @@ static void v253_receive(struct tty_struct *tty, | |||
277 | const unsigned char *cp, char *fp, int count) | 266 | const unsigned char *cp, char *fp, int count) |
278 | { | 267 | { |
279 | struct snd_soc_codec *codec = tty->disc_data; | 268 | struct snd_soc_codec *codec = tty->disc_data; |
269 | struct cx20442_priv *cx20442; | ||
280 | 270 | ||
281 | if (!codec) | 271 | if (!codec) |
282 | return; | 272 | return; |
283 | 273 | ||
284 | if (!codec->control_data) { | 274 | cx20442 = snd_soc_codec_get_drvdata(codec); |
275 | |||
276 | if (!cx20442->control_data) { | ||
285 | /* First modem response, complete setup procedure */ | 277 | /* First modem response, complete setup procedure */ |
286 | 278 | ||
287 | /* Set up codec driver access to modem controls */ | 279 | /* Set up codec driver access to modem controls */ |
288 | codec->control_data = tty; | 280 | cx20442->control_data = tty; |
289 | codec->hw_write = (hw_write_t)tty->ops->write; | 281 | codec->hw_write = (hw_write_t)tty->ops->write; |
290 | codec->pop_time = 1; | 282 | codec->card->pop_time = 1; |
291 | } | 283 | } |
292 | } | 284 | } |
293 | 285 | ||
@@ -313,8 +305,8 @@ EXPORT_SYMBOL_GPL(v253_ops); | |||
313 | * Codec DAI | 305 | * Codec DAI |
314 | */ | 306 | */ |
315 | 307 | ||
316 | struct snd_soc_dai cx20442_dai = { | 308 | static struct snd_soc_dai_driver cx20442_dai = { |
317 | .name = "CX20442", | 309 | .name = "cx20442-voice", |
318 | .playback = { | 310 | .playback = { |
319 | .stream_name = "Playback", | 311 | .stream_name = "Playback", |
320 | .channels_min = 1, | 312 | .channels_min = 1, |
@@ -330,142 +322,68 @@ struct snd_soc_dai cx20442_dai = { | |||
330 | .formats = SNDRV_PCM_FMTBIT_S16_LE, | 322 | .formats = SNDRV_PCM_FMTBIT_S16_LE, |
331 | }, | 323 | }, |
332 | }; | 324 | }; |
333 | EXPORT_SYMBOL_GPL(cx20442_dai); | ||
334 | 325 | ||
335 | static int cx20442_codec_probe(struct platform_device *pdev) | 326 | static int cx20442_codec_probe(struct snd_soc_codec *codec) |
336 | { | 327 | { |
337 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 328 | struct cx20442_priv *cx20442; |
338 | struct snd_soc_codec *codec; | ||
339 | int ret; | ||
340 | |||
341 | if (!cx20442_codec) { | ||
342 | dev_err(&pdev->dev, "cx20442 not yet discovered\n"); | ||
343 | return -ENODEV; | ||
344 | } | ||
345 | codec = cx20442_codec; | ||
346 | |||
347 | socdev->card->codec = codec; | ||
348 | 329 | ||
349 | /* register pcms */ | 330 | cx20442 = kzalloc(sizeof(struct cx20442_priv), GFP_KERNEL); |
350 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | 331 | if (cx20442 == NULL) |
351 | if (ret < 0) { | 332 | return -ENOMEM; |
352 | dev_err(&pdev->dev, "failed to create pcms\n"); | 333 | snd_soc_codec_set_drvdata(codec, cx20442); |
353 | goto pcm_err; | ||
354 | } | ||
355 | 334 | ||
356 | cx20442_add_widgets(codec); | 335 | cx20442->control_data = NULL; |
336 | codec->hw_write = NULL; | ||
337 | codec->card->pop_time = 0; | ||
357 | 338 | ||
358 | pcm_err: | 339 | return 0; |
359 | return ret; | ||
360 | } | 340 | } |
361 | 341 | ||
362 | /* power down chip */ | 342 | /* power down chip */ |
363 | static int cx20442_codec_remove(struct platform_device *pdev) | 343 | static int cx20442_codec_remove(struct snd_soc_codec *codec) |
364 | { | 344 | { |
365 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 345 | struct cx20442_priv *cx20442 = snd_soc_codec_get_drvdata(codec); |
366 | 346 | ||
367 | snd_soc_free_pcms(socdev); | 347 | if (cx20442->control_data) { |
368 | snd_soc_dapm_free(socdev); | 348 | struct tty_struct *tty = cx20442->control_data; |
349 | tty_hangup(tty); | ||
350 | } | ||
369 | 351 | ||
352 | kfree(cx20442); | ||
370 | return 0; | 353 | return 0; |
371 | } | 354 | } |
372 | 355 | ||
373 | struct snd_soc_codec_device cx20442_codec_dev = { | 356 | static const u8 cx20442_reg; |
357 | |||
358 | static struct snd_soc_codec_driver cx20442_codec_dev = { | ||
374 | .probe = cx20442_codec_probe, | 359 | .probe = cx20442_codec_probe, |
375 | .remove = cx20442_codec_remove, | 360 | .remove = cx20442_codec_remove, |
361 | .reg_cache_default = &cx20442_reg, | ||
362 | .reg_cache_size = 1, | ||
363 | .reg_word_size = sizeof(u8), | ||
364 | .read = cx20442_read_reg_cache, | ||
365 | .write = cx20442_write, | ||
366 | .dapm_widgets = cx20442_dapm_widgets, | ||
367 | .num_dapm_widgets = ARRAY_SIZE(cx20442_dapm_widgets), | ||
368 | .dapm_routes = cx20442_audio_map, | ||
369 | .num_dapm_routes = ARRAY_SIZE(cx20442_audio_map), | ||
376 | }; | 370 | }; |
377 | EXPORT_SYMBOL_GPL(cx20442_codec_dev); | ||
378 | |||
379 | static int cx20442_register(struct cx20442_priv *cx20442) | ||
380 | { | ||
381 | struct snd_soc_codec *codec = &cx20442->codec; | ||
382 | int ret; | ||
383 | |||
384 | mutex_init(&codec->mutex); | ||
385 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
386 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
387 | |||
388 | codec->name = "CX20442"; | ||
389 | codec->owner = THIS_MODULE; | ||
390 | snd_soc_codec_set_drvdata(codec, cx20442); | ||
391 | |||
392 | codec->dai = &cx20442_dai; | ||
393 | codec->num_dai = 1; | ||
394 | |||
395 | codec->reg_cache = &cx20442->reg_cache; | ||
396 | codec->reg_cache_size = ARRAY_SIZE(cx20442->reg_cache); | ||
397 | codec->read = cx20442_read_reg_cache; | ||
398 | codec->write = cx20442_write; | ||
399 | |||
400 | codec->bias_level = SND_SOC_BIAS_OFF; | ||
401 | |||
402 | cx20442_dai.dev = codec->dev; | ||
403 | |||
404 | cx20442_codec = codec; | ||
405 | |||
406 | ret = snd_soc_register_codec(codec); | ||
407 | if (ret != 0) { | ||
408 | dev_err(codec->dev, "Failed to register codec: %d\n", ret); | ||
409 | goto err; | ||
410 | } | ||
411 | |||
412 | ret = snd_soc_register_dai(&cx20442_dai); | ||
413 | if (ret != 0) { | ||
414 | dev_err(codec->dev, "Failed to register DAI: %d\n", ret); | ||
415 | goto err_codec; | ||
416 | } | ||
417 | |||
418 | return 0; | ||
419 | |||
420 | err_codec: | ||
421 | snd_soc_unregister_codec(codec); | ||
422 | err: | ||
423 | cx20442_codec = NULL; | ||
424 | kfree(cx20442); | ||
425 | return ret; | ||
426 | } | ||
427 | |||
428 | static void cx20442_unregister(struct cx20442_priv *cx20442) | ||
429 | { | ||
430 | snd_soc_unregister_dai(&cx20442_dai); | ||
431 | snd_soc_unregister_codec(&cx20442->codec); | ||
432 | |||
433 | cx20442_codec = NULL; | ||
434 | kfree(cx20442); | ||
435 | } | ||
436 | 371 | ||
437 | static int cx20442_platform_probe(struct platform_device *pdev) | 372 | static int cx20442_platform_probe(struct platform_device *pdev) |
438 | { | 373 | { |
439 | struct cx20442_priv *cx20442; | 374 | return snd_soc_register_codec(&pdev->dev, |
440 | struct snd_soc_codec *codec; | 375 | &cx20442_codec_dev, &cx20442_dai, 1); |
441 | |||
442 | cx20442 = kzalloc(sizeof(struct cx20442_priv), GFP_KERNEL); | ||
443 | if (cx20442 == NULL) | ||
444 | return -ENOMEM; | ||
445 | |||
446 | codec = &cx20442->codec; | ||
447 | |||
448 | codec->control_data = NULL; | ||
449 | codec->hw_write = NULL; | ||
450 | codec->pop_time = 0; | ||
451 | |||
452 | codec->dev = &pdev->dev; | ||
453 | platform_set_drvdata(pdev, cx20442); | ||
454 | |||
455 | return cx20442_register(cx20442); | ||
456 | } | 376 | } |
457 | 377 | ||
458 | static int __exit cx20442_platform_remove(struct platform_device *pdev) | 378 | static int __exit cx20442_platform_remove(struct platform_device *pdev) |
459 | { | 379 | { |
460 | struct cx20442_priv *cx20442 = platform_get_drvdata(pdev); | 380 | snd_soc_unregister_codec(&pdev->dev); |
461 | |||
462 | cx20442_unregister(cx20442); | ||
463 | return 0; | 381 | return 0; |
464 | } | 382 | } |
465 | 383 | ||
466 | static struct platform_driver cx20442_platform_driver = { | 384 | static struct platform_driver cx20442_platform_driver = { |
467 | .driver = { | 385 | .driver = { |
468 | .name = "cx20442", | 386 | .name = "cx20442-codec", |
469 | .owner = THIS_MODULE, | 387 | .owner = THIS_MODULE, |
470 | }, | 388 | }, |
471 | .probe = cx20442_platform_probe, | 389 | .probe = cx20442_platform_probe, |
@@ -487,4 +405,4 @@ module_exit(cx20442_exit); | |||
487 | MODULE_DESCRIPTION("ASoC CX20442-11 voice modem codec driver"); | 405 | MODULE_DESCRIPTION("ASoC CX20442-11 voice modem codec driver"); |
488 | MODULE_AUTHOR("Janusz Krzysztofik"); | 406 | MODULE_AUTHOR("Janusz Krzysztofik"); |
489 | MODULE_LICENSE("GPL"); | 407 | MODULE_LICENSE("GPL"); |
490 | MODULE_ALIAS("platform:cx20442"); | 408 | MODULE_ALIAS("platform:cx20442-codec"); |