| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * digi00x-stream.c - a part of driver for Digidesign Digi 002/003 family |
| * |
| * Copyright (c) 2014-2015 Takashi Sakamoto |
| */ |
| |
| #include "digi00x.h" |
| |
| #define CALLBACK_TIMEOUT 500 |
| |
| const unsigned int snd_dg00x_stream_rates[SND_DG00X_RATE_COUNT] = { |
| [SND_DG00X_RATE_44100] = 44100, |
| [SND_DG00X_RATE_48000] = 48000, |
| [SND_DG00X_RATE_88200] = 88200, |
| [SND_DG00X_RATE_96000] = 96000, |
| }; |
| |
| /* Multi Bit Linear Audio data channels for each sampling transfer frequency. */ |
| const unsigned int |
| snd_dg00x_stream_pcm_channels[SND_DG00X_RATE_COUNT] = { |
| /* Analog/ADAT/SPDIF */ |
| [SND_DG00X_RATE_44100] = (8 + 8 + 2), |
| [SND_DG00X_RATE_48000] = (8 + 8 + 2), |
| /* Analog/SPDIF */ |
| [SND_DG00X_RATE_88200] = (8 + 2), |
| [SND_DG00X_RATE_96000] = (8 + 2), |
| }; |
| |
| int snd_dg00x_stream_get_local_rate(struct snd_dg00x *dg00x, unsigned int *rate) |
| { |
| u32 data; |
| __be32 reg; |
| int err; |
| |
| err = snd_fw_transaction(dg00x->unit, TCODE_READ_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_LOCAL_RATE, |
| ®, sizeof(reg), 0); |
| if (err < 0) |
| return err; |
| |
| data = be32_to_cpu(reg) & 0x0f; |
| if (data < ARRAY_SIZE(snd_dg00x_stream_rates)) |
| *rate = snd_dg00x_stream_rates[data]; |
| else |
| err = -EIO; |
| |
| return err; |
| } |
| |
| int snd_dg00x_stream_set_local_rate(struct snd_dg00x *dg00x, unsigned int rate) |
| { |
| __be32 reg; |
| unsigned int i; |
| |
| for (i = 0; i < ARRAY_SIZE(snd_dg00x_stream_rates); i++) { |
| if (rate == snd_dg00x_stream_rates[i]) |
| break; |
| } |
| if (i == ARRAY_SIZE(snd_dg00x_stream_rates)) |
| return -EINVAL; |
| |
| reg = cpu_to_be32(i); |
| return snd_fw_transaction(dg00x->unit, TCODE_WRITE_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_LOCAL_RATE, |
| ®, sizeof(reg), 0); |
| } |
| |
| int snd_dg00x_stream_get_clock(struct snd_dg00x *dg00x, |
| enum snd_dg00x_clock *clock) |
| { |
| __be32 reg; |
| int err; |
| |
| err = snd_fw_transaction(dg00x->unit, TCODE_READ_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_CLOCK_SOURCE, |
| ®, sizeof(reg), 0); |
| if (err < 0) |
| return err; |
| |
| *clock = be32_to_cpu(reg) & 0x0f; |
| if (*clock >= SND_DG00X_CLOCK_COUNT) |
| err = -EIO; |
| |
| return err; |
| } |
| |
| int snd_dg00x_stream_check_external_clock(struct snd_dg00x *dg00x, bool *detect) |
| { |
| __be32 reg; |
| int err; |
| |
| err = snd_fw_transaction(dg00x->unit, TCODE_READ_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_DETECT_EXTERNAL, |
| ®, sizeof(reg), 0); |
| if (err >= 0) |
| *detect = be32_to_cpu(reg) > 0; |
| |
| return err; |
| } |
| |
| int snd_dg00x_stream_get_external_rate(struct snd_dg00x *dg00x, |
| unsigned int *rate) |
| { |
| u32 data; |
| __be32 reg; |
| int err; |
| |
| err = snd_fw_transaction(dg00x->unit, TCODE_READ_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_EXTERNAL_RATE, |
| ®, sizeof(reg), 0); |
| if (err < 0) |
| return err; |
| |
| data = be32_to_cpu(reg) & 0x0f; |
| if (data < ARRAY_SIZE(snd_dg00x_stream_rates)) |
| *rate = snd_dg00x_stream_rates[data]; |
| /* This means desync. */ |
| else |
| err = -EBUSY; |
| |
| return err; |
| } |
| |
| static void finish_session(struct snd_dg00x *dg00x) |
| { |
| __be32 data = cpu_to_be32(0x00000003); |
| |
| snd_fw_transaction(dg00x->unit, TCODE_WRITE_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_STREAMING_SET, |
| &data, sizeof(data), 0); |
| } |
| |
| static int begin_session(struct snd_dg00x *dg00x) |
| { |
| __be32 data; |
| u32 curr; |
| int err; |
| |
| err = snd_fw_transaction(dg00x->unit, TCODE_READ_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_STREAMING_STATE, |
| &data, sizeof(data), 0); |
| if (err < 0) |
| goto error; |
| curr = be32_to_cpu(data); |
| |
| if (curr == 0) |
| curr = 2; |
| |
| curr--; |
| while (curr > 0) { |
| data = cpu_to_be32(curr); |
| err = snd_fw_transaction(dg00x->unit, |
| TCODE_WRITE_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + |
| DG00X_OFFSET_STREAMING_SET, |
| &data, sizeof(data), 0); |
| if (err < 0) |
| goto error; |
| |
| msleep(20); |
| curr--; |
| } |
| |
| return 0; |
| error: |
| finish_session(dg00x); |
| return err; |
| } |
| |
| static void release_resources(struct snd_dg00x *dg00x) |
| { |
| __be32 data = 0; |
| |
| /* Unregister isochronous channels for both direction. */ |
| snd_fw_transaction(dg00x->unit, TCODE_WRITE_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_ISOC_CHANNELS, |
| &data, sizeof(data), 0); |
| |
| /* Release isochronous resources. */ |
| fw_iso_resources_free(&dg00x->tx_resources); |
| fw_iso_resources_free(&dg00x->rx_resources); |
| } |
| |
| static int keep_resources(struct snd_dg00x *dg00x, unsigned int rate) |
| { |
| unsigned int i; |
| __be32 data; |
| int err; |
| |
| /* Check sampling rate. */ |
| for (i = 0; i < SND_DG00X_RATE_COUNT; i++) { |
| if (snd_dg00x_stream_rates[i] == rate) |
| break; |
| } |
| if (i == SND_DG00X_RATE_COUNT) |
| return -EINVAL; |
| |
| /* Keep resources for out-stream. */ |
| err = amdtp_dot_set_parameters(&dg00x->rx_stream, rate, |
| snd_dg00x_stream_pcm_channels[i]); |
| if (err < 0) |
| return err; |
| err = fw_iso_resources_allocate(&dg00x->rx_resources, |
| amdtp_stream_get_max_payload(&dg00x->rx_stream), |
| fw_parent_device(dg00x->unit)->max_speed); |
| if (err < 0) |
| return err; |
| |
| /* Keep resources for in-stream. */ |
| err = amdtp_dot_set_parameters(&dg00x->tx_stream, rate, |
| snd_dg00x_stream_pcm_channels[i]); |
| if (err < 0) |
| return err; |
| err = fw_iso_resources_allocate(&dg00x->tx_resources, |
| amdtp_stream_get_max_payload(&dg00x->tx_stream), |
| fw_parent_device(dg00x->unit)->max_speed); |
| if (err < 0) |
| goto error; |
| |
| /* Register isochronous channels for both direction. */ |
| data = cpu_to_be32((dg00x->tx_resources.channel << 16) | |
| dg00x->rx_resources.channel); |
| err = snd_fw_transaction(dg00x->unit, TCODE_WRITE_QUADLET_REQUEST, |
| DG00X_ADDR_BASE + DG00X_OFFSET_ISOC_CHANNELS, |
| &data, sizeof(data), 0); |
| if (err < 0) |
| goto error; |
| |
| return 0; |
| error: |
| release_resources(dg00x); |
| return err; |
| } |
| |
| int snd_dg00x_stream_init_duplex(struct snd_dg00x *dg00x) |
| { |
| int err; |
| |
| /* For out-stream. */ |
| err = fw_iso_resources_init(&dg00x->rx_resources, dg00x->unit); |
| if (err < 0) |
| goto error; |
| err = amdtp_dot_init(&dg00x->rx_stream, dg00x->unit, AMDTP_OUT_STREAM); |
| if (err < 0) |
| goto error; |
| |
| /* For in-stream. */ |
| err = fw_iso_resources_init(&dg00x->tx_resources, dg00x->unit); |
| if (err < 0) |
| goto error; |
| err = amdtp_dot_init(&dg00x->tx_stream, dg00x->unit, AMDTP_IN_STREAM); |
| if (err < 0) |
| goto error; |
| |
| return 0; |
| error: |
| snd_dg00x_stream_destroy_duplex(dg00x); |
| return err; |
| } |
| |
| /* |
| * This function should be called before starting streams or after stopping |
| * streams. |
| */ |
| void snd_dg00x_stream_destroy_duplex(struct snd_dg00x *dg00x) |
| { |
| amdtp_stream_destroy(&dg00x->rx_stream); |
| fw_iso_resources_destroy(&dg00x->rx_resources); |
| |
| amdtp_stream_destroy(&dg00x->tx_stream); |
| fw_iso_resources_destroy(&dg00x->tx_resources); |
| } |
| |
| int snd_dg00x_stream_start_duplex(struct snd_dg00x *dg00x, unsigned int rate) |
| { |
| unsigned int curr_rate; |
| int err = 0; |
| |
| if (dg00x->substreams_counter == 0) |
| goto end; |
| |
| /* Check current sampling rate. */ |
| err = snd_dg00x_stream_get_local_rate(dg00x, &curr_rate); |
| if (err < 0) |
| goto error; |
| if (rate == 0) |
| rate = curr_rate; |
| if (curr_rate != rate || |
| amdtp_streaming_error(&dg00x->tx_stream) || |
| amdtp_streaming_error(&dg00x->rx_stream)) { |
| finish_session(dg00x); |
| |
| amdtp_stream_stop(&dg00x->tx_stream); |
| amdtp_stream_stop(&dg00x->rx_stream); |
| release_resources(dg00x); |
| } |
| |
| /* |
| * No packets are transmitted without receiving packets, reagardless of |
| * which source of clock is used. |
| */ |
| if (!amdtp_stream_running(&dg00x->rx_stream)) { |
| err = snd_dg00x_stream_set_local_rate(dg00x, rate); |
| if (err < 0) |
| goto error; |
| |
| err = keep_resources(dg00x, rate); |
| if (err < 0) |
| goto error; |
| |
| err = begin_session(dg00x); |
| if (err < 0) |
| goto error; |
| |
| err = amdtp_stream_start(&dg00x->rx_stream, |
| dg00x->rx_resources.channel, |
| fw_parent_device(dg00x->unit)->max_speed); |
| if (err < 0) |
| goto error; |
| |
| if (!amdtp_stream_wait_callback(&dg00x->rx_stream, |
| CALLBACK_TIMEOUT)) { |
| err = -ETIMEDOUT; |
| goto error; |
| } |
| } |
| |
| /* |
| * The value of SYT field in transmitted packets is always 0x0000. Thus, |
| * duplex streams with timestamp synchronization cannot be built. |
| */ |
| if (!amdtp_stream_running(&dg00x->tx_stream)) { |
| err = amdtp_stream_start(&dg00x->tx_stream, |
| dg00x->tx_resources.channel, |
| fw_parent_device(dg00x->unit)->max_speed); |
| if (err < 0) |
| goto error; |
| |
| if (!amdtp_stream_wait_callback(&dg00x->tx_stream, |
| CALLBACK_TIMEOUT)) { |
| err = -ETIMEDOUT; |
| goto error; |
| } |
| } |
| end: |
| return err; |
| error: |
| finish_session(dg00x); |
| |
| amdtp_stream_stop(&dg00x->tx_stream); |
| amdtp_stream_stop(&dg00x->rx_stream); |
| release_resources(dg00x); |
| |
| return err; |
| } |
| |
| void snd_dg00x_stream_stop_duplex(struct snd_dg00x *dg00x) |
| { |
| if (dg00x->substreams_counter > 0) |
| return; |
| |
| amdtp_stream_stop(&dg00x->tx_stream); |
| amdtp_stream_stop(&dg00x->rx_stream); |
| finish_session(dg00x); |
| release_resources(dg00x); |
| |
| /* |
| * Just after finishing the session, the device may lost transmitting |
| * functionality for a short time. |
| */ |
| msleep(50); |
| } |
| |
| void snd_dg00x_stream_update_duplex(struct snd_dg00x *dg00x) |
| { |
| fw_iso_resources_update(&dg00x->tx_resources); |
| fw_iso_resources_update(&dg00x->rx_resources); |
| |
| amdtp_stream_update(&dg00x->tx_stream); |
| amdtp_stream_update(&dg00x->rx_stream); |
| } |
| |
| void snd_dg00x_stream_lock_changed(struct snd_dg00x *dg00x) |
| { |
| dg00x->dev_lock_changed = true; |
| wake_up(&dg00x->hwdep_wait); |
| } |
| |
| int snd_dg00x_stream_lock_try(struct snd_dg00x *dg00x) |
| { |
| int err; |
| |
| spin_lock_irq(&dg00x->lock); |
| |
| /* user land lock this */ |
| if (dg00x->dev_lock_count < 0) { |
| err = -EBUSY; |
| goto end; |
| } |
| |
| /* this is the first time */ |
| if (dg00x->dev_lock_count++ == 0) |
| snd_dg00x_stream_lock_changed(dg00x); |
| err = 0; |
| end: |
| spin_unlock_irq(&dg00x->lock); |
| return err; |
| } |
| |
| void snd_dg00x_stream_lock_release(struct snd_dg00x *dg00x) |
| { |
| spin_lock_irq(&dg00x->lock); |
| |
| if (WARN_ON(dg00x->dev_lock_count <= 0)) |
| goto end; |
| if (--dg00x->dev_lock_count == 0) |
| snd_dg00x_stream_lock_changed(dg00x); |
| end: |
| spin_unlock_irq(&dg00x->lock); |
| } |