diff options
Diffstat (limited to 'sound/soc/jz4740')
-rw-r--r-- | sound/soc/jz4740/jz4740-i2s.c | 107 | ||||
-rw-r--r-- | sound/soc/jz4740/jz4740-i2s.h | 2 | ||||
-rw-r--r-- | sound/soc/jz4740/jz4740-pcm.c | 18 | ||||
-rw-r--r-- | sound/soc/jz4740/jz4740-pcm.h | 2 | ||||
-rw-r--r-- | sound/soc/jz4740/qi_lb60.c | 73 |
5 files changed, 86 insertions, 116 deletions
diff --git a/sound/soc/jz4740/jz4740-i2s.c b/sound/soc/jz4740/jz4740-i2s.c index eb518f0c5e01..cd22a54b2f14 100644 --- a/sound/soc/jz4740/jz4740-i2s.c +++ b/sound/soc/jz4740/jz4740-i2s.c | |||
@@ -28,7 +28,6 @@ | |||
28 | #include <sound/pcm.h> | 28 | #include <sound/pcm.h> |
29 | #include <sound/pcm_params.h> | 29 | #include <sound/pcm_params.h> |
30 | #include <sound/soc.h> | 30 | #include <sound/soc.h> |
31 | #include <sound/soc-dapm.h> | ||
32 | #include <sound/initval.h> | 31 | #include <sound/initval.h> |
33 | 32 | ||
34 | #include "jz4740-i2s.h" | 33 | #include "jz4740-i2s.h" |
@@ -106,15 +105,10 @@ static inline void jz4740_i2s_write(const struct jz4740_i2s *i2s, | |||
106 | writel(value, i2s->base + reg); | 105 | writel(value, i2s->base + reg); |
107 | } | 106 | } |
108 | 107 | ||
109 | static inline struct jz4740_i2s *jz4740_dai_to_i2s(struct snd_soc_dai *dai) | ||
110 | { | ||
111 | return dai->private_data; | ||
112 | } | ||
113 | |||
114 | static int jz4740_i2s_startup(struct snd_pcm_substream *substream, | 108 | static int jz4740_i2s_startup(struct snd_pcm_substream *substream, |
115 | struct snd_soc_dai *dai) | 109 | struct snd_soc_dai *dai) |
116 | { | 110 | { |
117 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 111 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
118 | uint32_t conf, ctrl; | 112 | uint32_t conf, ctrl; |
119 | 113 | ||
120 | if (dai->active) | 114 | if (dai->active) |
@@ -136,10 +130,10 @@ static int jz4740_i2s_startup(struct snd_pcm_substream *substream, | |||
136 | static void jz4740_i2s_shutdown(struct snd_pcm_substream *substream, | 130 | static void jz4740_i2s_shutdown(struct snd_pcm_substream *substream, |
137 | struct snd_soc_dai *dai) | 131 | struct snd_soc_dai *dai) |
138 | { | 132 | { |
139 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 133 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
140 | uint32_t conf; | 134 | uint32_t conf; |
141 | 135 | ||
142 | if (!dai->active) | 136 | if (dai->active) |
143 | return; | 137 | return; |
144 | 138 | ||
145 | conf = jz4740_i2s_read(i2s, JZ_REG_AIC_CONF); | 139 | conf = jz4740_i2s_read(i2s, JZ_REG_AIC_CONF); |
@@ -152,7 +146,7 @@ static void jz4740_i2s_shutdown(struct snd_pcm_substream *substream, | |||
152 | static int jz4740_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | 146 | static int jz4740_i2s_trigger(struct snd_pcm_substream *substream, int cmd, |
153 | struct snd_soc_dai *dai) | 147 | struct snd_soc_dai *dai) |
154 | { | 148 | { |
155 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 149 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
156 | 150 | ||
157 | uint32_t ctrl; | 151 | uint32_t ctrl; |
158 | uint32_t mask; | 152 | uint32_t mask; |
@@ -186,7 +180,7 @@ static int jz4740_i2s_trigger(struct snd_pcm_substream *substream, int cmd, | |||
186 | 180 | ||
187 | static int jz4740_i2s_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) | 181 | static int jz4740_i2s_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) |
188 | { | 182 | { |
189 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 183 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
190 | 184 | ||
191 | uint32_t format = 0; | 185 | uint32_t format = 0; |
192 | uint32_t conf; | 186 | uint32_t conf; |
@@ -238,7 +232,7 @@ static int jz4740_i2s_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) | |||
238 | static int jz4740_i2s_hw_params(struct snd_pcm_substream *substream, | 232 | static int jz4740_i2s_hw_params(struct snd_pcm_substream *substream, |
239 | struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) | 233 | struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) |
240 | { | 234 | { |
241 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 235 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
242 | enum jz4740_dma_width dma_width; | 236 | enum jz4740_dma_width dma_width; |
243 | struct jz4740_pcm_config *pcm_config; | 237 | struct jz4740_pcm_config *pcm_config; |
244 | unsigned int sample_size; | 238 | unsigned int sample_size; |
@@ -288,7 +282,7 @@ static int jz4740_i2s_hw_params(struct snd_pcm_substream *substream, | |||
288 | static int jz4740_i2s_set_sysclk(struct snd_soc_dai *dai, int clk_id, | 282 | static int jz4740_i2s_set_sysclk(struct snd_soc_dai *dai, int clk_id, |
289 | unsigned int freq, int dir) | 283 | unsigned int freq, int dir) |
290 | { | 284 | { |
291 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 285 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
292 | struct clk *parent; | 286 | struct clk *parent; |
293 | int ret = 0; | 287 | int ret = 0; |
294 | 288 | ||
@@ -312,7 +306,7 @@ static int jz4740_i2s_set_sysclk(struct snd_soc_dai *dai, int clk_id, | |||
312 | 306 | ||
313 | static int jz4740_i2s_suspend(struct snd_soc_dai *dai) | 307 | static int jz4740_i2s_suspend(struct snd_soc_dai *dai) |
314 | { | 308 | { |
315 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 309 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
316 | uint32_t conf; | 310 | uint32_t conf; |
317 | 311 | ||
318 | if (dai->active) { | 312 | if (dai->active) { |
@@ -330,7 +324,7 @@ static int jz4740_i2s_suspend(struct snd_soc_dai *dai) | |||
330 | 324 | ||
331 | static int jz4740_i2s_resume(struct snd_soc_dai *dai) | 325 | static int jz4740_i2s_resume(struct snd_soc_dai *dai) |
332 | { | 326 | { |
333 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 327 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
334 | uint32_t conf; | 328 | uint32_t conf; |
335 | 329 | ||
336 | clk_enable(i2s->clk_aic); | 330 | clk_enable(i2s->clk_aic); |
@@ -346,11 +340,38 @@ static int jz4740_i2s_resume(struct snd_soc_dai *dai) | |||
346 | return 0; | 340 | return 0; |
347 | } | 341 | } |
348 | 342 | ||
349 | static int jz4740_i2s_probe(struct platform_device *pdev, struct snd_soc_dai *dai) | 343 | static void jz4740_i2c_init_pcm_config(struct jz4740_i2s *i2s) |
344 | { | ||
345 | struct jz4740_dma_config *dma_config; | ||
346 | |||
347 | /* Playback */ | ||
348 | dma_config = &i2s->pcm_config_playback.dma_config; | ||
349 | dma_config->src_width = JZ4740_DMA_WIDTH_32BIT, | ||
350 | dma_config->transfer_size = JZ4740_DMA_TRANSFER_SIZE_16BYTE; | ||
351 | dma_config->request_type = JZ4740_DMA_TYPE_AIC_TRANSMIT; | ||
352 | dma_config->flags = JZ4740_DMA_SRC_AUTOINC; | ||
353 | dma_config->mode = JZ4740_DMA_MODE_SINGLE; | ||
354 | i2s->pcm_config_playback.fifo_addr = i2s->phys_base + JZ_REG_AIC_FIFO; | ||
355 | |||
356 | /* Capture */ | ||
357 | dma_config = &i2s->pcm_config_capture.dma_config; | ||
358 | dma_config->dst_width = JZ4740_DMA_WIDTH_32BIT, | ||
359 | dma_config->transfer_size = JZ4740_DMA_TRANSFER_SIZE_16BYTE; | ||
360 | dma_config->request_type = JZ4740_DMA_TYPE_AIC_RECEIVE; | ||
361 | dma_config->flags = JZ4740_DMA_DST_AUTOINC; | ||
362 | dma_config->mode = JZ4740_DMA_MODE_SINGLE; | ||
363 | i2s->pcm_config_capture.fifo_addr = i2s->phys_base + JZ_REG_AIC_FIFO; | ||
364 | } | ||
365 | |||
366 | static int jz4740_i2s_dai_probe(struct snd_soc_dai *dai) | ||
350 | { | 367 | { |
351 | struct jz4740_i2s *i2s = jz4740_dai_to_i2s(dai); | 368 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); |
352 | uint32_t conf; | 369 | uint32_t conf; |
353 | 370 | ||
371 | clk_enable(i2s->clk_aic); | ||
372 | |||
373 | jz4740_i2c_init_pcm_config(i2s); | ||
374 | |||
354 | conf = (7 << JZ_AIC_CONF_FIFO_RX_THRESHOLD_OFFSET) | | 375 | conf = (7 << JZ_AIC_CONF_FIFO_RX_THRESHOLD_OFFSET) | |
355 | (8 << JZ_AIC_CONF_FIFO_TX_THRESHOLD_OFFSET) | | 376 | (8 << JZ_AIC_CONF_FIFO_TX_THRESHOLD_OFFSET) | |
356 | JZ_AIC_CONF_OVERFLOW_PLAY_LAST | | 377 | JZ_AIC_CONF_OVERFLOW_PLAY_LAST | |
@@ -363,6 +384,14 @@ static int jz4740_i2s_probe(struct platform_device *pdev, struct snd_soc_dai *da | |||
363 | return 0; | 384 | return 0; |
364 | } | 385 | } |
365 | 386 | ||
387 | static int jz4740_i2s_dai_remove(struct snd_soc_dai *dai) | ||
388 | { | ||
389 | struct jz4740_i2s *i2s = snd_soc_dai_get_drvdata(dai); | ||
390 | |||
391 | clk_disable(i2s->clk_aic); | ||
392 | return 0; | ||
393 | } | ||
394 | |||
366 | static struct snd_soc_dai_ops jz4740_i2s_dai_ops = { | 395 | static struct snd_soc_dai_ops jz4740_i2s_dai_ops = { |
367 | .startup = jz4740_i2s_startup, | 396 | .startup = jz4740_i2s_startup, |
368 | .shutdown = jz4740_i2s_shutdown, | 397 | .shutdown = jz4740_i2s_shutdown, |
@@ -375,9 +404,9 @@ static struct snd_soc_dai_ops jz4740_i2s_dai_ops = { | |||
375 | #define JZ4740_I2S_FMTS (SNDRV_PCM_FMTBIT_S8 | \ | 404 | #define JZ4740_I2S_FMTS (SNDRV_PCM_FMTBIT_S8 | \ |
376 | SNDRV_PCM_FMTBIT_S16_LE) | 405 | SNDRV_PCM_FMTBIT_S16_LE) |
377 | 406 | ||
378 | struct snd_soc_dai jz4740_i2s_dai = { | 407 | static struct snd_soc_dai_driver jz4740_i2s_dai = { |
379 | .name = "jz4740-i2s", | 408 | .probe = jz4740_i2s_dai_probe, |
380 | .probe = jz4740_i2s_probe, | 409 | .remove = jz4740_i2s_dai_remove, |
381 | .playback = { | 410 | .playback = { |
382 | .channels_min = 1, | 411 | .channels_min = 1, |
383 | .channels_max = 2, | 412 | .channels_max = 2, |
@@ -395,30 +424,6 @@ struct snd_soc_dai jz4740_i2s_dai = { | |||
395 | .suspend = jz4740_i2s_suspend, | 424 | .suspend = jz4740_i2s_suspend, |
396 | .resume = jz4740_i2s_resume, | 425 | .resume = jz4740_i2s_resume, |
397 | }; | 426 | }; |
398 | EXPORT_SYMBOL_GPL(jz4740_i2s_dai); | ||
399 | |||
400 | static void __devinit jz4740_i2c_init_pcm_config(struct jz4740_i2s *i2s) | ||
401 | { | ||
402 | struct jz4740_dma_config *dma_config; | ||
403 | |||
404 | /* Playback */ | ||
405 | dma_config = &i2s->pcm_config_playback.dma_config; | ||
406 | dma_config->src_width = JZ4740_DMA_WIDTH_32BIT, | ||
407 | dma_config->transfer_size = JZ4740_DMA_TRANSFER_SIZE_16BYTE; | ||
408 | dma_config->request_type = JZ4740_DMA_TYPE_AIC_TRANSMIT; | ||
409 | dma_config->flags = JZ4740_DMA_SRC_AUTOINC; | ||
410 | dma_config->mode = JZ4740_DMA_MODE_SINGLE; | ||
411 | i2s->pcm_config_playback.fifo_addr = i2s->phys_base + JZ_REG_AIC_FIFO; | ||
412 | |||
413 | /* Capture */ | ||
414 | dma_config = &i2s->pcm_config_capture.dma_config; | ||
415 | dma_config->dst_width = JZ4740_DMA_WIDTH_32BIT, | ||
416 | dma_config->transfer_size = JZ4740_DMA_TRANSFER_SIZE_16BYTE; | ||
417 | dma_config->request_type = JZ4740_DMA_TYPE_AIC_RECEIVE; | ||
418 | dma_config->flags = JZ4740_DMA_DST_AUTOINC; | ||
419 | dma_config->mode = JZ4740_DMA_MODE_SINGLE; | ||
420 | i2s->pcm_config_capture.fifo_addr = i2s->phys_base + JZ_REG_AIC_FIFO; | ||
421 | } | ||
422 | 427 | ||
423 | static int __devinit jz4740_i2s_dev_probe(struct platform_device *pdev) | 428 | static int __devinit jz4740_i2s_dev_probe(struct platform_device *pdev) |
424 | { | 429 | { |
@@ -463,24 +468,17 @@ static int __devinit jz4740_i2s_dev_probe(struct platform_device *pdev) | |||
463 | goto err_clk_put_aic; | 468 | goto err_clk_put_aic; |
464 | } | 469 | } |
465 | 470 | ||
466 | clk_enable(i2s->clk_aic); | 471 | platform_set_drvdata(pdev, i2s); |
467 | 472 | ret = snd_soc_register_dai(&pdev->dev, &jz4740_i2s_dai); | |
468 | jz4740_i2c_init_pcm_config(i2s); | ||
469 | |||
470 | jz4740_i2s_dai.private_data = i2s; | ||
471 | ret = snd_soc_register_dai(&jz4740_i2s_dai); | ||
472 | 473 | ||
473 | if (ret) { | 474 | if (ret) { |
474 | dev_err(&pdev->dev, "Failed to register DAI\n"); | 475 | dev_err(&pdev->dev, "Failed to register DAI\n"); |
475 | goto err_clk_put_i2s; | 476 | goto err_clk_put_i2s; |
476 | } | 477 | } |
477 | 478 | ||
478 | platform_set_drvdata(pdev, i2s); | ||
479 | |||
480 | return 0; | 479 | return 0; |
481 | 480 | ||
482 | err_clk_put_i2s: | 481 | err_clk_put_i2s: |
483 | clk_disable(i2s->clk_aic); | ||
484 | clk_put(i2s->clk_i2s); | 482 | clk_put(i2s->clk_i2s); |
485 | err_clk_put_aic: | 483 | err_clk_put_aic: |
486 | clk_put(i2s->clk_aic); | 484 | clk_put(i2s->clk_aic); |
@@ -498,9 +496,8 @@ static int __devexit jz4740_i2s_dev_remove(struct platform_device *pdev) | |||
498 | { | 496 | { |
499 | struct jz4740_i2s *i2s = platform_get_drvdata(pdev); | 497 | struct jz4740_i2s *i2s = platform_get_drvdata(pdev); |
500 | 498 | ||
501 | snd_soc_unregister_dai(&jz4740_i2s_dai); | 499 | snd_soc_unregister_dai(&pdev->dev); |
502 | 500 | ||
503 | clk_disable(i2s->clk_aic); | ||
504 | clk_put(i2s->clk_i2s); | 501 | clk_put(i2s->clk_i2s); |
505 | clk_put(i2s->clk_aic); | 502 | clk_put(i2s->clk_aic); |
506 | 503 | ||
diff --git a/sound/soc/jz4740/jz4740-i2s.h b/sound/soc/jz4740/jz4740-i2s.h index da22ed88a589..5e49339d8b93 100644 --- a/sound/soc/jz4740/jz4740-i2s.h +++ b/sound/soc/jz4740/jz4740-i2s.h | |||
@@ -13,6 +13,4 @@ | |||
13 | 13 | ||
14 | #define JZ4740_I2S_BIT_CLK 0 | 14 | #define JZ4740_I2S_BIT_CLK 0 |
15 | 15 | ||
16 | extern struct snd_soc_dai jz4740_i2s_dai; | ||
17 | |||
18 | #endif | 16 | #endif |
diff --git a/sound/soc/jz4740/jz4740-pcm.c b/sound/soc/jz4740/jz4740-pcm.c index ee68d850c8dd..fb1483f7c966 100644 --- a/sound/soc/jz4740/jz4740-pcm.c +++ b/sound/soc/jz4740/jz4740-pcm.c | |||
@@ -109,7 +109,7 @@ static int jz4740_pcm_hw_params(struct snd_pcm_substream *substream, | |||
109 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 109 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
110 | struct jz4740_pcm_config *config; | 110 | struct jz4740_pcm_config *config; |
111 | 111 | ||
112 | config = snd_soc_dai_get_dma_data(rtd->dai->cpu_dai, substream); | 112 | config = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); |
113 | 113 | ||
114 | if (!config) | 114 | if (!config) |
115 | return 0; | 115 | return 0; |
@@ -310,14 +310,14 @@ int jz4740_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
310 | if (!card->dev->coherent_dma_mask) | 310 | if (!card->dev->coherent_dma_mask) |
311 | card->dev->coherent_dma_mask = DMA_BIT_MASK(32); | 311 | card->dev->coherent_dma_mask = DMA_BIT_MASK(32); |
312 | 312 | ||
313 | if (dai->playback.channels_min) { | 313 | if (dai->driver->playback.channels_min) { |
314 | ret = jz4740_pcm_preallocate_dma_buffer(pcm, | 314 | ret = jz4740_pcm_preallocate_dma_buffer(pcm, |
315 | SNDRV_PCM_STREAM_PLAYBACK); | 315 | SNDRV_PCM_STREAM_PLAYBACK); |
316 | if (ret) | 316 | if (ret) |
317 | goto err; | 317 | goto err; |
318 | } | 318 | } |
319 | 319 | ||
320 | if (dai->capture.channels_min) { | 320 | if (dai->driver->capture.channels_min) { |
321 | ret = jz4740_pcm_preallocate_dma_buffer(pcm, | 321 | ret = jz4740_pcm_preallocate_dma_buffer(pcm, |
322 | SNDRV_PCM_STREAM_CAPTURE); | 322 | SNDRV_PCM_STREAM_CAPTURE); |
323 | if (ret) | 323 | if (ret) |
@@ -328,22 +328,20 @@ err: | |||
328 | return ret; | 328 | return ret; |
329 | } | 329 | } |
330 | 330 | ||
331 | struct snd_soc_platform jz4740_soc_platform = { | 331 | static struct snd_soc_platform_driver jz4740_soc_platform = { |
332 | .name = "jz4740-pcm", | 332 | .ops = &jz4740_pcm_ops, |
333 | .pcm_ops = &jz4740_pcm_ops, | ||
334 | .pcm_new = jz4740_pcm_new, | 333 | .pcm_new = jz4740_pcm_new, |
335 | .pcm_free = jz4740_pcm_free, | 334 | .pcm_free = jz4740_pcm_free, |
336 | }; | 335 | }; |
337 | EXPORT_SYMBOL_GPL(jz4740_soc_platform); | ||
338 | 336 | ||
339 | static int __devinit jz4740_pcm_probe(struct platform_device *pdev) | 337 | static int __devinit jz4740_pcm_probe(struct platform_device *pdev) |
340 | { | 338 | { |
341 | return snd_soc_register_platform(&jz4740_soc_platform); | 339 | return snd_soc_register_platform(&pdev->dev, &jz4740_soc_platform); |
342 | } | 340 | } |
343 | 341 | ||
344 | static int __devexit jz4740_pcm_remove(struct platform_device *pdev) | 342 | static int __devexit jz4740_pcm_remove(struct platform_device *pdev) |
345 | { | 343 | { |
346 | snd_soc_unregister_platform(&jz4740_soc_platform); | 344 | snd_soc_unregister_platform(&pdev->dev); |
347 | return 0; | 345 | return 0; |
348 | } | 346 | } |
349 | 347 | ||
@@ -351,7 +349,7 @@ static struct platform_driver jz4740_pcm_driver = { | |||
351 | .probe = jz4740_pcm_probe, | 349 | .probe = jz4740_pcm_probe, |
352 | .remove = __devexit_p(jz4740_pcm_remove), | 350 | .remove = __devexit_p(jz4740_pcm_remove), |
353 | .driver = { | 351 | .driver = { |
354 | .name = "jz4740-pcm", | 352 | .name = "jz4740-pcm-audio", |
355 | .owner = THIS_MODULE, | 353 | .owner = THIS_MODULE, |
356 | }, | 354 | }, |
357 | }; | 355 | }; |
diff --git a/sound/soc/jz4740/jz4740-pcm.h b/sound/soc/jz4740/jz4740-pcm.h index e3f221e2779c..1220cbb4382c 100644 --- a/sound/soc/jz4740/jz4740-pcm.h +++ b/sound/soc/jz4740/jz4740-pcm.h | |||
@@ -11,8 +11,6 @@ | |||
11 | #include <linux/dma-mapping.h> | 11 | #include <linux/dma-mapping.h> |
12 | #include <asm/mach-jz4740/dma.h> | 12 | #include <asm/mach-jz4740/dma.h> |
13 | 13 | ||
14 | /* platform data */ | ||
15 | extern struct snd_soc_platform jz4740_soc_platform; | ||
16 | 14 | ||
17 | struct jz4740_pcm_config { | 15 | struct jz4740_pcm_config { |
18 | struct jz4740_dma_config dma_config; | 16 | struct jz4740_dma_config dma_config; |
diff --git a/sound/soc/jz4740/qi_lb60.c b/sound/soc/jz4740/qi_lb60.c index f15f4918f15f..c5fc339f68f1 100644 --- a/sound/soc/jz4740/qi_lb60.c +++ b/sound/soc/jz4740/qi_lb60.c | |||
@@ -19,25 +19,15 @@ | |||
19 | #include <sound/core.h> | 19 | #include <sound/core.h> |
20 | #include <sound/pcm.h> | 20 | #include <sound/pcm.h> |
21 | #include <sound/soc.h> | 21 | #include <sound/soc.h> |
22 | #include <sound/soc-dapm.h> | ||
23 | #include <linux/gpio.h> | 22 | #include <linux/gpio.h> |
24 | 23 | ||
25 | #include "../codecs/jz4740.h" | ||
26 | #include "jz4740-pcm.h" | ||
27 | #include "jz4740-i2s.h" | ||
28 | |||
29 | |||
30 | #define QI_LB60_SND_GPIO JZ_GPIO_PORTB(29) | 24 | #define QI_LB60_SND_GPIO JZ_GPIO_PORTB(29) |
31 | #define QI_LB60_AMP_GPIO JZ_GPIO_PORTD(4) | 25 | #define QI_LB60_AMP_GPIO JZ_GPIO_PORTD(4) |
32 | 26 | ||
33 | static int qi_lb60_spk_event(struct snd_soc_dapm_widget *widget, | 27 | static int qi_lb60_spk_event(struct snd_soc_dapm_widget *widget, |
34 | struct snd_kcontrol *ctrl, int event) | 28 | struct snd_kcontrol *ctrl, int event) |
35 | { | 29 | { |
36 | int on = 0; | 30 | int on = !SND_SOC_DAPM_EVENT_OFF(event); |
37 | if (event & SND_SOC_DAPM_POST_PMU) | ||
38 | on = 1; | ||
39 | else if (event & SND_SOC_DAPM_PRE_PMD) | ||
40 | on = 0; | ||
41 | 31 | ||
42 | gpio_set_value(QI_LB60_SND_GPIO, on); | 32 | gpio_set_value(QI_LB60_SND_GPIO, on); |
43 | gpio_set_value(QI_LB60_AMP_GPIO, on); | 33 | gpio_set_value(QI_LB60_AMP_GPIO, on); |
@@ -60,13 +50,15 @@ static const struct snd_soc_dapm_route qi_lb60_routes[] = { | |||
60 | SND_SOC_DAIFMT_NB_NF | \ | 50 | SND_SOC_DAIFMT_NB_NF | \ |
61 | SND_SOC_DAIFMT_CBM_CFM) | 51 | SND_SOC_DAIFMT_CBM_CFM) |
62 | 52 | ||
63 | static int qi_lb60_codec_init(struct snd_soc_codec *codec) | 53 | static int qi_lb60_codec_init(struct snd_soc_pcm_runtime *rtd) |
64 | { | 54 | { |
55 | struct snd_soc_codec *codec = rtd->codec; | ||
56 | struct snd_soc_dai *cpu_dai = rtd->cpu_dai; | ||
57 | struct snd_soc_dapm_context *dapm = &codec->dapm; | ||
65 | int ret; | 58 | int ret; |
66 | struct snd_soc_dai *cpu_dai = codec->socdev->card->dai_link->cpu_dai; | ||
67 | 59 | ||
68 | snd_soc_dapm_nc_pin(codec, "LIN"); | 60 | snd_soc_dapm_nc_pin(dapm, "LIN"); |
69 | snd_soc_dapm_nc_pin(codec, "RIN"); | 61 | snd_soc_dapm_nc_pin(dapm, "RIN"); |
70 | 62 | ||
71 | ret = snd_soc_dai_set_fmt(cpu_dai, QI_LB60_DAIFMT); | 63 | ret = snd_soc_dai_set_fmt(cpu_dai, QI_LB60_DAIFMT); |
72 | if (ret < 0) { | 64 | if (ret < 0) { |
@@ -74,18 +66,16 @@ static int qi_lb60_codec_init(struct snd_soc_codec *codec) | |||
74 | return ret; | 66 | return ret; |
75 | } | 67 | } |
76 | 68 | ||
77 | snd_soc_dapm_new_controls(codec, qi_lb60_widgets, ARRAY_SIZE(qi_lb60_widgets)); | ||
78 | snd_soc_dapm_add_routes(codec, qi_lb60_routes, ARRAY_SIZE(qi_lb60_routes)); | ||
79 | snd_soc_dapm_sync(codec); | ||
80 | |||
81 | return 0; | 69 | return 0; |
82 | } | 70 | } |
83 | 71 | ||
84 | static struct snd_soc_dai_link qi_lb60_dai = { | 72 | static struct snd_soc_dai_link qi_lb60_dai = { |
85 | .name = "jz4740", | 73 | .name = "jz4740", |
86 | .stream_name = "jz4740", | 74 | .stream_name = "jz4740", |
87 | .cpu_dai = &jz4740_i2s_dai, | 75 | .cpu_dai_name = "jz4740-i2s", |
88 | .codec_dai = &jz4740_codec_dai, | 76 | .platform_name = "jz4740-pcm-audio", |
77 | .codec_dai_name = "jz4740-hifi", | ||
78 | .codec_name = "jz4740-codec", | ||
89 | .init = qi_lb60_codec_init, | 79 | .init = qi_lb60_codec_init, |
90 | }; | 80 | }; |
91 | 81 | ||
@@ -93,16 +83,20 @@ static struct snd_soc_card qi_lb60 = { | |||
93 | .name = "QI LB60", | 83 | .name = "QI LB60", |
94 | .dai_link = &qi_lb60_dai, | 84 | .dai_link = &qi_lb60_dai, |
95 | .num_links = 1, | 85 | .num_links = 1, |
96 | .platform = &jz4740_soc_platform, | ||
97 | }; | ||
98 | 86 | ||
99 | static struct snd_soc_device qi_lb60_snd_devdata = { | 87 | .dapm_widgets = qi_lb60_widgets, |
100 | .card = &qi_lb60, | 88 | .num_dapm_widgets = ARRAY_SIZE(qi_lb60_widgets), |
101 | .codec_dev = &soc_codec_dev_jz4740_codec, | 89 | .dapm_routes = qi_lb60_routes, |
90 | .num_dapm_routes = ARRAY_SIZE(qi_lb60_routes), | ||
102 | }; | 91 | }; |
103 | 92 | ||
104 | static struct platform_device *qi_lb60_snd_device; | 93 | static struct platform_device *qi_lb60_snd_device; |
105 | 94 | ||
95 | static const struct gpio qi_lb60_gpios[] = { | ||
96 | { QI_LB60_SND_GPIO, GPIOF_OUT_INIT_LOW, "SND" }, | ||
97 | { QI_LB60_AMP_GPIO, GPIOF_OUT_INIT_LOW, "AMP" }, | ||
98 | }; | ||
99 | |||
106 | static int __init qi_lb60_init(void) | 100 | static int __init qi_lb60_init(void) |
107 | { | 101 | { |
108 | int ret; | 102 | int ret; |
@@ -112,25 +106,13 @@ static int __init qi_lb60_init(void) | |||
112 | if (!qi_lb60_snd_device) | 106 | if (!qi_lb60_snd_device) |
113 | return -ENOMEM; | 107 | return -ENOMEM; |
114 | 108 | ||
115 | ret = gpio_request(QI_LB60_SND_GPIO, "SND"); | 109 | ret = gpio_request_array(qi_lb60_gpios, ARRAY_SIZE(qi_lb60_gpios)); |
116 | if (ret) { | 110 | if (ret) { |
117 | pr_err("qi_lb60 snd: Failed to request SND GPIO(%d): %d\n", | 111 | pr_err("qi_lb60 snd: Failed to request gpios: %d\n", ret); |
118 | QI_LB60_SND_GPIO, ret); | ||
119 | goto err_device_put; | 112 | goto err_device_put; |
120 | } | 113 | } |
121 | 114 | ||
122 | ret = gpio_request(QI_LB60_AMP_GPIO, "AMP"); | 115 | platform_set_drvdata(qi_lb60_snd_device, &qi_lb60); |
123 | if (ret) { | ||
124 | pr_err("qi_lb60 snd: Failed to request AMP GPIO(%d): %d\n", | ||
125 | QI_LB60_AMP_GPIO, ret); | ||
126 | goto err_gpio_free_snd; | ||
127 | } | ||
128 | |||
129 | gpio_direction_output(QI_LB60_SND_GPIO, 0); | ||
130 | gpio_direction_output(QI_LB60_AMP_GPIO, 0); | ||
131 | |||
132 | platform_set_drvdata(qi_lb60_snd_device, &qi_lb60_snd_devdata); | ||
133 | qi_lb60_snd_devdata.dev = &qi_lb60_snd_device->dev; | ||
134 | 116 | ||
135 | ret = platform_device_add(qi_lb60_snd_device); | 117 | ret = platform_device_add(qi_lb60_snd_device); |
136 | if (ret) { | 118 | if (ret) { |
@@ -142,10 +124,8 @@ static int __init qi_lb60_init(void) | |||
142 | 124 | ||
143 | err_unset_pdata: | 125 | err_unset_pdata: |
144 | platform_set_drvdata(qi_lb60_snd_device, NULL); | 126 | platform_set_drvdata(qi_lb60_snd_device, NULL); |
145 | /*err_gpio_free_amp:*/ | 127 | /*err_gpio_free_array:*/ |
146 | gpio_free(QI_LB60_AMP_GPIO); | 128 | gpio_free_array(qi_lb60_gpios, ARRAY_SIZE(qi_lb60_gpios)); |
147 | err_gpio_free_snd: | ||
148 | gpio_free(QI_LB60_SND_GPIO); | ||
149 | err_device_put: | 129 | err_device_put: |
150 | platform_device_put(qi_lb60_snd_device); | 130 | platform_device_put(qi_lb60_snd_device); |
151 | 131 | ||
@@ -155,9 +135,8 @@ module_init(qi_lb60_init); | |||
155 | 135 | ||
156 | static void __exit qi_lb60_exit(void) | 136 | static void __exit qi_lb60_exit(void) |
157 | { | 137 | { |
158 | gpio_free(QI_LB60_AMP_GPIO); | ||
159 | gpio_free(QI_LB60_SND_GPIO); | ||
160 | platform_device_unregister(qi_lb60_snd_device); | 138 | platform_device_unregister(qi_lb60_snd_device); |
139 | gpio_free_array(qi_lb60_gpios, ARRAY_SIZE(qi_lb60_gpios)); | ||
161 | } | 140 | } |
162 | module_exit(qi_lb60_exit); | 141 | module_exit(qi_lb60_exit); |
163 | 142 | ||