2022-03-17 20:50:28 +03:00
// SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
//
// This file is provided under a dual BSD/GPLv2 license. When using or
// redistributing this file, you may do so under either license.
//
// Copyright(c) 2021 Intel Corporation. All rights reserved.
//
//
2022-04-05 20:26:58 +03:00
# include <sound/sof/stream.h>
# include <sound/sof/control.h>
2022-03-17 20:50:28 +03:00
# include "sof-priv.h"
2022-04-05 20:27:00 +03:00
# include "sof-audio.h"
2022-03-17 20:50:28 +03:00
# include "ipc3-ops.h"
2022-04-05 20:26:56 +03:00
# include "ops.h"
2022-04-05 20:27:00 +03:00
typedef void ( * ipc3_rx_callback ) ( struct snd_sof_dev * sdev , void * msg_buf ) ;
2022-04-05 20:26:56 +03:00
# if IS_ENABLED(CONFIG_SND_SOC_SOF_DEBUG_VERBOSE_IPC)
static void ipc3_log_header ( struct device * dev , u8 * text , u32 cmd )
{
u8 * str ;
u8 * str2 = NULL ;
u32 glb ;
u32 type ;
bool vdbg = false ;
glb = cmd & SOF_GLB_TYPE_MASK ;
type = cmd & SOF_CMD_TYPE_MASK ;
switch ( glb ) {
case SOF_IPC_GLB_REPLY :
str = " GLB_REPLY " ; break ;
case SOF_IPC_GLB_COMPOUND :
str = " GLB_COMPOUND " ; break ;
case SOF_IPC_GLB_TPLG_MSG :
str = " GLB_TPLG_MSG " ;
switch ( type ) {
case SOF_IPC_TPLG_COMP_NEW :
str2 = " COMP_NEW " ; break ;
case SOF_IPC_TPLG_COMP_FREE :
str2 = " COMP_FREE " ; break ;
case SOF_IPC_TPLG_COMP_CONNECT :
str2 = " COMP_CONNECT " ; break ;
case SOF_IPC_TPLG_PIPE_NEW :
str2 = " PIPE_NEW " ; break ;
case SOF_IPC_TPLG_PIPE_FREE :
str2 = " PIPE_FREE " ; break ;
case SOF_IPC_TPLG_PIPE_CONNECT :
str2 = " PIPE_CONNECT " ; break ;
case SOF_IPC_TPLG_PIPE_COMPLETE :
str2 = " PIPE_COMPLETE " ; break ;
case SOF_IPC_TPLG_BUFFER_NEW :
str2 = " BUFFER_NEW " ; break ;
case SOF_IPC_TPLG_BUFFER_FREE :
str2 = " BUFFER_FREE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_PM_MSG :
str = " GLB_PM_MSG " ;
switch ( type ) {
case SOF_IPC_PM_CTX_SAVE :
str2 = " CTX_SAVE " ; break ;
case SOF_IPC_PM_CTX_RESTORE :
str2 = " CTX_RESTORE " ; break ;
case SOF_IPC_PM_CTX_SIZE :
str2 = " CTX_SIZE " ; break ;
case SOF_IPC_PM_CLK_SET :
str2 = " CLK_SET " ; break ;
case SOF_IPC_PM_CLK_GET :
str2 = " CLK_GET " ; break ;
case SOF_IPC_PM_CLK_REQ :
str2 = " CLK_REQ " ; break ;
case SOF_IPC_PM_CORE_ENABLE :
str2 = " CORE_ENABLE " ; break ;
case SOF_IPC_PM_GATE :
str2 = " GATE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_COMP_MSG :
str = " GLB_COMP_MSG " ;
switch ( type ) {
case SOF_IPC_COMP_SET_VALUE :
str2 = " SET_VALUE " ; break ;
case SOF_IPC_COMP_GET_VALUE :
str2 = " GET_VALUE " ; break ;
case SOF_IPC_COMP_SET_DATA :
str2 = " SET_DATA " ; break ;
case SOF_IPC_COMP_GET_DATA :
str2 = " GET_DATA " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_STREAM_MSG :
str = " GLB_STREAM_MSG " ;
switch ( type ) {
case SOF_IPC_STREAM_PCM_PARAMS :
str2 = " PCM_PARAMS " ; break ;
case SOF_IPC_STREAM_PCM_PARAMS_REPLY :
str2 = " PCM_REPLY " ; break ;
case SOF_IPC_STREAM_PCM_FREE :
str2 = " PCM_FREE " ; break ;
case SOF_IPC_STREAM_TRIG_START :
str2 = " TRIG_START " ; break ;
case SOF_IPC_STREAM_TRIG_STOP :
str2 = " TRIG_STOP " ; break ;
case SOF_IPC_STREAM_TRIG_PAUSE :
str2 = " TRIG_PAUSE " ; break ;
case SOF_IPC_STREAM_TRIG_RELEASE :
str2 = " TRIG_RELEASE " ; break ;
case SOF_IPC_STREAM_TRIG_DRAIN :
str2 = " TRIG_DRAIN " ; break ;
case SOF_IPC_STREAM_TRIG_XRUN :
str2 = " TRIG_XRUN " ; break ;
case SOF_IPC_STREAM_POSITION :
vdbg = true ;
str2 = " POSITION " ; break ;
case SOF_IPC_STREAM_VORBIS_PARAMS :
str2 = " VORBIS_PARAMS " ; break ;
case SOF_IPC_STREAM_VORBIS_FREE :
str2 = " VORBIS_FREE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_FW_READY :
str = " FW_READY " ; break ;
case SOF_IPC_GLB_DAI_MSG :
str = " GLB_DAI_MSG " ;
switch ( type ) {
case SOF_IPC_DAI_CONFIG :
str2 = " CONFIG " ; break ;
case SOF_IPC_DAI_LOOPBACK :
str2 = " LOOPBACK " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_TRACE_MSG :
str = " GLB_TRACE_MSG " ;
switch ( type ) {
case SOF_IPC_TRACE_DMA_PARAMS :
str2 = " DMA_PARAMS " ; break ;
case SOF_IPC_TRACE_DMA_POSITION :
str2 = " DMA_POSITION " ; break ;
case SOF_IPC_TRACE_DMA_PARAMS_EXT :
str2 = " DMA_PARAMS_EXT " ; break ;
case SOF_IPC_TRACE_FILTER_UPDATE :
str2 = " FILTER_UPDATE " ; break ;
case SOF_IPC_TRACE_DMA_FREE :
str2 = " DMA_FREE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_TEST_MSG :
str = " GLB_TEST_MSG " ;
switch ( type ) {
case SOF_IPC_TEST_IPC_FLOOD :
str2 = " IPC_FLOOD " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_DEBUG :
str = " GLB_DEBUG " ;
switch ( type ) {
case SOF_IPC_DEBUG_MEM_USAGE :
str2 = " MEM_USAGE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
case SOF_IPC_GLB_PROBE :
str = " GLB_PROBE " ;
switch ( type ) {
case SOF_IPC_PROBE_INIT :
str2 = " INIT " ; break ;
case SOF_IPC_PROBE_DEINIT :
str2 = " DEINIT " ; break ;
case SOF_IPC_PROBE_DMA_ADD :
str2 = " DMA_ADD " ; break ;
case SOF_IPC_PROBE_DMA_INFO :
str2 = " DMA_INFO " ; break ;
case SOF_IPC_PROBE_DMA_REMOVE :
str2 = " DMA_REMOVE " ; break ;
case SOF_IPC_PROBE_POINT_ADD :
str2 = " POINT_ADD " ; break ;
case SOF_IPC_PROBE_POINT_INFO :
str2 = " POINT_INFO " ; break ;
case SOF_IPC_PROBE_POINT_REMOVE :
str2 = " POINT_REMOVE " ; break ;
default :
str2 = " unknown type " ; break ;
}
break ;
default :
str = " unknown GLB command " ; break ;
}
if ( str2 ) {
if ( vdbg )
dev_vdbg ( dev , " %s: 0x%x: %s: %s \n " , text , cmd , str , str2 ) ;
else
dev_dbg ( dev , " %s: 0x%x: %s: %s \n " , text , cmd , str , str2 ) ;
} else {
dev_dbg ( dev , " %s: 0x%x: %s \n " , text , cmd , str ) ;
}
}
# else
static inline void ipc3_log_header ( struct device * dev , u8 * text , u32 cmd )
{
if ( ( cmd & SOF_GLB_TYPE_MASK ) ! = SOF_IPC_GLB_TRACE_MSG )
dev_dbg ( dev , " %s: 0x%x \n " , text , cmd ) ;
}
# endif
2022-04-05 20:26:59 +03:00
static int sof_ipc3_get_reply ( struct snd_sof_dev * sdev )
{
struct snd_sof_ipc_msg * msg = sdev - > msg ;
struct sof_ipc_reply * reply ;
int ret = 0 ;
/* get the generic reply */
reply = msg - > reply_data ;
snd_sof_dsp_mailbox_read ( sdev , sdev - > host_box . offset , reply , sizeof ( * reply ) ) ;
if ( reply - > error < 0 )
return reply - > error ;
if ( ! reply - > hdr . size ) {
/* Reply should always be >= sizeof(struct sof_ipc_reply) */
if ( msg - > reply_size )
dev_err ( sdev - > dev ,
" empty reply received, expected %zu bytes \n " ,
msg - > reply_size ) ;
else
dev_err ( sdev - > dev , " empty reply received \n " ) ;
return - EINVAL ;
}
if ( msg - > reply_size > 0 ) {
if ( reply - > hdr . size = = msg - > reply_size ) {
ret = 0 ;
} else if ( reply - > hdr . size < msg - > reply_size ) {
dev_dbg ( sdev - > dev ,
" reply size (%u) is less than expected (%zu) \n " ,
reply - > hdr . size , msg - > reply_size ) ;
msg - > reply_size = reply - > hdr . size ;
ret = 0 ;
} else {
dev_err ( sdev - > dev ,
" reply size (%u) exceeds the buffer size (%zu) \n " ,
reply - > hdr . size , msg - > reply_size ) ;
ret = - EINVAL ;
}
/*
* get the full message if reply - > hdr . size < = msg - > reply_size
* and the reply - > hdr . size > sizeof ( struct sof_ipc_reply )
*/
if ( ! ret & & msg - > reply_size > sizeof ( * reply ) )
snd_sof_dsp_mailbox_read ( sdev , sdev - > host_box . offset ,
msg - > reply_data , msg - > reply_size ) ;
}
return ret ;
}
2022-04-05 20:26:56 +03:00
/* wait for IPC message reply */
static int ipc3_wait_tx_done ( struct snd_sof_ipc * ipc , void * reply_data )
{
struct snd_sof_ipc_msg * msg = & ipc - > msg ;
struct sof_ipc_cmd_hdr * hdr = msg - > msg_data ;
struct snd_sof_dev * sdev = ipc - > sdev ;
int ret ;
/* wait for DSP IPC completion */
ret = wait_event_timeout ( msg - > waitq , msg - > ipc_complete ,
msecs_to_jiffies ( sdev - > ipc_timeout ) ) ;
if ( ret = = 0 ) {
dev_err ( sdev - > dev ,
" ipc tx timed out for %#x (msg/reply size: %d/%zu) \n " ,
hdr - > cmd , hdr - > size , msg - > reply_size ) ;
snd_sof_handle_fw_exception ( ipc - > sdev ) ;
ret = - ETIMEDOUT ;
} else {
ret = msg - > reply_error ;
if ( ret < 0 ) {
dev_err ( sdev - > dev ,
" ipc tx error for %#x (msg/reply size: %d/%zu): %d \n " ,
hdr - > cmd , hdr - > size , msg - > reply_size , ret ) ;
} else {
ipc3_log_header ( sdev - > dev , " ipc tx succeeded " , hdr - > cmd ) ;
if ( msg - > reply_size )
/* copy the data returned from DSP */
memcpy ( reply_data , msg - > reply_data ,
msg - > reply_size ) ;
}
/* re-enable dumps after successful IPC tx */
if ( sdev - > ipc_dump_printed ) {
sdev - > dbg_dump_printed = false ;
sdev - > ipc_dump_printed = false ;
}
}
return ret ;
}
/* send IPC message from host to DSP */
static int ipc3_tx_msg_unlocked ( struct snd_sof_ipc * ipc ,
void * msg_data , size_t msg_bytes ,
void * reply_data , size_t reply_bytes )
{
struct sof_ipc_cmd_hdr * hdr = msg_data ;
struct snd_sof_dev * sdev = ipc - > sdev ;
int ret ;
ret = sof_ipc_send_msg ( sdev , msg_data , msg_bytes , reply_bytes ) ;
if ( ret ) {
dev_err_ratelimited ( sdev - > dev ,
" %s: ipc message send for %#x failed: %d \n " ,
__func__ , hdr - > cmd , ret ) ;
return ret ;
}
ipc3_log_header ( sdev - > dev , " ipc tx " , hdr - > cmd ) ;
/* now wait for completion */
return ipc3_wait_tx_done ( ipc , reply_data ) ;
}
static int sof_ipc3_tx_msg ( struct snd_sof_dev * sdev , void * msg_data , size_t msg_bytes ,
void * reply_data , size_t reply_bytes , bool no_pm )
{
struct snd_sof_ipc * ipc = sdev - > ipc ;
int ret ;
if ( ! msg_data | | msg_bytes < sizeof ( struct sof_ipc_cmd_hdr ) ) {
dev_err_ratelimited ( sdev - > dev , " No IPC message to send \n " ) ;
return - EINVAL ;
}
if ( ! no_pm ) {
const struct sof_dsp_power_state target_state = {
. state = SOF_DSP_PM_D0 ,
} ;
/* ensure the DSP is in D0 before sending a new IPC */
ret = snd_sof_dsp_set_power_state ( sdev , & target_state ) ;
if ( ret < 0 ) {
dev_err ( sdev - > dev , " %s: resuming DSP failed: %d \n " ,
__func__ , ret ) ;
return ret ;
}
}
/* Serialise IPC TX */
mutex_lock ( & ipc - > tx_mutex ) ;
ret = ipc3_tx_msg_unlocked ( ipc , msg_data , msg_bytes , reply_data , reply_bytes ) ;
mutex_unlock ( & ipc - > tx_mutex ) ;
return ret ;
}
2022-03-17 20:50:28 +03:00
2022-04-05 20:26:58 +03:00
static int sof_ipc3_set_get_data ( struct snd_sof_dev * sdev , void * data , size_t data_bytes ,
bool set )
{
size_t msg_bytes , hdr_bytes , payload_size , send_bytes ;
struct sof_ipc_ctrl_data * cdata = data ;
struct sof_ipc_ctrl_data * cdata_chunk ;
struct snd_sof_ipc * ipc = sdev - > ipc ;
size_t offset = 0 ;
u8 * src , * dst ;
u32 num_msg ;
int ret = 0 ;
int i ;
if ( ! cdata | | data_bytes < sizeof ( * cdata ) )
return - EINVAL ;
if ( ( cdata - > rhdr . hdr . cmd & SOF_GLB_TYPE_MASK ) ! = SOF_IPC_GLB_COMP_MSG ) {
dev_err ( sdev - > dev , " %s: Not supported message type of %#x \n " ,
__func__ , cdata - > rhdr . hdr . cmd ) ;
return - EINVAL ;
}
/* send normal size ipc in one part */
if ( cdata - > rhdr . hdr . size < = ipc - > max_payload_size )
return sof_ipc3_tx_msg ( sdev , cdata , cdata - > rhdr . hdr . size ,
cdata , cdata - > rhdr . hdr . size , false ) ;
cdata_chunk = kzalloc ( ipc - > max_payload_size , GFP_KERNEL ) ;
if ( ! cdata_chunk )
return - ENOMEM ;
switch ( cdata - > type ) {
case SOF_CTRL_TYPE_VALUE_CHAN_GET :
case SOF_CTRL_TYPE_VALUE_CHAN_SET :
hdr_bytes = sizeof ( struct sof_ipc_ctrl_data ) ;
if ( set ) {
src = ( u8 * ) cdata - > chanv ;
dst = ( u8 * ) cdata_chunk - > chanv ;
} else {
src = ( u8 * ) cdata_chunk - > chanv ;
dst = ( u8 * ) cdata - > chanv ;
}
break ;
case SOF_CTRL_TYPE_DATA_GET :
case SOF_CTRL_TYPE_DATA_SET :
hdr_bytes = sizeof ( struct sof_ipc_ctrl_data ) + sizeof ( struct sof_abi_hdr ) ;
if ( set ) {
src = ( u8 * ) cdata - > data - > data ;
dst = ( u8 * ) cdata_chunk - > data - > data ;
} else {
src = ( u8 * ) cdata_chunk - > data - > data ;
dst = ( u8 * ) cdata - > data - > data ;
}
break ;
default :
kfree ( cdata_chunk ) ;
return - EINVAL ;
}
msg_bytes = cdata - > rhdr . hdr . size - hdr_bytes ;
payload_size = ipc - > max_payload_size - hdr_bytes ;
num_msg = DIV_ROUND_UP ( msg_bytes , payload_size ) ;
/* copy the header data */
memcpy ( cdata_chunk , cdata , hdr_bytes ) ;
/* Serialise IPC TX */
mutex_lock ( & sdev - > ipc - > tx_mutex ) ;
/* copy the payload data in a loop */
for ( i = 0 ; i < num_msg ; i + + ) {
send_bytes = min ( msg_bytes , payload_size ) ;
cdata_chunk - > num_elems = send_bytes ;
cdata_chunk - > rhdr . hdr . size = hdr_bytes + send_bytes ;
cdata_chunk - > msg_index = i ;
msg_bytes - = send_bytes ;
cdata_chunk - > elems_remaining = msg_bytes ;
if ( set )
memcpy ( dst , src + offset , send_bytes ) ;
ret = ipc3_tx_msg_unlocked ( sdev - > ipc ,
cdata_chunk , cdata_chunk - > rhdr . hdr . size ,
cdata_chunk , cdata_chunk - > rhdr . hdr . size ) ;
if ( ret < 0 )
break ;
if ( ! set )
memcpy ( dst + offset , src , send_bytes ) ;
offset + = payload_size ;
}
mutex_unlock ( & sdev - > ipc - > tx_mutex ) ;
kfree ( cdata_chunk ) ;
return ret ;
}
2022-04-05 20:27:00 +03:00
/* IPC stream position. */
static void ipc3_period_elapsed ( struct snd_sof_dev * sdev , u32 msg_id )
{
struct snd_soc_component * scomp = sdev - > component ;
struct snd_sof_pcm_stream * stream ;
struct sof_ipc_stream_posn posn ;
struct snd_sof_pcm * spcm ;
int direction , ret ;
spcm = snd_sof_find_spcm_comp ( scomp , msg_id , & direction ) ;
if ( ! spcm ) {
dev_err ( sdev - > dev , " period elapsed for unknown stream, msg_id %d \n " ,
msg_id ) ;
return ;
}
stream = & spcm - > stream [ direction ] ;
ret = snd_sof_ipc_msg_data ( sdev , stream - > substream , & posn , sizeof ( posn ) ) ;
if ( ret < 0 ) {
dev_warn ( sdev - > dev , " failed to read stream position: %d \n " , ret ) ;
return ;
}
dev_vdbg ( sdev - > dev , " posn : host 0x%llx dai 0x%llx wall 0x%llx \n " ,
posn . host_posn , posn . dai_posn , posn . wallclock ) ;
memcpy ( & stream - > posn , & posn , sizeof ( posn ) ) ;
if ( spcm - > pcm . compress )
snd_sof_compr_fragment_elapsed ( stream - > cstream ) ;
else if ( stream - > substream - > runtime & &
! stream - > substream - > runtime - > no_period_wakeup )
/* only inform ALSA for period_wakeup mode */
snd_sof_pcm_period_elapsed ( stream - > substream ) ;
}
/* DSP notifies host of an XRUN within FW */
static void ipc3_xrun ( struct snd_sof_dev * sdev , u32 msg_id )
{
struct snd_soc_component * scomp = sdev - > component ;
struct snd_sof_pcm_stream * stream ;
struct sof_ipc_stream_posn posn ;
struct snd_sof_pcm * spcm ;
int direction , ret ;
spcm = snd_sof_find_spcm_comp ( scomp , msg_id , & direction ) ;
if ( ! spcm ) {
dev_err ( sdev - > dev , " XRUN for unknown stream, msg_id %d \n " ,
msg_id ) ;
return ;
}
stream = & spcm - > stream [ direction ] ;
ret = snd_sof_ipc_msg_data ( sdev , stream - > substream , & posn , sizeof ( posn ) ) ;
if ( ret < 0 ) {
dev_warn ( sdev - > dev , " failed to read overrun position: %d \n " , ret ) ;
return ;
}
dev_dbg ( sdev - > dev , " posn XRUN: host %llx comp %d size %d \n " ,
posn . host_posn , posn . xrun_comp_id , posn . xrun_size ) ;
# if defined(CONFIG_SND_SOC_SOF_DEBUG_XRUN_STOP)
/* stop PCM on XRUN - used for pipeline debug */
memcpy ( & stream - > posn , & posn , sizeof ( posn ) ) ;
snd_pcm_stop_xrun ( stream - > substream ) ;
# endif
}
/* stream notifications from firmware */
static void ipc3_stream_message ( struct snd_sof_dev * sdev , void * msg_buf )
{
struct sof_ipc_cmd_hdr * hdr = msg_buf ;
u32 msg_type = hdr - > cmd & SOF_CMD_TYPE_MASK ;
u32 msg_id = SOF_IPC_MESSAGE_ID ( hdr - > cmd ) ;
switch ( msg_type ) {
case SOF_IPC_STREAM_POSITION :
ipc3_period_elapsed ( sdev , msg_id ) ;
break ;
case SOF_IPC_STREAM_TRIG_XRUN :
ipc3_xrun ( sdev , msg_id ) ;
break ;
default :
dev_err ( sdev - > dev , " unhandled stream message %#x \n " ,
msg_id ) ;
break ;
}
}
/* component notifications from firmware */
static void ipc3_comp_notification ( struct snd_sof_dev * sdev , void * msg_buf )
{
const struct sof_ipc_tplg_ops * tplg_ops = sdev - > ipc - > ops - > tplg ;
struct sof_ipc_cmd_hdr * hdr = msg_buf ;
u32 msg_type = hdr - > cmd & SOF_CMD_TYPE_MASK ;
switch ( msg_type ) {
case SOF_IPC_COMP_GET_VALUE :
case SOF_IPC_COMP_GET_DATA :
break ;
default :
dev_err ( sdev - > dev , " unhandled component message %#x \n " , msg_type ) ;
return ;
}
if ( tplg_ops - > control - > update )
tplg_ops - > control - > update ( sdev , msg_buf ) ;
}
static void ipc3_trace_message ( struct snd_sof_dev * sdev , void * msg_buf )
{
struct sof_ipc_cmd_hdr * hdr = msg_buf ;
u32 msg_type = hdr - > cmd & SOF_CMD_TYPE_MASK ;
switch ( msg_type ) {
case SOF_IPC_TRACE_DMA_POSITION :
snd_sof_trace_update_pos ( sdev , msg_buf ) ;
break ;
default :
dev_err ( sdev - > dev , " unhandled trace message %#x \n " , msg_type ) ;
break ;
}
}
/* DSP firmware has sent host a message */
static void sof_ipc3_rx_msg ( struct snd_sof_dev * sdev )
{
ipc3_rx_callback rx_callback = NULL ;
struct sof_ipc_cmd_hdr hdr ;
void * msg_buf ;
u32 cmd ;
int err ;
/* read back header */
err = snd_sof_ipc_msg_data ( sdev , NULL , & hdr , sizeof ( hdr ) ) ;
if ( err < 0 ) {
dev_warn ( sdev - > dev , " failed to read IPC header: %d \n " , err ) ;
return ;
}
if ( hdr . size < sizeof ( hdr ) ) {
dev_err ( sdev - > dev , " The received message size is invalid \n " ) ;
return ;
}
ipc3_log_header ( sdev - > dev , " ipc rx " , hdr . cmd ) ;
cmd = hdr . cmd & SOF_GLB_TYPE_MASK ;
/* check message type */
switch ( cmd ) {
case SOF_IPC_GLB_REPLY :
dev_err ( sdev - > dev , " ipc reply unknown \n " ) ;
break ;
case SOF_IPC_FW_READY :
/* check for FW boot completion */
if ( sdev - > fw_state = = SOF_FW_BOOT_IN_PROGRESS ) {
err = sof_ops ( sdev ) - > fw_ready ( sdev , cmd ) ;
if ( err < 0 )
sof_set_fw_state ( sdev , SOF_FW_BOOT_READY_FAILED ) ;
else
sof_set_fw_state ( sdev , SOF_FW_BOOT_READY_OK ) ;
/* wake up firmware loader */
wake_up ( & sdev - > boot_wait ) ;
}
break ;
case SOF_IPC_GLB_COMPOUND :
case SOF_IPC_GLB_TPLG_MSG :
case SOF_IPC_GLB_PM_MSG :
break ;
case SOF_IPC_GLB_COMP_MSG :
rx_callback = ipc3_comp_notification ;
break ;
case SOF_IPC_GLB_STREAM_MSG :
rx_callback = ipc3_stream_message ;
break ;
case SOF_IPC_GLB_TRACE_MSG :
rx_callback = ipc3_trace_message ;
break ;
default :
dev_err ( sdev - > dev , " %s: Unknown DSP message: 0x%x \n " , __func__ , cmd ) ;
break ;
}
/* read the full message */
msg_buf = kmalloc ( hdr . size , GFP_KERNEL ) ;
if ( ! msg_buf )
return ;
err = snd_sof_ipc_msg_data ( sdev , NULL , msg_buf , hdr . size ) ;
if ( err < 0 ) {
dev_err ( sdev - > dev , " %s: Failed to read message: %d \n " , __func__ , err ) ;
} else {
/* Call local handler for the message */
if ( rx_callback )
rx_callback ( sdev , msg_buf ) ;
/* Notify registered clients */
sof_client_ipc_rx_dispatcher ( sdev , msg_buf ) ;
}
kfree ( msg_buf ) ;
ipc3_log_header ( sdev - > dev , " ipc rx done " , hdr . cmd ) ;
}
2022-03-17 20:50:28 +03:00
static int sof_ipc3_ctx_ipc ( struct snd_sof_dev * sdev , int cmd )
{
struct sof_ipc_pm_ctx pm_ctx = {
. hdr . size = sizeof ( pm_ctx ) ,
. hdr . cmd = SOF_IPC_GLB_PM_MSG | cmd ,
} ;
struct sof_ipc_reply reply ;
/* send ctx save ipc to dsp */
2022-04-05 20:26:57 +03:00
return sof_ipc3_tx_msg ( sdev , & pm_ctx , sizeof ( pm_ctx ) ,
& reply , sizeof ( reply ) , false ) ;
2022-03-17 20:50:28 +03:00
}
static int sof_ipc3_ctx_save ( struct snd_sof_dev * sdev )
{
return sof_ipc3_ctx_ipc ( sdev , SOF_IPC_PM_CTX_SAVE ) ;
}
static int sof_ipc3_ctx_restore ( struct snd_sof_dev * sdev )
{
return sof_ipc3_ctx_ipc ( sdev , SOF_IPC_PM_CTX_RESTORE ) ;
}
static const struct sof_ipc_pm_ops ipc3_pm_ops = {
. ctx_save = sof_ipc3_ctx_save ,
. ctx_restore = sof_ipc3_ctx_restore ,
} ;
const struct sof_ipc_ops ipc3_ops = {
. tplg = & ipc3_tplg_ops ,
. pm = & ipc3_pm_ops ,
2022-03-17 20:50:38 +03:00
. pcm = & ipc3_pcm_ops ,
2022-04-05 20:26:56 +03:00
. tx_msg = sof_ipc3_tx_msg ,
2022-04-05 20:27:00 +03:00
. rx_msg = sof_ipc3_rx_msg ,
2022-04-05 20:26:58 +03:00
. set_get_data = sof_ipc3_set_get_data ,
2022-04-05 20:26:59 +03:00
. get_reply = sof_ipc3_get_reply ,
2022-03-17 20:50:28 +03:00
} ;