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