• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * Copyright (C) 2008, Creative Technology Ltd. All Rights Reserved.
3  *
4  * This source file is released under GPL v2 license (no other versions).
5  * See the COPYING file included in the main directory of this source
6  * distribution for the license terms and conditions.
7  *
8  * @File	ctpcm.c
9  *
10  * @Brief
11  * This file contains the definition of the pcm device functions.
12  *
13  * @Author	Liu Chun
14  * @Date 	Apr 2 2008
15  *
16  */
17 
18 #include "ctpcm.h"
19 #include "cttimer.h"
20 #include <linux/slab.h>
21 #include <sound/pcm.h>
22 
23 /* Hardware descriptions for playback */
24 static const struct snd_pcm_hardware ct_pcm_playback_hw = {
25 	.info			= (SNDRV_PCM_INFO_MMAP |
26 				   SNDRV_PCM_INFO_INTERLEAVED |
27 				   SNDRV_PCM_INFO_BLOCK_TRANSFER |
28 				   SNDRV_PCM_INFO_MMAP_VALID |
29 				   SNDRV_PCM_INFO_PAUSE),
30 	.formats		= (SNDRV_PCM_FMTBIT_U8 |
31 				   SNDRV_PCM_FMTBIT_S16_LE |
32 				   SNDRV_PCM_FMTBIT_S24_3LE |
33 				   SNDRV_PCM_FMTBIT_S32_LE |
34 				   SNDRV_PCM_FMTBIT_FLOAT_LE),
35 	.rates			= (SNDRV_PCM_RATE_CONTINUOUS |
36 				   SNDRV_PCM_RATE_8000_192000),
37 	.rate_min		= 8000,
38 	.rate_max		= 192000,
39 	.channels_min		= 1,
40 	.channels_max		= 2,
41 	.buffer_bytes_max	= (128*1024),
42 	.period_bytes_min	= (64),
43 	.period_bytes_max	= (128*1024),
44 	.periods_min		= 2,
45 	.periods_max		= 1024,
46 	.fifo_size		= 0,
47 };
48 
49 static const struct snd_pcm_hardware ct_spdif_passthru_playback_hw = {
50 	.info			= (SNDRV_PCM_INFO_MMAP |
51 				   SNDRV_PCM_INFO_INTERLEAVED |
52 				   SNDRV_PCM_INFO_BLOCK_TRANSFER |
53 				   SNDRV_PCM_INFO_MMAP_VALID |
54 				   SNDRV_PCM_INFO_PAUSE),
55 	.formats		= SNDRV_PCM_FMTBIT_S16_LE,
56 	.rates			= (SNDRV_PCM_RATE_48000 |
57 				   SNDRV_PCM_RATE_44100 |
58 				   SNDRV_PCM_RATE_32000),
59 	.rate_min		= 32000,
60 	.rate_max		= 48000,
61 	.channels_min		= 2,
62 	.channels_max		= 2,
63 	.buffer_bytes_max	= (128*1024),
64 	.period_bytes_min	= (64),
65 	.period_bytes_max	= (128*1024),
66 	.periods_min		= 2,
67 	.periods_max		= 1024,
68 	.fifo_size		= 0,
69 };
70 
71 /* Hardware descriptions for capture */
72 static const struct snd_pcm_hardware ct_pcm_capture_hw = {
73 	.info			= (SNDRV_PCM_INFO_MMAP |
74 				   SNDRV_PCM_INFO_INTERLEAVED |
75 				   SNDRV_PCM_INFO_BLOCK_TRANSFER |
76 				   SNDRV_PCM_INFO_PAUSE |
77 				   SNDRV_PCM_INFO_MMAP_VALID),
78 	.formats		= (SNDRV_PCM_FMTBIT_U8 |
79 				   SNDRV_PCM_FMTBIT_S16_LE |
80 				   SNDRV_PCM_FMTBIT_S24_3LE |
81 				   SNDRV_PCM_FMTBIT_S32_LE |
82 				   SNDRV_PCM_FMTBIT_FLOAT_LE),
83 	.rates			= (SNDRV_PCM_RATE_CONTINUOUS |
84 				   SNDRV_PCM_RATE_8000_96000),
85 	.rate_min		= 8000,
86 	.rate_max		= 96000,
87 	.channels_min		= 1,
88 	.channels_max		= 2,
89 	.buffer_bytes_max	= (128*1024),
90 	.period_bytes_min	= (384),
91 	.period_bytes_max	= (64*1024),
92 	.periods_min		= 2,
93 	.periods_max		= 1024,
94 	.fifo_size		= 0,
95 };
96 
ct_atc_pcm_interrupt(struct ct_atc_pcm * atc_pcm)97 static void ct_atc_pcm_interrupt(struct ct_atc_pcm *atc_pcm)
98 {
99 	struct ct_atc_pcm *apcm = atc_pcm;
100 
101 	if (!apcm->substream)
102 		return;
103 
104 	snd_pcm_period_elapsed(apcm->substream);
105 }
106 
ct_atc_pcm_free_substream(struct snd_pcm_runtime * runtime)107 static void ct_atc_pcm_free_substream(struct snd_pcm_runtime *runtime)
108 {
109 	struct ct_atc_pcm *apcm = runtime->private_data;
110 	struct ct_atc *atc = snd_pcm_substream_chip(apcm->substream);
111 
112 	atc->pcm_release_resources(atc, apcm);
113 	ct_timer_instance_free(apcm->timer);
114 	kfree(apcm);
115 	runtime->private_data = NULL;
116 }
117 
118 /* pcm playback operations */
ct_pcm_playback_open(struct snd_pcm_substream * substream)119 static int ct_pcm_playback_open(struct snd_pcm_substream *substream)
120 {
121 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
122 	struct snd_pcm_runtime *runtime = substream->runtime;
123 	struct ct_atc_pcm *apcm;
124 	int err;
125 
126 	apcm = kzalloc(sizeof(*apcm), GFP_KERNEL);
127 	if (!apcm)
128 		return -ENOMEM;
129 
130 	apcm->substream = substream;
131 	apcm->interrupt = ct_atc_pcm_interrupt;
132 	if (IEC958 == substream->pcm->device) {
133 		runtime->hw = ct_spdif_passthru_playback_hw;
134 		atc->spdif_out_passthru(atc, 1);
135 	} else {
136 		runtime->hw = ct_pcm_playback_hw;
137 		if (FRONT == substream->pcm->device)
138 			runtime->hw.channels_max = 8;
139 	}
140 
141 	err = snd_pcm_hw_constraint_integer(runtime,
142 					    SNDRV_PCM_HW_PARAM_PERIODS);
143 	if (err < 0)
144 		goto free_pcm;
145 
146 	err = snd_pcm_hw_constraint_minmax(runtime,
147 					   SNDRV_PCM_HW_PARAM_BUFFER_BYTES,
148 					   1024, UINT_MAX);
149 	if (err < 0)
150 		goto free_pcm;
151 
152 	apcm->timer = ct_timer_instance_new(atc->timer, apcm);
153 	if (!apcm->timer) {
154 		err = -ENOMEM;
155 		goto free_pcm;
156 	}
157 	runtime->private_data = apcm;
158 	runtime->private_free = ct_atc_pcm_free_substream;
159 
160 	return 0;
161 
162 free_pcm:
163 	kfree(apcm);
164 	return err;
165 }
166 
ct_pcm_playback_close(struct snd_pcm_substream * substream)167 static int ct_pcm_playback_close(struct snd_pcm_substream *substream)
168 {
169 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
170 
171 	/* TODO: Notify mixer inactive. */
172 	if (IEC958 == substream->pcm->device)
173 		atc->spdif_out_passthru(atc, 0);
174 
175 	/* The ct_atc_pcm object will be freed by runtime->private_free */
176 
177 	return 0;
178 }
179 
ct_pcm_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * hw_params)180 static int ct_pcm_hw_params(struct snd_pcm_substream *substream,
181 				     struct snd_pcm_hw_params *hw_params)
182 {
183 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
184 	struct ct_atc_pcm *apcm = substream->runtime->private_data;
185 	int err;
186 
187 	err = snd_pcm_lib_malloc_pages(substream,
188 					params_buffer_bytes(hw_params));
189 	if (err < 0)
190 		return err;
191 	/* clear previous resources */
192 	atc->pcm_release_resources(atc, apcm);
193 	return err;
194 }
195 
ct_pcm_hw_free(struct snd_pcm_substream * substream)196 static int ct_pcm_hw_free(struct snd_pcm_substream *substream)
197 {
198 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
199 	struct ct_atc_pcm *apcm = substream->runtime->private_data;
200 
201 	/* clear previous resources */
202 	atc->pcm_release_resources(atc, apcm);
203 	/* Free snd-allocated pages */
204 	return snd_pcm_lib_free_pages(substream);
205 }
206 
207 
ct_pcm_playback_prepare(struct snd_pcm_substream * substream)208 static int ct_pcm_playback_prepare(struct snd_pcm_substream *substream)
209 {
210 	int err;
211 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
212 	struct snd_pcm_runtime *runtime = substream->runtime;
213 	struct ct_atc_pcm *apcm = runtime->private_data;
214 
215 	if (IEC958 == substream->pcm->device)
216 		err = atc->spdif_passthru_playback_prepare(atc, apcm);
217 	else
218 		err = atc->pcm_playback_prepare(atc, apcm);
219 
220 	if (err < 0) {
221 		dev_err(atc->card->dev,
222 			"Preparing pcm playback failed!!!\n");
223 		return err;
224 	}
225 
226 	return 0;
227 }
228 
229 static int
ct_pcm_playback_trigger(struct snd_pcm_substream * substream,int cmd)230 ct_pcm_playback_trigger(struct snd_pcm_substream *substream, int cmd)
231 {
232 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
233 	struct snd_pcm_runtime *runtime = substream->runtime;
234 	struct ct_atc_pcm *apcm = runtime->private_data;
235 
236 	switch (cmd) {
237 	case SNDRV_PCM_TRIGGER_START:
238 	case SNDRV_PCM_TRIGGER_RESUME:
239 	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
240 		atc->pcm_playback_start(atc, apcm);
241 		break;
242 	case SNDRV_PCM_TRIGGER_STOP:
243 	case SNDRV_PCM_TRIGGER_SUSPEND:
244 	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
245 		atc->pcm_playback_stop(atc, apcm);
246 		break;
247 	default:
248 		break;
249 	}
250 
251 	return 0;
252 }
253 
254 static snd_pcm_uframes_t
ct_pcm_playback_pointer(struct snd_pcm_substream * substream)255 ct_pcm_playback_pointer(struct snd_pcm_substream *substream)
256 {
257 	unsigned long position;
258 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
259 	struct snd_pcm_runtime *runtime = substream->runtime;
260 	struct ct_atc_pcm *apcm = runtime->private_data;
261 
262 	/* Read out playback position */
263 	position = atc->pcm_playback_position(atc, apcm);
264 	position = bytes_to_frames(runtime, position);
265 	if (position >= runtime->buffer_size)
266 		position = 0;
267 	return position;
268 }
269 
270 /* pcm capture operations */
ct_pcm_capture_open(struct snd_pcm_substream * substream)271 static int ct_pcm_capture_open(struct snd_pcm_substream *substream)
272 {
273 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
274 	struct snd_pcm_runtime *runtime = substream->runtime;
275 	struct ct_atc_pcm *apcm;
276 	int err;
277 
278 	apcm = kzalloc(sizeof(*apcm), GFP_KERNEL);
279 	if (!apcm)
280 		return -ENOMEM;
281 
282 	apcm->started = 0;
283 	apcm->substream = substream;
284 	apcm->interrupt = ct_atc_pcm_interrupt;
285 	runtime->hw = ct_pcm_capture_hw;
286 	runtime->hw.rate_max = atc->rsr * atc->msr;
287 
288 	err = snd_pcm_hw_constraint_integer(runtime,
289 					    SNDRV_PCM_HW_PARAM_PERIODS);
290 	if (err < 0)
291 		goto free_pcm;
292 
293 	err = snd_pcm_hw_constraint_minmax(runtime,
294 					   SNDRV_PCM_HW_PARAM_BUFFER_BYTES,
295 					   1024, UINT_MAX);
296 	if (err < 0)
297 		goto free_pcm;
298 
299 	apcm->timer = ct_timer_instance_new(atc->timer, apcm);
300 	if (!apcm->timer) {
301 		err = -ENOMEM;
302 		goto free_pcm;
303 	}
304 	runtime->private_data = apcm;
305 	runtime->private_free = ct_atc_pcm_free_substream;
306 
307 	return 0;
308 
309 free_pcm:
310 	kfree(apcm);
311 	return err;
312 }
313 
ct_pcm_capture_close(struct snd_pcm_substream * substream)314 static int ct_pcm_capture_close(struct snd_pcm_substream *substream)
315 {
316 	/* The ct_atc_pcm object will be freed by runtime->private_free */
317 	/* TODO: Notify mixer inactive. */
318 	return 0;
319 }
320 
ct_pcm_capture_prepare(struct snd_pcm_substream * substream)321 static int ct_pcm_capture_prepare(struct snd_pcm_substream *substream)
322 {
323 	int err;
324 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
325 	struct snd_pcm_runtime *runtime = substream->runtime;
326 	struct ct_atc_pcm *apcm = runtime->private_data;
327 
328 	err = atc->pcm_capture_prepare(atc, apcm);
329 	if (err < 0) {
330 		dev_err(atc->card->dev,
331 			"Preparing pcm capture failed!!!\n");
332 		return err;
333 	}
334 
335 	return 0;
336 }
337 
338 static int
ct_pcm_capture_trigger(struct snd_pcm_substream * substream,int cmd)339 ct_pcm_capture_trigger(struct snd_pcm_substream *substream, int cmd)
340 {
341 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
342 	struct snd_pcm_runtime *runtime = substream->runtime;
343 	struct ct_atc_pcm *apcm = runtime->private_data;
344 
345 	switch (cmd) {
346 	case SNDRV_PCM_TRIGGER_START:
347 		atc->pcm_capture_start(atc, apcm);
348 		break;
349 	case SNDRV_PCM_TRIGGER_STOP:
350 		atc->pcm_capture_stop(atc, apcm);
351 		break;
352 	default:
353 		atc->pcm_capture_stop(atc, apcm);
354 		break;
355 	}
356 
357 	return 0;
358 }
359 
360 static snd_pcm_uframes_t
ct_pcm_capture_pointer(struct snd_pcm_substream * substream)361 ct_pcm_capture_pointer(struct snd_pcm_substream *substream)
362 {
363 	unsigned long position;
364 	struct ct_atc *atc = snd_pcm_substream_chip(substream);
365 	struct snd_pcm_runtime *runtime = substream->runtime;
366 	struct ct_atc_pcm *apcm = runtime->private_data;
367 
368 	/* Read out playback position */
369 	position = atc->pcm_capture_position(atc, apcm);
370 	position = bytes_to_frames(runtime, position);
371 	if (position >= runtime->buffer_size)
372 		position = 0;
373 	return position;
374 }
375 
376 /* PCM operators for playback */
377 static const struct snd_pcm_ops ct_pcm_playback_ops = {
378 	.open	 	= ct_pcm_playback_open,
379 	.close		= ct_pcm_playback_close,
380 	.ioctl		= snd_pcm_lib_ioctl,
381 	.hw_params	= ct_pcm_hw_params,
382 	.hw_free	= ct_pcm_hw_free,
383 	.prepare	= ct_pcm_playback_prepare,
384 	.trigger	= ct_pcm_playback_trigger,
385 	.pointer	= ct_pcm_playback_pointer,
386 	.page		= snd_pcm_sgbuf_ops_page,
387 };
388 
389 /* PCM operators for capture */
390 static const struct snd_pcm_ops ct_pcm_capture_ops = {
391 	.open	 	= ct_pcm_capture_open,
392 	.close		= ct_pcm_capture_close,
393 	.ioctl		= snd_pcm_lib_ioctl,
394 	.hw_params	= ct_pcm_hw_params,
395 	.hw_free	= ct_pcm_hw_free,
396 	.prepare	= ct_pcm_capture_prepare,
397 	.trigger	= ct_pcm_capture_trigger,
398 	.pointer	= ct_pcm_capture_pointer,
399 	.page		= snd_pcm_sgbuf_ops_page,
400 };
401 
402 static const struct snd_pcm_chmap_elem surround_map[] = {
403 	{ .channels = 1,
404 	  .map = { SNDRV_CHMAP_MONO } },
405 	{ .channels = 2,
406 	  .map = { SNDRV_CHMAP_RL, SNDRV_CHMAP_RR } },
407 	{ }
408 };
409 
410 static const struct snd_pcm_chmap_elem clfe_map[] = {
411 	{ .channels = 1,
412 	  .map = { SNDRV_CHMAP_MONO } },
413 	{ .channels = 2,
414 	  .map = { SNDRV_CHMAP_FC, SNDRV_CHMAP_LFE } },
415 	{ }
416 };
417 
418 static const struct snd_pcm_chmap_elem side_map[] = {
419 	{ .channels = 1,
420 	  .map = { SNDRV_CHMAP_MONO } },
421 	{ .channels = 2,
422 	  .map = { SNDRV_CHMAP_SL, SNDRV_CHMAP_SR } },
423 	{ }
424 };
425 
426 /* Create ALSA pcm device */
ct_alsa_pcm_create(struct ct_atc * atc,enum CTALSADEVS device,const char * device_name)427 int ct_alsa_pcm_create(struct ct_atc *atc,
428 		       enum CTALSADEVS device,
429 		       const char *device_name)
430 {
431 	struct snd_pcm *pcm;
432 	const struct snd_pcm_chmap_elem *map;
433 	int chs;
434 	int err;
435 	int playback_count, capture_count;
436 
437 	playback_count = (IEC958 == device) ? 1 : 256;
438 	capture_count = (FRONT == device) ? 1 : 0;
439 	err = snd_pcm_new(atc->card, "ctxfi", device,
440 			  playback_count, capture_count, &pcm);
441 	if (err < 0) {
442 		dev_err(atc->card->dev, "snd_pcm_new failed!! Err=%d\n",
443 			err);
444 		return err;
445 	}
446 
447 	pcm->private_data = atc;
448 	pcm->info_flags = 0;
449 	pcm->dev_subclass = SNDRV_PCM_SUBCLASS_GENERIC_MIX;
450 	strlcpy(pcm->name, device_name, sizeof(pcm->name));
451 
452 	snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &ct_pcm_playback_ops);
453 
454 	if (FRONT == device)
455 		snd_pcm_set_ops(pcm,
456 				SNDRV_PCM_STREAM_CAPTURE, &ct_pcm_capture_ops);
457 
458 	snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV_SG,
459 			snd_dma_pci_data(atc->pci), 128*1024, 128*1024);
460 
461 	chs = 2;
462 	switch (device) {
463 	case FRONT:
464 		chs = 8;
465 		map = snd_pcm_std_chmaps;
466 		break;
467 	case SURROUND:
468 		map = surround_map;
469 		break;
470 	case CLFE:
471 		map = clfe_map;
472 		break;
473 	case SIDE:
474 		map = side_map;
475 		break;
476 	default:
477 		map = snd_pcm_std_chmaps;
478 		break;
479 	}
480 	err = snd_pcm_add_chmap_ctls(pcm, SNDRV_PCM_STREAM_PLAYBACK, map, chs,
481 				     0, NULL);
482 	if (err < 0)
483 		return err;
484 
485 #ifdef CONFIG_PM_SLEEP
486 	atc->pcms[device] = pcm;
487 #endif
488 
489 	return 0;
490 }
491