mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

The SOF drivers for imx chips have a lot of duplicate code and routines/code snippets that could certainly be reused among drivers. As such, introduce a new set of structures and functions that will help eliminate the redundancy and code size of the drivers. Signed-off-by: Laurentiu Mihalcea <laurentiu.mihalcea@nxp.com> Reviewed-by: Frank Li <Frank.Li@nxp.com> Link: https://patch.msgid.link/20250207162246.3104-2-laurentiumihalcea111@gmail.com Signed-off-by: Mark Brown <broonie@kernel.org>
169 lines
4.5 KiB
C
169 lines
4.5 KiB
C
/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause) */
|
|
|
|
#ifndef __IMX_COMMON_H__
|
|
#define __IMX_COMMON_H__
|
|
|
|
#include <linux/clk.h>
|
|
#include <linux/of_platform.h>
|
|
#include <sound/sof/xtensa.h>
|
|
|
|
#include "../sof-of-dev.h"
|
|
#include "../ops.h"
|
|
|
|
#define EXCEPT_MAX_HDR_SIZE 0x400
|
|
#define IMX8_STACK_DUMP_SIZE 32
|
|
|
|
/* chip_info refers to the data stored in struct sof_dev_desc's chip_info */
|
|
#define get_chip_info(sdev)\
|
|
((const struct imx_chip_info *)((sdev)->pdata->desc->chip_info))
|
|
|
|
/* chip_pdata refers to the data stored in struct imx_common_data's chip_pdata */
|
|
#define get_chip_pdata(sdev)\
|
|
(((struct imx_common_data *)((sdev)->pdata->hw_pdata))->chip_pdata)
|
|
|
|
/* can be used if:
|
|
* 1) The only supported IPC version is IPC3.
|
|
* 2) The default paths/FW name match values below.
|
|
*
|
|
* otherwise, just explicitly declare the structure
|
|
*/
|
|
#define IMX_SOF_DEV_DESC(mach_name, of_machs, \
|
|
mach_chip_info, mach_ops, mach_ops_init) \
|
|
static struct sof_dev_desc sof_of_##mach_name##_desc = { \
|
|
.of_machines = of_machs, \
|
|
.chip_info = mach_chip_info, \
|
|
.ipc_supported_mask = BIT(SOF_IPC_TYPE_3), \
|
|
.ipc_default = SOF_IPC_TYPE_3, \
|
|
.default_fw_path = { \
|
|
[SOF_IPC_TYPE_3] = "imx/sof", \
|
|
}, \
|
|
.default_tplg_path = { \
|
|
[SOF_IPC_TYPE_3] = "imx/sof-tplg", \
|
|
}, \
|
|
.default_fw_filename = { \
|
|
[SOF_IPC_TYPE_3] = "sof-" #mach_name ".ri", \
|
|
}, \
|
|
.ops = mach_ops, \
|
|
.ops_init = mach_ops_init, \
|
|
}
|
|
|
|
/* to be used alongside IMX_SOF_DEV_DESC() */
|
|
#define IMX_SOF_DEV_DESC_NAME(mach_name) sof_of_##mach_name##_desc
|
|
|
|
/* dai driver entry w/ playback and capture caps. If one direction is missing
|
|
* then set the channels to 0.
|
|
*/
|
|
#define IMX_SOF_DAI_DRV_ENTRY(dai_name, pb_cmin, pb_cmax, cap_cmin, cap_cmax) \
|
|
{ \
|
|
.name = dai_name, \
|
|
.playback = { \
|
|
.channels_min = pb_cmin, \
|
|
.channels_max = pb_cmax, \
|
|
}, \
|
|
.capture = { \
|
|
.channels_min = cap_cmin, \
|
|
.channels_max = cap_cmax, \
|
|
}, \
|
|
}
|
|
|
|
/* use if playback and capture have the same min/max channel count */
|
|
#define IMX_SOF_DAI_DRV_ENTRY_BIDIR(dai_name, cmin, cmax)\
|
|
IMX_SOF_DAI_DRV_ENTRY(dai_name, cmin, cmax, cmin, cmax)
|
|
|
|
struct imx_ipc_info {
|
|
/* true if core is able to write a panic code to the debug box */
|
|
bool has_panic_code;
|
|
/* offset to mailbox in which firmware initially writes FW_READY */
|
|
int boot_mbox_offset;
|
|
/* offset to region at which the mailboxes start */
|
|
int window_offset;
|
|
};
|
|
|
|
struct imx_chip_ops {
|
|
/* called after clocks and PDs are enabled */
|
|
int (*probe)(struct snd_sof_dev *sdev);
|
|
/* used directly by the SOF core */
|
|
int (*core_kick)(struct snd_sof_dev *sdev);
|
|
/* called during suspend()/remove() before clocks are disabled */
|
|
int (*core_shutdown)(struct snd_sof_dev *sdev);
|
|
/* used directly by the SOF core */
|
|
int (*core_reset)(struct snd_sof_dev *sdev);
|
|
};
|
|
|
|
struct imx_memory_info {
|
|
const char *name;
|
|
bool reserved;
|
|
};
|
|
|
|
struct imx_chip_info {
|
|
struct imx_ipc_info ipc_info;
|
|
/* does the chip have a reserved memory region for DMA? */
|
|
bool has_dma_reserved;
|
|
struct imx_memory_info *memory;
|
|
struct snd_soc_dai_driver *drv;
|
|
int num_drv;
|
|
/* optional */
|
|
const struct imx_chip_ops *ops;
|
|
};
|
|
|
|
struct imx_common_data {
|
|
struct platform_device *ipc_dev;
|
|
struct imx_dsp_ipc *ipc_handle;
|
|
/* core may have no clocks */
|
|
struct clk_bulk_data *clks;
|
|
int clk_num;
|
|
/* core may have no PDs */
|
|
struct dev_pm_domain_list *pd_list;
|
|
void *chip_pdata;
|
|
};
|
|
|
|
static inline int imx_chip_core_kick(struct snd_sof_dev *sdev)
|
|
{
|
|
const struct imx_chip_ops *ops = get_chip_info(sdev)->ops;
|
|
|
|
if (ops && ops->core_kick)
|
|
return ops->core_kick(sdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline int imx_chip_core_shutdown(struct snd_sof_dev *sdev)
|
|
{
|
|
const struct imx_chip_ops *ops = get_chip_info(sdev)->ops;
|
|
|
|
if (ops && ops->core_shutdown)
|
|
return ops->core_shutdown(sdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline int imx_chip_core_reset(struct snd_sof_dev *sdev)
|
|
{
|
|
const struct imx_chip_ops *ops = get_chip_info(sdev)->ops;
|
|
|
|
if (ops && ops->core_reset)
|
|
return ops->core_reset(sdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline int imx_chip_probe(struct snd_sof_dev *sdev)
|
|
{
|
|
const struct imx_chip_ops *ops = get_chip_info(sdev)->ops;
|
|
|
|
if (ops && ops->probe)
|
|
return ops->probe(sdev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void imx8_get_registers(struct snd_sof_dev *sdev,
|
|
struct sof_ipc_dsp_oops_xtensa *xoops,
|
|
struct sof_ipc_panic_info *panic_info,
|
|
u32 *stack, size_t stack_words);
|
|
|
|
void imx8_dump(struct snd_sof_dev *sdev, u32 flags);
|
|
|
|
extern const struct snd_sof_dsp_ops sof_imx_ops;
|
|
|
|
#endif
|