2017-12-14 11:19:43 +05:30
// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
// Copyright(c) 2015-17 Intel Corporation.
/*
* Soundwire Intel Master Driver
*/
# include <linux/acpi.h>
2019-08-21 13:58:20 -05:00
# include <linux/debugfs.h>
2017-12-14 11:19:43 +05:30
# include <linux/delay.h>
2019-09-16 13:57:38 -05:00
# include <linux/io.h>
2018-04-26 18:38:58 +05:30
# include <sound/pcm_params.h>
2020-07-16 23:09:46 +08:00
# include <linux/pm_runtime.h>
2018-04-26 18:38:58 +05:30
# include <sound/soc.h>
2017-12-14 11:19:43 +05:30
# include <linux/soundwire/sdw_registers.h>
# include <linux/soundwire/sdw.h>
# include <linux/soundwire/sdw_intel.h>
# include "cadence_master.h"
2019-08-21 13:58:20 -05:00
# include "bus.h"
2017-12-14 11:19:43 +05:30
# include "intel.h"
2020-07-16 23:09:39 +08:00
static int intel_wait_bit ( void __iomem * base , int offset , u32 mask , u32 target )
2017-12-14 11:19:43 +05:30
{
int timeout = 10 ;
u32 reg_read ;
do {
reg_read = readl ( base + offset ) ;
2020-07-16 23:09:39 +08:00
if ( ( reg_read & mask ) = = target )
2017-12-14 11:19:43 +05:30
return 0 ;
timeout - - ;
2020-07-16 23:09:39 +08:00
usleep_range ( 50 , 100 ) ;
2017-12-14 11:19:43 +05:30
} while ( timeout ! = 0 ) ;
return - EAGAIN ;
}
2020-07-16 23:09:39 +08:00
static int intel_clear_bit ( void __iomem * base , int offset , u32 value , u32 mask )
2017-12-14 11:19:43 +05:30
{
writel ( value , base + offset ) ;
2020-07-16 23:09:39 +08:00
return intel_wait_bit ( base , offset , mask , 0 ) ;
}
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:39 +08:00
static int intel_set_bit ( void __iomem * base , int offset , u32 value , u32 mask )
{
writel ( value , base + offset ) ;
return intel_wait_bit ( base , offset , mask , mask ) ;
2017-12-14 11:19:43 +05:30
}
2019-08-21 13:58:20 -05:00
/*
* debugfs
*/
# ifdef CONFIG_DEBUG_FS
# define RD_BUF (2 * PAGE_SIZE)
static ssize_t intel_sprintf ( void __iomem * mem , bool l ,
char * buf , size_t pos , unsigned int reg )
{
int value ;
if ( l )
value = intel_readl ( mem , reg ) ;
else
value = intel_readw ( mem , reg ) ;
return scnprintf ( buf + pos , RD_BUF - pos , " %4x \t %4x \n " , reg , value ) ;
}
static int intel_reg_show ( struct seq_file * s_file , void * data )
{
struct sdw_intel * sdw = s_file - > private ;
2020-02-14 19:47:36 -06:00
void __iomem * s = sdw - > link_res - > shim ;
void __iomem * a = sdw - > link_res - > alh ;
2019-08-21 13:58:20 -05:00
char * buf ;
ssize_t ret ;
int i , j ;
unsigned int links , reg ;
buf = kzalloc ( RD_BUF , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
2022-08-23 13:38:36 +08:00
links = intel_readl ( s , SDW_SHIM_LCAP ) & SDW_SHIM_LCAP_LCOUNT_MASK ;
2019-08-21 13:58:20 -05:00
ret = scnprintf ( buf , RD_BUF , " Register Value \n " ) ;
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n Shim \n " ) ;
for ( i = 0 ; i < links ; i + + ) {
reg = SDW_SHIM_LCAP + i * 4 ;
ret + = intel_sprintf ( s , true , buf , ret , reg ) ;
}
for ( i = 0 ; i < links ; i + + ) {
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n Link%d \n " , i ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTLSCAP ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTLS0CM ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTLS1CM ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTLS2CM ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTLS3CM ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_PCMSCAP ( i ) ) ;
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n PCMSyCH registers \n " ) ;
/*
* the value 10 is the number of PDIs . We will need a
* cleanup to remove hard - coded Intel configurations
* from cadence_master . c
*/
for ( j = 0 ; j < 10 ; j + + ) {
ret + = intel_sprintf ( s , false , buf , ret ,
SDW_SHIM_PCMSYCHM ( i , j ) ) ;
ret + = intel_sprintf ( s , false , buf , ret ,
SDW_SHIM_PCMSYCHC ( i , j ) ) ;
}
2022-08-23 13:38:40 +08:00
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n IOCTL, CTMCTL \n " ) ;
2019-08-21 13:58:20 -05:00
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_IOCTL ( i ) ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_CTMCTL ( i ) ) ;
}
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n Wake registers \n " ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_WAKEEN ) ;
ret + = intel_sprintf ( s , false , buf , ret , SDW_SHIM_WAKESTS ) ;
ret + = scnprintf ( buf + ret , RD_BUF - ret , " \n ALH STRMzCFG \n " ) ;
for ( i = 0 ; i < SDW_ALH_NUM_STREAMS ; i + + )
ret + = intel_sprintf ( a , true , buf , ret , SDW_ALH_STRMZCFG ( i ) ) ;
seq_printf ( s_file , " %s " , buf ) ;
kfree ( buf ) ;
return 0 ;
}
DEFINE_SHOW_ATTRIBUTE ( intel_reg ) ;
2020-09-21 03:32:06 +08:00
static int intel_set_m_datamode ( void * data , u64 value )
{
struct sdw_intel * sdw = data ;
struct sdw_bus * bus = & sdw - > cdns . bus ;
if ( value > SDW_PORT_DATA_MODE_STATIC_1 )
return - EINVAL ;
/* Userspace changed the hardware state behind the kernel's back */
add_taint ( TAINT_USER , LOCKDEP_STILL_OK ) ;
bus - > params . m_data_mode = value ;
return 0 ;
}
DEFINE_DEBUGFS_ATTRIBUTE ( intel_set_m_datamode_fops , NULL ,
intel_set_m_datamode , " %llu \n " ) ;
static int intel_set_s_datamode ( void * data , u64 value )
{
struct sdw_intel * sdw = data ;
struct sdw_bus * bus = & sdw - > cdns . bus ;
if ( value > SDW_PORT_DATA_MODE_STATIC_1 )
return - EINVAL ;
/* Userspace changed the hardware state behind the kernel's back */
add_taint ( TAINT_USER , LOCKDEP_STILL_OK ) ;
bus - > params . s_data_mode = value ;
return 0 ;
}
DEFINE_DEBUGFS_ATTRIBUTE ( intel_set_s_datamode_fops , NULL ,
intel_set_s_datamode , " %llu \n " ) ;
2019-08-21 13:58:20 -05:00
static void intel_debugfs_init ( struct sdw_intel * sdw )
{
struct dentry * root = sdw - > cdns . bus . debugfs ;
if ( ! root )
return ;
sdw - > debugfs = debugfs_create_dir ( " intel-sdw " , root ) ;
debugfs_create_file ( " intel-registers " , 0400 , sdw - > debugfs , sdw ,
& intel_reg_fops ) ;
2020-09-21 03:32:06 +08:00
debugfs_create_file ( " intel-m-datamode " , 0200 , sdw - > debugfs , sdw ,
& intel_set_m_datamode_fops ) ;
debugfs_create_file ( " intel-s-datamode " , 0200 , sdw - > debugfs , sdw ,
& intel_set_s_datamode_fops ) ;
2019-08-21 13:58:20 -05:00
sdw_cdns_debugfs_init ( & sdw - > cdns , sdw - > debugfs ) ;
}
static void intel_debugfs_exit ( struct sdw_intel * sdw )
{
debugfs_remove_recursive ( sdw - > debugfs ) ;
}
# else
static void intel_debugfs_init ( struct sdw_intel * sdw ) { }
static void intel_debugfs_exit ( struct sdw_intel * sdw ) { }
# endif /* CONFIG_DEBUG_FS */
2017-12-14 11:19:43 +05:30
/*
* shim ops
*/
2020-07-16 23:09:40 +08:00
/* this needs to be called with shim_lock */
static void intel_shim_glue_to_master_ip ( struct sdw_intel * sdw )
2017-12-14 11:19:43 +05:30
{
2020-02-14 19:47:36 -06:00
void __iomem * shim = sdw - > link_res - > shim ;
2017-12-14 11:19:43 +05:30
unsigned int link_id = sdw - > instance ;
2020-07-16 23:09:40 +08:00
u16 ioctl ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
/* Switch to MIP from Glue logic */
ioctl = intel_readw ( shim , SDW_SHIM_IOCTL ( link_id ) ) ;
ioctl & = ~ ( SDW_SHIM_IOCTL_DOE ) ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl & = ~ ( SDW_SHIM_IOCTL_DO ) ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl | = ( SDW_SHIM_IOCTL_MIF ) ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl & = ~ ( SDW_SHIM_IOCTL_BKE ) ;
ioctl & = ~ ( SDW_SHIM_IOCTL_COE ) ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
/* at this point Master IP has full control of the I/Os */
}
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
/* this needs to be called with shim_lock */
static void intel_shim_master_ip_to_glue ( struct sdw_intel * sdw )
{
unsigned int link_id = sdw - > instance ;
void __iomem * shim = sdw - > link_res - > shim ;
u16 ioctl ;
/* Glue logic */
ioctl = intel_readw ( shim , SDW_SHIM_IOCTL ( link_id ) ) ;
ioctl | = SDW_SHIM_IOCTL_BKE ;
ioctl | = SDW_SHIM_IOCTL_COE ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl & = ~ ( SDW_SHIM_IOCTL_MIF ) ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
/* at this point Integration Glue has full control of the I/Os */
}
2022-09-20 01:57:17 +08:00
/* this needs to be called with shim_lock */
static void intel_shim_init ( struct sdw_intel * sdw )
2020-07-16 23:09:40 +08:00
{
void __iomem * shim = sdw - > link_res - > shim ;
unsigned int link_id = sdw - > instance ;
2023-05-15 16:13:01 +08:00
u16 ioctl = 0 , act ;
2020-07-16 23:09:40 +08:00
/* Initialize Shim */
ioctl | = SDW_SHIM_IOCTL_BKE ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl | = SDW_SHIM_IOCTL_WPDD ;
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
usleep_range ( 10 , 15 ) ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:40 +08:00
ioctl | = SDW_SHIM_IOCTL_DO ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
ioctl | = SDW_SHIM_IOCTL_DOE ;
2017-12-14 11:19:43 +05:30
intel_writew ( shim , SDW_SHIM_IOCTL ( link_id ) , ioctl ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
intel_shim_glue_to_master_ip ( sdw ) ;
2017-12-14 11:19:43 +05:30
2023-05-15 16:13:01 +08:00
act = intel_readw ( shim , SDW_SHIM_CTMCTL ( link_id ) ) ;
2020-09-17 17:31:46 +05:30
u16p_replace_bits ( & act , 0x1 , SDW_SHIM_CTMCTL_DOAIS ) ;
2017-12-14 11:19:43 +05:30
act | = SDW_SHIM_CTMCTL_DACTQE ;
act | = SDW_SHIM_CTMCTL_DODS ;
intel_writew ( shim , SDW_SHIM_CTMCTL ( link_id ) , act ) ;
2020-07-16 23:09:40 +08:00
usleep_range ( 10 , 15 ) ;
}
2017-12-14 11:19:43 +05:30
2022-09-20 01:57:19 +08:00
static int intel_shim_check_wake ( struct sdw_intel * sdw )
{
void __iomem * shim ;
u16 wake_sts ;
2020-07-16 23:09:40 +08:00
2022-09-20 01:57:19 +08:00
shim = sdw - > link_res - > shim ;
wake_sts = intel_readw ( shim , SDW_SHIM_WAKESTS ) ;
return wake_sts & BIT ( sdw - > instance ) ;
2020-07-16 23:09:40 +08:00
}
2020-07-16 23:09:46 +08:00
static void intel_shim_wake ( struct sdw_intel * sdw , bool wake_enable )
2020-07-16 23:09:40 +08:00
{
void __iomem * shim = sdw - > link_res - > shim ;
unsigned int link_id = sdw - > instance ;
u16 wake_en , wake_sts ;
mutex_lock ( sdw - > link_res - > shim_lock ) ;
wake_en = intel_readw ( shim , SDW_SHIM_WAKEEN ) ;
if ( wake_enable ) {
/* Enable the wakeup */
wake_en | = ( SDW_SHIM_WAKEEN_ENABLE < < link_id ) ;
intel_writew ( shim , SDW_SHIM_WAKEEN , wake_en ) ;
} else {
/* Disable the wake up interrupt */
wake_en & = ~ ( SDW_SHIM_WAKEEN_ENABLE < < link_id ) ;
intel_writew ( shim , SDW_SHIM_WAKEEN , wake_en ) ;
/* Clear wake status */
wake_sts = intel_readw ( shim , SDW_SHIM_WAKESTS ) ;
2022-01-26 09:14:51 +08:00
wake_sts | = ( SDW_SHIM_WAKESTS_STATUS < < link_id ) ;
intel_writew ( shim , SDW_SHIM_WAKESTS , wake_sts ) ;
2020-07-16 23:09:40 +08:00
}
mutex_unlock ( sdw - > link_res - > shim_lock ) ;
}
2023-03-14 09:54:03 +08:00
static bool intel_check_cmdsync_unlocked ( struct sdw_intel * sdw )
{
void __iomem * shim = sdw - > link_res - > shim ;
int sync_reg ;
sync_reg = intel_readl ( shim , SDW_SHIM_SYNC ) ;
return ! ! ( sync_reg & SDW_SHIM_SYNC_CMDSYNC_MASK ) ;
}
2022-09-20 01:57:16 +08:00
static int intel_link_power_up ( struct sdw_intel * sdw )
{
unsigned int link_id = sdw - > instance ;
void __iomem * shim = sdw - > link_res - > shim ;
u32 * shim_mask = sdw - > link_res - > shim_mask ;
struct sdw_bus * bus = & sdw - > cdns . bus ;
struct sdw_master_prop * prop = & bus - > prop ;
u32 spa_mask , cpa_mask ;
u32 link_control ;
int ret = 0 ;
u32 syncprd ;
u32 sync_reg ;
mutex_lock ( sdw - > link_res - > shim_lock ) ;
/*
* The hardware relies on an internal counter , typically 4 kHz ,
* to generate the SoundWire SSP - which defines a ' safe '
* synchronization point between commands and audio transport
* and allows for multi link synchronization . The SYNCPRD value
* is only dependent on the oscillator clock provided to
* the IP , so adjust based on _DSD properties reported in DSDT
* tables . The values reported are based on either 24 MHz
* ( CNL / CML ) or 38.4 MHz ( ICL / TGL + ) .
*/
if ( prop - > mclk_freq % 6000000 )
syncprd = SDW_SHIM_SYNC_SYNCPRD_VAL_38_4 ;
else
syncprd = SDW_SHIM_SYNC_SYNCPRD_VAL_24 ;
if ( ! * shim_mask ) {
dev_dbg ( sdw - > cdns . dev , " powering up all links \n " ) ;
/* we first need to program the SyncPRD/CPU registers */
dev_dbg ( sdw - > cdns . dev ,
" first link up, programming SYNCPRD \n " ) ;
/* set SyncPRD period */
sync_reg = intel_readl ( shim , SDW_SHIM_SYNC ) ;
u32p_replace_bits ( & sync_reg , syncprd , SDW_SHIM_SYNC_SYNCPRD ) ;
/* Set SyncCPU bit */
sync_reg | = SDW_SHIM_SYNC_SYNCCPU ;
intel_writel ( shim , SDW_SHIM_SYNC , sync_reg ) ;
/* Link power up sequence */
link_control = intel_readl ( shim , SDW_SHIM_LCTL ) ;
/* only power-up enabled links */
spa_mask = FIELD_PREP ( SDW_SHIM_LCTL_SPA_MASK , sdw - > link_res - > link_mask ) ;
cpa_mask = FIELD_PREP ( SDW_SHIM_LCTL_CPA_MASK , sdw - > link_res - > link_mask ) ;
link_control | = spa_mask ;
ret = intel_set_bit ( shim , SDW_SHIM_LCTL , link_control , cpa_mask ) ;
if ( ret < 0 ) {
dev_err ( sdw - > cdns . dev , " Failed to power up link: %d \n " , ret ) ;
goto out ;
}
/* SyncCPU will change once link is active */
ret = intel_wait_bit ( shim , SDW_SHIM_SYNC ,
SDW_SHIM_SYNC_SYNCCPU , 0 ) ;
if ( ret < 0 ) {
dev_err ( sdw - > cdns . dev ,
" Failed to set SHIM_SYNC: %d \n " , ret ) ;
goto out ;
}
}
* shim_mask | = BIT ( link_id ) ;
sdw - > cdns . link_up = true ;
2022-09-20 01:57:17 +08:00
intel_shim_init ( sdw ) ;
2022-09-20 01:57:16 +08:00
out :
mutex_unlock ( sdw - > link_res - > shim_lock ) ;
return ret ;
}
2020-07-22 04:37:11 +08:00
static int intel_link_power_down ( struct sdw_intel * sdw )
2020-07-16 23:09:40 +08:00
{
2020-09-01 23:05:56 +08:00
u32 link_control , spa_mask , cpa_mask ;
2020-07-16 23:09:40 +08:00
unsigned int link_id = sdw - > instance ;
void __iomem * shim = sdw - > link_res - > shim ;
u32 * shim_mask = sdw - > link_res - > shim_mask ;
int ret = 0 ;
mutex_lock ( sdw - > link_res - > shim_lock ) ;
if ( ! ( * shim_mask & BIT ( link_id ) ) )
dev_err ( sdw - > cdns . dev ,
" %s: Unbalanced power-up/down calls \n " , __func__ ) ;
2021-08-18 10:49:52 +08:00
sdw - > cdns . link_up = false ;
intel_shim_master_ip_to_glue ( sdw ) ;
2020-07-16 23:09:40 +08:00
* shim_mask & = ~ BIT ( link_id ) ;
2020-09-01 23:05:56 +08:00
if ( ! * shim_mask ) {
2022-08-23 13:01:58 +08:00
dev_dbg ( sdw - > cdns . dev , " powering down all links \n " ) ;
2020-09-01 23:05:56 +08:00
/* Link power down sequence */
link_control = intel_readl ( shim , SDW_SHIM_LCTL ) ;
/* only power-down enabled links */
2020-09-03 17:15:02 +05:30
spa_mask = FIELD_PREP ( SDW_SHIM_LCTL_SPA_MASK , ~ sdw - > link_res - > link_mask ) ;
cpa_mask = FIELD_PREP ( SDW_SHIM_LCTL_CPA_MASK , sdw - > link_res - > link_mask ) ;
2020-09-01 23:05:56 +08:00
link_control & = spa_mask ;
ret = intel_clear_bit ( shim , SDW_SHIM_LCTL , link_control , cpa_mask ) ;
2021-08-18 10:49:52 +08:00
if ( ret < 0 ) {
dev_err ( sdw - > cdns . dev , " %s: could not power down link \n " , __func__ ) ;
/*
* we leave the sdw - > cdns . link_up flag as false since we ' ve disabled
* the link at this point and cannot handle interrupts any longer .
*/
}
2020-09-01 23:05:56 +08:00
}
2020-07-16 23:09:40 +08:00
mutex_unlock ( sdw - > link_res - > shim_lock ) ;
2017-12-14 11:19:43 +05:30
2021-08-18 10:49:52 +08:00
return ret ;
2017-12-14 11:19:43 +05:30
}
2020-07-16 23:09:41 +08:00
static void intel_shim_sync_arm ( struct sdw_intel * sdw )
{
void __iomem * shim = sdw - > link_res - > shim ;
u32 sync_reg ;
mutex_lock ( sdw - > link_res - > shim_lock ) ;
/* update SYNC register */
sync_reg = intel_readl ( shim , SDW_SHIM_SYNC ) ;
sync_reg | = ( SDW_SHIM_SYNC_CMDSYNC < < sdw - > instance ) ;
intel_writel ( shim , SDW_SHIM_SYNC , sync_reg ) ;
mutex_unlock ( sdw - > link_res - > shim_lock ) ;
}
2020-07-16 23:09:42 +08:00
static int intel_shim_sync_go_unlocked ( struct sdw_intel * sdw )
{
void __iomem * shim = sdw - > link_res - > shim ;
u32 sync_reg ;
2017-12-14 11:19:43 +05:30
2020-07-16 23:09:42 +08:00
/* Read SYNC register */
2017-12-14 11:19:43 +05:30
sync_reg = intel_readl ( shim , SDW_SHIM_SYNC ) ;
2020-07-16 23:09:42 +08:00
/*
* Set SyncGO bit to synchronously trigger a bank switch for
* all the masters . A write to SYNCGO bit clears CMDSYNC bit for all
* the Masters .
*/
sync_reg | = SDW_SHIM_SYNC_SYNCGO ;
2023-03-14 09:53:59 +08:00
intel_writel ( shim , SDW_SHIM_SYNC , sync_reg ) ;
2017-12-14 11:19:43 +05:30
2023-03-14 09:53:59 +08:00
return 0 ;
2017-12-14 11:19:43 +05:30
}
2020-09-01 23:05:49 +08:00
static int intel_shim_sync_go ( struct sdw_intel * sdw )
{
int ret ;
mutex_lock ( sdw - > link_res - > shim_lock ) ;
ret = intel_shim_sync_go_unlocked ( sdw ) ;
mutex_unlock ( sdw - > link_res - > shim_lock ) ;
return ret ;
}
2018-04-26 18:38:58 +05:30
/*
* PDI routines
*/
static void intel_pdi_init ( struct sdw_intel * sdw ,
2019-05-01 10:57:38 -05:00
struct sdw_cdns_stream_config * config )
2018-04-26 18:38:58 +05:30
{
2020-02-14 19:47:36 -06:00
void __iomem * shim = sdw - > link_res - > shim ;
2018-04-26 18:38:58 +05:30
unsigned int link_id = sdw - > instance ;
2021-12-24 10:10:34 +08:00
int pcm_cap ;
2018-04-26 18:38:58 +05:30
/* PCM Stream Capability */
pcm_cap = intel_readw ( shim , SDW_SHIM_PCMSCAP ( link_id ) ) ;
2020-09-03 17:15:02 +05:30
config - > pcm_bd = FIELD_GET ( SDW_SHIM_PCMSCAP_BSS , pcm_cap ) ;
config - > pcm_in = FIELD_GET ( SDW_SHIM_PCMSCAP_ISS , pcm_cap ) ;
config - > pcm_out = FIELD_GET ( SDW_SHIM_PCMSCAP_OSS , pcm_cap ) ;
2018-04-26 18:38:58 +05:30
2019-05-22 14:47:29 -05:00
dev_dbg ( sdw - > cdns . dev , " PCM cap bd:%d in:%d out:%d \n " ,
config - > pcm_bd , config - > pcm_in , config - > pcm_out ) ;
2018-04-26 18:38:58 +05:30
}
static int
2021-12-24 10:10:34 +08:00
intel_pdi_get_ch_cap ( struct sdw_intel * sdw , unsigned int pdi_num )
2018-04-26 18:38:58 +05:30
{
2020-02-14 19:47:36 -06:00
void __iomem * shim = sdw - > link_res - > shim ;
2018-04-26 18:38:58 +05:30
unsigned int link_id = sdw - > instance ;
int count ;
2021-12-24 10:10:34 +08:00
count = intel_readw ( shim , SDW_SHIM_PCMSYCHC ( link_id , pdi_num ) ) ;
2019-08-05 19:55:07 -05:00
2021-12-24 10:10:34 +08:00
/*
* WORKAROUND : on all existing Intel controllers , pdi
* number 2 reports channel count as 1 even though it
* supports 8 channels . Performing hardcoding for pdi
* number 2.
*/
if ( pdi_num = = 2 )
count = 7 ;
2018-04-26 18:38:58 +05:30
/* zero based values for channel count in register */
count + + ;
return count ;
}
static int intel_pdi_get_ch_update ( struct sdw_intel * sdw ,
2019-05-01 10:57:38 -05:00
struct sdw_cdns_pdi * pdi ,
unsigned int num_pdi ,
2021-12-24 10:10:34 +08:00
unsigned int * num_ch )
2018-04-26 18:38:58 +05:30
{
int i , ch_count = 0 ;
for ( i = 0 ; i < num_pdi ; i + + ) {
2021-12-24 10:10:34 +08:00
pdi - > ch_count = intel_pdi_get_ch_cap ( sdw , pdi - > num ) ;
2018-04-26 18:38:58 +05:30
ch_count + = pdi - > ch_count ;
pdi + + ;
}
* num_ch = ch_count ;
return 0 ;
}
static int intel_pdi_stream_ch_update ( struct sdw_intel * sdw ,
2021-12-24 10:10:34 +08:00
struct sdw_cdns_streams * stream )
2018-04-26 18:38:58 +05:30
{
intel_pdi_get_ch_update ( sdw , stream - > bd , stream - > num_bd ,
2021-12-24 10:10:34 +08:00
& stream - > num_ch_bd ) ;
2018-04-26 18:38:58 +05:30
intel_pdi_get_ch_update ( sdw , stream - > in , stream - > num_in ,
2021-12-24 10:10:34 +08:00
& stream - > num_ch_in ) ;
2018-04-26 18:38:58 +05:30
intel_pdi_get_ch_update ( sdw , stream - > out , stream - > num_out ,
2021-12-24 10:10:34 +08:00
& stream - > num_ch_out ) ;
2018-04-26 18:38:58 +05:30
return 0 ;
}
static void
intel_pdi_shim_configure ( struct sdw_intel * sdw , struct sdw_cdns_pdi * pdi )
{
2020-02-14 19:47:36 -06:00
void __iomem * shim = sdw - > link_res - > shim ;
2018-04-26 18:38:58 +05:30
unsigned int link_id = sdw - > instance ;
int pdi_conf = 0 ;
2019-10-22 18:29:48 -05:00
/* the Bulk and PCM streams are not contiguous */
pdi - > intel_alh_id = ( link_id * 16 ) + pdi - > num + 3 ;
if ( pdi - > num > = 2 )
pdi - > intel_alh_id + = 2 ;
2018-04-26 18:38:58 +05:30
/*
* Program stream parameters to stream SHIM register
* This is applicable for PCM stream only .
*/
if ( pdi - > type ! = SDW_STREAM_PCM )
return ;
if ( pdi - > dir = = SDW_DATA_DIR_RX )
pdi_conf | = SDW_SHIM_PCMSYCM_DIR ;
else
pdi_conf & = ~ ( SDW_SHIM_PCMSYCM_DIR ) ;
2020-09-17 17:31:46 +05:30
u32p_replace_bits ( & pdi_conf , pdi - > intel_alh_id , SDW_SHIM_PCMSYCM_STREAM ) ;
u32p_replace_bits ( & pdi_conf , pdi - > l_ch_num , SDW_SHIM_PCMSYCM_LCHN ) ;
u32p_replace_bits ( & pdi_conf , pdi - > h_ch_num , SDW_SHIM_PCMSYCM_HCHN ) ;
2018-04-26 18:38:58 +05:30
intel_writew ( shim , SDW_SHIM_PCMSYCHM ( link_id , pdi - > num ) , pdi_conf ) ;
}
static void
intel_pdi_alh_configure ( struct sdw_intel * sdw , struct sdw_cdns_pdi * pdi )
{
2020-02-14 19:47:36 -06:00
void __iomem * alh = sdw - > link_res - > alh ;
2018-04-26 18:38:58 +05:30
unsigned int link_id = sdw - > instance ;
unsigned int conf ;
2019-10-22 18:29:48 -05:00
/* the Bulk and PCM streams are not contiguous */
pdi - > intel_alh_id = ( link_id * 16 ) + pdi - > num + 3 ;
if ( pdi - > num > = 2 )
pdi - > intel_alh_id + = 2 ;
2018-04-26 18:38:58 +05:30
/* Program Stream config ALH register */
conf = intel_readl ( alh , SDW_ALH_STRMZCFG ( pdi - > intel_alh_id ) ) ;
2020-09-17 17:31:46 +05:30
u32p_replace_bits ( & conf , SDW_ALH_STRMZCFG_DMAT_VAL , SDW_ALH_STRMZCFG_DMAT ) ;
u32p_replace_bits ( & conf , pdi - > ch_count - 1 , SDW_ALH_STRMZCFG_CHN ) ;
2018-04-26 18:38:58 +05:30
intel_writel ( alh , SDW_ALH_STRMZCFG ( pdi - > intel_alh_id ) , conf ) ;
}
2019-12-11 19:45:02 -06:00
static int intel_params_stream ( struct sdw_intel * sdw ,
2023-05-15 15:10:38 +08:00
struct snd_pcm_substream * substream ,
2019-05-01 10:57:38 -05:00
struct snd_soc_dai * dai ,
2019-12-11 19:45:02 -06:00
struct snd_pcm_hw_params * hw_params ,
int link_id , int alh_stream_id )
2018-04-26 18:39:05 +05:30
{
2020-02-14 19:47:36 -06:00
struct sdw_intel_link_res * res = sdw - > link_res ;
2019-12-11 19:45:02 -06:00
struct sdw_intel_stream_params_data params_data ;
2019-08-05 19:55:06 -05:00
2023-05-15 15:10:38 +08:00
params_data . substream = substream ;
2019-12-11 19:45:02 -06:00
params_data . dai = dai ;
params_data . hw_params = hw_params ;
params_data . link_id = link_id ;
params_data . alh_stream_id = alh_stream_id ;
2018-04-26 18:39:05 +05:30
2019-12-11 19:45:02 -06:00
if ( res - > ops & & res - > ops - > params_stream & & res - > dev )
return res - > ops - > params_stream ( res - > dev ,
& params_data ) ;
2018-04-26 18:39:05 +05:30
return - EIO ;
}
/*
* DAI routines
*/
static int intel_hw_params ( struct snd_pcm_substream * substream ,
2019-05-01 10:57:38 -05:00
struct snd_pcm_hw_params * params ,
struct snd_soc_dai * dai )
2018-04-26 18:39:05 +05:30
{
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
struct sdw_intel * sdw = cdns_to_intel ( cdns ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
2019-09-16 14:23:46 -05:00
struct sdw_cdns_pdi * pdi ;
2018-04-26 18:39:05 +05:30
struct sdw_stream_config sconfig ;
struct sdw_port_config * pconfig ;
2019-09-16 14:23:46 -05:00
int ch , dir ;
int ret ;
2018-04-26 18:39:05 +05:30
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime )
2018-04-26 18:39:05 +05:30
return - EIO ;
ch = params_channels ( params ) ;
if ( substream - > stream = = SNDRV_PCM_STREAM_CAPTURE )
dir = SDW_DATA_DIR_RX ;
else
dir = SDW_DATA_DIR_TX ;
2021-12-24 10:10:34 +08:00
pdi = sdw_cdns_alloc_pdi ( cdns , & cdns - > pcm , ch , dir , dai - > id ) ;
2019-09-16 14:23:46 -05:00
if ( ! pdi ) {
ret = - EINVAL ;
goto error ;
2018-04-26 18:39:05 +05:30
}
2019-09-16 14:23:46 -05:00
/* do run-time configurations for SHIM, ALH and PDI/PORT */
intel_pdi_shim_configure ( sdw , pdi ) ;
intel_pdi_alh_configure ( sdw , pdi ) ;
sdw_cdns_config_stream ( cdns , ch , dir , pdi ) ;
2018-04-26 18:39:05 +05:30
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
/* store pdi and hw_params, may be needed in prepare step */
2022-11-01 10:35:20 +08:00
dai_runtime - > paused = false ;
dai_runtime - > suspended = false ;
dai_runtime - > pdi = pdi ;
2018-04-26 18:39:05 +05:30
/* Inform DSP about PDI stream number */
2023-05-15 15:10:38 +08:00
ret = intel_params_stream ( sdw , substream , dai , params ,
2019-12-11 19:45:02 -06:00
sdw - > instance ,
2019-09-16 14:23:46 -05:00
pdi - > intel_alh_id ) ;
if ( ret )
goto error ;
2018-04-26 18:39:05 +05:30
sconfig . direction = dir ;
sconfig . ch_count = ch ;
sconfig . frame_rate = params_rate ( params ) ;
2022-11-01 10:35:20 +08:00
sconfig . type = dai_runtime - > stream_type ;
2018-04-26 18:39:05 +05:30
2021-12-24 10:10:34 +08:00
sconfig . bps = snd_pcm_format_width ( params_format ( params ) ) ;
2018-04-26 18:39:05 +05:30
/* Port configuration */
2020-12-29 21:50:12 +08:00
pconfig = kzalloc ( sizeof ( * pconfig ) , GFP_KERNEL ) ;
2018-04-26 18:39:05 +05:30
if ( ! pconfig ) {
ret = - ENOMEM ;
2019-09-16 14:23:46 -05:00
goto error ;
2018-04-26 18:39:05 +05:30
}
2019-09-16 14:23:46 -05:00
pconfig - > num = pdi - > num ;
pconfig - > ch_mask = ( 1 < < ch ) - 1 ;
2018-04-26 18:39:05 +05:30
ret = sdw_stream_add_master ( & cdns - > bus , & sconfig ,
2022-11-01 10:35:20 +08:00
pconfig , 1 , dai_runtime - > stream ) ;
2019-09-16 14:23:46 -05:00
if ( ret )
2019-05-01 10:57:45 -05:00
dev_err ( cdns - > dev , " add master to stream failed:%d \n " , ret ) ;
2018-04-26 18:39:05 +05:30
kfree ( pconfig ) ;
2019-09-16 14:23:46 -05:00
error :
2018-04-26 18:39:05 +05:30
return ret ;
}
2020-02-14 19:47:37 -06:00
static int intel_prepare ( struct snd_pcm_substream * substream ,
struct snd_soc_dai * dai )
{
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
struct sdw_intel * sdw = cdns_to_intel ( cdns ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
int ch , dir ;
2020-09-04 04:47:39 +08:00
int ret = 0 ;
2020-02-14 19:47:37 -06:00
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime ) {
dev_err ( dai - > dev , " failed to get dai runtime in %s \n " ,
2020-02-14 19:47:37 -06:00
__func__ ) ;
return - EIO ;
}
2022-11-01 10:35:20 +08:00
if ( dai_runtime - > suspended ) {
soundwire: intel: don't save hw_params for use in prepare
The existing code copies the hw_params pointer and reuses it later in
.prepare, specifically to re-initialize the ALH DMA channel
information that's lost in suspend-resume cycles.
This is not needed, we can directly access the information from the
substream/rtd - as done for the HDAudio DAIs in
sound/soc/sof/intel/hda-dai.c
In addition, using the saved pointer causes the suspend-resume test
cases to fail on specific platforms, depending on which version of GCC
is used. Péter Ujfalusi and I have spent long hours to root-cause this
problem that was reported by the Intel CI first with 6.2-rc1 and again
v6.3-rc1. In the latter case we were lucky that the problem was 100%
reproducible on local test devices, and found out that adding a
dev_dbg() or adding a call to usleep_range() just before accessing the
saved pointer "fixed" the issue. With errors appearing just by
changing the compiler version or minor changes in the code generated,
clearly we have a memory management Heisenbug.
The root-cause seems to be that the hw_params pointer is not
persistent. The soc-pcm code allocates the hw_params structure on the
stack, and passes it to the BE dailink hw_params and DAIs
hw_params. Saving such a pointer and reusing it later during the
.prepare stage cannot possibly work reliably, it's broken-by-design
since v5.10. It's astonishing that the problem was not seen earlier.
This simple fix will have to be back-ported to -stable, due to changes
to avoid the use of the get/set_dmadata routines this patch will only
apply on kernels older than v6.1.
Fixes: a5a0239c27fe ("soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming")
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Reviewed-by: Ranjani Sridharan <ranjani.sridharan@linux.intel.com>
Reviewed-by: Péter Ujfalusi <peter.ujfalusi@linux.intel.com>
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Link: https://lore.kernel.org/r/20230321022642.1426611-1-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2023-03-21 10:26:42 +08:00
struct snd_soc_pcm_runtime * rtd = asoc_substream_to_rtd ( substream ) ;
struct snd_pcm_hw_params * hw_params ;
hw_params = & rtd - > dpcm [ substream - > stream ] . hw_params ;
2022-11-01 10:35:20 +08:00
dai_runtime - > suspended = false ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
/*
* . prepare ( ) is called after system resume , where we
* need to reinitialize the SHIM / ALH / Cadence IP .
* . prepare ( ) is also called to deal with underflows ,
* but in those cases we cannot touch ALH / SHIM
* registers
*/
/* configure stream */
soundwire: intel: don't save hw_params for use in prepare
The existing code copies the hw_params pointer and reuses it later in
.prepare, specifically to re-initialize the ALH DMA channel
information that's lost in suspend-resume cycles.
This is not needed, we can directly access the information from the
substream/rtd - as done for the HDAudio DAIs in
sound/soc/sof/intel/hda-dai.c
In addition, using the saved pointer causes the suspend-resume test
cases to fail on specific platforms, depending on which version of GCC
is used. Péter Ujfalusi and I have spent long hours to root-cause this
problem that was reported by the Intel CI first with 6.2-rc1 and again
v6.3-rc1. In the latter case we were lucky that the problem was 100%
reproducible on local test devices, and found out that adding a
dev_dbg() or adding a call to usleep_range() just before accessing the
saved pointer "fixed" the issue. With errors appearing just by
changing the compiler version or minor changes in the code generated,
clearly we have a memory management Heisenbug.
The root-cause seems to be that the hw_params pointer is not
persistent. The soc-pcm code allocates the hw_params structure on the
stack, and passes it to the BE dailink hw_params and DAIs
hw_params. Saving such a pointer and reusing it later during the
.prepare stage cannot possibly work reliably, it's broken-by-design
since v5.10. It's astonishing that the problem was not seen earlier.
This simple fix will have to be back-ported to -stable, due to changes
to avoid the use of the get/set_dmadata routines this patch will only
apply on kernels older than v6.1.
Fixes: a5a0239c27fe ("soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming")
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Reviewed-by: Ranjani Sridharan <ranjani.sridharan@linux.intel.com>
Reviewed-by: Péter Ujfalusi <peter.ujfalusi@linux.intel.com>
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Link: https://lore.kernel.org/r/20230321022642.1426611-1-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2023-03-21 10:26:42 +08:00
ch = params_channels ( hw_params ) ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
if ( substream - > stream = = SNDRV_PCM_STREAM_CAPTURE )
dir = SDW_DATA_DIR_RX ;
else
dir = SDW_DATA_DIR_TX ;
2022-11-01 10:35:20 +08:00
intel_pdi_shim_configure ( sdw , dai_runtime - > pdi ) ;
intel_pdi_alh_configure ( sdw , dai_runtime - > pdi ) ;
sdw_cdns_config_stream ( cdns , ch , dir , dai_runtime - > pdi ) ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
/* Inform DSP about PDI stream number */
2023-05-15 15:10:38 +08:00
ret = intel_params_stream ( sdw , substream , dai ,
soundwire: intel: don't save hw_params for use in prepare
The existing code copies the hw_params pointer and reuses it later in
.prepare, specifically to re-initialize the ALH DMA channel
information that's lost in suspend-resume cycles.
This is not needed, we can directly access the information from the
substream/rtd - as done for the HDAudio DAIs in
sound/soc/sof/intel/hda-dai.c
In addition, using the saved pointer causes the suspend-resume test
cases to fail on specific platforms, depending on which version of GCC
is used. Péter Ujfalusi and I have spent long hours to root-cause this
problem that was reported by the Intel CI first with 6.2-rc1 and again
v6.3-rc1. In the latter case we were lucky that the problem was 100%
reproducible on local test devices, and found out that adding a
dev_dbg() or adding a call to usleep_range() just before accessing the
saved pointer "fixed" the issue. With errors appearing just by
changing the compiler version or minor changes in the code generated,
clearly we have a memory management Heisenbug.
The root-cause seems to be that the hw_params pointer is not
persistent. The soc-pcm code allocates the hw_params structure on the
stack, and passes it to the BE dailink hw_params and DAIs
hw_params. Saving such a pointer and reusing it later during the
.prepare stage cannot possibly work reliably, it's broken-by-design
since v5.10. It's astonishing that the problem was not seen earlier.
This simple fix will have to be back-ported to -stable, due to changes
to avoid the use of the get/set_dmadata routines this patch will only
apply on kernels older than v6.1.
Fixes: a5a0239c27fe ("soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming")
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Reviewed-by: Ranjani Sridharan <ranjani.sridharan@linux.intel.com>
Reviewed-by: Péter Ujfalusi <peter.ujfalusi@linux.intel.com>
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Link: https://lore.kernel.org/r/20230321022642.1426611-1-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2023-03-21 10:26:42 +08:00
hw_params ,
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
sdw - > instance ,
2022-11-01 10:35:20 +08:00
dai_runtime - > pdi - > intel_alh_id ) ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
}
return ret ;
2020-02-14 19:47:37 -06:00
}
2018-04-26 18:39:05 +05:30
static int
intel_hw_free ( struct snd_pcm_substream * substream , struct snd_soc_dai * dai )
{
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
2018-04-26 18:39:05 +05:30
int ret ;
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime )
2018-04-26 18:39:05 +05:30
return - EIO ;
2020-09-04 04:47:39 +08:00
/*
* The sdw stream state will transition to RELEASED when stream - >
* master_list is empty . So the stream state will transition to
* DEPREPARED for the first cpu - dai and to RELEASED for the last
* cpu - dai .
*/
2022-11-01 10:35:20 +08:00
ret = sdw_stream_remove_master ( & cdns - > bus , dai_runtime - > stream ) ;
2020-02-14 19:47:40 -06:00
if ( ret < 0 ) {
2019-05-01 10:57:45 -05:00
dev_err ( dai - > dev , " remove master from stream %s failed: %d \n " ,
2022-11-01 10:35:20 +08:00
dai_runtime - > stream - > name , ret ) ;
2020-02-14 19:47:40 -06:00
return ret ;
}
2018-04-26 18:39:05 +05:30
2022-11-01 10:35:20 +08:00
dai_runtime - > pdi = NULL ;
soundwire: intel: reinitialize IP+DSP in .prepare(), but only when resuming
The .prepare() callback is invoked for normal streaming, underflows or
during the system resume transition. In the latter case, the context
for the ALH PDIs is lost, and the DSP is not initialized properly
either, but the bus parameters don't need to be recomputed.
Conversely, when doing a regular .prepare() during an underflow, the
ALH/SHIM registers shall not be changed as the hardware cannot be
reprogrammed after the DMA started (hardware spec requirement).
This patch adds storage of PDI and hw_params in the DAI dma context,
and the difference between the types of .prepare() usages is handled
via a simple boolean, updated when suspending, and tested for in the
.prepare() case.
Signed-off-by: Bard Liao <yung-chuan.liao@linux.intel.com>
Signed-off-by: Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Link: https://lore.kernel.org/r/20200817152923.3259-6-yung-chuan.liao@linux.intel.com
Signed-off-by: Vinod Koul <vkoul@kernel.org>
2020-08-17 23:29:16 +08:00
2020-02-14 19:47:40 -06:00
return 0 ;
2018-04-26 18:39:05 +05:30
}
static int intel_pcm_set_sdw_stream ( struct snd_soc_dai * dai ,
2019-05-01 10:57:38 -05:00
void * stream , int direction )
2018-04-26 18:39:05 +05:30
{
2021-12-24 10:10:34 +08:00
return cdns_set_sdw_stream ( dai , stream , direction ) ;
2018-04-26 18:39:05 +05:30
}
2020-07-01 02:43:52 +08:00
static void * intel_get_sdw_stream ( struct snd_soc_dai * dai ,
int direction )
{
2022-11-01 10:35:21 +08:00
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
2020-07-01 02:43:52 +08:00
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime )
2020-09-04 04:47:37 +08:00
return ERR_PTR ( - EINVAL ) ;
2020-07-01 02:43:52 +08:00
2022-11-01 10:35:20 +08:00
return dai_runtime - > stream ;
2020-07-01 02:43:52 +08:00
}
2021-12-24 10:10:30 +08:00
static int intel_trigger ( struct snd_pcm_substream * substream , int cmd , struct snd_soc_dai * dai )
{
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
2021-12-24 10:10:30 +08:00
int ret = 0 ;
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime ) {
dev_err ( dai - > dev , " failed to get dai runtime in %s \n " ,
2021-12-24 10:10:30 +08:00
__func__ ) ;
return - EIO ;
}
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_SUSPEND :
/*
* The . prepare callback is used to deal with xruns and resume operations .
* In the case of xruns , the DMAs and SHIM registers cannot be touched ,
* but for resume operations the DMAs and SHIM registers need to be initialized .
* the . trigger callback is used to track the suspend case only .
*/
2022-11-01 10:35:20 +08:00
dai_runtime - > suspended = true ;
2021-12-24 10:10:30 +08:00
break ;
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
2022-11-01 10:35:20 +08:00
dai_runtime - > paused = true ;
2021-12-24 10:10:30 +08:00
break ;
case SNDRV_PCM_TRIGGER_STOP :
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
2022-11-01 10:35:20 +08:00
dai_runtime - > paused = false ;
2021-12-24 10:10:30 +08:00
break ;
default :
break ;
}
return ret ;
}
2022-06-21 17:56:40 -05:00
static int intel_component_probe ( struct snd_soc_component * component )
{
int ret ;
/*
* make sure the device is pm_runtime_active before initiating
* bus transactions during the card registration .
* We use pm_runtime_resume ( ) here , without taking a reference
* and releasing it immediately .
*/
ret = pm_runtime_resume ( component - > dev ) ;
if ( ret < 0 & & ret ! = - EACCES )
return ret ;
return 0 ;
}
2021-12-24 10:10:30 +08:00
static int intel_component_dais_suspend ( struct snd_soc_component * component )
{
struct snd_soc_dai * dai ;
/*
* In the corner case where a SUSPEND happens during a PAUSE , the ALSA core
* does not throw the TRIGGER_SUSPEND . This leaves the DAIs in an unbalanced state .
* Since the component suspend is called last , we can trap this corner case
* and force the DAIs to release their resources .
*/
for_each_component_dais ( component , dai ) {
struct sdw_cdns * cdns = snd_soc_dai_get_drvdata ( dai ) ;
2022-11-01 10:35:20 +08:00
struct sdw_cdns_dai_runtime * dai_runtime ;
2021-12-24 10:10:30 +08:00
2022-11-01 10:35:21 +08:00
dai_runtime = cdns - > dai_runtime_array [ dai - > id ] ;
2021-12-24 10:10:30 +08:00
2022-11-01 10:35:20 +08:00
if ( ! dai_runtime )
2021-12-24 10:10:30 +08:00
continue ;
2022-11-01 10:35:20 +08:00
if ( dai_runtime - > suspended )
2021-12-24 10:10:30 +08:00
continue ;
2018-04-26 18:39:05 +05:30
2023-05-15 15:10:41 +08:00
if ( dai_runtime - > paused )
2022-11-01 10:35:20 +08:00
dai_runtime - > suspended = true ;
2021-12-24 10:10:30 +08:00
}
return 0 ;
}
2018-10-27 15:34:42 +02:00
static const struct snd_soc_dai_ops intel_pcm_dai_ops = {
2018-04-26 18:39:05 +05:30
. hw_params = intel_hw_params ,
2020-02-14 19:47:37 -06:00
. prepare = intel_prepare ,
2018-04-26 18:39:05 +05:30
. hw_free = intel_hw_free ,
2021-12-24 10:10:30 +08:00
. trigger = intel_trigger ,
2021-12-24 10:10:31 +08:00
. set_stream = intel_pcm_set_sdw_stream ,
. get_stream = intel_get_sdw_stream ,
2018-04-26 18:39:05 +05:30
} ;
static const struct snd_soc_component_driver dai_component = {
2022-06-23 13:51:16 +01:00
. name = " soundwire " ,
sound updates for 6.0-rc1
As diffstat shows, we've had lots of developments in a wide range
at this time; the majority of changes are about ASoC, including
subsystem-wide cleanups, continued SOF / Intel updates and a
bunch of new drivers (as usual), while there have been some
significant (but almost invisible) improvements in ALSA core
side, too. Below are some highlights:
Core:
- Faster lookups of control elements with Xarray; normal user
won't notice, but on the devices with tons of control elements,
it can be visibly faster
- Support for input validation for controls; this will harden
for badly written drivers in general with a slight overhead
- Deferred async signal handling for working around the potential
deadlocks
- Cleanup / refactoring raw MIDI locking code
ASoC:
- Restructing of the set_fmt() callbacks for making things clearer
in situations like CODEC to CODEC links
- Clean up and modernizing the DAI naming scheme setups
- Merge of more of the Intel AVS driver stack, including some
board integrations
- New version 4 mechanism for communication with SOF DSPs
- Suppoort for dynamically selecting the PLL to use at runtime on
i.MX platforms
- Improvements for CODEC to CODEC support in the generic cards
- Support for AMD Jadeite and various machines, AMD RPL, Intel
MetorLake DSPs, Mediatek MT8186 DSPs and MT6366, nVidia Tegra
MDDRC, OPE and PEQ, NXP TFA9890, Qualcomm SDM845, WCD9335 and
WAS883x, and Texas Instruments TAS2780
HD- and USB-audio:
- Continued improvement for CS35L41 (sub)codec support
- More quirks for various devices (HP, Lenovo, Dell, Clevo)
-----BEGIN PGP SIGNATURE-----
iQJCBAABCAAsFiEEIXTw5fNLNI7mMiVaLtJE4w1nLE8FAmLr5bcOHHRpd2FpQHN1
c2UuZGUACgkQLtJE4w1nLE93IQ/+OleeiGv7C487QN5MrBCkdFnSAiXsXDArcMgo
Gt6XLubH54et1tqi2ms4gRQOqr4HiBelERuqmaCLMZfEgVDc0VhJnf2jjhluYq9+
o9+kcYKul6qTZeNZLPjEX8pBvDe7HzOl7yep++ZnKeH6DAKNQQLDjVuOcQU/BXdY
kL8vYrLI3zfqj/pCePb5xpkBx4XdCrE3TfiCr3tAHVg9MyvSGOJyWs02mEBqQRnc
rlLmkjQVQyln/AGK4RAPMmrrFytktAvBjmIDyFL7kAzhdxe0ouNzTvdxESeojNJv
CVo/p3hl/+0LYjpD2x9v2pQuifOfpjwSCy6f8jsaF366sMwR1D45h051prILsxAF
05D5AOwMCnnJdFQFxw3mIkoDva3/PRX8GFfHsXoz+efc5Ibp8ksNYVgAJ3D2TTtt
7nAYMn0dimVDtw2LHiKantgWAs/rOqD3hDzGxFj2sR662ahsHr8pT8csnJAGoBvW
7kgx7ZzFo/wSyZJqVqV7p4g6J79ScehRwhqoiwZau9Eo+PhuxZUKvm4RwGFh0Vvg
GbiVRPfLV4xQd/pDin6qRX1M7cgPc62qGLkhQHAzrX6H5ipwIbQrpyDGLjwdSEp8
XcQmzCG1zGOvb9A8BY1VBOQXxZRTqN58XujbZ6hsms7Uw8sqagxpYLA/e1bvt1E1
RQoHQRw=
=1n0/
-----END PGP SIGNATURE-----
Merge tag 'sound-6.0-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound
Pull sound updates from Takashi Iwai:
"As the diffstat shows, we've had lots of developments in a wide range
at this time; the majority of changes are about ASoC, including
subsystem-wide cleanups, continued SOF / Intel updates and a bunch of
new drivers (as usual), while there have been some significant (but
almost invisible) improvements in ALSA core side, too.
Below are some highlights:
Core:
- Faster lookups of control elements with Xarray; normal user won't
notice, but on the devices with tons of control elements, it can be
visibly faster
- Support for input validation for controls; this will harden for
badly written drivers in general with a slight overhead
- Deferred async signal handling for working around the potential
deadlocks
- Cleanup / refactoring raw MIDI locking code
ASoC:
- Restructing of the set_fmt() callbacks for making things clearer in
situations like CODEC to CODEC links
- Clean up and modernizing the DAI naming scheme setups
- Merge of more of the Intel AVS driver stack, including some board
integrations
- New version 4 mechanism for communication with SOF DSPs
- Suppoort for dynamically selecting the PLL to use at runtime on
i.MX platforms
- Improvements for CODEC to CODEC support in the generic cards
- Support for AMD Jadeite and various machines, AMD RPL, Intel
MetorLake DSPs, Mediatek MT8186 DSPs and MT6366, nVidia Tegra
MDDRC, OPE and PEQ, NXP TFA9890, Qualcomm SDM845, WCD9335 and
WAS883x, and Texas Instruments TAS2780
HD- and USB-audio:
- Continued improvement for CS35L41 (sub)codec support
- More quirks for various devices (HP, Lenovo, Dell, Clevo)"
* tag 'sound-6.0-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound: (778 commits)
ALSA: hda/realtek: Add quirk for HP Spectre x360 15-eb0xxx
ALSA: line6: Replace sprintf() with sysfs_emit()
ALSA: hda: Replace sprintf() with sysfs_emit()
ALSA: pcm: Replace sprintf() with sysfs_emit()
ALSA: core: Replace scnprintf() with sysfs_emit()
ALSA: control-led: Replace sprintf() with sysfs_emit()
ALSA: aoa: Replace sprintf() with sysfs_emit()
ALSA: ac97: Replace sprintf() with sysfs_emit()
ALSA: hda/realtek: Add quirk for Clevo NV45PZ
ALSA: hda/realtek: Add quirk for Lenovo Yoga9 14IAP7
ALSA: control: Use deferred fasync helper
ALSA: pcm: Use deferred fasync helper
ALSA: timer: Use deferred fasync helper
ALSA: core: Add async signal helpers
ASoC: q6asm: use kcalloc() instead of kzalloc()
ACPI: scan: Add CLSA0101 Laptop Support
ALSA: hda: cs35l41: Support CLSA0101
ALSA: hda: cs35l41: Use the CS35L41 HDA internal define
ASoC: dt-bindings: use spi-peripheral-props.yaml
ASoC: codecs: va-macro: use fsgen as clock
...
2022-08-06 10:19:51 -07:00
. probe = intel_component_probe ,
2022-06-23 13:51:16 +01:00
. suspend = intel_component_dais_suspend ,
. legacy_dai_naming = 1 ,
2018-04-26 18:39:05 +05:30
} ;
static int intel_create_dai ( struct sdw_cdns * cdns ,
2019-05-01 10:57:38 -05:00
struct snd_soc_dai_driver * dais ,
enum intel_pdi_type type ,
2021-12-24 10:10:34 +08:00
u32 num , u32 off , u32 max_ch )
2018-04-26 18:39:05 +05:30
{
int i ;
if ( num = = 0 )
return 0 ;
for ( i = off ; i < ( off + num ) ; i + + ) {
2020-06-17 11:35:36 -05:00
dais [ i ] . name = devm_kasprintf ( cdns - > dev , GFP_KERNEL ,
" SDW%d Pin%d " ,
cdns - > instance , i ) ;
2018-04-26 18:39:05 +05:30
if ( ! dais [ i ] . name )
return - ENOMEM ;
if ( type = = INTEL_PDI_BD | | type = = INTEL_PDI_OUT ) {
dais [ i ] . playback . channels_min = 1 ;
dais [ i ] . playback . channels_max = max_ch ;
}
if ( type = = INTEL_PDI_BD | | type = = INTEL_PDI_IN ) {
2019-06-06 12:23:04 +01:00
dais [ i ] . capture . channels_min = 1 ;
dais [ i ] . capture . channels_max = max_ch ;
2018-04-26 18:39:05 +05:30
}
2021-12-24 10:10:34 +08:00
dais [ i ] . ops = & intel_pcm_dai_ops ;
2018-04-26 18:39:05 +05:30
}
return 0 ;
}
static int intel_register_dai ( struct sdw_intel * sdw )
{
2022-11-01 10:35:21 +08:00
struct sdw_cdns_dai_runtime * * dai_runtime_array ;
2022-09-20 01:57:14 +08:00
struct sdw_cdns_stream_config config ;
2018-04-26 18:39:05 +05:30
struct sdw_cdns * cdns = & sdw - > cdns ;
struct sdw_cdns_streams * stream ;
struct snd_soc_dai_driver * dais ;
int num_dai , ret , off = 0 ;
2022-09-20 01:57:14 +08:00
/* Read the PDI config and initialize cadence PDI */
intel_pdi_init ( sdw , & config ) ;
ret = sdw_cdns_pdi_init ( cdns , config ) ;
if ( ret )
return ret ;
2023-03-14 09:53:58 +08:00
intel_pdi_stream_ch_update ( sdw , & sdw - > cdns . pcm ) ;
2022-09-20 01:57:14 +08:00
2018-04-26 18:39:05 +05:30
/* DAIs are created based on total number of PDIs supported */
2021-12-24 10:10:34 +08:00
num_dai = cdns - > pcm . num_pdi ;
2018-04-26 18:39:05 +05:30
2022-11-01 10:35:21 +08:00
dai_runtime_array = devm_kcalloc ( cdns - > dev , num_dai ,
sizeof ( struct sdw_cdns_dai_runtime * ) ,
GFP_KERNEL ) ;
if ( ! dai_runtime_array )
return - ENOMEM ;
cdns - > dai_runtime_array = dai_runtime_array ;
2018-04-26 18:39:05 +05:30
dais = devm_kcalloc ( cdns - > dev , num_dai , sizeof ( * dais ) , GFP_KERNEL ) ;
if ( ! dais )
return - ENOMEM ;
/* Create PCM DAIs */
stream = & cdns - > pcm ;
2019-09-16 14:23:43 -05:00
ret = intel_create_dai ( cdns , dais , INTEL_PDI_IN , cdns - > pcm . num_in ,
2021-12-24 10:10:34 +08:00
off , stream - > num_ch_in ) ;
2018-04-26 18:39:05 +05:30
if ( ret )
return ret ;
off + = cdns - > pcm . num_in ;
2019-05-02 16:29:25 +05:30
ret = intel_create_dai ( cdns , dais , INTEL_PDI_OUT , cdns - > pcm . num_out ,
2021-12-24 10:10:34 +08:00
off , stream - > num_ch_out ) ;
2018-04-26 18:39:05 +05:30
if ( ret )
return ret ;
off + = cdns - > pcm . num_out ;
2019-05-02 16:29:25 +05:30
ret = intel_create_dai ( cdns , dais , INTEL_PDI_BD , cdns - > pcm . num_bd ,
2021-12-24 10:10:34 +08:00
off , stream - > num_ch_bd ) ;
2018-04-26 18:39:05 +05:30
if ( ret )
return ret ;
2022-09-20 01:57:12 +08:00
return devm_snd_soc_register_component ( cdns - > dev , & dai_component ,
dais , num_dai ) ;
2018-04-26 18:39:05 +05:30
}
2022-11-11 09:31:28 +08:00
const struct sdw_intel_hw_ops sdw_intel_cnl_hw_ops = {
2022-11-11 09:31:29 +08:00
. debugfs_init = intel_debugfs_init ,
. debugfs_exit = intel_debugfs_exit ,
2022-11-11 09:31:30 +08:00
. register_dai = intel_register_dai ,
2022-11-11 09:31:31 +08:00
. check_clock_stop = intel_check_clock_stop ,
. start_bus = intel_start_bus ,
. start_bus_after_reset = intel_start_bus_after_reset ,
. start_bus_after_clock_stop = intel_start_bus_after_clock_stop ,
. stop_bus = intel_stop_bus ,
2022-11-11 09:31:32 +08:00
. link_power_up = intel_link_power_up ,
. link_power_down = intel_link_power_down ,
2022-11-11 09:31:33 +08:00
. shim_check_wake = intel_shim_check_wake ,
. shim_wake = intel_shim_wake ,
2022-11-11 09:31:28 +08:00
. pre_bank_switch = intel_pre_bank_switch ,
. post_bank_switch = intel_post_bank_switch ,
2023-03-14 09:54:00 +08:00
. sync_arm = intel_shim_sync_arm ,
. sync_go_unlocked = intel_shim_sync_go_unlocked ,
. sync_go = intel_shim_sync_go ,
2023-03-14 09:54:03 +08:00
. sync_check_cmdsync_unlocked = intel_check_cmdsync_unlocked ,
2022-11-11 09:31:28 +08:00
} ;
EXPORT_SYMBOL_NS ( sdw_intel_cnl_hw_ops , SOUNDWIRE_INTEL ) ;