mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-07 14:32:23 +00:00
8c462d5648
SMCCCv1.3 added a hint bit which callers can set in an SMCCC function ID (AKA "FID") to indicate that it is acceptable for the SMCCC implementation to discard SVE and/or SME state over a specific SMCCC call. The kernel support for using this hint is broken and SMCCC calls may clobber the SVE and/or SME state of arbitrary tasks, though FPSIMD state is unaffected. The kernel support is intended to use the hint when there is no SVE or SME state to save, and to do this it checks whether TIF_FOREIGN_FPSTATE is set or TIF_SVE is clear in assembly code: | ldr <flags>, [<current_task>, #TSK_TI_FLAGS] | tbnz <flags>, #TIF_FOREIGN_FPSTATE, 1f // Any live FP state? | tbnz <flags>, #TIF_SVE, 2f // Does that state include SVE? | | 1: orr <fid>, <fid>, ARM_SMCCC_1_3_SVE_HINT | 2: | << SMCCC call using FID >> This is not safe as-is: (1) SMCCC calls can be made in a preemptible context and preemption can result in TIF_FOREIGN_FPSTATE being set or cleared at arbitrary points in time. Thus checking for TIF_FOREIGN_FPSTATE provides no guarantee. (2) TIF_FOREIGN_FPSTATE only indicates that the live FP/SVE/SME state in the CPU does not belong to the current task, and does not indicate that clobbering this state is acceptable. When the live CPU state is clobbered it is necessary to update fpsimd_last_state.st to ensure that a subsequent context switch will reload FP/SVE/SME state from memory rather than consuming the clobbered state. This and the SMCCC call itself must happen in a critical section with preemption disabled to avoid races. (3) Live SVE/SME state can exist with TIF_SVE clear (e.g. with only TIF_SME set), and checking TIF_SVE alone is insufficient. Remove the broken support for the SMCCCv1.3 SVE saving hint. This is effectively a revert of commits: *cfa7ff959a
("arm64: smccc: Support SMCCC v1.3 SVE register saving hint") *a7c3acca53
("arm64: smccc: Save lr before calling __arm_smccc_sve_check()") ... leaving behind the ARM_SMCCC_VERSION_1_3 and ARM_SMCCC_1_3_SVE_HINT definitions, since these are simply definitions from the SMCCC specification, and the latter is used in KVM via ARM_SMCCC_CALL_HINTS. If we want to bring this back in future, we'll probably want to handle this logic in C where we can use all the usual FPSIMD/SVE/SME helper functions, and that'll likely require some rework of the SMCCC code and/or its callers. Fixes:cfa7ff959a
("arm64: smccc: Support SMCCC v1.3 SVE register saving hint") Signed-off-by: Mark Rutland <mark.rutland@arm.com> Cc: Ard Biesheuvel <ardb@kernel.org> Cc: Catalin Marinas <catalin.marinas@arm.com> Cc: Marc Zyngier <maz@kernel.org> Cc: Mark Brown <broonie@kernel.org> Cc: Will Deacon <will@kernel.org> Cc: stable@vger.kernel.org Reviewed-by: Mark Brown <broonie@kernel.org> Link: https://lore.kernel.org/r/20241106160448.2712997-1-mark.rutland@arm.com Signed-off-by: Will Deacon <will@kernel.org>
85 lines
2.0 KiB
C
85 lines
2.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 2020 Arm Limited
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "smccc: " fmt
|
|
|
|
#include <linux/cache.h>
|
|
#include <linux/init.h>
|
|
#include <linux/arm-smccc.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/platform_device.h>
|
|
#include <asm/archrandom.h>
|
|
|
|
static u32 smccc_version = ARM_SMCCC_VERSION_1_0;
|
|
static enum arm_smccc_conduit smccc_conduit = SMCCC_CONDUIT_NONE;
|
|
|
|
bool __ro_after_init smccc_trng_available = false;
|
|
s32 __ro_after_init smccc_soc_id_version = SMCCC_RET_NOT_SUPPORTED;
|
|
s32 __ro_after_init smccc_soc_id_revision = SMCCC_RET_NOT_SUPPORTED;
|
|
|
|
void __init arm_smccc_version_init(u32 version, enum arm_smccc_conduit conduit)
|
|
{
|
|
struct arm_smccc_res res;
|
|
|
|
smccc_version = version;
|
|
smccc_conduit = conduit;
|
|
|
|
smccc_trng_available = smccc_probe_trng();
|
|
|
|
if ((smccc_version >= ARM_SMCCC_VERSION_1_2) &&
|
|
(smccc_conduit != SMCCC_CONDUIT_NONE)) {
|
|
arm_smccc_1_1_invoke(ARM_SMCCC_ARCH_FEATURES_FUNC_ID,
|
|
ARM_SMCCC_ARCH_SOC_ID, &res);
|
|
if ((s32)res.a0 >= 0) {
|
|
arm_smccc_1_1_invoke(ARM_SMCCC_ARCH_SOC_ID, 0, &res);
|
|
smccc_soc_id_version = (s32)res.a0;
|
|
arm_smccc_1_1_invoke(ARM_SMCCC_ARCH_SOC_ID, 1, &res);
|
|
smccc_soc_id_revision = (s32)res.a0;
|
|
}
|
|
}
|
|
}
|
|
|
|
enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void)
|
|
{
|
|
if (smccc_version < ARM_SMCCC_VERSION_1_1)
|
|
return SMCCC_CONDUIT_NONE;
|
|
|
|
return smccc_conduit;
|
|
}
|
|
EXPORT_SYMBOL_GPL(arm_smccc_1_1_get_conduit);
|
|
|
|
u32 arm_smccc_get_version(void)
|
|
{
|
|
return smccc_version;
|
|
}
|
|
EXPORT_SYMBOL_GPL(arm_smccc_get_version);
|
|
|
|
s32 arm_smccc_get_soc_id_version(void)
|
|
{
|
|
return smccc_soc_id_version;
|
|
}
|
|
|
|
s32 arm_smccc_get_soc_id_revision(void)
|
|
{
|
|
return smccc_soc_id_revision;
|
|
}
|
|
EXPORT_SYMBOL_GPL(arm_smccc_get_soc_id_revision);
|
|
|
|
static int __init smccc_devices_init(void)
|
|
{
|
|
struct platform_device *pdev;
|
|
|
|
if (smccc_trng_available) {
|
|
pdev = platform_device_register_simple("smccc_trng", -1,
|
|
NULL, 0);
|
|
if (IS_ERR(pdev))
|
|
pr_err("smccc_trng: could not register device: %ld\n",
|
|
PTR_ERR(pdev));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
device_initcall(smccc_devices_init);
|