|
@@ -24,6 +24,7 @@
|
|
|
#include <asm/cpu.h>
|
|
|
#include <asm/cpufeature.h>
|
|
|
#include <asm/cpu_ops.h>
|
|
|
+#include <asm/mmu_context.h>
|
|
|
#include <asm/processor.h>
|
|
|
#include <asm/sysreg.h>
|
|
|
#include <asm/virt.h>
|
|
@@ -55,19 +56,23 @@ DECLARE_BITMAP(cpu_hwcaps, ARM64_NCAPS);
|
|
|
.safe_val = SAFE_VAL, \
|
|
|
}
|
|
|
|
|
|
-/* Define a feature with signed values */
|
|
|
+/* Define a feature with unsigned values */
|
|
|
#define ARM64_FTR_BITS(STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL) \
|
|
|
- __ARM64_FTR_BITS(FTR_SIGNED, STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL)
|
|
|
-
|
|
|
-/* Define a feature with unsigned value */
|
|
|
-#define U_ARM64_FTR_BITS(STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL) \
|
|
|
__ARM64_FTR_BITS(FTR_UNSIGNED, STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL)
|
|
|
|
|
|
+/* Define a feature with a signed value */
|
|
|
+#define S_ARM64_FTR_BITS(STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL) \
|
|
|
+ __ARM64_FTR_BITS(FTR_SIGNED, STRICT, TYPE, SHIFT, WIDTH, SAFE_VAL)
|
|
|
+
|
|
|
#define ARM64_FTR_END \
|
|
|
{ \
|
|
|
.width = 0, \
|
|
|
}
|
|
|
|
|
|
+/* meta feature for alternatives */
|
|
|
+static bool __maybe_unused
|
|
|
+cpufeature_pan_not_uao(const struct arm64_cpu_capabilities *entry);
|
|
|
+
|
|
|
static struct arm64_ftr_bits ftr_id_aa64isar0[] = {
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 32, 32, 0),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64ISAR0_RDM_SHIFT, 4, 0),
|
|
@@ -85,8 +90,8 @@ static struct arm64_ftr_bits ftr_id_aa64pfr0[] = {
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 32, 32, 0),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 28, 4, 0),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64PFR0_GIC_SHIFT, 4, 0),
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64PFR0_ASIMD_SHIFT, 4, ID_AA64PFR0_ASIMD_NI),
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64PFR0_FP_SHIFT, 4, ID_AA64PFR0_FP_NI),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64PFR0_ASIMD_SHIFT, 4, ID_AA64PFR0_ASIMD_NI),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64PFR0_FP_SHIFT, 4, ID_AA64PFR0_FP_NI),
|
|
|
/* Linux doesn't care about the EL3 */
|
|
|
ARM64_FTR_BITS(FTR_NONSTRICT, FTR_EXACT, ID_AA64PFR0_EL3_SHIFT, 4, 0),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64PFR0_EL2_SHIFT, 4, 0),
|
|
@@ -97,8 +102,8 @@ static struct arm64_ftr_bits ftr_id_aa64pfr0[] = {
|
|
|
|
|
|
static struct arm64_ftr_bits ftr_id_aa64mmfr0[] = {
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 32, 32, 0),
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_TGRAN4_SHIFT, 4, ID_AA64MMFR0_TGRAN4_NI),
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_TGRAN64_SHIFT, 4, ID_AA64MMFR0_TGRAN64_NI),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_TGRAN4_SHIFT, 4, ID_AA64MMFR0_TGRAN4_NI),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_TGRAN64_SHIFT, 4, ID_AA64MMFR0_TGRAN64_NI),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_TGRAN16_SHIFT, 4, ID_AA64MMFR0_TGRAN16_NI),
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR0_BIGENDEL0_SHIFT, 4, 0),
|
|
|
/* Linux shouldn't care about secure memory */
|
|
@@ -109,7 +114,7 @@ static struct arm64_ftr_bits ftr_id_aa64mmfr0[] = {
|
|
|
* Differing PARange is fine as long as all peripherals and memory are mapped
|
|
|
* within the minimum PARange of all CPUs
|
|
|
*/
|
|
|
- U_ARM64_FTR_BITS(FTR_NONSTRICT, FTR_LOWER_SAFE, ID_AA64MMFR0_PARANGE_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_NONSTRICT, FTR_LOWER_SAFE, ID_AA64MMFR0_PARANGE_SHIFT, 4, 0),
|
|
|
ARM64_FTR_END,
|
|
|
};
|
|
|
|
|
@@ -124,29 +129,34 @@ static struct arm64_ftr_bits ftr_id_aa64mmfr1[] = {
|
|
|
ARM64_FTR_END,
|
|
|
};
|
|
|
|
|
|
+static struct arm64_ftr_bits ftr_id_aa64mmfr2[] = {
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64MMFR2_UAO_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_END,
|
|
|
+};
|
|
|
+
|
|
|
static struct arm64_ftr_bits ftr_ctr[] = {
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 31, 1, 1), /* RAO */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 31, 1, 1), /* RAO */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 28, 3, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_HIGHER_SAFE, 24, 4, 0), /* CWG */
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 20, 4, 0), /* ERG */
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 16, 4, 1), /* DminLine */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_HIGHER_SAFE, 24, 4, 0), /* CWG */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 20, 4, 0), /* ERG */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 16, 4, 1), /* DminLine */
|
|
|
/*
|
|
|
* Linux can handle differing I-cache policies. Userspace JITs will
|
|
|
* make use of *minLine
|
|
|
*/
|
|
|
- U_ARM64_FTR_BITS(FTR_NONSTRICT, FTR_EXACT, 14, 2, 0), /* L1Ip */
|
|
|
+ ARM64_FTR_BITS(FTR_NONSTRICT, FTR_EXACT, 14, 2, 0), /* L1Ip */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 4, 10, 0), /* RAZ */
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 0, 4, 0), /* IminLine */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 0, 4, 0), /* IminLine */
|
|
|
ARM64_FTR_END,
|
|
|
};
|
|
|
|
|
|
static struct arm64_ftr_bits ftr_id_mmfr0[] = {
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 28, 4, 0), /* InnerShr */
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 28, 4, 0xf), /* InnerShr */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 24, 4, 0), /* FCSE */
|
|
|
ARM64_FTR_BITS(FTR_NONSTRICT, FTR_LOWER_SAFE, 20, 4, 0), /* AuxReg */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 16, 4, 0), /* TCM */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 12, 4, 0), /* ShareLvl */
|
|
|
- ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 8, 4, 0), /* OuterShr */
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 8, 4, 0xf), /* OuterShr */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 4, 4, 0), /* PMSA */
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 0, 4, 0), /* VMSA */
|
|
|
ARM64_FTR_END,
|
|
@@ -154,12 +164,12 @@ static struct arm64_ftr_bits ftr_id_mmfr0[] = {
|
|
|
|
|
|
static struct arm64_ftr_bits ftr_id_aa64dfr0[] = {
|
|
|
ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, 32, 32, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_CTX_CMPS_SHIFT, 4, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_WRPS_SHIFT, 4, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_BRPS_SHIFT, 4, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_PMUVER_SHIFT, 4, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_TRACEVER_SHIFT, 4, 0),
|
|
|
- U_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_DEBUGVER_SHIFT, 4, 0x6),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_CTX_CMPS_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_WRPS_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, ID_AA64DFR0_BRPS_SHIFT, 4, 0),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_PMUVER_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_TRACEVER_SHIFT, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_EXACT, ID_AA64DFR0_DEBUGVER_SHIFT, 4, 0x6),
|
|
|
ARM64_FTR_END,
|
|
|
};
|
|
|
|
|
@@ -205,6 +215,18 @@ static struct arm64_ftr_bits ftr_id_pfr0[] = {
|
|
|
ARM64_FTR_END,
|
|
|
};
|
|
|
|
|
|
+static struct arm64_ftr_bits ftr_id_dfr0[] = {
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 28, 4, 0),
|
|
|
+ S_ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 24, 4, 0xf), /* PerfMon */
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 20, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 16, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 12, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 8, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 4, 4, 0),
|
|
|
+ ARM64_FTR_BITS(FTR_STRICT, FTR_LOWER_SAFE, 0, 4, 0),
|
|
|
+ ARM64_FTR_END,
|
|
|
+};
|
|
|
+
|
|
|
/*
|
|
|
* Common ftr bits for a 32bit register with all hidden, strict
|
|
|
* attributes, with 4bit feature fields and a default safe value of
|
|
@@ -250,7 +272,7 @@ static struct arm64_ftr_reg arm64_ftr_regs[] = {
|
|
|
/* Op1 = 0, CRn = 0, CRm = 1 */
|
|
|
ARM64_FTR_REG(SYS_ID_PFR0_EL1, ftr_id_pfr0),
|
|
|
ARM64_FTR_REG(SYS_ID_PFR1_EL1, ftr_generic_32bits),
|
|
|
- ARM64_FTR_REG(SYS_ID_DFR0_EL1, ftr_generic_32bits),
|
|
|
+ ARM64_FTR_REG(SYS_ID_DFR0_EL1, ftr_id_dfr0),
|
|
|
ARM64_FTR_REG(SYS_ID_MMFR0_EL1, ftr_id_mmfr0),
|
|
|
ARM64_FTR_REG(SYS_ID_MMFR1_EL1, ftr_generic_32bits),
|
|
|
ARM64_FTR_REG(SYS_ID_MMFR2_EL1, ftr_generic_32bits),
|
|
@@ -285,6 +307,7 @@ static struct arm64_ftr_reg arm64_ftr_regs[] = {
|
|
|
/* Op1 = 0, CRn = 0, CRm = 7 */
|
|
|
ARM64_FTR_REG(SYS_ID_AA64MMFR0_EL1, ftr_id_aa64mmfr0),
|
|
|
ARM64_FTR_REG(SYS_ID_AA64MMFR1_EL1, ftr_id_aa64mmfr1),
|
|
|
+ ARM64_FTR_REG(SYS_ID_AA64MMFR2_EL1, ftr_id_aa64mmfr2),
|
|
|
|
|
|
/* Op1 = 3, CRn = 0, CRm = 0 */
|
|
|
ARM64_FTR_REG(SYS_CTR_EL0, ftr_ctr),
|
|
@@ -409,6 +432,7 @@ void __init init_cpu_features(struct cpuinfo_arm64 *info)
|
|
|
init_cpu_ftr_reg(SYS_ID_AA64ISAR1_EL1, info->reg_id_aa64isar1);
|
|
|
init_cpu_ftr_reg(SYS_ID_AA64MMFR0_EL1, info->reg_id_aa64mmfr0);
|
|
|
init_cpu_ftr_reg(SYS_ID_AA64MMFR1_EL1, info->reg_id_aa64mmfr1);
|
|
|
+ init_cpu_ftr_reg(SYS_ID_AA64MMFR2_EL1, info->reg_id_aa64mmfr2);
|
|
|
init_cpu_ftr_reg(SYS_ID_AA64PFR0_EL1, info->reg_id_aa64pfr0);
|
|
|
init_cpu_ftr_reg(SYS_ID_AA64PFR1_EL1, info->reg_id_aa64pfr1);
|
|
|
init_cpu_ftr_reg(SYS_ID_DFR0_EL1, info->reg_id_dfr0);
|
|
@@ -518,6 +542,8 @@ void update_cpu_features(int cpu,
|
|
|
info->reg_id_aa64mmfr0, boot->reg_id_aa64mmfr0);
|
|
|
taint |= check_update_ftr_reg(SYS_ID_AA64MMFR1_EL1, cpu,
|
|
|
info->reg_id_aa64mmfr1, boot->reg_id_aa64mmfr1);
|
|
|
+ taint |= check_update_ftr_reg(SYS_ID_AA64MMFR2_EL1, cpu,
|
|
|
+ info->reg_id_aa64mmfr2, boot->reg_id_aa64mmfr2);
|
|
|
|
|
|
/*
|
|
|
* EL3 is not our concern.
|
|
@@ -593,7 +619,7 @@ u64 read_system_reg(u32 id)
|
|
|
static bool
|
|
|
feature_matches(u64 reg, const struct arm64_cpu_capabilities *entry)
|
|
|
{
|
|
|
- int val = cpuid_feature_extract_field(reg, entry->field_pos);
|
|
|
+ int val = cpuid_feature_extract_field(reg, entry->field_pos, entry->sign);
|
|
|
|
|
|
return val >= entry->min_field_value;
|
|
|
}
|
|
@@ -622,6 +648,18 @@ static bool has_useable_gicv3_cpuif(const struct arm64_cpu_capabilities *entry)
|
|
|
return has_sre;
|
|
|
}
|
|
|
|
|
|
+static bool has_no_hw_prefetch(const struct arm64_cpu_capabilities *entry)
|
|
|
+{
|
|
|
+ u32 midr = read_cpuid_id();
|
|
|
+ u32 rv_min, rv_max;
|
|
|
+
|
|
|
+ /* Cavium ThunderX pass 1.x and 2.x */
|
|
|
+ rv_min = 0;
|
|
|
+ rv_max = (1 << MIDR_VARIANT_SHIFT) | MIDR_REVISION_MASK;
|
|
|
+
|
|
|
+ return MIDR_IS_CPU_MODEL_RANGE(midr, MIDR_THUNDERX, rv_min, rv_max);
|
|
|
+}
|
|
|
+
|
|
|
static bool runs_at_el2(const struct arm64_cpu_capabilities *entry)
|
|
|
{
|
|
|
return is_kernel_in_hyp_mode();
|
|
@@ -634,6 +672,7 @@ static const struct arm64_cpu_capabilities arm64_features[] = {
|
|
|
.matches = has_useable_gicv3_cpuif,
|
|
|
.sys_reg = SYS_ID_AA64PFR0_EL1,
|
|
|
.field_pos = ID_AA64PFR0_GIC_SHIFT,
|
|
|
+ .sign = FTR_UNSIGNED,
|
|
|
.min_field_value = 1,
|
|
|
},
|
|
|
#ifdef CONFIG_ARM64_PAN
|
|
@@ -643,6 +682,7 @@ static const struct arm64_cpu_capabilities arm64_features[] = {
|
|
|
.matches = has_cpuid_feature,
|
|
|
.sys_reg = SYS_ID_AA64MMFR1_EL1,
|
|
|
.field_pos = ID_AA64MMFR1_PAN_SHIFT,
|
|
|
+ .sign = FTR_UNSIGNED,
|
|
|
.min_field_value = 1,
|
|
|
.enable = cpu_enable_pan,
|
|
|
},
|
|
@@ -654,9 +694,32 @@ static const struct arm64_cpu_capabilities arm64_features[] = {
|
|
|
.matches = has_cpuid_feature,
|
|
|
.sys_reg = SYS_ID_AA64ISAR0_EL1,
|
|
|
.field_pos = ID_AA64ISAR0_ATOMICS_SHIFT,
|
|
|
+ .sign = FTR_UNSIGNED,
|
|
|
.min_field_value = 2,
|
|
|
},
|
|
|
#endif /* CONFIG_AS_LSE && CONFIG_ARM64_LSE_ATOMICS */
|
|
|
+ {
|
|
|
+ .desc = "Software prefetching using PRFM",
|
|
|
+ .capability = ARM64_HAS_NO_HW_PREFETCH,
|
|
|
+ .matches = has_no_hw_prefetch,
|
|
|
+ },
|
|
|
+#ifdef CONFIG_ARM64_UAO
|
|
|
+ {
|
|
|
+ .desc = "User Access Override",
|
|
|
+ .capability = ARM64_HAS_UAO,
|
|
|
+ .matches = has_cpuid_feature,
|
|
|
+ .sys_reg = SYS_ID_AA64MMFR2_EL1,
|
|
|
+ .field_pos = ID_AA64MMFR2_UAO_SHIFT,
|
|
|
+ .min_field_value = 1,
|
|
|
+ .enable = cpu_enable_uao,
|
|
|
+ },
|
|
|
+#endif /* CONFIG_ARM64_UAO */
|
|
|
+#ifdef CONFIG_ARM64_PAN
|
|
|
+ {
|
|
|
+ .capability = ARM64_ALT_PAN_NOT_UAO,
|
|
|
+ .matches = cpufeature_pan_not_uao,
|
|
|
+ },
|
|
|
+#endif /* CONFIG_ARM64_PAN */
|
|
|
{
|
|
|
.desc = "Virtualization Host Extensions",
|
|
|
.capability = ARM64_HAS_VIRT_HOST_EXTN,
|
|
@@ -665,32 +728,35 @@ static const struct arm64_cpu_capabilities arm64_features[] = {
|
|
|
{},
|
|
|
};
|
|
|
|
|
|
-#define HWCAP_CAP(reg, field, min_value, type, cap) \
|
|
|
+#define HWCAP_CAP(reg, field, s, min_value, type, cap) \
|
|
|
{ \
|
|
|
.desc = #cap, \
|
|
|
.matches = has_cpuid_feature, \
|
|
|
.sys_reg = reg, \
|
|
|
.field_pos = field, \
|
|
|
+ .sign = s, \
|
|
|
.min_field_value = min_value, \
|
|
|
.hwcap_type = type, \
|
|
|
.hwcap = cap, \
|
|
|
}
|
|
|
|
|
|
static const struct arm64_cpu_capabilities arm64_hwcaps[] = {
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, 2, CAP_HWCAP, HWCAP_PMULL),
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_AES_SHIFT, 1, CAP_HWCAP, HWCAP_AES),
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_SHA1_SHIFT, 1, CAP_HWCAP, HWCAP_SHA1),
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_SHA2_SHIFT, 1, CAP_HWCAP, HWCAP_SHA2),
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_CRC32_SHIFT, 1, CAP_HWCAP, HWCAP_CRC32),
|
|
|
- HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_ATOMICS_SHIFT, 2, CAP_HWCAP, HWCAP_ATOMICS),
|
|
|
- HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_FP_SHIFT, 0, CAP_HWCAP, HWCAP_FP),
|
|
|
- HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_ASIMD_SHIFT, 0, CAP_HWCAP, HWCAP_ASIMD),
|
|
|
+ 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_SHA1_SHIFT, FTR_UNSIGNED, 1, CAP_HWCAP, HWCAP_SHA1),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_SHA2_SHIFT, FTR_UNSIGNED, 1, CAP_HWCAP, HWCAP_SHA2),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_CRC32_SHIFT, FTR_UNSIGNED, 1, CAP_HWCAP, HWCAP_CRC32),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64ISAR0_EL1, ID_AA64ISAR0_ATOMICS_SHIFT, FTR_UNSIGNED, 2, CAP_HWCAP, HWCAP_ATOMICS),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_FP_SHIFT, FTR_SIGNED, 0, CAP_HWCAP, HWCAP_FP),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_FP_SHIFT, FTR_SIGNED, 1, CAP_HWCAP, HWCAP_FPHP),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_ASIMD_SHIFT, FTR_SIGNED, 0, CAP_HWCAP, HWCAP_ASIMD),
|
|
|
+ HWCAP_CAP(SYS_ID_AA64PFR0_EL1, ID_AA64PFR0_ASIMD_SHIFT, FTR_SIGNED, 1, CAP_HWCAP, HWCAP_ASIMDHP),
|
|
|
#ifdef CONFIG_COMPAT
|
|
|
- HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_AES_SHIFT, 2, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_PMULL),
|
|
|
- HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_AES_SHIFT, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_AES),
|
|
|
- HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_SHA1_SHIFT, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_SHA1),
|
|
|
- HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_SHA2_SHIFT, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_SHA2),
|
|
|
- HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_CRC32_SHIFT, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_CRC32),
|
|
|
+ HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_AES_SHIFT, FTR_UNSIGNED, 2, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_PMULL),
|
|
|
+ HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_AES_SHIFT, FTR_UNSIGNED, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_AES),
|
|
|
+ HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_SHA1_SHIFT, FTR_UNSIGNED, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_SHA1),
|
|
|
+ HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_SHA2_SHIFT, FTR_UNSIGNED, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_SHA2),
|
|
|
+ HWCAP_CAP(SYS_ID_ISAR5_EL1, ID_ISAR5_CRC32_SHIFT, FTR_UNSIGNED, 1, CAP_COMPAT_HWCAP2, COMPAT_HWCAP2_CRC32),
|
|
|
#endif
|
|
|
{},
|
|
|
};
|
|
@@ -745,7 +811,7 @@ static void __init setup_cpu_hwcaps(void)
|
|
|
int i;
|
|
|
const struct arm64_cpu_capabilities *hwcaps = arm64_hwcaps;
|
|
|
|
|
|
- for (i = 0; hwcaps[i].desc; i++)
|
|
|
+ for (i = 0; hwcaps[i].matches; i++)
|
|
|
if (hwcaps[i].matches(&hwcaps[i]))
|
|
|
cap_set_hwcap(&hwcaps[i]);
|
|
|
}
|
|
@@ -755,11 +821,11 @@ void update_cpu_capabilities(const struct arm64_cpu_capabilities *caps,
|
|
|
{
|
|
|
int i;
|
|
|
|
|
|
- for (i = 0; caps[i].desc; i++) {
|
|
|
+ for (i = 0; caps[i].matches; i++) {
|
|
|
if (!caps[i].matches(&caps[i]))
|
|
|
continue;
|
|
|
|
|
|
- if (!cpus_have_cap(caps[i].capability))
|
|
|
+ if (!cpus_have_cap(caps[i].capability) && caps[i].desc)
|
|
|
pr_info("%s %s\n", info, caps[i].desc);
|
|
|
cpus_set_cap(caps[i].capability);
|
|
|
}
|
|
@@ -774,13 +840,11 @@ enable_cpu_capabilities(const struct arm64_cpu_capabilities *caps)
|
|
|
{
|
|
|
int i;
|
|
|
|
|
|
- for (i = 0; caps[i].desc; i++)
|
|
|
+ for (i = 0; caps[i].matches; i++)
|
|
|
if (caps[i].enable && cpus_have_cap(caps[i].capability))
|
|
|
on_each_cpu(caps[i].enable, NULL, true);
|
|
|
}
|
|
|
|
|
|
-#ifdef CONFIG_HOTPLUG_CPU
|
|
|
-
|
|
|
/*
|
|
|
* Flag to indicate if we have computed the system wide
|
|
|
* capabilities based on the boot time active CPUs. This
|
|
@@ -802,35 +866,36 @@ static inline void set_sys_caps_initialised(void)
|
|
|
static u64 __raw_read_system_reg(u32 sys_id)
|
|
|
{
|
|
|
switch (sys_id) {
|
|
|
- case SYS_ID_PFR0_EL1: return (u64)read_cpuid(ID_PFR0_EL1);
|
|
|
- case SYS_ID_PFR1_EL1: return (u64)read_cpuid(ID_PFR1_EL1);
|
|
|
- case SYS_ID_DFR0_EL1: return (u64)read_cpuid(ID_DFR0_EL1);
|
|
|
- case SYS_ID_MMFR0_EL1: return (u64)read_cpuid(ID_MMFR0_EL1);
|
|
|
- case SYS_ID_MMFR1_EL1: return (u64)read_cpuid(ID_MMFR1_EL1);
|
|
|
- case SYS_ID_MMFR2_EL1: return (u64)read_cpuid(ID_MMFR2_EL1);
|
|
|
- case SYS_ID_MMFR3_EL1: return (u64)read_cpuid(ID_MMFR3_EL1);
|
|
|
- case SYS_ID_ISAR0_EL1: return (u64)read_cpuid(ID_ISAR0_EL1);
|
|
|
- case SYS_ID_ISAR1_EL1: return (u64)read_cpuid(ID_ISAR1_EL1);
|
|
|
- case SYS_ID_ISAR2_EL1: return (u64)read_cpuid(ID_ISAR2_EL1);
|
|
|
- case SYS_ID_ISAR3_EL1: return (u64)read_cpuid(ID_ISAR3_EL1);
|
|
|
- case SYS_ID_ISAR4_EL1: return (u64)read_cpuid(ID_ISAR4_EL1);
|
|
|
- case SYS_ID_ISAR5_EL1: return (u64)read_cpuid(ID_ISAR4_EL1);
|
|
|
- case SYS_MVFR0_EL1: return (u64)read_cpuid(MVFR0_EL1);
|
|
|
- case SYS_MVFR1_EL1: return (u64)read_cpuid(MVFR1_EL1);
|
|
|
- case SYS_MVFR2_EL1: return (u64)read_cpuid(MVFR2_EL1);
|
|
|
-
|
|
|
- case SYS_ID_AA64PFR0_EL1: return (u64)read_cpuid(ID_AA64PFR0_EL1);
|
|
|
- case SYS_ID_AA64PFR1_EL1: return (u64)read_cpuid(ID_AA64PFR0_EL1);
|
|
|
- case SYS_ID_AA64DFR0_EL1: return (u64)read_cpuid(ID_AA64DFR0_EL1);
|
|
|
- case SYS_ID_AA64DFR1_EL1: return (u64)read_cpuid(ID_AA64DFR0_EL1);
|
|
|
- case SYS_ID_AA64MMFR0_EL1: return (u64)read_cpuid(ID_AA64MMFR0_EL1);
|
|
|
- case SYS_ID_AA64MMFR1_EL1: return (u64)read_cpuid(ID_AA64MMFR1_EL1);
|
|
|
- case SYS_ID_AA64ISAR0_EL1: return (u64)read_cpuid(ID_AA64ISAR0_EL1);
|
|
|
- case SYS_ID_AA64ISAR1_EL1: return (u64)read_cpuid(ID_AA64ISAR1_EL1);
|
|
|
-
|
|
|
- case SYS_CNTFRQ_EL0: return (u64)read_cpuid(CNTFRQ_EL0);
|
|
|
- case SYS_CTR_EL0: return (u64)read_cpuid(CTR_EL0);
|
|
|
- case SYS_DCZID_EL0: return (u64)read_cpuid(DCZID_EL0);
|
|
|
+ case SYS_ID_PFR0_EL1: return read_cpuid(ID_PFR0_EL1);
|
|
|
+ case SYS_ID_PFR1_EL1: return read_cpuid(ID_PFR1_EL1);
|
|
|
+ case SYS_ID_DFR0_EL1: return read_cpuid(ID_DFR0_EL1);
|
|
|
+ case SYS_ID_MMFR0_EL1: return read_cpuid(ID_MMFR0_EL1);
|
|
|
+ case SYS_ID_MMFR1_EL1: return read_cpuid(ID_MMFR1_EL1);
|
|
|
+ case SYS_ID_MMFR2_EL1: return read_cpuid(ID_MMFR2_EL1);
|
|
|
+ case SYS_ID_MMFR3_EL1: return read_cpuid(ID_MMFR3_EL1);
|
|
|
+ case SYS_ID_ISAR0_EL1: return read_cpuid(ID_ISAR0_EL1);
|
|
|
+ case SYS_ID_ISAR1_EL1: return read_cpuid(ID_ISAR1_EL1);
|
|
|
+ case SYS_ID_ISAR2_EL1: return read_cpuid(ID_ISAR2_EL1);
|
|
|
+ case SYS_ID_ISAR3_EL1: return read_cpuid(ID_ISAR3_EL1);
|
|
|
+ case SYS_ID_ISAR4_EL1: return read_cpuid(ID_ISAR4_EL1);
|
|
|
+ case SYS_ID_ISAR5_EL1: return read_cpuid(ID_ISAR4_EL1);
|
|
|
+ case SYS_MVFR0_EL1: return read_cpuid(MVFR0_EL1);
|
|
|
+ case SYS_MVFR1_EL1: return read_cpuid(MVFR1_EL1);
|
|
|
+ case SYS_MVFR2_EL1: return read_cpuid(MVFR2_EL1);
|
|
|
+
|
|
|
+ case SYS_ID_AA64PFR0_EL1: return read_cpuid(ID_AA64PFR0_EL1);
|
|
|
+ case SYS_ID_AA64PFR1_EL1: return read_cpuid(ID_AA64PFR0_EL1);
|
|
|
+ case SYS_ID_AA64DFR0_EL1: return read_cpuid(ID_AA64DFR0_EL1);
|
|
|
+ case SYS_ID_AA64DFR1_EL1: return read_cpuid(ID_AA64DFR0_EL1);
|
|
|
+ case SYS_ID_AA64MMFR0_EL1: return read_cpuid(ID_AA64MMFR0_EL1);
|
|
|
+ case SYS_ID_AA64MMFR1_EL1: return read_cpuid(ID_AA64MMFR1_EL1);
|
|
|
+ case SYS_ID_AA64MMFR2_EL1: return read_cpuid(ID_AA64MMFR2_EL1);
|
|
|
+ case SYS_ID_AA64ISAR0_EL1: return read_cpuid(ID_AA64ISAR0_EL1);
|
|
|
+ case SYS_ID_AA64ISAR1_EL1: return read_cpuid(ID_AA64ISAR1_EL1);
|
|
|
+
|
|
|
+ case SYS_CNTFRQ_EL0: return read_cpuid(CNTFRQ_EL0);
|
|
|
+ case SYS_CTR_EL0: return read_cpuid(CTR_EL0);
|
|
|
+ case SYS_DCZID_EL0: return read_cpuid(DCZID_EL0);
|
|
|
default:
|
|
|
BUG();
|
|
|
return 0;
|
|
@@ -838,25 +903,12 @@ static u64 __raw_read_system_reg(u32 sys_id)
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
- * Park the CPU which doesn't have the capability as advertised
|
|
|
- * by the system.
|
|
|
+ * Check for CPU features that are used in early boot
|
|
|
+ * based on the Boot CPU value.
|
|
|
*/
|
|
|
-static void fail_incapable_cpu(char *cap_type,
|
|
|
- const struct arm64_cpu_capabilities *cap)
|
|
|
+static void check_early_cpu_features(void)
|
|
|
{
|
|
|
- int cpu = smp_processor_id();
|
|
|
-
|
|
|
- pr_crit("CPU%d: missing %s : %s\n", cpu, cap_type, cap->desc);
|
|
|
- /* Mark this CPU absent */
|
|
|
- set_cpu_present(cpu, 0);
|
|
|
-
|
|
|
- /* Check if we can park ourselves */
|
|
|
- if (cpu_ops[cpu] && cpu_ops[cpu]->cpu_die)
|
|
|
- cpu_ops[cpu]->cpu_die(cpu);
|
|
|
- asm(
|
|
|
- "1: wfe\n"
|
|
|
- " wfi\n"
|
|
|
- " b 1b");
|
|
|
+ verify_cpu_asid_bits();
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -872,6 +924,8 @@ void verify_local_cpu_capabilities(void)
|
|
|
int i;
|
|
|
const struct arm64_cpu_capabilities *caps;
|
|
|
|
|
|
+ check_early_cpu_features();
|
|
|
+
|
|
|
/*
|
|
|
* If we haven't computed the system capabilities, there is nothing
|
|
|
* to verify.
|
|
@@ -880,35 +934,33 @@ void verify_local_cpu_capabilities(void)
|
|
|
return;
|
|
|
|
|
|
caps = arm64_features;
|
|
|
- for (i = 0; caps[i].desc; i++) {
|
|
|
+ for (i = 0; caps[i].matches; i++) {
|
|
|
if (!cpus_have_cap(caps[i].capability) || !caps[i].sys_reg)
|
|
|
continue;
|
|
|
/*
|
|
|
* If the new CPU misses an advertised feature, we cannot proceed
|
|
|
* further, park the cpu.
|
|
|
*/
|
|
|
- if (!feature_matches(__raw_read_system_reg(caps[i].sys_reg), &caps[i]))
|
|
|
- fail_incapable_cpu("arm64_features", &caps[i]);
|
|
|
+ if (!feature_matches(__raw_read_system_reg(caps[i].sys_reg), &caps[i])) {
|
|
|
+ pr_crit("CPU%d: missing feature: %s\n",
|
|
|
+ smp_processor_id(), caps[i].desc);
|
|
|
+ cpu_die_early();
|
|
|
+ }
|
|
|
if (caps[i].enable)
|
|
|
caps[i].enable(NULL);
|
|
|
}
|
|
|
|
|
|
- for (i = 0, caps = arm64_hwcaps; caps[i].desc; i++) {
|
|
|
+ for (i = 0, caps = arm64_hwcaps; caps[i].matches; i++) {
|
|
|
if (!cpus_have_hwcap(&caps[i]))
|
|
|
continue;
|
|
|
- if (!feature_matches(__raw_read_system_reg(caps[i].sys_reg), &caps[i]))
|
|
|
- fail_incapable_cpu("arm64_hwcaps", &caps[i]);
|
|
|
+ if (!feature_matches(__raw_read_system_reg(caps[i].sys_reg), &caps[i])) {
|
|
|
+ pr_crit("CPU%d: missing HWCAP: %s\n",
|
|
|
+ smp_processor_id(), caps[i].desc);
|
|
|
+ cpu_die_early();
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-#else /* !CONFIG_HOTPLUG_CPU */
|
|
|
-
|
|
|
-static inline void set_sys_caps_initialised(void)
|
|
|
-{
|
|
|
-}
|
|
|
-
|
|
|
-#endif /* CONFIG_HOTPLUG_CPU */
|
|
|
-
|
|
|
static void __init setup_feature_capabilities(void)
|
|
|
{
|
|
|
update_cpu_capabilities(arm64_features, "detected feature:");
|
|
@@ -939,3 +991,9 @@ void __init setup_cpu_features(void)
|
|
|
pr_warn("L1_CACHE_BYTES smaller than the Cache Writeback Granule (%d < %d)\n",
|
|
|
L1_CACHE_BYTES, cls);
|
|
|
}
|
|
|
+
|
|
|
+static bool __maybe_unused
|
|
|
+cpufeature_pan_not_uao(const struct arm64_cpu_capabilities *entry)
|
|
|
+{
|
|
|
+ return (cpus_have_cap(ARM64_HAS_PAN) && !cpus_have_cap(ARM64_HAS_UAO));
|
|
|
+}
|