1 /*
2 * motu-stream.c - a part of driver for MOTU FireWire series
3 *
4 * Copyright (c) 2015-2017 Takashi Sakamoto <o-takashi@sakamocchi.jp>
5 *
6 * Licensed under the terms of the GNU General Public License, version 2.
7 */
8
9 #include "motu.h"
10
11 #define CALLBACK_TIMEOUT 200
12
13 #define ISOC_COMM_CONTROL_OFFSET 0x0b00
14 #define ISOC_COMM_CONTROL_MASK 0xffff0000
15 #define CHANGE_RX_ISOC_COMM_STATE 0x80000000
16 #define RX_ISOC_COMM_IS_ACTIVATED 0x40000000
17 #define RX_ISOC_COMM_CHANNEL_MASK 0x3f000000
18 #define RX_ISOC_COMM_CHANNEL_SHIFT 24
19 #define CHANGE_TX_ISOC_COMM_STATE 0x00800000
20 #define TX_ISOC_COMM_IS_ACTIVATED 0x00400000
21 #define TX_ISOC_COMM_CHANNEL_MASK 0x003f0000
22 #define TX_ISOC_COMM_CHANNEL_SHIFT 16
23
24 #define PACKET_FORMAT_OFFSET 0x0b10
25 #define TX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS 0x00000080
26 #define RX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS 0x00000040
27 #define TX_PACKET_TRANSMISSION_SPEED_MASK 0x0000000f
28
start_both_streams(struct snd_motu * motu,unsigned int rate)29 static int start_both_streams(struct snd_motu *motu, unsigned int rate)
30 {
31 unsigned int midi_ports = 0;
32 __be32 reg;
33 u32 data;
34 int err;
35
36 if ((motu->spec->flags & SND_MOTU_SPEC_RX_MIDI_2ND_Q) ||
37 (motu->spec->flags & SND_MOTU_SPEC_RX_MIDI_3RD_Q))
38 midi_ports = 1;
39
40 /* Set packet formation to our packet streaming engine. */
41 err = amdtp_motu_set_parameters(&motu->rx_stream, rate, midi_ports,
42 &motu->rx_packet_formats);
43 if (err < 0)
44 return err;
45
46 if ((motu->spec->flags & SND_MOTU_SPEC_TX_MIDI_2ND_Q) ||
47 (motu->spec->flags & SND_MOTU_SPEC_TX_MIDI_3RD_Q))
48 midi_ports = 1;
49 else
50 midi_ports = 0;
51
52 err = amdtp_motu_set_parameters(&motu->tx_stream, rate, midi_ports,
53 &motu->tx_packet_formats);
54 if (err < 0)
55 return err;
56
57 /* Get isochronous resources on the bus. */
58 err = fw_iso_resources_allocate(&motu->rx_resources,
59 amdtp_stream_get_max_payload(&motu->rx_stream),
60 fw_parent_device(motu->unit)->max_speed);
61 if (err < 0)
62 return err;
63
64 err = fw_iso_resources_allocate(&motu->tx_resources,
65 amdtp_stream_get_max_payload(&motu->tx_stream),
66 fw_parent_device(motu->unit)->max_speed);
67 if (err < 0)
68 return err;
69
70 /* Configure the unit to start isochronous communication. */
71 err = snd_motu_transaction_read(motu, ISOC_COMM_CONTROL_OFFSET, ®,
72 sizeof(reg));
73 if (err < 0)
74 return err;
75 data = be32_to_cpu(reg) & ~ISOC_COMM_CONTROL_MASK;
76
77 data |= CHANGE_RX_ISOC_COMM_STATE | RX_ISOC_COMM_IS_ACTIVATED |
78 (motu->rx_resources.channel << RX_ISOC_COMM_CHANNEL_SHIFT) |
79 CHANGE_TX_ISOC_COMM_STATE | TX_ISOC_COMM_IS_ACTIVATED |
80 (motu->tx_resources.channel << TX_ISOC_COMM_CHANNEL_SHIFT);
81
82 reg = cpu_to_be32(data);
83 return snd_motu_transaction_write(motu, ISOC_COMM_CONTROL_OFFSET, ®,
84 sizeof(reg));
85 }
86
stop_both_streams(struct snd_motu * motu)87 static void stop_both_streams(struct snd_motu *motu)
88 {
89 __be32 reg;
90 u32 data;
91 int err;
92
93 err = motu->spec->protocol->switch_fetching_mode(motu, false);
94 if (err < 0)
95 return;
96
97 err = snd_motu_transaction_read(motu, ISOC_COMM_CONTROL_OFFSET, ®,
98 sizeof(reg));
99 if (err < 0)
100 return;
101 data = be32_to_cpu(reg);
102
103 data &= ~(RX_ISOC_COMM_IS_ACTIVATED | TX_ISOC_COMM_IS_ACTIVATED);
104 data |= CHANGE_RX_ISOC_COMM_STATE | CHANGE_TX_ISOC_COMM_STATE;
105
106 reg = cpu_to_be32(data);
107 snd_motu_transaction_write(motu, ISOC_COMM_CONTROL_OFFSET, ®,
108 sizeof(reg));
109
110 fw_iso_resources_free(&motu->tx_resources);
111 fw_iso_resources_free(&motu->rx_resources);
112 }
113
start_isoc_ctx(struct snd_motu * motu,struct amdtp_stream * stream)114 static int start_isoc_ctx(struct snd_motu *motu, struct amdtp_stream *stream)
115 {
116 struct fw_iso_resources *resources;
117 int err;
118
119 if (stream == &motu->rx_stream)
120 resources = &motu->rx_resources;
121 else
122 resources = &motu->tx_resources;
123
124 err = amdtp_stream_start(stream, resources->channel,
125 fw_parent_device(motu->unit)->max_speed);
126 if (err < 0)
127 return err;
128
129 if (!amdtp_stream_wait_callback(stream, CALLBACK_TIMEOUT)) {
130 amdtp_stream_stop(stream);
131 fw_iso_resources_free(resources);
132 return -ETIMEDOUT;
133 }
134
135 return 0;
136 }
137
stop_isoc_ctx(struct snd_motu * motu,struct amdtp_stream * stream)138 static void stop_isoc_ctx(struct snd_motu *motu, struct amdtp_stream *stream)
139 {
140 struct fw_iso_resources *resources;
141
142 if (stream == &motu->rx_stream)
143 resources = &motu->rx_resources;
144 else
145 resources = &motu->tx_resources;
146
147 amdtp_stream_stop(stream);
148 fw_iso_resources_free(resources);
149 }
150
snd_motu_stream_cache_packet_formats(struct snd_motu * motu)151 int snd_motu_stream_cache_packet_formats(struct snd_motu *motu)
152 {
153 int err;
154
155 err = motu->spec->protocol->cache_packet_formats(motu);
156 if (err < 0)
157 return err;
158
159 if (motu->spec->flags & SND_MOTU_SPEC_TX_MIDI_2ND_Q) {
160 motu->tx_packet_formats.midi_flag_offset = 4;
161 motu->tx_packet_formats.midi_byte_offset = 6;
162 } else if (motu->spec->flags & SND_MOTU_SPEC_TX_MIDI_3RD_Q) {
163 motu->tx_packet_formats.midi_flag_offset = 8;
164 motu->tx_packet_formats.midi_byte_offset = 7;
165 }
166
167 if (motu->spec->flags & SND_MOTU_SPEC_RX_MIDI_2ND_Q) {
168 motu->rx_packet_formats.midi_flag_offset = 4;
169 motu->rx_packet_formats.midi_byte_offset = 6;
170 } else if (motu->spec->flags & SND_MOTU_SPEC_RX_MIDI_3RD_Q) {
171 motu->rx_packet_formats.midi_flag_offset = 8;
172 motu->rx_packet_formats.midi_byte_offset = 7;
173 }
174
175 return 0;
176 }
177
ensure_packet_formats(struct snd_motu * motu)178 static int ensure_packet_formats(struct snd_motu *motu)
179 {
180 __be32 reg;
181 u32 data;
182 int err;
183
184 err = snd_motu_transaction_read(motu, PACKET_FORMAT_OFFSET, ®,
185 sizeof(reg));
186 if (err < 0)
187 return err;
188 data = be32_to_cpu(reg);
189
190 data &= ~(TX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS |
191 RX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS|
192 TX_PACKET_TRANSMISSION_SPEED_MASK);
193 if (motu->tx_packet_formats.differed_part_pcm_chunks[0] == 0)
194 data |= TX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS;
195 if (motu->rx_packet_formats.differed_part_pcm_chunks[0] == 0)
196 data |= RX_PACKET_EXCLUDE_DIFFERED_DATA_CHUNKS;
197 data |= fw_parent_device(motu->unit)->max_speed;
198
199 reg = cpu_to_be32(data);
200 return snd_motu_transaction_write(motu, PACKET_FORMAT_OFFSET, ®,
201 sizeof(reg));
202 }
203
snd_motu_stream_start_duplex(struct snd_motu * motu,unsigned int rate)204 int snd_motu_stream_start_duplex(struct snd_motu *motu, unsigned int rate)
205 {
206 const struct snd_motu_protocol *protocol = motu->spec->protocol;
207 unsigned int curr_rate;
208 int err = 0;
209
210 if (motu->capture_substreams == 0 && motu->playback_substreams == 0)
211 return 0;
212
213 /* Some packet queueing errors. */
214 if (amdtp_streaming_error(&motu->rx_stream) ||
215 amdtp_streaming_error(&motu->tx_stream)) {
216 amdtp_stream_stop(&motu->rx_stream);
217 amdtp_stream_stop(&motu->tx_stream);
218 stop_both_streams(motu);
219 }
220
221 err = snd_motu_stream_cache_packet_formats(motu);
222 if (err < 0)
223 return err;
224
225 /* Stop stream if rate is different. */
226 err = protocol->get_clock_rate(motu, &curr_rate);
227 if (err < 0) {
228 dev_err(&motu->unit->device,
229 "fail to get sampling rate: %d\n", err);
230 return err;
231 }
232 if (rate == 0)
233 rate = curr_rate;
234 if (rate != curr_rate) {
235 amdtp_stream_stop(&motu->rx_stream);
236 amdtp_stream_stop(&motu->tx_stream);
237 stop_both_streams(motu);
238 }
239
240 if (!amdtp_stream_running(&motu->rx_stream)) {
241 err = protocol->set_clock_rate(motu, rate);
242 if (err < 0) {
243 dev_err(&motu->unit->device,
244 "fail to set sampling rate: %d\n", err);
245 return err;
246 }
247
248 err = ensure_packet_formats(motu);
249 if (err < 0)
250 return err;
251
252 err = start_both_streams(motu, rate);
253 if (err < 0) {
254 dev_err(&motu->unit->device,
255 "fail to start isochronous comm: %d\n", err);
256 goto stop_streams;
257 }
258
259 err = start_isoc_ctx(motu, &motu->rx_stream);
260 if (err < 0) {
261 dev_err(&motu->unit->device,
262 "fail to start IT context: %d\n", err);
263 goto stop_streams;
264 }
265
266 err = protocol->switch_fetching_mode(motu, true);
267 if (err < 0) {
268 dev_err(&motu->unit->device,
269 "fail to enable frame fetching: %d\n", err);
270 goto stop_streams;
271 }
272 }
273
274 if (!amdtp_stream_running(&motu->tx_stream) &&
275 motu->capture_substreams > 0) {
276 err = start_isoc_ctx(motu, &motu->tx_stream);
277 if (err < 0) {
278 dev_err(&motu->unit->device,
279 "fail to start IR context: %d", err);
280 amdtp_stream_stop(&motu->rx_stream);
281 goto stop_streams;
282 }
283 }
284
285 return 0;
286
287 stop_streams:
288 stop_both_streams(motu);
289 return err;
290 }
291
snd_motu_stream_stop_duplex(struct snd_motu * motu)292 void snd_motu_stream_stop_duplex(struct snd_motu *motu)
293 {
294 if (motu->capture_substreams == 0) {
295 if (amdtp_stream_running(&motu->tx_stream))
296 stop_isoc_ctx(motu, &motu->tx_stream);
297
298 if (motu->playback_substreams == 0) {
299 if (amdtp_stream_running(&motu->rx_stream))
300 stop_isoc_ctx(motu, &motu->rx_stream);
301 stop_both_streams(motu);
302 }
303 }
304 }
305
init_stream(struct snd_motu * motu,enum amdtp_stream_direction dir)306 static int init_stream(struct snd_motu *motu, enum amdtp_stream_direction dir)
307 {
308 int err;
309 struct amdtp_stream *stream;
310 struct fw_iso_resources *resources;
311
312 if (dir == AMDTP_IN_STREAM) {
313 stream = &motu->tx_stream;
314 resources = &motu->tx_resources;
315 } else {
316 stream = &motu->rx_stream;
317 resources = &motu->rx_resources;
318 }
319
320 err = fw_iso_resources_init(resources, motu->unit);
321 if (err < 0)
322 return err;
323
324 err = amdtp_motu_init(stream, motu->unit, dir, motu->spec->protocol);
325 if (err < 0) {
326 amdtp_stream_destroy(stream);
327 fw_iso_resources_destroy(resources);
328 }
329
330 return err;
331 }
332
destroy_stream(struct snd_motu * motu,enum amdtp_stream_direction dir)333 static void destroy_stream(struct snd_motu *motu,
334 enum amdtp_stream_direction dir)
335 {
336 struct amdtp_stream *stream;
337 struct fw_iso_resources *resources;
338
339 if (dir == AMDTP_IN_STREAM) {
340 stream = &motu->tx_stream;
341 resources = &motu->tx_resources;
342 } else {
343 stream = &motu->rx_stream;
344 resources = &motu->rx_resources;
345 }
346
347 amdtp_stream_destroy(stream);
348 fw_iso_resources_free(resources);
349 }
350
snd_motu_stream_init_duplex(struct snd_motu * motu)351 int snd_motu_stream_init_duplex(struct snd_motu *motu)
352 {
353 int err;
354
355 err = init_stream(motu, AMDTP_IN_STREAM);
356 if (err < 0)
357 return err;
358
359 err = init_stream(motu, AMDTP_OUT_STREAM);
360 if (err < 0)
361 destroy_stream(motu, AMDTP_IN_STREAM);
362
363 return err;
364 }
365
366 /*
367 * This function should be called before starting streams or after stopping
368 * streams.
369 */
snd_motu_stream_destroy_duplex(struct snd_motu * motu)370 void snd_motu_stream_destroy_duplex(struct snd_motu *motu)
371 {
372 destroy_stream(motu, AMDTP_IN_STREAM);
373 destroy_stream(motu, AMDTP_OUT_STREAM);
374
375 motu->playback_substreams = 0;
376 motu->capture_substreams = 0;
377 }
378
motu_lock_changed(struct snd_motu * motu)379 static void motu_lock_changed(struct snd_motu *motu)
380 {
381 motu->dev_lock_changed = true;
382 wake_up(&motu->hwdep_wait);
383 }
384
snd_motu_stream_lock_try(struct snd_motu * motu)385 int snd_motu_stream_lock_try(struct snd_motu *motu)
386 {
387 int err;
388
389 spin_lock_irq(&motu->lock);
390
391 if (motu->dev_lock_count < 0) {
392 err = -EBUSY;
393 goto out;
394 }
395
396 if (motu->dev_lock_count++ == 0)
397 motu_lock_changed(motu);
398 err = 0;
399 out:
400 spin_unlock_irq(&motu->lock);
401 return err;
402 }
403
snd_motu_stream_lock_release(struct snd_motu * motu)404 void snd_motu_stream_lock_release(struct snd_motu *motu)
405 {
406 spin_lock_irq(&motu->lock);
407
408 if (WARN_ON(motu->dev_lock_count <= 0))
409 goto out;
410
411 if (--motu->dev_lock_count == 0)
412 motu_lock_changed(motu);
413 out:
414 spin_unlock_irq(&motu->lock);
415 }
416