[PATCH] firmware: xilinx: add support for new SMC call format
Michal Simek
michal.simek at amd.com
Wed Oct 2 03:55:24 PDT 2024
On 9/20/24 07:55, Ronak Jain wrote:
> Added zynqmp_pm_invoke_fw_fn() to use new SMC format in which
> lower 12 bits of SMC id are fixed and firmware header is moved to
> subsequent SMC arguments. The new SMC format supports full request and
> response buffers.
>
> Added zynqmp_pm_get_sip_svc_version() to get SiP SVC version
> number to check if TF-A is newer or older and use the SMC format
> accordingly to handle backward compatibility.
>
> Used new SMC format for PM_QUERY_DATA API as more response values are
> required in it.
>
> Signed-off-by: Ronak Jain <ronak.jain at amd.com>
> Signed-off-by: Jay Buddhabhatti <jay.buddhabhatti at amd.com>
> ---
> drivers/firmware/xilinx/zynqmp.c | 137 ++++++++++++++++++++++++++-
> include/linux/firmware/xlnx-zynqmp.h | 26 ++++-
> 2 files changed, 157 insertions(+), 6 deletions(-)
>
> diff --git a/drivers/firmware/xilinx/zynqmp.c b/drivers/firmware/xilinx/zynqmp.c
> index c8be32d9c6af..5ab0dfb2b225 100644
> --- a/drivers/firmware/xilinx/zynqmp.c
> +++ b/drivers/firmware/xilinx/zynqmp.c
> @@ -3,7 +3,7 @@
> * Xilinx Zynq MPSoC Firmware layer
> *
> * Copyright (C) 2014-2022 Xilinx, Inc.
> - * Copyright (C) 2022 - 2023, Advanced Micro Devices, Inc.
> + * Copyright (C) 2022 - 2024, Advanced Micro Devices, Inc.
> *
> * Michal Simek <michal.simek at amd.com>
> * Davorin Mista <davorin.mista at aggios.com>
> @@ -46,6 +46,7 @@ static DEFINE_HASHTABLE(pm_api_features_map, PM_API_FEATURE_CHECK_MAX_ORDER);
> static u32 ioctl_features[FEATURE_PAYLOAD_SIZE];
> static u32 query_features[FEATURE_PAYLOAD_SIZE];
>
> +static u32 sip_svc_version;
> static struct platform_device *em_dev;
>
> /**
> @@ -151,6 +152,9 @@ static noinline int do_fw_call_smc(u32 *ret_payload, u32 num_args, ...)
> ret_payload[1] = upper_32_bits(res.a0);
> ret_payload[2] = lower_32_bits(res.a1);
> ret_payload[3] = upper_32_bits(res.a1);
> + ret_payload[4] = lower_32_bits(res.a2);
> + ret_payload[5] = upper_32_bits(res.a2);
> + ret_payload[6] = lower_32_bits(res.a3);
> }
>
> return zynqmp_pm_ret_code((enum pm_ret_status)res.a0);
> @@ -191,6 +195,9 @@ static noinline int do_fw_call_hvc(u32 *ret_payload, u32 num_args, ...)
> ret_payload[1] = upper_32_bits(res.a0);
> ret_payload[2] = lower_32_bits(res.a1);
> ret_payload[3] = upper_32_bits(res.a1);
> + ret_payload[4] = lower_32_bits(res.a2);
> + ret_payload[5] = upper_32_bits(res.a2);
> + ret_payload[6] = lower_32_bits(res.a3);
> }
>
> return zynqmp_pm_ret_code((enum pm_ret_status)res.a0);
> @@ -331,6 +338,70 @@ int zynqmp_pm_is_function_supported(const u32 api_id, const u32 id)
> }
> EXPORT_SYMBOL_GPL(zynqmp_pm_is_function_supported);
>
> +/**
> + * zynqmp_pm_invoke_fw_fn() - Invoke the system-level platform management layer
> + * caller function depending on the configuration
> + * @pm_api_id: Requested PM-API call
> + * @ret_payload: Returned value array
> + * @num_args: Number of arguments to requested PM-API call
> + *
> + * Invoke platform management function for SMC or HVC call, depending on
> + * configuration.
> + * Following SMC Calling Convention (SMCCC) for SMC64:
> + * Pm Function Identifier,
> + * PM_SIP_SVC + PASS_THROUGH_FW_CMD_ID =
> + * ((SMC_TYPE_FAST << FUNCID_TYPE_SHIFT)
> + * ((SMC_64) << FUNCID_CC_SHIFT)
> + * ((SIP_START) << FUNCID_OEN_SHIFT)
> + * (PASS_THROUGH_FW_CMD_ID))
> + *
> + * PM_SIP_SVC - Registered ZynqMP SIP Service Call.
> + * PASS_THROUGH_FW_CMD_ID - Fixed SiP SVC call ID for FW specific calls.
> + *
> + * Return: Returns status, either success or error+reason
> + */
> +int zynqmp_pm_invoke_fw_fn(u32 pm_api_id, u32 *ret_payload, u32 num_args, ...)
> +{
> + /*
> + * Added SIP service call Function Identifier
> + * Make sure to stay in x0 register
> + */
> + u64 smc_arg[SMC_ARG_CNT_64];
> + int ret, i;
> + va_list arg_list;
> + u32 args[SMC_ARG_CNT_32] = {0};
> + u32 module_id;
> +
> + if (num_args > SMC_ARG_CNT_32)
> + return -EINVAL;
> +
> + va_start(arg_list, num_args);
> +
> + /* Check if feature is supported or not */
> + ret = zynqmp_pm_feature(pm_api_id);
> + if (ret < 0)
> + return ret;
> +
> + for (i = 0; i < num_args; i++)
> + args[i] = va_arg(arg_list, u32);
> +
> + va_end(arg_list);
> +
> + module_id = FIELD_GET(PLM_MODULE_ID_MASK, pm_api_id);
> +
> + if (module_id == 0)
> + module_id = XPM_MODULE_ID;
> +
> + smc_arg[0] = PM_SIP_SVC | PASS_THROUGH_FW_CMD_ID;
> + smc_arg[1] = ((u64)args[0] << 32U) | FIELD_PREP(PLM_MODULE_ID_MASK, module_id) |
> + (pm_api_id & API_ID_MASK);
> + for (i = 1; i < (SMC_ARG_CNT_64 - 1); i++)
> + smc_arg[i + 1] = ((u64)args[(i * 2)] << 32U) | args[(i * 2) - 1];
> +
> + return do_fw_call(ret_payload, 8, smc_arg[0], smc_arg[1], smc_arg[2], smc_arg[3],
> + smc_arg[4], smc_arg[5], smc_arg[6], smc_arg[7]);
> +}
> +
> /**
> * zynqmp_pm_invoke_fn() - Invoke the system-level platform management layer
> * caller function depending on the configuration
> @@ -488,6 +559,35 @@ int zynqmp_pm_get_family_info(u32 *family, u32 *subfamily)
> }
> EXPORT_SYMBOL_GPL(zynqmp_pm_get_family_info);
>
> +/**
> + * zynqmp_pm_get_sip_svc_version() - Get SiP service call version
> + * @version: Returned version value
> + *
> + * Return: Returns status, either success or error+reason
> + */
> +static int zynqmp_pm_get_sip_svc_version(u32 *version)
> +{
> + struct arm_smccc_res res;
> + u64 args[SMC_ARG_CNT_64] = {0};
> +
> + if (!version)
> + return -EINVAL;
> +
> + /* Check if SiP SVC version already verified */
> + if (sip_svc_version > 0) {
> + *version = sip_svc_version;
> + return 0;
> + }
> +
> + args[0] = GET_SIP_SVC_VERSION;
> +
> + arm_smccc_smc(args[0], args[1], args[2], args[3], args[4], args[5], args[6], args[7], &res);
> +
> + *version = ((lower_32_bits(res.a0) << 16U) | lower_32_bits(res.a1));
> +
> + return zynqmp_pm_ret_code(XST_PM_SUCCESS);
> +}
> +
> /**
> * zynqmp_pm_get_trustzone_version() - Get secure trustzone firmware version
> * @version: Returned version value
> @@ -552,10 +652,34 @@ static int get_set_conduit_method(struct device_node *np)
> */
> int zynqmp_pm_query_data(struct zynqmp_pm_query_data qdata, u32 *out)
> {
> - int ret;
> + int ret, i = 0;
> + u32 ret_payload[PAYLOAD_ARG_CNT] = {0};
> +
> + if (sip_svc_version >= SIP_SVC_PASSTHROUGH_VERSION) {
> + ret = zynqmp_pm_invoke_fw_fn(PM_QUERY_DATA, ret_payload, 4,
> + qdata.qid, qdata.arg1,
> + qdata.arg2, qdata.arg3);
> + /* To support backward compatibility */
> + if (!ret && !ret_payload[0]) {
> + /*
> + * TF-A passes return status on 0th index but
> + * api to get clock name reads data from 0th
> + * index so pass data at 0th index instead of
> + * return status
> + */
> + if (qdata.qid == PM_QID_CLOCK_GET_NAME ||
> + qdata.qid == PM_QID_PINCTRL_GET_FUNCTION_NAME)
> + i = 1;
> +
> + for (; i < PAYLOAD_ARG_CNT; i++, out++)
> + *out = ret_payload[i];
>
> - ret = zynqmp_pm_invoke_fn(PM_QUERY_DATA, out, 4, qdata.qid, qdata.arg1, qdata.arg2,
> - qdata.arg3);
> + return ret;
> + }
> + }
> +
> + ret = zynqmp_pm_invoke_fn(PM_QUERY_DATA, out, 4, qdata.qid,
> + qdata.arg1, qdata.arg2, qdata.arg3);
>
> /*
> * For clock name query, all bytes in SMC response are clock name
> @@ -1887,6 +2011,11 @@ static int zynqmp_firmware_probe(struct platform_device *pdev)
> if (ret)
> return ret;
>
> + /* Get SiP SVC version number */
> + ret = zynqmp_pm_get_sip_svc_version(&sip_svc_version);
> + if (ret)
> + return ret;
> +
> ret = do_feature_check_call(PM_FEATURE_CHECK);
> if (ret >= 0 && ((ret & FIRMWARE_VERSION_MASK) >= PM_API_VERSION_1))
> feature_check_enabled = true;
> diff --git a/include/linux/firmware/xlnx-zynqmp.h b/include/linux/firmware/xlnx-zynqmp.h
> index 5b938fc2adad..76d85ad82ec0 100644
> --- a/include/linux/firmware/xlnx-zynqmp.h
> +++ b/include/linux/firmware/xlnx-zynqmp.h
> @@ -3,7 +3,7 @@
> * Xilinx Zynq MPSoC Firmware layer
> *
> * Copyright (C) 2014-2021 Xilinx
> - * Copyright (C) 2022 - 2023, Advanced Micro Devices, Inc.
> + * Copyright (C) 2022 - 2024, Advanced Micro Devices, Inc.
> *
> * Michal Simek <michal.simek at amd.com>
> * Davorin Mista <davorin.mista at aggios.com>
> @@ -32,6 +32,19 @@
> /* SMC SIP service Call Function Identifier Prefix */
> #define PM_SIP_SVC 0xC2000000
>
> +/* SMC function ID to get SiP SVC version */
> +#define GET_SIP_SVC_VERSION (0x8200ff03U)
> +
> +/* SiP Service Calls version numbers */
> +#define SIP_SVC_VERSION_MAJOR (0U)
> +#define SIP_SVC_VERSION_MINOR (2U)
> +
> +#define SIP_SVC_PASSTHROUGH_VERSION ((SIP_SVC_VERSION_MAJOR << 16) | \
> + SIP_SVC_VERSION_MINOR)
> +
> +/* Fixed ID for FW specific APIs */
> +#define PASS_THROUGH_FW_CMD_ID GENMASK(11, 0)
> +
> /* PM API versions */
> #define PM_API_VERSION_1 1
> #define PM_API_VERSION_2 2
> @@ -51,6 +64,7 @@
>
> #define API_ID_MASK GENMASK(7, 0)
> #define MODULE_ID_MASK GENMASK(11, 8)
> +#define PLM_MODULE_ID_MASK GENMASK(15, 8)
>
> /* Firmware feature check version mask */
> #define FIRMWARE_VERSION_MASK 0xFFFFU
> @@ -62,7 +76,13 @@
> #define GET_CALLBACK_DATA 0xa01
>
> /* Number of 32bits values in payload */
> -#define PAYLOAD_ARG_CNT 4U
> +#define PAYLOAD_ARG_CNT 7U
> +
> +/* Number of 64bits arguments for SMC call */
> +#define SMC_ARG_CNT_64 8U
> +
> +/* Number of 32bits arguments for SMC call */
> +#define SMC_ARG_CNT_32 13U
>
> /* Number of arguments for a callback */
> #define CB_ARG_CNT 4
> @@ -130,6 +150,7 @@
>
> enum pm_module_id {
> PM_MODULE_ID = 0x0,
> + XPM_MODULE_ID = 0x2,
> XSEM_MODULE_ID = 0x3,
> TF_A_MODULE_ID = 0xa,
> };
> @@ -537,6 +558,7 @@ struct zynqmp_pm_query_data {
> };
>
> int zynqmp_pm_invoke_fn(u32 pm_api_id, u32 *ret_payload, u32 num_args, ...);
> +int zynqmp_pm_invoke_fw_fn(u32 pm_api_id, u32 *ret_payload, u32 num_args, ...);
>
> #if IS_REACHABLE(CONFIG_ZYNQMP_FIRMWARE)
> int zynqmp_pm_get_api_version(u32 *version);
Applied.
M
More information about the linux-arm-kernel
mailing list