mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-26 15:45:14 +08:00
arm64: cpufeature: Rework ptr auth hwcaps using multi_entry_cap_matches
Open-coding the pointer-auth HWCAPs is a mess and can be avoided by reusing the multi-cap logic from the CPU errata framework. Move the multi_entry_cap_matches code to cpufeature.h and reuse it for the pointer auth HWCAPs. Reviewed-by: Suzuki Poulose <suzuki.poulose@arm.com> Signed-off-by: Will Deacon <will.deacon@arm.com>
This commit is contained in:
parent
a56005d321
commit
1e013d0612
@ -321,19 +321,20 @@ struct arm64_cpu_capabilities {
|
|||||||
bool sign;
|
bool sign;
|
||||||
unsigned long hwcap;
|
unsigned long hwcap;
|
||||||
};
|
};
|
||||||
/*
|
|
||||||
* A list of "matches/cpu_enable" pair for the same
|
|
||||||
* "capability" of the same "type" as described by the parent.
|
|
||||||
* Only matches(), cpu_enable() and fields relevant to these
|
|
||||||
* methods are significant in the list. The cpu_enable is
|
|
||||||
* invoked only if the corresponding entry "matches()".
|
|
||||||
* However, if a cpu_enable() method is associated
|
|
||||||
* with multiple matches(), care should be taken that either
|
|
||||||
* the match criteria are mutually exclusive, or that the
|
|
||||||
* method is robust against being called multiple times.
|
|
||||||
*/
|
|
||||||
const struct arm64_cpu_capabilities *match_list;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* An optional list of "matches/cpu_enable" pair for the same
|
||||||
|
* "capability" of the same "type" as described by the parent.
|
||||||
|
* Only matches(), cpu_enable() and fields relevant to these
|
||||||
|
* methods are significant in the list. The cpu_enable is
|
||||||
|
* invoked only if the corresponding entry "matches()".
|
||||||
|
* However, if a cpu_enable() method is associated
|
||||||
|
* with multiple matches(), care should be taken that either
|
||||||
|
* the match criteria are mutually exclusive, or that the
|
||||||
|
* method is robust against being called multiple times.
|
||||||
|
*/
|
||||||
|
const struct arm64_cpu_capabilities *match_list;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline int cpucap_default_scope(const struct arm64_cpu_capabilities *cap)
|
static inline int cpucap_default_scope(const struct arm64_cpu_capabilities *cap)
|
||||||
@ -353,6 +354,39 @@ cpucap_late_cpu_permitted(const struct arm64_cpu_capabilities *cap)
|
|||||||
return !!(cap->type & ARM64_CPUCAP_PERMITTED_FOR_LATE_CPU);
|
return !!(cap->type & ARM64_CPUCAP_PERMITTED_FOR_LATE_CPU);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Generic helper for handling capabilties with multiple (match,enable) pairs
|
||||||
|
* of call backs, sharing the same capability bit.
|
||||||
|
* Iterate over each entry to see if at least one matches.
|
||||||
|
*/
|
||||||
|
static inline bool
|
||||||
|
cpucap_multi_entry_cap_matches(const struct arm64_cpu_capabilities *entry,
|
||||||
|
int scope)
|
||||||
|
{
|
||||||
|
const struct arm64_cpu_capabilities *caps;
|
||||||
|
|
||||||
|
for (caps = entry->match_list; caps->matches; caps++)
|
||||||
|
if (caps->matches(caps, scope))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Take appropriate action for all matching entries in the shared capability
|
||||||
|
* entry.
|
||||||
|
*/
|
||||||
|
static inline void
|
||||||
|
cpucap_multi_entry_cap_cpu_enable(const struct arm64_cpu_capabilities *entry)
|
||||||
|
{
|
||||||
|
const struct arm64_cpu_capabilities *caps;
|
||||||
|
|
||||||
|
for (caps = entry->match_list; caps->matches; caps++)
|
||||||
|
if (caps->matches(caps, SCOPE_LOCAL_CPU) &&
|
||||||
|
caps->cpu_enable)
|
||||||
|
caps->cpu_enable(caps);
|
||||||
|
}
|
||||||
|
|
||||||
extern DECLARE_BITMAP(cpu_hwcaps, ARM64_NCAPS);
|
extern DECLARE_BITMAP(cpu_hwcaps, ARM64_NCAPS);
|
||||||
extern struct static_key_false cpu_hwcap_keys[ARM64_NCAPS];
|
extern struct static_key_false cpu_hwcap_keys[ARM64_NCAPS];
|
||||||
extern struct static_key_false arm64_const_caps_ready;
|
extern struct static_key_false arm64_const_caps_ready;
|
||||||
@ -476,7 +510,6 @@ static inline bool id_aa64pfr0_sve(u64 pfr0)
|
|||||||
void __init setup_cpu_features(void);
|
void __init setup_cpu_features(void);
|
||||||
void check_local_cpu_capabilities(void);
|
void check_local_cpu_capabilities(void);
|
||||||
|
|
||||||
|
|
||||||
u64 read_sanitised_ftr_reg(u32 id);
|
u64 read_sanitised_ftr_reg(u32 id);
|
||||||
|
|
||||||
static inline bool cpu_supports_mixed_endian_el0(void)
|
static inline bool cpu_supports_mixed_endian_el0(void)
|
||||||
|
@ -507,38 +507,6 @@ cpu_enable_cache_maint_trap(const struct arm64_cpu_capabilities *__unused)
|
|||||||
.type = ARM64_CPUCAP_LOCAL_CPU_ERRATUM, \
|
.type = ARM64_CPUCAP_LOCAL_CPU_ERRATUM, \
|
||||||
CAP_MIDR_RANGE_LIST(midr_list)
|
CAP_MIDR_RANGE_LIST(midr_list)
|
||||||
|
|
||||||
/*
|
|
||||||
* Generic helper for handling capabilties with multiple (match,enable) pairs
|
|
||||||
* of call backs, sharing the same capability bit.
|
|
||||||
* Iterate over each entry to see if at least one matches.
|
|
||||||
*/
|
|
||||||
static bool __maybe_unused
|
|
||||||
multi_entry_cap_matches(const struct arm64_cpu_capabilities *entry, int scope)
|
|
||||||
{
|
|
||||||
const struct arm64_cpu_capabilities *caps;
|
|
||||||
|
|
||||||
for (caps = entry->match_list; caps->matches; caps++)
|
|
||||||
if (caps->matches(caps, scope))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Take appropriate action for all matching entries in the shared capability
|
|
||||||
* entry.
|
|
||||||
*/
|
|
||||||
static void __maybe_unused
|
|
||||||
multi_entry_cap_cpu_enable(const struct arm64_cpu_capabilities *entry)
|
|
||||||
{
|
|
||||||
const struct arm64_cpu_capabilities *caps;
|
|
||||||
|
|
||||||
for (caps = entry->match_list; caps->matches; caps++)
|
|
||||||
if (caps->matches(caps, SCOPE_LOCAL_CPU) &&
|
|
||||||
caps->cpu_enable)
|
|
||||||
caps->cpu_enable(caps);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef CONFIG_HARDEN_BRANCH_PREDICTOR
|
#ifdef CONFIG_HARDEN_BRANCH_PREDICTOR
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -700,7 +668,7 @@ const struct arm64_cpu_capabilities arm64_errata[] = {
|
|||||||
{
|
{
|
||||||
.desc = "Qualcomm Technologies Falkor/Kryo erratum 1003",
|
.desc = "Qualcomm Technologies Falkor/Kryo erratum 1003",
|
||||||
.capability = ARM64_WORKAROUND_QCOM_FALKOR_E1003,
|
.capability = ARM64_WORKAROUND_QCOM_FALKOR_E1003,
|
||||||
.matches = multi_entry_cap_matches,
|
.matches = cpucap_multi_entry_cap_matches,
|
||||||
.match_list = qcom_erratum_1003_list,
|
.match_list = qcom_erratum_1003_list,
|
||||||
},
|
},
|
||||||
#endif
|
#endif
|
||||||
|
@ -1196,34 +1196,6 @@ static void cpu_enable_address_auth(struct arm64_cpu_capabilities const *cap)
|
|||||||
sysreg_clear_set(sctlr_el1, 0, SCTLR_ELx_ENIA | SCTLR_ELx_ENIB |
|
sysreg_clear_set(sctlr_el1, 0, SCTLR_ELx_ENIA | SCTLR_ELx_ENIB |
|
||||||
SCTLR_ELx_ENDA | SCTLR_ELx_ENDB);
|
SCTLR_ELx_ENDA | SCTLR_ELx_ENDB);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool has_address_auth(const struct arm64_cpu_capabilities *entry,
|
|
||||||
int __unused)
|
|
||||||
{
|
|
||||||
u64 isar1 = read_sanitised_ftr_reg(SYS_ID_AA64ISAR1_EL1);
|
|
||||||
bool api, apa;
|
|
||||||
|
|
||||||
apa = cpuid_feature_extract_unsigned_field(isar1,
|
|
||||||
ID_AA64ISAR1_APA_SHIFT) > 0;
|
|
||||||
api = cpuid_feature_extract_unsigned_field(isar1,
|
|
||||||
ID_AA64ISAR1_API_SHIFT) > 0;
|
|
||||||
|
|
||||||
return apa || api;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool has_generic_auth(const struct arm64_cpu_capabilities *entry,
|
|
||||||
int __unused)
|
|
||||||
{
|
|
||||||
u64 isar1 = read_sanitised_ftr_reg(SYS_ID_AA64ISAR1_EL1);
|
|
||||||
bool gpi, gpa;
|
|
||||||
|
|
||||||
gpa = cpuid_feature_extract_unsigned_field(isar1,
|
|
||||||
ID_AA64ISAR1_GPA_SHIFT) > 0;
|
|
||||||
gpi = cpuid_feature_extract_unsigned_field(isar1,
|
|
||||||
ID_AA64ISAR1_GPI_SHIFT) > 0;
|
|
||||||
|
|
||||||
return gpa || gpi;
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_ARM64_PTR_AUTH */
|
#endif /* CONFIG_ARM64_PTR_AUTH */
|
||||||
|
|
||||||
static const struct arm64_cpu_capabilities arm64_features[] = {
|
static const struct arm64_cpu_capabilities arm64_features[] = {
|
||||||
@ -1506,19 +1478,58 @@ static const struct arm64_cpu_capabilities arm64_features[] = {
|
|||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
|
|
||||||
#define HWCAP_CAP(reg, field, s, min_value, cap_type, cap) \
|
#define HWCAP_CPUID_MATCH(reg, field, s, min_value) \
|
||||||
{ \
|
.matches = has_cpuid_feature, \
|
||||||
.desc = #cap, \
|
.sys_reg = reg, \
|
||||||
.type = ARM64_CPUCAP_SYSTEM_FEATURE, \
|
.field_pos = field, \
|
||||||
.matches = has_cpuid_feature, \
|
.sign = s, \
|
||||||
.sys_reg = reg, \
|
.min_field_value = min_value,
|
||||||
.field_pos = field, \
|
|
||||||
.sign = s, \
|
#define __HWCAP_CAP(name, cap_type, cap) \
|
||||||
.min_field_value = min_value, \
|
.desc = name, \
|
||||||
.hwcap_type = cap_type, \
|
.type = ARM64_CPUCAP_SYSTEM_FEATURE, \
|
||||||
.hwcap = cap, \
|
.hwcap_type = cap_type, \
|
||||||
|
.hwcap = cap, \
|
||||||
|
|
||||||
|
#define HWCAP_CAP(reg, field, s, min_value, cap_type, cap) \
|
||||||
|
{ \
|
||||||
|
__HWCAP_CAP(#cap, cap_type, cap) \
|
||||||
|
HWCAP_CPUID_MATCH(reg, field, s, min_value) \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define HWCAP_MULTI_CAP(list, cap_type, cap) \
|
||||||
|
{ \
|
||||||
|
__HWCAP_CAP(#cap, cap_type, cap) \
|
||||||
|
.matches = cpucap_multi_entry_cap_matches, \
|
||||||
|
.match_list = list, \
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_ARM64_PTR_AUTH
|
||||||
|
static const struct arm64_cpu_capabilities ptr_auth_hwcap_addr_matches[] = {
|
||||||
|
{
|
||||||
|
HWCAP_CPUID_MATCH(SYS_ID_AA64ISAR1_EL1, ID_AA64ISAR1_APA_SHIFT,
|
||||||
|
FTR_UNSIGNED, ID_AA64ISAR1_APA_ARCHITECTED)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
HWCAP_CPUID_MATCH(SYS_ID_AA64ISAR1_EL1, ID_AA64ISAR1_API_SHIFT,
|
||||||
|
FTR_UNSIGNED, ID_AA64ISAR1_API_IMP_DEF)
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct arm64_cpu_capabilities ptr_auth_hwcap_gen_matches[] = {
|
||||||
|
{
|
||||||
|
HWCAP_CPUID_MATCH(SYS_ID_AA64ISAR1_EL1, ID_AA64ISAR1_GPA_SHIFT,
|
||||||
|
FTR_UNSIGNED, ID_AA64ISAR1_GPA_ARCHITECTED)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
HWCAP_CPUID_MATCH(SYS_ID_AA64ISAR1_EL1, ID_AA64ISAR1_GPI_SHIFT,
|
||||||
|
FTR_UNSIGNED, ID_AA64ISAR1_GPI_IMP_DEF)
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
static const struct arm64_cpu_capabilities arm64_elf_hwcaps[] = {
|
static const struct arm64_cpu_capabilities arm64_elf_hwcaps[] = {
|
||||||
HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, FTR_UNSIGNED, 2, CAP_HWCAP, HWCAP_PMULL),
|
HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, FTR_UNSIGNED, 2, CAP_HWCAP, HWCAP_PMULL),
|
||||||
HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, FTR_UNSIGNED, 1, CAP_HWCAP, HWCAP_AES),
|
HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, FTR_UNSIGNED, 1, CAP_HWCAP, HWCAP_AES),
|
||||||
@ -1551,10 +1562,8 @@ static const struct arm64_cpu_capabilities arm64_elf_hwcaps[] = {
|
|||||||
#endif
|
#endif
|
||||||
HWCAP_CAP(SYS_ID_AA64PFR1_EL1, ID_AA64PFR1_SSBS_SHIFT, FTR_UNSIGNED, ID_AA64PFR1_SSBS_PSTATE_INSNS, CAP_HWCAP, HWCAP_SSBS),
|
HWCAP_CAP(SYS_ID_AA64PFR1_EL1, ID_AA64PFR1_SSBS_SHIFT, FTR_UNSIGNED, ID_AA64PFR1_SSBS_PSTATE_INSNS, CAP_HWCAP, HWCAP_SSBS),
|
||||||
#ifdef CONFIG_ARM64_PTR_AUTH
|
#ifdef CONFIG_ARM64_PTR_AUTH
|
||||||
{ .desc = "HWCAP_PACA", .type = ARM64_CPUCAP_SYSTEM_FEATURE, .matches = has_address_auth,
|
HWCAP_MULTI_CAP(ptr_auth_hwcap_addr_matches, CAP_HWCAP, HWCAP_PACA),
|
||||||
.hwcap_type = CAP_HWCAP, .hwcap = HWCAP_PACA },
|
HWCAP_MULTI_CAP(ptr_auth_hwcap_gen_matches, CAP_HWCAP, HWCAP_PACG),
|
||||||
{ .desc = "HWCAP_PACG", .type = ARM64_CPUCAP_SYSTEM_FEATURE, .matches = has_generic_auth,
|
|
||||||
.hwcap_type = CAP_HWCAP, .hwcap = HWCAP_PACG },
|
|
||||||
#endif
|
#endif
|
||||||
{},
|
{},
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user