2010-03-04 19:46:13 +01:00
/*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
# include <linux/init.h>
2010-03-29 16:02:50 +11:00
# include <linux/slab.h>
2010-03-04 19:46:13 +01:00
# include <linux/usb.h>
# include <linux/usb/audio.h>
2010-03-11 21:13:20 +01:00
# include <linux/usb/audio-v2.h>
2018-03-21 02:03:59 +02:00
# include <linux/usb/audio-v3.h>
2010-03-04 19:46:13 +01:00
# include <sound/core.h>
# include <sound/pcm.h>
# include "usbaudio.h"
# include "card.h"
# include "quirks.h"
# include "helper.h"
# include "debug.h"
2010-05-31 14:51:31 +02:00
# include "clock.h"
2011-05-18 11:28:40 +02:00
# include "format.h"
2010-03-04 19:46:13 +01:00
/*
* parse the audio format type I descriptor
* and returns the corresponding pcm format
*
* @ dev : usb device
* @ fp : audioformat record
* @ format : the format tag ( wFormatTag )
2018-03-21 02:03:59 +02:00
* @ fmt : the format type descriptor ( v1 / v2 ) or AudioStreaming descriptor ( v3 )
2010-03-04 19:46:13 +01:00
*/
2010-03-04 19:46:16 +01:00
static u64 parse_audio_format_i_type ( struct snd_usb_audio * chip ,
2010-03-04 19:46:13 +01:00
struct audioformat * fp ,
2018-03-21 02:03:59 +02:00
u64 format , void * _fmt )
2010-03-04 19:46:13 +01:00
{
int sample_width , sample_bytes ;
2013-03-17 20:07:25 +08:00
u64 pcm_formats = 0 ;
2010-03-04 19:46:13 +01:00
2013-01-31 21:39:17 +01:00
switch ( fp - > protocol ) {
2010-09-03 10:53:11 +02:00
case UAC_VERSION_1 :
default : {
2010-03-04 19:46:13 +01:00
struct uac_format_type_i_discrete_descriptor * fmt = _fmt ;
sample_width = fmt - > bBitResolution ;
sample_bytes = fmt - > bSubframeSize ;
2018-03-29 12:03:19 +03:00
format = 1ULL < < format ;
2010-03-04 19:46:13 +01:00
break ;
}
case UAC_VERSION_2 : {
struct uac_format_type_i_ext_descriptor * fmt = _fmt ;
sample_width = fmt - > bBitResolution ;
sample_bytes = fmt - > bSubslotSize ;
2013-03-17 20:07:25 +08:00
if ( format & UAC2_FORMAT_TYPE_I_RAW_DATA )
pcm_formats | = SNDRV_PCM_FMTBIT_SPECIAL ;
2010-03-04 19:46:16 +01:00
format < < = 1 ;
2010-03-04 19:46:13 +01:00
break ;
}
2018-03-21 02:03:59 +02:00
case UAC_VERSION_3 : {
struct uac3_as_header_descriptor * as = _fmt ;
sample_width = as - > bBitResolution ;
sample_bytes = as - > bSubslotSize ;
if ( format & UAC3_FORMAT_TYPE_I_RAW_DATA )
pcm_formats | = SNDRV_PCM_FMTBIT_SPECIAL ;
format < < = 1 ;
break ;
}
2010-03-04 19:46:13 +01:00
}
2013-03-17 20:07:25 +08:00
if ( ( pcm_formats = = 0 ) & &
( format = = 0 | | format = = ( 1 < < UAC_FORMAT_TYPE_I_UNDEFINED ) ) ) {
2010-03-04 19:46:16 +01:00
/* some devices don't define this correctly... */
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip , " %u:%d : format type 0 is detected, processed as PCM \n " ,
fp - > iface , fp - > altsetting ) ;
2010-03-04 19:46:16 +01:00
format = 1 < < UAC_FORMAT_TYPE_I_PCM ;
}
if ( format & ( 1 < < UAC_FORMAT_TYPE_I_PCM ) ) {
2015-04-21 11:23:57 +09:00
if ( ( ( chip - > usb_id = = USB_ID ( 0x0582 , 0x0016 ) ) | |
/* Edirol SD-90 */
( chip - > usb_id = = USB_ID ( 0x0582 , 0x000c ) ) ) & &
/* Roland SC-D70 */
2011-01-10 16:30:54 +01:00
sample_width = = 24 & & sample_bytes = = 2 )
sample_bytes = 3 ;
else if ( sample_width > sample_bytes * 8 ) {
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip , " %u:%d : sample bitwidth %d in over sample bytes %d \n " ,
fp - > iface , fp - > altsetting ,
sample_width , sample_bytes ) ;
2010-03-04 19:46:13 +01:00
}
/* check the format byte size */
switch ( sample_bytes ) {
case 1 :
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S8 ;
2010-03-04 19:46:13 +01:00
break ;
case 2 :
if ( snd_usb_is_big_endian_format ( chip , fp ) )
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S16_BE ; /* grrr, big endian!! */
2010-03-04 19:46:13 +01:00
else
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S16_LE ;
2010-03-04 19:46:13 +01:00
break ;
case 3 :
if ( snd_usb_is_big_endian_format ( chip , fp ) )
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S24_3BE ; /* grrr, big endian!! */
2010-03-04 19:46:13 +01:00
else
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S24_3LE ;
2010-03-04 19:46:13 +01:00
break ;
case 4 :
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S32_LE ;
2010-03-04 19:46:13 +01:00
break ;
default :
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip ,
" %u:%d : unsupported sample bitwidth %d in %d bytes \n " ,
fp - > iface , fp - > altsetting ,
sample_width , sample_bytes ) ;
2010-03-04 19:46:13 +01:00
break ;
}
2010-03-04 19:46:16 +01:00
}
if ( format & ( 1 < < UAC_FORMAT_TYPE_I_PCM8 ) ) {
2010-03-04 19:46:13 +01:00
/* Dallas DS4201 workaround: it advertises U8 format, but really
supports S8 . */
if ( chip - > usb_id = = USB_ID ( 0x04fa , 0x4201 ) )
2010-03-04 19:46:16 +01:00
pcm_formats | = SNDRV_PCM_FMTBIT_S8 ;
else
pcm_formats | = SNDRV_PCM_FMTBIT_U8 ;
}
if ( format & ( 1 < < UAC_FORMAT_TYPE_I_IEEE_FLOAT ) ) {
pcm_formats | = SNDRV_PCM_FMTBIT_FLOAT_LE ;
}
if ( format & ( 1 < < UAC_FORMAT_TYPE_I_ALAW ) ) {
pcm_formats | = SNDRV_PCM_FMTBIT_A_LAW ;
}
if ( format & ( 1 < < UAC_FORMAT_TYPE_I_MULAW ) ) {
pcm_formats | = SNDRV_PCM_FMTBIT_MU_LAW ;
}
if ( format & ~ 0x3f ) {
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip ,
2018-03-21 02:03:59 +02:00
" %u:%d : unsupported format bits %#llx \n " ,
2014-02-26 13:02:17 +01:00
fp - > iface , fp - > altsetting , format ) ;
2010-03-04 19:46:13 +01:00
}
2013-04-17 00:01:40 +08:00
pcm_formats | = snd_usb_interface_dsd_format_quirks ( chip , fp , sample_bytes ) ;
2010-03-04 19:46:16 +01:00
return pcm_formats ;
2010-03-04 19:46:13 +01:00
}
/*
* parse the format descriptor and stores the possible sample rates
* on the audioformat table ( audio class v1 ) .
*
* @ dev : usb device
* @ fp : audioformat record
* @ fmt : the format descriptor
* @ offset : the start offset of descriptor pointing the rate type
* ( 7 for type I and II , 8 for type II )
*/
static int parse_audio_format_rates_v1 ( struct snd_usb_audio * chip , struct audioformat * fp ,
unsigned char * fmt , int offset )
{
int nr_rates = fmt [ offset ] ;
if ( fmt [ 0 ] < offset + 1 + 3 * ( nr_rates ? nr_rates : 2 ) ) {
2014-02-26 13:02:17 +01:00
usb_audio_err ( chip ,
" %u:%d : invalid UAC_FORMAT_TYPE desc \n " ,
fp - > iface , fp - > altsetting ) ;
2012-11-21 15:25:56 +05:30
return - EINVAL ;
2010-03-04 19:46:13 +01:00
}
if ( nr_rates ) {
/*
* build the rate table and bitmap flags
*/
int r , idx ;
fp - > rate_table = kmalloc ( sizeof ( int ) * nr_rates , GFP_KERNEL ) ;
2016-08-21 10:17:36 +08:00
if ( fp - > rate_table = = NULL )
2012-11-21 15:25:56 +05:30
return - ENOMEM ;
2010-03-04 19:46:13 +01:00
fp - > nr_rates = 0 ;
fp - > rate_min = fp - > rate_max = 0 ;
for ( r = 0 , idx = offset + 1 ; r < nr_rates ; r + + , idx + = 3 ) {
unsigned int rate = combine_triple ( & fmt [ idx ] ) ;
if ( ! rate )
continue ;
/* C-Media CM6501 mislabels its 96 kHz altsetting */
2011-04-28 16:18:40 +02:00
/* Terratec Aureon 7.1 USB C-Media 6206, too */
2010-03-04 19:46:13 +01:00
if ( rate = = 48000 & & nr_rates = = 1 & &
( chip - > usb_id = = USB_ID ( 0x0d8c , 0x0201 ) | |
2011-04-28 16:18:40 +02:00
chip - > usb_id = = USB_ID ( 0x0d8c , 0x0102 ) | |
chip - > usb_id = = USB_ID ( 0x0ccd , 0x00b1 ) ) & &
2010-03-04 19:46:13 +01:00
fp - > altsetting = = 5 & & fp - > maxpacksize = = 392 )
rate = 96000 ;
2014-01-14 13:42:57 +01:00
/* Creative VF0420/VF0470 Live Cams report 16 kHz instead of 8kHz */
if ( rate = = 16000 & &
( chip - > usb_id = = USB_ID ( 0x041e , 0x4064 ) | |
chip - > usb_id = = USB_ID ( 0x041e , 0x4068 ) ) )
2010-03-04 19:46:13 +01:00
rate = 8000 ;
fp - > rate_table [ fp - > nr_rates ] = rate ;
if ( ! fp - > rate_min | | rate < fp - > rate_min )
fp - > rate_min = rate ;
if ( ! fp - > rate_max | | rate > fp - > rate_max )
fp - > rate_max = rate ;
fp - > rates | = snd_pcm_rate_to_rate_bit ( rate ) ;
fp - > nr_rates + + ;
}
if ( ! fp - > nr_rates ) {
hwc_debug ( " All rates were zero. Skipping format! \n " ) ;
2012-11-21 15:25:56 +05:30
return - EINVAL ;
2010-03-04 19:46:13 +01:00
}
} else {
/* continuous rates */
fp - > rates = SNDRV_PCM_RATE_CONTINUOUS ;
fp - > rate_min = combine_triple ( & fmt [ offset + 1 ] ) ;
fp - > rate_max = combine_triple ( & fmt [ offset + 4 ] ) ;
}
return 0 ;
}
2010-06-11 17:46:33 +02:00
/*
* Helper function to walk the array of sample rate triplets reported by
* the device . The problem is that we need to parse whole array first to
* get to know how many sample rates we have to expect .
* Then fp - > rate_table can be allocated and filled .
*/
2014-02-26 13:02:17 +01:00
static int parse_uac2_sample_rate_range ( struct snd_usb_audio * chip ,
struct audioformat * fp , int nr_triplets ,
2010-06-11 17:46:33 +02:00
const unsigned char * data )
{
int i , nr_rates = 0 ;
fp - > rates = fp - > rate_min = fp - > rate_max = 0 ;
for ( i = 0 ; i < nr_triplets ; i + + ) {
int min = combine_quad ( & data [ 2 + 12 * i ] ) ;
int max = combine_quad ( & data [ 6 + 12 * i ] ) ;
int res = combine_quad ( & data [ 10 + 12 * i ] ) ;
2012-01-08 09:02:52 -05:00
unsigned int rate ;
2010-06-11 17:46:33 +02:00
if ( ( max < 0 ) | | ( min < 0 ) | | ( res < 0 ) | | ( max < min ) )
continue ;
/*
* for ranges with res = = 1 , we announce a continuous sample
* rate range , and this function should return 0 for no further
* parsing .
*/
if ( res = = 1 ) {
fp - > rate_min = min ;
fp - > rate_max = max ;
fp - > rates = SNDRV_PCM_RATE_CONTINUOUS ;
return 0 ;
}
for ( rate = min ; rate < = max ; rate + = res ) {
if ( fp - > rate_table )
fp - > rate_table [ nr_rates ] = rate ;
if ( ! fp - > rate_min | | rate < fp - > rate_min )
fp - > rate_min = rate ;
if ( ! fp - > rate_max | | rate > fp - > rate_max )
fp - > rate_max = rate ;
fp - > rates | = snd_pcm_rate_to_rate_bit ( rate ) ;
nr_rates + + ;
2012-02-14 05:18:48 -05:00
if ( nr_rates > = MAX_NR_RATES ) {
2014-02-26 13:02:17 +01:00
usb_audio_err ( chip , " invalid uac2 rates \n " ) ;
2012-01-08 09:02:52 -05:00
break ;
}
2010-06-11 17:46:33 +02:00
/* avoid endless loop */
if ( res = = 0 )
break ;
}
}
return nr_rates ;
}
2010-03-04 19:46:13 +01:00
/*
* parse the format descriptor and stores the possible sample rates
2018-03-21 02:03:59 +02:00
* on the audioformat table ( audio class v2 and v3 ) .
2010-03-04 19:46:13 +01:00
*/
2018-03-21 02:03:59 +02:00
static int parse_audio_format_rates_v2v3 ( struct snd_usb_audio * chip ,
2010-06-16 17:57:31 +02:00
struct audioformat * fp )
2010-03-04 19:46:13 +01:00
{
struct usb_device * dev = chip - > dev ;
unsigned char tmp [ 2 ] , * data ;
2010-06-11 17:46:33 +02:00
int nr_triplets , data_size , ret = 0 ;
2018-03-21 02:03:59 +02:00
int clock = snd_usb_clock_find_source ( chip , fp - > protocol ,
fp - > clock , false ) ;
2010-03-04 19:46:13 +01:00
2010-06-11 17:34:19 +02:00
if ( clock < 0 ) {
2014-02-26 13:02:17 +01:00
dev_err ( & dev - > dev ,
" %s(): unable to find clock source (clock %d) \n " ,
2010-06-11 17:34:19 +02:00
__func__ , clock ) ;
goto err ;
}
2010-03-04 19:46:13 +01:00
/* get the number of sample rates first by only fetching 2 bytes */
ret = snd_usb_ctl_msg ( dev , usb_rcvctrlpipe ( dev , 0 ) , UAC2_CS_RANGE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
2010-06-11 17:34:20 +02:00
UAC2_CS_CONTROL_SAM_FREQ < < 8 ,
snd_usb_ctrl_intf ( chip ) | ( clock < < 8 ) ,
2011-09-26 21:15:27 +02:00
tmp , sizeof ( tmp ) ) ;
2010-03-04 19:46:13 +01:00
if ( ret < 0 ) {
2014-02-26 13:02:17 +01:00
dev_err ( & dev - > dev ,
" %s(): unable to retrieve number of sample rates (clock %d) \n " ,
2010-05-31 14:51:31 +02:00
__func__ , clock ) ;
2010-03-04 19:46:13 +01:00
goto err ;
}
2010-06-11 17:46:33 +02:00
nr_triplets = ( tmp [ 1 ] < < 8 ) | tmp [ 0 ] ;
data_size = 2 + 12 * nr_triplets ;
2010-03-04 19:46:13 +01:00
data = kzalloc ( data_size , GFP_KERNEL ) ;
if ( ! data ) {
ret = - ENOMEM ;
goto err ;
}
/* now get the full information */
ret = snd_usb_ctl_msg ( dev , usb_rcvctrlpipe ( dev , 0 ) , UAC2_CS_RANGE ,
2010-05-31 14:51:31 +02:00
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
2010-06-11 17:34:20 +02:00
UAC2_CS_CONTROL_SAM_FREQ < < 8 ,
snd_usb_ctrl_intf ( chip ) | ( clock < < 8 ) ,
2011-09-26 21:15:27 +02:00
data , data_size ) ;
2010-03-04 19:46:13 +01:00
if ( ret < 0 ) {
2014-02-26 13:02:17 +01:00
dev_err ( & dev - > dev ,
" %s(): unable to retrieve sample rate range (clock %d) \n " ,
2010-05-31 14:51:31 +02:00
__func__ , clock ) ;
2010-03-04 19:46:13 +01:00
ret = - EINVAL ;
goto err_free ;
}
2010-06-11 17:46:33 +02:00
/* Call the triplet parser, and make sure fp->rate_table is NULL.
* We just use the return value to know how many sample rates we
* will have to deal with . */
kfree ( fp - > rate_table ) ;
fp - > rate_table = NULL ;
2014-02-26 13:02:17 +01:00
fp - > nr_rates = parse_uac2_sample_rate_range ( chip , fp , nr_triplets , data ) ;
2010-06-11 17:46:33 +02:00
if ( fp - > nr_rates = = 0 ) {
/* SNDRV_PCM_RATE_CONTINUOUS */
ret = 0 ;
goto err_free ;
}
fp - > rate_table = kmalloc ( sizeof ( int ) * fp - > nr_rates , GFP_KERNEL ) ;
2010-03-04 19:46:13 +01:00
if ( ! fp - > rate_table ) {
ret = - ENOMEM ;
goto err_free ;
}
2010-06-11 17:46:33 +02:00
/* Call the triplet parser again, but this time, fp->rate_table is
* allocated , so the rates will be stored */
2014-02-26 13:02:17 +01:00
parse_uac2_sample_rate_range ( chip , fp , nr_triplets , data ) ;
2010-03-04 19:46:13 +01:00
err_free :
kfree ( data ) ;
err :
return ret ;
}
/*
* parse the format type I and III descriptors
*/
static int parse_audio_format_i ( struct snd_usb_audio * chip ,
2018-03-21 02:03:59 +02:00
struct audioformat * fp , u64 format ,
void * _fmt )
2010-03-04 19:46:13 +01:00
{
2013-04-23 01:00:41 +02:00
snd_pcm_format_t pcm_format ;
2018-03-21 02:03:59 +02:00
unsigned int fmt_type ;
2013-04-23 01:00:41 +02:00
int ret ;
2010-03-04 19:46:13 +01:00
2018-03-21 02:03:59 +02:00
switch ( fp - > protocol ) {
default :
case UAC_VERSION_1 :
case UAC_VERSION_2 : {
struct uac_format_type_i_continuous_descriptor * fmt = _fmt ;
fmt_type = fmt - > bFormatType ;
break ;
}
case UAC_VERSION_3 : {
/* fp->fmt_type is already set in this case */
fmt_type = fp - > fmt_type ;
break ;
}
}
if ( fmt_type = = UAC_FORMAT_TYPE_III ) {
2010-03-04 19:46:13 +01:00
/* FIXME: the format type is really IECxxx
* but we give normal PCM format to get the existing
* apps working . . .
*/
switch ( chip - > usb_id ) {
case USB_ID ( 0x0763 , 0x2003 ) : /* M-Audio Audiophile USB */
if ( chip - > setup = = 0x00 & &
fp - > altsetting = = 6 )
pcm_format = SNDRV_PCM_FORMAT_S16_BE ;
else
pcm_format = SNDRV_PCM_FORMAT_S16_LE ;
break ;
default :
pcm_format = SNDRV_PCM_FORMAT_S16_LE ;
}
2013-04-23 01:00:41 +02:00
fp - > formats = pcm_format_to_bits ( pcm_format ) ;
2010-03-04 19:46:13 +01:00
} else {
2018-03-21 02:03:59 +02:00
fp - > formats = parse_audio_format_i_type ( chip , fp , format , _fmt ) ;
2010-03-04 19:46:16 +01:00
if ( ! fp - > formats )
2012-11-21 15:25:56 +05:30
return - EINVAL ;
2010-03-04 19:46:13 +01:00
}
/* gather possible sample rates */
/* audio class v1 reports possible sample rates as part of the
* proprietary class specific descriptor .
* audio class v2 uses class specific EP0 range requests for that .
*/
2013-01-31 21:39:17 +01:00
switch ( fp - > protocol ) {
2010-09-03 10:53:11 +02:00
default :
2018-03-21 02:03:59 +02:00
case UAC_VERSION_1 : {
struct uac_format_type_i_continuous_descriptor * fmt = _fmt ;
2010-03-04 19:46:13 +01:00
fp - > channels = fmt - > bNrChannels ;
2010-05-26 18:11:36 +02:00
ret = parse_audio_format_rates_v1 ( chip , fp , ( unsigned char * ) fmt , 7 ) ;
2010-03-04 19:46:13 +01:00
break ;
2018-03-21 02:03:59 +02:00
}
2010-03-04 19:46:13 +01:00
case UAC_VERSION_2 :
2018-03-21 02:03:59 +02:00
case UAC_VERSION_3 : {
2010-03-04 19:46:13 +01:00
/* fp->channels is already set in this case */
2018-03-21 02:03:59 +02:00
ret = parse_audio_format_rates_v2v3 ( chip , fp ) ;
2010-03-04 19:46:13 +01:00
break ;
}
2018-03-21 02:03:59 +02:00
}
2010-03-04 19:46:13 +01:00
if ( fp - > channels < 1 ) {
2014-02-26 13:02:17 +01:00
usb_audio_err ( chip ,
" %u:%d : invalid channels %d \n " ,
fp - > iface , fp - > altsetting , fp - > channels ) ;
2012-11-21 15:25:56 +05:30
return - EINVAL ;
2010-03-04 19:46:13 +01:00
}
return ret ;
}
/*
* parse the format type II descriptor
*/
static int parse_audio_format_ii ( struct snd_usb_audio * chip ,
struct audioformat * fp ,
2018-03-21 02:03:59 +02:00
u64 format , void * _fmt )
2010-03-04 19:46:13 +01:00
{
int brate , framesize , ret ;
switch ( format ) {
case UAC_FORMAT_TYPE_II_AC3 :
/* FIXME: there is no AC3 format defined yet */
2010-03-04 19:46:15 +01:00
// fp->formats = SNDRV_PCM_FMTBIT_AC3;
fp - > formats = SNDRV_PCM_FMTBIT_U8 ; /* temporary hack to receive byte streams */
2010-03-04 19:46:13 +01:00
break ;
case UAC_FORMAT_TYPE_II_MPEG :
2010-03-04 19:46:15 +01:00
fp - > formats = SNDRV_PCM_FMTBIT_MPEG ;
2010-03-04 19:46:13 +01:00
break ;
default :
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip ,
2018-03-21 02:03:59 +02:00
" %u:%d : unknown format tag %#llx is detected. processed as MPEG. \n " ,
2014-02-26 13:02:17 +01:00
fp - > iface , fp - > altsetting , format ) ;
2010-03-04 19:46:15 +01:00
fp - > formats = SNDRV_PCM_FMTBIT_MPEG ;
2010-03-04 19:46:13 +01:00
break ;
}
fp - > channels = 1 ;
2013-01-31 21:39:17 +01:00
switch ( fp - > protocol ) {
2010-09-03 10:53:11 +02:00
default :
2010-03-04 19:46:13 +01:00
case UAC_VERSION_1 : {
struct uac_format_type_ii_discrete_descriptor * fmt = _fmt ;
brate = le16_to_cpu ( fmt - > wMaxBitRate ) ;
framesize = le16_to_cpu ( fmt - > wSamplesPerFrame ) ;
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip , " found format II with max.bitrate = %d, frame size=%d \n " , brate , framesize ) ;
2010-03-04 19:46:13 +01:00
fp - > frame_size = framesize ;
ret = parse_audio_format_rates_v1 ( chip , fp , _fmt , 8 ) ; /* fmt[8..] sample rates */
break ;
}
case UAC_VERSION_2 : {
struct uac_format_type_ii_ext_descriptor * fmt = _fmt ;
brate = le16_to_cpu ( fmt - > wMaxBitRate ) ;
framesize = le16_to_cpu ( fmt - > wSamplesPerFrame ) ;
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip , " found format II with max.bitrate = %d, frame size=%d \n " , brate , framesize ) ;
2010-03-04 19:46:13 +01:00
fp - > frame_size = framesize ;
2018-03-21 02:03:59 +02:00
ret = parse_audio_format_rates_v2v3 ( chip , fp ) ;
2010-03-04 19:46:13 +01:00
break ;
}
}
return ret ;
}
2013-03-17 20:07:24 +08:00
int snd_usb_parse_audio_format ( struct snd_usb_audio * chip ,
2018-03-21 02:03:59 +02:00
struct audioformat * fp , u64 format ,
2013-03-17 20:07:24 +08:00
struct uac_format_type_i_continuous_descriptor * fmt ,
2013-01-31 21:39:17 +01:00
int stream )
2010-03-04 19:46:13 +01:00
{
int err ;
2010-05-26 18:11:36 +02:00
switch ( fmt - > bFormatType ) {
2010-03-04 19:46:13 +01:00
case UAC_FORMAT_TYPE_I :
case UAC_FORMAT_TYPE_III :
2013-01-31 21:39:17 +01:00
err = parse_audio_format_i ( chip , fp , format , fmt ) ;
2010-03-04 19:46:13 +01:00
break ;
case UAC_FORMAT_TYPE_II :
2013-01-31 21:39:17 +01:00
err = parse_audio_format_ii ( chip , fp , format , fmt ) ;
2010-03-04 19:46:13 +01:00
break ;
default :
2014-02-26 13:02:17 +01:00
usb_audio_info ( chip ,
" %u:%d : format type %d is not supported yet \n " ,
fp - > iface , fp - > altsetting ,
fmt - > bFormatType ) ;
2010-05-26 18:11:37 +02:00
return - ENOTSUPP ;
2010-03-04 19:46:13 +01:00
}
2010-05-26 18:11:36 +02:00
fp - > fmt_type = fmt - > bFormatType ;
2010-03-04 19:46:13 +01:00
if ( err < 0 )
return err ;
# if 1
/* FIXME: temporary hack for extigy/audigy 2 nx/zs */
/* extigy apparently supports sample rates other than 48k
* but not in ordinary way . so we enable only 48 k atm .
*/
if ( chip - > usb_id = = USB_ID ( 0x041e , 0x3000 ) | |
chip - > usb_id = = USB_ID ( 0x041e , 0x3020 ) | |
chip - > usb_id = = USB_ID ( 0x041e , 0x3061 ) ) {
2010-05-26 18:11:36 +02:00
if ( fmt - > bFormatType = = UAC_FORMAT_TYPE_I & &
2010-03-04 19:46:13 +01:00
fp - > rates ! = SNDRV_PCM_RATE_48000 & &
fp - > rates ! = SNDRV_PCM_RATE_96000 )
2010-05-26 18:11:37 +02:00
return - ENOTSUPP ;
2010-03-04 19:46:13 +01:00
}
# endif
return 0 ;
}
2018-03-21 02:03:59 +02:00
int snd_usb_parse_audio_format_v3 ( struct snd_usb_audio * chip ,
struct audioformat * fp ,
struct uac3_as_header_descriptor * as ,
int stream )
{
u64 format = le64_to_cpu ( as - > bmFormats ) ;
int err ;
/*
* Type I format bits are D0 . . D6
* This test works because type IV is not supported
*/
if ( format & 0x7f )
fp - > fmt_type = UAC_FORMAT_TYPE_I ;
else
fp - > fmt_type = UAC_FORMAT_TYPE_III ;
err = parse_audio_format_i ( chip , fp , format , as ) ;
if ( err < 0 )
return err ;
return 0 ;
}