aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/jz4740
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/jz4740')
-rw-r--r--sound/soc/jz4740/jz4740-i2s.c107
-rw-r--r--sound/soc/jz4740/jz4740-i2s.h2
-rw-r--r--sound/soc/jz4740/jz4740-pcm.c18
-rw-r--r--sound/soc/jz4740/jz4740-pcm.h2
-rw-r--r--sound/soc/jz4740/qi_lb60.c73
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
109static inline struct jz4740_i2s *jz4740_dai_to_i2s(struct snd_soc_dai *dai)
110{
111 return dai->private_data;
112}
113
114static int jz4740_i2s_startup(struct snd_pcm_substream *substream, 108static 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,
136static void jz4740_i2s_shutdown(struct snd_pcm_substream *substream, 130static 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,
152static int jz4740_i2s_trigger(struct snd_pcm_substream *substream, int cmd, 146static 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
187static int jz4740_i2s_set_fmt(struct snd_soc_dai *dai, unsigned int fmt) 181static 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)
238static int jz4740_i2s_hw_params(struct snd_pcm_substream *substream, 232static 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,
288static int jz4740_i2s_set_sysclk(struct snd_soc_dai *dai, int clk_id, 282static 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
313static int jz4740_i2s_suspend(struct snd_soc_dai *dai) 307static 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
331static int jz4740_i2s_resume(struct snd_soc_dai *dai) 325static 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
349static int jz4740_i2s_probe(struct platform_device *pdev, struct snd_soc_dai *dai) 343static 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
366static 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
387static 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
366static struct snd_soc_dai_ops jz4740_i2s_dai_ops = { 395static 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
378struct snd_soc_dai jz4740_i2s_dai = { 407static 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};
398EXPORT_SYMBOL_GPL(jz4740_i2s_dai);
399
400static 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
423static int __devinit jz4740_i2s_dev_probe(struct platform_device *pdev) 428static 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
482err_clk_put_i2s: 481err_clk_put_i2s:
483 clk_disable(i2s->clk_aic);
484 clk_put(i2s->clk_i2s); 482 clk_put(i2s->clk_i2s);
485err_clk_put_aic: 483err_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
16extern 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
331struct snd_soc_platform jz4740_soc_platform = { 331static 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};
337EXPORT_SYMBOL_GPL(jz4740_soc_platform);
338 336
339static int __devinit jz4740_pcm_probe(struct platform_device *pdev) 337static 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
344static int __devexit jz4740_pcm_remove(struct platform_device *pdev) 342static 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 */
15extern struct snd_soc_platform jz4740_soc_platform;
16 14
17struct jz4740_pcm_config { 15struct 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
33static int qi_lb60_spk_event(struct snd_soc_dapm_widget *widget, 27static 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
63static int qi_lb60_codec_init(struct snd_soc_codec *codec) 53static 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
84static struct snd_soc_dai_link qi_lb60_dai = { 72static 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
99static 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
104static struct platform_device *qi_lb60_snd_device; 93static struct platform_device *qi_lb60_snd_device;
105 94
95static 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
106static int __init qi_lb60_init(void) 100static 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
143err_unset_pdata: 125err_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));
147err_gpio_free_snd:
148 gpio_free(QI_LB60_SND_GPIO);
149err_device_put: 129err_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
156static void __exit qi_lb60_exit(void) 136static 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}
162module_exit(qi_lb60_exit); 141module_exit(qi_lb60_exit);
163 142