diff options
author | Liam Girdwood <lg@opensource.wolfsonmicro.com> | 2006-10-19 14:35:56 -0400 |
---|---|---|
committer | Jaroslav Kysela <perex@suse.cz> | 2007-02-09 03:01:07 -0500 |
commit | a71a468a50f1385855e28864e26251b02df829bb (patch) | |
tree | 243daee96ea5c55c88a186aa03b7917f7ad533f6 /sound/soc/soc-core.c | |
parent | 543a0fbe18d0b44f3d037fe6b59458fa0c0d5e4b (diff) |
[ALSA] ASoC: Add support for BCLK based on (Rate * Chn * Word Size)
This patch adds support for the DAI BCLK to be generated by multiplying
Rate * Channels * Word Size (RCW).
This now gives 3 options for BCLK clocking and synchronisation :-
1. BCLK = Rate * x
2. BCLK = MCLK / x
3. BCLK = Rate * Chn * Word Size. (New)
Changes:-
o Add support for RCW generation of BCLK
o Update Documentation to include RCW.
o Update DAI documentation for label = value DAI modes.
o Add RCW support to wm8731, wm8750 and pxa2xx-i2s drivers.
Signed-off-by: Liam Girdwood <lg@opensource.wolfsonmicro.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
Diffstat (limited to 'sound/soc/soc-core.c')
-rw-r--r-- | sound/soc/soc-core.c | 215 |
1 files changed, 167 insertions, 48 deletions
diff --git a/sound/soc/soc-core.c b/sound/soc/soc-core.c index 2ce0c8251dc3..6da1616bf776 100644 --- a/sound/soc/soc-core.c +++ b/sound/soc/soc-core.c | |||
@@ -51,6 +51,8 @@ | |||
51 | #define dbgc(format, arg...) | 51 | #define dbgc(format, arg...) |
52 | #endif | 52 | #endif |
53 | 53 | ||
54 | #define CODEC_CPU(codec, cpu) ((codec << 4) | cpu) | ||
55 | |||
54 | static DEFINE_MUTEX(pcm_mutex); | 56 | static DEFINE_MUTEX(pcm_mutex); |
55 | static DEFINE_MUTEX(io_mutex); | 57 | static DEFINE_MUTEX(io_mutex); |
56 | static struct workqueue_struct *soc_workq; | 58 | static struct workqueue_struct *soc_workq; |
@@ -150,11 +152,11 @@ static unsigned inline int soc_get_mclk(struct snd_soc_pcm_runtime *rtd, | |||
150 | } | 152 | } |
151 | 153 | ||
152 | /* changes a bitclk multiplier mask to a divider mask */ | 154 | /* changes a bitclk multiplier mask to a divider mask */ |
153 | static u16 soc_bfs_mult_to_div(u16 bfs, int rate, unsigned int mclk, | 155 | static u64 soc_bfs_rcw_to_div(u64 bfs, int rate, unsigned int mclk, |
154 | unsigned int pcmfmt, unsigned int chn) | 156 | unsigned int pcmfmt, unsigned int chn) |
155 | { | 157 | { |
156 | int i, j; | 158 | int i, j; |
157 | u16 bfs_ = 0; | 159 | u64 bfs_ = 0; |
158 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; | 160 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; |
159 | 161 | ||
160 | if (size <= 0) | 162 | if (size <= 0) |
@@ -162,17 +164,14 @@ static u16 soc_bfs_mult_to_div(u16 bfs, int rate, unsigned int mclk, | |||
162 | 164 | ||
163 | /* the minimum bit clock that has enough bandwidth */ | 165 | /* the minimum bit clock that has enough bandwidth */ |
164 | min = size * rate * chn; | 166 | min = size * rate * chn; |
165 | dbgc("mult --> div min bclk %d with mclk %d\n", min, mclk); | 167 | dbgc("rcw --> div min bclk %d with mclk %d\n", min, mclk); |
166 | 168 | ||
167 | for (i = 0; i < 16; i++) { | 169 | for (i = 0; i < 64; i++) { |
168 | if ((bfs >> i) & 0x1) { | 170 | if ((bfs >> i) & 0x1) { |
169 | j = rate * SND_SOC_FSB_REAL(1<<i); | 171 | j = min * (i + 1); |
170 | 172 | bfs_ |= SND_SOC_FSBD(mclk/j); | |
171 | if (j >= min) { | 173 | dbgc("rcw --> div support mult %d\n", |
172 | bfs_ |= SND_SOC_FSBD(mclk/j); | 174 | SND_SOC_FSBD_REAL(1<<i)); |
173 | dbgc("mult --> div support mult %d\n", | ||
174 | SND_SOC_FSB_REAL(1<<i)); | ||
175 | } | ||
176 | } | 175 | } |
177 | } | 176 | } |
178 | 177 | ||
@@ -180,11 +179,11 @@ static u16 soc_bfs_mult_to_div(u16 bfs, int rate, unsigned int mclk, | |||
180 | } | 179 | } |
181 | 180 | ||
182 | /* changes a bitclk divider mask to a multiplier mask */ | 181 | /* changes a bitclk divider mask to a multiplier mask */ |
183 | static u16 soc_bfs_div_to_mult(u16 bfs, int rate, unsigned int mclk, | 182 | static u64 soc_bfs_div_to_rcw(u64 bfs, int rate, unsigned int mclk, |
184 | unsigned int pcmfmt, unsigned int chn) | 183 | unsigned int pcmfmt, unsigned int chn) |
185 | { | 184 | { |
186 | int i, j; | 185 | int i, j; |
187 | u16 bfs_ = 0; | 186 | u64 bfs_ = 0; |
188 | 187 | ||
189 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; | 188 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; |
190 | 189 | ||
@@ -193,15 +192,15 @@ static u16 soc_bfs_div_to_mult(u16 bfs, int rate, unsigned int mclk, | |||
193 | 192 | ||
194 | /* the minimum bit clock that has enough bandwidth */ | 193 | /* the minimum bit clock that has enough bandwidth */ |
195 | min = size * rate * chn; | 194 | min = size * rate * chn; |
196 | dbgc("div to mult min bclk %d with mclk %d\n", min, mclk); | 195 | dbgc("div to rcw min bclk %d with mclk %d\n", min, mclk); |
197 | 196 | ||
198 | for (i = 0; i < 16; i++) { | 197 | for (i = 0; i < 64; i++) { |
199 | if ((bfs >> i) & 0x1) { | 198 | if ((bfs >> i) & 0x1) { |
200 | j = mclk / (SND_SOC_FSBD_REAL(1<<i)); | 199 | j = mclk / (i + 1); |
201 | if (j >= min) { | 200 | if (j >= min) { |
202 | bfs_ |= SND_SOC_FSB(j/rate); | 201 | bfs_ |= SND_SOC_FSBW(j/min); |
203 | dbgc("div --> mult support div %d\n", | 202 | dbgc("div --> rcw support div %d\n", |
204 | SND_SOC_FSBD_REAL(1<<i)); | 203 | SND_SOC_FSBW_REAL(1<<i)); |
205 | } | 204 | } |
206 | } | 205 | } |
207 | } | 206 | } |
@@ -209,6 +208,52 @@ static u16 soc_bfs_div_to_mult(u16 bfs, int rate, unsigned int mclk, | |||
209 | return bfs_; | 208 | return bfs_; |
210 | } | 209 | } |
211 | 210 | ||
211 | /* changes a constant bitclk to a multiplier mask */ | ||
212 | static u64 soc_bfs_rate_to_rcw(u64 bfs, int rate, unsigned int mclk, | ||
213 | unsigned int pcmfmt, unsigned int chn) | ||
214 | { | ||
215 | unsigned int bfs_ = rate * bfs; | ||
216 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; | ||
217 | |||
218 | if (size <= 0) | ||
219 | return 0; | ||
220 | |||
221 | /* the minimum bit clock that has enough bandwidth */ | ||
222 | min = size * rate * chn; | ||
223 | dbgc("rate --> rcw min bclk %d with mclk %d\n", min, mclk); | ||
224 | |||
225 | if (bfs_ < min) | ||
226 | return 0; | ||
227 | else { | ||
228 | bfs_ = SND_SOC_FSBW(bfs_/min); | ||
229 | dbgc("rate --> rcw support div %d\n", SND_SOC_FSBW_REAL(bfs_)); | ||
230 | return bfs_; | ||
231 | } | ||
232 | } | ||
233 | |||
234 | /* changes a bitclk multiplier mask to a divider mask */ | ||
235 | static u64 soc_bfs_rate_to_div(u64 bfs, int rate, unsigned int mclk, | ||
236 | unsigned int pcmfmt, unsigned int chn) | ||
237 | { | ||
238 | unsigned int bfs_ = rate * bfs; | ||
239 | int size = snd_pcm_format_physical_width(pcmfmt), min = 0; | ||
240 | |||
241 | if (size <= 0) | ||
242 | return 0; | ||
243 | |||
244 | /* the minimum bit clock that has enough bandwidth */ | ||
245 | min = size * rate * chn; | ||
246 | dbgc("rate --> div min bclk %d with mclk %d\n", min, mclk); | ||
247 | |||
248 | if (bfs_ < min) | ||
249 | return 0; | ||
250 | else { | ||
251 | bfs_ = SND_SOC_FSBW(mclk/bfs_); | ||
252 | dbgc("rate --> div support div %d\n", SND_SOC_FSBD_REAL(bfs_)); | ||
253 | return bfs_; | ||
254 | } | ||
255 | } | ||
256 | |||
212 | /* Matches codec DAI and SoC CPU DAI hardware parameters */ | 257 | /* Matches codec DAI and SoC CPU DAI hardware parameters */ |
213 | static int soc_hw_match_params(struct snd_pcm_substream *substream, | 258 | static int soc_hw_match_params(struct snd_pcm_substream *substream, |
214 | struct snd_pcm_hw_params *params) | 259 | struct snd_pcm_hw_params *params) |
@@ -217,9 +262,10 @@ static int soc_hw_match_params(struct snd_pcm_substream *substream, | |||
217 | struct snd_soc_dai_mode *codec_dai_mode = NULL; | 262 | struct snd_soc_dai_mode *codec_dai_mode = NULL; |
218 | struct snd_soc_dai_mode *cpu_dai_mode = NULL; | 263 | struct snd_soc_dai_mode *cpu_dai_mode = NULL; |
219 | struct snd_soc_clock_info clk_info; | 264 | struct snd_soc_clock_info clk_info; |
220 | unsigned int fs, mclk, codec_bfs, cpu_bfs, rate = params_rate(params), | 265 | unsigned int fs, mclk, rate = params_rate(params), |
221 | chn, j, k, cpu_bclk, codec_bclk, pcmrate; | 266 | chn, j, k, cpu_bclk, codec_bclk, pcmrate; |
222 | u16 fmt = 0; | 267 | u16 fmt = 0; |
268 | u64 codec_bfs, cpu_bfs; | ||
223 | 269 | ||
224 | dbg("asoc: match version %s\n", SND_SOC_VERSION); | 270 | dbg("asoc: match version %s\n", SND_SOC_VERSION); |
225 | clk_info.rate = rate; | 271 | clk_info.rate = rate; |
@@ -309,44 +355,98 @@ static int soc_hw_match_params(struct snd_pcm_substream *substream, | |||
309 | * used in the codec and cpu DAI modes. We always choose the | 355 | * used in the codec and cpu DAI modes. We always choose the |
310 | * lowest possible clocks to reduce power. | 356 | * lowest possible clocks to reduce power. |
311 | */ | 357 | */ |
312 | if (codec_dai_mode->flags & cpu_dai_mode->flags & | 358 | switch (CODEC_CPU(codec_dai_mode->flags, cpu_dai_mode->flags)) { |
313 | SND_SOC_DAI_BFS_DIV) { | 359 | case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_DIV): |
314 | /* cpu & codec bfs dividers */ | 360 | /* cpu & codec bfs dividers */ |
315 | rtd->cpu_dai->dai_runtime.bfs = | 361 | rtd->cpu_dai->dai_runtime.bfs = |
316 | rtd->codec_dai->dai_runtime.bfs = | 362 | rtd->codec_dai->dai_runtime.bfs = |
317 | 1 << (fls(codec_dai_mode->bfs & cpu_dai_mode->bfs) - 1); | 363 | 1 << (fls(codec_dai_mode->bfs & cpu_dai_mode->bfs) - 1); |
318 | } else if (codec_dai_mode->flags & SND_SOC_DAI_BFS_DIV) { | 364 | break; |
319 | /* normalise bfs codec divider & cpu mult */ | 365 | case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_RCW): |
320 | codec_bfs = soc_bfs_div_to_mult(codec_dai_mode->bfs, rate, | 366 | /* normalise bfs codec divider & cpu rcw mult */ |
367 | codec_bfs = soc_bfs_div_to_rcw(codec_dai_mode->bfs, rate, | ||
321 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | 368 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); |
322 | rtd->cpu_dai->dai_runtime.bfs = | 369 | rtd->cpu_dai->dai_runtime.bfs = |
323 | 1 << (ffs(codec_bfs & cpu_dai_mode->bfs) - 1); | 370 | 1 << (ffs(codec_bfs & cpu_dai_mode->bfs) - 1); |
324 | cpu_bfs = soc_bfs_mult_to_div(cpu_dai_mode->bfs, rate, mclk, | 371 | cpu_bfs = soc_bfs_rcw_to_div(cpu_dai_mode->bfs, rate, mclk, |
325 | rtd->codec_dai->dai_runtime.pcmfmt, chn); | 372 | rtd->codec_dai->dai_runtime.pcmfmt, chn); |
326 | rtd->codec_dai->dai_runtime.bfs = | 373 | rtd->codec_dai->dai_runtime.bfs = |
327 | 1 << (fls(codec_dai_mode->bfs & cpu_bfs) - 1); | 374 | 1 << (fls(codec_dai_mode->bfs & cpu_bfs) - 1); |
328 | } else if (cpu_dai_mode->flags & SND_SOC_DAI_BFS_DIV) { | 375 | break; |
329 | /* normalise bfs codec mult & cpu divider */ | 376 | case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_DIV): |
330 | codec_bfs = soc_bfs_mult_to_div(codec_dai_mode->bfs, rate, | 377 | /* normalise bfs codec rcw mult & cpu divider */ |
378 | codec_bfs = soc_bfs_rcw_to_div(codec_dai_mode->bfs, rate, | ||
331 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | 379 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); |
332 | rtd->cpu_dai->dai_runtime.bfs = | 380 | rtd->cpu_dai->dai_runtime.bfs = |
333 | 1 << (fls(codec_bfs & cpu_dai_mode->bfs) -1); | 381 | 1 << (fls(codec_bfs & cpu_dai_mode->bfs) -1); |
334 | cpu_bfs = soc_bfs_div_to_mult(cpu_dai_mode->bfs, rate, mclk, | 382 | cpu_bfs = soc_bfs_div_to_rcw(cpu_dai_mode->bfs, rate, mclk, |
335 | rtd->codec_dai->dai_runtime.pcmfmt, chn); | 383 | rtd->codec_dai->dai_runtime.pcmfmt, chn); |
336 | rtd->codec_dai->dai_runtime.bfs = | 384 | rtd->codec_dai->dai_runtime.bfs = |
337 | 1 << (ffs(codec_dai_mode->bfs & cpu_bfs) -1); | 385 | 1 << (ffs(codec_dai_mode->bfs & cpu_bfs) -1); |
338 | } else { | 386 | break; |
339 | /* codec & cpu bfs rate multipliers */ | 387 | case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_RCW): |
388 | /* codec & cpu bfs rate rcw multipliers */ | ||
340 | rtd->cpu_dai->dai_runtime.bfs = | 389 | rtd->cpu_dai->dai_runtime.bfs = |
341 | rtd->codec_dai->dai_runtime.bfs = | 390 | rtd->codec_dai->dai_runtime.bfs = |
342 | 1 << (ffs(codec_dai_mode->bfs & cpu_dai_mode->bfs) -1); | 391 | 1 << (ffs(codec_dai_mode->bfs & cpu_dai_mode->bfs) -1); |
392 | break; | ||
393 | case CODEC_CPU(SND_SOC_DAI_BFS_DIV, SND_SOC_DAI_BFS_RATE): | ||
394 | /* normalise cpu bfs rate const multiplier & codec div */ | ||
395 | cpu_bfs = soc_bfs_rate_to_div(cpu_dai_mode->bfs, rate, | ||
396 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | ||
397 | if(codec_dai_mode->bfs & cpu_bfs) { | ||
398 | rtd->codec_dai->dai_runtime.bfs = cpu_bfs; | ||
399 | rtd->cpu_dai->dai_runtime.bfs = cpu_dai_mode->bfs; | ||
400 | } else | ||
401 | rtd->cpu_dai->dai_runtime.bfs = 0; | ||
402 | break; | ||
403 | case CODEC_CPU(SND_SOC_DAI_BFS_RCW, SND_SOC_DAI_BFS_RATE): | ||
404 | /* normalise cpu bfs rate const multiplier & codec rcw mult */ | ||
405 | cpu_bfs = soc_bfs_rate_to_rcw(cpu_dai_mode->bfs, rate, | ||
406 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | ||
407 | if(codec_dai_mode->bfs & cpu_bfs) { | ||
408 | rtd->codec_dai->dai_runtime.bfs = cpu_bfs; | ||
409 | rtd->cpu_dai->dai_runtime.bfs = cpu_dai_mode->bfs; | ||
410 | } else | ||
411 | rtd->cpu_dai->dai_runtime.bfs = 0; | ||
412 | break; | ||
413 | case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_RCW): | ||
414 | /* normalise cpu bfs rate rcw multiplier & codec const mult */ | ||
415 | codec_bfs = soc_bfs_rate_to_rcw(codec_dai_mode->bfs, rate, | ||
416 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | ||
417 | if(cpu_dai_mode->bfs & codec_bfs) { | ||
418 | rtd->cpu_dai->dai_runtime.bfs = codec_bfs; | ||
419 | rtd->codec_dai->dai_runtime.bfs = codec_dai_mode->bfs; | ||
420 | } else | ||
421 | rtd->cpu_dai->dai_runtime.bfs = 0; | ||
422 | break; | ||
423 | case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_DIV): | ||
424 | /* normalise cpu bfs div & codec const mult */ | ||
425 | codec_bfs = soc_bfs_rate_to_div(codec_dai_mode->bfs, rate, | ||
426 | mclk, rtd->codec_dai->dai_runtime.pcmfmt, chn); | ||
427 | if(codec_dai_mode->bfs & codec_bfs) { | ||
428 | rtd->cpu_dai->dai_runtime.bfs = codec_bfs; | ||
429 | rtd->codec_dai->dai_runtime.bfs = codec_dai_mode->bfs; | ||
430 | } else | ||
431 | rtd->cpu_dai->dai_runtime.bfs = 0; | ||
432 | break; | ||
433 | case CODEC_CPU(SND_SOC_DAI_BFS_RATE, SND_SOC_DAI_BFS_RATE): | ||
434 | /* cpu & codec constant mult */ | ||
435 | if(codec_dai_mode->bfs == cpu_dai_mode->bfs) | ||
436 | rtd->cpu_dai->dai_runtime.bfs = | ||
437 | rtd->codec_dai->dai_runtime.bfs = | ||
438 | codec_dai_mode->bfs; | ||
439 | else | ||
440 | rtd->cpu_dai->dai_runtime.bfs = | ||
441 | rtd->codec_dai->dai_runtime.bfs = 0; | ||
442 | break; | ||
343 | } | 443 | } |
344 | 444 | ||
345 | /* make sure the bit clock speed is acceptable */ | 445 | /* make sure the bit clock speed is acceptable */ |
346 | if (!rtd->cpu_dai->dai_runtime.bfs || | 446 | if (!rtd->cpu_dai->dai_runtime.bfs || |
347 | !rtd->codec_dai->dai_runtime.bfs) { | 447 | !rtd->codec_dai->dai_runtime.bfs) { |
348 | dbgc("asoc: DAI[%d:%d] failed to match BFS\n", j, k); | 448 | dbgc("asoc: DAI[%d:%d] failed to match BFS\n", j, k); |
349 | dbgc("asoc: cpu_dai %x codec %x\n", | 449 | dbgc("asoc: cpu_dai %llu codec %llu\n", |
350 | rtd->cpu_dai->dai_runtime.bfs, | 450 | rtd->cpu_dai->dai_runtime.bfs, |
351 | rtd->codec_dai->dai_runtime.bfs); | 451 | rtd->codec_dai->dai_runtime.bfs); |
352 | dbgc("asoc: mclk %d hwfmt %x\n", mclk, fmt); | 452 | dbgc("asoc: mclk %d hwfmt %x\n", mclk, fmt); |
@@ -378,26 +478,41 @@ found: | |||
378 | dbg("asoc: codec fs %d mclk %d bfs div %d bclk %d\n", | 478 | dbg("asoc: codec fs %d mclk %d bfs div %d bclk %d\n", |
379 | rtd->codec_dai->dai_runtime.fs, mclk, | 479 | rtd->codec_dai->dai_runtime.fs, mclk, |
380 | SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); | 480 | SND_SOC_FSBD_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); |
381 | } else { | 481 | } else if(rtd->codec_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RATE) { |
382 | codec_bclk = params_rate(params) * | 482 | codec_bclk = params_rate(params) * rtd->codec_dai->dai_runtime.bfs; |
383 | SND_SOC_FSB_REAL(rtd->codec_dai->dai_runtime.bfs); | 483 | dbg("asoc: codec fs %d mclk %d bfs rate mult %llu bclk %d\n", |
384 | dbg("asoc: codec fs %d mclk %d bfs mult %d bclk %d\n", | ||
385 | rtd->codec_dai->dai_runtime.fs, mclk, | 484 | rtd->codec_dai->dai_runtime.fs, mclk, |
386 | SND_SOC_FSB_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); | 485 | rtd->codec_dai->dai_runtime.bfs, codec_bclk); |
387 | } | 486 | } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RCW) { |
487 | codec_bclk = params_rate(params) * params_channels(params) * | ||
488 | snd_pcm_format_physical_width(rtd->codec_dai->dai_runtime.pcmfmt) * | ||
489 | SND_SOC_FSBW_REAL(rtd->codec_dai->dai_runtime.bfs); | ||
490 | dbg("asoc: codec fs %d mclk %d bfs rcw mult %d bclk %d\n", | ||
491 | rtd->codec_dai->dai_runtime.fs, mclk, | ||
492 | SND_SOC_FSBW_REAL(rtd->codec_dai->dai_runtime.bfs), codec_bclk); | ||
493 | } else | ||
494 | codec_bclk = 0; | ||
495 | |||
388 | if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_DIV) { | 496 | if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_DIV) { |
389 | cpu_bclk = (rtd->cpu_dai->dai_runtime.fs * params_rate(params)) / | 497 | cpu_bclk = (rtd->cpu_dai->dai_runtime.fs * params_rate(params)) / |
390 | SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); | 498 | SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs); |
391 | dbg("asoc: cpu fs %d mclk %d bfs div %d bclk %d\n", | 499 | dbg("asoc: cpu fs %d mclk %d bfs div %d bclk %d\n", |
392 | rtd->cpu_dai->dai_runtime.fs, mclk, | 500 | rtd->cpu_dai->dai_runtime.fs, mclk, |
393 | SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); | 501 | SND_SOC_FSBD_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); |
394 | } else { | 502 | } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RATE) { |
395 | cpu_bclk = params_rate(params) * | 503 | cpu_bclk = params_rate(params) * rtd->cpu_dai->dai_runtime.bfs; |
396 | SND_SOC_FSB_REAL(rtd->cpu_dai->dai_runtime.bfs); | 504 | dbg("asoc: cpu fs %d mclk %d bfs rate mult %llu bclk %d\n", |
397 | dbg("asoc: cpu fs %d mclk %d bfs mult %d bclk %d\n", | ||
398 | rtd->cpu_dai->dai_runtime.fs, mclk, | 505 | rtd->cpu_dai->dai_runtime.fs, mclk, |
399 | SND_SOC_FSB_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); | 506 | rtd->cpu_dai->dai_runtime.bfs, cpu_bclk); |
400 | } | 507 | } else if (rtd->cpu_dai->dai_runtime.flags == SND_SOC_DAI_BFS_RCW) { |
508 | cpu_bclk = params_rate(params) * params_channels(params) * | ||
509 | snd_pcm_format_physical_width(rtd->cpu_dai->dai_runtime.pcmfmt) * | ||
510 | SND_SOC_FSBW_REAL(rtd->cpu_dai->dai_runtime.bfs); | ||
511 | dbg("asoc: cpu fs %d mclk %d bfs mult rcw %d bclk %d\n", | ||
512 | rtd->cpu_dai->dai_runtime.fs, mclk, | ||
513 | SND_SOC_FSBW_REAL(rtd->cpu_dai->dai_runtime.bfs), cpu_bclk); | ||
514 | } else | ||
515 | cpu_bclk = 0; | ||
401 | 516 | ||
402 | /* | 517 | /* |
403 | * Check we have matching bitclocks. If we don't then it means the | 518 | * Check we have matching bitclocks. If we don't then it means the |
@@ -405,7 +520,7 @@ found: | |||
405 | * machine sysclock function) is wrong compared with the supported DAI | 520 | * machine sysclock function) is wrong compared with the supported DAI |
406 | * modes for the codec or cpu DAI. | 521 | * modes for the codec or cpu DAI. |
407 | */ | 522 | */ |
408 | if (cpu_bclk != codec_bclk){ | 523 | if (cpu_bclk != codec_bclk && cpu_bclk){ |
409 | printk(KERN_ERR | 524 | printk(KERN_ERR |
410 | "asoc: codec and cpu bitclocks differ, audio may be wrong speed\n" | 525 | "asoc: codec and cpu bitclocks differ, audio may be wrong speed\n" |
411 | ); | 526 | ); |
@@ -723,14 +838,18 @@ static int soc_pcm_prepare(struct snd_pcm_substream *substream) | |||
723 | mutex_lock(&pcm_mutex); | 838 | mutex_lock(&pcm_mutex); |
724 | if (platform->pcm_ops->prepare) { | 839 | if (platform->pcm_ops->prepare) { |
725 | ret = platform->pcm_ops->prepare(substream); | 840 | ret = platform->pcm_ops->prepare(substream); |
726 | if (ret < 0) | 841 | if (ret < 0) { |
842 | printk(KERN_ERR "asoc: platform prepare error\n"); | ||
727 | goto out; | 843 | goto out; |
844 | } | ||
728 | } | 845 | } |
729 | 846 | ||
730 | if (rtd->codec_dai->ops.prepare) { | 847 | if (rtd->codec_dai->ops.prepare) { |
731 | ret = rtd->codec_dai->ops.prepare(substream); | 848 | ret = rtd->codec_dai->ops.prepare(substream); |
732 | if (ret < 0) | 849 | if (ret < 0) { |
850 | printk(KERN_ERR "asoc: codec DAI prepare error\n"); | ||
733 | goto out; | 851 | goto out; |
852 | } | ||
734 | } | 853 | } |
735 | 854 | ||
736 | if (rtd->cpu_dai->ops.prepare) | 855 | if (rtd->cpu_dai->ops.prepare) |