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, &reg,
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, &reg,
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, &reg,
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, &reg,
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, &reg,
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, &reg,
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