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/imx | |
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/imx')
-rw-r--r-- | sound/soc/imx/Kconfig | 22 | ||||
-rw-r--r-- | sound/soc/imx/Makefile | 12 | ||||
-rw-r--r-- | sound/soc/imx/eukrea-tlv320.c | 24 | ||||
-rw-r--r-- | sound/soc/imx/imx-pcm-dma-mx2.c | 267 | ||||
-rw-r--r-- | sound/soc/imx/imx-pcm-fiq.c | 68 | ||||
-rw-r--r-- | sound/soc/imx/imx-ssi.c | 182 | ||||
-rw-r--r-- | sound/soc/imx/imx-ssi.h | 14 | ||||
-rw-r--r-- | sound/soc/imx/mx27vis-aic32x4.c | 137 | ||||
-rw-r--r-- | sound/soc/imx/phycore-ac97.c | 45 | ||||
-rw-r--r-- | sound/soc/imx/wm1133-ev1.c | 35 |
10 files changed, 506 insertions, 300 deletions
diff --git a/sound/soc/imx/Kconfig b/sound/soc/imx/Kconfig index 687c76fc0839..bb699bb55a50 100644 --- a/sound/soc/imx/Kconfig +++ b/sound/soc/imx/Kconfig | |||
@@ -8,20 +8,38 @@ menuconfig SND_IMX_SOC | |||
8 | Say Y or M if you want to add support for codecs attached to | 8 | Say Y or M if you want to add support for codecs attached to |
9 | the i.MX SSI interface. | 9 | the i.MX SSI interface. |
10 | 10 | ||
11 | |||
11 | if SND_IMX_SOC | 12 | if SND_IMX_SOC |
12 | 13 | ||
14 | config SND_MXC_SOC_FIQ | ||
15 | tristate | ||
16 | |||
17 | config SND_MXC_SOC_MX2 | ||
18 | tristate | ||
19 | |||
13 | config SND_MXC_SOC_WM1133_EV1 | 20 | config SND_MXC_SOC_WM1133_EV1 |
14 | tristate "Audio on the the i.MX31ADS with WM1133-EV1 fitted" | 21 | tristate "Audio on the the i.MX31ADS with WM1133-EV1 fitted" |
15 | depends on MACH_MX31ADS_WM1133_EV1 && EXPERIMENTAL | 22 | depends on MACH_MX31ADS_WM1133_EV1 && EXPERIMENTAL |
16 | select SND_SOC_WM8350 | 23 | select SND_SOC_WM8350 |
24 | select SND_MXC_SOC_FIQ | ||
17 | help | 25 | help |
18 | Enable support for audio on the i.MX31ADS with the WM1133-EV1 | 26 | Enable support for audio on the i.MX31ADS with the WM1133-EV1 |
19 | PMIC board with WM8835x fitted. | 27 | PMIC board with WM8835x fitted. |
20 | 28 | ||
29 | config SND_SOC_MX27VIS_AIC32X4 | ||
30 | tristate "SoC audio support for Visstrim M10 boards" | ||
31 | depends on MACH_IMX27_VISSTRIM_M10 | ||
32 | select SND_SOC_TVL320AIC32X4 | ||
33 | select SND_MXC_SOC_MX2 | ||
34 | help | ||
35 | Say Y if you want to add support for SoC audio on Visstrim SM10 | ||
36 | board with TLV320AIC32X4 codec. | ||
37 | |||
21 | config SND_SOC_PHYCORE_AC97 | 38 | config SND_SOC_PHYCORE_AC97 |
22 | tristate "SoC Audio support for Phytec phyCORE (and phyCARD) boards" | 39 | tristate "SoC Audio support for Phytec phyCORE (and phyCARD) boards" |
23 | depends on MACH_PCM043 || MACH_PCA100 | 40 | depends on MACH_PCM043 || MACH_PCA100 |
24 | select SND_SOC_WM9712 | 41 | select SND_SOC_WM9712 |
42 | select SND_MXC_SOC_FIQ | ||
25 | help | 43 | help |
26 | Say Y if you want to add support for SoC audio on Phytec phyCORE | 44 | Say Y if you want to add support for SoC audio on Phytec phyCORE |
27 | and phyCARD boards in AC97 mode | 45 | and phyCARD boards in AC97 mode |
@@ -30,8 +48,10 @@ config SND_SOC_EUKREA_TLV320 | |||
30 | tristate "Eukrea TLV320" | 48 | tristate "Eukrea TLV320" |
31 | depends on MACH_EUKREA_MBIMX27_BASEBOARD \ | 49 | depends on MACH_EUKREA_MBIMX27_BASEBOARD \ |
32 | || MACH_EUKREA_MBIMXSD25_BASEBOARD \ | 50 | || MACH_EUKREA_MBIMXSD25_BASEBOARD \ |
33 | || MACH_EUKREA_MBIMXSD35_BASEBOARD | 51 | || MACH_EUKREA_MBIMXSD35_BASEBOARD \ |
52 | || MACH_EUKREA_MBIMXSD51_BASEBOARD | ||
34 | select SND_SOC_TLV320AIC23 | 53 | select SND_SOC_TLV320AIC23 |
54 | select SND_MXC_SOC_FIQ | ||
35 | help | 55 | help |
36 | Enable I2S based access to the TLV320AIC23B codec attached | 56 | Enable I2S based access to the TLV320AIC23B codec attached |
37 | to the SSI interface | 57 | to the SSI interface |
diff --git a/sound/soc/imx/Makefile b/sound/soc/imx/Makefile index 7bc57baf2b0e..d6d609ba7e24 100644 --- a/sound/soc/imx/Makefile +++ b/sound/soc/imx/Makefile | |||
@@ -1,17 +1,19 @@ | |||
1 | # i.MX Platform Support | 1 | # i.MX Platform Support |
2 | snd-soc-imx-objs := imx-ssi.o imx-pcm-fiq.o | 2 | snd-soc-imx-objs := imx-ssi.o |
3 | 3 | snd-soc-imx-fiq-objs := imx-pcm-fiq.o | |
4 | ifdef CONFIG_MACH_MX27 | 4 | snd-soc-imx-mx2-objs := imx-pcm-dma-mx2.o |
5 | snd-soc-imx-objs += imx-pcm-dma-mx2.o | ||
6 | endif | ||
7 | 5 | ||
8 | obj-$(CONFIG_SND_IMX_SOC) += snd-soc-imx.o | 6 | obj-$(CONFIG_SND_IMX_SOC) += snd-soc-imx.o |
7 | obj-$(CONFIG_SND_MXC_SOC_FIQ) += snd-soc-imx-fiq.o | ||
8 | obj-$(CONFIG_SND_MXC_SOC_MX2) += snd-soc-imx-mx2.o | ||
9 | 9 | ||
10 | # i.MX Machine Support | 10 | # i.MX Machine Support |
11 | snd-soc-eukrea-tlv320-objs := eukrea-tlv320.o | 11 | snd-soc-eukrea-tlv320-objs := eukrea-tlv320.o |
12 | snd-soc-phycore-ac97-objs := phycore-ac97.o | 12 | snd-soc-phycore-ac97-objs := phycore-ac97.o |
13 | snd-soc-mx27vis-aic32x4-objs := mx27vis-aic32x4.o | ||
13 | snd-soc-wm1133-ev1-objs := wm1133-ev1.o | 14 | snd-soc-wm1133-ev1-objs := wm1133-ev1.o |
14 | 15 | ||
15 | obj-$(CONFIG_SND_SOC_EUKREA_TLV320) += snd-soc-eukrea-tlv320.o | 16 | obj-$(CONFIG_SND_SOC_EUKREA_TLV320) += snd-soc-eukrea-tlv320.o |
16 | obj-$(CONFIG_SND_SOC_PHYCORE_AC97) += snd-soc-phycore-ac97.o | 17 | obj-$(CONFIG_SND_SOC_PHYCORE_AC97) += snd-soc-phycore-ac97.o |
18 | obj-$(CONFIG_SND_SOC_MX27VIS_AIC32X4) += snd-soc-mx27vis-aic32x4.o | ||
17 | obj-$(CONFIG_SND_MXC_SOC_WM1133_EV1) += snd-soc-wm1133-ev1.o | 19 | obj-$(CONFIG_SND_MXC_SOC_WM1133_EV1) += snd-soc-wm1133-ev1.o |
diff --git a/sound/soc/imx/eukrea-tlv320.c b/sound/soc/imx/eukrea-tlv320.c index f15dfbdc47ee..75fb4b83548b 100644 --- a/sound/soc/imx/eukrea-tlv320.c +++ b/sound/soc/imx/eukrea-tlv320.c | |||
@@ -22,7 +22,6 @@ | |||
22 | #include <sound/core.h> | 22 | #include <sound/core.h> |
23 | #include <sound/pcm.h> | 23 | #include <sound/pcm.h> |
24 | #include <sound/soc.h> | 24 | #include <sound/soc.h> |
25 | #include <sound/soc-dapm.h> | ||
26 | #include <asm/mach-types.h> | 25 | #include <asm/mach-types.h> |
27 | 26 | ||
28 | #include "../codecs/tlv320aic23.h" | 27 | #include "../codecs/tlv320aic23.h" |
@@ -34,8 +33,8 @@ static int eukrea_tlv320_hw_params(struct snd_pcm_substream *substream, | |||
34 | struct snd_pcm_hw_params *params) | 33 | struct snd_pcm_hw_params *params) |
35 | { | 34 | { |
36 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 35 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
37 | struct snd_soc_dai *codec_dai = rtd->dai->codec_dai; | 36 | struct snd_soc_dai *codec_dai = rtd->codec_dai; |
38 | struct snd_soc_dai *cpu_dai = rtd->dai->cpu_dai; | 37 | struct snd_soc_dai *cpu_dai = rtd->cpu_dai; |
39 | int ret; | 38 | int ret; |
40 | 39 | ||
41 | ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_I2S | | 40 | ret = snd_soc_dai_set_fmt(cpu_dai, SND_SOC_DAIFMT_I2S | |
@@ -79,22 +78,19 @@ static struct snd_soc_ops eukrea_tlv320_snd_ops = { | |||
79 | static struct snd_soc_dai_link eukrea_tlv320_dai = { | 78 | static struct snd_soc_dai_link eukrea_tlv320_dai = { |
80 | .name = "tlv320aic23", | 79 | .name = "tlv320aic23", |
81 | .stream_name = "TLV320AIC23", | 80 | .stream_name = "TLV320AIC23", |
82 | .codec_dai = &tlv320aic23_dai, | 81 | .codec_dai_name = "tlv320aic23-hifi", |
82 | .platform_name = "imx-fiq-pcm-audio.0", | ||
83 | .codec_name = "tlv320aic23-codec.0-001a", | ||
84 | .cpu_dai_name = "imx-ssi.0", | ||
83 | .ops = &eukrea_tlv320_snd_ops, | 85 | .ops = &eukrea_tlv320_snd_ops, |
84 | }; | 86 | }; |
85 | 87 | ||
86 | static struct snd_soc_card eukrea_tlv320 = { | 88 | static struct snd_soc_card eukrea_tlv320 = { |
87 | .name = "cpuimx-audio", | 89 | .name = "cpuimx-audio", |
88 | .platform = &imx_soc_platform, | ||
89 | .dai_link = &eukrea_tlv320_dai, | 90 | .dai_link = &eukrea_tlv320_dai, |
90 | .num_links = 1, | 91 | .num_links = 1, |
91 | }; | 92 | }; |
92 | 93 | ||
93 | static struct snd_soc_device eukrea_tlv320_snd_devdata = { | ||
94 | .card = &eukrea_tlv320, | ||
95 | .codec_dev = &soc_codec_dev_tlv320aic23, | ||
96 | }; | ||
97 | |||
98 | static struct platform_device *eukrea_tlv320_snd_device; | 94 | static struct platform_device *eukrea_tlv320_snd_device; |
99 | 95 | ||
100 | static int __init eukrea_tlv320_init(void) | 96 | static int __init eukrea_tlv320_init(void) |
@@ -102,7 +98,8 @@ static int __init eukrea_tlv320_init(void) | |||
102 | int ret; | 98 | int ret; |
103 | 99 | ||
104 | if (!machine_is_eukrea_cpuimx27() && !machine_is_eukrea_cpuimx25sd() | 100 | if (!machine_is_eukrea_cpuimx27() && !machine_is_eukrea_cpuimx25sd() |
105 | && !machine_is_eukrea_cpuimx35sd()) | 101 | && !machine_is_eukrea_cpuimx35sd() |
102 | && !machine_is_eukrea_cpuimx51sd()) | ||
106 | /* return happy. We might run on a totally different machine */ | 103 | /* return happy. We might run on a totally different machine */ |
107 | return 0; | 104 | return 0; |
108 | 105 | ||
@@ -110,10 +107,7 @@ static int __init eukrea_tlv320_init(void) | |||
110 | if (!eukrea_tlv320_snd_device) | 107 | if (!eukrea_tlv320_snd_device) |
111 | return -ENOMEM; | 108 | return -ENOMEM; |
112 | 109 | ||
113 | eukrea_tlv320_dai.cpu_dai = &imx_ssi_pcm_dai[0]; | 110 | platform_set_drvdata(eukrea_tlv320_snd_device, &eukrea_tlv320); |
114 | |||
115 | platform_set_drvdata(eukrea_tlv320_snd_device, &eukrea_tlv320_snd_devdata); | ||
116 | eukrea_tlv320_snd_devdata.dev = &eukrea_tlv320_snd_device->dev; | ||
117 | ret = platform_device_add(eukrea_tlv320_snd_device); | 111 | ret = platform_device_add(eukrea_tlv320_snd_device); |
118 | 112 | ||
119 | if (ret) { | 113 | if (ret) { |
diff --git a/sound/soc/imx/imx-pcm-dma-mx2.c b/sound/soc/imx/imx-pcm-dma-mx2.c index 0a595da4811d..4173b3d87f97 100644 --- a/sound/soc/imx/imx-pcm-dma-mx2.c +++ b/sound/soc/imx/imx-pcm-dma-mx2.c | |||
@@ -20,6 +20,7 @@ | |||
20 | #include <linux/module.h> | 20 | #include <linux/module.h> |
21 | #include <linux/platform_device.h> | 21 | #include <linux/platform_device.h> |
22 | #include <linux/slab.h> | 22 | #include <linux/slab.h> |
23 | #include <linux/dmaengine.h> | ||
23 | 24 | ||
24 | #include <sound/core.h> | 25 | #include <sound/core.h> |
25 | #include <sound/initval.h> | 26 | #include <sound/initval.h> |
@@ -27,165 +28,146 @@ | |||
27 | #include <sound/pcm_params.h> | 28 | #include <sound/pcm_params.h> |
28 | #include <sound/soc.h> | 29 | #include <sound/soc.h> |
29 | 30 | ||
30 | #include <mach/dma-mx1-mx2.h> | 31 | #include <mach/dma.h> |
31 | 32 | ||
32 | #include "imx-ssi.h" | 33 | #include "imx-ssi.h" |
33 | 34 | ||
34 | struct imx_pcm_runtime_data { | 35 | struct imx_pcm_runtime_data { |
35 | int sg_count; | 36 | int period_bytes; |
36 | struct scatterlist *sg_list; | ||
37 | int period; | ||
38 | int periods; | 37 | int periods; |
39 | unsigned long dma_addr; | ||
40 | int dma; | 38 | int dma; |
41 | struct snd_pcm_substream *substream; | ||
42 | unsigned long offset; | 39 | unsigned long offset; |
43 | unsigned long size; | 40 | unsigned long size; |
44 | unsigned long period_cnt; | ||
45 | void *buf; | 41 | void *buf; |
46 | int period_time; | 42 | int period_time; |
43 | struct dma_async_tx_descriptor *desc; | ||
44 | struct dma_chan *dma_chan; | ||
45 | struct imx_dma_data dma_data; | ||
47 | }; | 46 | }; |
48 | 47 | ||
49 | /* Called by the DMA framework when a period has elapsed */ | 48 | static void audio_dma_irq(void *data) |
50 | static void imx_ssi_dma_progression(int channel, void *data, | ||
51 | struct scatterlist *sg) | ||
52 | { | 49 | { |
53 | struct snd_pcm_substream *substream = data; | 50 | struct snd_pcm_substream *substream = (struct snd_pcm_substream *)data; |
54 | struct snd_pcm_runtime *runtime = substream->runtime; | 51 | struct snd_pcm_runtime *runtime = substream->runtime; |
55 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | 52 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; |
56 | 53 | ||
57 | if (!sg) | 54 | iprtd->offset += iprtd->period_bytes; |
58 | return; | 55 | iprtd->offset %= iprtd->period_bytes * iprtd->periods; |
59 | |||
60 | runtime = iprtd->substream->runtime; | ||
61 | 56 | ||
62 | iprtd->offset = sg->dma_address - runtime->dma_addr; | 57 | snd_pcm_period_elapsed(substream); |
63 | |||
64 | snd_pcm_period_elapsed(iprtd->substream); | ||
65 | } | 58 | } |
66 | 59 | ||
67 | static void imx_ssi_dma_callback(int channel, void *data) | 60 | static bool filter(struct dma_chan *chan, void *param) |
68 | { | 61 | { |
69 | pr_err("%s shouldn't be called\n", __func__); | 62 | struct imx_pcm_runtime_data *iprtd = param; |
70 | } | ||
71 | 63 | ||
72 | static void snd_imx_dma_err_callback(int channel, void *data, int err) | 64 | if (!imx_dma_is_general_purpose(chan)) |
73 | { | 65 | return false; |
74 | struct snd_pcm_substream *substream = data; | ||
75 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | ||
76 | struct imx_pcm_dma_params *dma_params = | ||
77 | snd_soc_dai_get_dma_data(rtd->dai->cpu_dai, substream); | ||
78 | struct snd_pcm_runtime *runtime = substream->runtime; | ||
79 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | ||
80 | int ret; | ||
81 | 66 | ||
82 | pr_err("DMA timeout on channel %d -%s%s%s%s\n", | 67 | chan->private = &iprtd->dma_data; |
83 | channel, | ||
84 | err & IMX_DMA_ERR_BURST ? " burst" : "", | ||
85 | err & IMX_DMA_ERR_REQUEST ? " request" : "", | ||
86 | err & IMX_DMA_ERR_TRANSFER ? " transfer" : "", | ||
87 | err & IMX_DMA_ERR_BUFFER ? " buffer" : ""); | ||
88 | 68 | ||
89 | imx_dma_disable(iprtd->dma); | 69 | return true; |
90 | ret = imx_dma_setup_sg(iprtd->dma, iprtd->sg_list, iprtd->sg_count, | ||
91 | IMX_DMA_LENGTH_LOOP, dma_params->dma_addr, | ||
92 | substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? | ||
93 | DMA_MODE_WRITE : DMA_MODE_READ); | ||
94 | if (!ret) | ||
95 | imx_dma_enable(iprtd->dma); | ||
96 | } | 70 | } |
97 | 71 | ||
98 | static int imx_ssi_dma_alloc(struct snd_pcm_substream *substream) | 72 | static int imx_ssi_dma_alloc(struct snd_pcm_substream *substream, |
73 | struct snd_pcm_hw_params *params) | ||
99 | { | 74 | { |
100 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 75 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
101 | struct imx_pcm_dma_params *dma_params; | 76 | struct imx_pcm_dma_params *dma_params; |
102 | struct snd_pcm_runtime *runtime = substream->runtime; | 77 | struct snd_pcm_runtime *runtime = substream->runtime; |
103 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | 78 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; |
79 | struct dma_slave_config slave_config; | ||
80 | dma_cap_mask_t mask; | ||
81 | enum dma_slave_buswidth buswidth; | ||
104 | int ret; | 82 | int ret; |
105 | 83 | ||
106 | dma_params = snd_soc_dai_get_dma_data(rtd->dai->cpu_dai, substream); | 84 | dma_params = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); |
107 | 85 | ||
108 | iprtd->dma = imx_dma_request_by_prio(DRV_NAME, DMA_PRIO_HIGH); | 86 | iprtd->dma_data.peripheral_type = IMX_DMATYPE_SSI; |
109 | if (iprtd->dma < 0) { | 87 | iprtd->dma_data.priority = DMA_PRIO_HIGH; |
110 | pr_err("Failed to claim the audio DMA\n"); | 88 | iprtd->dma_data.dma_request = dma_params->dma; |
111 | return -ENODEV; | ||
112 | } | ||
113 | 89 | ||
114 | ret = imx_dma_setup_handlers(iprtd->dma, | 90 | /* Try to grab a DMA channel */ |
115 | imx_ssi_dma_callback, | 91 | dma_cap_zero(mask); |
116 | snd_imx_dma_err_callback, substream); | 92 | dma_cap_set(DMA_SLAVE, mask); |
117 | if (ret) | 93 | iprtd->dma_chan = dma_request_channel(mask, filter, iprtd); |
118 | goto out; | 94 | if (!iprtd->dma_chan) |
95 | return -EINVAL; | ||
119 | 96 | ||
120 | ret = imx_dma_setup_progression_handler(iprtd->dma, | 97 | switch (params_format(params)) { |
121 | imx_ssi_dma_progression); | 98 | case SNDRV_PCM_FORMAT_S16_LE: |
122 | if (ret) { | 99 | buswidth = DMA_SLAVE_BUSWIDTH_2_BYTES; |
123 | pr_err("Failed to setup the DMA handler\n"); | 100 | break; |
124 | goto out; | 101 | case SNDRV_PCM_FORMAT_S20_3LE: |
102 | case SNDRV_PCM_FORMAT_S24_LE: | ||
103 | buswidth = DMA_SLAVE_BUSWIDTH_4_BYTES; | ||
104 | break; | ||
105 | default: | ||
106 | return 0; | ||
125 | } | 107 | } |
126 | 108 | ||
127 | ret = imx_dma_config_channel(iprtd->dma, | 109 | if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) { |
128 | IMX_DMA_MEMSIZE_16 | IMX_DMA_TYPE_FIFO, | 110 | slave_config.direction = DMA_TO_DEVICE; |
129 | IMX_DMA_MEMSIZE_32 | IMX_DMA_TYPE_LINEAR, | 111 | slave_config.dst_addr = dma_params->dma_addr; |
130 | dma_params->dma, 1); | 112 | slave_config.dst_addr_width = buswidth; |
131 | if (ret < 0) { | 113 | slave_config.dst_maxburst = dma_params->burstsize * buswidth; |
132 | pr_err("Cannot configure DMA channel: %d\n", ret); | 114 | } else { |
133 | goto out; | 115 | slave_config.direction = DMA_FROM_DEVICE; |
116 | slave_config.src_addr = dma_params->dma_addr; | ||
117 | slave_config.src_addr_width = buswidth; | ||
118 | slave_config.src_maxburst = dma_params->burstsize * buswidth; | ||
134 | } | 119 | } |
135 | 120 | ||
136 | imx_dma_config_burstlen(iprtd->dma, dma_params->burstsize * 2); | 121 | ret = dmaengine_slave_config(iprtd->dma_chan, &slave_config); |
122 | if (ret) | ||
123 | return ret; | ||
137 | 124 | ||
138 | return 0; | 125 | return 0; |
139 | out: | ||
140 | imx_dma_free(iprtd->dma); | ||
141 | return ret; | ||
142 | } | 126 | } |
143 | 127 | ||
144 | static int snd_imx_pcm_hw_params(struct snd_pcm_substream *substream, | 128 | static int snd_imx_pcm_hw_params(struct snd_pcm_substream *substream, |
145 | struct snd_pcm_hw_params *params) | 129 | struct snd_pcm_hw_params *params) |
146 | { | 130 | { |
131 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | ||
147 | struct snd_pcm_runtime *runtime = substream->runtime; | 132 | struct snd_pcm_runtime *runtime = substream->runtime; |
148 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | 133 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; |
149 | int i; | ||
150 | unsigned long dma_addr; | 134 | unsigned long dma_addr; |
135 | struct dma_chan *chan; | ||
136 | struct imx_pcm_dma_params *dma_params; | ||
137 | int ret; | ||
151 | 138 | ||
152 | imx_ssi_dma_alloc(substream); | 139 | dma_params = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); |
140 | ret = imx_ssi_dma_alloc(substream, params); | ||
141 | if (ret) | ||
142 | return ret; | ||
143 | chan = iprtd->dma_chan; | ||
153 | 144 | ||
154 | iprtd->size = params_buffer_bytes(params); | 145 | iprtd->size = params_buffer_bytes(params); |
155 | iprtd->periods = params_periods(params); | 146 | iprtd->periods = params_periods(params); |
156 | iprtd->period = params_period_bytes(params); | 147 | iprtd->period_bytes = params_period_bytes(params); |
157 | iprtd->offset = 0; | 148 | iprtd->offset = 0; |
158 | iprtd->period_time = HZ / (params_rate(params) / | 149 | iprtd->period_time = HZ / (params_rate(params) / |
159 | params_period_size(params)); | 150 | params_period_size(params)); |
160 | 151 | ||
161 | snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); | 152 | snd_pcm_set_runtime_buffer(substream, &substream->dma_buffer); |
162 | 153 | ||
163 | if (iprtd->sg_count != iprtd->periods) { | ||
164 | kfree(iprtd->sg_list); | ||
165 | |||
166 | iprtd->sg_list = kcalloc(iprtd->periods + 1, | ||
167 | sizeof(struct scatterlist), GFP_KERNEL); | ||
168 | if (!iprtd->sg_list) | ||
169 | return -ENOMEM; | ||
170 | iprtd->sg_count = iprtd->periods + 1; | ||
171 | } | ||
172 | |||
173 | sg_init_table(iprtd->sg_list, iprtd->sg_count); | ||
174 | dma_addr = runtime->dma_addr; | 154 | dma_addr = runtime->dma_addr; |
175 | 155 | ||
176 | for (i = 0; i < iprtd->periods; i++) { | 156 | iprtd->buf = (unsigned int *)substream->dma_buffer.area; |
177 | iprtd->sg_list[i].page_link = 0; | 157 | |
178 | iprtd->sg_list[i].offset = 0; | 158 | iprtd->desc = chan->device->device_prep_dma_cyclic(chan, dma_addr, |
179 | iprtd->sg_list[i].dma_address = dma_addr; | 159 | iprtd->period_bytes * iprtd->periods, |
180 | iprtd->sg_list[i].length = iprtd->period; | 160 | iprtd->period_bytes, |
181 | dma_addr += iprtd->period; | 161 | substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? |
162 | DMA_TO_DEVICE : DMA_FROM_DEVICE); | ||
163 | if (!iprtd->desc) { | ||
164 | dev_err(&chan->dev->device, "cannot prepare slave dma\n"); | ||
165 | return -EINVAL; | ||
182 | } | 166 | } |
183 | 167 | ||
184 | /* close the loop */ | 168 | iprtd->desc->callback = audio_dma_irq; |
185 | iprtd->sg_list[iprtd->sg_count - 1].offset = 0; | 169 | iprtd->desc->callback_param = substream; |
186 | iprtd->sg_list[iprtd->sg_count - 1].length = 0; | 170 | |
187 | iprtd->sg_list[iprtd->sg_count - 1].page_link = | ||
188 | ((unsigned long) iprtd->sg_list | 0x01) & ~0x02; | ||
189 | return 0; | 171 | return 0; |
190 | } | 172 | } |
191 | 173 | ||
@@ -194,40 +176,20 @@ static int snd_imx_pcm_hw_free(struct snd_pcm_substream *substream) | |||
194 | struct snd_pcm_runtime *runtime = substream->runtime; | 176 | struct snd_pcm_runtime *runtime = substream->runtime; |
195 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | 177 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; |
196 | 178 | ||
197 | if (iprtd->dma >= 0) { | 179 | if (iprtd->dma_chan) { |
198 | imx_dma_free(iprtd->dma); | 180 | dma_release_channel(iprtd->dma_chan); |
199 | iprtd->dma = -EINVAL; | 181 | iprtd->dma_chan = NULL; |
200 | } | 182 | } |
201 | 183 | ||
202 | kfree(iprtd->sg_list); | ||
203 | iprtd->sg_list = NULL; | ||
204 | |||
205 | return 0; | 184 | return 0; |
206 | } | 185 | } |
207 | 186 | ||
208 | static int snd_imx_pcm_prepare(struct snd_pcm_substream *substream) | 187 | static int snd_imx_pcm_prepare(struct snd_pcm_substream *substream) |
209 | { | 188 | { |
210 | struct snd_pcm_runtime *runtime = substream->runtime; | ||
211 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 189 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
212 | struct imx_pcm_dma_params *dma_params; | 190 | struct imx_pcm_dma_params *dma_params; |
213 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | ||
214 | int err; | ||
215 | |||
216 | dma_params = snd_soc_dai_get_dma_data(rtd->dai->cpu_dai, substream); | ||
217 | |||
218 | iprtd->substream = substream; | ||
219 | iprtd->buf = (unsigned int *)substream->dma_buffer.area; | ||
220 | iprtd->period_cnt = 0; | ||
221 | |||
222 | pr_debug("%s: buf: %p period: %d periods: %d\n", | ||
223 | __func__, iprtd->buf, iprtd->period, iprtd->periods); | ||
224 | 191 | ||
225 | err = imx_dma_setup_sg(iprtd->dma, iprtd->sg_list, iprtd->sg_count, | 192 | dma_params = snd_soc_dai_get_dma_data(rtd->cpu_dai, substream); |
226 | IMX_DMA_LENGTH_LOOP, dma_params->dma_addr, | ||
227 | substream->stream == SNDRV_PCM_STREAM_PLAYBACK ? | ||
228 | DMA_MODE_WRITE : DMA_MODE_READ); | ||
229 | if (err) | ||
230 | return err; | ||
231 | 193 | ||
232 | return 0; | 194 | return 0; |
233 | } | 195 | } |
@@ -241,14 +203,14 @@ static int snd_imx_pcm_trigger(struct snd_pcm_substream *substream, int cmd) | |||
241 | case SNDRV_PCM_TRIGGER_START: | 203 | case SNDRV_PCM_TRIGGER_START: |
242 | case SNDRV_PCM_TRIGGER_RESUME: | 204 | case SNDRV_PCM_TRIGGER_RESUME: |
243 | case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: | 205 | case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: |
244 | imx_dma_enable(iprtd->dma); | 206 | dmaengine_submit(iprtd->desc); |
245 | 207 | ||
246 | break; | 208 | break; |
247 | 209 | ||
248 | case SNDRV_PCM_TRIGGER_STOP: | 210 | case SNDRV_PCM_TRIGGER_STOP: |
249 | case SNDRV_PCM_TRIGGER_SUSPEND: | 211 | case SNDRV_PCM_TRIGGER_SUSPEND: |
250 | case SNDRV_PCM_TRIGGER_PAUSE_PUSH: | 212 | case SNDRV_PCM_TRIGGER_PAUSE_PUSH: |
251 | imx_dma_disable(iprtd->dma); | 213 | dmaengine_terminate_all(iprtd->dma_chan); |
252 | 214 | ||
253 | break; | 215 | break; |
254 | default: | 216 | default: |
@@ -263,6 +225,9 @@ static snd_pcm_uframes_t snd_imx_pcm_pointer(struct snd_pcm_substream *substream | |||
263 | struct snd_pcm_runtime *runtime = substream->runtime; | 225 | struct snd_pcm_runtime *runtime = substream->runtime; |
264 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | 226 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; |
265 | 227 | ||
228 | pr_debug("%s: %ld %ld\n", __func__, iprtd->offset, | ||
229 | bytes_to_frames(substream->runtime, iprtd->offset)); | ||
230 | |||
266 | return bytes_to_frames(substream->runtime, iprtd->offset); | 231 | return bytes_to_frames(substream->runtime, iprtd->offset); |
267 | } | 232 | } |
268 | 233 | ||
@@ -279,7 +244,7 @@ static struct snd_pcm_hardware snd_imx_hardware = { | |||
279 | .channels_max = 2, | 244 | .channels_max = 2, |
280 | .buffer_bytes_max = IMX_SSI_DMABUF_SIZE, | 245 | .buffer_bytes_max = IMX_SSI_DMABUF_SIZE, |
281 | .period_bytes_min = 128, | 246 | .period_bytes_min = 128, |
282 | .period_bytes_max = 16 * 1024, | 247 | .period_bytes_max = 65535, /* Limited by SDMA engine */ |
283 | .periods_min = 2, | 248 | .periods_min = 2, |
284 | .periods_max = 255, | 249 | .periods_max = 255, |
285 | .fifo_size = 0, | 250 | .fifo_size = 0, |
@@ -304,11 +269,23 @@ static int snd_imx_open(struct snd_pcm_substream *substream) | |||
304 | } | 269 | } |
305 | 270 | ||
306 | snd_soc_set_runtime_hwparams(substream, &snd_imx_hardware); | 271 | snd_soc_set_runtime_hwparams(substream, &snd_imx_hardware); |
272 | |||
273 | return 0; | ||
274 | } | ||
275 | |||
276 | static int snd_imx_close(struct snd_pcm_substream *substream) | ||
277 | { | ||
278 | struct snd_pcm_runtime *runtime = substream->runtime; | ||
279 | struct imx_pcm_runtime_data *iprtd = runtime->private_data; | ||
280 | |||
281 | kfree(iprtd); | ||
282 | |||
307 | return 0; | 283 | return 0; |
308 | } | 284 | } |
309 | 285 | ||
310 | static struct snd_pcm_ops imx_pcm_ops = { | 286 | static struct snd_pcm_ops imx_pcm_ops = { |
311 | .open = snd_imx_open, | 287 | .open = snd_imx_open, |
288 | .close = snd_imx_close, | ||
312 | .ioctl = snd_pcm_lib_ioctl, | 289 | .ioctl = snd_pcm_lib_ioctl, |
313 | .hw_params = snd_imx_pcm_hw_params, | 290 | .hw_params = snd_imx_pcm_hw_params, |
314 | .hw_free = snd_imx_pcm_hw_free, | 291 | .hw_free = snd_imx_pcm_hw_free, |
@@ -318,19 +295,47 @@ static struct snd_pcm_ops imx_pcm_ops = { | |||
318 | .mmap = snd_imx_pcm_mmap, | 295 | .mmap = snd_imx_pcm_mmap, |
319 | }; | 296 | }; |
320 | 297 | ||
321 | static struct snd_soc_platform imx_soc_platform_dma = { | 298 | static struct snd_soc_platform_driver imx_soc_platform_mx2 = { |
322 | .name = "imx-audio", | 299 | .ops = &imx_pcm_ops, |
323 | .pcm_ops = &imx_pcm_ops, | ||
324 | .pcm_new = imx_pcm_new, | 300 | .pcm_new = imx_pcm_new, |
325 | .pcm_free = imx_pcm_free, | 301 | .pcm_free = imx_pcm_free, |
326 | }; | 302 | }; |
327 | 303 | ||
328 | struct snd_soc_platform *imx_ssi_dma_mx2_init(struct platform_device *pdev, | 304 | static int __devinit imx_soc_platform_probe(struct platform_device *pdev) |
329 | struct imx_ssi *ssi) | ||
330 | { | 305 | { |
331 | ssi->dma_params_tx.burstsize = DMA_TXFIFO_BURST; | 306 | struct imx_ssi *ssi = platform_get_drvdata(pdev); |
332 | ssi->dma_params_rx.burstsize = DMA_RXFIFO_BURST; | 307 | |
308 | ssi->dma_params_tx.burstsize = 6; | ||
309 | ssi->dma_params_rx.burstsize = 4; | ||
333 | 310 | ||
334 | return &imx_soc_platform_dma; | 311 | return snd_soc_register_platform(&pdev->dev, &imx_soc_platform_mx2); |
312 | } | ||
313 | |||
314 | static int __devexit imx_soc_platform_remove(struct platform_device *pdev) | ||
315 | { | ||
316 | snd_soc_unregister_platform(&pdev->dev); | ||
317 | return 0; | ||
335 | } | 318 | } |
336 | 319 | ||
320 | static struct platform_driver imx_pcm_driver = { | ||
321 | .driver = { | ||
322 | .name = "imx-pcm-audio", | ||
323 | .owner = THIS_MODULE, | ||
324 | }, | ||
325 | .probe = imx_soc_platform_probe, | ||
326 | .remove = __devexit_p(imx_soc_platform_remove), | ||
327 | }; | ||
328 | |||
329 | static int __init snd_imx_pcm_init(void) | ||
330 | { | ||
331 | return platform_driver_register(&imx_pcm_driver); | ||
332 | } | ||
333 | module_init(snd_imx_pcm_init); | ||
334 | |||
335 | static void __exit snd_imx_pcm_exit(void) | ||
336 | { | ||
337 | platform_driver_unregister(&imx_pcm_driver); | ||
338 | } | ||
339 | module_exit(snd_imx_pcm_exit); | ||
340 | MODULE_LICENSE("GPL"); | ||
341 | MODULE_ALIAS("platform:imx-pcm-audio"); | ||
diff --git a/sound/soc/imx/imx-pcm-fiq.c b/sound/soc/imx/imx-pcm-fiq.c index b2bf27282cd2..413b78da248f 100644 --- a/sound/soc/imx/imx-pcm-fiq.c +++ b/sound/soc/imx/imx-pcm-fiq.c | |||
@@ -236,6 +236,8 @@ static struct snd_pcm_ops imx_pcm_ops = { | |||
236 | .mmap = snd_imx_pcm_mmap, | 236 | .mmap = snd_imx_pcm_mmap, |
237 | }; | 237 | }; |
238 | 238 | ||
239 | static int ssi_irq = 0; | ||
240 | |||
239 | static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, | 241 | static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, |
240 | struct snd_pcm *pcm) | 242 | struct snd_pcm *pcm) |
241 | { | 243 | { |
@@ -245,7 +247,7 @@ static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
245 | if (ret) | 247 | if (ret) |
246 | return ret; | 248 | return ret; |
247 | 249 | ||
248 | if (dai->playback.channels_min) { | 250 | if (dai->driver->playback.channels_min) { |
249 | struct snd_pcm_substream *substream = | 251 | struct snd_pcm_substream *substream = |
250 | pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream; | 252 | pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream; |
251 | struct snd_dma_buffer *buf = &substream->dma_buffer; | 253 | struct snd_dma_buffer *buf = &substream->dma_buffer; |
@@ -253,7 +255,7 @@ static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
253 | imx_ssi_fiq_tx_buffer = (unsigned long)buf->area; | 255 | imx_ssi_fiq_tx_buffer = (unsigned long)buf->area; |
254 | } | 256 | } |
255 | 257 | ||
256 | if (dai->capture.channels_min) { | 258 | if (dai->driver->capture.channels_min) { |
257 | struct snd_pcm_substream *substream = | 259 | struct snd_pcm_substream *substream = |
258 | pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream; | 260 | pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream; |
259 | struct snd_dma_buffer *buf = &substream->dma_buffer; | 261 | struct snd_dma_buffer *buf = &substream->dma_buffer; |
@@ -267,24 +269,32 @@ static int imx_pcm_fiq_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
267 | return 0; | 269 | return 0; |
268 | } | 270 | } |
269 | 271 | ||
270 | static struct snd_soc_platform imx_soc_platform_fiq = { | 272 | static void imx_pcm_fiq_free(struct snd_pcm *pcm) |
271 | .pcm_ops = &imx_pcm_ops, | 273 | { |
274 | mxc_set_irq_fiq(ssi_irq, 0); | ||
275 | release_fiq(&fh); | ||
276 | imx_pcm_free(pcm); | ||
277 | } | ||
278 | |||
279 | static struct snd_soc_platform_driver imx_soc_platform_fiq = { | ||
280 | .ops = &imx_pcm_ops, | ||
272 | .pcm_new = imx_pcm_fiq_new, | 281 | .pcm_new = imx_pcm_fiq_new, |
273 | .pcm_free = imx_pcm_free, | 282 | .pcm_free = imx_pcm_fiq_free, |
274 | }; | 283 | }; |
275 | 284 | ||
276 | struct snd_soc_platform *imx_ssi_fiq_init(struct platform_device *pdev, | 285 | static int __devinit imx_soc_platform_probe(struct platform_device *pdev) |
277 | struct imx_ssi *ssi) | ||
278 | { | 286 | { |
279 | int ret = 0; | 287 | struct imx_ssi *ssi = platform_get_drvdata(pdev); |
288 | int ret; | ||
280 | 289 | ||
281 | ret = claim_fiq(&fh); | 290 | ret = claim_fiq(&fh); |
282 | if (ret) { | 291 | if (ret) { |
283 | dev_err(&pdev->dev, "failed to claim fiq: %d", ret); | 292 | dev_err(&pdev->dev, "failed to claim fiq: %d", ret); |
284 | return ERR_PTR(ret); | 293 | return ret; |
285 | } | 294 | } |
286 | 295 | ||
287 | mxc_set_irq_fiq(ssi->irq, 1); | 296 | mxc_set_irq_fiq(ssi->irq, 1); |
297 | ssi_irq = ssi->irq; | ||
288 | 298 | ||
289 | imx_pcm_fiq = ssi->irq; | 299 | imx_pcm_fiq = ssi->irq; |
290 | 300 | ||
@@ -293,13 +303,43 @@ struct snd_soc_platform *imx_ssi_fiq_init(struct platform_device *pdev, | |||
293 | ssi->dma_params_tx.burstsize = 4; | 303 | ssi->dma_params_tx.burstsize = 4; |
294 | ssi->dma_params_rx.burstsize = 6; | 304 | ssi->dma_params_rx.burstsize = 6; |
295 | 305 | ||
296 | return &imx_soc_platform_fiq; | 306 | ret = snd_soc_register_platform(&pdev->dev, &imx_soc_platform_fiq); |
307 | if (ret) | ||
308 | goto failed_register; | ||
309 | |||
310 | return 0; | ||
311 | |||
312 | failed_register: | ||
313 | mxc_set_irq_fiq(ssi_irq, 0); | ||
314 | release_fiq(&fh); | ||
315 | |||
316 | return ret; | ||
297 | } | 317 | } |
298 | 318 | ||
299 | void imx_ssi_fiq_exit(struct platform_device *pdev, | 319 | static int __devexit imx_soc_platform_remove(struct platform_device *pdev) |
300 | struct imx_ssi *ssi) | ||
301 | { | 320 | { |
302 | mxc_set_irq_fiq(ssi->irq, 0); | 321 | snd_soc_unregister_platform(&pdev->dev); |
303 | release_fiq(&fh); | 322 | return 0; |
304 | } | 323 | } |
305 | 324 | ||
325 | static struct platform_driver imx_pcm_driver = { | ||
326 | .driver = { | ||
327 | .name = "imx-fiq-pcm-audio", | ||
328 | .owner = THIS_MODULE, | ||
329 | }, | ||
330 | |||
331 | .probe = imx_soc_platform_probe, | ||
332 | .remove = __devexit_p(imx_soc_platform_remove), | ||
333 | }; | ||
334 | |||
335 | static int __init snd_imx_pcm_init(void) | ||
336 | { | ||
337 | return platform_driver_register(&imx_pcm_driver); | ||
338 | } | ||
339 | module_init(snd_imx_pcm_init); | ||
340 | |||
341 | static void __exit snd_imx_pcm_exit(void) | ||
342 | { | ||
343 | platform_driver_unregister(&imx_pcm_driver); | ||
344 | } | ||
345 | module_exit(snd_imx_pcm_exit); | ||
diff --git a/sound/soc/imx/imx-ssi.c b/sound/soc/imx/imx-ssi.c index c81da05a4f11..61fceb09cdb5 100644 --- a/sound/soc/imx/imx-ssi.c +++ b/sound/soc/imx/imx-ssi.c | |||
@@ -16,7 +16,7 @@ | |||
16 | * sane processor vendors have a FIFO per AC97 slot, the i.MX has only | 16 | * sane processor vendors have a FIFO per AC97 slot, the i.MX has only |
17 | * one FIFO which combines all valid receive slots. We cannot even select | 17 | * one FIFO which combines all valid receive slots. We cannot even select |
18 | * which slots we want to receive. The WM9712 with which this driver | 18 | * which slots we want to receive. The WM9712 with which this driver |
19 | * was developped with always sends GPIO status data in slot 12 which | 19 | * was developed with always sends GPIO status data in slot 12 which |
20 | * we receive in our (PCM-) data stream. The only chance we have is to | 20 | * we receive in our (PCM-) data stream. The only chance we have is to |
21 | * manually skip this data in the FIQ handler. With sampling rates different | 21 | * manually skip this data in the FIQ handler. With sampling rates different |
22 | * from 48000Hz not every frame has valid receive data, so the ratio | 22 | * from 48000Hz not every frame has valid receive data, so the ratio |
@@ -61,7 +61,7 @@ | |||
61 | static int imx_ssi_set_dai_tdm_slot(struct snd_soc_dai *cpu_dai, | 61 | static int imx_ssi_set_dai_tdm_slot(struct snd_soc_dai *cpu_dai, |
62 | unsigned int tx_mask, unsigned int rx_mask, int slots, int slot_width) | 62 | unsigned int tx_mask, unsigned int rx_mask, int slots, int slot_width) |
63 | { | 63 | { |
64 | struct imx_ssi *ssi = cpu_dai->private_data; | 64 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(cpu_dai); |
65 | u32 sccr; | 65 | u32 sccr; |
66 | 66 | ||
67 | sccr = readl(ssi->base + SSI_STCCR); | 67 | sccr = readl(ssi->base + SSI_STCCR); |
@@ -86,7 +86,7 @@ static int imx_ssi_set_dai_tdm_slot(struct snd_soc_dai *cpu_dai, | |||
86 | */ | 86 | */ |
87 | static int imx_ssi_set_dai_fmt(struct snd_soc_dai *cpu_dai, unsigned int fmt) | 87 | static int imx_ssi_set_dai_fmt(struct snd_soc_dai *cpu_dai, unsigned int fmt) |
88 | { | 88 | { |
89 | struct imx_ssi *ssi = cpu_dai->private_data; | 89 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(cpu_dai); |
90 | u32 strcr = 0, scr; | 90 | u32 strcr = 0, scr; |
91 | 91 | ||
92 | scr = readl(ssi->base + SSI_SCR) & ~(SSI_SCR_SYN | SSI_SCR_NET); | 92 | scr = readl(ssi->base + SSI_SCR) & ~(SSI_SCR_SYN | SSI_SCR_NET); |
@@ -108,7 +108,7 @@ static int imx_ssi_set_dai_fmt(struct snd_soc_dai *cpu_dai, unsigned int fmt) | |||
108 | break; | 108 | break; |
109 | case SND_SOC_DAIFMT_DSP_B: | 109 | case SND_SOC_DAIFMT_DSP_B: |
110 | /* data on rising edge of bclk, frame high with data */ | 110 | /* data on rising edge of bclk, frame high with data */ |
111 | strcr |= SSI_STCR_TFSL; | 111 | strcr |= SSI_STCR_TFSL | SSI_STCR_TXBIT0; |
112 | break; | 112 | break; |
113 | case SND_SOC_DAIFMT_DSP_A: | 113 | case SND_SOC_DAIFMT_DSP_A: |
114 | /* data on rising edge of bclk, frame high 1clk before data */ | 114 | /* data on rising edge of bclk, frame high 1clk before data */ |
@@ -164,7 +164,7 @@ static int imx_ssi_set_dai_fmt(struct snd_soc_dai *cpu_dai, unsigned int fmt) | |||
164 | static int imx_ssi_set_dai_sysclk(struct snd_soc_dai *cpu_dai, | 164 | static int imx_ssi_set_dai_sysclk(struct snd_soc_dai *cpu_dai, |
165 | int clk_id, unsigned int freq, int dir) | 165 | int clk_id, unsigned int freq, int dir) |
166 | { | 166 | { |
167 | struct imx_ssi *ssi = cpu_dai->private_data; | 167 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(cpu_dai); |
168 | u32 scr; | 168 | u32 scr; |
169 | 169 | ||
170 | scr = readl(ssi->base + SSI_SCR); | 170 | scr = readl(ssi->base + SSI_SCR); |
@@ -192,7 +192,7 @@ static int imx_ssi_set_dai_sysclk(struct snd_soc_dai *cpu_dai, | |||
192 | static int imx_ssi_set_dai_clkdiv(struct snd_soc_dai *cpu_dai, | 192 | static int imx_ssi_set_dai_clkdiv(struct snd_soc_dai *cpu_dai, |
193 | int div_id, int div) | 193 | int div_id, int div) |
194 | { | 194 | { |
195 | struct imx_ssi *ssi = cpu_dai->private_data; | 195 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(cpu_dai); |
196 | u32 stccr, srccr; | 196 | u32 stccr, srccr; |
197 | 197 | ||
198 | stccr = readl(ssi->base + SSI_STCCR); | 198 | stccr = readl(ssi->base + SSI_STCCR); |
@@ -241,7 +241,7 @@ static int imx_ssi_hw_params(struct snd_pcm_substream *substream, | |||
241 | struct snd_pcm_hw_params *params, | 241 | struct snd_pcm_hw_params *params, |
242 | struct snd_soc_dai *cpu_dai) | 242 | struct snd_soc_dai *cpu_dai) |
243 | { | 243 | { |
244 | struct imx_ssi *ssi = cpu_dai->private_data; | 244 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(cpu_dai); |
245 | struct imx_pcm_dma_params *dma_data; | 245 | struct imx_pcm_dma_params *dma_data; |
246 | u32 reg, sccr; | 246 | u32 reg, sccr; |
247 | 247 | ||
@@ -282,9 +282,7 @@ static int imx_ssi_hw_params(struct snd_pcm_substream *substream, | |||
282 | static int imx_ssi_trigger(struct snd_pcm_substream *substream, int cmd, | 282 | static int imx_ssi_trigger(struct snd_pcm_substream *substream, int cmd, |
283 | struct snd_soc_dai *dai) | 283 | struct snd_soc_dai *dai) |
284 | { | 284 | { |
285 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 285 | struct imx_ssi *ssi = snd_soc_dai_get_drvdata(dai); |
286 | struct snd_soc_dai *cpu_dai = rtd->dai->cpu_dai; | ||
287 | struct imx_ssi *ssi = cpu_dai->private_data; | ||
288 | unsigned int sier_bits, sier; | 286 | unsigned int sier_bits, sier; |
289 | unsigned int scr; | 287 | unsigned int scr; |
290 | 288 | ||
@@ -353,22 +351,6 @@ static struct snd_soc_dai_ops imx_ssi_pcm_dai_ops = { | |||
353 | .trigger = imx_ssi_trigger, | 351 | .trigger = imx_ssi_trigger, |
354 | }; | 352 | }; |
355 | 353 | ||
356 | static struct snd_soc_dai imx_ssi_dai = { | ||
357 | .playback = { | ||
358 | .channels_min = 2, | ||
359 | .channels_max = 2, | ||
360 | .rates = SNDRV_PCM_RATE_8000_96000, | ||
361 | .formats = SNDRV_PCM_FMTBIT_S16_LE, | ||
362 | }, | ||
363 | .capture = { | ||
364 | .channels_min = 2, | ||
365 | .channels_max = 2, | ||
366 | .rates = SNDRV_PCM_RATE_8000_96000, | ||
367 | .formats = SNDRV_PCM_FMTBIT_S16_LE, | ||
368 | }, | ||
369 | .ops = &imx_ssi_pcm_dai_ops, | ||
370 | }; | ||
371 | |||
372 | int snd_imx_pcm_mmap(struct snd_pcm_substream *substream, | 354 | int snd_imx_pcm_mmap(struct snd_pcm_substream *substream, |
373 | struct vm_area_struct *vma) | 355 | struct vm_area_struct *vma) |
374 | { | 356 | { |
@@ -384,6 +366,7 @@ int snd_imx_pcm_mmap(struct snd_pcm_substream *substream, | |||
384 | runtime->dma_bytes); | 366 | runtime->dma_bytes); |
385 | return ret; | 367 | return ret; |
386 | } | 368 | } |
369 | EXPORT_SYMBOL_GPL(snd_imx_pcm_mmap); | ||
387 | 370 | ||
388 | static int imx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) | 371 | static int imx_pcm_preallocate_dma_buffer(struct snd_pcm *pcm, int stream) |
389 | { | 372 | { |
@@ -415,14 +398,14 @@ int imx_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
415 | card->dev->dma_mask = &imx_pcm_dmamask; | 398 | card->dev->dma_mask = &imx_pcm_dmamask; |
416 | if (!card->dev->coherent_dma_mask) | 399 | if (!card->dev->coherent_dma_mask) |
417 | card->dev->coherent_dma_mask = DMA_BIT_MASK(32); | 400 | card->dev->coherent_dma_mask = DMA_BIT_MASK(32); |
418 | if (dai->playback.channels_min) { | 401 | if (dai->driver->playback.channels_min) { |
419 | ret = imx_pcm_preallocate_dma_buffer(pcm, | 402 | ret = imx_pcm_preallocate_dma_buffer(pcm, |
420 | SNDRV_PCM_STREAM_PLAYBACK); | 403 | SNDRV_PCM_STREAM_PLAYBACK); |
421 | if (ret) | 404 | if (ret) |
422 | goto out; | 405 | goto out; |
423 | } | 406 | } |
424 | 407 | ||
425 | if (dai->capture.channels_min) { | 408 | if (dai->driver->capture.channels_min) { |
426 | ret = imx_pcm_preallocate_dma_buffer(pcm, | 409 | ret = imx_pcm_preallocate_dma_buffer(pcm, |
427 | SNDRV_PCM_STREAM_CAPTURE); | 410 | SNDRV_PCM_STREAM_CAPTURE); |
428 | if (ret) | 411 | if (ret) |
@@ -432,6 +415,7 @@ int imx_pcm_new(struct snd_card *card, struct snd_soc_dai *dai, | |||
432 | out: | 415 | out: |
433 | return ret; | 416 | return ret; |
434 | } | 417 | } |
418 | EXPORT_SYMBOL_GPL(imx_pcm_new); | ||
435 | 419 | ||
436 | void imx_pcm_free(struct snd_pcm *pcm) | 420 | void imx_pcm_free(struct snd_pcm *pcm) |
437 | { | 421 | { |
@@ -453,14 +437,41 @@ void imx_pcm_free(struct snd_pcm *pcm) | |||
453 | buf->area = NULL; | 437 | buf->area = NULL; |
454 | } | 438 | } |
455 | } | 439 | } |
440 | EXPORT_SYMBOL_GPL(imx_pcm_free); | ||
441 | |||
442 | static int imx_ssi_dai_probe(struct snd_soc_dai *dai) | ||
443 | { | ||
444 | struct imx_ssi *ssi = dev_get_drvdata(dai->dev); | ||
445 | uint32_t val; | ||
446 | |||
447 | snd_soc_dai_set_drvdata(dai, ssi); | ||
456 | 448 | ||
457 | struct snd_soc_platform imx_soc_platform = { | 449 | val = SSI_SFCSR_TFWM0(ssi->dma_params_tx.burstsize) | |
458 | .name = "imx-audio", | 450 | SSI_SFCSR_RFWM0(ssi->dma_params_rx.burstsize); |
451 | writel(val, ssi->base + SSI_SFCSR); | ||
452 | |||
453 | return 0; | ||
454 | } | ||
455 | |||
456 | static struct snd_soc_dai_driver imx_ssi_dai = { | ||
457 | .probe = imx_ssi_dai_probe, | ||
458 | .playback = { | ||
459 | .channels_min = 1, | ||
460 | .channels_max = 2, | ||
461 | .rates = SNDRV_PCM_RATE_8000_96000, | ||
462 | .formats = SNDRV_PCM_FMTBIT_S16_LE, | ||
463 | }, | ||
464 | .capture = { | ||
465 | .channels_min = 1, | ||
466 | .channels_max = 2, | ||
467 | .rates = SNDRV_PCM_RATE_8000_96000, | ||
468 | .formats = SNDRV_PCM_FMTBIT_S16_LE, | ||
469 | }, | ||
470 | .ops = &imx_ssi_pcm_dai_ops, | ||
459 | }; | 471 | }; |
460 | EXPORT_SYMBOL_GPL(imx_soc_platform); | ||
461 | 472 | ||
462 | static struct snd_soc_dai imx_ac97_dai = { | 473 | static struct snd_soc_dai_driver imx_ac97_dai = { |
463 | .name = "AC97", | 474 | .probe = imx_ssi_dai_probe, |
464 | .ac97_control = 1, | 475 | .ac97_control = 1, |
465 | .playback = { | 476 | .playback = { |
466 | .stream_name = "AC97 Playback", | 477 | .stream_name = "AC97 Playback", |
@@ -580,25 +591,18 @@ struct snd_ac97_bus_ops soc_ac97_ops = { | |||
580 | }; | 591 | }; |
581 | EXPORT_SYMBOL_GPL(soc_ac97_ops); | 592 | EXPORT_SYMBOL_GPL(soc_ac97_ops); |
582 | 593 | ||
583 | struct snd_soc_dai imx_ssi_pcm_dai[2]; | ||
584 | EXPORT_SYMBOL_GPL(imx_ssi_pcm_dai); | ||
585 | |||
586 | static int imx_ssi_probe(struct platform_device *pdev) | 594 | static int imx_ssi_probe(struct platform_device *pdev) |
587 | { | 595 | { |
588 | struct resource *res; | 596 | struct resource *res; |
589 | struct imx_ssi *ssi; | 597 | struct imx_ssi *ssi; |
590 | struct imx_ssi_platform_data *pdata = pdev->dev.platform_data; | 598 | struct imx_ssi_platform_data *pdata = pdev->dev.platform_data; |
591 | struct snd_soc_platform *platform; | ||
592 | int ret = 0; | 599 | int ret = 0; |
593 | unsigned int val; | 600 | struct snd_soc_dai_driver *dai; |
594 | struct snd_soc_dai *dai = &imx_ssi_pcm_dai[pdev->id]; | ||
595 | |||
596 | if (dai->id >= ARRAY_SIZE(imx_ssi_pcm_dai)) | ||
597 | return -EINVAL; | ||
598 | 601 | ||
599 | ssi = kzalloc(sizeof(*ssi), GFP_KERNEL); | 602 | ssi = kzalloc(sizeof(*ssi), GFP_KERNEL); |
600 | if (!ssi) | 603 | if (!ssi) |
601 | return -ENOMEM; | 604 | return -ENOMEM; |
605 | dev_set_drvdata(&pdev->dev, ssi); | ||
602 | 606 | ||
603 | if (pdata) { | 607 | if (pdata) { |
604 | ssi->ac97_reset = pdata->ac97_reset; | 608 | ssi->ac97_reset = pdata->ac97_reset; |
@@ -643,15 +647,18 @@ static int imx_ssi_probe(struct platform_device *pdev) | |||
643 | } | 647 | } |
644 | ac97_ssi = ssi; | 648 | ac97_ssi = ssi; |
645 | setup_channel_to_ac97(ssi); | 649 | setup_channel_to_ac97(ssi); |
646 | memcpy(dai, &imx_ac97_dai, sizeof(imx_ac97_dai)); | 650 | dai = &imx_ac97_dai; |
647 | } else | 651 | } else |
648 | memcpy(dai, &imx_ssi_dai, sizeof(imx_ssi_dai)); | 652 | dai = &imx_ssi_dai; |
649 | 653 | ||
650 | writel(0x0, ssi->base + SSI_SIER); | 654 | writel(0x0, ssi->base + SSI_SIER); |
651 | 655 | ||
652 | ssi->dma_params_rx.dma_addr = res->start + SSI_SRX0; | 656 | ssi->dma_params_rx.dma_addr = res->start + SSI_SRX0; |
653 | ssi->dma_params_tx.dma_addr = res->start + SSI_STX0; | 657 | ssi->dma_params_tx.dma_addr = res->start + SSI_STX0; |
654 | 658 | ||
659 | ssi->dma_params_tx.burstsize = 4; | ||
660 | ssi->dma_params_rx.burstsize = 4; | ||
661 | |||
655 | res = platform_get_resource_byname(pdev, IORESOURCE_DMA, "tx0"); | 662 | res = platform_get_resource_byname(pdev, IORESOURCE_DMA, "tx0"); |
656 | if (res) | 663 | if (res) |
657 | ssi->dma_params_tx.dma = res->start; | 664 | ssi->dma_params_tx.dma = res->start; |
@@ -660,37 +667,50 @@ static int imx_ssi_probe(struct platform_device *pdev) | |||
660 | if (res) | 667 | if (res) |
661 | ssi->dma_params_rx.dma = res->start; | 668 | ssi->dma_params_rx.dma = res->start; |
662 | 669 | ||
663 | dai->id = pdev->id; | 670 | platform_set_drvdata(pdev, ssi); |
664 | dai->dev = &pdev->dev; | ||
665 | dai->name = kasprintf(GFP_KERNEL, "imx-ssi.%d", pdev->id); | ||
666 | dai->private_data = ssi; | ||
667 | |||
668 | if ((cpu_is_mx27() || cpu_is_mx21()) && | ||
669 | !(ssi->flags & IMX_SSI_USE_AC97) && | ||
670 | (ssi->flags & IMX_SSI_DMA)) { | ||
671 | ssi->flags |= IMX_SSI_DMA; | ||
672 | platform = imx_ssi_dma_mx2_init(pdev, ssi); | ||
673 | } else | ||
674 | platform = imx_ssi_fiq_init(pdev, ssi); | ||
675 | |||
676 | imx_soc_platform.pcm_ops = platform->pcm_ops; | ||
677 | imx_soc_platform.pcm_new = platform->pcm_new; | ||
678 | imx_soc_platform.pcm_free = platform->pcm_free; | ||
679 | |||
680 | val = SSI_SFCSR_TFWM0(ssi->dma_params_tx.burstsize) | | ||
681 | SSI_SFCSR_RFWM0(ssi->dma_params_rx.burstsize); | ||
682 | writel(val, ssi->base + SSI_SFCSR); | ||
683 | 671 | ||
684 | ret = snd_soc_register_dai(dai); | 672 | ret = snd_soc_register_dai(&pdev->dev, dai); |
685 | if (ret) { | 673 | if (ret) { |
686 | dev_err(&pdev->dev, "register DAI failed\n"); | 674 | dev_err(&pdev->dev, "register DAI failed\n"); |
687 | goto failed_register; | 675 | goto failed_register; |
688 | } | 676 | } |
689 | 677 | ||
690 | platform_set_drvdata(pdev, ssi); | 678 | ssi->soc_platform_pdev_fiq = platform_device_alloc("imx-fiq-pcm-audio", pdev->id); |
679 | if (!ssi->soc_platform_pdev_fiq) { | ||
680 | ret = -ENOMEM; | ||
681 | goto failed_pdev_fiq_alloc; | ||
682 | } | ||
683 | |||
684 | platform_set_drvdata(ssi->soc_platform_pdev_fiq, ssi); | ||
685 | ret = platform_device_add(ssi->soc_platform_pdev_fiq); | ||
686 | if (ret) { | ||
687 | dev_err(&pdev->dev, "failed to add platform device\n"); | ||
688 | goto failed_pdev_fiq_add; | ||
689 | } | ||
690 | |||
691 | ssi->soc_platform_pdev = platform_device_alloc("imx-pcm-audio", pdev->id); | ||
692 | if (!ssi->soc_platform_pdev) { | ||
693 | ret = -ENOMEM; | ||
694 | goto failed_pdev_alloc; | ||
695 | } | ||
696 | |||
697 | platform_set_drvdata(ssi->soc_platform_pdev, ssi); | ||
698 | ret = platform_device_add(ssi->soc_platform_pdev); | ||
699 | if (ret) { | ||
700 | dev_err(&pdev->dev, "failed to add platform device\n"); | ||
701 | goto failed_pdev_add; | ||
702 | } | ||
691 | 703 | ||
692 | return 0; | 704 | return 0; |
693 | 705 | ||
706 | failed_pdev_add: | ||
707 | platform_device_put(ssi->soc_platform_pdev); | ||
708 | failed_pdev_alloc: | ||
709 | platform_device_del(ssi->soc_platform_pdev_fiq); | ||
710 | failed_pdev_fiq_add: | ||
711 | platform_device_put(ssi->soc_platform_pdev_fiq); | ||
712 | failed_pdev_fiq_alloc: | ||
713 | snd_soc_unregister_dai(&pdev->dev); | ||
694 | failed_register: | 714 | failed_register: |
695 | failed_ac97: | 715 | failed_ac97: |
696 | iounmap(ssi->base); | 716 | iounmap(ssi->base); |
@@ -709,16 +729,15 @@ static int __devexit imx_ssi_remove(struct platform_device *pdev) | |||
709 | { | 729 | { |
710 | struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 730 | struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
711 | struct imx_ssi *ssi = platform_get_drvdata(pdev); | 731 | struct imx_ssi *ssi = platform_get_drvdata(pdev); |
712 | struct snd_soc_dai *dai = &imx_ssi_pcm_dai[pdev->id]; | ||
713 | 732 | ||
714 | snd_soc_unregister_dai(dai); | 733 | platform_device_unregister(ssi->soc_platform_pdev); |
734 | platform_device_unregister(ssi->soc_platform_pdev_fiq); | ||
735 | |||
736 | snd_soc_unregister_dai(&pdev->dev); | ||
715 | 737 | ||
716 | if (ssi->flags & IMX_SSI_USE_AC97) | 738 | if (ssi->flags & IMX_SSI_USE_AC97) |
717 | ac97_ssi = NULL; | 739 | ac97_ssi = NULL; |
718 | 740 | ||
719 | if (!(ssi->flags & IMX_SSI_DMA)) | ||
720 | imx_ssi_fiq_exit(pdev, ssi); | ||
721 | |||
722 | iounmap(ssi->base); | 741 | iounmap(ssi->base); |
723 | release_mem_region(res->start, resource_size(res)); | 742 | release_mem_region(res->start, resource_size(res)); |
724 | clk_disable(ssi->clk); | 743 | clk_disable(ssi->clk); |
@@ -733,34 +752,19 @@ static struct platform_driver imx_ssi_driver = { | |||
733 | .remove = __devexit_p(imx_ssi_remove), | 752 | .remove = __devexit_p(imx_ssi_remove), |
734 | 753 | ||
735 | .driver = { | 754 | .driver = { |
736 | .name = DRV_NAME, | 755 | .name = "imx-ssi", |
737 | .owner = THIS_MODULE, | 756 | .owner = THIS_MODULE, |
738 | }, | 757 | }, |
739 | }; | 758 | }; |
740 | 759 | ||
741 | static int __init imx_ssi_init(void) | 760 | static int __init imx_ssi_init(void) |
742 | { | 761 | { |
743 | int ret; | 762 | return platform_driver_register(&imx_ssi_driver); |
744 | |||
745 | ret = snd_soc_register_platform(&imx_soc_platform); | ||
746 | if (ret) { | ||
747 | pr_err("failed to register soc platform: %d\n", ret); | ||
748 | return ret; | ||
749 | } | ||
750 | |||
751 | ret = platform_driver_register(&imx_ssi_driver); | ||
752 | if (ret) { | ||
753 | snd_soc_unregister_platform(&imx_soc_platform); | ||
754 | return ret; | ||
755 | } | ||
756 | |||
757 | return 0; | ||
758 | } | 763 | } |
759 | 764 | ||
760 | static void __exit imx_ssi_exit(void) | 765 | static void __exit imx_ssi_exit(void) |
761 | { | 766 | { |
762 | platform_driver_unregister(&imx_ssi_driver); | 767 | platform_driver_unregister(&imx_ssi_driver); |
763 | snd_soc_unregister_platform(&imx_soc_platform); | ||
764 | } | 768 | } |
765 | 769 | ||
766 | module_init(imx_ssi_init); | 770 | module_init(imx_ssi_init); |
@@ -770,4 +774,4 @@ module_exit(imx_ssi_exit); | |||
770 | MODULE_AUTHOR("Sascha Hauer, <s.hauer@pengutronix.de>"); | 774 | MODULE_AUTHOR("Sascha Hauer, <s.hauer@pengutronix.de>"); |
771 | MODULE_DESCRIPTION("i.MX I2S/ac97 SoC Interface"); | 775 | MODULE_DESCRIPTION("i.MX I2S/ac97 SoC Interface"); |
772 | MODULE_LICENSE("GPL"); | 776 | MODULE_LICENSE("GPL"); |
773 | 777 | MODULE_ALIAS("platform:imx-ssi"); | |
diff --git a/sound/soc/imx/imx-ssi.h b/sound/soc/imx/imx-ssi.h index 55f26ebcd8c2..dc8a87530e3e 100644 --- a/sound/soc/imx/imx-ssi.h +++ b/sound/soc/imx/imx-ssi.h | |||
@@ -183,11 +183,11 @@ | |||
183 | #define IMX_SSI_RX_DIV_PSR 4 | 183 | #define IMX_SSI_RX_DIV_PSR 4 |
184 | #define IMX_SSI_RX_DIV_PM 5 | 184 | #define IMX_SSI_RX_DIV_PM 5 |
185 | 185 | ||
186 | extern struct snd_soc_dai imx_ssi_pcm_dai[2]; | ||
187 | extern struct snd_soc_platform imx_soc_platform; | ||
188 | |||
189 | #define DRV_NAME "imx-ssi" | 186 | #define DRV_NAME "imx-ssi" |
190 | 187 | ||
188 | #include <linux/dmaengine.h> | ||
189 | #include <mach/dma.h> | ||
190 | |||
191 | struct imx_pcm_dma_params { | 191 | struct imx_pcm_dma_params { |
192 | int dma; | 192 | int dma; |
193 | unsigned long dma_addr; | 193 | unsigned long dma_addr; |
@@ -197,7 +197,7 @@ struct imx_pcm_dma_params { | |||
197 | struct imx_ssi { | 197 | struct imx_ssi { |
198 | struct platform_device *ac97_dev; | 198 | struct platform_device *ac97_dev; |
199 | 199 | ||
200 | struct snd_soc_device imx_ac97; | 200 | struct snd_soc_dai *imx_ac97; |
201 | struct clk *clk; | 201 | struct clk *clk; |
202 | void __iomem *base; | 202 | void __iomem *base; |
203 | int irq; | 203 | int irq; |
@@ -213,6 +213,9 @@ struct imx_ssi { | |||
213 | struct imx_pcm_dma_params dma_params_tx; | 213 | struct imx_pcm_dma_params dma_params_tx; |
214 | 214 | ||
215 | int enabled; | 215 | int enabled; |
216 | |||
217 | struct platform_device *soc_platform_pdev; | ||
218 | struct platform_device *soc_platform_pdev_fiq; | ||
216 | }; | 219 | }; |
217 | 220 | ||
218 | struct snd_soc_platform *imx_ssi_fiq_init(struct platform_device *pdev, | 221 | struct snd_soc_platform *imx_ssi_fiq_init(struct platform_device *pdev, |
@@ -231,7 +234,4 @@ void imx_pcm_free(struct snd_pcm *pcm); | |||
231 | */ | 234 | */ |
232 | #define IMX_SSI_DMABUF_SIZE (64 * 1024) | 235 | #define IMX_SSI_DMABUF_SIZE (64 * 1024) |
233 | 236 | ||
234 | #define DMA_RXFIFO_BURST 0x4 | ||
235 | #define DMA_TXFIFO_BURST 0x6 | ||
236 | |||
237 | #endif /* _IMX_SSI_H */ | 237 | #endif /* _IMX_SSI_H */ |
diff --git a/sound/soc/imx/mx27vis-aic32x4.c b/sound/soc/imx/mx27vis-aic32x4.c new file mode 100644 index 000000000000..054110b91d42 --- /dev/null +++ b/sound/soc/imx/mx27vis-aic32x4.c | |||
@@ -0,0 +1,137 @@ | |||
1 | /* | ||
2 | * mx27vis-aic32x4.c | ||
3 | * | ||
4 | * Copyright 2011 Vista Silicon S.L. | ||
5 | * | ||
6 | * Author: Javier Martin <javier.martin@vista-silicon.com> | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify it | ||
9 | * under the terms of the GNU General Public License as published by the | ||
10 | * Free Software Foundation; either version 2 of the License, or (at your | ||
11 | * option) any later version. | ||
12 | * | ||
13 | * This program is distributed in the hope that it will be useful, | ||
14 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
15 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
16 | * GNU General Public License for more details. | ||
17 | * | ||
18 | * You should have received a copy of the GNU General Public License | ||
19 | * along with this program; if not, write to the Free Software | ||
20 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, | ||
21 | * MA 02110-1301, USA. | ||
22 | */ | ||
23 | |||
24 | #include <linux/module.h> | ||
25 | #include <linux/moduleparam.h> | ||
26 | #include <linux/device.h> | ||
27 | #include <linux/i2c.h> | ||
28 | #include <sound/core.h> | ||
29 | #include <sound/pcm.h> | ||
30 | #include <sound/soc.h> | ||
31 | #include <sound/soc-dapm.h> | ||
32 | #include <asm/mach-types.h> | ||
33 | #include <mach/audmux.h> | ||
34 | |||
35 | #include "../codecs/tlv320aic32x4.h" | ||
36 | #include "imx-ssi.h" | ||
37 | |||
38 | static int mx27vis_aic32x4_hw_params(struct snd_pcm_substream *substream, | ||
39 | struct snd_pcm_hw_params *params) | ||
40 | { | ||
41 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | ||
42 | struct snd_soc_dai *codec_dai = rtd->codec_dai; | ||
43 | struct snd_soc_dai *cpu_dai = rtd->cpu_dai; | ||
44 | int ret; | ||
45 | u32 dai_format; | ||
46 | |||
47 | dai_format = SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_NB_NF | | ||
48 | SND_SOC_DAIFMT_CBM_CFM; | ||
49 | |||
50 | /* set codec DAI configuration */ | ||
51 | snd_soc_dai_set_fmt(codec_dai, dai_format); | ||
52 | |||
53 | /* set cpu DAI configuration */ | ||
54 | snd_soc_dai_set_fmt(cpu_dai, dai_format); | ||
55 | |||
56 | ret = snd_soc_dai_set_sysclk(codec_dai, 0, | ||
57 | 25000000, SND_SOC_CLOCK_OUT); | ||
58 | if (ret) { | ||
59 | pr_err("%s: failed setting codec sysclk\n", __func__); | ||
60 | return ret; | ||
61 | } | ||
62 | |||
63 | ret = snd_soc_dai_set_sysclk(cpu_dai, IMX_SSP_SYS_CLK, 0, | ||
64 | SND_SOC_CLOCK_IN); | ||
65 | if (ret) { | ||
66 | pr_err("can't set CPU system clock IMX_SSP_SYS_CLK\n"); | ||
67 | return ret; | ||
68 | } | ||
69 | |||
70 | return 0; | ||
71 | } | ||
72 | |||
73 | static struct snd_soc_ops mx27vis_aic32x4_snd_ops = { | ||
74 | .hw_params = mx27vis_aic32x4_hw_params, | ||
75 | }; | ||
76 | |||
77 | static struct snd_soc_dai_link mx27vis_aic32x4_dai = { | ||
78 | .name = "tlv320aic32x4", | ||
79 | .stream_name = "TLV320AIC32X4", | ||
80 | .codec_dai_name = "tlv320aic32x4-hifi", | ||
81 | .platform_name = "imx-pcm-audio.0", | ||
82 | .codec_name = "tlv320aic32x4.0-0018", | ||
83 | .cpu_dai_name = "imx-ssi.0", | ||
84 | .ops = &mx27vis_aic32x4_snd_ops, | ||
85 | }; | ||
86 | |||
87 | static struct snd_soc_card mx27vis_aic32x4 = { | ||
88 | .name = "visstrim_m10-audio", | ||
89 | .dai_link = &mx27vis_aic32x4_dai, | ||
90 | .num_links = 1, | ||
91 | }; | ||
92 | |||
93 | static struct platform_device *mx27vis_aic32x4_snd_device; | ||
94 | |||
95 | static int __init mx27vis_aic32x4_init(void) | ||
96 | { | ||
97 | int ret; | ||
98 | |||
99 | mx27vis_aic32x4_snd_device = platform_device_alloc("soc-audio", -1); | ||
100 | if (!mx27vis_aic32x4_snd_device) | ||
101 | return -ENOMEM; | ||
102 | |||
103 | platform_set_drvdata(mx27vis_aic32x4_snd_device, &mx27vis_aic32x4); | ||
104 | ret = platform_device_add(mx27vis_aic32x4_snd_device); | ||
105 | |||
106 | if (ret) { | ||
107 | printk(KERN_ERR "ASoC: Platform device allocation failed\n"); | ||
108 | platform_device_put(mx27vis_aic32x4_snd_device); | ||
109 | } | ||
110 | |||
111 | /* Connect SSI0 as clock slave to SSI1 external pins */ | ||
112 | mxc_audmux_v1_configure_port(MX27_AUDMUX_HPCR1_SSI0, | ||
113 | MXC_AUDMUX_V1_PCR_SYN | | ||
114 | MXC_AUDMUX_V1_PCR_TFSDIR | | ||
115 | MXC_AUDMUX_V1_PCR_TCLKDIR | | ||
116 | MXC_AUDMUX_V1_PCR_TFCSEL(MX27_AUDMUX_PPCR1_SSI_PINS_1) | | ||
117 | MXC_AUDMUX_V1_PCR_RXDSEL(MX27_AUDMUX_PPCR1_SSI_PINS_1) | ||
118 | ); | ||
119 | mxc_audmux_v1_configure_port(MX27_AUDMUX_PPCR1_SSI_PINS_1, | ||
120 | MXC_AUDMUX_V1_PCR_SYN | | ||
121 | MXC_AUDMUX_V1_PCR_RXDSEL(MX27_AUDMUX_HPCR1_SSI0) | ||
122 | ); | ||
123 | |||
124 | return ret; | ||
125 | } | ||
126 | |||
127 | static void __exit mx27vis_aic32x4_exit(void) | ||
128 | { | ||
129 | platform_device_unregister(mx27vis_aic32x4_snd_device); | ||
130 | } | ||
131 | |||
132 | module_init(mx27vis_aic32x4_init); | ||
133 | module_exit(mx27vis_aic32x4_exit); | ||
134 | |||
135 | MODULE_AUTHOR("Javier Martin <javier.martin@vista-silicon.com>"); | ||
136 | MODULE_DESCRIPTION("ALSA SoC AIC32X4 mx27 visstrim"); | ||
137 | MODULE_LICENSE("GPL"); | ||
diff --git a/sound/soc/imx/phycore-ac97.c b/sound/soc/imx/phycore-ac97.c index a8307d55c70e..a7deb5cb2433 100644 --- a/sound/soc/imx/phycore-ac97.c +++ b/sound/soc/imx/phycore-ac97.c | |||
@@ -17,12 +17,8 @@ | |||
17 | #include <sound/core.h> | 17 | #include <sound/core.h> |
18 | #include <sound/pcm.h> | 18 | #include <sound/pcm.h> |
19 | #include <sound/soc.h> | 19 | #include <sound/soc.h> |
20 | #include <sound/soc-dapm.h> | ||
21 | #include <asm/mach-types.h> | 20 | #include <asm/mach-types.h> |
22 | 21 | ||
23 | #include "../codecs/wm9712.h" | ||
24 | #include "imx-ssi.h" | ||
25 | |||
26 | static struct snd_soc_card imx_phycore; | 22 | static struct snd_soc_card imx_phycore; |
27 | 23 | ||
28 | static struct snd_soc_ops imx_phycore_hifi_ops = { | 24 | static struct snd_soc_ops imx_phycore_hifi_ops = { |
@@ -32,23 +28,21 @@ static struct snd_soc_dai_link imx_phycore_dai_ac97[] = { | |||
32 | { | 28 | { |
33 | .name = "HiFi", | 29 | .name = "HiFi", |
34 | .stream_name = "HiFi", | 30 | .stream_name = "HiFi", |
35 | .codec_dai = &wm9712_dai[WM9712_DAI_AC97_HIFI], | 31 | .codec_dai_name = "wm9712-hifi", |
32 | .codec_name = "wm9712-codec", | ||
33 | .cpu_dai_name = "imx-ssi.0", | ||
34 | .platform_name = "imx-fiq-pcm-audio.0", | ||
36 | .ops = &imx_phycore_hifi_ops, | 35 | .ops = &imx_phycore_hifi_ops, |
37 | }, | 36 | }, |
38 | }; | 37 | }; |
39 | 38 | ||
40 | static struct snd_soc_card imx_phycore = { | 39 | static struct snd_soc_card imx_phycore = { |
41 | .name = "PhyCORE-audio", | 40 | .name = "PhyCORE-ac97-audio", |
42 | .platform = &imx_soc_platform, | ||
43 | .dai_link = imx_phycore_dai_ac97, | 41 | .dai_link = imx_phycore_dai_ac97, |
44 | .num_links = ARRAY_SIZE(imx_phycore_dai_ac97), | 42 | .num_links = ARRAY_SIZE(imx_phycore_dai_ac97), |
45 | }; | 43 | }; |
46 | 44 | ||
47 | static struct snd_soc_device imx_phycore_snd_devdata = { | 45 | static struct platform_device *imx_phycore_snd_ac97_device; |
48 | .card = &imx_phycore, | ||
49 | .codec_dev = &soc_codec_dev_wm9712, | ||
50 | }; | ||
51 | |||
52 | static struct platform_device *imx_phycore_snd_device; | 46 | static struct platform_device *imx_phycore_snd_device; |
53 | 47 | ||
54 | static int __init imx_phycore_init(void) | 48 | static int __init imx_phycore_init(void) |
@@ -59,27 +53,42 @@ static int __init imx_phycore_init(void) | |||
59 | /* return happy. We might run on a totally different machine */ | 53 | /* return happy. We might run on a totally different machine */ |
60 | return 0; | 54 | return 0; |
61 | 55 | ||
62 | imx_phycore_snd_device = platform_device_alloc("soc-audio", -1); | 56 | imx_phycore_snd_ac97_device = platform_device_alloc("soc-audio", -1); |
63 | if (!imx_phycore_snd_device) | 57 | if (!imx_phycore_snd_ac97_device) |
64 | return -ENOMEM; | 58 | return -ENOMEM; |
65 | 59 | ||
66 | imx_phycore_dai_ac97[0].cpu_dai = &imx_ssi_pcm_dai[0]; | 60 | platform_set_drvdata(imx_phycore_snd_ac97_device, &imx_phycore); |
61 | ret = platform_device_add(imx_phycore_snd_ac97_device); | ||
62 | if (ret) | ||
63 | goto fail1; | ||
67 | 64 | ||
68 | platform_set_drvdata(imx_phycore_snd_device, &imx_phycore_snd_devdata); | 65 | imx_phycore_snd_device = platform_device_alloc("wm9712-codec", -1); |
69 | imx_phycore_snd_devdata.dev = &imx_phycore_snd_device->dev; | 66 | if (!imx_phycore_snd_device) { |
67 | ret = -ENOMEM; | ||
68 | goto fail2; | ||
69 | } | ||
70 | ret = platform_device_add(imx_phycore_snd_device); | 70 | ret = platform_device_add(imx_phycore_snd_device); |
71 | 71 | ||
72 | if (ret) { | 72 | if (ret) { |
73 | printk(KERN_ERR "ASoC: Platform device allocation failed\n"); | 73 | printk(KERN_ERR "ASoC: Platform device allocation failed\n"); |
74 | platform_device_put(imx_phycore_snd_device); | 74 | goto fail3; |
75 | } | 75 | } |
76 | 76 | ||
77 | return 0; | ||
78 | |||
79 | fail3: | ||
80 | platform_device_put(imx_phycore_snd_device); | ||
81 | fail2: | ||
82 | platform_device_del(imx_phycore_snd_ac97_device); | ||
83 | fail1: | ||
84 | platform_device_put(imx_phycore_snd_ac97_device); | ||
77 | return ret; | 85 | return ret; |
78 | } | 86 | } |
79 | 87 | ||
80 | static void __exit imx_phycore_exit(void) | 88 | static void __exit imx_phycore_exit(void) |
81 | { | 89 | { |
82 | platform_device_unregister(imx_phycore_snd_device); | 90 | platform_device_unregister(imx_phycore_snd_device); |
91 | platform_device_unregister(imx_phycore_snd_ac97_device); | ||
83 | } | 92 | } |
84 | 93 | ||
85 | late_initcall(imx_phycore_init); | 94 | late_initcall(imx_phycore_init); |
diff --git a/sound/soc/imx/wm1133-ev1.c b/sound/soc/imx/wm1133-ev1.c index a6e7d9497639..75b4c72787e2 100644 --- a/sound/soc/imx/wm1133-ev1.c +++ b/sound/soc/imx/wm1133-ev1.c | |||
@@ -19,7 +19,6 @@ | |||
19 | #include <sound/pcm.h> | 19 | #include <sound/pcm.h> |
20 | #include <sound/pcm_params.h> | 20 | #include <sound/pcm_params.h> |
21 | #include <sound/soc.h> | 21 | #include <sound/soc.h> |
22 | #include <sound/soc-dapm.h> | ||
23 | 22 | ||
24 | #include <mach/audmux.h> | 23 | #include <mach/audmux.h> |
25 | 24 | ||
@@ -82,8 +81,8 @@ static int wm1133_ev1_hw_params(struct snd_pcm_substream *substream, | |||
82 | struct snd_pcm_hw_params *params) | 81 | struct snd_pcm_hw_params *params) |
83 | { | 82 | { |
84 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 83 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
85 | struct snd_soc_dai *codec_dai = rtd->dai->codec_dai; | 84 | struct snd_soc_dai *codec_dai = rtd->codec_dai; |
86 | struct snd_soc_dai *cpu_dai = rtd->dai->cpu_dai; | 85 | struct snd_soc_dai *cpu_dai = rtd->cpu_dai; |
87 | int i, found = 0; | 86 | int i, found = 0; |
88 | snd_pcm_format_t format = params_format(params); | 87 | snd_pcm_format_t format = params_format(params); |
89 | unsigned int rate = params_rate(params); | 88 | unsigned int rate = params_rate(params); |
@@ -210,31 +209,32 @@ static struct snd_soc_jack_pin mic_jack_pins[] = { | |||
210 | { .pin = "Mic2 Jack", .mask = SND_JACK_MICROPHONE }, | 209 | { .pin = "Mic2 Jack", .mask = SND_JACK_MICROPHONE }, |
211 | }; | 210 | }; |
212 | 211 | ||
213 | static int wm1133_ev1_init(struct snd_soc_codec *codec) | 212 | static int wm1133_ev1_init(struct snd_soc_pcm_runtime *rtd) |
214 | { | 213 | { |
215 | struct snd_soc_card *card = codec->socdev->card; | 214 | struct snd_soc_codec *codec = rtd->codec; |
215 | struct snd_soc_dapm_context *dapm = &codec->dapm; | ||
216 | 216 | ||
217 | snd_soc_dapm_new_controls(codec, wm1133_ev1_widgets, | 217 | snd_soc_dapm_new_controls(dapm, wm1133_ev1_widgets, |
218 | ARRAY_SIZE(wm1133_ev1_widgets)); | 218 | ARRAY_SIZE(wm1133_ev1_widgets)); |
219 | 219 | ||
220 | snd_soc_dapm_add_routes(codec, wm1133_ev1_map, | 220 | snd_soc_dapm_add_routes(dapm, wm1133_ev1_map, |
221 | ARRAY_SIZE(wm1133_ev1_map)); | 221 | ARRAY_SIZE(wm1133_ev1_map)); |
222 | 222 | ||
223 | /* Headphone jack detection */ | 223 | /* Headphone jack detection */ |
224 | snd_soc_jack_new(card, "Headphone", SND_JACK_HEADPHONE, &hp_jack); | 224 | snd_soc_jack_new(codec, "Headphone", SND_JACK_HEADPHONE, &hp_jack); |
225 | snd_soc_jack_add_pins(&hp_jack, ARRAY_SIZE(hp_jack_pins), | 225 | snd_soc_jack_add_pins(&hp_jack, ARRAY_SIZE(hp_jack_pins), |
226 | hp_jack_pins); | 226 | hp_jack_pins); |
227 | wm8350_hp_jack_detect(codec, WM8350_JDR, &hp_jack, SND_JACK_HEADPHONE); | 227 | wm8350_hp_jack_detect(codec, WM8350_JDR, &hp_jack, SND_JACK_HEADPHONE); |
228 | 228 | ||
229 | /* Microphone jack detection */ | 229 | /* Microphone jack detection */ |
230 | snd_soc_jack_new(card, "Microphone", | 230 | snd_soc_jack_new(codec, "Microphone", |
231 | SND_JACK_MICROPHONE | SND_JACK_BTN_0, &mic_jack); | 231 | SND_JACK_MICROPHONE | SND_JACK_BTN_0, &mic_jack); |
232 | snd_soc_jack_add_pins(&mic_jack, ARRAY_SIZE(mic_jack_pins), | 232 | snd_soc_jack_add_pins(&mic_jack, ARRAY_SIZE(mic_jack_pins), |
233 | mic_jack_pins); | 233 | mic_jack_pins); |
234 | wm8350_mic_jack_detect(codec, &mic_jack, SND_JACK_MICROPHONE, | 234 | wm8350_mic_jack_detect(codec, &mic_jack, SND_JACK_MICROPHONE, |
235 | SND_JACK_BTN_0); | 235 | SND_JACK_BTN_0); |
236 | 236 | ||
237 | snd_soc_dapm_force_enable_pin(codec, "Mic Bias"); | 237 | snd_soc_dapm_force_enable_pin(dapm, "Mic Bias"); |
238 | 238 | ||
239 | return 0; | 239 | return 0; |
240 | } | 240 | } |
@@ -243,8 +243,10 @@ static int wm1133_ev1_init(struct snd_soc_codec *codec) | |||
243 | static struct snd_soc_dai_link wm1133_ev1_dai = { | 243 | static struct snd_soc_dai_link wm1133_ev1_dai = { |
244 | .name = "WM1133-EV1", | 244 | .name = "WM1133-EV1", |
245 | .stream_name = "Audio", | 245 | .stream_name = "Audio", |
246 | .cpu_dai = &imx_ssi_pcm_dai[0], | 246 | .cpu_dai_name = "imx-ssi.0", |
247 | .codec_dai = &wm8350_dai, | 247 | .codec_dai_name = "wm8350-hifi", |
248 | .platform_name = "imx-fiq-pcm-audio.0", | ||
249 | .codec_name = "wm8350-codec.0-0x1a", | ||
248 | .init = wm1133_ev1_init, | 250 | .init = wm1133_ev1_init, |
249 | .ops = &wm1133_ev1_ops, | 251 | .ops = &wm1133_ev1_ops, |
250 | .symmetric_rates = 1, | 252 | .symmetric_rates = 1, |
@@ -252,16 +254,10 @@ static struct snd_soc_dai_link wm1133_ev1_dai = { | |||
252 | 254 | ||
253 | static struct snd_soc_card wm1133_ev1 = { | 255 | static struct snd_soc_card wm1133_ev1 = { |
254 | .name = "WM1133-EV1", | 256 | .name = "WM1133-EV1", |
255 | .platform = &imx_soc_platform, | ||
256 | .dai_link = &wm1133_ev1_dai, | 257 | .dai_link = &wm1133_ev1_dai, |
257 | .num_links = 1, | 258 | .num_links = 1, |
258 | }; | 259 | }; |
259 | 260 | ||
260 | static struct snd_soc_device wm1133_ev1_snd_devdata = { | ||
261 | .card = &wm1133_ev1, | ||
262 | .codec_dev = &soc_codec_dev_wm8350, | ||
263 | }; | ||
264 | |||
265 | static struct platform_device *wm1133_ev1_snd_device; | 261 | static struct platform_device *wm1133_ev1_snd_device; |
266 | 262 | ||
267 | static int __init wm1133_ev1_audio_init(void) | 263 | static int __init wm1133_ev1_audio_init(void) |
@@ -286,8 +282,7 @@ static int __init wm1133_ev1_audio_init(void) | |||
286 | if (!wm1133_ev1_snd_device) | 282 | if (!wm1133_ev1_snd_device) |
287 | return -ENOMEM; | 283 | return -ENOMEM; |
288 | 284 | ||
289 | platform_set_drvdata(wm1133_ev1_snd_device, &wm1133_ev1_snd_devdata); | 285 | platform_set_drvdata(wm1133_ev1_snd_device, &wm1133_ev1); |
290 | wm1133_ev1_snd_devdata.dev = &wm1133_ev1_snd_device->dev; | ||
291 | ret = platform_device_add(wm1133_ev1_snd_device); | 286 | ret = platform_device_add(wm1133_ev1_snd_device); |
292 | 287 | ||
293 | if (ret) | 288 | if (ret) |