Commit 436b5abe authored by Takashi Sakamoto's avatar Takashi Sakamoto Committed by Takashi Iwai

ALSA: dice: handle whole available isochronous streams

This commit enables ALSA dice driver to handle whole available streams.

In Dice, certain registers represent the number of available streams at
current sampling transfer frequency for both directions. The parameters
of each stream are represented in a block of register. This block is
aligned sequentially. These streams start simultaneously by writing
enable bit to a register.

This commit operates these registers when starting/stopping streams.
Signed-off-by: default avatarTakashi Sakamoto <o-takashi@sakamocchi.jp>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent 8ae25b76
...@@ -65,85 +65,84 @@ static int ensure_phase_lock(struct snd_dice *dice) ...@@ -65,85 +65,84 @@ static int ensure_phase_lock(struct snd_dice *dice)
return 0; return 0;
} }
static void release_resources(struct snd_dice *dice, static int get_register_params(struct snd_dice *dice, unsigned int params[4])
struct fw_iso_resources *resources)
{ {
__be32 channel; __be32 reg[2];
/* Reset channel number */
channel = cpu_to_be32((u32)-1);
if (resources == &dice->tx_resources[0])
snd_dice_transaction_write_tx(dice, TX_ISOCHRONOUS,
&channel, sizeof(channel));
else
snd_dice_transaction_write_rx(dice, RX_ISOCHRONOUS,
&channel, sizeof(channel));
fw_iso_resources_free(resources);
}
static int keep_resources(struct snd_dice *dice,
struct fw_iso_resources *resources,
unsigned int max_payload_bytes)
{
__be32 channel;
int err; int err;
err = fw_iso_resources_allocate(resources, max_payload_bytes, err = snd_dice_transaction_read_tx(dice, TX_NUMBER, reg, sizeof(reg));
fw_parent_device(dice->unit)->max_speed);
if (err < 0) if (err < 0)
goto end; return err;
params[0] = min_t(unsigned int, be32_to_cpu(reg[0]), MAX_STREAMS);
params[1] = be32_to_cpu(reg[1]) * 4;
/* Set channel number */ err = snd_dice_transaction_read_rx(dice, RX_NUMBER, reg, sizeof(reg));
channel = cpu_to_be32(resources->channel);
if (resources == &dice->tx_resources[0])
err = snd_dice_transaction_write_tx(dice, TX_ISOCHRONOUS,
&channel, sizeof(channel));
else
err = snd_dice_transaction_write_rx(dice, RX_ISOCHRONOUS,
&channel, sizeof(channel));
if (err < 0) if (err < 0)
release_resources(dice, resources); return err;
end: params[2] = min_t(unsigned int, be32_to_cpu(reg[0]), MAX_STREAMS);
return err; params[3] = be32_to_cpu(reg[1]) * 4;
return 0;
} }
static void stop_stream(struct snd_dice *dice, struct amdtp_stream *stream) static void release_resources(struct snd_dice *dice)
{ {
amdtp_stream_pcm_abort(stream); unsigned int i;
amdtp_stream_stop(stream);
for (i = 0; i < MAX_STREAMS; i++) {
if (amdtp_stream_running(&dice->tx_stream[i])) {
amdtp_stream_pcm_abort(&dice->tx_stream[i]);
amdtp_stream_stop(&dice->tx_stream[i]);
}
if (amdtp_stream_running(&dice->rx_stream[i])) {
amdtp_stream_pcm_abort(&dice->rx_stream[i]);
amdtp_stream_stop(&dice->rx_stream[i]);
}
if (stream == &dice->tx_stream[0]) fw_iso_resources_free(&dice->tx_resources[i]);
release_resources(dice, &dice->tx_resources[0]); fw_iso_resources_free(&dice->rx_resources[i]);
else }
release_resources(dice, &dice->rx_resources[0]);
} }
static int start_stream(struct snd_dice *dice, struct amdtp_stream *stream, static void stop_streams(struct snd_dice *dice, enum amdtp_stream_direction dir,
unsigned int rate) unsigned int count, unsigned int size)
{ {
__be32 reg;
unsigned int i;
for (i = 0; i < count; i++) {
reg = cpu_to_be32((u32)-1);
if (dir == AMDTP_IN_STREAM) {
snd_dice_transaction_write_tx(dice,
size * i + TX_ISOCHRONOUS,
&reg, sizeof(reg));
} else {
snd_dice_transaction_write_rx(dice,
size * i + RX_ISOCHRONOUS,
&reg, sizeof(reg));
}
}
}
static int keep_resources(struct snd_dice *dice,
enum amdtp_stream_direction dir, unsigned int index,
unsigned int rate, unsigned int pcm_chs,
unsigned int midi_ports)
{
struct amdtp_stream *stream;
struct fw_iso_resources *resources; struct fw_iso_resources *resources;
__be32 reg[2];
unsigned int i, pcm_chs, midi_ports;
bool double_pcm_frames; bool double_pcm_frames;
unsigned int i;
int err; int err;
if (stream == &dice->tx_stream[0]) { if (dir == AMDTP_IN_STREAM) {
resources = &dice->tx_resources[0]; stream = &dice->tx_stream[index];
err = snd_dice_transaction_read_tx(dice, TX_NUMBER_AUDIO, resources = &dice->tx_resources[index];
reg, sizeof(reg));
} else { } else {
resources = &dice->rx_resources[0]; stream = &dice->rx_stream[index];
err = snd_dice_transaction_read_rx(dice, RX_NUMBER_AUDIO, resources = &dice->rx_resources[index];
reg, sizeof(reg));
} }
if (err < 0)
goto end;
pcm_chs = be32_to_cpu(reg[0]);
midi_ports = be32_to_cpu(reg[1]);
/* /*
* At 176.4/192.0 kHz, Dice has a quirk to transfer two PCM frames in * At 176.4/192.0 kHz, Dice has a quirk to transfer two PCM frames in
* one data block of AMDTP packet. Thus sampling transfer frequency is * one data block of AMDTP packet. Thus sampling transfer frequency is
...@@ -163,7 +162,7 @@ static int start_stream(struct snd_dice *dice, struct amdtp_stream *stream, ...@@ -163,7 +162,7 @@ static int start_stream(struct snd_dice *dice, struct amdtp_stream *stream,
err = amdtp_am824_set_parameters(stream, rate, pcm_chs, midi_ports, err = amdtp_am824_set_parameters(stream, rate, pcm_chs, midi_ports,
double_pcm_frames); double_pcm_frames);
if (err < 0) if (err < 0)
goto end; return err;
if (double_pcm_frames) { if (double_pcm_frames) {
pcm_chs /= 2; pcm_chs /= 2;
...@@ -175,122 +174,208 @@ static int start_stream(struct snd_dice *dice, struct amdtp_stream *stream, ...@@ -175,122 +174,208 @@ static int start_stream(struct snd_dice *dice, struct amdtp_stream *stream,
} }
} }
err = keep_resources(dice, resources, return fw_iso_resources_allocate(resources,
amdtp_stream_get_max_payload(stream)); amdtp_stream_get_max_payload(stream),
if (err < 0) { fw_parent_device(dice->unit)->max_speed);
dev_err(&dice->unit->device, }
"fail to keep isochronous resources\n");
goto end; static int start_streams(struct snd_dice *dice, enum amdtp_stream_direction dir,
unsigned int rate, unsigned int count,
unsigned int size)
{
__be32 reg[2];
unsigned int i, pcm_chs, midi_ports;
struct amdtp_stream *streams;
struct fw_iso_resources *resources;
int err = 0;
if (dir == AMDTP_IN_STREAM) {
streams = dice->tx_stream;
resources = dice->tx_resources;
} else {
streams = dice->rx_stream;
resources = dice->rx_resources;
}
for (i = 0; i < count; i++) {
if (dir == AMDTP_IN_STREAM) {
err = snd_dice_transaction_read_tx(dice,
size * i + TX_NUMBER_AUDIO,
reg, sizeof(reg));
} else {
err = snd_dice_transaction_read_rx(dice,
size * i + RX_NUMBER_AUDIO,
reg, sizeof(reg));
}
if (err < 0)
return err;
pcm_chs = be32_to_cpu(reg[0]);
midi_ports = be32_to_cpu(reg[1]);
err = keep_resources(dice, dir, i, rate, pcm_chs, midi_ports);
if (err < 0)
return err;
reg[0] = cpu_to_be32(resources[i].channel);
if (dir == AMDTP_IN_STREAM) {
err = snd_dice_transaction_write_tx(dice,
size * i + TX_ISOCHRONOUS,
reg, sizeof(reg[0]));
} else {
err = snd_dice_transaction_write_rx(dice,
size * i + RX_ISOCHRONOUS,
reg, sizeof(reg[0]));
}
if (err < 0)
return err;
err = amdtp_stream_start(&streams[i], resources[i].channel,
fw_parent_device(dice->unit)->max_speed);
if (err < 0)
return err;
} }
err = amdtp_stream_start(stream, resources->channel,
fw_parent_device(dice->unit)->max_speed);
if (err < 0)
release_resources(dice, resources);
end:
return err; return err;
} }
/*
* MEMO: After this function, there're two states of streams:
* - None streams are running.
* - All streams are running.
*/
int snd_dice_stream_start_duplex(struct snd_dice *dice, unsigned int rate) int snd_dice_stream_start_duplex(struct snd_dice *dice, unsigned int rate)
{ {
struct amdtp_stream *master, *slave;
unsigned int curr_rate; unsigned int curr_rate;
int err = 0; unsigned int i;
unsigned int reg_params[4];
bool need_to_start;
int err;
if (dice->substreams_counter == 0) if (dice->substreams_counter == 0)
goto end; return -EIO;
master = &dice->rx_stream[0];
slave = &dice->tx_stream[0];
/* Some packet queueing errors. */ err = get_register_params(dice, reg_params);
if (amdtp_streaming_error(master) || amdtp_streaming_error(slave)) if (err < 0)
stop_stream(dice, master); return err;
/* Stop stream if rate is different. */
err = snd_dice_transaction_get_rate(dice, &curr_rate); err = snd_dice_transaction_get_rate(dice, &curr_rate);
if (err < 0) { if (err < 0) {
dev_err(&dice->unit->device, dev_err(&dice->unit->device,
"fail to get sampling rate\n"); "fail to get sampling rate\n");
goto end; return err;
} }
if (rate == 0) if (rate == 0)
rate = curr_rate; rate = curr_rate;
if (rate != curr_rate) { if (rate != curr_rate)
err = -EINVAL; return -EINVAL;
goto end;
/* Judge to need to restart streams. */
for (i = 0; i < MAX_STREAMS; i++) {
if (i < reg_params[0]) {
if (amdtp_streaming_error(&dice->tx_stream[i]) ||
!amdtp_stream_running(&dice->tx_stream[i]))
break;
}
if (i < reg_params[2]) {
if (amdtp_streaming_error(&dice->rx_stream[i]) ||
!amdtp_stream_running(&dice->rx_stream[i]))
break;
}
} }
need_to_start = (i < MAX_STREAMS);
if (!amdtp_stream_running(master)) { if (need_to_start) {
stop_stream(dice, slave); /* Stop transmission. */
snd_dice_transaction_clear_enable(dice); snd_dice_transaction_clear_enable(dice);
stop_streams(dice, AMDTP_IN_STREAM, reg_params[0],
reg_params[1]);
stop_streams(dice, AMDTP_OUT_STREAM, reg_params[2],
reg_params[3]);
release_resources(dice);
err = ensure_phase_lock(dice); err = ensure_phase_lock(dice);
if (err < 0) { if (err < 0) {
dev_err(&dice->unit->device, dev_err(&dice->unit->device,
"fail to ensure phase lock\n"); "fail to ensure phase lock\n");
goto end; return err;
} }
/* Start both streams. */ /* Start both streams. */
err = start_stream(dice, master, rate); err = start_streams(dice, AMDTP_IN_STREAM, rate, reg_params[0],
if (err < 0) { reg_params[1]);
dev_err(&dice->unit->device, if (err < 0)
"fail to start AMDTP master stream\n"); goto error;
goto end; err = start_streams(dice, AMDTP_OUT_STREAM, rate, reg_params[2],
} reg_params[3]);
err = start_stream(dice, slave, rate); if (err < 0)
if (err < 0) { goto error;
dev_err(&dice->unit->device,
"fail to start AMDTP slave stream\n");
stop_stream(dice, master);
goto end;
}
err = snd_dice_transaction_set_enable(dice); err = snd_dice_transaction_set_enable(dice);
if (err < 0) { if (err < 0) {
dev_err(&dice->unit->device, dev_err(&dice->unit->device,
"fail to enable interface\n"); "fail to enable interface\n");
stop_stream(dice, master); goto error;
stop_stream(dice, slave);
goto end;
} }
/* Wait first callbacks */ for (i = 0; i < MAX_STREAMS; i++) {
if (!amdtp_stream_wait_callback(master, CALLBACK_TIMEOUT) || if ((i < reg_params[0] &&
!amdtp_stream_wait_callback(slave, CALLBACK_TIMEOUT)) { !amdtp_stream_wait_callback(&dice->tx_stream[i],
snd_dice_transaction_clear_enable(dice); CALLBACK_TIMEOUT)) ||
stop_stream(dice, master); (i < reg_params[2] &&
stop_stream(dice, slave); !amdtp_stream_wait_callback(&dice->rx_stream[i],
err = -ETIMEDOUT; CALLBACK_TIMEOUT))) {
err = -ETIMEDOUT;
goto error;
}
} }
} }
end:
return err;
error:
snd_dice_transaction_clear_enable(dice);
stop_streams(dice, AMDTP_IN_STREAM, reg_params[0], reg_params[1]);
stop_streams(dice, AMDTP_OUT_STREAM, reg_params[2], reg_params[3]);
release_resources(dice);
return err; return err;
} }
/*
* MEMO: After this function, there're two states of streams:
* - None streams are running.
* - All streams are running.
*/
void snd_dice_stream_stop_duplex(struct snd_dice *dice) void snd_dice_stream_stop_duplex(struct snd_dice *dice)
{ {
unsigned int reg_params[4];
if (dice->substreams_counter > 0) if (dice->substreams_counter > 0)
return; return;
snd_dice_transaction_clear_enable(dice); snd_dice_transaction_clear_enable(dice);
stop_stream(dice, &dice->tx_stream[0]); if (get_register_params(dice, reg_params) == 0) {
stop_stream(dice, &dice->rx_stream[0]); stop_streams(dice, AMDTP_IN_STREAM, reg_params[0],
reg_params[1]);
stop_streams(dice, AMDTP_OUT_STREAM, reg_params[2],
reg_params[3]);
}
release_resources(dice);
} }
static int init_stream(struct snd_dice *dice, struct amdtp_stream *stream) static int init_stream(struct snd_dice *dice, enum amdtp_stream_direction dir,
unsigned int index)
{ {
int err; struct amdtp_stream *stream;
struct fw_iso_resources *resources; struct fw_iso_resources *resources;
enum amdtp_stream_direction dir; int err;
if (stream == &dice->tx_stream[0]) { if (dir == AMDTP_IN_STREAM) {
resources = &dice->tx_resources[0]; stream = &dice->tx_stream[index];
dir = AMDTP_IN_STREAM; resources = &dice->tx_resources[index];
} else { } else {
resources = &dice->rx_resources[0]; stream = &dice->rx_stream[index];
dir = AMDTP_OUT_STREAM; resources = &dice->rx_resources[index];
} }
err = fw_iso_resources_init(resources, dice->unit); err = fw_iso_resources_init(resources, dice->unit);
...@@ -311,14 +396,20 @@ static int init_stream(struct snd_dice *dice, struct amdtp_stream *stream) ...@@ -311,14 +396,20 @@ static int init_stream(struct snd_dice *dice, struct amdtp_stream *stream)
* This function should be called before starting streams or after stopping * This function should be called before starting streams or after stopping
* streams. * streams.
*/ */
static void destroy_stream(struct snd_dice *dice, struct amdtp_stream *stream) static void destroy_stream(struct snd_dice *dice,
enum amdtp_stream_direction dir,
unsigned int index)
{ {
struct amdtp_stream *stream;
struct fw_iso_resources *resources; struct fw_iso_resources *resources;
if (stream == &dice->tx_stream[0]) if (dir == AMDTP_IN_STREAM) {
resources = &dice->tx_resources[0]; stream = &dice->tx_stream[index];
else resources = &dice->tx_resources[index];
resources = &dice->rx_resources[0]; } else {
stream = &dice->rx_stream[index];
resources = &dice->rx_resources[index];
}
amdtp_stream_destroy(stream); amdtp_stream_destroy(stream);
fw_iso_resources_destroy(resources); fw_iso_resources_destroy(resources);
...@@ -326,33 +417,53 @@ static void destroy_stream(struct snd_dice *dice, struct amdtp_stream *stream) ...@@ -326,33 +417,53 @@ static void destroy_stream(struct snd_dice *dice, struct amdtp_stream *stream)
int snd_dice_stream_init_duplex(struct snd_dice *dice) int snd_dice_stream_init_duplex(struct snd_dice *dice)
{ {
int err; int i, err;
dice->substreams_counter = 0; for (i = 0; i < MAX_STREAMS; i++) {
err = init_stream(dice, AMDTP_IN_STREAM, i);
err = init_stream(dice, &dice->tx_stream[0]); if (err < 0) {
if (err < 0) for (; i >= 0; i--)
goto end; destroy_stream(dice, AMDTP_OUT_STREAM, i);
goto end;
}
}
err = init_stream(dice, &dice->rx_stream[0]); for (i = 0; i < MAX_STREAMS; i++) {
if (err < 0) err = init_stream(dice, AMDTP_OUT_STREAM, i);
destroy_stream(dice, &dice->tx_stream[0]); if (err < 0) {
for (; i >= 0; i--)
destroy_stream(dice, AMDTP_OUT_STREAM, i);
for (i = 0; i < MAX_STREAMS; i++)
destroy_stream(dice, AMDTP_IN_STREAM, i);
break;
}
}
end: end:
return err; return err;
} }
void snd_dice_stream_destroy_duplex(struct snd_dice *dice) void snd_dice_stream_destroy_duplex(struct snd_dice *dice)
{ {
unsigned int reg_params[4];
snd_dice_transaction_clear_enable(dice); snd_dice_transaction_clear_enable(dice);
destroy_stream(dice, &dice->tx_stream[0]); if (get_register_params(dice, reg_params) == 0) {
destroy_stream(dice, &dice->rx_stream[0]); stop_streams(dice, AMDTP_IN_STREAM, reg_params[0],
reg_params[1]);
stop_streams(dice, AMDTP_OUT_STREAM, reg_params[2],
reg_params[3]);
}
release_resources(dice);
dice->substreams_counter = 0; dice->substreams_counter = 0;
} }
void snd_dice_stream_update_duplex(struct snd_dice *dice) void snd_dice_stream_update_duplex(struct snd_dice *dice)
{ {
unsigned int reg_params[4];
/* /*
* On a bus reset, the DICE firmware disables streaming and then goes * On a bus reset, the DICE firmware disables streaming and then goes
* off contemplating its own navel for hundreds of milliseconds before * off contemplating its own navel for hundreds of milliseconds before
...@@ -363,11 +474,12 @@ void snd_dice_stream_update_duplex(struct snd_dice *dice) ...@@ -363,11 +474,12 @@ void snd_dice_stream_update_duplex(struct snd_dice *dice)
*/ */
dice->global_enabled = false; dice->global_enabled = false;
stop_stream(dice, &dice->rx_stream[0]); if (get_register_params(dice, reg_params) == 0) {
stop_stream(dice, &dice->tx_stream[0]); stop_streams(dice, AMDTP_IN_STREAM, reg_params[0],
reg_params[1]);
fw_iso_resources_update(&dice->rx_resources[0]); stop_streams(dice, AMDTP_OUT_STREAM, reg_params[2],
fw_iso_resources_update(&dice->tx_resources[0]); reg_params[3]);
}
} }
static void dice_lock_changed(struct snd_dice *dice) static void dice_lock_changed(struct snd_dice *dice)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment