iommu/mediatek: Add enable IOMMU SMC command for INFRA masters
Prepare for MT8188. In MT8188, the register which enables IOMMU for INFRA masters are in the secure world for security concerns, therefore we add a SMC command for INFRA masters to enable IOMMU in ATF. Signed-off-by: Chengci.Xu <chengci.xu@mediatek.com> Signed-off-by: Yong Wu <yong.wu@mediatek.com> Reviewed-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com> Reviewed-by: Alexandre Mergnat <amergnat@baylibre.com> Link: https://lore.kernel.org/r/20230602090227.7264-5-yong.wu@mediatek.com Signed-off-by: Joerg Roedel <jroedel@suse.de>
This commit is contained in:
parent
9a89051084
commit
946e719ce6
@ -3,6 +3,7 @@
|
|||||||
* Copyright (c) 2015-2016 MediaTek Inc.
|
* Copyright (c) 2015-2016 MediaTek Inc.
|
||||||
* Author: Yong Wu <yong.wu@mediatek.com>
|
* Author: Yong Wu <yong.wu@mediatek.com>
|
||||||
*/
|
*/
|
||||||
|
#include <linux/arm-smccc.h>
|
||||||
#include <linux/bitfield.h>
|
#include <linux/bitfield.h>
|
||||||
#include <linux/bug.h>
|
#include <linux/bug.h>
|
||||||
#include <linux/clk.h>
|
#include <linux/clk.h>
|
||||||
@ -27,6 +28,7 @@
|
|||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
#include <linux/soc/mediatek/infracfg.h>
|
#include <linux/soc/mediatek/infracfg.h>
|
||||||
|
#include <linux/soc/mediatek/mtk_sip_svc.h>
|
||||||
#include <asm/barrier.h>
|
#include <asm/barrier.h>
|
||||||
#include <soc/mediatek/smi.h>
|
#include <soc/mediatek/smi.h>
|
||||||
|
|
||||||
@ -143,6 +145,7 @@
|
|||||||
#define PGTABLE_PA_35_EN BIT(17)
|
#define PGTABLE_PA_35_EN BIT(17)
|
||||||
#define TF_PORT_TO_ADDR_MT8173 BIT(18)
|
#define TF_PORT_TO_ADDR_MT8173 BIT(18)
|
||||||
#define INT_ID_PORT_WIDTH_6 BIT(19)
|
#define INT_ID_PORT_WIDTH_6 BIT(19)
|
||||||
|
#define CFG_IFA_MASTER_IN_ATF BIT(20)
|
||||||
|
|
||||||
#define MTK_IOMMU_HAS_FLAG_MASK(pdata, _x, mask) \
|
#define MTK_IOMMU_HAS_FLAG_MASK(pdata, _x, mask) \
|
||||||
((((pdata)->flags) & (mask)) == (_x))
|
((((pdata)->flags) & (mask)) == (_x))
|
||||||
@ -580,6 +583,7 @@ static int mtk_iommu_config(struct mtk_iommu_data *data, struct device *dev,
|
|||||||
struct iommu_fwspec *fwspec = dev_iommu_fwspec_get(dev);
|
struct iommu_fwspec *fwspec = dev_iommu_fwspec_get(dev);
|
||||||
const struct mtk_iommu_iova_region *region;
|
const struct mtk_iommu_iova_region *region;
|
||||||
unsigned long portid_msk = 0;
|
unsigned long portid_msk = 0;
|
||||||
|
struct arm_smccc_res res;
|
||||||
int i, ret = 0;
|
int i, ret = 0;
|
||||||
|
|
||||||
for (i = 0; i < fwspec->num_ids; ++i) {
|
for (i = 0; i < fwspec->num_ids; ++i) {
|
||||||
@ -605,17 +609,24 @@ static int mtk_iommu_config(struct mtk_iommu_data *data, struct device *dev,
|
|||||||
else
|
else
|
||||||
larb_mmu->mmu &= ~portid_msk;
|
larb_mmu->mmu &= ~portid_msk;
|
||||||
} else if (MTK_IOMMU_IS_TYPE(data->plat_data, MTK_IOMMU_TYPE_INFRA)) {
|
} else if (MTK_IOMMU_IS_TYPE(data->plat_data, MTK_IOMMU_TYPE_INFRA)) {
|
||||||
/* PCI dev has only one output id, enable the next writing bit for PCIe */
|
if (MTK_IOMMU_HAS_FLAG(data->plat_data, CFG_IFA_MASTER_IN_ATF)) {
|
||||||
if (dev_is_pci(dev)) {
|
arm_smccc_smc(MTK_SIP_KERNEL_IOMMU_CONTROL,
|
||||||
if (fwspec->num_ids != 1) {
|
IOMMU_ATF_CMD_CONFIG_INFRA_IOMMU,
|
||||||
dev_err(dev, "PCI dev can only have one port.\n");
|
portid_msk, enable, 0, 0, 0, 0, &res);
|
||||||
return -ENODEV;
|
ret = res.a0;
|
||||||
|
} else {
|
||||||
|
/* PCI dev has only one output id, enable the next writing bit for PCIe */
|
||||||
|
if (dev_is_pci(dev)) {
|
||||||
|
if (fwspec->num_ids != 1) {
|
||||||
|
dev_err(dev, "PCI dev can only have one port.\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
portid_msk |= BIT(portid + 1);
|
||||||
}
|
}
|
||||||
portid_msk |= BIT(portid + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = regmap_update_bits(data->pericfg, PERICFG_IOMMU_1,
|
ret = regmap_update_bits(data->pericfg, PERICFG_IOMMU_1,
|
||||||
(u32)portid_msk, enable ? (u32)portid_msk : 0);
|
(u32)portid_msk, enable ? (u32)portid_msk : 0);
|
||||||
|
}
|
||||||
if (ret)
|
if (ret)
|
||||||
dev_err(dev, "%s iommu(%s) inframaster 0x%lx fail(%d).\n",
|
dev_err(dev, "%s iommu(%s) inframaster 0x%lx fail(%d).\n",
|
||||||
enable ? "enable" : "disable",
|
enable ? "enable" : "disable",
|
||||||
@ -1330,7 +1341,8 @@ static int mtk_iommu_probe(struct platform_device *pdev)
|
|||||||
dev_err_probe(dev, ret, "mm dts parse fail\n");
|
dev_err_probe(dev, ret, "mm dts parse fail\n");
|
||||||
goto out_runtime_disable;
|
goto out_runtime_disable;
|
||||||
}
|
}
|
||||||
} else if (MTK_IOMMU_IS_TYPE(data->plat_data, MTK_IOMMU_TYPE_INFRA)) {
|
} else if (MTK_IOMMU_IS_TYPE(data->plat_data, MTK_IOMMU_TYPE_INFRA) &&
|
||||||
|
!MTK_IOMMU_HAS_FLAG(data->plat_data, CFG_IFA_MASTER_IN_ATF)) {
|
||||||
p = data->plat_data->pericfg_comp_str;
|
p = data->plat_data->pericfg_comp_str;
|
||||||
data->pericfg = syscon_regmap_lookup_by_compatible(p);
|
data->pericfg = syscon_regmap_lookup_by_compatible(p);
|
||||||
if (IS_ERR(data->pericfg)) {
|
if (IS_ERR(data->pericfg)) {
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
|
|
||||||
enum iommu_atf_cmd {
|
enum iommu_atf_cmd {
|
||||||
IOMMU_ATF_CMD_CONFIG_SMI_LARB, /* For mm master to en/disable iommu */
|
IOMMU_ATF_CMD_CONFIG_SMI_LARB, /* For mm master to en/disable iommu */
|
||||||
|
IOMMU_ATF_CMD_CONFIG_INFRA_IOMMU, /* For infra master to enable iommu */
|
||||||
IOMMU_ATF_CMD_MAX,
|
IOMMU_ATF_CMD_MAX,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user