blob: ec4db3a514fce3c4f16c2b8c38a71ad2e9a77d4e [file] [log] [blame]
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +09001/*
2 * dice_stream.c - a part of driver for DICE based devices
3 *
4 * Copyright (c) Clemens Ladisch <clemens@ladisch.de>
5 * Copyright (c) 2014 Takashi Sakamoto <o-takashi@sakamocchi.jp>
6 *
7 * Licensed under the terms of the GNU General Public License, version 2.
8 */
9
10#include "dice.h"
11
Takashi Sakamoto288a8d02014-12-09 00:10:35 +090012#define CALLBACK_TIMEOUT 200
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +090013#define NOTIFICATION_TIMEOUT_MS (2 * MSEC_PER_SEC)
Takashi Sakamoto288a8d02014-12-09 00:10:35 +090014
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +090015struct reg_params {
16 unsigned int count;
17 unsigned int size;
18};
19
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +090020const unsigned int snd_dice_rates[SND_DICE_RATES_COUNT] = {
21 /* mode 0 */
22 [0] = 32000,
23 [1] = 44100,
24 [2] = 48000,
25 /* mode 1 */
26 [3] = 88200,
27 [4] = 96000,
28 /* mode 2 */
29 [5] = 176400,
30 [6] = 192000,
31};
32
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +090033/*
34 * This operation has an effect to synchronize GLOBAL_STATUS/GLOBAL_SAMPLE_RATE
35 * to GLOBAL_STATUS. Especially, just after powering on, these are different.
36 */
37static int ensure_phase_lock(struct snd_dice *dice)
38{
Takashi Sakamotofbeac842016-02-11 20:18:38 +090039 __be32 reg, nominal;
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +090040 int err;
41
42 err = snd_dice_transaction_read_global(dice, GLOBAL_CLOCK_SELECT,
43 &reg, sizeof(reg));
44 if (err < 0)
45 return err;
46
47 if (completion_done(&dice->clock_accepted))
48 reinit_completion(&dice->clock_accepted);
49
50 err = snd_dice_transaction_write_global(dice, GLOBAL_CLOCK_SELECT,
51 &reg, sizeof(reg));
52 if (err < 0)
53 return err;
54
55 if (wait_for_completion_timeout(&dice->clock_accepted,
Takashi Sakamotofbeac842016-02-11 20:18:38 +090056 msecs_to_jiffies(NOTIFICATION_TIMEOUT_MS)) == 0) {
57 /*
58 * Old versions of Dice firmware transfer no notification when
59 * the same clock status as current one is set. In this case,
60 * just check current clock status.
61 */
62 err = snd_dice_transaction_read_global(dice, GLOBAL_STATUS,
63 &nominal, sizeof(nominal));
64 if (err < 0)
65 return err;
66 if (!(be32_to_cpu(nominal) & STATUS_SOURCE_LOCKED))
67 return -ETIMEDOUT;
68 }
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +090069
70 return 0;
71}
72
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +090073static int get_register_params(struct snd_dice *dice,
74 struct reg_params *tx_params,
75 struct reg_params *rx_params)
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +090076{
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +090077 __be32 reg[2];
78 int err;
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +090079
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +090080 err = snd_dice_transaction_read_tx(dice, TX_NUMBER, reg, sizeof(reg));
81 if (err < 0)
82 return err;
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +090083 tx_params->count =
84 min_t(unsigned int, be32_to_cpu(reg[0]), MAX_STREAMS);
85 tx_params->size = be32_to_cpu(reg[1]) * 4;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +090086
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +090087 err = snd_dice_transaction_read_rx(dice, RX_NUMBER, reg, sizeof(reg));
88 if (err < 0)
89 return err;
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +090090 rx_params->count =
91 min_t(unsigned int, be32_to_cpu(reg[0]), MAX_STREAMS);
92 rx_params->size = be32_to_cpu(reg[1]) * 4;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +090093
94 return 0;
95}
96
97static void release_resources(struct snd_dice *dice)
98{
99 unsigned int i;
100
101 for (i = 0; i < MAX_STREAMS; i++) {
102 if (amdtp_stream_running(&dice->tx_stream[i])) {
103 amdtp_stream_pcm_abort(&dice->tx_stream[i]);
104 amdtp_stream_stop(&dice->tx_stream[i]);
105 }
106 if (amdtp_stream_running(&dice->rx_stream[i])) {
107 amdtp_stream_pcm_abort(&dice->rx_stream[i]);
108 amdtp_stream_stop(&dice->rx_stream[i]);
109 }
110
111 fw_iso_resources_free(&dice->tx_resources[i]);
112 fw_iso_resources_free(&dice->rx_resources[i]);
113 }
114}
115
116static void stop_streams(struct snd_dice *dice, enum amdtp_stream_direction dir,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900117 struct reg_params *params)
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900118{
119 __be32 reg;
120 unsigned int i;
121
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900122 for (i = 0; i < params->count; i++) {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900123 reg = cpu_to_be32((u32)-1);
124 if (dir == AMDTP_IN_STREAM) {
125 snd_dice_transaction_write_tx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900126 params->size * i + TX_ISOCHRONOUS,
127 &reg, sizeof(reg));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900128 } else {
129 snd_dice_transaction_write_rx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900130 params->size * i + RX_ISOCHRONOUS,
131 &reg, sizeof(reg));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900132 }
133 }
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900134}
135
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900136static int keep_resources(struct snd_dice *dice,
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900137 enum amdtp_stream_direction dir, unsigned int index,
138 unsigned int rate, unsigned int pcm_chs,
139 unsigned int midi_ports)
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900140{
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900141 struct amdtp_stream *stream;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900142 struct fw_iso_resources *resources;
Takashi Sakamoto27ec83b2015-09-19 11:21:50 +0900143 bool double_pcm_frames;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900144 unsigned int i;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900145 int err;
146
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900147 if (dir == AMDTP_IN_STREAM) {
148 stream = &dice->tx_stream[index];
149 resources = &dice->tx_resources[index];
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900150 } else {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900151 stream = &dice->rx_stream[index];
152 resources = &dice->rx_resources[index];
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900153 }
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900154
155 /*
156 * At 176.4/192.0 kHz, Dice has a quirk to transfer two PCM frames in
157 * one data block of AMDTP packet. Thus sampling transfer frequency is
158 * a half of PCM sampling frequency, i.e. PCM frames at 192.0 kHz are
159 * transferred on AMDTP packets at 96 kHz. Two successive samples of a
160 * channel are stored consecutively in the packet. This quirk is called
161 * as 'Dual Wire'.
162 * For this quirk, blocking mode is required and PCM buffer size should
163 * be aligned to SYT_INTERVAL.
164 */
Takashi Sakamoto6f688262016-02-08 22:54:19 +0900165 double_pcm_frames = rate > 96000;
Takashi Sakamoto27ec83b2015-09-19 11:21:50 +0900166 if (double_pcm_frames) {
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900167 rate /= 2;
168 pcm_chs *= 2;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900169 }
170
Takashi Sakamoto51c29fd2015-09-19 11:21:56 +0900171 err = amdtp_am824_set_parameters(stream, rate, pcm_chs, midi_ports,
172 double_pcm_frames);
Takashi Sakamoto547e6312015-09-19 11:21:49 +0900173 if (err < 0)
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900174 return err;
Takashi Sakamoto547e6312015-09-19 11:21:49 +0900175
Takashi Sakamoto27ec83b2015-09-19 11:21:50 +0900176 if (double_pcm_frames) {
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900177 pcm_chs /= 2;
178
179 for (i = 0; i < pcm_chs; i++) {
Takashi Sakamotof65be912015-09-19 11:21:58 +0900180 amdtp_am824_set_pcm_position(stream, i, i * 2);
181 amdtp_am824_set_pcm_position(stream, i + pcm_chs,
182 i * 2 + 1);
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900183 }
184 }
185
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900186 return fw_iso_resources_allocate(resources,
187 amdtp_stream_get_max_payload(stream),
188 fw_parent_device(dice->unit)->max_speed);
189}
190
191static int start_streams(struct snd_dice *dice, enum amdtp_stream_direction dir,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900192 unsigned int rate, struct reg_params *params)
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900193{
194 __be32 reg[2];
195 unsigned int i, pcm_chs, midi_ports;
196 struct amdtp_stream *streams;
197 struct fw_iso_resources *resources;
198 int err = 0;
199
200 if (dir == AMDTP_IN_STREAM) {
201 streams = dice->tx_stream;
202 resources = dice->tx_resources;
203 } else {
204 streams = dice->rx_stream;
205 resources = dice->rx_resources;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900206 }
207
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900208 for (i = 0; i < params->count; i++) {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900209 if (dir == AMDTP_IN_STREAM) {
210 err = snd_dice_transaction_read_tx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900211 params->size * i + TX_NUMBER_AUDIO,
212 reg, sizeof(reg));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900213 } else {
214 err = snd_dice_transaction_read_rx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900215 params->size * i + RX_NUMBER_AUDIO,
216 reg, sizeof(reg));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900217 }
218 if (err < 0)
219 return err;
220 pcm_chs = be32_to_cpu(reg[0]);
221 midi_ports = be32_to_cpu(reg[1]);
222
223 err = keep_resources(dice, dir, i, rate, pcm_chs, midi_ports);
224 if (err < 0)
225 return err;
226
227 reg[0] = cpu_to_be32(resources[i].channel);
228 if (dir == AMDTP_IN_STREAM) {
229 err = snd_dice_transaction_write_tx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900230 params->size * i + TX_ISOCHRONOUS,
231 reg, sizeof(reg[0]));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900232 } else {
233 err = snd_dice_transaction_write_rx(dice,
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900234 params->size * i + RX_ISOCHRONOUS,
235 reg, sizeof(reg[0]));
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900236 }
237 if (err < 0)
238 return err;
239
240 err = amdtp_stream_start(&streams[i], resources[i].channel,
241 fw_parent_device(dice->unit)->max_speed);
242 if (err < 0)
243 return err;
244 }
245
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900246 return err;
247}
248
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900249/*
250 * MEMO: After this function, there're two states of streams:
251 * - None streams are running.
252 * - All streams are running.
253 */
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900254int snd_dice_stream_start_duplex(struct snd_dice *dice, unsigned int rate)
255{
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900256 unsigned int curr_rate;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900257 unsigned int i;
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900258 struct reg_params tx_params, rx_params;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900259 bool need_to_start;
260 int err;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900261
262 if (dice->substreams_counter == 0)
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900263 return -EIO;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900264
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900265 err = get_register_params(dice, &tx_params, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900266 if (err < 0)
267 return err;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900268
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900269 err = snd_dice_transaction_get_rate(dice, &curr_rate);
270 if (err < 0) {
271 dev_err(&dice->unit->device,
272 "fail to get sampling rate\n");
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900273 return err;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900274 }
Takashi Sakamotoa113ff82014-12-09 00:10:39 +0900275 if (rate == 0)
276 rate = curr_rate;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900277 if (rate != curr_rate)
278 return -EINVAL;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900279
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900280 /* Judge to need to restart streams. */
281 for (i = 0; i < MAX_STREAMS; i++) {
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900282 if (i < tx_params.count) {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900283 if (amdtp_streaming_error(&dice->tx_stream[i]) ||
284 !amdtp_stream_running(&dice->tx_stream[i]))
285 break;
286 }
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900287 if (i < rx_params.count) {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900288 if (amdtp_streaming_error(&dice->rx_stream[i]) ||
289 !amdtp_stream_running(&dice->rx_stream[i]))
290 break;
291 }
292 }
293 need_to_start = (i < MAX_STREAMS);
294
295 if (need_to_start) {
296 /* Stop transmission. */
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900297 snd_dice_transaction_clear_enable(dice);
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900298 stop_streams(dice, AMDTP_IN_STREAM, &tx_params);
299 stop_streams(dice, AMDTP_OUT_STREAM, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900300 release_resources(dice);
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900301
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +0900302 err = ensure_phase_lock(dice);
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900303 if (err < 0) {
304 dev_err(&dice->unit->device,
Takashi Sakamotodfabc0e2016-02-08 22:54:20 +0900305 "fail to ensure phase lock\n");
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900306 return err;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900307 }
308
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900309 /* Start both streams. */
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900310 err = start_streams(dice, AMDTP_IN_STREAM, rate, &tx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900311 if (err < 0)
312 goto error;
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900313 err = start_streams(dice, AMDTP_OUT_STREAM, rate, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900314 if (err < 0)
315 goto error;
316
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900317 err = snd_dice_transaction_set_enable(dice);
318 if (err < 0) {
319 dev_err(&dice->unit->device,
320 "fail to enable interface\n");
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900321 goto error;
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900322 }
323
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900324 for (i = 0; i < MAX_STREAMS; i++) {
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900325 if ((i < tx_params.count &&
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900326 !amdtp_stream_wait_callback(&dice->tx_stream[i],
327 CALLBACK_TIMEOUT)) ||
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900328 (i < rx_params.count &&
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900329 !amdtp_stream_wait_callback(&dice->rx_stream[i],
330 CALLBACK_TIMEOUT))) {
331 err = -ETIMEDOUT;
332 goto error;
333 }
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900334 }
335 }
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900336
337 return err;
338error:
339 snd_dice_transaction_clear_enable(dice);
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900340 stop_streams(dice, AMDTP_IN_STREAM, &tx_params);
341 stop_streams(dice, AMDTP_OUT_STREAM, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900342 release_resources(dice);
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900343 return err;
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900344}
345
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900346/*
347 * MEMO: After this function, there're two states of streams:
348 * - None streams are running.
349 * - All streams are running.
350 */
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900351void snd_dice_stream_stop_duplex(struct snd_dice *dice)
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900352{
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900353 struct reg_params tx_params, rx_params;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900354
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900355 if (dice->substreams_counter > 0)
356 return;
357
Takashi Sakamoto288a8d02014-12-09 00:10:35 +0900358 snd_dice_transaction_clear_enable(dice);
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900359
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900360 if (get_register_params(dice, &tx_params, &rx_params) == 0) {
361 stop_streams(dice, AMDTP_IN_STREAM, &tx_params);
362 stop_streams(dice, AMDTP_OUT_STREAM, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900363 }
364
365 release_resources(dice);
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900366}
367
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900368static int init_stream(struct snd_dice *dice, enum amdtp_stream_direction dir,
369 unsigned int index)
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900370{
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900371 struct amdtp_stream *stream;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900372 struct fw_iso_resources *resources;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900373 int err;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900374
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900375 if (dir == AMDTP_IN_STREAM) {
376 stream = &dice->tx_stream[index];
377 resources = &dice->tx_resources[index];
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900378 } else {
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900379 stream = &dice->rx_stream[index];
380 resources = &dice->rx_resources[index];
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900381 }
382
383 err = fw_iso_resources_init(resources, dice->unit);
384 if (err < 0)
385 goto end;
386 resources->channels_mask = 0x00000000ffffffffuLL;
387
Takashi Sakamoto59558152015-09-19 11:21:55 +0900388 err = amdtp_am824_init(stream, dice->unit, dir, CIP_BLOCKING);
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900389 if (err < 0) {
390 amdtp_stream_destroy(stream);
391 fw_iso_resources_destroy(resources);
392 }
393end:
394 return err;
395}
396
Takashi Sakamotod23c2cc2015-02-21 23:54:59 +0900397/*
398 * This function should be called before starting streams or after stopping
399 * streams.
400 */
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900401static void destroy_stream(struct snd_dice *dice,
402 enum amdtp_stream_direction dir,
403 unsigned int index)
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900404{
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900405 struct amdtp_stream *stream;
Takashi Sakamotod23c2cc2015-02-21 23:54:59 +0900406 struct fw_iso_resources *resources;
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900407
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900408 if (dir == AMDTP_IN_STREAM) {
409 stream = &dice->tx_stream[index];
410 resources = &dice->tx_resources[index];
411 } else {
412 stream = &dice->rx_stream[index];
413 resources = &dice->rx_resources[index];
414 }
Takashi Sakamotod23c2cc2015-02-21 23:54:59 +0900415
416 amdtp_stream_destroy(stream);
417 fw_iso_resources_destroy(resources);
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900418}
419
420int snd_dice_stream_init_duplex(struct snd_dice *dice)
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900421{
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900422 int i, err;
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900423
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900424 for (i = 0; i < MAX_STREAMS; i++) {
425 err = init_stream(dice, AMDTP_IN_STREAM, i);
426 if (err < 0) {
427 for (; i >= 0; i--)
428 destroy_stream(dice, AMDTP_OUT_STREAM, i);
429 goto end;
430 }
431 }
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900432
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900433 for (i = 0; i < MAX_STREAMS; i++) {
434 err = init_stream(dice, AMDTP_OUT_STREAM, i);
435 if (err < 0) {
436 for (; i >= 0; i--)
437 destroy_stream(dice, AMDTP_OUT_STREAM, i);
438 for (i = 0; i < MAX_STREAMS; i++)
439 destroy_stream(dice, AMDTP_IN_STREAM, i);
440 break;
441 }
442 }
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900443end:
444 return err;
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900445}
446
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900447void snd_dice_stream_destroy_duplex(struct snd_dice *dice)
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900448{
Takashi Sakamoto6b94fb12016-03-28 08:29:32 +0900449 unsigned int i;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900450
Takashi Sakamoto6b94fb12016-03-28 08:29:32 +0900451 for (i = 0; i < MAX_STREAMS; i++) {
452 destroy_stream(dice, AMDTP_IN_STREAM, i);
453 destroy_stream(dice, AMDTP_OUT_STREAM, i);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900454 }
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900455}
456
Takashi Sakamoto9a028432014-12-09 00:10:36 +0900457void snd_dice_stream_update_duplex(struct snd_dice *dice)
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900458{
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900459 struct reg_params tx_params, rx_params;
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900460
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900461 /*
462 * On a bus reset, the DICE firmware disables streaming and then goes
463 * off contemplating its own navel for hundreds of milliseconds before
464 * it can react to any of our attempts to reenable streaming. This
465 * means that we lose synchronization anyway, so we force our streams
466 * to stop so that the application can restart them in an orderly
467 * manner.
468 */
469 dice->global_enabled = false;
470
Takashi Sakamoto8cc1a8a2016-03-10 21:44:28 +0900471 if (get_register_params(dice, &tx_params, &rx_params) == 0) {
472 stop_streams(dice, AMDTP_IN_STREAM, &tx_params);
473 stop_streams(dice, AMDTP_OUT_STREAM, &rx_params);
Takashi Sakamoto436b5ab2016-03-07 22:35:43 +0900474 }
Takashi Sakamoto6eb6c812014-11-29 00:59:14 +0900475}
476
477static void dice_lock_changed(struct snd_dice *dice)
478{
479 dice->dev_lock_changed = true;
480 wake_up(&dice->hwdep_wait);
481}
482
483int snd_dice_stream_lock_try(struct snd_dice *dice)
484{
485 int err;
486
487 spin_lock_irq(&dice->lock);
488
489 if (dice->dev_lock_count < 0) {
490 err = -EBUSY;
491 goto out;
492 }
493
494 if (dice->dev_lock_count++ == 0)
495 dice_lock_changed(dice);
496 err = 0;
497out:
498 spin_unlock_irq(&dice->lock);
499 return err;
500}
501
502void snd_dice_stream_lock_release(struct snd_dice *dice)
503{
504 spin_lock_irq(&dice->lock);
505
506 if (WARN_ON(dice->dev_lock_count <= 0))
507 goto out;
508
509 if (--dice->dev_lock_count == 0)
510 dice_lock_changed(dice);
511out:
512 spin_unlock_irq(&dice->lock);
513}