Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 | /* SPDX-License-Identifier: GPL-2.0 */ #ifndef SOUND_FIREWIRE_AMDTP_H_INCLUDED #define SOUND_FIREWIRE_AMDTP_H_INCLUDED #include <linux/err.h> #include <linux/interrupt.h> #include <linux/mutex.h> #include <linux/sched.h> #include <sound/asound.h> #include "packets-buffer.h" /** * enum cip_flags - describes details of the streaming protocol * @CIP_NONBLOCKING: In non-blocking mode, each packet contains * sample_rate/8000 samples, with rounding up or down to adjust * for clock skew and left-over fractional samples. This should * be used if supported by the device. * @CIP_BLOCKING: In blocking mode, each packet contains either zero or * SYT_INTERVAL samples, with these two types alternating so that * the overall sample rate comes out right. * @CIP_EMPTY_WITH_TAG0: Only for in-stream. Empty in-packets have TAG0. * @CIP_DBC_IS_END_EVENT: The value of dbc in an packet corresponds to the end * of event in the packet. Out of IEC 61883. * @CIP_WRONG_DBS: Only for in-stream. The value of dbs is wrong in in-packets. * The value of data_block_quadlets is used instead of reported value. * @CIP_SKIP_DBC_ZERO_CHECK: Only for in-stream. Packets with zero in dbc is * skipped for detecting discontinuity. * @CIP_EMPTY_HAS_WRONG_DBC: Only for in-stream. The value of dbc in empty * packet is wrong but the others are correct. * @CIP_JUMBO_PAYLOAD: Only for in-stream. The number of data blocks in an * packet is larger than IEC 61883-6 defines. Current implementation * allows 5 times as large as IEC 61883-6 defines. * @CIP_HEADER_WITHOUT_EOH: Only for in-stream. CIP Header doesn't include * valid EOH. * @CIP_NO_HEADERS: a lack of headers in packets * @CIP_UNALIGHED_DBC: Only for in-stream. The value of dbc is not alighed to * the value of current SYT_INTERVAL; e.g. initial value is not zero. * @CIP_UNAWARE_SYT: For outgoing packet, the value in SYT field of CIP is 0xffff. * For incoming packet, the value in SYT field of CIP is not handled. */ enum cip_flags { CIP_NONBLOCKING = 0x00, CIP_BLOCKING = 0x01, CIP_EMPTY_WITH_TAG0 = 0x02, CIP_DBC_IS_END_EVENT = 0x04, CIP_WRONG_DBS = 0x08, CIP_SKIP_DBC_ZERO_CHECK = 0x10, CIP_EMPTY_HAS_WRONG_DBC = 0x20, CIP_JUMBO_PAYLOAD = 0x40, CIP_HEADER_WITHOUT_EOH = 0x80, CIP_NO_HEADER = 0x100, CIP_UNALIGHED_DBC = 0x200, CIP_UNAWARE_SYT = 0x400, }; /** * enum cip_sfc - supported Sampling Frequency Codes (SFCs) * @CIP_SFC_32000: 32,000 data blocks * @CIP_SFC_44100: 44,100 data blocks * @CIP_SFC_48000: 48,000 data blocks * @CIP_SFC_88200: 88,200 data blocks * @CIP_SFC_96000: 96,000 data blocks * @CIP_SFC_176400: 176,400 data blocks * @CIP_SFC_192000: 192,000 data blocks * @CIP_SFC_COUNT: the number of supported SFCs * * These values are used to show nominal Sampling Frequency Code in * Format Dependent Field (FDF) of AMDTP packet header. In IEC 61883-6:2002, * this code means the number of events per second. Actually the code * represents the number of data blocks transferred per second in an AMDTP * stream. * * In IEC 61883-6:2005, some extensions were added to support more types of * data such as 'One Bit LInear Audio', therefore the meaning of SFC became * different depending on the types. * * Currently our implementation is compatible with IEC 61883-6:2002. */ enum cip_sfc { CIP_SFC_32000 = 0, CIP_SFC_44100 = 1, CIP_SFC_48000 = 2, CIP_SFC_88200 = 3, CIP_SFC_96000 = 4, CIP_SFC_176400 = 5, CIP_SFC_192000 = 6, CIP_SFC_COUNT }; struct fw_unit; struct fw_iso_context; struct snd_pcm_substream; struct snd_pcm_runtime; enum amdtp_stream_direction { AMDTP_OUT_STREAM = 0, AMDTP_IN_STREAM }; struct pkt_desc { u32 cycle; u32 syt; unsigned int data_blocks; unsigned int data_block_counter; __be32 *ctx_payload; }; struct amdtp_stream; typedef unsigned int (*amdtp_stream_process_ctx_payloads_t)( struct amdtp_stream *s, const struct pkt_desc *desc, unsigned int packets, struct snd_pcm_substream *pcm); struct amdtp_domain; struct amdtp_stream { struct fw_unit *unit; // The combination of cip_flags enumeration-constants. unsigned int flags; enum amdtp_stream_direction direction; struct mutex mutex; /* For packet processing. */ struct fw_iso_context *context; struct iso_packets_buffer buffer; unsigned int queue_size; int packet_index; struct pkt_desc *pkt_descs; int tag; union { struct { unsigned int ctx_header_size; // limit for payload of iso packet. unsigned int max_ctx_payload_length; // For quirks of CIP headers. // Fixed interval of dbc between previos/current // packets. unsigned int dbc_interval; // The device starts multiplexing events to the packet. bool event_starts; struct { struct seq_desc *descs; unsigned int size; unsigned int tail; } cache; } tx; struct { // To generate CIP header. unsigned int fdf; // To generate constant hardware IRQ. unsigned int event_count; // To calculate CIP data blocks and tstamp. struct { struct seq_desc *descs; unsigned int size; unsigned int tail; unsigned int head; } seq; unsigned int data_block_state; unsigned int syt_offset_state; unsigned int last_syt_offset; struct amdtp_stream *replay_target; unsigned int cache_head; } rx; } ctx_data; /* For CIP headers. */ unsigned int source_node_id_field; unsigned int data_block_quadlets; unsigned int data_block_counter; unsigned int sph; unsigned int fmt; // Internal flags. unsigned int transfer_delay; enum cip_sfc sfc; unsigned int syt_interval; /* For a PCM substream processing. */ struct snd_pcm_substream *pcm; snd_pcm_uframes_t pcm_buffer_pointer; unsigned int pcm_period_pointer; // To start processing content of packets at the same cycle in several contexts for // each direction. bool ready_processing; wait_queue_head_t ready_wait; unsigned int next_cycle; /* For backends to process data blocks. */ void *protocol; amdtp_stream_process_ctx_payloads_t process_ctx_payloads; // For domain. int channel; int speed; struct list_head list; struct amdtp_domain *domain; }; int amdtp_stream_init(struct amdtp_stream *s, struct fw_unit *unit, enum amdtp_stream_direction dir, unsigned int flags, unsigned int fmt, amdtp_stream_process_ctx_payloads_t process_ctx_payloads, unsigned int protocol_size); void amdtp_stream_destroy(struct amdtp_stream *s); int amdtp_stream_set_parameters(struct amdtp_stream *s, unsigned int rate, unsigned int data_block_quadlets); unsigned int amdtp_stream_get_max_payload(struct amdtp_stream *s); void amdtp_stream_update(struct amdtp_stream *s); int amdtp_stream_add_pcm_hw_constraints(struct amdtp_stream *s, struct snd_pcm_runtime *runtime); void amdtp_stream_pcm_prepare(struct amdtp_stream *s); void amdtp_stream_pcm_abort(struct amdtp_stream *s); extern const unsigned int amdtp_syt_intervals[CIP_SFC_COUNT]; extern const unsigned int amdtp_rate_table[CIP_SFC_COUNT]; /** * amdtp_stream_running - check stream is running or not * @s: the AMDTP stream * * If this function returns true, the stream is running. */ static inline bool amdtp_stream_running(struct amdtp_stream *s) { return !IS_ERR(s->context); } /** * amdtp_streaming_error - check for streaming error * @s: the AMDTP stream * * If this function returns true, the stream's packet queue has stopped due to * an asynchronous error. */ static inline bool amdtp_streaming_error(struct amdtp_stream *s) { return s->packet_index < 0; } /** * amdtp_stream_pcm_running - check PCM substream is running or not * @s: the AMDTP stream * * If this function returns true, PCM substream in the AMDTP stream is running. */ static inline bool amdtp_stream_pcm_running(struct amdtp_stream *s) { return !!s->pcm; } /** * amdtp_stream_pcm_trigger - start/stop playback from a PCM device * @s: the AMDTP stream * @pcm: the PCM device to be started, or %NULL to stop the current device * * Call this function on a running isochronous stream to enable the actual * transmission of PCM data. This function should be called from the PCM * device's .trigger callback. */ static inline void amdtp_stream_pcm_trigger(struct amdtp_stream *s, struct snd_pcm_substream *pcm) { WRITE_ONCE(s->pcm, pcm); } static inline bool cip_sfc_is_base_44100(enum cip_sfc sfc) { return sfc & 1; } struct seq_desc { unsigned int syt_offset; unsigned int data_blocks; }; struct amdtp_domain { struct list_head streams; unsigned int events_per_period; unsigned int events_per_buffer; struct amdtp_stream *irq_target; struct { unsigned int tx_init_skip; unsigned int tx_start; unsigned int rx_start; } processing_cycle; struct { bool enable:1; bool on_the_fly:1; } replay; }; int amdtp_domain_init(struct amdtp_domain *d); void amdtp_domain_destroy(struct amdtp_domain *d); int amdtp_domain_add_stream(struct amdtp_domain *d, struct amdtp_stream *s, int channel, int speed); int amdtp_domain_start(struct amdtp_domain *d, unsigned int tx_init_skip_cycles, bool replay_seq, bool replay_on_the_fly); void amdtp_domain_stop(struct amdtp_domain *d); static inline int amdtp_domain_set_events_per_period(struct amdtp_domain *d, unsigned int events_per_period, unsigned int events_per_buffer) { d->events_per_period = events_per_period; d->events_per_buffer = events_per_buffer; return 0; } unsigned long amdtp_domain_stream_pcm_pointer(struct amdtp_domain *d, struct amdtp_stream *s); int amdtp_domain_stream_pcm_ack(struct amdtp_domain *d, struct amdtp_stream *s); /** * amdtp_domain_wait_ready - sleep till being ready to process packets or timeout * @d: the AMDTP domain * @timeout_ms: msec till timeout * * If this function return false, the AMDTP domain should be stopped. */ static inline bool amdtp_domain_wait_ready(struct amdtp_domain *d, unsigned int timeout_ms) { struct amdtp_stream *s; list_for_each_entry(s, &d->streams, list) { unsigned int j = msecs_to_jiffies(timeout_ms); if (wait_event_interruptible_timeout(s->ready_wait, s->ready_processing, j) <= 0) return false; } return true; } #endif |