ALSA: line6: Rearrange PCM structure

Introduce a new line6_pcm_stream structure and group individual
fields of snd_line6_pcm struct to playback and capture groups.

This patch itself just does rename and nothing else.  More
meaningful cleanups based on these fields shuffling will follow.

Tested-by: Chris Rorvick <chris@rorvick.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
Takashi Iwai 2015-01-23 16:10:57 +01:00
parent ab5cdcbab2
commit ad0119abe2
4 changed files with 142 additions and 208 deletions

View file

@ -29,17 +29,17 @@ static int submit_audio_in_urb(struct snd_line6_pcm *line6pcm)
int ret; int ret;
struct urb *urb_in; struct urb *urb_in;
spin_lock_irqsave(&line6pcm->lock_audio_in, flags); spin_lock_irqsave(&line6pcm->in.lock, flags);
index = index =
find_first_zero_bit(&line6pcm->active_urb_in, LINE6_ISO_BUFFERS); find_first_zero_bit(&line6pcm->in.active_urbs, LINE6_ISO_BUFFERS);
if (index < 0 || index >= LINE6_ISO_BUFFERS) { if (index < 0 || index >= LINE6_ISO_BUFFERS) {
spin_unlock_irqrestore(&line6pcm->lock_audio_in, flags); spin_unlock_irqrestore(&line6pcm->in.lock, flags);
dev_err(line6pcm->line6->ifcdev, "no free URB found\n"); dev_err(line6pcm->line6->ifcdev, "no free URB found\n");
return -EINVAL; return -EINVAL;
} }
urb_in = line6pcm->urb_audio_in[index]; urb_in = line6pcm->in.urbs[index];
urb_size = 0; urb_size = 0;
for (i = 0; i < LINE6_ISO_PACKETS; ++i) { for (i = 0; i < LINE6_ISO_PACKETS; ++i) {
@ -51,7 +51,7 @@ static int submit_audio_in_urb(struct snd_line6_pcm *line6pcm)
} }
urb_in->transfer_buffer = urb_in->transfer_buffer =
line6pcm->buffer_in + line6pcm->in.buffer +
index * LINE6_ISO_PACKETS * line6pcm->max_packet_size; index * LINE6_ISO_PACKETS * line6pcm->max_packet_size;
urb_in->transfer_buffer_length = urb_size; urb_in->transfer_buffer_length = urb_size;
urb_in->context = line6pcm; urb_in->context = line6pcm;
@ -59,12 +59,12 @@ static int submit_audio_in_urb(struct snd_line6_pcm *line6pcm)
ret = usb_submit_urb(urb_in, GFP_ATOMIC); ret = usb_submit_urb(urb_in, GFP_ATOMIC);
if (ret == 0) if (ret == 0)
set_bit(index, &line6pcm->active_urb_in); set_bit(index, &line6pcm->in.active_urbs);
else else
dev_err(line6pcm->line6->ifcdev, dev_err(line6pcm->line6->ifcdev,
"URB in #%d submission failed (%d)\n", index, ret); "URB in #%d submission failed (%d)\n", index, ret);
spin_unlock_irqrestore(&line6pcm->lock_audio_in, flags); spin_unlock_irqrestore(&line6pcm->in.lock, flags);
return 0; return 0;
} }
@ -92,9 +92,9 @@ void line6_unlink_audio_in_urbs(struct snd_line6_pcm *line6pcm)
unsigned int i; unsigned int i;
for (i = 0; i < LINE6_ISO_BUFFERS; i++) { for (i = 0; i < LINE6_ISO_BUFFERS; i++) {
if (test_bit(i, &line6pcm->active_urb_in)) { if (test_bit(i, &line6pcm->in.active_urbs)) {
if (!test_and_set_bit(i, &line6pcm->unlink_urb_in)) { if (!test_and_set_bit(i, &line6pcm->in.unlink_urbs)) {
struct urb *u = line6pcm->urb_audio_in[i]; struct urb *u = line6pcm->in.urbs[i];
usb_unlink_urb(u); usb_unlink_urb(u);
} }
@ -115,7 +115,7 @@ void line6_wait_clear_audio_in_urbs(struct snd_line6_pcm *line6pcm)
do { do {
alive = 0; alive = 0;
for (i = 0; i < LINE6_ISO_BUFFERS; i++) { for (i = 0; i < LINE6_ISO_BUFFERS; i++) {
if (test_bit(i, &line6pcm->active_urb_in)) if (test_bit(i, &line6pcm->in.active_urbs))
alive++; alive++;
} }
if (!alive) if (!alive)
@ -150,18 +150,18 @@ void line6_capture_copy(struct snd_line6_pcm *line6pcm, char *fbuf, int fsize)
if (runtime == NULL) if (runtime == NULL)
return; return;
if (line6pcm->pos_in_done + frames > runtime->buffer_size) { if (line6pcm->in.pos_done + frames > runtime->buffer_size) {
/* /*
The transferred area goes over buffer boundary, The transferred area goes over buffer boundary,
copy two separate chunks. copy two separate chunks.
*/ */
int len; int len;
len = runtime->buffer_size - line6pcm->pos_in_done; len = runtime->buffer_size - line6pcm->in.pos_done;
if (len > 0) { if (len > 0) {
memcpy(runtime->dma_area + memcpy(runtime->dma_area +
line6pcm->pos_in_done * bytes_per_frame, fbuf, line6pcm->in.pos_done * bytes_per_frame, fbuf,
len * bytes_per_frame); len * bytes_per_frame);
memcpy(runtime->dma_area, fbuf + len * bytes_per_frame, memcpy(runtime->dma_area, fbuf + len * bytes_per_frame,
(frames - len) * bytes_per_frame); (frames - len) * bytes_per_frame);
@ -173,12 +173,12 @@ void line6_capture_copy(struct snd_line6_pcm *line6pcm, char *fbuf, int fsize)
} else { } else {
/* copy single chunk */ /* copy single chunk */
memcpy(runtime->dma_area + memcpy(runtime->dma_area +
line6pcm->pos_in_done * bytes_per_frame, fbuf, fsize); line6pcm->in.pos_done * bytes_per_frame, fbuf, fsize);
} }
line6pcm->pos_in_done += frames; line6pcm->in.pos_done += frames;
if (line6pcm->pos_in_done >= runtime->buffer_size) if (line6pcm->in.pos_done >= runtime->buffer_size)
line6pcm->pos_in_done -= runtime->buffer_size; line6pcm->in.pos_done -= runtime->buffer_size;
} }
void line6_capture_check_period(struct snd_line6_pcm *line6pcm, int length) void line6_capture_check_period(struct snd_line6_pcm *line6pcm, int length)
@ -186,17 +186,17 @@ void line6_capture_check_period(struct snd_line6_pcm *line6pcm, int length)
struct snd_pcm_substream *substream = struct snd_pcm_substream *substream =
get_substream(line6pcm, SNDRV_PCM_STREAM_CAPTURE); get_substream(line6pcm, SNDRV_PCM_STREAM_CAPTURE);
line6pcm->bytes_in += length; line6pcm->in.bytes += length;
if (line6pcm->bytes_in >= line6pcm->period_in) { if (line6pcm->in.bytes >= line6pcm->in.period) {
line6pcm->bytes_in %= line6pcm->period_in; line6pcm->in.bytes %= line6pcm->in.period;
snd_pcm_period_elapsed(substream); snd_pcm_period_elapsed(substream);
} }
} }
void line6_free_capture_buffer(struct snd_line6_pcm *line6pcm) void line6_free_capture_buffer(struct snd_line6_pcm *line6pcm)
{ {
kfree(line6pcm->buffer_in); kfree(line6pcm->in.buffer);
line6pcm->buffer_in = NULL; line6pcm->in.buffer = NULL;
} }
/* /*
@ -209,14 +209,14 @@ static void audio_in_callback(struct urb *urb)
struct snd_line6_pcm *line6pcm = (struct snd_line6_pcm *)urb->context; struct snd_line6_pcm *line6pcm = (struct snd_line6_pcm *)urb->context;
line6pcm->last_frame_in = urb->start_frame; line6pcm->in.last_frame = urb->start_frame;
/* find index of URB */ /* find index of URB */
for (index = 0; index < LINE6_ISO_BUFFERS; ++index) for (index = 0; index < LINE6_ISO_BUFFERS; ++index)
if (urb == line6pcm->urb_audio_in[index]) if (urb == line6pcm->in.urbs[index])
break; break;
spin_lock_irqsave(&line6pcm->lock_audio_in, flags); spin_lock_irqsave(&line6pcm->in.lock, flags);
for (i = 0; i < LINE6_ISO_PACKETS; ++i) { for (i = 0; i < LINE6_ISO_PACKETS; ++i) {
char *fbuf; char *fbuf;
@ -249,12 +249,12 @@ static void audio_in_callback(struct urb *urb)
line6_capture_copy(line6pcm, fbuf, fsize); line6_capture_copy(line6pcm, fbuf, fsize);
} }
clear_bit(index, &line6pcm->active_urb_in); clear_bit(index, &line6pcm->in.active_urbs);
if (test_and_clear_bit(index, &line6pcm->unlink_urb_in)) if (test_and_clear_bit(index, &line6pcm->in.unlink_urbs))
shutdown = 1; shutdown = 1;
spin_unlock_irqrestore(&line6pcm->lock_audio_in, flags); spin_unlock_irqrestore(&line6pcm->in.lock, flags);
if (!shutdown) { if (!shutdown) {
submit_audio_in_urb(line6pcm); submit_audio_in_urb(line6pcm);
@ -309,7 +309,7 @@ static int snd_line6_capture_hw_params(struct snd_pcm_substream *substream,
return ret; return ret;
} }
line6pcm->period_in = params_period_bytes(hw_params); line6pcm->in.period = params_period_bytes(hw_params);
return 0; return 0;
} }
@ -361,7 +361,7 @@ snd_line6_capture_pointer(struct snd_pcm_substream *substream)
{ {
struct snd_line6_pcm *line6pcm = snd_pcm_substream_chip(substream); struct snd_line6_pcm *line6pcm = snd_pcm_substream_chip(substream);
return line6pcm->pos_in_done; return line6pcm->in.pos_done;
} }
/* capture operators */ /* capture operators */
@ -386,7 +386,7 @@ int line6_create_audio_in_urbs(struct snd_line6_pcm *line6pcm)
struct urb *urb; struct urb *urb;
/* URB for audio in: */ /* URB for audio in: */
urb = line6pcm->urb_audio_in[i] = urb = line6pcm->in.urbs[i] =
usb_alloc_urb(LINE6_ISO_PACKETS, GFP_KERNEL); usb_alloc_urb(LINE6_ISO_PACKETS, GFP_KERNEL);
if (urb == NULL) if (urb == NULL)

View file

@ -112,11 +112,11 @@ int line6_pcm_acquire(struct snd_line6_pcm *line6pcm, int channels)
if (test_flags(flags_old, flags_new, LINE6_BITS_CAPTURE_BUFFER)) { if (test_flags(flags_old, flags_new, LINE6_BITS_CAPTURE_BUFFER)) {
/* Invoked multiple times in a row so allocate once only */ /* Invoked multiple times in a row so allocate once only */
if (!line6pcm->buffer_in) { if (!line6pcm->in.buffer) {
line6pcm->buffer_in = line6pcm->in.buffer =
kmalloc(LINE6_ISO_BUFFERS * LINE6_ISO_PACKETS * kmalloc(LINE6_ISO_BUFFERS * LINE6_ISO_PACKETS *
line6pcm->max_packet_size, GFP_KERNEL); line6pcm->max_packet_size, GFP_KERNEL);
if (!line6pcm->buffer_in) { if (!line6pcm->in.buffer) {
err = -ENOMEM; err = -ENOMEM;
goto pcm_acquire_error; goto pcm_acquire_error;
} }
@ -131,13 +131,13 @@ int line6_pcm_acquire(struct snd_line6_pcm *line6pcm, int channels)
a bug, we therefore report an error if capturing is restarted a bug, we therefore report an error if capturing is restarted
too soon. too soon.
*/ */
if (line6pcm->active_urb_in || line6pcm->unlink_urb_in) { if (line6pcm->in.active_urbs || line6pcm->in.unlink_urbs) {
dev_err(line6pcm->line6->ifcdev, "Device not yet ready\n"); dev_err(line6pcm->line6->ifcdev, "Device not yet ready\n");
err = -EBUSY; err = -EBUSY;
goto pcm_acquire_error; goto pcm_acquire_error;
} }
line6pcm->count_in = 0; line6pcm->in.count = 0;
line6pcm->prev_fsize = 0; line6pcm->prev_fsize = 0;
err = line6_submit_audio_in_all_urbs(line6pcm); err = line6_submit_audio_in_all_urbs(line6pcm);
@ -149,11 +149,11 @@ int line6_pcm_acquire(struct snd_line6_pcm *line6pcm, int channels)
if (test_flags(flags_old, flags_new, LINE6_BITS_PLAYBACK_BUFFER)) { if (test_flags(flags_old, flags_new, LINE6_BITS_PLAYBACK_BUFFER)) {
/* Invoked multiple times in a row so allocate once only */ /* Invoked multiple times in a row so allocate once only */
if (!line6pcm->buffer_out) { if (!line6pcm->out.buffer) {
line6pcm->buffer_out = line6pcm->out.buffer =
kmalloc(LINE6_ISO_BUFFERS * LINE6_ISO_PACKETS * kmalloc(LINE6_ISO_BUFFERS * LINE6_ISO_PACKETS *
line6pcm->max_packet_size, GFP_KERNEL); line6pcm->max_packet_size, GFP_KERNEL);
if (!line6pcm->buffer_out) { if (!line6pcm->out.buffer) {
err = -ENOMEM; err = -ENOMEM;
goto pcm_acquire_error; goto pcm_acquire_error;
} }
@ -166,12 +166,12 @@ int line6_pcm_acquire(struct snd_line6_pcm *line6pcm, int channels)
/* /*
See comment above regarding PCM restart. See comment above regarding PCM restart.
*/ */
if (line6pcm->active_urb_out || line6pcm->unlink_urb_out) { if (line6pcm->out.active_urbs || line6pcm->out.unlink_urbs) {
dev_err(line6pcm->line6->ifcdev, "Device not yet ready\n"); dev_err(line6pcm->line6->ifcdev, "Device not yet ready\n");
return -EBUSY; return -EBUSY;
} }
line6pcm->count_out = 0; line6pcm->out.count = 0;
err = line6_submit_audio_out_all_urbs(line6pcm); err = line6_submit_audio_out_all_urbs(line6pcm);
if (err < 0) if (err < 0)
@ -331,13 +331,13 @@ static void line6_cleanup_pcm(struct snd_pcm *pcm)
struct snd_line6_pcm *line6pcm = snd_pcm_chip(pcm); struct snd_line6_pcm *line6pcm = snd_pcm_chip(pcm);
for (i = 0; i < LINE6_ISO_BUFFERS; i++) { for (i = 0; i < LINE6_ISO_BUFFERS; i++) {
if (line6pcm->urb_audio_out[i]) { if (line6pcm->out.urbs[i]) {
usb_kill_urb(line6pcm->urb_audio_out[i]); usb_kill_urb(line6pcm->out.urbs[i]);
usb_free_urb(line6pcm->urb_audio_out[i]); usb_free_urb(line6pcm->out.urbs[i]);
} }
if (line6pcm->urb_audio_in[i]) { if (line6pcm->in.urbs[i]) {
usb_kill_urb(line6pcm->urb_audio_in[i]); usb_kill_urb(line6pcm->in.urbs[i]);
usb_free_urb(line6pcm->urb_audio_in[i]); usb_free_urb(line6pcm->in.urbs[i]);
} }
} }
kfree(line6pcm); kfree(line6pcm);
@ -415,8 +415,8 @@ int line6_init_pcm(struct usb_line6 *line6,
usb_maxpacket(line6->usbdev, usb_maxpacket(line6->usbdev,
usb_sndisocpipe(line6->usbdev, ep_write), 1)); usb_sndisocpipe(line6->usbdev, ep_write), 1));
spin_lock_init(&line6pcm->lock_audio_out); spin_lock_init(&line6pcm->out.lock);
spin_lock_init(&line6pcm->lock_audio_in); spin_lock_init(&line6pcm->in.lock);
line6pcm->impulse_period = LINE6_IMPULSE_DEFAULT_PERIOD; line6pcm->impulse_period = LINE6_IMPULSE_DEFAULT_PERIOD;
line6->line6pcm = line6pcm; line6->line6pcm = line6pcm;
@ -464,13 +464,13 @@ int snd_line6_prepare(struct snd_pcm_substream *substream)
} }
if (!test_and_set_bit(LINE6_INDEX_PREPARED, &line6pcm->flags)) { if (!test_and_set_bit(LINE6_INDEX_PREPARED, &line6pcm->flags)) {
line6pcm->count_out = 0; line6pcm->out.count = 0;
line6pcm->pos_out = 0; line6pcm->out.pos = 0;
line6pcm->pos_out_done = 0; line6pcm->out.pos_done = 0;
line6pcm->bytes_out = 0; line6pcm->out.bytes = 0;
line6pcm->count_in = 0; line6pcm->in.count = 0;
line6pcm->pos_in_done = 0; line6pcm->in.pos_done = 0;
line6pcm->bytes_in = 0; line6pcm->in.bytes = 0;
} }
return 0; return 0;

View file

@ -165,6 +165,49 @@ struct line6_pcm_properties {
int bytes_per_frame; int bytes_per_frame;
}; };
struct line6_pcm_stream {
/* allocated URBs */
struct urb *urbs[LINE6_ISO_BUFFERS];
/* Temporary buffer;
* Since the packet size is not known in advance, this buffer is
* large enough to store maximum size packets.
*/
unsigned char *buffer;
/* Free frame position in the buffer. */
snd_pcm_uframes_t pos;
/* Count processed bytes;
* This is modulo period size (to determine when a period is finished).
*/
unsigned bytes;
/* Counter to create desired sample rate */
unsigned count;
/* period size in bytes */
unsigned period;
/* Processed frame position in the buffer;
* The contents of the ring buffer have been consumed by the USB
* subsystem (i.e., sent to the USB device) up to this position.
*/
snd_pcm_uframes_t pos_done;
/* Bit mask of active URBs */
unsigned long active_urbs;
/* Bit mask of URBs currently being unlinked */
unsigned long unlink_urbs;
/* Spin lock to protect updates of the buffer positions (not contents)
*/
spinlock_t lock;
int last_frame;
};
struct snd_line6_pcm { struct snd_line6_pcm {
/** /**
Pointer back to the Line 6 driver data structure. Pointer back to the Line 6 driver data structure.
@ -181,29 +224,9 @@ struct snd_line6_pcm {
*/ */
struct snd_pcm *pcm; struct snd_pcm *pcm;
/** /* Capture and playback streams */
URBs for audio playback. struct line6_pcm_stream in;
*/ struct line6_pcm_stream out;
struct urb *urb_audio_out[LINE6_ISO_BUFFERS];
/**
URBs for audio capture.
*/
struct urb *urb_audio_in[LINE6_ISO_BUFFERS];
/**
Temporary buffer for playback.
Since the packet size is not known in advance, this buffer is
large enough to store maximum size packets.
*/
unsigned char *buffer_out;
/**
Temporary buffer for capture.
Since the packet size is not known in advance, this buffer is
large enough to store maximum size packets.
*/
unsigned char *buffer_in;
/** /**
Previously captured frame (for software monitoring). Previously captured frame (for software monitoring).
@ -215,98 +238,11 @@ struct snd_line6_pcm {
*/ */
int prev_fsize; int prev_fsize;
/**
Free frame position in the playback buffer.
*/
snd_pcm_uframes_t pos_out;
/**
Count processed bytes for playback.
This is modulo period size (to determine when a period is
finished).
*/
unsigned bytes_out;
/**
Counter to create desired playback sample rate.
*/
unsigned count_out;
/**
Playback period size in bytes
*/
unsigned period_out;
/**
Processed frame position in the playback buffer.
The contents of the output ring buffer have been consumed by
the USB subsystem (i.e., sent to the USB device) up to this
position.
*/
snd_pcm_uframes_t pos_out_done;
/**
Count processed bytes for capture.
This is modulo period size (to determine when a period is
finished).
*/
unsigned bytes_in;
/**
Counter to create desired capture sample rate.
*/
unsigned count_in;
/**
Capture period size in bytes
*/
unsigned period_in;
/**
Processed frame position in the capture buffer.
The contents of the output ring buffer have been consumed by
the USB subsystem (i.e., sent to the USB device) up to this
position.
*/
snd_pcm_uframes_t pos_in_done;
/**
Bit mask of active playback URBs.
*/
unsigned long active_urb_out;
/** /**
Maximum size of USB packet. Maximum size of USB packet.
*/ */
int max_packet_size; int max_packet_size;
/**
Bit mask of active capture URBs.
*/
unsigned long active_urb_in;
/**
Bit mask of playback URBs currently being unlinked.
*/
unsigned long unlink_urb_out;
/**
Bit mask of capture URBs currently being unlinked.
*/
unsigned long unlink_urb_in;
/**
Spin lock to protect updates of the playback buffer positions (not
contents!)
*/
spinlock_t lock_audio_out;
/**
Spin lock to protect updates of the capture buffer positions (not
contents!)
*/
spinlock_t lock_audio_in;
/** /**
PCM playback volume (left and right). PCM playback volume (left and right).
*/ */
@ -336,8 +272,6 @@ struct snd_line6_pcm {
Several status bits (see LINE6_BIT_*). Several status bits (see LINE6_BIT_*).
*/ */
unsigned long flags; unsigned long flags;
int last_frame_in, last_frame_out;
}; };
extern int line6_init_pcm(struct usb_line6 *line6, extern int line6_init_pcm(struct usb_line6 *line6,

View file

@ -145,17 +145,17 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
(USB_INTERVALS_PER_SECOND / LINE6_ISO_INTERVAL); (USB_INTERVALS_PER_SECOND / LINE6_ISO_INTERVAL);
struct urb *urb_out; struct urb *urb_out;
spin_lock_irqsave(&line6pcm->lock_audio_out, flags); spin_lock_irqsave(&line6pcm->out.lock, flags);
index = index =
find_first_zero_bit(&line6pcm->active_urb_out, LINE6_ISO_BUFFERS); find_first_zero_bit(&line6pcm->out.active_urbs, LINE6_ISO_BUFFERS);
if (index < 0 || index >= LINE6_ISO_BUFFERS) { if (index < 0 || index >= LINE6_ISO_BUFFERS) {
spin_unlock_irqrestore(&line6pcm->lock_audio_out, flags); spin_unlock_irqrestore(&line6pcm->out.lock, flags);
dev_err(line6pcm->line6->ifcdev, "no free URB found\n"); dev_err(line6pcm->line6->ifcdev, "no free URB found\n");
return -EINVAL; return -EINVAL;
} }
urb_out = line6pcm->urb_audio_out[index]; urb_out = line6pcm->out.urbs[index];
urb_size = 0; urb_size = 0;
for (i = 0; i < LINE6_ISO_PACKETS; ++i) { for (i = 0; i < LINE6_ISO_PACKETS; ++i) {
@ -170,9 +170,9 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
if (fsize == 0) { if (fsize == 0) {
int n; int n;
line6pcm->count_out += frame_increment; line6pcm->out.count += frame_increment;
n = line6pcm->count_out / frame_factor; n = line6pcm->out.count / frame_factor;
line6pcm->count_out -= n * frame_factor; line6pcm->out.count -= n * frame_factor;
fsize = n * bytes_per_frame; fsize = n * bytes_per_frame;
} }
@ -183,14 +183,14 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
if (urb_size == 0) { if (urb_size == 0) {
/* can't determine URB size */ /* can't determine URB size */
spin_unlock_irqrestore(&line6pcm->lock_audio_out, flags); spin_unlock_irqrestore(&line6pcm->out.lock, flags);
dev_err(line6pcm->line6->ifcdev, "driver bug: urb_size = 0\n"); dev_err(line6pcm->line6->ifcdev, "driver bug: urb_size = 0\n");
return -EINVAL; return -EINVAL;
} }
urb_frames = urb_size / bytes_per_frame; urb_frames = urb_size / bytes_per_frame;
urb_out->transfer_buffer = urb_out->transfer_buffer =
line6pcm->buffer_out + line6pcm->out.buffer +
index * LINE6_ISO_PACKETS * line6pcm->max_packet_size; index * LINE6_ISO_PACKETS * line6pcm->max_packet_size;
urb_out->transfer_buffer_length = urb_size; urb_out->transfer_buffer_length = urb_size;
urb_out->context = line6pcm; urb_out->context = line6pcm;
@ -200,19 +200,19 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
struct snd_pcm_runtime *runtime = struct snd_pcm_runtime *runtime =
get_substream(line6pcm, SNDRV_PCM_STREAM_PLAYBACK)->runtime; get_substream(line6pcm, SNDRV_PCM_STREAM_PLAYBACK)->runtime;
if (line6pcm->pos_out + urb_frames > runtime->buffer_size) { if (line6pcm->out.pos + urb_frames > runtime->buffer_size) {
/* /*
The transferred area goes over buffer boundary, The transferred area goes over buffer boundary,
copy the data to the temp buffer. copy the data to the temp buffer.
*/ */
int len; int len;
len = runtime->buffer_size - line6pcm->pos_out; len = runtime->buffer_size - line6pcm->out.pos;
if (len > 0) { if (len > 0) {
memcpy(urb_out->transfer_buffer, memcpy(urb_out->transfer_buffer,
runtime->dma_area + runtime->dma_area +
line6pcm->pos_out * bytes_per_frame, line6pcm->out.pos * bytes_per_frame,
len * bytes_per_frame); len * bytes_per_frame);
memcpy(urb_out->transfer_buffer + memcpy(urb_out->transfer_buffer +
len * bytes_per_frame, runtime->dma_area, len * bytes_per_frame, runtime->dma_area,
@ -223,13 +223,13 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
} else { } else {
memcpy(urb_out->transfer_buffer, memcpy(urb_out->transfer_buffer,
runtime->dma_area + runtime->dma_area +
line6pcm->pos_out * bytes_per_frame, line6pcm->out.pos * bytes_per_frame,
urb_out->transfer_buffer_length); urb_out->transfer_buffer_length);
} }
line6pcm->pos_out += urb_frames; line6pcm->out.pos += urb_frames;
if (line6pcm->pos_out >= runtime->buffer_size) if (line6pcm->out.pos >= runtime->buffer_size)
line6pcm->pos_out -= runtime->buffer_size; line6pcm->out.pos -= runtime->buffer_size;
} else { } else {
memset(urb_out->transfer_buffer, 0, memset(urb_out->transfer_buffer, 0,
urb_out->transfer_buffer_length); urb_out->transfer_buffer_length);
@ -265,12 +265,12 @@ static int submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
ret = usb_submit_urb(urb_out, GFP_ATOMIC); ret = usb_submit_urb(urb_out, GFP_ATOMIC);
if (ret == 0) if (ret == 0)
set_bit(index, &line6pcm->active_urb_out); set_bit(index, &line6pcm->out.active_urbs);
else else
dev_err(line6pcm->line6->ifcdev, dev_err(line6pcm->line6->ifcdev,
"URB out #%d submission failed (%d)\n", index, ret); "URB out #%d submission failed (%d)\n", index, ret);
spin_unlock_irqrestore(&line6pcm->lock_audio_out, flags); spin_unlock_irqrestore(&line6pcm->out.lock, flags);
return 0; return 0;
} }
@ -298,9 +298,9 @@ void line6_unlink_audio_out_urbs(struct snd_line6_pcm *line6pcm)
unsigned int i; unsigned int i;
for (i = 0; i < LINE6_ISO_BUFFERS; i++) { for (i = 0; i < LINE6_ISO_BUFFERS; i++) {
if (test_bit(i, &line6pcm->active_urb_out)) { if (test_bit(i, &line6pcm->out.active_urbs)) {
if (!test_and_set_bit(i, &line6pcm->unlink_urb_out)) { if (!test_and_set_bit(i, &line6pcm->out.unlink_urbs)) {
struct urb *u = line6pcm->urb_audio_out[i]; struct urb *u = line6pcm->out.urbs[i];
usb_unlink_urb(u); usb_unlink_urb(u);
} }
@ -321,7 +321,7 @@ void line6_wait_clear_audio_out_urbs(struct snd_line6_pcm *line6pcm)
do { do {
alive = 0; alive = 0;
for (i = 0; i < LINE6_ISO_BUFFERS; i++) { for (i = 0; i < LINE6_ISO_BUFFERS; i++) {
if (test_bit(i, &line6pcm->active_urb_out)) if (test_bit(i, &line6pcm->out.active_urbs))
alive++; alive++;
} }
if (!alive) if (!alive)
@ -344,8 +344,8 @@ void line6_unlink_wait_clear_audio_out_urbs(struct snd_line6_pcm *line6pcm)
void line6_free_playback_buffer(struct snd_line6_pcm *line6pcm) void line6_free_playback_buffer(struct snd_line6_pcm *line6pcm)
{ {
kfree(line6pcm->buffer_out); kfree(line6pcm->out.buffer);
line6pcm->buffer_out = NULL; line6pcm->out.buffer = NULL;
} }
/* /*
@ -363,11 +363,11 @@ static void audio_out_callback(struct urb *urb)
memset(urb->transfer_buffer, 0, urb->transfer_buffer_length); memset(urb->transfer_buffer, 0, urb->transfer_buffer_length);
#endif #endif
line6pcm->last_frame_out = urb->start_frame; line6pcm->out.last_frame = urb->start_frame;
/* find index of URB */ /* find index of URB */
for (index = 0; index < LINE6_ISO_BUFFERS; index++) for (index = 0; index < LINE6_ISO_BUFFERS; index++)
if (urb == line6pcm->urb_audio_out[index]) if (urb == line6pcm->out.urbs[index])
break; break;
if (index >= LINE6_ISO_BUFFERS) if (index >= LINE6_ISO_BUFFERS)
@ -376,19 +376,19 @@ static void audio_out_callback(struct urb *urb)
for (i = 0; i < LINE6_ISO_PACKETS; i++) for (i = 0; i < LINE6_ISO_PACKETS; i++)
length += urb->iso_frame_desc[i].length; length += urb->iso_frame_desc[i].length;
spin_lock_irqsave(&line6pcm->lock_audio_out, flags); spin_lock_irqsave(&line6pcm->out.lock, flags);
if (test_bit(LINE6_INDEX_PCM_ALSA_PLAYBACK_STREAM, &line6pcm->flags)) { if (test_bit(LINE6_INDEX_PCM_ALSA_PLAYBACK_STREAM, &line6pcm->flags)) {
struct snd_pcm_runtime *runtime = substream->runtime; struct snd_pcm_runtime *runtime = substream->runtime;
line6pcm->pos_out_done += line6pcm->out.pos_done +=
length / line6pcm->properties->bytes_per_frame; length / line6pcm->properties->bytes_per_frame;
if (line6pcm->pos_out_done >= runtime->buffer_size) if (line6pcm->out.pos_done >= runtime->buffer_size)
line6pcm->pos_out_done -= runtime->buffer_size; line6pcm->out.pos_done -= runtime->buffer_size;
} }
clear_bit(index, &line6pcm->active_urb_out); clear_bit(index, &line6pcm->out.active_urbs);
for (i = 0; i < LINE6_ISO_PACKETS; i++) for (i = 0; i < LINE6_ISO_PACKETS; i++)
if (urb->iso_frame_desc[i].status == -EXDEV) { if (urb->iso_frame_desc[i].status == -EXDEV) {
@ -396,19 +396,19 @@ static void audio_out_callback(struct urb *urb)
break; break;
} }
if (test_and_clear_bit(index, &line6pcm->unlink_urb_out)) if (test_and_clear_bit(index, &line6pcm->out.unlink_urbs))
shutdown = 1; shutdown = 1;
spin_unlock_irqrestore(&line6pcm->lock_audio_out, flags); spin_unlock_irqrestore(&line6pcm->out.lock, flags);
if (!shutdown) { if (!shutdown) {
submit_audio_out_urb(line6pcm); submit_audio_out_urb(line6pcm);
if (test_bit(LINE6_INDEX_PCM_ALSA_PLAYBACK_STREAM, if (test_bit(LINE6_INDEX_PCM_ALSA_PLAYBACK_STREAM,
&line6pcm->flags)) { &line6pcm->flags)) {
line6pcm->bytes_out += length; line6pcm->out.bytes += length;
if (line6pcm->bytes_out >= line6pcm->period_out) { if (line6pcm->out.bytes >= line6pcm->out.period) {
line6pcm->bytes_out %= line6pcm->period_out; line6pcm->out.bytes %= line6pcm->out.period;
snd_pcm_period_elapsed(substream); snd_pcm_period_elapsed(substream);
} }
} }
@ -457,7 +457,7 @@ static int snd_line6_playback_hw_params(struct snd_pcm_substream *substream,
return ret; return ret;
} }
line6pcm->period_out = params_period_bytes(hw_params); line6pcm->out.period = params_period_bytes(hw_params);
return 0; return 0;
} }
@ -517,7 +517,7 @@ snd_line6_playback_pointer(struct snd_pcm_substream *substream)
{ {
struct snd_line6_pcm *line6pcm = snd_pcm_substream_chip(substream); struct snd_line6_pcm *line6pcm = snd_pcm_substream_chip(substream);
return line6pcm->pos_out_done; return line6pcm->out.pos_done;
} }
/* playback operators */ /* playback operators */
@ -542,7 +542,7 @@ int line6_create_audio_out_urbs(struct snd_line6_pcm *line6pcm)
struct urb *urb; struct urb *urb;
/* URB for audio out: */ /* URB for audio out: */
urb = line6pcm->urb_audio_out[i] = urb = line6pcm->out.urbs[i] =
usb_alloc_urb(LINE6_ISO_PACKETS, GFP_KERNEL); usb_alloc_urb(LINE6_ISO_PACKETS, GFP_KERNEL);
if (urb == NULL) if (urb == NULL)