1
0
mirror of https://github.com/physwizz/a155-U-u1.git synced 2025-08-04 15:30:24 +00:00
Files
physwizz 99537be4e2 first
2024-03-11 06:53:12 +11:00

329 lines
7.9 KiB
C

// SPDX-License-Identifier: GPL-2.0
//
// Copyright (C) 2018 MediaTek Inc.
#include "mtk-scp-ultra-common.h"
#include "mtk-base-scp-ultra.h"
#include <sound/soc.h>
#include <linux/device.h>
#include <linux/compat.h>
//#include "audio_ultra_msg_id.h"
#include "ultra_ipi.h"
#include "mtk-base-afe.h"
#include "mtk-scp-ultra.h"
#include "mtk-afe-external.h"
#define AFE_AGENT_SET_OFFSET 4
#define AFE_AGENT_CLR_OFFSET 8
/* don't use this directly if not necessary */
static struct mtk_base_afe *local_scp_ultra_afe;
static void *ipi_recv_private;
int ultra_set_dsp_afe(struct mtk_base_afe *afe)
{
if (!afe) {
pr_err("%s(), afe is NULL", __func__);
return -1;
}
local_scp_ultra_afe = afe;
return 0;
}
EXPORT_SYMBOL_GPL(ultra_set_dsp_afe);
struct mtk_base_afe *get_afe_base(void)
{
if (!local_scp_ultra_afe)
pr_err("%s(), local_scp_ultra_afe is NULL", __func__);
return local_scp_ultra_afe;
}
void set_ipi_recv_private(void *priv)
{
pr_debug("%s\n", __func__);
if (priv != NULL)
ipi_recv_private = priv;
else
pr_debug("%s ipi_recv_private has been set\n", __func__);
}
void *ultra_get_ipi_recv_private(void)
{
if (!ipi_recv_private)
pr_info("%s(), ipi_recv_private is NULL", __func__);
return ipi_recv_private;
}
void ultra_set_ipi_recv_private(void *priv)
{
pr_debug("%s()\n", __func__);
if (!ipi_recv_private)
ipi_recv_private = priv;
else
pr_info("%s() has been set\n", __func__);
}
void set_afe_dl_irq_target(int scp_enable)
{
struct mtk_base_afe *afe = get_afe_base();
struct mtk_base_scp_ultra *scp_ultra = get_scp_ultra_base();
struct mtk_base_afe_memif *memif =
&afe->memif[scp_ultra->ultra_mem.ultra_dl_memif_id];
struct mtk_base_afe_irq *irqs = &afe->irqs[memif->irq_usage];
const struct mtk_base_irq_data *irq_data = irqs->irq_data;
if (scp_enable) {
regmap_update_bits(afe->regmap,
irq_data->irq_ap_en_reg,
0x1 << irq_data->irq_ap_en_shift,
0x0);
regmap_update_bits(afe->regmap,
irq_data->irq_scp_en_reg,
0x1 << irq_data->irq_scp_en_shift,
0x1 << irq_data->irq_scp_en_shift);
} else {
regmap_update_bits(afe->regmap,
irq_data->irq_scp_en_reg,
0x1 << irq_data->irq_scp_en_shift,
0);
regmap_update_bits(afe->regmap,
irq_data->irq_ap_en_reg,
0x1 << irq_data->irq_ap_en_shift,
0x1 << irq_data->irq_ap_en_shift);
}
pr_debug("%s(), scp_en=%d,memif=%d,ap_en_reg:0x%x,scp_en_reg:0x%x\n",
__func__,
scp_enable,
scp_ultra->ultra_mem.ultra_dl_memif_id,
irq_data->irq_ap_en_reg,
irq_data->irq_scp_en_reg);
}
void set_afe_ul_irq_target(int scp_enable)
{
struct mtk_base_afe *afe = get_afe_base();
struct mtk_base_scp_ultra *scp_ultra = get_scp_ultra_base();
struct mtk_base_afe_memif *memif =
&afe->memif[scp_ultra->ultra_mem.ultra_ul_memif_id];
struct mtk_base_afe_irq *irqs = &afe->irqs[memif->irq_usage];
const struct mtk_base_irq_data *irq_data = irqs->irq_data;
if (scp_enable) {
regmap_update_bits(afe->regmap,
irq_data->irq_ap_en_reg,
0x1 << irq_data->irq_ap_en_shift,
0x0);
regmap_update_bits(afe->regmap,
irq_data->irq_scp_en_reg,
0x1 << irq_data->irq_scp_en_shift,
0x1 << irq_data->irq_scp_en_shift);
} else {
regmap_update_bits(afe->regmap,
irq_data->irq_scp_en_reg,
0x1 << irq_data->irq_scp_en_shift,
0);
regmap_update_bits(afe->regmap,
irq_data->irq_ap_en_reg,
0x1 << irq_data->irq_ap_en_shift,
0x1 << irq_data->irq_ap_en_shift);
}
pr_debug("%s(),scp_en=%d,memif=%d,ap_en_reg:0x%x,scp_en_reg:0x%x\n",
__func__,
scp_enable,
scp_ultra->ultra_mem.ultra_ul_memif_id,
irq_data->irq_ap_en_reg,
irq_data->irq_scp_en_reg);
}
int ultra_memif_set_enable(struct mtk_base_afe *afe, int afe_id)
{
struct mtk_base_afe_memif *memif = &afe->memif[afe_id];
int reg = 0;
if (memif->data->enable_shift < 0) {
pr_info("%s(), error, id %d, enable_shift < 0\n",
__func__, afe_id);
return 0;
}
if (memif->data->enable_reg < 0) {
pr_info("%s(), error, id %d, enable_reg < 0\n",
__func__, afe_id);
return 0;
}
if (afe->is_memif_bit_banding)
reg = memif->data->enable_reg + AFE_AGENT_SET_OFFSET;
else
reg = memif->data->enable_reg;
return regmap_update_bits(afe->regmap,
reg,
1 << memif->data->enable_shift,
1 << memif->data->enable_shift);
}
int ultra_memif_set_disable(struct mtk_base_afe *afe, int afe_id)
{
struct mtk_base_afe_memif *memif = &afe->memif[afe_id];
int reg = 0;
int val = 0;
if (memif->data->enable_shift < 0) {
pr_info("%s(), error, id %d, enable_shift < 0\n",
__func__, afe_id);
return 0;
}
if (memif->data->enable_reg < 0) {
pr_info("%s(), error, id %d, enable_reg < 0\n",
__func__, afe_id);
return 0;
}
if (afe->is_memif_bit_banding) {
reg = memif->data->enable_reg + AFE_AGENT_CLR_OFFSET;
val = 1;
} else {
reg = memif->data->enable_reg;
val = 0;
}
return regmap_update_bits(afe->regmap,
reg,
1 << memif->data->enable_shift,
val << memif->data->enable_shift);
}
int ultra_memif_set_enable_hw_sema(struct mtk_base_afe *afe, int afe_id)
{
int ret = 0;
int scp_sem_ret = NOTIFY_STOP;
if (!afe)
return -EPERM;
if (!afe->is_scp_sema_support)
return ultra_memif_set_enable(afe, afe_id);
if (afe->is_memif_bit_banding == 0) {
scp_sem_ret = notify_3way_semaphore_control(
NOTIFIER_SCP_3WAY_SEMAPHORE_GET, NULL);
if (scp_sem_ret != NOTIFY_STOP) {
pr_info("%s error, adsp_sem_ret[%d]\n", __func__, ret);
return -EBUSY;
}
}
ret = ultra_memif_set_enable(afe, afe_id);
if (afe->is_memif_bit_banding == 0) {
notify_3way_semaphore_control(
NOTIFIER_SCP_3WAY_SEMAPHORE_RELEASE, NULL);
}
return ret;
}
int ultra_memif_set_disable_hw_sema(struct mtk_base_afe *afe, int afe_id)
{
int ret = 0;
int scp_sem_ret = NOTIFY_STOP;
if (!afe)
return -EPERM;
if (!afe->is_scp_sema_support)
return ultra_memif_set_disable(afe, afe_id);
if (afe->is_memif_bit_banding == 0) {
scp_sem_ret = notify_3way_semaphore_control(
NOTIFIER_SCP_3WAY_SEMAPHORE_GET, NULL);
if (scp_sem_ret != NOTIFY_STOP) {
pr_info("%s error, adsp_sem_ret[%d]\n", __func__, ret);
return -EBUSY;
}
}
ret = ultra_memif_set_disable(afe, afe_id);
if (afe->is_memif_bit_banding == 0) {
notify_3way_semaphore_control(
NOTIFIER_SCP_3WAY_SEMAPHORE_RELEASE, NULL);
}
return ret;
}
int ultra_irq_set_enable_hw_sema(struct mtk_base_afe *afe,
const struct mtk_base_irq_data *irq_data,
int afe_id)
{
int ret = 0;
int scp_sem_ret = NOTIFY_STOP;
if (!afe)
return -ENODEV;
if (!irq_data)
return -ENODEV;
if (!afe->is_scp_sema_support)
return regmap_update_bits(afe->regmap, irq_data->irq_en_reg,
1 << irq_data->irq_en_shift,
1 << irq_data->irq_en_shift);
scp_sem_ret =
notify_3way_semaphore_control(NOTIFIER_SCP_3WAY_SEMAPHORE_GET,
NULL);
if (scp_sem_ret != NOTIFY_STOP) {
pr_info("%s error, scp_sem_ret[%d]\n", __func__, scp_sem_ret);
return -EBUSY;
}
regmap_update_bits(afe->regmap, irq_data->irq_en_reg,
1 << irq_data->irq_en_shift,
1 << irq_data->irq_en_shift);
notify_3way_semaphore_control(NOTIFIER_SCP_3WAY_SEMAPHORE_RELEASE,
NULL);
return ret;
}
int ultra_irq_set_disable_hw_sema(struct mtk_base_afe *afe,
const struct mtk_base_irq_data *irq_data,
int afe_id)
{
int ret = 0;
int scp_sem_ret = NOTIFY_STOP;
if (!afe)
return -ENODEV;
if (!irq_data)
return -ENODEV;
if (!afe->is_scp_sema_support)
return regmap_update_bits(afe->regmap, irq_data->irq_en_reg,
1 << irq_data->irq_en_shift,
0 << irq_data->irq_en_shift);
scp_sem_ret =
notify_3way_semaphore_control(NOTIFIER_SCP_3WAY_SEMAPHORE_GET,
NULL);
if (scp_sem_ret != NOTIFY_STOP) {
pr_info("%s error, scp_sem_ret[%d]\n", __func__, scp_sem_ret);
return -EBUSY;
}
regmap_update_bits(afe->regmap, irq_data->irq_en_reg,
1 << irq_data->irq_en_shift,
0 << irq_data->irq_en_shift);
notify_3way_semaphore_control(NOTIFIER_SCP_3WAY_SEMAPHORE_RELEASE,
NULL);
return ret;
}