2019-05-27 09:55:05 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
* Routines for control of the CS8427 via i2c bus
* IEC958 ( S / PDIF ) receiver & transmitter by Cirrus Logic
2007-10-15 11:50:19 +04:00
* Copyright ( c ) by Jaroslav Kysela < perex @ perex . cz >
2005-04-17 02:20:36 +04:00
*/
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/init.h>
2009-11-13 10:02:56 +03:00
# include <linux/bitrev.h>
2011-07-15 20:38:28 +04:00
# include <linux/module.h>
2008-05-08 15:01:32 +04:00
# include <asm/unaligned.h>
2005-04-17 02:20:36 +04:00
# include <sound/core.h>
# include <sound/control.h>
# include <sound/pcm.h>
# include <sound/cs8427.h>
# include <sound/asoundef.h>
2005-11-17 16:17:19 +03:00
static void snd_cs8427_reset ( struct snd_i2c_device * cs8427 ) ;
2005-04-17 02:20:36 +04:00
2007-10-15 11:50:19 +04:00
MODULE_AUTHOR ( " Jaroslav Kysela <perex@perex.cz> " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " IEC958 (S/PDIF) receiver & transmitter by Cirrus Logic " ) ;
MODULE_LICENSE ( " GPL " ) ;
# define CS8427_ADDR (0x20>>1) /* fixed address */
2005-11-17 16:17:19 +03:00
struct cs8427_stream {
struct snd_pcm_substream * substream ;
2005-04-17 02:20:36 +04:00
char hw_status [ 24 ] ; /* hardware status */
char def_status [ 24 ] ; /* default status */
char pcm_status [ 24 ] ; /* PCM private status */
char hw_udata [ 32 ] ;
2005-11-17 16:17:19 +03:00
struct snd_kcontrol * pcm_ctl ;
} ;
2005-04-17 02:20:36 +04:00
2005-11-17 16:17:19 +03:00
struct cs8427 {
2005-04-17 02:20:36 +04:00
unsigned char regmap [ 0x14 ] ; /* map of first 1 + 13 registers */
unsigned int rate ;
unsigned int reset_timeout ;
2005-11-17 16:17:19 +03:00
struct cs8427_stream playback ;
struct cs8427_stream capture ;
} ;
2005-04-17 02:20:36 +04:00
2005-11-17 16:17:19 +03:00
int snd_cs8427_reg_write ( struct snd_i2c_device * device , unsigned char reg ,
unsigned char val )
2005-04-17 02:20:36 +04:00
{
int err ;
unsigned char buf [ 2 ] ;
buf [ 0 ] = reg & 0x7f ;
buf [ 1 ] = val ;
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , buf , 2 ) ;
if ( err ! = 2 ) {
2006-06-30 17:31:25 +04:00
snd_printk ( KERN_ERR " unable to send bytes 0x%02x:0x%02x "
" to CS8427 (%i) \n " , buf [ 0 ] , buf [ 1 ] , err ) ;
2005-04-17 02:20:36 +04:00
return err < 0 ? err : - EIO ;
}
return 0 ;
}
2006-06-30 17:31:25 +04:00
EXPORT_SYMBOL ( snd_cs8427_reg_write ) ;
2005-11-17 16:17:19 +03:00
static int snd_cs8427_reg_read ( struct snd_i2c_device * device , unsigned char reg )
2005-04-17 02:20:36 +04:00
{
int err ;
unsigned char buf ;
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , & reg , 1 ) ;
if ( err ! = 1 ) {
2006-06-30 17:31:25 +04:00
snd_printk ( KERN_ERR " unable to send register 0x%x byte "
" to CS8427 \n " , reg ) ;
2005-04-17 02:20:36 +04:00
return err < 0 ? err : - EIO ;
}
2021-06-08 17:05:39 +03:00
err = snd_i2c_readbytes ( device , & buf , 1 ) ;
if ( err ! = 1 ) {
2006-06-30 17:31:25 +04:00
snd_printk ( KERN_ERR " unable to read register 0x%x byte "
" from CS8427 \n " , reg ) ;
2005-04-17 02:20:36 +04:00
return err < 0 ? err : - EIO ;
}
return buf ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_select_corudata ( struct snd_i2c_device * device , int udata )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip = device - > private_data ;
2005-04-17 02:20:36 +04:00
int err ;
udata = udata ? CS8427_BSEL : 0 ;
if ( udata ! = ( chip - > regmap [ CS8427_REG_CSDATABUF ] & udata ) ) {
chip - > regmap [ CS8427_REG_CSDATABUF ] & = ~ CS8427_BSEL ;
chip - > regmap [ CS8427_REG_CSDATABUF ] | = udata ;
2005-11-17 16:17:19 +03:00
err = snd_cs8427_reg_write ( device , CS8427_REG_CSDATABUF ,
chip - > regmap [ CS8427_REG_CSDATABUF ] ) ;
2005-04-17 02:20:36 +04:00
if ( err < 0 )
return err ;
}
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_send_corudata ( struct snd_i2c_device * device ,
2005-04-17 02:20:36 +04:00
int udata ,
unsigned char * ndata ,
int count )
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip = device - > private_data ;
2006-06-30 17:31:25 +04:00
char * hw_data = udata ?
chip - > playback . hw_udata : chip - > playback . hw_status ;
2018-10-18 13:33:02 +03:00
unsigned char data [ 32 ] ;
2005-04-17 02:20:36 +04:00
int err , idx ;
if ( ! memcmp ( hw_data , ndata , count ) )
return 0 ;
2021-06-08 17:05:39 +03:00
err = snd_cs8427_select_corudata ( device , udata ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
memcpy ( hw_data , ndata , count ) ;
if ( udata ) {
memset ( data , 0 , sizeof ( data ) ) ;
if ( memcmp ( hw_data , data , count ) = = 0 ) {
chip - > regmap [ CS8427_REG_UDATABUF ] & = ~ CS8427_UBMMASK ;
2006-06-30 17:31:25 +04:00
chip - > regmap [ CS8427_REG_UDATABUF ] | = CS8427_UBMZEROS |
CS8427_EFTUI ;
err = snd_cs8427_reg_write ( device , CS8427_REG_UDATABUF ,
chip - > regmap [ CS8427_REG_UDATABUF ] ) ;
return err < 0 ? err : 0 ;
2005-04-17 02:20:36 +04:00
}
}
data [ 0 ] = CS8427_REG_AUTOINC | CS8427_REG_CORU_DATABUF ;
for ( idx = 0 ; idx < count ; idx + + )
2009-11-13 10:02:56 +03:00
data [ idx + 1 ] = bitrev8 ( ndata [ idx ] ) ;
2005-04-17 02:20:36 +04:00
if ( snd_i2c_sendbytes ( device , data , count + 1 ) ! = count + 1 )
return - EIO ;
return 1 ;
}
2005-11-17 16:17:19 +03:00
static void snd_cs8427_free ( struct snd_i2c_device * device )
2005-04-17 02:20:36 +04:00
{
kfree ( device - > private_data ) ;
}
2014-04-03 16:49:46 +04:00
int snd_cs8427_init ( struct snd_i2c_bus * bus ,
struct snd_i2c_device * device )
2005-04-17 02:20:36 +04:00
{
static unsigned char initvals1 [ ] = {
CS8427_REG_CONTROL1 | CS8427_REG_AUTOINC ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_CONTROL1: RMCK to OMCK, valid PCM audio, disable mutes,
TCBL = output */
2005-04-17 02:20:36 +04:00
CS8427_SWCLK | CS8427_TCBLDIR ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_CONTROL2: hold last valid audio sample, RMCK=256*Fs,
normal stereo operation */
2005-04-17 02:20:36 +04:00
0x00 ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_DATAFLOW: output drivers normal operation, Tx<=serial,
Rx = > serial */
2005-04-17 02:20:36 +04:00
CS8427_TXDSERIAL | CS8427_SPDAES3RECEIVER ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_CLOCKSOURCE: Run off, CMCK=256*Fs,
output time base = OMCK , input time base = recovered input clock ,
recovered input clock source is ILRCK changed to AES3INPUT
( workaround , see snd_cs8427_reset ) */
2005-04-17 02:20:36 +04:00
CS8427_RXDILRCK ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_SERIALINPUT: Serial audio input port data format = I2S,
24 - bit , 64 * Fsi */
2005-04-17 02:20:36 +04:00
CS8427_SIDEL | CS8427_SILRPOL ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_SERIALOUTPUT: Serial audio output port data format
= I2S , 24 - bit , 64 * Fsi */
2005-04-17 02:20:36 +04:00
CS8427_SODEL | CS8427_SOLRPOL ,
} ;
static unsigned char initvals2 [ ] = {
CS8427_REG_RECVERRMASK | CS8427_REG_AUTOINC ,
2006-06-30 17:31:25 +04:00
/* CS8427_REG_RECVERRMASK: unmask the input PLL clock, V, confidence,
biphase , parity status bits */
/* CS8427_UNLOCK | CS8427_V | CS8427_CONF | CS8427_BIP | CS8427_PAR,*/
2005-04-17 02:20:36 +04:00
0xff , /* set everything */
/* CS8427_REG_CSDATABUF:
Registers 32 - 55 window to CS buffer
Inhibit D - > E transfers from overwriting first 5 bytes of CS data .
Inhibit D - > E transfers ( all ) of CS data .
Allow E - > F transfer of CS data .
One byte mode ; both A / B channels get same written CB data .
A channel info is output to chip ' s EMPH * pin . */
CS8427_CBMR | CS8427_DETCI ,
/* CS8427_REG_UDATABUF:
Use internal buffer to transmit User ( U ) data .
Chip ' s U pin is an output .
Transmit all O ' s for user data .
Inhibit D - > E transfers .
Inhibit E - > F transfers . */
CS8427_UD | CS8427_EFTUI | CS8427_DETUI ,
} ;
2014-04-03 16:49:46 +04:00
struct cs8427 * chip = device - > private_data ;
2005-04-17 02:20:36 +04:00
int err ;
unsigned char buf [ 24 ] ;
snd_i2c_lock ( bus ) ;
2006-06-30 17:31:25 +04:00
err = snd_cs8427_reg_read ( device , CS8427_REG_ID_AND_VER ) ;
2007-09-20 19:34:57 +04:00
if ( err ! = CS8427_VER8427A ) {
/* give second chance */
snd_printk ( KERN_WARNING " invalid CS8427 signature 0x%x: "
" let me try again... \n " , err ) ;
err = snd_cs8427_reg_read ( device , CS8427_REG_ID_AND_VER ) ;
}
2006-06-30 17:31:25 +04:00
if ( err ! = CS8427_VER8427A ) {
2005-04-17 02:20:36 +04:00
snd_i2c_unlock ( bus ) ;
2005-11-17 16:17:19 +03:00
snd_printk ( KERN_ERR " unable to find CS8427 signature "
" (expected 0x%x, read 0x%x), \n " ,
CS8427_VER8427A , err ) ;
snd_printk ( KERN_ERR " initialization is not completed \n " ) ;
2005-04-17 02:20:36 +04:00
return - EFAULT ;
}
/* turn off run bit while making changes to configuration */
2006-06-30 17:31:25 +04:00
err = snd_cs8427_reg_write ( device , CS8427_REG_CLOCKSOURCE , 0x00 ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
goto __fail ;
/* send initial values */
memcpy ( chip - > regmap + ( initvals1 [ 0 ] & 0x7f ) , initvals1 + 1 , 6 ) ;
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , initvals1 , 7 ) ;
if ( err ! = 7 ) {
2005-04-17 02:20:36 +04:00
err = err < 0 ? err : - EIO ;
goto __fail ;
}
/* Turn off CS8427 interrupt stuff that is not used in hardware */
memset ( buf , 0 , 7 ) ;
/* from address 9 to 15 */
buf [ 0 ] = 9 ; /* register */
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , buf , 7 ) ;
if ( err ! = 7 )
2005-04-17 02:20:36 +04:00
goto __fail ;
/* send transfer initialization sequence */
memcpy ( chip - > regmap + ( initvals2 [ 0 ] & 0x7f ) , initvals2 + 1 , 3 ) ;
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , initvals2 , 4 ) ;
if ( err ! = 4 ) {
2005-04-17 02:20:36 +04:00
err = err < 0 ? err : - EIO ;
goto __fail ;
}
/* write default channel status bytes */
2008-05-08 15:01:32 +04:00
put_unaligned_le32 ( SNDRV_PCM_DEFAULT_CON_SPDIF , buf ) ;
2005-04-17 02:20:36 +04:00
memset ( buf + 4 , 0 , 24 - 4 ) ;
if ( snd_cs8427_send_corudata ( device , 0 , buf , 24 ) < 0 )
goto __fail ;
memcpy ( chip - > playback . def_status , buf , 24 ) ;
memcpy ( chip - > playback . pcm_status , buf , 24 ) ;
snd_i2c_unlock ( bus ) ;
/* turn on run bit and rock'n'roll */
2014-04-03 16:49:46 +04:00
snd_cs8427_reset ( device ) ;
return 0 ;
__fail :
snd_i2c_unlock ( bus ) ;
return err ;
}
EXPORT_SYMBOL ( snd_cs8427_init ) ;
int snd_cs8427_create ( struct snd_i2c_bus * bus ,
unsigned char addr ,
unsigned int reset_timeout ,
struct snd_i2c_device * * r_cs8427 )
{
int err ;
struct cs8427 * chip ;
struct snd_i2c_device * device ;
err = snd_i2c_device_create ( bus , " CS8427 " , CS8427_ADDR | ( addr & 7 ) ,
& device ) ;
if ( err < 0 )
return err ;
chip = device - > private_data = kzalloc ( sizeof ( * chip ) , GFP_KERNEL ) ;
if ( chip = = NULL ) {
snd_i2c_device_free ( device ) ;
return - ENOMEM ;
}
device - > private_free = snd_cs8427_free ;
2005-04-17 02:20:36 +04:00
if ( reset_timeout < 1 )
reset_timeout = 1 ;
chip - > reset_timeout = reset_timeout ;
2014-04-03 16:49:46 +04:00
err = snd_cs8427_init ( bus , device ) ;
if ( err )
goto __fail ;
2005-04-17 02:20:36 +04:00
#if 0 // it's nice for read tests
{
char buf [ 128 ] ;
int xx ;
buf [ 0 ] = 0x81 ;
snd_i2c_sendbytes ( device , buf , 1 ) ;
snd_i2c_readbytes ( device , buf , 127 ) ;
for ( xx = 0 ; xx < 127 ; xx + + )
2005-10-20 20:26:44 +04:00
printk ( KERN_DEBUG " reg[0x%x] = 0x%x \n " , xx + 1 , buf [ xx ] ) ;
2005-04-17 02:20:36 +04:00
}
# endif
if ( r_cs8427 )
* r_cs8427 = device ;
return 0 ;
__fail :
snd_i2c_device_free ( device ) ;
return err < 0 ? err : - EIO ;
}
2006-06-30 17:31:25 +04:00
EXPORT_SYMBOL ( snd_cs8427_create ) ;
2005-04-17 02:20:36 +04:00
/*
* Reset the chip using run bit , also lock PLL using ILRCK and
* put back AES3INPUT . This workaround is described in latest
* CS8427 datasheet , otherwise TXDSERIAL will not work .
*/
2005-11-17 16:17:19 +03:00
static void snd_cs8427_reset ( struct snd_i2c_device * cs8427 )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip ;
2005-04-17 02:20:36 +04:00
unsigned long end_time ;
2006-02-02 09:56:54 +03:00
int data , aes3input = 0 ;
2005-04-17 02:20:36 +04:00
2008-08-08 19:12:47 +04:00
if ( snd_BUG_ON ( ! cs8427 ) )
return ;
2005-04-17 02:20:36 +04:00
chip = cs8427 - > private_data ;
snd_i2c_lock ( cs8427 - > bus ) ;
2006-06-30 17:31:25 +04:00
if ( ( chip - > regmap [ CS8427_REG_CLOCKSOURCE ] & CS8427_RXDAES3INPUT ) = =
CS8427_RXDAES3INPUT ) /* AES3 bit is set */
2006-02-02 09:56:54 +03:00
aes3input = 1 ;
2005-04-17 02:20:36 +04:00
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] & = ~ ( CS8427_RUN | CS8427_RXDMASK ) ;
2005-11-17 16:17:19 +03:00
snd_cs8427_reg_write ( cs8427 , CS8427_REG_CLOCKSOURCE ,
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] ) ;
2005-04-17 02:20:36 +04:00
udelay ( 200 ) ;
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] | = CS8427_RUN | CS8427_RXDILRCK ;
2005-11-17 16:17:19 +03:00
snd_cs8427_reg_write ( cs8427 , CS8427_REG_CLOCKSOURCE ,
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] ) ;
2005-04-17 02:20:36 +04:00
udelay ( 200 ) ;
snd_i2c_unlock ( cs8427 - > bus ) ;
end_time = jiffies + chip - > reset_timeout ;
while ( time_after_eq ( end_time , jiffies ) ) {
snd_i2c_lock ( cs8427 - > bus ) ;
data = snd_cs8427_reg_read ( cs8427 , CS8427_REG_RECVERRORS ) ;
snd_i2c_unlock ( cs8427 - > bus ) ;
if ( ! ( data & CS8427_UNLOCK ) )
break ;
2005-10-24 17:02:37 +04:00
schedule_timeout_uninterruptible ( 1 ) ;
2005-04-17 02:20:36 +04:00
}
snd_i2c_lock ( cs8427 - > bus ) ;
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] & = ~ CS8427_RXDMASK ;
2006-02-02 09:56:54 +03:00
if ( aes3input )
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] | = CS8427_RXDAES3INPUT ;
2005-11-17 16:17:19 +03:00
snd_cs8427_reg_write ( cs8427 , CS8427_REG_CLOCKSOURCE ,
chip - > regmap [ CS8427_REG_CLOCKSOURCE ] ) ;
2005-04-17 02:20:36 +04:00
snd_i2c_unlock ( cs8427 - > bus ) ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_in_status_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 255 ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_in_status_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct snd_i2c_device * device = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int data ;
snd_i2c_lock ( device - > bus ) ;
data = snd_cs8427_reg_read ( device , kcontrol - > private_value ) ;
snd_i2c_unlock ( device - > bus ) ;
if ( data < 0 )
return data ;
ucontrol - > value . integer . value [ 0 ] = data ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_qsubcode_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_BYTES ;
uinfo - > count = 10 ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_qsubcode_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct snd_i2c_device * device = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
unsigned char reg = CS8427_REG_QSUBCODE ;
int err ;
snd_i2c_lock ( device - > bus ) ;
2021-06-08 17:05:39 +03:00
err = snd_i2c_sendbytes ( device , & reg , 1 ) ;
if ( err ! = 1 ) {
2006-06-30 17:31:25 +04:00
snd_printk ( KERN_ERR " unable to send register 0x%x byte "
" to CS8427 \n " , reg ) ;
2005-04-17 02:20:36 +04:00
snd_i2c_unlock ( device - > bus ) ;
return err < 0 ? err : - EIO ;
}
2006-06-30 17:31:25 +04:00
err = snd_i2c_readbytes ( device , ucontrol - > value . bytes . data , 10 ) ;
if ( err ! = 10 ) {
snd_printk ( KERN_ERR " unable to read Q-subcode bytes "
" from CS8427 \n " ) ;
2005-04-17 02:20:36 +04:00
snd_i2c_unlock ( device - > bus ) ;
return err < 0 ? err : - EIO ;
}
snd_i2c_unlock ( device - > bus ) ;
return 0 ;
}
2006-06-30 17:31:25 +04:00
static int snd_cs8427_spdif_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_IEC958 ;
uinfo - > count = 1 ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_spdif_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct snd_i2c_device * device = snd_kcontrol_chip ( kcontrol ) ;
struct cs8427 * chip = device - > private_data ;
2005-04-17 02:20:36 +04:00
snd_i2c_lock ( device - > bus ) ;
memcpy ( ucontrol - > value . iec958 . status , chip - > playback . def_status , 24 ) ;
snd_i2c_unlock ( device - > bus ) ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_spdif_put ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct snd_i2c_device * device = snd_kcontrol_chip ( kcontrol ) ;
struct cs8427 * chip = device - > private_data ;
unsigned char * status = kcontrol - > private_value ?
chip - > playback . pcm_status : chip - > playback . def_status ;
struct snd_pcm_runtime * runtime = chip - > playback . substream ?
chip - > playback . substream - > runtime : NULL ;
2005-04-17 02:20:36 +04:00
int err , change ;
snd_i2c_lock ( device - > bus ) ;
change = memcmp ( ucontrol - > value . iec958 . status , status , 24 ) ! = 0 ;
memcpy ( status , ucontrol - > value . iec958 . status , 24 ) ;
2006-06-30 17:31:25 +04:00
if ( change & & ( kcontrol - > private_value ?
runtime ! = NULL : runtime = = NULL ) ) {
2005-04-17 02:20:36 +04:00
err = snd_cs8427_send_corudata ( device , 0 , status , 24 ) ;
if ( err < 0 )
change = err ;
}
snd_i2c_unlock ( device - > bus ) ;
return change ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_spdif_mask_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_IEC958 ;
uinfo - > count = 1 ;
return 0 ;
}
2005-11-17 16:17:19 +03:00
static int snd_cs8427_spdif_mask_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
memset ( ucontrol - > value . iec958 . status , 0xff , 24 ) ;
return 0 ;
}
2020-01-03 11:16:50 +03:00
static const struct snd_kcontrol_new snd_cs8427_iec958_controls [ ] = {
2005-04-17 02:20:36 +04:00
{
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. info = snd_cs8427_in_status_info ,
. name = " IEC958 CS8427 Input Status " ,
2006-06-30 17:31:25 +04:00
. access = ( SNDRV_CTL_ELEM_ACCESS_READ |
SNDRV_CTL_ELEM_ACCESS_VOLATILE ) ,
2005-04-17 02:20:36 +04:00
. get = snd_cs8427_in_status_get ,
. private_value = 15 ,
} ,
{
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. info = snd_cs8427_in_status_info ,
. name = " IEC958 CS8427 Error Status " ,
2006-06-30 17:31:25 +04:00
. access = ( SNDRV_CTL_ELEM_ACCESS_READ |
SNDRV_CTL_ELEM_ACCESS_VOLATILE ) ,
2005-04-17 02:20:36 +04:00
. get = snd_cs8427_in_status_get ,
. private_value = 16 ,
} ,
{
. access = SNDRV_CTL_ELEM_ACCESS_READ ,
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. name = SNDRV_CTL_NAME_IEC958 ( " " , PLAYBACK , MASK ) ,
. info = snd_cs8427_spdif_mask_info ,
. get = snd_cs8427_spdif_mask_get ,
} ,
{
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. name = SNDRV_CTL_NAME_IEC958 ( " " , PLAYBACK , DEFAULT ) ,
. info = snd_cs8427_spdif_info ,
. get = snd_cs8427_spdif_get ,
. put = snd_cs8427_spdif_put ,
. private_value = 0
} ,
{
2006-06-30 17:31:25 +04:00
. access = ( SNDRV_CTL_ELEM_ACCESS_READWRITE |
SNDRV_CTL_ELEM_ACCESS_INACTIVE ) ,
2005-04-17 02:20:36 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. name = SNDRV_CTL_NAME_IEC958 ( " " , PLAYBACK , PCM_STREAM ) ,
. info = snd_cs8427_spdif_info ,
. get = snd_cs8427_spdif_get ,
. put = snd_cs8427_spdif_put ,
. private_value = 1
} ,
{
. iface = SNDRV_CTL_ELEM_IFACE_PCM ,
. info = snd_cs8427_qsubcode_info ,
. name = " IEC958 Q-subcode Capture Default " ,
2006-06-30 17:31:25 +04:00
. access = ( SNDRV_CTL_ELEM_ACCESS_READ |
SNDRV_CTL_ELEM_ACCESS_VOLATILE ) ,
2005-04-17 02:20:36 +04:00
. get = snd_cs8427_qsubcode_get
} } ;
2005-11-17 16:17:19 +03:00
int snd_cs8427_iec958_build ( struct snd_i2c_device * cs8427 ,
struct snd_pcm_substream * play_substream ,
struct snd_pcm_substream * cap_substream )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip = cs8427 - > private_data ;
struct snd_kcontrol * kctl ;
2005-04-17 02:20:36 +04:00
unsigned int idx ;
int err ;
2008-08-08 19:12:47 +04:00
if ( snd_BUG_ON ( ! play_substream | | ! cap_substream ) )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
for ( idx = 0 ; idx < ARRAY_SIZE ( snd_cs8427_iec958_controls ) ; idx + + ) {
kctl = snd_ctl_new1 ( & snd_cs8427_iec958_controls [ idx ] , cs8427 ) ;
if ( kctl = = NULL )
return - ENOMEM ;
kctl - > id . device = play_substream - > pcm - > device ;
kctl - > id . subdevice = play_substream - > number ;
err = snd_ctl_add ( cs8427 - > bus - > card , kctl ) ;
if ( err < 0 )
return err ;
2006-06-30 17:31:25 +04:00
if ( ! strcmp ( kctl - > id . name ,
SNDRV_CTL_NAME_IEC958 ( " " , PLAYBACK , PCM_STREAM ) ) )
2005-04-17 02:20:36 +04:00
chip - > playback . pcm_ctl = kctl ;
}
chip - > playback . substream = play_substream ;
chip - > capture . substream = cap_substream ;
2008-08-08 19:12:47 +04:00
if ( snd_BUG_ON ( ! chip - > playback . pcm_ctl ) )
return - EIO ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2006-06-30 17:31:25 +04:00
EXPORT_SYMBOL ( snd_cs8427_iec958_build ) ;
2005-11-17 16:17:19 +03:00
int snd_cs8427_iec958_active ( struct snd_i2c_device * cs8427 , int active )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip ;
2005-04-17 02:20:36 +04:00
2008-08-08 19:12:47 +04:00
if ( snd_BUG_ON ( ! cs8427 ) )
return - ENXIO ;
2005-04-17 02:20:36 +04:00
chip = cs8427 - > private_data ;
2023-04-05 23:12:19 +03:00
if ( active ) {
2006-06-30 17:31:25 +04:00
memcpy ( chip - > playback . pcm_status ,
chip - > playback . def_status , 24 ) ;
2023-04-05 23:12:19 +03:00
chip - > playback . pcm_ctl - > vd [ 0 ] . access & = ~ SNDRV_CTL_ELEM_ACCESS_INACTIVE ;
} else {
chip - > playback . pcm_ctl - > vd [ 0 ] . access | = SNDRV_CTL_ELEM_ACCESS_INACTIVE ;
}
2006-06-30 17:31:25 +04:00
snd_ctl_notify ( cs8427 - > bus - > card ,
SNDRV_CTL_EVENT_MASK_VALUE | SNDRV_CTL_EVENT_MASK_INFO ,
& chip - > playback . pcm_ctl - > id ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2006-06-30 17:31:25 +04:00
EXPORT_SYMBOL ( snd_cs8427_iec958_active ) ;
2005-11-17 16:17:19 +03:00
int snd_cs8427_iec958_pcm ( struct snd_i2c_device * cs8427 , unsigned int rate )
2005-04-17 02:20:36 +04:00
{
2005-11-17 16:17:19 +03:00
struct cs8427 * chip ;
2005-04-17 02:20:36 +04:00
char * status ;
int err , reset ;
2008-08-08 19:12:47 +04:00
if ( snd_BUG_ON ( ! cs8427 ) )
return - ENXIO ;
2005-04-17 02:20:36 +04:00
chip = cs8427 - > private_data ;
status = chip - > playback . pcm_status ;
snd_i2c_lock ( cs8427 - > bus ) ;
if ( status [ 0 ] & IEC958_AES0_PROFESSIONAL ) {
status [ 0 ] & = ~ IEC958_AES0_PRO_FS ;
switch ( rate ) {
case 32000 : status [ 0 ] | = IEC958_AES0_PRO_FS_32000 ; break ;
case 44100 : status [ 0 ] | = IEC958_AES0_PRO_FS_44100 ; break ;
case 48000 : status [ 0 ] | = IEC958_AES0_PRO_FS_48000 ; break ;
default : status [ 0 ] | = IEC958_AES0_PRO_FS_NOTID ; break ;
}
} else {
status [ 3 ] & = ~ IEC958_AES3_CON_FS ;
switch ( rate ) {
case 32000 : status [ 3 ] | = IEC958_AES3_CON_FS_32000 ; break ;
case 44100 : status [ 3 ] | = IEC958_AES3_CON_FS_44100 ; break ;
case 48000 : status [ 3 ] | = IEC958_AES3_CON_FS_48000 ; break ;
}
}
err = snd_cs8427_send_corudata ( cs8427 , 0 , status , 24 ) ;
if ( err > 0 )
snd_ctl_notify ( cs8427 - > bus - > card ,
SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > playback . pcm_ctl - > id ) ;
reset = chip - > rate ! = rate ;
chip - > rate = rate ;
snd_i2c_unlock ( cs8427 - > bus ) ;
if ( reset )
snd_cs8427_reset ( cs8427 ) ;
return err < 0 ? err : 0 ;
}
2006-06-30 17:31:25 +04:00
EXPORT_SYMBOL ( snd_cs8427_iec958_pcm ) ;