2019-05-27 09:55:05 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2010-03-04 21:46:13 +03:00
/*
*/
2011-09-14 14:46:57 +04:00
# include <linux/gfp.h>
# include <linux/init.h>
2012-01-09 14:37:20 +04:00
# include <linux/ratelimit.h>
2011-09-14 14:46:57 +04:00
# include <linux/usb.h>
# include <linux/usb/audio.h>
2012-04-12 15:51:11 +04:00
# include <linux/slab.h>
2011-09-14 14:46:57 +04:00
# include <sound/core.h>
# include <sound/pcm.h>
2012-04-12 15:51:11 +04:00
# include <sound/pcm_params.h>
2011-09-14 14:46:57 +04:00
# include "usbaudio.h"
# include "helper.h"
# include "card.h"
# include "endpoint.h"
# include "pcm.h"
2012-09-04 12:23:07 +04:00
# include "quirks.h"
2011-09-14 14:46:57 +04:00
2012-04-12 15:51:11 +04:00
# define EP_FLAG_RUNNING 1
2012-11-08 11:52:45 +04:00
# define EP_FLAG_STOPPING 2
2012-04-12 15:51:11 +04:00
2012-04-12 15:51:15 +04:00
/*
* snd_usb_endpoint is a model that abstracts everything related to an
* USB endpoint and its streaming .
*
* There are functions to activate and deactivate the streaming URBs and
2012-04-24 21:31:24 +04:00
* optional callbacks to let the pcm logic handle the actual content of the
2012-04-12 15:51:15 +04:00
* packets for playback and record . Thus , the bus streaming and the audio
* handlers are fully decoupled .
*
2012-04-24 21:31:24 +04:00
* There are two different types of endpoints in audio applications .
2012-04-12 15:51:15 +04:00
*
* SND_USB_ENDPOINT_TYPE_DATA handles full audio data payload for both
* inbound and outbound traffic .
*
2012-04-24 21:31:24 +04:00
* SND_USB_ENDPOINT_TYPE_SYNC endpoints are for inbound traffic only and
* expect the payload to carry Q10 .14 / Q16 .16 formatted sync information
* ( 3 or 4 bytes ) .
2012-04-12 15:51:15 +04:00
*
2012-04-24 21:31:24 +04:00
* Each endpoint has to be configured prior to being used by calling
* snd_usb_endpoint_set_params ( ) .
2012-04-12 15:51:15 +04:00
*
* The model incorporates a reference counting , so that multiple users
* can call snd_usb_endpoint_start ( ) and snd_usb_endpoint_stop ( ) , and
* only the first user will effectively start the URBs , and only the last
2012-04-24 21:31:24 +04:00
* one to stop it will tear the URBs down again .
2012-04-12 15:51:15 +04:00
*/
2011-09-14 14:46:57 +04:00
/*
* convert a sampling rate into our full speed format ( fs / 1000 in Q16 .16 )
* this will overflow at approx 524 kHz
*/
static inline unsigned get_usb_full_speed_rate ( unsigned int rate )
{
return ( ( rate < < 13 ) + 62 ) / 125 ;
}
/*
* convert a sampling rate into USB high speed format ( fs / 8000 in Q16 .16 )
* this will overflow at approx 4 MHz
*/
static inline unsigned get_usb_high_speed_rate ( unsigned int rate )
{
return ( ( rate < < 10 ) + 62 ) / 125 ;
}
/*
* release a urb data
*/
static void release_urb_ctx ( struct snd_urb_ctx * u )
{
2012-04-12 15:51:13 +04:00
if ( u - > buffer_size )
usb_free_coherent ( u - > ep - > chip - > dev , u - > buffer_size ,
u - > urb - > transfer_buffer ,
u - > urb - > transfer_dma ) ;
usb_free_urb ( u - > urb ) ;
u - > urb = NULL ;
2011-09-14 14:46:57 +04:00
}
static const char * usb_error_string ( int err )
{
switch ( err ) {
case - ENODEV :
return " no device " ;
case - ENOENT :
return " endpoint not enabled " ;
case - EPIPE :
return " endpoint stalled " ;
case - ENOSPC :
return " not enough bandwidth " ;
case - ESHUTDOWN :
return " device disabled " ;
case - EHOSTUNREACH :
return " device suspended " ;
case - EINVAL :
case - EAGAIN :
case - EFBIG :
case - EMSGSIZE :
return " internal error " ;
default :
return " unknown error " ;
}
}
2012-04-12 15:51:15 +04:00
/**
* snd_usb_endpoint_implicit_feedback_sink : Report endpoint usage type
*
2012-04-24 21:31:24 +04:00
* @ ep : The snd_usb_endpoint
2012-04-12 15:51:15 +04:00
*
* Determine whether an endpoint is driven by an implicit feedback
* data endpoint source .
*/
2013-04-04 01:18:52 +04:00
int snd_usb_endpoint_implicit_feedback_sink ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
return ep - > sync_master & &
ep - > sync_master - > type = = SND_USB_ENDPOINT_TYPE_DATA & &
ep - > type = = SND_USB_ENDPOINT_TYPE_DATA & &
usb_pipeout ( ep - > pipe ) ;
}
2012-04-12 15:51:15 +04:00
/*
* For streaming based on information derived from sync endpoints ,
2020-04-24 05:24:48 +03:00
* prepare_outbound_urb_sizes ( ) will call slave_next_packet_size ( ) to
2012-04-12 15:51:15 +04:00
* determine the number of samples to be sent in the next packet .
*
2020-04-24 05:24:48 +03:00
* For implicit feedback , slave_next_packet_size ( ) is unused .
2012-04-12 15:51:15 +04:00
*/
2020-04-24 05:24:48 +03:00
int snd_usb_endpoint_slave_next_packet_size ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
unsigned long flags ;
int ret ;
if ( ep - > fill_max )
return ep - > maxframesize ;
spin_lock_irqsave ( & ep - > lock , flags ) ;
ep - > phase = ( ep - > phase & 0xffff )
+ ( ep - > freqm < < ep - > datainterval ) ;
ret = min ( ep - > phase > > 16 , ep - > maxframesize ) ;
spin_unlock_irqrestore ( & ep - > lock , flags ) ;
return ret ;
}
2020-04-24 05:24:48 +03:00
/*
* For adaptive and synchronous endpoints , prepare_outbound_urb_sizes ( )
* will call next_packet_size ( ) to determine the number of samples to be
* sent in the next packet .
*/
int snd_usb_endpoint_next_packet_size ( struct snd_usb_endpoint * ep )
{
int ret ;
if ( ep - > fill_max )
return ep - > maxframesize ;
ep - > sample_accum + = ep - > sample_rem ;
2020-06-29 05:59:34 +03:00
if ( ep - > sample_accum > = ep - > pps ) {
ep - > sample_accum - = ep - > pps ;
ret = ep - > packsize [ 1 ] ;
2020-04-24 05:24:48 +03:00
} else {
2020-06-29 05:59:34 +03:00
ret = ep - > packsize [ 0 ] ;
2020-04-24 05:24:48 +03:00
}
return ret ;
}
2012-04-12 15:51:11 +04:00
static void retire_outbound_urb ( struct snd_usb_endpoint * ep ,
struct snd_urb_ctx * urb_ctx )
{
if ( ep - > retire_data_urb )
ep - > retire_data_urb ( ep - > data_subs , urb_ctx - > urb ) ;
}
static void retire_inbound_urb ( struct snd_usb_endpoint * ep ,
struct snd_urb_ctx * urb_ctx )
{
struct urb * urb = urb_ctx - > urb ;
2012-09-04 12:23:07 +04:00
if ( unlikely ( ep - > skip_packets > 0 ) ) {
ep - > skip_packets - - ;
return ;
}
2012-04-12 15:51:11 +04:00
if ( ep - > sync_slave )
snd_usb_handle_sync_urb ( ep - > sync_slave , ep , urb ) ;
if ( ep - > retire_data_urb )
ep - > retire_data_urb ( ep - > data_subs , urb ) ;
}
2015-10-19 09:52:51 +03:00
static void prepare_silent_urb ( struct snd_usb_endpoint * ep ,
struct snd_urb_ctx * ctx )
{
struct urb * urb = ctx - > urb ;
unsigned int offs = 0 ;
ALSA: USB-audio: Add quirk for Zoom R16/24 playback
The Zoom R16/24 have a nonstandard playback format where each isochronous
packet contains a length descriptor in the first four bytes. (Curiously,
capture data does not contain this and requires no quirk.)
The quirk involves adding the extra length descriptor whenever outgoing
isochronous packets are generated, both in pcm.c (outgoing audio) and
endpoint.c (silent data).
In order to make the quirk as unintrusive as possible, for
pcm.c:prepare_playback_urb(), the isochronous packet descriptors are
initially set up in the same way no matter if the quirk is enabled or not.
Once it is time to actually copy the data into the outgoing packet buffer
(together with the added length descriptors) the isochronous descriptors
are adjusted in order take the increased payload length into account.
For endpoint.c:prepare_silent_urb() it makes more sense to modify the
actual function, partly because the function is less complex to start with
and partly because it is not as time-critical as prepare_playback_urb()
(whose bulk is run with interrupts disabled), so the (minute) additional
time spent in the non-quirk case is motivated by the simplicity of having
a single function for all cases.
The quirk is controlled by the new tx_length_quirk member in struct
snd_usb_substream and struct snd_usb_audio, which is conveyed to pcm.c
and endpoint.c from quirks.c in a similar manner to the txfr_quirk member
in the same structs.
In contrast to txfr_quirk however, the quirk is enabled directly in
quirks.c:create_standard_audio_quirk() by checking the USB ID in that
function. Another option would be to introduce a new
QUIRK_AUDIO_ZOOM_INTERFACE or somesuch, which would have made the quirk
very plain to see in the quirk table, but it was felt that the additional
code needed to implement it this way would just make the implementation
more complex with no real gain.
Tested with a Zoom R16, both by doing capture and playback separately
using arecord and aplay (8 channel capture and 2 channel playback,
respectively), as well as capture and playback together using Ardour, as
well as Audacity and Qtractor together with jackd.
The R24 is reportedly compatible with the R16 when used as an audio
interface. Both devices share the same USB ID and have the same number of
inputs (8) and outputs (2). Therefore "R16/24" is mentioned throughout the
patch.
Regression tested using an Edirol UA-5 in both class compliant (16-bit)
and "advanced" (24 bit, forces the use of quirks) modes.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Tested-by: Panu Matilainen <pmatilai@laiskiainen.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-19 09:52:53 +03:00
unsigned int extra = 0 ;
__le32 packet_length ;
2015-10-19 09:52:51 +03:00
int i ;
ALSA: USB-audio: Add quirk for Zoom R16/24 playback
The Zoom R16/24 have a nonstandard playback format where each isochronous
packet contains a length descriptor in the first four bytes. (Curiously,
capture data does not contain this and requires no quirk.)
The quirk involves adding the extra length descriptor whenever outgoing
isochronous packets are generated, both in pcm.c (outgoing audio) and
endpoint.c (silent data).
In order to make the quirk as unintrusive as possible, for
pcm.c:prepare_playback_urb(), the isochronous packet descriptors are
initially set up in the same way no matter if the quirk is enabled or not.
Once it is time to actually copy the data into the outgoing packet buffer
(together with the added length descriptors) the isochronous descriptors
are adjusted in order take the increased payload length into account.
For endpoint.c:prepare_silent_urb() it makes more sense to modify the
actual function, partly because the function is less complex to start with
and partly because it is not as time-critical as prepare_playback_urb()
(whose bulk is run with interrupts disabled), so the (minute) additional
time spent in the non-quirk case is motivated by the simplicity of having
a single function for all cases.
The quirk is controlled by the new tx_length_quirk member in struct
snd_usb_substream and struct snd_usb_audio, which is conveyed to pcm.c
and endpoint.c from quirks.c in a similar manner to the txfr_quirk member
in the same structs.
In contrast to txfr_quirk however, the quirk is enabled directly in
quirks.c:create_standard_audio_quirk() by checking the USB ID in that
function. Another option would be to introduce a new
QUIRK_AUDIO_ZOOM_INTERFACE or somesuch, which would have made the quirk
very plain to see in the quirk table, but it was felt that the additional
code needed to implement it this way would just make the implementation
more complex with no real gain.
Tested with a Zoom R16, both by doing capture and playback separately
using arecord and aplay (8 channel capture and 2 channel playback,
respectively), as well as capture and playback together using Ardour, as
well as Audacity and Qtractor together with jackd.
The R24 is reportedly compatible with the R16 when used as an audio
interface. Both devices share the same USB ID and have the same number of
inputs (8) and outputs (2). Therefore "R16/24" is mentioned throughout the
patch.
Regression tested using an Edirol UA-5 in both class compliant (16-bit)
and "advanced" (24 bit, forces the use of quirks) modes.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Tested-by: Panu Matilainen <pmatilai@laiskiainen.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-19 09:52:53 +03:00
/* For tx_length_quirk, put packet length at start of packet */
if ( ep - > chip - > tx_length_quirk )
extra = sizeof ( packet_length ) ;
2015-10-19 09:52:51 +03:00
for ( i = 0 ; i < ctx - > packets ; + + i ) {
ALSA: USB-audio: Add quirk for Zoom R16/24 playback
The Zoom R16/24 have a nonstandard playback format where each isochronous
packet contains a length descriptor in the first four bytes. (Curiously,
capture data does not contain this and requires no quirk.)
The quirk involves adding the extra length descriptor whenever outgoing
isochronous packets are generated, both in pcm.c (outgoing audio) and
endpoint.c (silent data).
In order to make the quirk as unintrusive as possible, for
pcm.c:prepare_playback_urb(), the isochronous packet descriptors are
initially set up in the same way no matter if the quirk is enabled or not.
Once it is time to actually copy the data into the outgoing packet buffer
(together with the added length descriptors) the isochronous descriptors
are adjusted in order take the increased payload length into account.
For endpoint.c:prepare_silent_urb() it makes more sense to modify the
actual function, partly because the function is less complex to start with
and partly because it is not as time-critical as prepare_playback_urb()
(whose bulk is run with interrupts disabled), so the (minute) additional
time spent in the non-quirk case is motivated by the simplicity of having
a single function for all cases.
The quirk is controlled by the new tx_length_quirk member in struct
snd_usb_substream and struct snd_usb_audio, which is conveyed to pcm.c
and endpoint.c from quirks.c in a similar manner to the txfr_quirk member
in the same structs.
In contrast to txfr_quirk however, the quirk is enabled directly in
quirks.c:create_standard_audio_quirk() by checking the USB ID in that
function. Another option would be to introduce a new
QUIRK_AUDIO_ZOOM_INTERFACE or somesuch, which would have made the quirk
very plain to see in the quirk table, but it was felt that the additional
code needed to implement it this way would just make the implementation
more complex with no real gain.
Tested with a Zoom R16, both by doing capture and playback separately
using arecord and aplay (8 channel capture and 2 channel playback,
respectively), as well as capture and playback together using Ardour, as
well as Audacity and Qtractor together with jackd.
The R24 is reportedly compatible with the R16 when used as an audio
interface. Both devices share the same USB ID and have the same number of
inputs (8) and outputs (2). Therefore "R16/24" is mentioned throughout the
patch.
Regression tested using an Edirol UA-5 in both class compliant (16-bit)
and "advanced" (24 bit, forces the use of quirks) modes.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Tested-by: Panu Matilainen <pmatilai@laiskiainen.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-19 09:52:53 +03:00
unsigned int offset ;
unsigned int length ;
2015-10-19 09:52:51 +03:00
int counts ;
if ( ctx - > packet_size [ i ] )
counts = ctx - > packet_size [ i ] ;
2020-04-24 05:24:48 +03:00
else if ( ep - > sync_master )
counts = snd_usb_endpoint_slave_next_packet_size ( ep ) ;
2015-10-19 09:52:51 +03:00
else
counts = snd_usb_endpoint_next_packet_size ( ep ) ;
ALSA: USB-audio: Add quirk for Zoom R16/24 playback
The Zoom R16/24 have a nonstandard playback format where each isochronous
packet contains a length descriptor in the first four bytes. (Curiously,
capture data does not contain this and requires no quirk.)
The quirk involves adding the extra length descriptor whenever outgoing
isochronous packets are generated, both in pcm.c (outgoing audio) and
endpoint.c (silent data).
In order to make the quirk as unintrusive as possible, for
pcm.c:prepare_playback_urb(), the isochronous packet descriptors are
initially set up in the same way no matter if the quirk is enabled or not.
Once it is time to actually copy the data into the outgoing packet buffer
(together with the added length descriptors) the isochronous descriptors
are adjusted in order take the increased payload length into account.
For endpoint.c:prepare_silent_urb() it makes more sense to modify the
actual function, partly because the function is less complex to start with
and partly because it is not as time-critical as prepare_playback_urb()
(whose bulk is run with interrupts disabled), so the (minute) additional
time spent in the non-quirk case is motivated by the simplicity of having
a single function for all cases.
The quirk is controlled by the new tx_length_quirk member in struct
snd_usb_substream and struct snd_usb_audio, which is conveyed to pcm.c
and endpoint.c from quirks.c in a similar manner to the txfr_quirk member
in the same structs.
In contrast to txfr_quirk however, the quirk is enabled directly in
quirks.c:create_standard_audio_quirk() by checking the USB ID in that
function. Another option would be to introduce a new
QUIRK_AUDIO_ZOOM_INTERFACE or somesuch, which would have made the quirk
very plain to see in the quirk table, but it was felt that the additional
code needed to implement it this way would just make the implementation
more complex with no real gain.
Tested with a Zoom R16, both by doing capture and playback separately
using arecord and aplay (8 channel capture and 2 channel playback,
respectively), as well as capture and playback together using Ardour, as
well as Audacity and Qtractor together with jackd.
The R24 is reportedly compatible with the R16 when used as an audio
interface. Both devices share the same USB ID and have the same number of
inputs (8) and outputs (2). Therefore "R16/24" is mentioned throughout the
patch.
Regression tested using an Edirol UA-5 in both class compliant (16-bit)
and "advanced" (24 bit, forces the use of quirks) modes.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Tested-by: Panu Matilainen <pmatilai@laiskiainen.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-19 09:52:53 +03:00
length = counts * ep - > stride ; /* number of silent bytes */
offset = offs * ep - > stride + extra * i ;
urb - > iso_frame_desc [ i ] . offset = offset ;
urb - > iso_frame_desc [ i ] . length = length + extra ;
if ( extra ) {
packet_length = cpu_to_le32 ( length ) ;
memcpy ( urb - > transfer_buffer + offset ,
& packet_length , sizeof ( packet_length ) ) ;
}
memset ( urb - > transfer_buffer + offset + extra ,
ep - > silence_value , length ) ;
2015-10-19 09:52:51 +03:00
offs + = counts ;
}
urb - > number_of_packets = ctx - > packets ;
ALSA: USB-audio: Add quirk for Zoom R16/24 playback
The Zoom R16/24 have a nonstandard playback format where each isochronous
packet contains a length descriptor in the first four bytes. (Curiously,
capture data does not contain this and requires no quirk.)
The quirk involves adding the extra length descriptor whenever outgoing
isochronous packets are generated, both in pcm.c (outgoing audio) and
endpoint.c (silent data).
In order to make the quirk as unintrusive as possible, for
pcm.c:prepare_playback_urb(), the isochronous packet descriptors are
initially set up in the same way no matter if the quirk is enabled or not.
Once it is time to actually copy the data into the outgoing packet buffer
(together with the added length descriptors) the isochronous descriptors
are adjusted in order take the increased payload length into account.
For endpoint.c:prepare_silent_urb() it makes more sense to modify the
actual function, partly because the function is less complex to start with
and partly because it is not as time-critical as prepare_playback_urb()
(whose bulk is run with interrupts disabled), so the (minute) additional
time spent in the non-quirk case is motivated by the simplicity of having
a single function for all cases.
The quirk is controlled by the new tx_length_quirk member in struct
snd_usb_substream and struct snd_usb_audio, which is conveyed to pcm.c
and endpoint.c from quirks.c in a similar manner to the txfr_quirk member
in the same structs.
In contrast to txfr_quirk however, the quirk is enabled directly in
quirks.c:create_standard_audio_quirk() by checking the USB ID in that
function. Another option would be to introduce a new
QUIRK_AUDIO_ZOOM_INTERFACE or somesuch, which would have made the quirk
very plain to see in the quirk table, but it was felt that the additional
code needed to implement it this way would just make the implementation
more complex with no real gain.
Tested with a Zoom R16, both by doing capture and playback separately
using arecord and aplay (8 channel capture and 2 channel playback,
respectively), as well as capture and playback together using Ardour, as
well as Audacity and Qtractor together with jackd.
The R24 is reportedly compatible with the R16 when used as an audio
interface. Both devices share the same USB ID and have the same number of
inputs (8) and outputs (2). Therefore "R16/24" is mentioned throughout the
patch.
Regression tested using an Edirol UA-5 in both class compliant (16-bit)
and "advanced" (24 bit, forces the use of quirks) modes.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Tested-by: Panu Matilainen <pmatilai@laiskiainen.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-19 09:52:53 +03:00
urb - > transfer_buffer_length = offs * ep - > stride + ctx - > packets * extra ;
2015-10-19 09:52:51 +03:00
}
2012-04-12 15:51:11 +04:00
/*
* Prepare a PLAYBACK urb for submission to the bus .
*/
static void prepare_outbound_urb ( struct snd_usb_endpoint * ep ,
struct snd_urb_ctx * ctx )
{
struct urb * urb = ctx - > urb ;
unsigned char * cp = urb - > transfer_buffer ;
urb - > dev = ep - > chip - > dev ; /* we need to set this at each time */
switch ( ep - > type ) {
case SND_USB_ENDPOINT_TYPE_DATA :
if ( ep - > prepare_data_urb ) {
ep - > prepare_data_urb ( ep - > data_subs , urb ) ;
} else {
/* no data provider, so send silence */
2015-10-19 09:52:51 +03:00
prepare_silent_urb ( ep , ctx ) ;
2012-04-12 15:51:11 +04:00
}
break ;
case SND_USB_ENDPOINT_TYPE_SYNC :
if ( snd_usb_get_speed ( ep - > chip - > dev ) > = USB_SPEED_HIGH ) {
/*
* fill the length and offset of each urb descriptor .
* the fixed 12.13 frequency is passed as 16.16 through the pipe .
*/
urb - > iso_frame_desc [ 0 ] . length = 4 ;
urb - > iso_frame_desc [ 0 ] . offset = 0 ;
cp [ 0 ] = ep - > freqn ;
cp [ 1 ] = ep - > freqn > > 8 ;
cp [ 2 ] = ep - > freqn > > 16 ;
cp [ 3 ] = ep - > freqn > > 24 ;
} else {
/*
* fill the length and offset of each urb descriptor .
* the fixed 10.14 frequency is passed through the pipe .
*/
urb - > iso_frame_desc [ 0 ] . length = 3 ;
urb - > iso_frame_desc [ 0 ] . offset = 0 ;
cp [ 0 ] = ep - > freqn > > 2 ;
cp [ 1 ] = ep - > freqn > > 10 ;
cp [ 2 ] = ep - > freqn > > 18 ;
}
break ;
}
}
/*
* Prepare a CAPTURE or SYNC urb for submission to the bus .
*/
static inline void prepare_inbound_urb ( struct snd_usb_endpoint * ep ,
struct snd_urb_ctx * urb_ctx )
{
int i , offs ;
struct urb * urb = urb_ctx - > urb ;
urb - > dev = ep - > chip - > dev ; /* we need to set this at each time */
switch ( ep - > type ) {
case SND_USB_ENDPOINT_TYPE_DATA :
offs = 0 ;
for ( i = 0 ; i < urb_ctx - > packets ; i + + ) {
urb - > iso_frame_desc [ i ] . offset = offs ;
urb - > iso_frame_desc [ i ] . length = ep - > curpacksize ;
offs + = ep - > curpacksize ;
}
urb - > transfer_buffer_length = offs ;
urb - > number_of_packets = urb_ctx - > packets ;
break ;
case SND_USB_ENDPOINT_TYPE_SYNC :
urb - > iso_frame_desc [ 0 ] . length = min ( 4u , ep - > syncmaxsize ) ;
urb - > iso_frame_desc [ 0 ] . offset = 0 ;
break ;
}
}
2012-04-12 15:51:15 +04:00
/*
2012-04-24 21:31:24 +04:00
* Send output urbs that have been prepared previously . URBs are dequeued
2020-10-05 22:12:44 +03:00
* from ep - > ready_playback_urbs and in case there aren ' t any available
2012-04-12 15:51:15 +04:00
* or there are no packets that have been prepared , this function does
* nothing .
*
2012-04-24 21:31:24 +04:00
* The reason why the functionality of sending and preparing URBs is separated
* is that host controllers don ' t guarantee the order in which they return
* inbound and outbound packets to their submitters .
2012-04-12 15:51:15 +04:00
*
* This function is only used for implicit feedback endpoints . For endpoints
2012-04-24 21:31:24 +04:00
* driven by dedicated sync endpoints , URBs are immediately re - submitted
* from their completion handler .
2012-04-12 15:51:15 +04:00
*/
2012-04-12 15:51:11 +04:00
static void queue_pending_output_urbs ( struct snd_usb_endpoint * ep )
{
while ( test_bit ( EP_FLAG_RUNNING , & ep - > flags ) ) {
unsigned long flags ;
treewide: Remove uninitialized_var() usage
Using uninitialized_var() is dangerous as it papers over real bugs[1]
(or can in the future), and suppresses unrelated compiler warnings
(e.g. "unused variable"). If the compiler thinks it is uninitialized,
either simply initialize the variable or make compiler changes.
In preparation for removing[2] the[3] macro[4], remove all remaining
needless uses with the following script:
git grep '\buninitialized_var\b' | cut -d: -f1 | sort -u | \
xargs perl -pi -e \
's/\buninitialized_var\(([^\)]+)\)/\1/g;
s:\s*/\* (GCC be quiet|to make compiler happy) \*/$::g;'
drivers/video/fbdev/riva/riva_hw.c was manually tweaked to avoid
pathological white-space.
No outstanding warnings were found building allmodconfig with GCC 9.3.0
for x86_64, i386, arm64, arm, powerpc, powerpc64le, s390x, mips, sparc64,
alpha, and m68k.
[1] https://lore.kernel.org/lkml/20200603174714.192027-1-glider@google.com/
[2] https://lore.kernel.org/lkml/CA+55aFw+Vbj0i=1TGqCR5vQkCzWJ0QxK6CernOU6eedsudAixw@mail.gmail.com/
[3] https://lore.kernel.org/lkml/CA+55aFwgbgqhbp1fkxvRKEpzyR5J8n1vKT1VZdz9knmPuXhOeg@mail.gmail.com/
[4] https://lore.kernel.org/lkml/CA+55aFz2500WfbKXAx8s67wrm9=yVJu65TpLgN_ybYNv0VEOKA@mail.gmail.com/
Reviewed-by: Leon Romanovsky <leonro@mellanox.com> # drivers/infiniband and mlx4/mlx5
Acked-by: Jason Gunthorpe <jgg@mellanox.com> # IB
Acked-by: Kalle Valo <kvalo@codeaurora.org> # wireless drivers
Reviewed-by: Chao Yu <yuchao0@huawei.com> # erofs
Signed-off-by: Kees Cook <keescook@chromium.org>
2020-06-03 23:09:38 +03:00
struct snd_usb_packet_info * packet ;
2012-04-12 15:51:11 +04:00
struct snd_urb_ctx * ctx = NULL ;
int err , i ;
spin_lock_irqsave ( & ep - > lock , flags ) ;
if ( ep - > next_packet_read_pos ! = ep - > next_packet_write_pos ) {
packet = ep - > next_packet + ep - > next_packet_read_pos ;
ep - > next_packet_read_pos + + ;
ep - > next_packet_read_pos % = MAX_URBS ;
/* take URB out of FIFO */
2020-04-24 10:40:16 +03:00
if ( ! list_empty ( & ep - > ready_playback_urbs ) ) {
2012-04-12 15:51:11 +04:00
ctx = list_first_entry ( & ep - > ready_playback_urbs ,
struct snd_urb_ctx , ready_list ) ;
2020-04-24 10:40:16 +03:00
list_del_init ( & ctx - > ready_list ) ;
}
2012-04-12 15:51:11 +04:00
}
spin_unlock_irqrestore ( & ep - > lock , flags ) ;
if ( ctx = = NULL )
return ;
/* copy over the length information */
for ( i = 0 ; i < packet - > packets ; i + + )
ctx - > packet_size [ i ] = packet - > packet_size [ i ] ;
2012-04-12 15:51:15 +04:00
/* call the data handler to fill in playback data */
2012-04-12 15:51:11 +04:00
prepare_outbound_urb ( ep , ctx ) ;
err = usb_submit_urb ( ctx - > urb , GFP_ATOMIC ) ;
if ( err < 0 )
2014-02-26 16:02:17 +04:00
usb_audio_err ( ep - > chip ,
2020-11-23 11:53:13 +03:00
" Unable to submit urb #%d: %d at %s \n " ,
ctx - > index , err , __func__ ) ;
2012-04-12 15:51:11 +04:00
else
set_bit ( ctx - > index , & ep - > active_mask ) ;
}
}
/*
* complete callback for urbs
*/
static void snd_complete_urb ( struct urb * urb )
{
struct snd_urb_ctx * ctx = urb - > context ;
struct snd_usb_endpoint * ep = ctx - > ep ;
2014-11-06 15:04:49 +03:00
struct snd_pcm_substream * substream ;
unsigned long flags ;
2012-04-12 15:51:11 +04:00
int err ;
if ( unlikely ( urb - > status = = - ENOENT | | /* unlinked */
urb - > status = = - ENODEV | | /* device removed */
urb - > status = = - ECONNRESET | | /* unlinked */
ALSA: usb-audio: Avoid nested autoresume calls
After the recent fix of runtime PM for USB-audio driver, we got a
lockdep warning like:
=============================================
[ INFO: possible recursive locking detected ]
4.2.0-rc8+ #61 Not tainted
---------------------------------------------
pulseaudio/980 is trying to acquire lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
but task is already holding lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
This comes from snd_usb_autoresume() invoking down_read() and it's
used in a nested way. Although it's basically safe, per se (as these
are read locks), it's better to reduce such spurious warnings.
The read lock is needed to guarantee the execution of "shutdown"
(cleanup at disconnection) task after all concurrent tasks are
finished. This can be implemented in another better way.
Also, the current check of chip->in_pm isn't good enough for
protecting the racy execution of multiple auto-resumes.
This patch rewrites the logic of snd_usb_autoresume() & co; namely,
- The recursive call of autopm is avoided by the new refcount,
chip->active. The chip->in_pm flag is removed accordingly.
- Instead of rwsem, another refcount, chip->usage_count, is introduced
for tracking the period to delay the shutdown procedure. At
the last clear of this refcount, wake_up() to the shutdown waiter is
called.
- The shutdown flag is replaced with shutdown atomic count; this is
for reducing the lock.
- Two new helpers are introduced to simplify the management of these
refcounts; snd_usb_lock_shutdown() increases the usage_count, checks
the shutdown state, and does autoresume. snd_usb_unlock_shutdown()
does the opposite. Most of mixer and other codes just need this,
and simply returns an error if it receives an error from lock.
Fixes: 9003ebb13f61 ('ALSA: usb-audio: Fix runtime PM unbalance')
Reported-and-tested-by: Alexnader Kuleshov <kuleshovmail@gmail.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-08-25 17:09:00 +03:00
urb - > status = = - ESHUTDOWN ) ) /* device disabled */
goto exit_clear ;
/* device disconnected */
if ( unlikely ( atomic_read ( & ep - > chip - > shutdown ) ) )
2012-04-12 15:51:11 +04:00
goto exit_clear ;
2017-01-05 01:37:47 +03:00
if ( unlikely ( ! test_bit ( EP_FLAG_RUNNING , & ep - > flags ) ) )
goto exit_clear ;
2012-04-12 15:51:11 +04:00
if ( usb_pipeout ( ep - > pipe ) ) {
retire_outbound_urb ( ep , ctx ) ;
/* can be stopped during retire callback */
if ( unlikely ( ! test_bit ( EP_FLAG_RUNNING , & ep - > flags ) ) )
goto exit_clear ;
2013-04-04 01:18:52 +04:00
if ( snd_usb_endpoint_implicit_feedback_sink ( ep ) ) {
2012-04-12 15:51:11 +04:00
spin_lock_irqsave ( & ep - > lock , flags ) ;
list_add_tail ( & ctx - > ready_list , & ep - > ready_playback_urbs ) ;
spin_unlock_irqrestore ( & ep - > lock , flags ) ;
queue_pending_output_urbs ( ep ) ;
goto exit_clear ;
}
prepare_outbound_urb ( ep , ctx ) ;
2019-11-13 05:14:19 +03:00
/* can be stopped during prepare callback */
if ( unlikely ( ! test_bit ( EP_FLAG_RUNNING , & ep - > flags ) ) )
goto exit_clear ;
2012-04-12 15:51:11 +04:00
} else {
retire_inbound_urb ( ep , ctx ) ;
/* can be stopped during retire callback */
if ( unlikely ( ! test_bit ( EP_FLAG_RUNNING , & ep - > flags ) ) )
goto exit_clear ;
prepare_inbound_urb ( ep , ctx ) ;
}
err = usb_submit_urb ( urb , GFP_ATOMIC ) ;
if ( err = = 0 )
return ;
2014-02-26 16:02:17 +04:00
usb_audio_err ( ep - > chip , " cannot submit urb (err = %d) \n " , err ) ;
2014-11-06 15:04:49 +03:00
if ( ep - > data_subs & & ep - > data_subs - > pcm_substream ) {
substream = ep - > data_subs - > pcm_substream ;
2014-11-07 19:08:28 +03:00
snd_pcm_stop_xrun ( substream ) ;
2014-11-06 15:04:49 +03:00
}
2012-04-12 15:51:11 +04:00
exit_clear :
clear_bit ( ctx - > index , & ep - > active_mask ) ;
}
2020-11-23 11:53:11 +03:00
/*
* Get the existing endpoint object corresponding EP , iface and alt numbers
* Returns NULL if not present .
* Call inside chip - > mutex locking for avoiding the race .
*/
struct snd_usb_endpoint *
snd_usb_get_endpoint ( struct snd_usb_audio * chip ,
int ep_num , int iface , int altsetting )
{
struct snd_usb_endpoint * ep ;
list_for_each_entry ( ep , & chip - > ep_list , list ) {
if ( ep - > ep_num = = ep_num & &
ep - > iface = = iface & &
ep - > altsetting = = altsetting )
return ep ;
}
return NULL ;
}
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
# define ep_type_name(type) \
( type = = SND_USB_ENDPOINT_TYPE_DATA ? " data " : " sync " )
2012-04-12 15:51:15 +04:00
/**
2012-04-24 21:31:24 +04:00
* snd_usb_add_endpoint : Add an endpoint to an USB audio chip
2012-04-12 15:51:15 +04:00
*
* @ chip : The chip
* @ alts : The USB host interface
* @ ep_num : The number of the endpoint to use
* @ direction : SNDRV_PCM_STREAM_PLAYBACK or SNDRV_PCM_STREAM_CAPTURE
* @ type : SND_USB_ENDPOINT_TYPE_DATA or SND_USB_ENDPOINT_TYPE_SYNC
*
* If the requested endpoint has not been added to the given chip before ,
* a new instance is created . Otherwise , a pointer to the previoulsy
* created instance is returned . In case of any error , NULL is returned .
*
* New endpoints will be added to chip - > ep_list and must be freed by
* calling snd_usb_endpoint_free ( ) .
2016-03-15 17:20:58 +03:00
*
* For SND_USB_ENDPOINT_TYPE_SYNC , the caller needs to guarantee that
* bNumEndpoints > 1 beforehand .
2012-04-12 15:51:15 +04:00
*/
2012-04-12 15:51:11 +04:00
struct snd_usb_endpoint * snd_usb_add_endpoint ( struct snd_usb_audio * chip ,
struct usb_host_interface * alts ,
int ep_num , int direction , int type )
{
struct snd_usb_endpoint * ep ;
2012-07-12 15:08:40 +04:00
int is_playback = direction = = SNDRV_PCM_STREAM_PLAYBACK ;
2012-04-12 15:51:11 +04:00
2013-08-03 12:51:15 +04:00
if ( WARN_ON ( ! alts ) )
return NULL ;
2012-04-12 15:51:11 +04:00
mutex_lock ( & chip - > mutex ) ;
2020-11-23 11:53:11 +03:00
ep = snd_usb_get_endpoint ( chip , ep_num ,
alts - > desc . bInterfaceNumber ,
alts - > desc . bAlternateSetting ) ;
if ( ep ) {
2020-11-23 11:53:13 +03:00
usb_audio_dbg ( ep - > chip , " Re-using EP %x in iface %d,%d \n " ,
ep_num , ep - > iface , ep - > altsetting ) ;
2020-11-23 11:53:11 +03:00
goto __exit_unlock ;
2012-04-12 15:51:11 +04:00
}
2014-02-26 16:02:17 +04:00
usb_audio_dbg ( chip , " Creating new %s %s endpoint #%x \n " ,
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
is_playback ? " playback " : " capture " ,
ep_type_name ( type ) ,
ep_num ) ;
2012-04-12 15:51:11 +04:00
ep = kzalloc ( sizeof ( * ep ) , GFP_KERNEL ) ;
if ( ! ep )
goto __exit_unlock ;
ep - > chip = chip ;
spin_lock_init ( & ep - > lock ) ;
ep - > type = type ;
ep - > ep_num = ep_num ;
ep - > iface = alts - > desc . bInterfaceNumber ;
2013-10-07 00:31:13 +04:00
ep - > altsetting = alts - > desc . bAlternateSetting ;
2012-04-12 15:51:11 +04:00
INIT_LIST_HEAD ( & ep - > ready_playback_urbs ) ;
ep_num & = USB_ENDPOINT_NUMBER_MASK ;
if ( is_playback )
ep - > pipe = usb_sndisocpipe ( chip - > dev , ep_num ) ;
else
ep - > pipe = usb_rcvisocpipe ( chip - > dev , ep_num ) ;
if ( type = = SND_USB_ENDPOINT_TYPE_SYNC ) {
if ( get_endpoint ( alts , 1 ) - > bLength > = USB_DT_ENDPOINT_AUDIO_SIZE & &
get_endpoint ( alts , 1 ) - > bRefresh > = 1 & &
get_endpoint ( alts , 1 ) - > bRefresh < = 9 )
ep - > syncinterval = get_endpoint ( alts , 1 ) - > bRefresh ;
else if ( snd_usb_get_speed ( chip - > dev ) = = USB_SPEED_FULL )
ep - > syncinterval = 1 ;
else if ( get_endpoint ( alts , 1 ) - > bInterval > = 1 & &
get_endpoint ( alts , 1 ) - > bInterval < = 16 )
ep - > syncinterval = get_endpoint ( alts , 1 ) - > bInterval - 1 ;
else
ep - > syncinterval = 3 ;
ep - > syncmaxsize = le16_to_cpu ( get_endpoint ( alts , 1 ) - > wMaxPacketSize ) ;
}
list_add_tail ( & ep - > list , & chip - > ep_list ) ;
2020-05-10 21:29:11 +03:00
ep - > is_implicit_feedback = 0 ;
2012-04-12 15:51:11 +04:00
__exit_unlock :
mutex_unlock ( & chip - > mutex ) ;
return ep ;
}
/*
* wait until all urbs are processed .
*/
static int wait_clear_urbs ( struct snd_usb_endpoint * ep )
{
unsigned long end_time = jiffies + msecs_to_jiffies ( 1000 ) ;
int alive ;
do {
2012-11-17 11:35:16 +04:00
alive = bitmap_weight ( & ep - > active_mask , ep - > nurbs ) ;
2012-04-12 15:51:11 +04:00
if ( ! alive )
break ;
schedule_timeout_uninterruptible ( 1 ) ;
} while ( time_before ( jiffies , end_time ) ) ;
if ( alive )
2014-02-26 16:02:17 +04:00
usb_audio_err ( ep - > chip ,
" timeout: still %d active urbs on EP #%x \n " ,
alive , ep - > ep_num ) ;
2012-11-08 11:52:45 +04:00
clear_bit ( EP_FLAG_STOPPING , & ep - > flags ) ;
2012-04-12 15:51:11 +04:00
2017-01-05 01:37:46 +03:00
ep - > data_subs = NULL ;
ep - > sync_slave = NULL ;
ep - > retire_data_urb = NULL ;
ep - > prepare_data_urb = NULL ;
2012-04-12 15:51:11 +04:00
return 0 ;
}
2012-11-08 11:52:45 +04:00
/* sync the pending stop operation;
* this function itself doesn ' t trigger the stop operation
*/
void snd_usb_endpoint_sync_pending_stop ( struct snd_usb_endpoint * ep )
{
if ( ep & & test_bit ( EP_FLAG_STOPPING , & ep - > flags ) )
wait_clear_urbs ( ep ) ;
}
2012-04-12 15:51:11 +04:00
/*
* unlink active urbs .
*/
2012-11-21 11:22:52 +04:00
static int deactivate_urbs ( struct snd_usb_endpoint * ep , bool force )
2012-04-12 15:51:11 +04:00
{
unsigned int i ;
ALSA: usb-audio: Avoid nested autoresume calls
After the recent fix of runtime PM for USB-audio driver, we got a
lockdep warning like:
=============================================
[ INFO: possible recursive locking detected ]
4.2.0-rc8+ #61 Not tainted
---------------------------------------------
pulseaudio/980 is trying to acquire lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
but task is already holding lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
This comes from snd_usb_autoresume() invoking down_read() and it's
used in a nested way. Although it's basically safe, per se (as these
are read locks), it's better to reduce such spurious warnings.
The read lock is needed to guarantee the execution of "shutdown"
(cleanup at disconnection) task after all concurrent tasks are
finished. This can be implemented in another better way.
Also, the current check of chip->in_pm isn't good enough for
protecting the racy execution of multiple auto-resumes.
This patch rewrites the logic of snd_usb_autoresume() & co; namely,
- The recursive call of autopm is avoided by the new refcount,
chip->active. The chip->in_pm flag is removed accordingly.
- Instead of rwsem, another refcount, chip->usage_count, is introduced
for tracking the period to delay the shutdown procedure. At
the last clear of this refcount, wake_up() to the shutdown waiter is
called.
- The shutdown flag is replaced with shutdown atomic count; this is
for reducing the lock.
- Two new helpers are introduced to simplify the management of these
refcounts; snd_usb_lock_shutdown() increases the usage_count, checks
the shutdown state, and does autoresume. snd_usb_unlock_shutdown()
does the opposite. Most of mixer and other codes just need this,
and simply returns an error if it receives an error from lock.
Fixes: 9003ebb13f61 ('ALSA: usb-audio: Fix runtime PM unbalance')
Reported-and-tested-by: Alexnader Kuleshov <kuleshovmail@gmail.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-08-25 17:09:00 +03:00
if ( ! force & & atomic_read ( & ep - > chip - > shutdown ) ) /* to be sure... */
2012-04-12 15:51:11 +04:00
return - EBADFD ;
clear_bit ( EP_FLAG_RUNNING , & ep - > flags ) ;
INIT_LIST_HEAD ( & ep - > ready_playback_urbs ) ;
ep - > next_packet_read_pos = 0 ;
ep - > next_packet_write_pos = 0 ;
for ( i = 0 ; i < ep - > nurbs ; i + + ) {
if ( test_bit ( i , & ep - > active_mask ) ) {
if ( ! test_and_set_bit ( i , & ep - > unlink_mask ) ) {
struct urb * u = ep - > urb [ i ] . urb ;
2012-11-21 11:22:52 +04:00
usb_unlink_urb ( u ) ;
2012-04-12 15:51:11 +04:00
}
}
}
return 0 ;
}
/*
* release an endpoint ' s urbs
*/
static void release_urbs ( struct snd_usb_endpoint * ep , int force )
{
int i ;
/* route incoming urbs to nirvana */
ep - > retire_data_urb = NULL ;
ep - > prepare_data_urb = NULL ;
/* stop urbs */
2012-11-21 11:22:52 +04:00
deactivate_urbs ( ep , force ) ;
2012-04-12 15:51:11 +04:00
wait_clear_urbs ( ep ) ;
for ( i = 0 ; i < ep - > nurbs ; i + + )
release_urb_ctx ( & ep - > urb [ i ] ) ;
2020-07-27 05:52:08 +03:00
usb_free_coherent ( ep - > chip - > dev , SYNC_URBS * 4 ,
ep - > syncbuf , ep - > sync_dma ) ;
2012-04-12 15:51:11 +04:00
ep - > syncbuf = NULL ;
ep - > nurbs = 0 ;
}
2020-05-10 21:29:11 +03:00
/*
* Check data endpoint for format differences
*/
static bool check_ep_params ( struct snd_usb_endpoint * ep ,
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
snd_pcm_format_t pcm_format ,
unsigned int channels ,
unsigned int period_bytes ,
unsigned int frames_per_period ,
unsigned int periods_per_buffer ,
unsigned int rate ,
struct audioformat * fmt ,
struct snd_usb_endpoint * sync_ep )
2020-05-10 21:29:11 +03:00
{
unsigned int maxsize , minsize , packs_per_ms , max_packs_per_urb ;
unsigned int max_packs_per_period , urbs_per_period , urb_packs ;
unsigned int max_urbs ;
int frame_bits = snd_pcm_format_physical_width ( pcm_format ) * channels ;
int tx_length_quirk = ( ep - > chip - > tx_length_quirk & &
usb_pipeout ( ep - > pipe ) ) ;
bool ret = 1 ;
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
/* matching with the saved parameters? */
if ( ep - > cur_rate = = rate & &
ep - > cur_format = = pcm_format & &
ep - > cur_channels = = channels & &
ep - > cur_period_frames = = frames_per_period & &
ep - > cur_buffer_periods = = periods_per_buffer )
return true ;
2020-05-10 21:29:11 +03:00
if ( pcm_format = = SNDRV_PCM_FORMAT_DSD_U16_LE & & fmt - > dsd_dop ) {
/*
* When operating in DSD DOP mode , the size of a sample frame
* in hardware differs from the actual physical format width
* because we need to make room for the DOP markers .
*/
frame_bits + = channels < < 3 ;
}
ret = ret & & ( ep - > datainterval = = fmt - > datainterval ) ;
ret = ret & & ( ep - > stride = = frame_bits > > 3 ) ;
switch ( pcm_format ) {
case SNDRV_PCM_FORMAT_U8 :
ret = ret & & ( ep - > silence_value = = 0x80 ) ;
break ;
case SNDRV_PCM_FORMAT_DSD_U8 :
case SNDRV_PCM_FORMAT_DSD_U16_LE :
case SNDRV_PCM_FORMAT_DSD_U32_LE :
case SNDRV_PCM_FORMAT_DSD_U16_BE :
case SNDRV_PCM_FORMAT_DSD_U32_BE :
ret = ret & & ( ep - > silence_value = = 0x69 ) ;
break ;
default :
ret = ret & & ( ep - > silence_value = = 0 ) ;
}
/* assume max. frequency is 50% higher than nominal */
ret = ret & & ( ep - > freqmax = = ep - > freqn + ( ep - > freqn > > 1 ) ) ;
/* Round up freqmax to nearest integer in order to calculate maximum
* packet size , which must represent a whole number of frames .
* This is accomplished by adding 0x0 . ffff before converting the
* Q16 .16 format into integer .
* In order to accurately calculate the maximum packet size when
* the data interval is more than 1 ( i . e . ep - > datainterval > 0 ) ,
* multiply by the data interval prior to rounding . For instance ,
* a freqmax of 41 kHz will result in a max packet size of 6 ( 5.125 )
* frames with a data interval of 1 , but 11 ( 10.25 ) frames with a
* data interval of 2.
* ( ep - > freqmax < < ep - > datainterval overflows at 8.192 MHz for the
* maximum datainterval value of 3 , at USB full speed , higher for
* USB high speed , noting that ep - > freqmax is in units of
* frames per packet in Q16 .16 format . )
*/
maxsize = ( ( ( ep - > freqmax < < ep - > datainterval ) + 0xffff ) > > 16 ) *
( frame_bits > > 3 ) ;
if ( tx_length_quirk )
maxsize + = sizeof ( __le32 ) ; /* Space for length descriptor */
/* but wMaxPacketSize might reduce this */
if ( ep - > maxpacksize & & ep - > maxpacksize < maxsize ) {
/* whatever fits into a max. size packet */
unsigned int data_maxsize = maxsize = ep - > maxpacksize ;
if ( tx_length_quirk )
/* Need to remove the length descriptor to calc freq */
data_maxsize - = sizeof ( __le32 ) ;
ret = ret & & ( ep - > freqmax = = ( data_maxsize / ( frame_bits > > 3 ) )
< < ( 16 - ep - > datainterval ) ) ;
}
if ( ep - > fill_max )
ret = ret & & ( ep - > curpacksize = = ep - > maxpacksize ) ;
else
ret = ret & & ( ep - > curpacksize = = maxsize ) ;
if ( snd_usb_get_speed ( ep - > chip - > dev ) ! = USB_SPEED_FULL ) {
packs_per_ms = 8 > > ep - > datainterval ;
max_packs_per_urb = MAX_PACKS_HS ;
} else {
packs_per_ms = 1 ;
max_packs_per_urb = MAX_PACKS ;
}
if ( sync_ep & & ! snd_usb_endpoint_implicit_feedback_sink ( ep ) )
max_packs_per_urb = min ( max_packs_per_urb ,
1U < < sync_ep - > syncinterval ) ;
max_packs_per_urb = max ( 1u , max_packs_per_urb > > ep - > datainterval ) ;
/*
* Capture endpoints need to use small URBs because there ' s no way
* to tell in advance where the next period will end , and we don ' t
* want the next URB to complete much after the period ends .
*
* Playback endpoints with implicit sync much use the same parameters
* as their corresponding capture endpoint .
*/
if ( usb_pipein ( ep - > pipe ) | |
snd_usb_endpoint_implicit_feedback_sink ( ep ) ) {
urb_packs = packs_per_ms ;
/*
* Wireless devices can poll at a max rate of once per 4 ms .
* For dataintervals less than 5 , increase the packet count to
* allow the host controller to use bursting to fill in the
* gaps .
*/
if ( snd_usb_get_speed ( ep - > chip - > dev ) = = USB_SPEED_WIRELESS ) {
int interval = ep - > datainterval ;
while ( interval < 5 ) {
urb_packs < < = 1 ;
+ + interval ;
}
}
/* make capture URBs <= 1 ms and smaller than a period */
urb_packs = min ( max_packs_per_urb , urb_packs ) ;
while ( urb_packs > 1 & & urb_packs * maxsize > = period_bytes )
urb_packs > > = 1 ;
ret = ret & & ( ep - > nurbs = = MAX_URBS ) ;
/*
* Playback endpoints without implicit sync are adjusted so that
* a period fits as evenly as possible in the smallest number of
* URBs . The total number of URBs is adjusted to the size of the
* ALSA buffer , subject to the MAX_URBS and MAX_QUEUE limits .
*/
} else {
/* determine how small a packet can be */
minsize = ( ep - > freqn > > ( 16 - ep - > datainterval ) ) *
( frame_bits > > 3 ) ;
/* with sync from device, assume it can be 12% lower */
if ( sync_ep )
minsize - = minsize > > 3 ;
minsize = max ( minsize , 1u ) ;
/* how many packets will contain an entire ALSA period? */
max_packs_per_period = DIV_ROUND_UP ( period_bytes , minsize ) ;
/* how many URBs will contain a period? */
urbs_per_period = DIV_ROUND_UP ( max_packs_per_period ,
max_packs_per_urb ) ;
/* how many packets are needed in each URB? */
urb_packs = DIV_ROUND_UP ( max_packs_per_period , urbs_per_period ) ;
/* limit the number of frames in a single URB */
ret = ret & & ( ep - > max_urb_frames = =
DIV_ROUND_UP ( frames_per_period , urbs_per_period ) ) ;
/* try to use enough URBs to contain an entire ALSA buffer */
max_urbs = min ( ( unsigned ) MAX_URBS ,
MAX_QUEUE * packs_per_ms / urb_packs ) ;
ret = ret & & ( ep - > nurbs = = min ( max_urbs ,
urbs_per_period * periods_per_buffer ) ) ;
}
ret = ret & & ( ep - > datainterval = = fmt - > datainterval ) ;
ret = ret & & ( ep - > maxpacksize = = fmt - > maxpacksize ) ;
ret = ret & &
( ep - > fill_max = = ! ! ( fmt - > attributes & UAC_EP_CS_ATTR_FILL_MAX ) ) ;
return ret ;
}
2012-04-12 15:51:15 +04:00
/*
* configure a data endpoint
*/
2012-04-12 15:51:11 +04:00
static int data_ep_set_params ( struct snd_usb_endpoint * ep ,
2012-09-18 20:49:47 +04:00
snd_pcm_format_t pcm_format ,
unsigned int channels ,
unsigned int period_bytes ,
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
unsigned int frames_per_period ,
unsigned int periods_per_buffer ,
2012-04-12 15:51:11 +04:00
struct audioformat * fmt ,
struct snd_usb_endpoint * sync_ep )
{
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
unsigned int maxsize , minsize , packs_per_ms , max_packs_per_urb ;
unsigned int max_packs_per_period , urbs_per_period , urb_packs ;
unsigned int max_urbs , i ;
2012-09-18 20:49:47 +04:00
int frame_bits = snd_pcm_format_physical_width ( pcm_format ) * channels ;
2015-10-19 09:52:54 +03:00
int tx_length_quirk = ( ep - > chip - > tx_length_quirk & &
usb_pipeout ( ep - > pipe ) ) ;
2012-04-12 15:51:11 +04:00
2013-04-16 20:01:38 +04:00
if ( pcm_format = = SNDRV_PCM_FORMAT_DSD_U16_LE & & fmt - > dsd_dop ) {
/*
* When operating in DSD DOP mode , the size of a sample frame
* in hardware differs from the actual physical format width
* because we need to make room for the DOP markers .
*/
frame_bits + = channels < < 3 ;
}
2012-04-12 15:51:11 +04:00
ep - > datainterval = fmt - > datainterval ;
ep - > stride = frame_bits > > 3 ;
2016-12-12 20:52:58 +03:00
switch ( pcm_format ) {
case SNDRV_PCM_FORMAT_U8 :
ep - > silence_value = 0x80 ;
break ;
case SNDRV_PCM_FORMAT_DSD_U8 :
case SNDRV_PCM_FORMAT_DSD_U16_LE :
case SNDRV_PCM_FORMAT_DSD_U32_LE :
case SNDRV_PCM_FORMAT_DSD_U16_BE :
case SNDRV_PCM_FORMAT_DSD_U32_BE :
ep - > silence_value = 0x69 ;
break ;
default :
ep - > silence_value = 0 ;
}
2012-04-12 15:51:11 +04:00
2016-12-06 08:46:14 +03:00
/* assume max. frequency is 50% higher than nominal */
ep - > freqmax = ep - > freqn + ( ep - > freqn > > 1 ) ;
ALSA: usb-audio: Fix max packet size calculation for USB audio
Rounding must take place before multiplication with the frame size, since
each packet contains a whole number of frames.
We must also properly consider the data interval, as a larger data
interval will result in larger packets, which, depending on the sampling
frequency, can result in packet sizes that are less than integral
multiples of the packet size for a lower data interval.
Detailed explanation and rationale:
The code before this commit had the following expression on line 613 to
calculate the maximum isochronous packet size:
maxsize = ((ep->freqmax + 0xffff) * (frame_bits >> 3))
>> (16 - ep->datainterval);
Here, ep->freqmax is the maximum assumed sample frequency, calculated from the
nominal sample frequency plus 25%. It is ultimately derived from ep->freqn,
which is in the units of frames per packet, from get_usb_full_speed_rate()
or usb_high_speed_rate(), as applicable, in Q16.16 format.
The expression essentially adds the Q16.16 equivalent of 0.999... (i.e.
the largest number less than one) to the sample rate, in order to get a
rate whose integer part is rounded up from the fractional value. The
multiplication with (frame_bits >> 3) yields the number of bytes in a
packet, and the (16 >> ep->datainterval) then converts it from Q16.16 back
to an integer, taking into consideration the bDataInterval field of the
endpoint descriptor (which describes how often isochronous packets are
transmitted relative to the (micro)frame rate (125us or 1ms, for USB high
speed and full speed, respectively)). For this discussion we will initially
assume a bDataInterval of 0, so the second line of the expression just
converts the Q16.16 value to an integer.
In order to illustrate the problem, we will set frame_bits 64, which
corresponds to a frame size of 8 bytes.
The problem here is twofold. First, the rounding operation consists
of the addition of 0x0.ffff and subsequent conversion to integer, but as the
expression stands, the conversion to integer is done after multiplication
with the frame size, rather than before. This results in the resulting
maxsize becoming too large.
Let's take an example. We have a sample rate of 96 kHz, so our ep->freqn is
0xc0000 (see usb_high_speed_rate()). Add 25% (line 612) and we get 0xf0000.
The calculated maxsize is then ((0xf0000 + 0x0ffff) * 8) >> 16 = 127 .
However, if we do the number of bytes calculation in a less obscure way it's
more apparent what the true corresponding packet size is: we get
ceil(96000 * 1.25 / 8000) * 8 = 120, where 1.25 is the 25% from line 612,
and the 8000 is the number of isochronous packets per second on a high
speed USB connection (125 us microframe interval).
This is fixed by performing the complete rounding operation prior to
multiplication with the frame rate.
The second problem is that when considering the ep->datainterval, this
must be done before rounding, in order to take the advantage of the fact
that if the number of bytes per packet is not an integer, the resulting
rounded-up integer is not necessarily a factor of two when the data
interval is increased by the same factor.
For instance, assuming a freqency of 41 kHz, the resulting
bytes-per-packet value for USB high speed is 41 kHz / 8000 = 5.125, or
0x52000 in Q16.16 format. With a data interval of 1 (ep->datainterval = 0),
this means that 6 frames per packet are needed, whereas with a data
interval of 2 we need 10.25, i.e. 11 frames needed.
Rephrasing the maxsize expression to:
maxsize = (((ep->freqmax << ep->datainterval) + 0xffff) >> 16) *
(frame_bits >> 3);
for the above 96 kHz example we instead get
((0xf0000 + 0xffff) >> 16) * 8 = 120 which is the correct value.
We can also do the calculation with a non-integer sample rate which is when
rounding comes into effect: say we have 44.1 kHz (resulting ep->freqn =
0x58333, and resulting ep->freqmax 0x58333 * 1.25 = 0x6e3ff (rounded down)):
Original maxsize = ((0x6e3ff + 0xffff) * 8) << 16 = 63 (63.124.. rounded down)
True maxsize = ceil(44100 * 1.25 / 8000) * 8 = 7 * 8 = 56
New maxsize = ((0x6e3ff + 0xffff) >> 16) * 8 = 7 * 8 = 56
This is also corroborated by the wMaxPacketSize check on line 616. Assume
that wMaxPacketSize = 104, with ep->maxpacksize then having the same value.
As 104 < 127, we get maxsize = 104. ep->freqmax is then recalculated to
(104 / 8) << 16 = 0xd0000 . Putting that rate into the original maxsize
calculation yields a maxsize of ((0xd0000 + 0xffff) * 8) >> 16 = 111
(with decimals 111.99988). Clearly, we should get back the 104 here,
which we would with the new expression: ((0xd0000 + 0xffff) >> 16) * 8 = 104 .
(The error has not been a problem because it only results in maxsize being
a bit too big which just wastes a couple of bytes, either as a result of
the first maxsize calculation, or because the resulting calculation will
hit the wMaxPacketSize value before the packet is too big, resulting in
fixing the size to wMaxPacketSize even though the packet is actually not
too long.)
Tested with an Edirol UA-5 both at 44.1 kHz and 96 kHz.
Signed-off-by: Ricard Wanderlof <ricardw@axis.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-10-11 21:54:51 +03:00
/* Round up freqmax to nearest integer in order to calculate maximum
* packet size , which must represent a whole number of frames .
* This is accomplished by adding 0x0 . ffff before converting the
* Q16 .16 format into integer .
* In order to accurately calculate the maximum packet size when
* the data interval is more than 1 ( i . e . ep - > datainterval > 0 ) ,
* multiply by the data interval prior to rounding . For instance ,
* a freqmax of 41 kHz will result in a max packet size of 6 ( 5.125 )
* frames with a data interval of 1 , but 11 ( 10.25 ) frames with a
* data interval of 2.
* ( ep - > freqmax < < ep - > datainterval overflows at 8.192 MHz for the
* maximum datainterval value of 3 , at USB full speed , higher for
* USB high speed , noting that ep - > freqmax is in units of
* frames per packet in Q16 .16 format . )
*/
maxsize = ( ( ( ep - > freqmax < < ep - > datainterval ) + 0xffff ) > > 16 ) *
( frame_bits > > 3 ) ;
2015-10-19 09:52:54 +03:00
if ( tx_length_quirk )
maxsize + = sizeof ( __le32 ) ; /* Space for length descriptor */
2013-08-08 13:24:55 +04:00
/* but wMaxPacketSize might reduce this */
if ( ep - > maxpacksize & & ep - > maxpacksize < maxsize ) {
2012-04-12 15:51:11 +04:00
/* whatever fits into a max. size packet */
2015-10-19 09:52:54 +03:00
unsigned int data_maxsize = maxsize = ep - > maxpacksize ;
if ( tx_length_quirk )
/* Need to remove the length descriptor to calc freq */
data_maxsize - = sizeof ( __le32 ) ;
ep - > freqmax = ( data_maxsize / ( frame_bits > > 3 ) )
2012-04-12 15:51:11 +04:00
< < ( 16 - ep - > datainterval ) ;
}
if ( ep - > fill_max )
ep - > curpacksize = ep - > maxpacksize ;
else
ep - > curpacksize = maxsize ;
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
if ( snd_usb_get_speed ( ep - > chip - > dev ) ! = USB_SPEED_FULL ) {
2012-04-12 15:51:11 +04:00
packs_per_ms = 8 > > ep - > datainterval ;
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
max_packs_per_urb = MAX_PACKS_HS ;
2012-04-12 15:51:11 +04:00
} else {
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
packs_per_ms = 1 ;
max_packs_per_urb = MAX_PACKS ;
2012-04-12 15:51:11 +04:00
}
2013-04-04 01:18:52 +04:00
if ( sync_ep & & ! snd_usb_endpoint_implicit_feedback_sink ( ep ) )
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
max_packs_per_urb = min ( max_packs_per_urb ,
1U < < sync_ep - > syncinterval ) ;
max_packs_per_urb = max ( 1u , max_packs_per_urb > > ep - > datainterval ) ;
/*
* Capture endpoints need to use small URBs because there ' s no way
* to tell in advance where the next period will end , and we don ' t
* want the next URB to complete much after the period ends .
*
* Playback endpoints with implicit sync much use the same parameters
* as their corresponding capture endpoint .
*/
if ( usb_pipein ( ep - > pipe ) | |
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
ep - > is_implicit_feedback | |
snd_usb_endpoint_implicit_feedback_sink ( ep ) ) {
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
2013-11-26 23:58:15 +04:00
urb_packs = packs_per_ms ;
/*
* Wireless devices can poll at a max rate of once per 4 ms .
* For dataintervals less than 5 , increase the packet count to
* allow the host controller to use bursting to fill in the
* gaps .
*/
if ( snd_usb_get_speed ( ep - > chip - > dev ) = = USB_SPEED_WIRELESS ) {
int interval = ep - > datainterval ;
while ( interval < 5 ) {
urb_packs < < = 1 ;
+ + interval ;
}
}
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
/* make capture URBs <= 1 ms and smaller than a period */
2013-11-26 23:58:15 +04:00
urb_packs = min ( max_packs_per_urb , urb_packs ) ;
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
while ( urb_packs > 1 & & urb_packs * maxsize > = period_bytes )
urb_packs > > = 1 ;
ep - > nurbs = MAX_URBS ;
2012-04-12 15:51:11 +04:00
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
/*
* Playback endpoints without implicit sync are adjusted so that
* a period fits as evenly as possible in the smallest number of
* URBs . The total number of URBs is adjusted to the size of the
* ALSA buffer , subject to the MAX_URBS and MAX_QUEUE limits .
*/
} else {
2012-04-12 15:51:11 +04:00
/* determine how small a packet can be */
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
minsize = ( ep - > freqn > > ( 16 - ep - > datainterval ) ) *
( frame_bits > > 3 ) ;
2012-04-12 15:51:11 +04:00
/* with sync from device, assume it can be 12% lower */
if ( sync_ep )
minsize - = minsize > > 3 ;
minsize = max ( minsize , 1u ) ;
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
/* how many packets will contain an entire ALSA period? */
max_packs_per_period = DIV_ROUND_UP ( period_bytes , minsize ) ;
/* how many URBs will contain a period? */
urbs_per_period = DIV_ROUND_UP ( max_packs_per_period ,
max_packs_per_urb ) ;
/* how many packets are needed in each URB? */
urb_packs = DIV_ROUND_UP ( max_packs_per_period , urbs_per_period ) ;
/* limit the number of frames in a single URB */
ep - > max_urb_frames = DIV_ROUND_UP ( frames_per_period ,
urbs_per_period ) ;
/* try to use enough URBs to contain an entire ALSA buffer */
max_urbs = min ( ( unsigned ) MAX_URBS ,
MAX_QUEUE * packs_per_ms / urb_packs ) ;
ep - > nurbs = min ( max_urbs , urbs_per_period * periods_per_buffer ) ;
2012-04-12 15:51:11 +04:00
}
/* allocate and initialize data urbs */
for ( i = 0 ; i < ep - > nurbs ; i + + ) {
struct snd_urb_ctx * u = & ep - > urb [ i ] ;
u - > index = i ;
u - > ep = ep ;
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
u - > packets = urb_packs ;
2012-04-12 15:51:11 +04:00
u - > buffer_size = maxsize * u - > packets ;
if ( fmt - > fmt_type = = UAC_FORMAT_TYPE_II )
u - > packets + + ; /* for transfer delimiter */
u - > urb = usb_alloc_urb ( u - > packets , GFP_KERNEL ) ;
if ( ! u - > urb )
goto out_of_memory ;
u - > urb - > transfer_buffer =
usb_alloc_coherent ( ep - > chip - > dev , u - > buffer_size ,
GFP_KERNEL , & u - > urb - > transfer_dma ) ;
if ( ! u - > urb - > transfer_buffer )
goto out_of_memory ;
u - > urb - > pipe = ep - > pipe ;
2013-04-27 14:10:32 +04:00
u - > urb - > transfer_flags = URB_NO_TRANSFER_DMA_MAP ;
2012-04-12 15:51:11 +04:00
u - > urb - > interval = 1 < < ep - > datainterval ;
u - > urb - > context = u ;
u - > urb - > complete = snd_complete_urb ;
INIT_LIST_HEAD ( & u - > ready_list ) ;
}
return 0 ;
out_of_memory :
release_urbs ( ep , 0 ) ;
return - ENOMEM ;
}
2012-04-12 15:51:15 +04:00
/*
* configure a sync endpoint
*/
2013-10-07 00:31:06 +04:00
static int sync_ep_set_params ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
int i ;
ep - > syncbuf = usb_alloc_coherent ( ep - > chip - > dev , SYNC_URBS * 4 ,
GFP_KERNEL , & ep - > sync_dma ) ;
if ( ! ep - > syncbuf )
return - ENOMEM ;
for ( i = 0 ; i < SYNC_URBS ; i + + ) {
struct snd_urb_ctx * u = & ep - > urb [ i ] ;
u - > index = i ;
u - > ep = ep ;
u - > packets = 1 ;
u - > urb = usb_alloc_urb ( 1 , GFP_KERNEL ) ;
if ( ! u - > urb )
goto out_of_memory ;
u - > urb - > transfer_buffer = ep - > syncbuf + i * 4 ;
u - > urb - > transfer_dma = ep - > sync_dma + i * 4 ;
u - > urb - > transfer_buffer_length = 4 ;
u - > urb - > pipe = ep - > pipe ;
2013-04-27 14:10:32 +04:00
u - > urb - > transfer_flags = URB_NO_TRANSFER_DMA_MAP ;
2012-04-12 15:51:11 +04:00
u - > urb - > number_of_packets = 1 ;
u - > urb - > interval = 1 < < ep - > syncinterval ;
u - > urb - > context = u ;
u - > urb - > complete = snd_complete_urb ;
}
ep - > nurbs = SYNC_URBS ;
return 0 ;
out_of_memory :
release_urbs ( ep , 0 ) ;
return - ENOMEM ;
}
2012-04-12 15:51:15 +04:00
/**
2012-04-24 21:31:24 +04:00
* snd_usb_endpoint_set_params : configure an snd_usb_endpoint
2012-04-12 15:51:15 +04:00
*
2012-04-24 21:31:24 +04:00
* @ ep : the snd_usb_endpoint to configure
2012-09-18 20:49:47 +04:00
* @ pcm_format : the audio fomat .
* @ channels : the number of audio channels .
* @ period_bytes : the number of bytes in one alsa period .
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
* @ period_frames : the number of frames in one alsa period .
* @ buffer_periods : the number of periods in one alsa buffer .
2012-09-18 20:49:47 +04:00
* @ rate : the frame rate .
2012-04-24 21:31:24 +04:00
* @ fmt : the USB audio format information
* @ sync_ep : the sync endpoint to use , if any
2012-04-12 15:51:15 +04:00
*
2012-04-24 21:31:24 +04:00
* Determine the number of URBs to be used on this endpoint .
2012-04-12 15:51:15 +04:00
* An endpoint must be configured before it can be started .
* An endpoint that is already running can not be reconfigured .
*/
2012-04-12 15:51:11 +04:00
int snd_usb_endpoint_set_params ( struct snd_usb_endpoint * ep ,
2012-09-18 20:49:47 +04:00
snd_pcm_format_t pcm_format ,
unsigned int channels ,
unsigned int period_bytes ,
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
unsigned int period_frames ,
unsigned int buffer_periods ,
2012-09-18 20:49:47 +04:00
unsigned int rate ,
2012-04-12 15:51:11 +04:00
struct audioformat * fmt ,
struct snd_usb_endpoint * sync_ep )
{
int err ;
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
usb_audio_dbg ( ep - > chip ,
" Setting params for ep %x (type %s, count %d), rate=%d, format=%s, channels=%d, period_bytes=%d, periods=%d \n " ,
ep - > ep_num , ep_type_name ( ep - > type ) , ep - > use_count ,
rate , snd_pcm_format_name ( pcm_format ) , channels ,
period_bytes , buffer_periods ) ;
2012-04-12 15:51:11 +04:00
if ( ep - > use_count ! = 0 ) {
2020-05-10 21:29:11 +03:00
bool check = ep - > is_implicit_feedback & &
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
check_ep_params ( ep , pcm_format , channels , period_bytes ,
period_frames , buffer_periods , rate ,
fmt , sync_ep ) ;
2020-05-10 21:29:11 +03:00
if ( ! check ) {
usb_audio_warn ( ep - > chip ,
" Unable to change format on ep #%x: already in use \n " ,
ep - > ep_num ) ;
return - EBUSY ;
}
usb_audio_dbg ( ep - > chip ,
" Ep #%x already in use as implicit feedback but format not changed \n " ,
ep - > ep_num ) ;
return 0 ;
2012-04-12 15:51:11 +04:00
}
/* release old buffers, if any */
release_urbs ( ep , 0 ) ;
ep - > datainterval = fmt - > datainterval ;
ep - > maxpacksize = fmt - > maxpacksize ;
2012-04-13 14:41:54 +04:00
ep - > fill_max = ! ! ( fmt - > attributes & UAC_EP_CS_ATTR_FILL_MAX ) ;
2012-04-12 15:51:11 +04:00
2020-04-24 05:24:48 +03:00
if ( snd_usb_get_speed ( ep - > chip - > dev ) = = USB_SPEED_FULL ) {
2012-09-18 20:49:47 +04:00
ep - > freqn = get_usb_full_speed_rate ( rate ) ;
2020-06-29 05:59:34 +03:00
ep - > pps = 1000 > > ep - > datainterval ;
2020-04-24 05:24:48 +03:00
} else {
2012-09-18 20:49:47 +04:00
ep - > freqn = get_usb_high_speed_rate ( rate ) ;
2020-06-29 05:59:34 +03:00
ep - > pps = 8000 > > ep - > datainterval ;
2020-04-24 05:24:48 +03:00
}
2020-06-29 05:59:34 +03:00
ep - > sample_rem = rate % ep - > pps ;
ep - > packsize [ 0 ] = rate / ep - > pps ;
ep - > packsize [ 1 ] = ( rate + ( ep - > pps - 1 ) ) / ep - > pps ;
2012-04-12 15:51:11 +04:00
/* calculate the frequency in 16.16 format */
ep - > freqm = ep - > freqn ;
ep - > freqshift = INT_MIN ;
ep - > phase = 0 ;
switch ( ep - > type ) {
case SND_USB_ENDPOINT_TYPE_DATA :
2012-09-18 20:49:47 +04:00
err = data_ep_set_params ( ep , pcm_format , channels ,
ALSA: improve buffer size computations for USB PCM audio
This patch changes the way URBs are allocated and their sizes are
determined for PCM playback in the snd-usb-audio driver. Currently
the driver allocates too few URBs for endpoints that don't use
implicit sync, making underruns more likely to occur. This may be a
holdover from before I/O delays could be measured accurately; in any
case, it is no longer necessary.
The patch allocates as many URBs as possible, subject to four
limitations:
The total number of URBs for the endpoint is not allowed to
exceed MAX_URBS (which the patch increases from 8 to 12).
The total number of packets per URB is not allowed to exceed
MAX_PACKS (or MAX_PACKS_HS for high-speed devices), which is
decreased from 20 to 6.
The total duration of queued data is not allowed to exceed
MAX_QUEUE, which is decreased from 24 ms to 18 ms.
The total number of ALSA frames in the output queue is not
allowed to exceed the ALSA buffer size.
The last requirement is the hardest to implement. Currently the
number of URBs needed to fill a buffer cannot be determined in
advance, because a buffer contains a fixed number of frames whereas
the number of frames in an URB varies to match shifts in the device's
clock rate. To solve this problem, the patch changes the logic for
deciding how many packets an URB should contain. Rather than using as
many as possible without exceeding an ALSA period boundary, now the
driver uses only as many packets as needed to transfer a predetermined
number of frames. As a result, unless the device's clock has an
exceedingly variable rate, the number of URBs making up each period
(and hence each buffer) will remain constant.
The overall effect of the patch is that playback works better in
low-latency settings. The user can still specify values for
frames/period and periods/buffer that exceed the capabilities of the
hardware, of course. But for values that are within those
capabilities, the performance will be improved. For example, testing
shows that a high-speed device can handle 32 frames/period and 3
periods/buffer at 48 KHz, whereas the current driver starts to get
glitchy at 64 frames/period and 2 periods/buffer.
A side effect of these changes is that the "nrpacks" module parameter
is no longer used. The patch removes it.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
CC: Clemens Ladisch <clemens@ladisch.de>
Tested-by: Daniel Mack <zonque@gmail.com>
Tested-by: Eldad Zack <eldad@fogrefinery.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2013-09-24 23:51:58 +04:00
period_bytes , period_frames ,
buffer_periods , fmt , sync_ep ) ;
2012-04-12 15:51:11 +04:00
break ;
case SND_USB_ENDPOINT_TYPE_SYNC :
2013-10-07 00:31:06 +04:00
err = sync_ep_set_params ( ep ) ;
2012-04-12 15:51:11 +04:00
break ;
default :
err = - EINVAL ;
}
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
usb_audio_dbg ( ep - > chip , " Set up %d URBS, ret=%d \n " , ep - > nurbs , err ) ;
2012-04-12 15:51:11 +04:00
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
if ( err < 0 )
return err ;
/* record the current set up in the endpoint (for implicit fb) */
spin_lock_irq ( & ep - > lock ) ;
ep - > cur_rate = rate ;
ep - > cur_channels = channels ;
ep - > cur_format = pcm_format ;
ep - > cur_period_frames = period_frames ;
ep - > cur_period_bytes = period_bytes ;
ep - > cur_buffer_periods = buffer_periods ;
spin_unlock_irq ( & ep - > lock ) ;
return 0 ;
2012-04-12 15:51:11 +04:00
}
2012-04-12 15:51:15 +04:00
/**
* snd_usb_endpoint_start : start an snd_usb_endpoint
*
2017-01-05 01:37:46 +03:00
* @ ep : the endpoint to start
2012-04-12 15:51:15 +04:00
*
* A call to this function will increment the use count of the endpoint .
2012-04-24 21:31:24 +04:00
* In case it is not already running , the URBs for this endpoint will be
2012-04-12 15:51:15 +04:00
* submitted . Otherwise , this function does nothing .
*
* Must be balanced to calls of snd_usb_endpoint_stop ( ) .
*
* Returns an error if the URB submission failed , 0 in all other cases .
*/
2017-01-05 01:37:46 +03:00
int snd_usb_endpoint_start ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
int err ;
unsigned int i ;
ALSA: usb-audio: Avoid nested autoresume calls
After the recent fix of runtime PM for USB-audio driver, we got a
lockdep warning like:
=============================================
[ INFO: possible recursive locking detected ]
4.2.0-rc8+ #61 Not tainted
---------------------------------------------
pulseaudio/980 is trying to acquire lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
but task is already holding lock:
(&chip->shutdown_rwsem){.+.+.+}, at: [<ffffffffa0355dac>] snd_usb_autoresume+0x1d/0x52 [snd_usb_audio]
This comes from snd_usb_autoresume() invoking down_read() and it's
used in a nested way. Although it's basically safe, per se (as these
are read locks), it's better to reduce such spurious warnings.
The read lock is needed to guarantee the execution of "shutdown"
(cleanup at disconnection) task after all concurrent tasks are
finished. This can be implemented in another better way.
Also, the current check of chip->in_pm isn't good enough for
protecting the racy execution of multiple auto-resumes.
This patch rewrites the logic of snd_usb_autoresume() & co; namely,
- The recursive call of autopm is avoided by the new refcount,
chip->active. The chip->in_pm flag is removed accordingly.
- Instead of rwsem, another refcount, chip->usage_count, is introduced
for tracking the period to delay the shutdown procedure. At
the last clear of this refcount, wake_up() to the shutdown waiter is
called.
- The shutdown flag is replaced with shutdown atomic count; this is
for reducing the lock.
- Two new helpers are introduced to simplify the management of these
refcounts; snd_usb_lock_shutdown() increases the usage_count, checks
the shutdown state, and does autoresume. snd_usb_unlock_shutdown()
does the opposite. Most of mixer and other codes just need this,
and simply returns an error if it receives an error from lock.
Fixes: 9003ebb13f61 ('ALSA: usb-audio: Fix runtime PM unbalance')
Reported-and-tested-by: Alexnader Kuleshov <kuleshovmail@gmail.com>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2015-08-25 17:09:00 +03:00
if ( atomic_read ( & ep - > chip - > shutdown ) )
2012-04-12 15:51:11 +04:00
return - EBADFD ;
/* already running? */
if ( + + ep - > use_count ! = 1 )
return 0 ;
2012-08-29 15:17:05 +04:00
/* just to be sure */
2012-11-21 11:22:52 +04:00
deactivate_urbs ( ep , false ) ;
2012-08-29 15:17:05 +04:00
2012-04-12 15:51:11 +04:00
ep - > active_mask = 0 ;
ep - > unlink_mask = 0 ;
ep - > phase = 0 ;
2020-04-24 05:24:48 +03:00
ep - > sample_accum = 0 ;
2012-04-12 15:51:11 +04:00
2012-09-04 12:23:07 +04:00
snd_usb_endpoint_start_quirk ( ep ) ;
2012-04-12 15:51:11 +04:00
/*
* If this endpoint has a data endpoint as implicit feedback source ,
* don ' t start the urbs here . Instead , mark them all as available ,
2012-04-24 21:31:24 +04:00
* wait for the record urbs to return and queue the playback urbs
* from that context .
2012-04-12 15:51:11 +04:00
*/
set_bit ( EP_FLAG_RUNNING , & ep - > flags ) ;
2013-04-04 01:18:52 +04:00
if ( snd_usb_endpoint_implicit_feedback_sink ( ep ) ) {
2012-04-12 15:51:11 +04:00
for ( i = 0 ; i < ep - > nurbs ; i + + ) {
struct snd_urb_ctx * ctx = ep - > urb + i ;
list_add_tail ( & ctx - > ready_list , & ep - > ready_playback_urbs ) ;
}
return 0 ;
}
for ( i = 0 ; i < ep - > nurbs ; i + + ) {
struct urb * urb = ep - > urb [ i ] . urb ;
if ( snd_BUG_ON ( ! urb ) )
goto __error ;
if ( usb_pipeout ( ep - > pipe ) ) {
prepare_outbound_urb ( ep , urb - > context ) ;
} else {
prepare_inbound_urb ( ep , urb - > context ) ;
}
err = usb_submit_urb ( urb , GFP_ATOMIC ) ;
if ( err < 0 ) {
2014-02-26 16:02:17 +04:00
usb_audio_err ( ep - > chip ,
" cannot submit urb %d, error %d: %s \n " ,
i , err , usb_error_string ( err ) ) ;
2012-04-12 15:51:11 +04:00
goto __error ;
}
set_bit ( i , & ep - > active_mask ) ;
}
return 0 ;
__error :
clear_bit ( EP_FLAG_RUNNING , & ep - > flags ) ;
ep - > use_count - - ;
2012-11-21 11:22:52 +04:00
deactivate_urbs ( ep , false ) ;
2012-04-12 15:51:11 +04:00
return - EPIPE ;
}
2012-04-12 15:51:15 +04:00
/**
* snd_usb_endpoint_stop : stop an snd_usb_endpoint
*
* @ ep : the endpoint to stop ( may be NULL )
*
* A call to this function will decrement the use count of the endpoint .
* In case the last user has requested the endpoint stop , the URBs will
2012-04-24 21:31:24 +04:00
* actually be deactivated .
2012-04-12 15:51:15 +04:00
*
* Must be balanced to calls of snd_usb_endpoint_start ( ) .
2012-11-21 11:30:48 +04:00
*
* The caller needs to synchronize the pending stop operation via
* snd_usb_endpoint_sync_pending_stop ( ) .
2012-04-12 15:51:15 +04:00
*/
2012-11-21 11:30:48 +04:00
void snd_usb_endpoint_stop ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
if ( ! ep )
return ;
if ( snd_BUG_ON ( ep - > use_count = = 0 ) )
return ;
if ( - - ep - > use_count = = 0 ) {
2012-11-21 11:22:52 +04:00
deactivate_urbs ( ep , false ) ;
2012-11-21 11:30:48 +04:00
set_bit ( EP_FLAG_STOPPING , & ep - > flags ) ;
2012-04-12 15:51:11 +04:00
}
}
2012-04-12 15:51:15 +04:00
/**
* snd_usb_endpoint_deactivate : deactivate an snd_usb_endpoint
*
* @ ep : the endpoint to deactivate
*
2013-10-07 00:31:10 +04:00
* If the endpoint is not currently in use , this functions will
* deactivate its associated URBs .
2012-04-12 15:51:15 +04:00
*
* In case of any active users , this functions does nothing .
*/
2013-10-07 00:31:10 +04:00
void snd_usb_endpoint_deactivate ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
if ( ! ep )
2013-10-07 00:31:10 +04:00
return ;
2012-04-12 15:51:11 +04:00
if ( ep - > use_count ! = 0 )
2013-10-07 00:31:10 +04:00
return ;
2012-04-12 15:51:11 +04:00
2013-10-07 00:31:09 +04:00
deactivate_urbs ( ep , true ) ;
wait_clear_urbs ( ep ) ;
ALSA: usb-audio: Add hw constraint for implicit fb sync
In the current code, there is no check at the stream open time whether
the endpoint is being already used by others. In the normal
operations, this shouldn't happen, but in the case of the implicit
feedback mode, it's a common problem with the full duplex operation,
because the capture stream is always opened by the playback stream as
an implicit sync source.
Although we recently introduced the check of such a conflict of
parameters at the PCM hw_params time, it doesn't give any hint at the
hw_params itself and just gives the error. This isn't quite
comfortable, and it caused problems on many applications.
This patch attempts to make the parameter handling easier by
introducing the strict hw constraint matching with the counterpart
stream that is being used. That said, when an implicit feedback
playback stream is running before a capture stream is opened, the
capture stream carries the PCM hw-constraint to allow only the same
sample rate, format, periods and period frames as the running playback
stream. If not opened or there is no conflict of endpoints, the
behavior remains as same as before.
Note that this kind of "weak link" should work for most cases, but
this is no concrete solution; e.g. if an application changes the hw
params multiple times while another stream is opened, this would lead
to inconsistencies.
Tested-by: Keith Milner <kamilner@superlative.org>
Tested-by: Dylan Robinson <dylan_robinson@motu.com>
Link: https://lore.kernel.org/r/20201123085347.19667-11-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2020-11-23 11:53:16 +03:00
/* clear the saved hw params */
spin_lock_irq ( & ep - > lock ) ;
ep - > cur_rate = 0 ;
spin_unlock_irq ( & ep - > lock ) ;
2012-04-12 15:51:11 +04:00
}
2014-06-25 16:24:47 +04:00
/**
* snd_usb_endpoint_release : Tear down an snd_usb_endpoint
*
* @ ep : the endpoint to release
*
* This function does not care for the endpoint ' s use count but will tear
* down all the streaming URBs immediately .
*/
void snd_usb_endpoint_release ( struct snd_usb_endpoint * ep )
{
release_urbs ( ep , 1 ) ;
}
2012-04-24 21:31:24 +04:00
/**
* snd_usb_endpoint_free : Free the resources of an snd_usb_endpoint
2012-04-12 15:51:15 +04:00
*
2014-10-31 13:24:32 +03:00
* @ ep : the endpoint to free
2012-04-12 15:51:15 +04:00
*
2014-06-25 16:24:47 +04:00
* This free all resources of the given ep .
2012-04-12 15:51:15 +04:00
*/
2014-10-31 13:24:32 +03:00
void snd_usb_endpoint_free ( struct snd_usb_endpoint * ep )
2012-04-12 15:51:11 +04:00
{
kfree ( ep ) ;
}
2012-04-12 15:51:15 +04:00
/**
* snd_usb_handle_sync_urb : parse an USB sync packet
*
* @ ep : the endpoint to handle the packet
* @ sender : the sending endpoint
* @ urb : the received packet
*
* This function is called from the context of an endpoint that received
* the packet and is used to let another endpoint object handle the payload .
2012-04-12 15:51:11 +04:00
*/
void snd_usb_handle_sync_urb ( struct snd_usb_endpoint * ep ,
struct snd_usb_endpoint * sender ,
const struct urb * urb )
{
int shift ;
unsigned int f ;
unsigned long flags ;
snd_BUG_ON ( ep = = sender ) ;
2012-04-12 15:51:15 +04:00
/*
* In case the endpoint is operating in implicit feedback mode , prepare
2012-04-24 21:31:24 +04:00
* a new outbound URB that has the same layout as the received packet
* and add it to the list of pending urbs . queue_pending_output_urbs ( )
* will take care of them later .
2012-04-12 15:51:15 +04:00
*/
2013-04-04 01:18:52 +04:00
if ( snd_usb_endpoint_implicit_feedback_sink ( ep ) & &
2012-04-12 15:51:11 +04:00
ep - > use_count ! = 0 ) {
/* implicit feedback case */
int i , bytes = 0 ;
struct snd_urb_ctx * in_ctx ;
struct snd_usb_packet_info * out_packet ;
in_ctx = urb - > context ;
/* Count overall packet size */
for ( i = 0 ; i < in_ctx - > packets ; i + + )
if ( urb - > iso_frame_desc [ i ] . status = = 0 )
bytes + = urb - > iso_frame_desc [ i ] . actual_length ;
/*
* skip empty packets . At least M - Audio ' s Fast Track Ultra stops
* streaming once it received a 0 - byte OUT URB
*/
if ( bytes = = 0 )
return ;
spin_lock_irqsave ( & ep - > lock , flags ) ;
out_packet = ep - > next_packet + ep - > next_packet_write_pos ;
/*
* Iterate through the inbound packet and prepare the lengths
* for the output packet . The OUT packet we are about to send
2012-11-29 02:55:34 +04:00
* will have the same amount of payload bytes per stride as the
* IN packet we just received . Since the actual size is scaled
* by the stride , use the sender stride to calculate the length
* in case the number of channels differ between the implicitly
* fed - back endpoint and the synchronizing endpoint .
2012-04-12 15:51:11 +04:00
*/
out_packet - > packets = in_ctx - > packets ;
for ( i = 0 ; i < in_ctx - > packets ; i + + ) {
if ( urb - > iso_frame_desc [ i ] . status = = 0 )
out_packet - > packet_size [ i ] =
2012-11-29 02:55:34 +04:00
urb - > iso_frame_desc [ i ] . actual_length / sender - > stride ;
2012-04-12 15:51:11 +04:00
else
out_packet - > packet_size [ i ] = 0 ;
}
ep - > next_packet_write_pos + + ;
ep - > next_packet_write_pos % = MAX_URBS ;
spin_unlock_irqrestore ( & ep - > lock , flags ) ;
queue_pending_output_urbs ( ep ) ;
return ;
}
2012-04-12 15:51:15 +04:00
/*
* process after playback sync complete
*
* Full speed devices report feedback values in 10.14 format as samples
* per frame , high speed devices in 16.16 format as samples per
* microframe .
*
* Because the Audio Class 1 spec was written before USB 2.0 , many high
* speed devices use a wrong interpretation , some others use an
* entirely different format .
*
* Therefore , we cannot predict what format any particular device uses
* and must detect it automatically .
*/
2012-04-12 15:51:11 +04:00
if ( urb - > iso_frame_desc [ 0 ] . status ! = 0 | |
urb - > iso_frame_desc [ 0 ] . actual_length < 3 )
return ;
f = le32_to_cpup ( urb - > transfer_buffer ) ;
if ( urb - > iso_frame_desc [ 0 ] . actual_length = = 3 )
f & = 0x00ffffff ;
else
f & = 0x0fffffff ;
if ( f = = 0 )
return ;
2016-08-22 09:53:37 +03:00
if ( unlikely ( sender - > tenor_fb_quirk ) ) {
2014-05-01 14:20:22 +04:00
/*
2016-08-22 09:53:37 +03:00
* Devices based on Tenor 8802 chipsets ( TEAC UD - H01
* and others ) sometimes change the feedback value
2014-05-01 14:20:22 +04:00
* by + / - 0x1 .0000 .
*/
if ( f < ep - > freqn - 0x8000 )
2016-08-22 09:53:38 +03:00
f + = 0xf000 ;
2014-05-01 14:20:22 +04:00
else if ( f > ep - > freqn + 0x8000 )
2016-08-22 09:53:38 +03:00
f - = 0xf000 ;
2014-05-01 14:20:22 +04:00
} else if ( unlikely ( ep - > freqshift = = INT_MIN ) ) {
2012-04-12 15:51:11 +04:00
/*
* The first time we see a feedback value , determine its format
* by shifting it left or right until it matches the nominal
* frequency value . This assumes that the feedback does not
* differ from the nominal value more than + 50 % or - 25 % .
*/
shift = 0 ;
while ( f < ep - > freqn - ep - > freqn / 4 ) {
f < < = 1 ;
shift + + ;
}
while ( f > ep - > freqn + ep - > freqn / 2 ) {
f > > = 1 ;
shift - - ;
}
ep - > freqshift = shift ;
} else if ( ep - > freqshift > = 0 )
f < < = ep - > freqshift ;
else
f > > = - ep - > freqshift ;
if ( likely ( f > = ep - > freqn - ep - > freqn / 8 & & f < = ep - > freqmax ) ) {
/*
* If the frequency looks valid , set it .
* This value is referred to in prepare_playback_urb ( ) .
*/
spin_lock_irqsave ( & ep - > lock , flags ) ;
ep - > freqm = f ;
spin_unlock_irqrestore ( & ep - > lock , flags ) ;
} else {
/*
* Out of range ; maybe the shift value is wrong .
* Reset it so that we autodetect again the next time .
*/
ep - > freqshift = INT_MIN ;
}
}