mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-18 22:14:16 +00:00 
			
		
		
		
	Merge branches 'for-next/elf-hwcap-docs', 'for-next/smccc-conduit-cleanup', 'for-next/zone-dma', 'for-next/relax-icc_pmr_el1-sync', 'for-next/double-page-fault', 'for-next/misc', 'for-next/kselftest-arm64-signal' and 'for-next/kaslr-diagnostics' into for-next/core
* for-next/elf-hwcap-docs:
  : Update the arm64 ELF HWCAP documentation
  docs/arm64: cpu-feature-registers: Rewrite bitfields that don't follow [e, s]
  docs/arm64: cpu-feature-registers: Documents missing visible fields
  docs/arm64: elf_hwcaps: Document HWCAP_SB
  docs/arm64: elf_hwcaps: sort the HWCAP{, 2} documentation by ascending value
* for-next/smccc-conduit-cleanup:
  : SMC calling convention conduit clean-up
  firmware: arm_sdei: use common SMCCC_CONDUIT_*
  firmware/psci: use common SMCCC_CONDUIT_*
  arm: spectre-v2: use arm_smccc_1_1_get_conduit()
  arm64: errata: use arm_smccc_1_1_get_conduit()
  arm/arm64: smccc/psci: add arm_smccc_1_1_get_conduit()
* for-next/zone-dma:
  : Reintroduction of ZONE_DMA for Raspberry Pi 4 support
  arm64: mm: reserve CMA and crashkernel in ZONE_DMA32
  dma/direct: turn ARCH_ZONE_DMA_BITS into a variable
  arm64: Make arm64_dma32_phys_limit static
  arm64: mm: Fix unused variable warning in zone_sizes_init
  mm: refresh ZONE_DMA and ZONE_DMA32 comments in 'enum zone_type'
  arm64: use both ZONE_DMA and ZONE_DMA32
  arm64: rename variables used to calculate ZONE_DMA32's size
  arm64: mm: use arm64_dma_phys_limit instead of calling max_zone_dma_phys()
* for-next/relax-icc_pmr_el1-sync:
  : Relax ICC_PMR_EL1 (GICv3) accesses when ICC_CTLR_EL1.PMHE is clear
  arm64: Document ICC_CTLR_EL3.PMHE setting requirements
  arm64: Relax ICC_PMR_EL1 accesses when ICC_CTLR_EL1.PMHE is clear
* for-next/double-page-fault:
  : Avoid a double page fault in __copy_from_user_inatomic() if hw does not support auto Access Flag
  mm: fix double page fault on arm64 if PTE_AF is cleared
  x86/mm: implement arch_faults_on_old_pte() stub on x86
  arm64: mm: implement arch_faults_on_old_pte() on arm64
  arm64: cpufeature: introduce helper cpu_has_hw_af()
* for-next/misc:
  : Various fixes and clean-ups
  arm64: kpti: Add NVIDIA's Carmel core to the KPTI whitelist
  arm64: mm: Remove MAX_USER_VA_BITS definition
  arm64: mm: simplify the page end calculation in __create_pgd_mapping()
  arm64: print additional fault message when executing non-exec memory
  arm64: psci: Reduce the waiting time for cpu_psci_cpu_kill()
  arm64: pgtable: Correct typo in comment
  arm64: docs: cpu-feature-registers: Document ID_AA64PFR1_EL1
  arm64: cpufeature: Fix typos in comment
  arm64/mm: Poison initmem while freeing with free_reserved_area()
  arm64: use generic free_initrd_mem()
  arm64: simplify syscall wrapper ifdeffery
* for-next/kselftest-arm64-signal:
  : arm64-specific kselftest support with signal-related test-cases
  kselftest: arm64: fake_sigreturn_misaligned_sp
  kselftest: arm64: fake_sigreturn_bad_size
  kselftest: arm64: fake_sigreturn_duplicated_fpsimd
  kselftest: arm64: fake_sigreturn_missing_fpsimd
  kselftest: arm64: fake_sigreturn_bad_size_for_magic0
  kselftest: arm64: fake_sigreturn_bad_magic
  kselftest: arm64: add helper get_current_context
  kselftest: arm64: extend test_init functionalities
  kselftest: arm64: mangle_pstate_invalid_mode_el[123][ht]
  kselftest: arm64: mangle_pstate_invalid_daif_bits
  kselftest: arm64: mangle_pstate_invalid_compat_toggle and common utils
  kselftest: arm64: extend toplevel skeleton Makefile
* for-next/kaslr-diagnostics:
  : Provide diagnostics on boot for KASLR
  arm64: kaslr: Check command line before looking for a seed
  arm64: kaslr: Announce KASLR status on boot
			
			
This commit is contained in:
		
							parent
							
								
									51effa6d11
								
							
								
									478016c383
								
							
								
									e6ea46511b
								
							
								
									bff3b04460
								
							
								
									7e3a57fa6c
								
							
								
									83d116c530
								
							
								
									918e1946c8
								
							
								
									3f484ce375
								
							
								
									2203e1adb9
								
							
						
					
					
						commit
						6be22809e5
					
				
					 75 changed files with 2083 additions and 240 deletions
				
			
		|  | @ -213,6 +213,9 @@ Before jumping into the kernel, the following conditions must be met: | ||||||
| 
 | 
 | ||||||
|       - ICC_SRE_EL3.Enable (bit 3) must be initialiased to 0b1. |       - ICC_SRE_EL3.Enable (bit 3) must be initialiased to 0b1. | ||||||
|       - ICC_SRE_EL3.SRE (bit 0) must be initialised to 0b1. |       - ICC_SRE_EL3.SRE (bit 0) must be initialised to 0b1. | ||||||
|  |       - ICC_CTLR_EL3.PMHE (bit 6) must be set to the same value across | ||||||
|  |         all CPUs the kernel is executing on, and must stay constant | ||||||
|  |         for the lifetime of the kernel. | ||||||
| 
 | 
 | ||||||
|   - If the kernel is entered at EL1: |   - If the kernel is entered at EL1: | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -168,8 +168,15 @@ infrastructure: | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|   3) MIDR_EL1 - Main ID Register |   3) ID_AA64PFR1_EL1 - Processor Feature Register 1 | ||||||
|  |      +------------------------------+---------+---------+ | ||||||
|  |      | Name                         |  bits   | visible | | ||||||
|  |      +------------------------------+---------+---------+ | ||||||
|  |      | SSBS                         | [7-4]   |    y    | | ||||||
|  |      +------------------------------+---------+---------+ | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  |   4) MIDR_EL1 - Main ID Register | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|      | Name                         |  bits   | visible | |      | Name                         |  bits   | visible | | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|  | @ -188,11 +195,15 @@ infrastructure: | ||||||
|    as available on the CPU where it is fetched and is not a system |    as available on the CPU where it is fetched and is not a system | ||||||
|    wide safe value. |    wide safe value. | ||||||
| 
 | 
 | ||||||
|   4) ID_AA64ISAR1_EL1 - Instruction set attribute register 1 |   5) ID_AA64ISAR1_EL1 - Instruction set attribute register 1 | ||||||
| 
 | 
 | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|      | Name                         |  bits   | visible | |      | Name                         |  bits   | visible | | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|  |      | SB                           | [39-36] |    y    | | ||||||
|  |      +------------------------------+---------+---------+ | ||||||
|  |      | FRINTTS                      | [35-32] |    y    | | ||||||
|  |      +------------------------------+---------+---------+ | ||||||
|      | GPI                          | [31-28] |    y    | |      | GPI                          | [31-28] |    y    | | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|      | GPA                          | [27-24] |    y    | |      | GPA                          | [27-24] |    y    | | ||||||
|  | @ -210,7 +221,7 @@ infrastructure: | ||||||
|      | DPB                          | [3-0]   |    y    | |      | DPB                          | [3-0]   |    y    | | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
| 
 | 
 | ||||||
|   5) ID_AA64MMFR2_EL1 - Memory model feature register 2 |   6) ID_AA64MMFR2_EL1 - Memory model feature register 2 | ||||||
| 
 | 
 | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|      | Name                         |  bits   | visible | |      | Name                         |  bits   | visible | | ||||||
|  | @ -218,7 +229,7 @@ infrastructure: | ||||||
|      | AT                           | [35-32] |    y    | |      | AT                           | [35-32] |    y    | | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
| 
 | 
 | ||||||
|   6) ID_AA64ZFR0_EL1 - SVE feature ID register 0 |   7) ID_AA64ZFR0_EL1 - SVE feature ID register 0 | ||||||
| 
 | 
 | ||||||
|      +------------------------------+---------+---------+ |      +------------------------------+---------+---------+ | ||||||
|      | Name                         |  bits   | visible | |      | Name                         |  bits   | visible | | ||||||
|  |  | ||||||
|  | @ -119,10 +119,6 @@ HWCAP_LRCPC | ||||||
| HWCAP_DCPOP | HWCAP_DCPOP | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.DPB == 0b0001. |     Functionality implied by ID_AA64ISAR1_EL1.DPB == 0b0001. | ||||||
| 
 | 
 | ||||||
| HWCAP2_DCPODP |  | ||||||
| 
 |  | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.DPB == 0b0010. |  | ||||||
| 
 |  | ||||||
| HWCAP_SHA3 | HWCAP_SHA3 | ||||||
|     Functionality implied by ID_AA64ISAR0_EL1.SHA3 == 0b0001. |     Functionality implied by ID_AA64ISAR0_EL1.SHA3 == 0b0001. | ||||||
| 
 | 
 | ||||||
|  | @ -141,6 +137,41 @@ HWCAP_SHA512 | ||||||
| HWCAP_SVE | HWCAP_SVE | ||||||
|     Functionality implied by ID_AA64PFR0_EL1.SVE == 0b0001. |     Functionality implied by ID_AA64PFR0_EL1.SVE == 0b0001. | ||||||
| 
 | 
 | ||||||
|  | HWCAP_ASIMDFHM | ||||||
|  |    Functionality implied by ID_AA64ISAR0_EL1.FHM == 0b0001. | ||||||
|  | 
 | ||||||
|  | HWCAP_DIT | ||||||
|  |     Functionality implied by ID_AA64PFR0_EL1.DIT == 0b0001. | ||||||
|  | 
 | ||||||
|  | HWCAP_USCAT | ||||||
|  |     Functionality implied by ID_AA64MMFR2_EL1.AT == 0b0001. | ||||||
|  | 
 | ||||||
|  | HWCAP_ILRCPC | ||||||
|  |     Functionality implied by ID_AA64ISAR1_EL1.LRCPC == 0b0010. | ||||||
|  | 
 | ||||||
|  | HWCAP_FLAGM | ||||||
|  |     Functionality implied by ID_AA64ISAR0_EL1.TS == 0b0001. | ||||||
|  | 
 | ||||||
|  | HWCAP_SSBS | ||||||
|  |     Functionality implied by ID_AA64PFR1_EL1.SSBS == 0b0010. | ||||||
|  | 
 | ||||||
|  | HWCAP_SB | ||||||
|  |     Functionality implied by ID_AA64ISAR1_EL1.SB == 0b0001. | ||||||
|  | 
 | ||||||
|  | HWCAP_PACA | ||||||
|  |     Functionality implied by ID_AA64ISAR1_EL1.APA == 0b0001 or | ||||||
|  |     ID_AA64ISAR1_EL1.API == 0b0001, as described by | ||||||
|  |     Documentation/arm64/pointer-authentication.rst. | ||||||
|  | 
 | ||||||
|  | HWCAP_PACG | ||||||
|  |     Functionality implied by ID_AA64ISAR1_EL1.GPA == 0b0001 or | ||||||
|  |     ID_AA64ISAR1_EL1.GPI == 0b0001, as described by | ||||||
|  |     Documentation/arm64/pointer-authentication.rst. | ||||||
|  | 
 | ||||||
|  | HWCAP2_DCPODP | ||||||
|  | 
 | ||||||
|  |     Functionality implied by ID_AA64ISAR1_EL1.DPB == 0b0010. | ||||||
|  | 
 | ||||||
| HWCAP2_SVE2 | HWCAP2_SVE2 | ||||||
| 
 | 
 | ||||||
|     Functionality implied by ID_AA64ZFR0_EL1.SVEVer == 0b0001. |     Functionality implied by ID_AA64ZFR0_EL1.SVEVer == 0b0001. | ||||||
|  | @ -165,38 +196,10 @@ HWCAP2_SVESM4 | ||||||
| 
 | 
 | ||||||
|     Functionality implied by ID_AA64ZFR0_EL1.SM4 == 0b0001. |     Functionality implied by ID_AA64ZFR0_EL1.SM4 == 0b0001. | ||||||
| 
 | 
 | ||||||
| HWCAP_ASIMDFHM |  | ||||||
|    Functionality implied by ID_AA64ISAR0_EL1.FHM == 0b0001. |  | ||||||
| 
 |  | ||||||
| HWCAP_DIT |  | ||||||
|     Functionality implied by ID_AA64PFR0_EL1.DIT == 0b0001. |  | ||||||
| 
 |  | ||||||
| HWCAP_USCAT |  | ||||||
|     Functionality implied by ID_AA64MMFR2_EL1.AT == 0b0001. |  | ||||||
| 
 |  | ||||||
| HWCAP_ILRCPC |  | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.LRCPC == 0b0010. |  | ||||||
| 
 |  | ||||||
| HWCAP_FLAGM |  | ||||||
|     Functionality implied by ID_AA64ISAR0_EL1.TS == 0b0001. |  | ||||||
| 
 |  | ||||||
| HWCAP2_FLAGM2 | HWCAP2_FLAGM2 | ||||||
| 
 | 
 | ||||||
|     Functionality implied by ID_AA64ISAR0_EL1.TS == 0b0010. |     Functionality implied by ID_AA64ISAR0_EL1.TS == 0b0010. | ||||||
| 
 | 
 | ||||||
| HWCAP_SSBS |  | ||||||
|     Functionality implied by ID_AA64PFR1_EL1.SSBS == 0b0010. |  | ||||||
| 
 |  | ||||||
| HWCAP_PACA |  | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.APA == 0b0001 or |  | ||||||
|     ID_AA64ISAR1_EL1.API == 0b0001, as described by |  | ||||||
|     Documentation/arm64/pointer-authentication.rst. |  | ||||||
| 
 |  | ||||||
| HWCAP_PACG |  | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.GPA == 0b0001 or |  | ||||||
|     ID_AA64ISAR1_EL1.GPI == 0b0001, as described by |  | ||||||
|     Documentation/arm64/pointer-authentication.rst. |  | ||||||
| 
 |  | ||||||
| HWCAP2_FRINT | HWCAP2_FRINT | ||||||
| 
 | 
 | ||||||
|     Functionality implied by ID_AA64ISAR1_EL1.FRINTTS == 0b0001. |     Functionality implied by ID_AA64ISAR1_EL1.FRINTTS == 0b0001. | ||||||
|  |  | ||||||
|  | @ -1,7 +1,6 @@ | ||||||
| // SPDX-License-Identifier: GPL-2.0
 | // SPDX-License-Identifier: GPL-2.0
 | ||||||
| #include <linux/arm-smccc.h> | #include <linux/arm-smccc.h> | ||||||
| #include <linux/kernel.h> | #include <linux/kernel.h> | ||||||
| #include <linux/psci.h> |  | ||||||
| #include <linux/smp.h> | #include <linux/smp.h> | ||||||
| 
 | 
 | ||||||
| #include <asm/cp15.h> | #include <asm/cp15.h> | ||||||
|  | @ -75,11 +74,8 @@ static void cpu_v7_spectre_init(void) | ||||||
| 	case ARM_CPU_PART_CORTEX_A72: { | 	case ARM_CPU_PART_CORTEX_A72: { | ||||||
| 		struct arm_smccc_res res; | 		struct arm_smccc_res res; | ||||||
| 
 | 
 | ||||||
| 		if (psci_ops.smccc_version == SMCCC_VERSION_1_0) | 		switch (arm_smccc_1_1_get_conduit()) { | ||||||
| 			break; | 		case SMCCC_CONDUIT_HVC: | ||||||
| 
 |  | ||||||
| 		switch (psci_ops.conduit) { |  | ||||||
| 		case PSCI_CONDUIT_HVC: |  | ||||||
| 			arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 			arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 					  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | 					  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | ||||||
| 			if ((int)res.a0 != 0) | 			if ((int)res.a0 != 0) | ||||||
|  | @ -90,7 +86,7 @@ static void cpu_v7_spectre_init(void) | ||||||
| 			spectre_v2_method = "hypervisor"; | 			spectre_v2_method = "hypervisor"; | ||||||
| 			break; | 			break; | ||||||
| 
 | 
 | ||||||
| 		case PSCI_CONDUIT_SMC: | 		case SMCCC_CONDUIT_SMC: | ||||||
| 			arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 			arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 					  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | 					  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | ||||||
| 			if ((int)res.a0 != 0) | 			if ((int)res.a0 != 0) | ||||||
|  |  | ||||||
|  | @ -268,6 +268,10 @@ config GENERIC_CSUM | ||||||
| config GENERIC_CALIBRATE_DELAY | config GENERIC_CALIBRATE_DELAY | ||||||
| 	def_bool y | 	def_bool y | ||||||
| 
 | 
 | ||||||
|  | config ZONE_DMA | ||||||
|  | 	bool "Support DMA zone" if EXPERT | ||||||
|  | 	default y | ||||||
|  | 
 | ||||||
| config ZONE_DMA32 | config ZONE_DMA32 | ||||||
| 	bool "Support DMA32 zone" if EXPERT | 	bool "Support DMA32 zone" if EXPERT | ||||||
| 	default y | 	default y | ||||||
|  |  | ||||||
|  | @ -29,6 +29,18 @@ | ||||||
| 						 SB_BARRIER_INSN"nop\n",	\ | 						 SB_BARRIER_INSN"nop\n",	\ | ||||||
| 						 ARM64_HAS_SB)) | 						 ARM64_HAS_SB)) | ||||||
| 
 | 
 | ||||||
|  | #ifdef CONFIG_ARM64_PSEUDO_NMI | ||||||
|  | #define pmr_sync()						\ | ||||||
|  | 	do {							\ | ||||||
|  | 		extern struct static_key_false gic_pmr_sync;	\ | ||||||
|  | 								\ | ||||||
|  | 		if (static_branch_unlikely(&gic_pmr_sync))	\ | ||||||
|  | 			dsb(sy);				\ | ||||||
|  | 	} while(0) | ||||||
|  | #else | ||||||
|  | #define pmr_sync()	do {} while (0) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
| #define mb()		dsb(sy) | #define mb()		dsb(sy) | ||||||
| #define rmb()		dsb(ld) | #define rmb()		dsb(ld) | ||||||
| #define wmb()		dsb(st) | #define wmb()		dsb(st) | ||||||
|  |  | ||||||
|  | @ -659,6 +659,20 @@ static inline u32 id_aa64mmfr0_parange_to_phys_shift(int parange) | ||||||
| 	default: return CONFIG_ARM64_PA_BITS; | 	default: return CONFIG_ARM64_PA_BITS; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | /* Check whether hardware update of the Access flag is supported */ | ||||||
|  | static inline bool cpu_has_hw_af(void) | ||||||
|  | { | ||||||
|  | 	u64 mmfr1; | ||||||
|  | 
 | ||||||
|  | 	if (!IS_ENABLED(CONFIG_ARM64_HW_AFDBM)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	mmfr1 = read_cpuid(ID_AA64MMFR1_EL1); | ||||||
|  | 	return cpuid_feature_extract_unsigned_field(mmfr1, | ||||||
|  | 						ID_AA64MMFR1_HADBS_SHIFT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #endif /* __ASSEMBLY__ */ | #endif /* __ASSEMBLY__ */ | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -8,6 +8,7 @@ | ||||||
| #include <linux/irqflags.h> | #include <linux/irqflags.h> | ||||||
| 
 | 
 | ||||||
| #include <asm/arch_gicv3.h> | #include <asm/arch_gicv3.h> | ||||||
|  | #include <asm/barrier.h> | ||||||
| #include <asm/cpufeature.h> | #include <asm/cpufeature.h> | ||||||
| #include <asm/ptrace.h> | #include <asm/ptrace.h> | ||||||
| 
 | 
 | ||||||
|  | @ -66,7 +67,7 @@ static inline void local_daif_restore(unsigned long flags) | ||||||
| 
 | 
 | ||||||
| 		if (system_uses_irq_prio_masking()) { | 		if (system_uses_irq_prio_masking()) { | ||||||
| 			gic_write_pmr(GIC_PRIO_IRQON); | 			gic_write_pmr(GIC_PRIO_IRQON); | ||||||
| 			dsb(sy); | 			pmr_sync(); | ||||||
| 		} | 		} | ||||||
| 	} else if (system_uses_irq_prio_masking()) { | 	} else if (system_uses_irq_prio_masking()) { | ||||||
| 		u64 pmr; | 		u64 pmr; | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
| #define __ASM_IRQFLAGS_H | #define __ASM_IRQFLAGS_H | ||||||
| 
 | 
 | ||||||
| #include <asm/alternative.h> | #include <asm/alternative.h> | ||||||
|  | #include <asm/barrier.h> | ||||||
| #include <asm/ptrace.h> | #include <asm/ptrace.h> | ||||||
| #include <asm/sysreg.h> | #include <asm/sysreg.h> | ||||||
| 
 | 
 | ||||||
|  | @ -34,14 +35,14 @@ static inline void arch_local_irq_enable(void) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	asm volatile(ALTERNATIVE( | 	asm volatile(ALTERNATIVE( | ||||||
| 		"msr	daifclr, #2		// arch_local_irq_enable\n" | 		"msr	daifclr, #2		// arch_local_irq_enable", | ||||||
| 		"nop", | 		__msr_s(SYS_ICC_PMR_EL1, "%0"), | ||||||
| 		__msr_s(SYS_ICC_PMR_EL1, "%0") |  | ||||||
| 		"dsb	sy", |  | ||||||
| 		ARM64_HAS_IRQ_PRIO_MASKING) | 		ARM64_HAS_IRQ_PRIO_MASKING) | ||||||
| 		: | 		: | ||||||
| 		: "r" ((unsigned long) GIC_PRIO_IRQON) | 		: "r" ((unsigned long) GIC_PRIO_IRQON) | ||||||
| 		: "memory"); | 		: "memory"); | ||||||
|  | 
 | ||||||
|  | 	pmr_sync(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline void arch_local_irq_disable(void) | static inline void arch_local_irq_disable(void) | ||||||
|  | @ -116,14 +117,14 @@ static inline unsigned long arch_local_irq_save(void) | ||||||
| static inline void arch_local_irq_restore(unsigned long flags) | static inline void arch_local_irq_restore(unsigned long flags) | ||||||
| { | { | ||||||
| 	asm volatile(ALTERNATIVE( | 	asm volatile(ALTERNATIVE( | ||||||
| 			"msr	daif, %0\n" | 		"msr	daif, %0", | ||||||
| 			"nop", | 		__msr_s(SYS_ICC_PMR_EL1, "%0"), | ||||||
| 			__msr_s(SYS_ICC_PMR_EL1, "%0") | 		ARM64_HAS_IRQ_PRIO_MASKING) | ||||||
| 			"dsb	sy", |  | ||||||
| 			ARM64_HAS_IRQ_PRIO_MASKING) |  | ||||||
| 		: | 		: | ||||||
| 		: "r" (flags) | 		: "r" (flags) | ||||||
| 		: "memory"); | 		: "memory"); | ||||||
|  | 
 | ||||||
|  | 	pmr_sync(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #endif /* __ASM_IRQFLAGS_H */ | #endif /* __ASM_IRQFLAGS_H */ | ||||||
|  |  | ||||||
|  | @ -600,8 +600,7 @@ static inline void kvm_arm_vhe_guest_enter(void) | ||||||
| 	 * local_daif_mask() already sets GIC_PRIO_PSR_I_SET, we just need a | 	 * local_daif_mask() already sets GIC_PRIO_PSR_I_SET, we just need a | ||||||
| 	 * dsb to ensure the redistributor is forwards EL2 IRQs to the CPU. | 	 * dsb to ensure the redistributor is forwards EL2 IRQs to the CPU. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (system_uses_irq_prio_masking()) | 	pmr_sync(); | ||||||
| 		dsb(sy); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline void kvm_arm_vhe_guest_exit(void) | static inline void kvm_arm_vhe_guest_exit(void) | ||||||
|  |  | ||||||
|  | @ -69,12 +69,6 @@ | ||||||
| #define KERNEL_START		_text | #define KERNEL_START		_text | ||||||
| #define KERNEL_END		_end | #define KERNEL_END		_end | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_ARM64_VA_BITS_52 |  | ||||||
| #define MAX_USER_VA_BITS	52 |  | ||||||
| #else |  | ||||||
| #define MAX_USER_VA_BITS	VA_BITS |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| /*
 | /*
 | ||||||
|  * Generic and tag-based KASAN require 1/8th and 1/16th of the kernel virtual |  * Generic and tag-based KASAN require 1/8th and 1/16th of the kernel virtual | ||||||
|  * address space for the shadow region respectively. They can bloat the stack |  * address space for the shadow region respectively. They can bloat the stack | ||||||
|  |  | ||||||
|  | @ -69,7 +69,7 @@ | ||||||
| #define PGDIR_SHIFT		ARM64_HW_PGTABLE_LEVEL_SHIFT(4 - CONFIG_PGTABLE_LEVELS) | #define PGDIR_SHIFT		ARM64_HW_PGTABLE_LEVEL_SHIFT(4 - CONFIG_PGTABLE_LEVELS) | ||||||
| #define PGDIR_SIZE		(_AC(1, UL) << PGDIR_SHIFT) | #define PGDIR_SIZE		(_AC(1, UL) << PGDIR_SHIFT) | ||||||
| #define PGDIR_MASK		(~(PGDIR_SIZE-1)) | #define PGDIR_MASK		(~(PGDIR_SIZE-1)) | ||||||
| #define PTRS_PER_PGD		(1 << (MAX_USER_VA_BITS - PGDIR_SHIFT)) | #define PTRS_PER_PGD		(1 << (VA_BITS - PGDIR_SHIFT)) | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Section address mask and size definitions. |  * Section address mask and size definitions. | ||||||
|  |  | ||||||
|  | @ -17,7 +17,7 @@ | ||||||
|  * VMALLOC range. |  * VMALLOC range. | ||||||
|  * |  * | ||||||
|  * VMALLOC_START: beginning of the kernel vmalloc space |  * VMALLOC_START: beginning of the kernel vmalloc space | ||||||
|  * VMALLOC_END: extends to the available space below vmmemmap, PCI I/O space |  * VMALLOC_END: extends to the available space below vmemmap, PCI I/O space | ||||||
|  *	and fixed mappings |  *	and fixed mappings | ||||||
|  */ |  */ | ||||||
| #define VMALLOC_START		(MODULES_END) | #define VMALLOC_START		(MODULES_END) | ||||||
|  | @ -882,6 +882,20 @@ static inline void update_mmu_cache(struct vm_area_struct *vma, | ||||||
| #define phys_to_ttbr(addr)	(addr) | #define phys_to_ttbr(addr)	(addr) | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * On arm64 without hardware Access Flag, copying from user will fail because | ||||||
|  |  * the pte is old and cannot be marked young. So we always end up with zeroed | ||||||
|  |  * page after fork() + CoW for pfn mappings. We don't always have a | ||||||
|  |  * hardware-managed access flag on arm64. | ||||||
|  |  */ | ||||||
|  | static inline bool arch_faults_on_old_pte(void) | ||||||
|  | { | ||||||
|  | 	WARN_ON(preemptible()); | ||||||
|  | 
 | ||||||
|  | 	return !cpu_has_hw_af(); | ||||||
|  | } | ||||||
|  | #define arch_faults_on_old_pte arch_faults_on_old_pte | ||||||
|  | 
 | ||||||
| #endif /* !__ASSEMBLY__ */ | #endif /* !__ASSEMBLY__ */ | ||||||
| 
 | 
 | ||||||
| #endif /* __ASM_PGTABLE_H */ | #endif /* __ASM_PGTABLE_H */ | ||||||
|  |  | ||||||
|  | @ -9,7 +9,7 @@ | ||||||
| #define __ASM_PROCESSOR_H | #define __ASM_PROCESSOR_H | ||||||
| 
 | 
 | ||||||
| #define KERNEL_DS		UL(-1) | #define KERNEL_DS		UL(-1) | ||||||
| #define USER_DS			((UL(1) << MAX_USER_VA_BITS) - 1) | #define USER_DS			((UL(1) << VA_BITS) - 1) | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * On arm64 systems, unaligned accesses by the CPU are cheap, and so there is |  * On arm64 systems, unaligned accesses by the CPU are cheap, and so there is | ||||||
|  |  | ||||||
|  | @ -66,24 +66,18 @@ struct pt_regs; | ||||||
| 	}									\ | 	}									\ | ||||||
| 	static inline long __do_sys##name(__MAP(x,__SC_DECL,__VA_ARGS__)) | 	static inline long __do_sys##name(__MAP(x,__SC_DECL,__VA_ARGS__)) | ||||||
| 
 | 
 | ||||||
| #ifndef SYSCALL_DEFINE0 |  | ||||||
| #define SYSCALL_DEFINE0(sname)							\ | #define SYSCALL_DEFINE0(sname)							\ | ||||||
| 	SYSCALL_METADATA(_##sname, 0);						\ | 	SYSCALL_METADATA(_##sname, 0);						\ | ||||||
| 	asmlinkage long __arm64_sys_##sname(const struct pt_regs *__unused);	\ | 	asmlinkage long __arm64_sys_##sname(const struct pt_regs *__unused);	\ | ||||||
| 	ALLOW_ERROR_INJECTION(__arm64_sys_##sname, ERRNO);			\ | 	ALLOW_ERROR_INJECTION(__arm64_sys_##sname, ERRNO);			\ | ||||||
| 	asmlinkage long __arm64_sys_##sname(const struct pt_regs *__unused) | 	asmlinkage long __arm64_sys_##sname(const struct pt_regs *__unused) | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| #ifndef COND_SYSCALL |  | ||||||
| #define COND_SYSCALL(name)							\ | #define COND_SYSCALL(name)							\ | ||||||
| 	asmlinkage long __weak __arm64_sys_##name(const struct pt_regs *regs)	\ | 	asmlinkage long __weak __arm64_sys_##name(const struct pt_regs *regs)	\ | ||||||
| 	{									\ | 	{									\ | ||||||
| 		return sys_ni_syscall();					\ | 		return sys_ni_syscall();					\ | ||||||
| 	} | 	} | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| #ifndef SYS_NI |  | ||||||
| #define SYS_NI(name) SYSCALL_ALIAS(__arm64_sys_##name, sys_ni_posix_timers); | #define SYS_NI(name) SYSCALL_ALIAS(__arm64_sys_##name, sys_ni_posix_timers); | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| #endif /* __ASM_SYSCALL_WRAPPER_H */ | #endif /* __ASM_SYSCALL_WRAPPER_H */ | ||||||
|  |  | ||||||
|  | @ -6,7 +6,6 @@ | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #include <linux/arm-smccc.h> | #include <linux/arm-smccc.h> | ||||||
| #include <linux/psci.h> |  | ||||||
| #include <linux/types.h> | #include <linux/types.h> | ||||||
| #include <linux/cpu.h> | #include <linux/cpu.h> | ||||||
| #include <asm/cpu.h> | #include <asm/cpu.h> | ||||||
|  | @ -175,9 +174,7 @@ static void install_bp_hardening_cb(bp_hardening_cb_t fn, | ||||||
| } | } | ||||||
| #endif	/* CONFIG_KVM_INDIRECT_VECTORS */ | #endif	/* CONFIG_KVM_INDIRECT_VECTORS */ | ||||||
| 
 | 
 | ||||||
| #include <uapi/linux/psci.h> |  | ||||||
| #include <linux/arm-smccc.h> | #include <linux/arm-smccc.h> | ||||||
| #include <linux/psci.h> |  | ||||||
| 
 | 
 | ||||||
| static void call_smc_arch_workaround_1(void) | static void call_smc_arch_workaround_1(void) | ||||||
| { | { | ||||||
|  | @ -221,11 +218,8 @@ static int detect_harden_bp_fw(void) | ||||||
| 	struct arm_smccc_res res; | 	struct arm_smccc_res res; | ||||||
| 	u32 midr = read_cpuid_id(); | 	u32 midr = read_cpuid_id(); | ||||||
| 
 | 
 | ||||||
| 	if (psci_ops.smccc_version == SMCCC_VERSION_1_0) | 	switch (arm_smccc_1_1_get_conduit()) { | ||||||
| 		return -1; | 	case SMCCC_CONDUIT_HVC: | ||||||
| 
 |  | ||||||
| 	switch (psci_ops.conduit) { |  | ||||||
| 	case PSCI_CONDUIT_HVC: |  | ||||||
| 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 				  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | 				  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | ||||||
| 		switch ((int)res.a0) { | 		switch ((int)res.a0) { | ||||||
|  | @ -243,7 +237,7 @@ static int detect_harden_bp_fw(void) | ||||||
| 		} | 		} | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
| 	case PSCI_CONDUIT_SMC: | 	case SMCCC_CONDUIT_SMC: | ||||||
| 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 				  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | 				  ARM_SMCCC_ARCH_WORKAROUND_1, &res); | ||||||
| 		switch ((int)res.a0) { | 		switch ((int)res.a0) { | ||||||
|  | @ -317,11 +311,11 @@ void __init arm64_update_smccc_conduit(struct alt_instr *alt, | ||||||
| 
 | 
 | ||||||
| 	BUG_ON(nr_inst != 1); | 	BUG_ON(nr_inst != 1); | ||||||
| 
 | 
 | ||||||
| 	switch (psci_ops.conduit) { | 	switch (arm_smccc_1_1_get_conduit()) { | ||||||
| 	case PSCI_CONDUIT_HVC: | 	case SMCCC_CONDUIT_HVC: | ||||||
| 		insn = aarch64_insn_get_hvc_value(); | 		insn = aarch64_insn_get_hvc_value(); | ||||||
| 		break; | 		break; | ||||||
| 	case PSCI_CONDUIT_SMC: | 	case SMCCC_CONDUIT_SMC: | ||||||
| 		insn = aarch64_insn_get_smc_value(); | 		insn = aarch64_insn_get_smc_value(); | ||||||
| 		break; | 		break; | ||||||
| 	default: | 	default: | ||||||
|  | @ -360,12 +354,12 @@ void arm64_set_ssbd_mitigation(bool state) | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	switch (psci_ops.conduit) { | 	switch (arm_smccc_1_1_get_conduit()) { | ||||||
| 	case PSCI_CONDUIT_HVC: | 	case SMCCC_CONDUIT_HVC: | ||||||
| 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_WORKAROUND_2, state, NULL); | 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_WORKAROUND_2, state, NULL); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
| 	case PSCI_CONDUIT_SMC: | 	case SMCCC_CONDUIT_SMC: | ||||||
| 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_WORKAROUND_2, state, NULL); | 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_WORKAROUND_2, state, NULL); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
|  | @ -399,20 +393,13 @@ static bool has_ssbd_mitigation(const struct arm64_cpu_capabilities *entry, | ||||||
| 		goto out_printmsg; | 		goto out_printmsg; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (psci_ops.smccc_version == SMCCC_VERSION_1_0) { | 	switch (arm_smccc_1_1_get_conduit()) { | ||||||
| 		ssbd_state = ARM64_SSBD_UNKNOWN; | 	case SMCCC_CONDUIT_HVC: | ||||||
| 		if (!this_cpu_safe) |  | ||||||
| 			__ssb_safe = false; |  | ||||||
| 		return false; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	switch (psci_ops.conduit) { |  | ||||||
| 	case PSCI_CONDUIT_HVC: |  | ||||||
| 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 		arm_smccc_1_1_hvc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 				  ARM_SMCCC_ARCH_WORKAROUND_2, &res); | 				  ARM_SMCCC_ARCH_WORKAROUND_2, &res); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
| 	case PSCI_CONDUIT_SMC: | 	case SMCCC_CONDUIT_SMC: | ||||||
| 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | 		arm_smccc_1_1_smc(ARM_SMCCC_ARCH_FEATURES_FUNC_ID, | ||||||
| 				  ARM_SMCCC_ARCH_WORKAROUND_2, &res); | 				  ARM_SMCCC_ARCH_WORKAROUND_2, &res); | ||||||
| 		break; | 		break; | ||||||
|  |  | ||||||
|  | @ -982,6 +982,7 @@ static bool unmap_kernel_at_el0(const struct arm64_cpu_capabilities *entry, | ||||||
| 		MIDR_ALL_VERSIONS(MIDR_CORTEX_A72), | 		MIDR_ALL_VERSIONS(MIDR_CORTEX_A72), | ||||||
| 		MIDR_ALL_VERSIONS(MIDR_CORTEX_A73), | 		MIDR_ALL_VERSIONS(MIDR_CORTEX_A73), | ||||||
| 		MIDR_ALL_VERSIONS(MIDR_HISI_TSV110), | 		MIDR_ALL_VERSIONS(MIDR_HISI_TSV110), | ||||||
|  | 		MIDR_ALL_VERSIONS(MIDR_NVIDIA_CARMEL), | ||||||
| 		{ /* sentinel */ } | 		{ /* sentinel */ } | ||||||
| 	}; | 	}; | ||||||
| 	char const *str = "kpti command line option"; | 	char const *str = "kpti command line option"; | ||||||
|  |  | ||||||
|  | @ -329,7 +329,7 @@ static void __cpuinfo_store_cpu(struct cpuinfo_arm64 *info) | ||||||
| 	info->reg_cntfrq = arch_timer_get_cntfrq(); | 	info->reg_cntfrq = arch_timer_get_cntfrq(); | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Use the effective value of the CTR_EL0 than the raw value | 	 * Use the effective value of the CTR_EL0 than the raw value | ||||||
| 	 * exposed by the CPU. CTR_E0.IDC field value must be interpreted | 	 * exposed by the CPU. CTR_EL0.IDC field value must be interpreted | ||||||
| 	 * with the CLIDR_EL1 fields to avoid triggering false warnings | 	 * with the CLIDR_EL1 fields to avoid triggering false warnings | ||||||
| 	 * when there is a mismatch across the CPUs. Keep track of the | 	 * when there is a mismatch across the CPUs. Keep track of the | ||||||
| 	 * effective value of the CTR_EL0 in our internal records for | 	 * effective value of the CTR_EL0 in our internal records for | ||||||
|  |  | ||||||
|  | @ -269,8 +269,10 @@ alternative_else_nop_endif | ||||||
| alternative_if ARM64_HAS_IRQ_PRIO_MASKING | alternative_if ARM64_HAS_IRQ_PRIO_MASKING | ||||||
| 	ldr	x20, [sp, #S_PMR_SAVE] | 	ldr	x20, [sp, #S_PMR_SAVE] | ||||||
| 	msr_s	SYS_ICC_PMR_EL1, x20 | 	msr_s	SYS_ICC_PMR_EL1, x20 | ||||||
| 	/* Ensure priority change is seen by redistributor */ | 	mrs_s	x21, SYS_ICC_CTLR_EL1 | ||||||
| 	dsb	sy | 	tbz	x21, #6, .L__skip_pmr_sync\@	// Check for ICC_CTLR_EL1.PMHE
 | ||||||
|  | 	dsb	sy				// Ensure priority change is seen by redistributor | ||||||
|  | .L__skip_pmr_sync\@:
 | ||||||
| alternative_else_nop_endif | alternative_else_nop_endif | ||||||
| 
 | 
 | ||||||
| 	ldp	x21, x22, [sp, #S_PC]		// load ELR, SPSR | 	ldp	x21, x22, [sp, #S_PC]		// load ELR, SPSR | ||||||
|  |  | ||||||
|  | @ -19,6 +19,14 @@ | ||||||
| #include <asm/pgtable.h> | #include <asm/pgtable.h> | ||||||
| #include <asm/sections.h> | #include <asm/sections.h> | ||||||
| 
 | 
 | ||||||
|  | enum kaslr_status { | ||||||
|  | 	KASLR_ENABLED, | ||||||
|  | 	KASLR_DISABLED_CMDLINE, | ||||||
|  | 	KASLR_DISABLED_NO_SEED, | ||||||
|  | 	KASLR_DISABLED_FDT_REMAP, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static enum kaslr_status __initdata kaslr_status; | ||||||
| u64 __ro_after_init module_alloc_base; | u64 __ro_after_init module_alloc_base; | ||||||
| u16 __initdata memstart_offset_seed; | u16 __initdata memstart_offset_seed; | ||||||
| 
 | 
 | ||||||
|  | @ -91,15 +99,15 @@ u64 __init kaslr_early_init(u64 dt_phys) | ||||||
| 	 */ | 	 */ | ||||||
| 	early_fixmap_init(); | 	early_fixmap_init(); | ||||||
| 	fdt = fixmap_remap_fdt(dt_phys, &size, PAGE_KERNEL); | 	fdt = fixmap_remap_fdt(dt_phys, &size, PAGE_KERNEL); | ||||||
| 	if (!fdt) | 	if (!fdt) { | ||||||
|  | 		kaslr_status = KASLR_DISABLED_FDT_REMAP; | ||||||
| 		return 0; | 		return 0; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Retrieve (and wipe) the seed from the FDT | 	 * Retrieve (and wipe) the seed from the FDT | ||||||
| 	 */ | 	 */ | ||||||
| 	seed = get_kaslr_seed(fdt); | 	seed = get_kaslr_seed(fdt); | ||||||
| 	if (!seed) |  | ||||||
| 		return 0; |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Check if 'nokaslr' appears on the command line, and | 	 * Check if 'nokaslr' appears on the command line, and | ||||||
|  | @ -107,8 +115,15 @@ u64 __init kaslr_early_init(u64 dt_phys) | ||||||
| 	 */ | 	 */ | ||||||
| 	cmdline = kaslr_get_cmdline(fdt); | 	cmdline = kaslr_get_cmdline(fdt); | ||||||
| 	str = strstr(cmdline, "nokaslr"); | 	str = strstr(cmdline, "nokaslr"); | ||||||
| 	if (str == cmdline || (str > cmdline && *(str - 1) == ' ')) | 	if (str == cmdline || (str > cmdline && *(str - 1) == ' ')) { | ||||||
|  | 		kaslr_status = KASLR_DISABLED_CMDLINE; | ||||||
| 		return 0; | 		return 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!seed) { | ||||||
|  | 		kaslr_status = KASLR_DISABLED_NO_SEED; | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * OK, so we are proceeding with KASLR enabled. Calculate a suitable | 	 * OK, so we are proceeding with KASLR enabled. Calculate a suitable | ||||||
|  | @ -170,3 +185,24 @@ u64 __init kaslr_early_init(u64 dt_phys) | ||||||
| 
 | 
 | ||||||
| 	return offset; | 	return offset; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | static int __init kaslr_init(void) | ||||||
|  | { | ||||||
|  | 	switch (kaslr_status) { | ||||||
|  | 	case KASLR_ENABLED: | ||||||
|  | 		pr_info("KASLR enabled\n"); | ||||||
|  | 		break; | ||||||
|  | 	case KASLR_DISABLED_CMDLINE: | ||||||
|  | 		pr_info("KASLR disabled on command line\n"); | ||||||
|  | 		break; | ||||||
|  | 	case KASLR_DISABLED_NO_SEED: | ||||||
|  | 		pr_warn("KASLR disabled due to lack of seed\n"); | ||||||
|  | 		break; | ||||||
|  | 	case KASLR_DISABLED_FDT_REMAP: | ||||||
|  | 		pr_warn("KASLR disabled due to FDT remapping failure\n"); | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | core_initcall(kaslr_init) | ||||||
|  |  | ||||||
|  | @ -81,7 +81,8 @@ static void cpu_psci_cpu_die(unsigned int cpu) | ||||||
| 
 | 
 | ||||||
| static int cpu_psci_cpu_kill(unsigned int cpu) | static int cpu_psci_cpu_kill(unsigned int cpu) | ||||||
| { | { | ||||||
| 	int err, i; | 	int err; | ||||||
|  | 	unsigned long start, end; | ||||||
| 
 | 
 | ||||||
| 	if (!psci_ops.affinity_info) | 	if (!psci_ops.affinity_info) | ||||||
| 		return 0; | 		return 0; | ||||||
|  | @ -91,16 +92,18 @@ static int cpu_psci_cpu_kill(unsigned int cpu) | ||||||
| 	 * while it is dying. So, try again a few times. | 	 * while it is dying. So, try again a few times. | ||||||
| 	 */ | 	 */ | ||||||
| 
 | 
 | ||||||
| 	for (i = 0; i < 10; i++) { | 	start = jiffies; | ||||||
|  | 	end = start + msecs_to_jiffies(100); | ||||||
|  | 	do { | ||||||
| 		err = psci_ops.affinity_info(cpu_logical_map(cpu), 0); | 		err = psci_ops.affinity_info(cpu_logical_map(cpu), 0); | ||||||
| 		if (err == PSCI_0_2_AFFINITY_LEVEL_OFF) { | 		if (err == PSCI_0_2_AFFINITY_LEVEL_OFF) { | ||||||
| 			pr_info("CPU%d killed.\n", cpu); | 			pr_info("CPU%d killed (polled %d ms)\n", cpu, | ||||||
|  | 				jiffies_to_msecs(jiffies - start)); | ||||||
| 			return 0; | 			return 0; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		msleep(10); | 		usleep_range(100, 1000); | ||||||
| 		pr_info("Retrying again to check for CPU kill\n"); | 	} while (time_before(jiffies, end)); | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	pr_warn("CPU%d may not have shut down cleanly (AFFINITY_INFO reports %d)\n", | 	pr_warn("CPU%d may not have shut down cleanly (AFFINITY_INFO reports %d)\n", | ||||||
| 			cpu, err); | 			cpu, err); | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ | ||||||
| // Copyright (C) 2017 Arm Ltd.
 | // Copyright (C) 2017 Arm Ltd.
 | ||||||
| #define pr_fmt(fmt) "sdei: " fmt | #define pr_fmt(fmt) "sdei: " fmt | ||||||
| 
 | 
 | ||||||
|  | #include <linux/arm-smccc.h> | ||||||
| #include <linux/arm_sdei.h> | #include <linux/arm_sdei.h> | ||||||
| #include <linux/hardirq.h> | #include <linux/hardirq.h> | ||||||
| #include <linux/irqflags.h> | #include <linux/irqflags.h> | ||||||
|  | @ -161,7 +162,7 @@ unsigned long sdei_arch_get_entry_point(int conduit) | ||||||
| 			return 0; | 			return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	sdei_exit_mode = (conduit == CONDUIT_HVC) ? SDEI_EXIT_HVC : SDEI_EXIT_SMC; | 	sdei_exit_mode = (conduit == SMCCC_CONDUIT_HVC) ? SDEI_EXIT_HVC : SDEI_EXIT_SMC; | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_UNMAP_KERNEL_AT_EL0 | #ifdef CONFIG_UNMAP_KERNEL_AT_EL0 | ||||||
| 	if (arm64_kernel_unmapped_at_el0()) { | 	if (arm64_kernel_unmapped_at_el0()) { | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ | ||||||
| 
 | 
 | ||||||
| #include <kvm/arm_psci.h> | #include <kvm/arm_psci.h> | ||||||
| 
 | 
 | ||||||
| #include <asm/arch_gicv3.h> | #include <asm/barrier.h> | ||||||
| #include <asm/cpufeature.h> | #include <asm/cpufeature.h> | ||||||
| #include <asm/kprobes.h> | #include <asm/kprobes.h> | ||||||
| #include <asm/kvm_asm.h> | #include <asm/kvm_asm.h> | ||||||
|  | @ -688,7 +688,7 @@ int __hyp_text __kvm_vcpu_run_nvhe(struct kvm_vcpu *vcpu) | ||||||
| 	 */ | 	 */ | ||||||
| 	if (system_uses_irq_prio_masking()) { | 	if (system_uses_irq_prio_masking()) { | ||||||
| 		gic_write_pmr(GIC_PRIO_IRQON | GIC_PRIO_PSR_I_SET); | 		gic_write_pmr(GIC_PRIO_IRQON | GIC_PRIO_PSR_I_SET); | ||||||
| 		dsb(sy); | 		pmr_sync(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	vcpu = kern_hyp_va(vcpu); | 	vcpu = kern_hyp_va(vcpu); | ||||||
|  |  | ||||||
|  | @ -307,6 +307,8 @@ static void __do_kernel_fault(unsigned long addr, unsigned int esr, | ||||||
| 	if (is_el1_permission_fault(addr, esr, regs)) { | 	if (is_el1_permission_fault(addr, esr, regs)) { | ||||||
| 		if (esr & ESR_ELx_WNR) | 		if (esr & ESR_ELx_WNR) | ||||||
| 			msg = "write to read-only memory"; | 			msg = "write to read-only memory"; | ||||||
|  | 		else if (is_el1_instruction_abort(esr)) | ||||||
|  | 			msg = "execute from non-executable memory"; | ||||||
| 		else | 		else | ||||||
| 			msg = "read from unreadable memory"; | 			msg = "read from unreadable memory"; | ||||||
| 	} else if (addr < PAGE_SIZE) { | 	} else if (addr < PAGE_SIZE) { | ||||||
|  |  | ||||||
|  | @ -20,6 +20,7 @@ | ||||||
| #include <linux/sort.h> | #include <linux/sort.h> | ||||||
| #include <linux/of.h> | #include <linux/of.h> | ||||||
| #include <linux/of_fdt.h> | #include <linux/of_fdt.h> | ||||||
|  | #include <linux/dma-direct.h> | ||||||
| #include <linux/dma-mapping.h> | #include <linux/dma-mapping.h> | ||||||
| #include <linux/dma-contiguous.h> | #include <linux/dma-contiguous.h> | ||||||
| #include <linux/efi.h> | #include <linux/efi.h> | ||||||
|  | @ -41,6 +42,8 @@ | ||||||
| #include <asm/tlb.h> | #include <asm/tlb.h> | ||||||
| #include <asm/alternative.h> | #include <asm/alternative.h> | ||||||
| 
 | 
 | ||||||
|  | #define ARM64_ZONE_DMA_BITS	30 | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * We need to be able to catch inadvertent references to memstart_addr |  * We need to be able to catch inadvertent references to memstart_addr | ||||||
|  * that occur (potentially in generic code) before arm64_memblock_init() |  * that occur (potentially in generic code) before arm64_memblock_init() | ||||||
|  | @ -56,7 +59,14 @@ EXPORT_SYMBOL(physvirt_offset); | ||||||
| struct page *vmemmap __ro_after_init; | struct page *vmemmap __ro_after_init; | ||||||
| EXPORT_SYMBOL(vmemmap); | EXPORT_SYMBOL(vmemmap); | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * We create both ZONE_DMA and ZONE_DMA32. ZONE_DMA covers the first 1G of | ||||||
|  |  * memory as some devices, namely the Raspberry Pi 4, have peripherals with | ||||||
|  |  * this limited view of the memory. ZONE_DMA32 will cover the rest of the 32 | ||||||
|  |  * bit addressable memory area. | ||||||
|  |  */ | ||||||
| phys_addr_t arm64_dma_phys_limit __ro_after_init; | phys_addr_t arm64_dma_phys_limit __ro_after_init; | ||||||
|  | static phys_addr_t arm64_dma32_phys_limit __ro_after_init; | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_KEXEC_CORE | #ifdef CONFIG_KEXEC_CORE | ||||||
| /*
 | /*
 | ||||||
|  | @ -81,7 +91,7 @@ static void __init reserve_crashkernel(void) | ||||||
| 
 | 
 | ||||||
| 	if (crash_base == 0) { | 	if (crash_base == 0) { | ||||||
| 		/* Current arm64 boot protocol requires 2MB alignment */ | 		/* Current arm64 boot protocol requires 2MB alignment */ | ||||||
| 		crash_base = memblock_find_in_range(0, ARCH_LOW_ADDRESS_LIMIT, | 		crash_base = memblock_find_in_range(0, arm64_dma32_phys_limit, | ||||||
| 				crash_size, SZ_2M); | 				crash_size, SZ_2M); | ||||||
| 		if (crash_base == 0) { | 		if (crash_base == 0) { | ||||||
| 			pr_warn("cannot allocate crashkernel (size:0x%llx)\n", | 			pr_warn("cannot allocate crashkernel (size:0x%llx)\n", | ||||||
|  | @ -169,15 +179,16 @@ static void __init reserve_elfcorehdr(void) | ||||||
| { | { | ||||||
| } | } | ||||||
| #endif /* CONFIG_CRASH_DUMP */ | #endif /* CONFIG_CRASH_DUMP */ | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Return the maximum physical address for ZONE_DMA32 (DMA_BIT_MASK(32)). It |  * Return the maximum physical address for a zone with a given address size | ||||||
|  * currently assumes that for memory starting above 4G, 32-bit devices will |  * limit. It currently assumes that for memory starting above 4G, 32-bit | ||||||
|  * use a DMA offset. |  * devices will use a DMA offset. | ||||||
|  */ |  */ | ||||||
| static phys_addr_t __init max_zone_dma_phys(void) | static phys_addr_t __init max_zone_phys(unsigned int zone_bits) | ||||||
| { | { | ||||||
| 	phys_addr_t offset = memblock_start_of_DRAM() & GENMASK_ULL(63, 32); | 	phys_addr_t offset = memblock_start_of_DRAM() & GENMASK_ULL(63, zone_bits); | ||||||
| 	return min(offset + (1ULL << 32), memblock_end_of_DRAM()); | 	return min(offset + (1ULL << zone_bits), memblock_end_of_DRAM()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_NUMA | #ifdef CONFIG_NUMA | ||||||
|  | @ -186,8 +197,11 @@ static void __init zone_sizes_init(unsigned long min, unsigned long max) | ||||||
| { | { | ||||||
| 	unsigned long max_zone_pfns[MAX_NR_ZONES]  = {0}; | 	unsigned long max_zone_pfns[MAX_NR_ZONES]  = {0}; | ||||||
| 
 | 
 | ||||||
|  | #ifdef CONFIG_ZONE_DMA | ||||||
|  | 	max_zone_pfns[ZONE_DMA] = PFN_DOWN(arm64_dma_phys_limit); | ||||||
|  | #endif | ||||||
| #ifdef CONFIG_ZONE_DMA32 | #ifdef CONFIG_ZONE_DMA32 | ||||||
| 	max_zone_pfns[ZONE_DMA32] = PFN_DOWN(max_zone_dma_phys()); | 	max_zone_pfns[ZONE_DMA32] = PFN_DOWN(arm64_dma32_phys_limit); | ||||||
| #endif | #endif | ||||||
| 	max_zone_pfns[ZONE_NORMAL] = max; | 	max_zone_pfns[ZONE_NORMAL] = max; | ||||||
| 
 | 
 | ||||||
|  | @ -200,16 +214,21 @@ static void __init zone_sizes_init(unsigned long min, unsigned long max) | ||||||
| { | { | ||||||
| 	struct memblock_region *reg; | 	struct memblock_region *reg; | ||||||
| 	unsigned long zone_size[MAX_NR_ZONES], zhole_size[MAX_NR_ZONES]; | 	unsigned long zone_size[MAX_NR_ZONES], zhole_size[MAX_NR_ZONES]; | ||||||
| 	unsigned long max_dma = min; | 	unsigned long max_dma32 = min; | ||||||
|  | 	unsigned long __maybe_unused max_dma = min; | ||||||
| 
 | 
 | ||||||
| 	memset(zone_size, 0, sizeof(zone_size)); | 	memset(zone_size, 0, sizeof(zone_size)); | ||||||
| 
 | 
 | ||||||
| 	/* 4GB maximum for 32-bit only capable devices */ | #ifdef CONFIG_ZONE_DMA | ||||||
| #ifdef CONFIG_ZONE_DMA32 |  | ||||||
| 	max_dma = PFN_DOWN(arm64_dma_phys_limit); | 	max_dma = PFN_DOWN(arm64_dma_phys_limit); | ||||||
| 	zone_size[ZONE_DMA32] = max_dma - min; | 	zone_size[ZONE_DMA] = max_dma - min; | ||||||
|  | 	max_dma32 = max_dma; | ||||||
| #endif | #endif | ||||||
| 	zone_size[ZONE_NORMAL] = max - max_dma; | #ifdef CONFIG_ZONE_DMA32 | ||||||
|  | 	max_dma32 = PFN_DOWN(arm64_dma32_phys_limit); | ||||||
|  | 	zone_size[ZONE_DMA32] = max_dma32 - max_dma; | ||||||
|  | #endif | ||||||
|  | 	zone_size[ZONE_NORMAL] = max - max_dma32; | ||||||
| 
 | 
 | ||||||
| 	memcpy(zhole_size, zone_size, sizeof(zhole_size)); | 	memcpy(zhole_size, zone_size, sizeof(zhole_size)); | ||||||
| 
 | 
 | ||||||
|  | @ -219,16 +238,22 @@ static void __init zone_sizes_init(unsigned long min, unsigned long max) | ||||||
| 
 | 
 | ||||||
| 		if (start >= max) | 		if (start >= max) | ||||||
| 			continue; | 			continue; | ||||||
| 
 | #ifdef CONFIG_ZONE_DMA | ||||||
| #ifdef CONFIG_ZONE_DMA32 |  | ||||||
| 		if (start < max_dma) { | 		if (start < max_dma) { | ||||||
| 			unsigned long dma_end = min(end, max_dma); | 			unsigned long dma_end = min_not_zero(end, max_dma); | ||||||
| 			zhole_size[ZONE_DMA32] -= dma_end - start; | 			zhole_size[ZONE_DMA] -= dma_end - start; | ||||||
| 		} | 		} | ||||||
| #endif | #endif | ||||||
| 		if (end > max_dma) { | #ifdef CONFIG_ZONE_DMA32 | ||||||
|  | 		if (start < max_dma32) { | ||||||
|  | 			unsigned long dma32_end = min(end, max_dma32); | ||||||
|  | 			unsigned long dma32_start = max(start, max_dma); | ||||||
|  | 			zhole_size[ZONE_DMA32] -= dma32_end - dma32_start; | ||||||
|  | 		} | ||||||
|  | #endif | ||||||
|  | 		if (end > max_dma32) { | ||||||
| 			unsigned long normal_end = min(end, max); | 			unsigned long normal_end = min(end, max); | ||||||
| 			unsigned long normal_start = max(start, max_dma); | 			unsigned long normal_start = max(start, max_dma32); | ||||||
| 			zhole_size[ZONE_NORMAL] -= normal_end - normal_start; | 			zhole_size[ZONE_NORMAL] -= normal_end - normal_start; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | @ -418,11 +443,15 @@ void __init arm64_memblock_init(void) | ||||||
| 
 | 
 | ||||||
| 	early_init_fdt_scan_reserved_mem(); | 	early_init_fdt_scan_reserved_mem(); | ||||||
| 
 | 
 | ||||||
| 	/* 4GB maximum for 32-bit only capable devices */ | 	if (IS_ENABLED(CONFIG_ZONE_DMA)) { | ||||||
|  | 		zone_dma_bits = ARM64_ZONE_DMA_BITS; | ||||||
|  | 		arm64_dma_phys_limit = max_zone_phys(ARM64_ZONE_DMA_BITS); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if (IS_ENABLED(CONFIG_ZONE_DMA32)) | 	if (IS_ENABLED(CONFIG_ZONE_DMA32)) | ||||||
| 		arm64_dma_phys_limit = max_zone_dma_phys(); | 		arm64_dma32_phys_limit = max_zone_phys(32); | ||||||
| 	else | 	else | ||||||
| 		arm64_dma_phys_limit = PHYS_MASK + 1; | 		arm64_dma32_phys_limit = PHYS_MASK + 1; | ||||||
| 
 | 
 | ||||||
| 	reserve_crashkernel(); | 	reserve_crashkernel(); | ||||||
| 
 | 
 | ||||||
|  | @ -430,7 +459,7 @@ void __init arm64_memblock_init(void) | ||||||
| 
 | 
 | ||||||
| 	high_memory = __va(memblock_end_of_DRAM() - 1) + 1; | 	high_memory = __va(memblock_end_of_DRAM() - 1) + 1; | ||||||
| 
 | 
 | ||||||
| 	dma_contiguous_reserve(arm64_dma_phys_limit); | 	dma_contiguous_reserve(arm64_dma32_phys_limit); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void __init bootmem_init(void) | void __init bootmem_init(void) | ||||||
|  | @ -534,7 +563,7 @@ static void __init free_unused_memmap(void) | ||||||
| void __init mem_init(void) | void __init mem_init(void) | ||||||
| { | { | ||||||
| 	if (swiotlb_force == SWIOTLB_FORCE || | 	if (swiotlb_force == SWIOTLB_FORCE || | ||||||
| 	    max_pfn > (arm64_dma_phys_limit >> PAGE_SHIFT)) | 	    max_pfn > PFN_DOWN(arm64_dma_phys_limit ? : arm64_dma32_phys_limit)) | ||||||
| 		swiotlb_init(1); | 		swiotlb_init(1); | ||||||
| 	else | 	else | ||||||
| 		swiotlb_force = SWIOTLB_NO_FORCE; | 		swiotlb_force = SWIOTLB_NO_FORCE; | ||||||
|  | @ -571,7 +600,7 @@ void free_initmem(void) | ||||||
| { | { | ||||||
| 	free_reserved_area(lm_alias(__init_begin), | 	free_reserved_area(lm_alias(__init_begin), | ||||||
| 			   lm_alias(__init_end), | 			   lm_alias(__init_end), | ||||||
| 			   0, "unused kernel"); | 			   POISON_FREE_INITMEM, "unused kernel"); | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Unmap the __init region but leave the VM area in place. This | 	 * Unmap the __init region but leave the VM area in place. This | ||||||
| 	 * prevents the region from being reused for kernel modules, which | 	 * prevents the region from being reused for kernel modules, which | ||||||
|  | @ -580,18 +609,6 @@ void free_initmem(void) | ||||||
| 	unmap_kernel_range((u64)__init_begin, (u64)(__init_end - __init_begin)); | 	unmap_kernel_range((u64)__init_begin, (u64)(__init_end - __init_begin)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_BLK_DEV_INITRD |  | ||||||
| void __init free_initrd_mem(unsigned long start, unsigned long end) |  | ||||||
| { |  | ||||||
| 	unsigned long aligned_start, aligned_end; |  | ||||||
| 
 |  | ||||||
| 	aligned_start = __virt_to_phys(start) & PAGE_MASK; |  | ||||||
| 	aligned_end = PAGE_ALIGN(__virt_to_phys(end)); |  | ||||||
| 	memblock_free(aligned_start, aligned_end - aligned_start); |  | ||||||
| 	free_reserved_area((void *)start, (void *)end, 0, "initrd"); |  | ||||||
| } |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| /*
 | /*
 | ||||||
|  * Dump out memory limit information on panic. |  * Dump out memory limit information on panic. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | @ -338,7 +338,7 @@ static void __create_pgd_mapping(pgd_t *pgdir, phys_addr_t phys, | ||||||
| 				 phys_addr_t (*pgtable_alloc)(int), | 				 phys_addr_t (*pgtable_alloc)(int), | ||||||
| 				 int flags) | 				 int flags) | ||||||
| { | { | ||||||
| 	unsigned long addr, length, end, next; | 	unsigned long addr, end, next; | ||||||
| 	pgd_t *pgdp = pgd_offset_raw(pgdir, virt); | 	pgd_t *pgdp = pgd_offset_raw(pgdir, virt); | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
|  | @ -350,9 +350,8 @@ static void __create_pgd_mapping(pgd_t *pgdir, phys_addr_t phys, | ||||||
| 
 | 
 | ||||||
| 	phys &= PAGE_MASK; | 	phys &= PAGE_MASK; | ||||||
| 	addr = virt & PAGE_MASK; | 	addr = virt & PAGE_MASK; | ||||||
| 	length = PAGE_ALIGN(size + (virt & ~PAGE_MASK)); | 	end = PAGE_ALIGN(virt + size); | ||||||
| 
 | 
 | ||||||
| 	end = addr + length; |  | ||||||
| 	do { | 	do { | ||||||
| 		next = pgd_addr_end(addr, end); | 		next = pgd_addr_end(addr, end); | ||||||
| 		alloc_init_pud(pgdp, addr, next, phys, prot, pgtable_alloc, | 		alloc_init_pud(pgdp, addr, next, phys, prot, pgtable_alloc, | ||||||
|  |  | ||||||
|  | @ -329,13 +329,4 @@ struct vm_area_struct; | ||||||
| #endif /* __ASSEMBLY__ */ | #endif /* __ASSEMBLY__ */ | ||||||
| #include <asm/slice.h> | #include <asm/slice.h> | ||||||
| 
 | 
 | ||||||
| /*
 |  | ||||||
|  * Allow 30-bit DMA for very limited Broadcom wifi chips on many powerbooks. |  | ||||||
|  */ |  | ||||||
| #ifdef CONFIG_PPC32 |  | ||||||
| #define ARCH_ZONE_DMA_BITS 30 |  | ||||||
| #else |  | ||||||
| #define ARCH_ZONE_DMA_BITS 31 |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
| #endif /* _ASM_POWERPC_PAGE_H */ | #endif /* _ASM_POWERPC_PAGE_H */ | ||||||
|  |  | ||||||
|  | @ -31,6 +31,7 @@ | ||||||
| #include <linux/slab.h> | #include <linux/slab.h> | ||||||
| #include <linux/vmalloc.h> | #include <linux/vmalloc.h> | ||||||
| #include <linux/memremap.h> | #include <linux/memremap.h> | ||||||
|  | #include <linux/dma-direct.h> | ||||||
| 
 | 
 | ||||||
| #include <asm/pgalloc.h> | #include <asm/pgalloc.h> | ||||||
| #include <asm/prom.h> | #include <asm/prom.h> | ||||||
|  | @ -201,10 +202,10 @@ static int __init mark_nonram_nosave(void) | ||||||
|  * everything else. GFP_DMA32 page allocations automatically fall back to |  * everything else. GFP_DMA32 page allocations automatically fall back to | ||||||
|  * ZONE_DMA. |  * ZONE_DMA. | ||||||
|  * |  * | ||||||
|  * By using 31-bit unconditionally, we can exploit ARCH_ZONE_DMA_BITS to |  * By using 31-bit unconditionally, we can exploit zone_dma_bits to inform the | ||||||
|  * inform the generic DMA mapping code.  32-bit only devices (if not handled |  * generic DMA mapping code.  32-bit only devices (if not handled by an IOMMU | ||||||
|  * by an IOMMU anyway) will take a first dip into ZONE_NORMAL and get |  * anyway) will take a first dip into ZONE_NORMAL and get otherwise served by | ||||||
|  * otherwise served by ZONE_DMA. |  * ZONE_DMA. | ||||||
|  */ |  */ | ||||||
| static unsigned long max_zone_pfns[MAX_NR_ZONES]; | static unsigned long max_zone_pfns[MAX_NR_ZONES]; | ||||||
| 
 | 
 | ||||||
|  | @ -237,9 +238,18 @@ void __init paging_init(void) | ||||||
| 	printk(KERN_DEBUG "Memory hole size: %ldMB\n", | 	printk(KERN_DEBUG "Memory hole size: %ldMB\n", | ||||||
| 	       (long int)((top_of_ram - total_ram) >> 20)); | 	       (long int)((top_of_ram - total_ram) >> 20)); | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Allow 30-bit DMA for very limited Broadcom wifi chips on many | ||||||
|  | 	 * powerbooks. | ||||||
|  | 	 */ | ||||||
|  | 	if (IS_ENABLED(CONFIG_PPC32)) | ||||||
|  | 		zone_dma_bits = 30; | ||||||
|  | 	else | ||||||
|  | 		zone_dma_bits = 31; | ||||||
|  | 
 | ||||||
| #ifdef CONFIG_ZONE_DMA | #ifdef CONFIG_ZONE_DMA | ||||||
| 	max_zone_pfns[ZONE_DMA]	= min(max_low_pfn, | 	max_zone_pfns[ZONE_DMA]	= min(max_low_pfn, | ||||||
| 				      1UL << (ARCH_ZONE_DMA_BITS - PAGE_SHIFT)); | 				      1UL << (zone_dma_bits - PAGE_SHIFT)); | ||||||
| #endif | #endif | ||||||
| 	max_zone_pfns[ZONE_NORMAL] = max_low_pfn; | 	max_zone_pfns[ZONE_NORMAL] = max_low_pfn; | ||||||
| #ifdef CONFIG_HIGHMEM | #ifdef CONFIG_HIGHMEM | ||||||
|  |  | ||||||
|  | @ -177,8 +177,6 @@ static inline int devmem_is_allowed(unsigned long pfn) | ||||||
| #define VM_DATA_DEFAULT_FLAGS	(VM_READ | VM_WRITE | \ | #define VM_DATA_DEFAULT_FLAGS	(VM_READ | VM_WRITE | \ | ||||||
| 				 VM_MAYREAD | VM_MAYWRITE | VM_MAYEXEC) | 				 VM_MAYREAD | VM_MAYWRITE | VM_MAYEXEC) | ||||||
| 
 | 
 | ||||||
| #define ARCH_ZONE_DMA_BITS	31 |  | ||||||
| 
 |  | ||||||
| #include <asm-generic/memory_model.h> | #include <asm-generic/memory_model.h> | ||||||
| #include <asm-generic/getorder.h> | #include <asm-generic/getorder.h> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -118,6 +118,7 @@ void __init paging_init(void) | ||||||
| 
 | 
 | ||||||
| 	sparse_memory_present_with_active_regions(MAX_NUMNODES); | 	sparse_memory_present_with_active_regions(MAX_NUMNODES); | ||||||
| 	sparse_init(); | 	sparse_init(); | ||||||
|  | 	zone_dma_bits = 31; | ||||||
| 	memset(max_zone_pfns, 0, sizeof(max_zone_pfns)); | 	memset(max_zone_pfns, 0, sizeof(max_zone_pfns)); | ||||||
| 	max_zone_pfns[ZONE_DMA] = PFN_DOWN(MAX_DMA_ADDRESS); | 	max_zone_pfns[ZONE_DMA] = PFN_DOWN(MAX_DMA_ADDRESS); | ||||||
| 	max_zone_pfns[ZONE_NORMAL] = max_low_pfn; | 	max_zone_pfns[ZONE_NORMAL] = max_low_pfn; | ||||||
|  |  | ||||||
|  | @ -1463,6 +1463,12 @@ static inline bool arch_has_pfn_modify_check(void) | ||||||
| 	return boot_cpu_has_bug(X86_BUG_L1TF); | 	return boot_cpu_has_bug(X86_BUG_L1TF); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #define arch_faults_on_old_pte arch_faults_on_old_pte | ||||||
|  | static inline bool arch_faults_on_old_pte(void) | ||||||
|  | { | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #include <asm-generic/pgtable.h> | #include <asm-generic/pgtable.h> | ||||||
| #endif	/* __ASSEMBLY__ */ | #endif	/* __ASSEMBLY__ */ | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -967,29 +967,29 @@ static int sdei_get_conduit(struct platform_device *pdev) | ||||||
| 	if (np) { | 	if (np) { | ||||||
| 		if (of_property_read_string(np, "method", &method)) { | 		if (of_property_read_string(np, "method", &method)) { | ||||||
| 			pr_warn("missing \"method\" property\n"); | 			pr_warn("missing \"method\" property\n"); | ||||||
| 			return CONDUIT_INVALID; | 			return SMCCC_CONDUIT_NONE; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (!strcmp("hvc", method)) { | 		if (!strcmp("hvc", method)) { | ||||||
| 			sdei_firmware_call = &sdei_smccc_hvc; | 			sdei_firmware_call = &sdei_smccc_hvc; | ||||||
| 			return CONDUIT_HVC; | 			return SMCCC_CONDUIT_HVC; | ||||||
| 		} else if (!strcmp("smc", method)) { | 		} else if (!strcmp("smc", method)) { | ||||||
| 			sdei_firmware_call = &sdei_smccc_smc; | 			sdei_firmware_call = &sdei_smccc_smc; | ||||||
| 			return CONDUIT_SMC; | 			return SMCCC_CONDUIT_SMC; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		pr_warn("invalid \"method\" property: %s\n", method); | 		pr_warn("invalid \"method\" property: %s\n", method); | ||||||
| 	} else if (IS_ENABLED(CONFIG_ACPI) && !acpi_disabled) { | 	} else if (IS_ENABLED(CONFIG_ACPI) && !acpi_disabled) { | ||||||
| 		if (acpi_psci_use_hvc()) { | 		if (acpi_psci_use_hvc()) { | ||||||
| 			sdei_firmware_call = &sdei_smccc_hvc; | 			sdei_firmware_call = &sdei_smccc_hvc; | ||||||
| 			return CONDUIT_HVC; | 			return SMCCC_CONDUIT_HVC; | ||||||
| 		} else { | 		} else { | ||||||
| 			sdei_firmware_call = &sdei_smccc_smc; | 			sdei_firmware_call = &sdei_smccc_smc; | ||||||
| 			return CONDUIT_SMC; | 			return SMCCC_CONDUIT_SMC; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return CONDUIT_INVALID; | 	return SMCCC_CONDUIT_NONE; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int sdei_probe(struct platform_device *pdev) | static int sdei_probe(struct platform_device *pdev) | ||||||
|  |  | ||||||
|  | @ -53,10 +53,18 @@ bool psci_tos_resident_on(int cpu) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct psci_operations psci_ops = { | struct psci_operations psci_ops = { | ||||||
| 	.conduit = PSCI_CONDUIT_NONE, | 	.conduit = SMCCC_CONDUIT_NONE, | ||||||
| 	.smccc_version = SMCCC_VERSION_1_0, | 	.smccc_version = SMCCC_VERSION_1_0, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void) | ||||||
|  | { | ||||||
|  | 	if (psci_ops.smccc_version < SMCCC_VERSION_1_1) | ||||||
|  | 		return SMCCC_CONDUIT_NONE; | ||||||
|  | 
 | ||||||
|  | 	return psci_ops.conduit; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| typedef unsigned long (psci_fn)(unsigned long, unsigned long, | typedef unsigned long (psci_fn)(unsigned long, unsigned long, | ||||||
| 				unsigned long, unsigned long); | 				unsigned long, unsigned long); | ||||||
| static psci_fn *invoke_psci_fn; | static psci_fn *invoke_psci_fn; | ||||||
|  | @ -212,13 +220,13 @@ static unsigned long psci_migrate_info_up_cpu(void) | ||||||
| 			      0, 0, 0); | 			      0, 0, 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void set_conduit(enum psci_conduit conduit) | static void set_conduit(enum arm_smccc_conduit conduit) | ||||||
| { | { | ||||||
| 	switch (conduit) { | 	switch (conduit) { | ||||||
| 	case PSCI_CONDUIT_HVC: | 	case SMCCC_CONDUIT_HVC: | ||||||
| 		invoke_psci_fn = __invoke_psci_fn_hvc; | 		invoke_psci_fn = __invoke_psci_fn_hvc; | ||||||
| 		break; | 		break; | ||||||
| 	case PSCI_CONDUIT_SMC: | 	case SMCCC_CONDUIT_SMC: | ||||||
| 		invoke_psci_fn = __invoke_psci_fn_smc; | 		invoke_psci_fn = __invoke_psci_fn_smc; | ||||||
| 		break; | 		break; | ||||||
| 	default: | 	default: | ||||||
|  | @ -240,9 +248,9 @@ static int get_set_conduit_method(struct device_node *np) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (!strcmp("hvc", method)) { | 	if (!strcmp("hvc", method)) { | ||||||
| 		set_conduit(PSCI_CONDUIT_HVC); | 		set_conduit(SMCCC_CONDUIT_HVC); | ||||||
| 	} else if (!strcmp("smc", method)) { | 	} else if (!strcmp("smc", method)) { | ||||||
| 		set_conduit(PSCI_CONDUIT_SMC); | 		set_conduit(SMCCC_CONDUIT_SMC); | ||||||
| 	} else { | 	} else { | ||||||
| 		pr_warn("invalid \"method\" property: %s\n", method); | 		pr_warn("invalid \"method\" property: %s\n", method); | ||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
|  | @ -583,9 +591,9 @@ int __init psci_acpi_init(void) | ||||||
| 	pr_info("probing for conduit method from ACPI.\n"); | 	pr_info("probing for conduit method from ACPI.\n"); | ||||||
| 
 | 
 | ||||||
| 	if (acpi_psci_use_hvc()) | 	if (acpi_psci_use_hvc()) | ||||||
| 		set_conduit(PSCI_CONDUIT_HVC); | 		set_conduit(SMCCC_CONDUIT_HVC); | ||||||
| 	else | 	else | ||||||
| 		set_conduit(PSCI_CONDUIT_SMC); | 		set_conduit(SMCCC_CONDUIT_SMC); | ||||||
| 
 | 
 | ||||||
| 	return psci_probe(); | 	return psci_probe(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -87,6 +87,15 @@ static DEFINE_STATIC_KEY_TRUE(supports_deactivate_key); | ||||||
|  */ |  */ | ||||||
| static DEFINE_STATIC_KEY_FALSE(supports_pseudo_nmis); | static DEFINE_STATIC_KEY_FALSE(supports_pseudo_nmis); | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Global static key controlling whether an update to PMR allowing more | ||||||
|  |  * interrupts requires to be propagated to the redistributor (DSB SY). | ||||||
|  |  * And this needs to be exported for modules to be able to enable | ||||||
|  |  * interrupts... | ||||||
|  |  */ | ||||||
|  | DEFINE_STATIC_KEY_FALSE(gic_pmr_sync); | ||||||
|  | EXPORT_SYMBOL(gic_pmr_sync); | ||||||
|  | 
 | ||||||
| /* ppi_nmi_refs[n] == number of cpus having ppi[n + 16] set as NMI */ | /* ppi_nmi_refs[n] == number of cpus having ppi[n + 16] set as NMI */ | ||||||
| static refcount_t *ppi_nmi_refs; | static refcount_t *ppi_nmi_refs; | ||||||
| 
 | 
 | ||||||
|  | @ -1502,6 +1511,17 @@ static void gic_enable_nmi_support(void) | ||||||
| 	for (i = 0; i < gic_data.ppi_nr; i++) | 	for (i = 0; i < gic_data.ppi_nr; i++) | ||||||
| 		refcount_set(&ppi_nmi_refs[i], 0); | 		refcount_set(&ppi_nmi_refs[i], 0); | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Linux itself doesn't use 1:N distribution, so has no need to | ||||||
|  | 	 * set PMHE. The only reason to have it set is if EL3 requires it | ||||||
|  | 	 * (and we can't change it). | ||||||
|  | 	 */ | ||||||
|  | 	if (gic_read_ctlr() & ICC_CTLR_EL1_PMHE_MASK) | ||||||
|  | 		static_branch_enable(&gic_pmr_sync); | ||||||
|  | 
 | ||||||
|  | 	pr_info("%s ICC_PMR_EL1 synchronisation\n", | ||||||
|  | 		static_branch_unlikely(&gic_pmr_sync) ? "Forcing" : "Relaxing"); | ||||||
|  | 
 | ||||||
| 	static_branch_enable(&supports_pseudo_nmis); | 	static_branch_enable(&supports_pseudo_nmis); | ||||||
| 
 | 
 | ||||||
| 	if (static_branch_likely(&supports_deactivate_key)) | 	if (static_branch_likely(&supports_deactivate_key)) | ||||||
|  |  | ||||||
|  | @ -80,6 +80,22 @@ | ||||||
| 
 | 
 | ||||||
| #include <linux/linkage.h> | #include <linux/linkage.h> | ||||||
| #include <linux/types.h> | #include <linux/types.h> | ||||||
|  | 
 | ||||||
|  | enum arm_smccc_conduit { | ||||||
|  | 	SMCCC_CONDUIT_NONE, | ||||||
|  | 	SMCCC_CONDUIT_SMC, | ||||||
|  | 	SMCCC_CONDUIT_HVC, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * arm_smccc_1_1_get_conduit() | ||||||
|  |  * | ||||||
|  |  * Returns the conduit to be used for SMCCCv1.1 or later. | ||||||
|  |  * | ||||||
|  |  * When SMCCCv1.1 is not present, returns SMCCC_CONDUIT_NONE. | ||||||
|  |  */ | ||||||
|  | enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void); | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * struct arm_smccc_res - Result from SMC/HVC call |  * struct arm_smccc_res - Result from SMC/HVC call | ||||||
|  * @a0-a3 result values from registers 0 to 3 |  * @a0-a3 result values from registers 0 to 3 | ||||||
|  |  | ||||||
|  | @ -5,12 +5,6 @@ | ||||||
| 
 | 
 | ||||||
| #include <uapi/linux/arm_sdei.h> | #include <uapi/linux/arm_sdei.h> | ||||||
| 
 | 
 | ||||||
| enum sdei_conduit_types { |  | ||||||
| 	CONDUIT_INVALID = 0, |  | ||||||
| 	CONDUIT_SMC, |  | ||||||
| 	CONDUIT_HVC, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| #include <acpi/ghes.h> | #include <acpi/ghes.h> | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_ARM_SDE_INTERFACE | #ifdef CONFIG_ARM_SDE_INTERFACE | ||||||
|  |  | ||||||
|  | @ -5,6 +5,8 @@ | ||||||
| #include <linux/dma-mapping.h> | #include <linux/dma-mapping.h> | ||||||
| #include <linux/mem_encrypt.h> | #include <linux/mem_encrypt.h> | ||||||
| 
 | 
 | ||||||
|  | extern unsigned int zone_dma_bits; | ||||||
|  | 
 | ||||||
| #ifdef CONFIG_ARCH_HAS_PHYS_TO_DMA | #ifdef CONFIG_ARCH_HAS_PHYS_TO_DMA | ||||||
| #include <asm/dma-direct.h> | #include <asm/dma-direct.h> | ||||||
| #else | #else | ||||||
|  |  | ||||||
|  | @ -487,6 +487,8 @@ | ||||||
| #define ICC_CTLR_EL1_EOImode_MASK	(1 << ICC_CTLR_EL1_EOImode_SHIFT) | #define ICC_CTLR_EL1_EOImode_MASK	(1 << ICC_CTLR_EL1_EOImode_SHIFT) | ||||||
| #define ICC_CTLR_EL1_CBPR_SHIFT		0 | #define ICC_CTLR_EL1_CBPR_SHIFT		0 | ||||||
| #define ICC_CTLR_EL1_CBPR_MASK		(1 << ICC_CTLR_EL1_CBPR_SHIFT) | #define ICC_CTLR_EL1_CBPR_MASK		(1 << ICC_CTLR_EL1_CBPR_SHIFT) | ||||||
|  | #define ICC_CTLR_EL1_PMHE_SHIFT		6 | ||||||
|  | #define ICC_CTLR_EL1_PMHE_MASK		(1 << ICC_CTLR_EL1_PMHE_SHIFT) | ||||||
| #define ICC_CTLR_EL1_PRI_BITS_SHIFT	8 | #define ICC_CTLR_EL1_PRI_BITS_SHIFT	8 | ||||||
| #define ICC_CTLR_EL1_PRI_BITS_MASK	(0x7 << ICC_CTLR_EL1_PRI_BITS_SHIFT) | #define ICC_CTLR_EL1_PRI_BITS_MASK	(0x7 << ICC_CTLR_EL1_PRI_BITS_SHIFT) | ||||||
| #define ICC_CTLR_EL1_ID_BITS_SHIFT	11 | #define ICC_CTLR_EL1_ID_BITS_SHIFT	11 | ||||||
|  |  | ||||||
|  | @ -359,33 +359,40 @@ struct per_cpu_nodestat { | ||||||
| #endif /* !__GENERATING_BOUNDS.H */ | #endif /* !__GENERATING_BOUNDS.H */ | ||||||
| 
 | 
 | ||||||
| enum zone_type { | enum zone_type { | ||||||
| #ifdef CONFIG_ZONE_DMA |  | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * ZONE_DMA is used when there are devices that are not able | 	 * ZONE_DMA and ZONE_DMA32 are used when there are peripherals not able | ||||||
| 	 * to do DMA to all of addressable memory (ZONE_NORMAL). Then we | 	 * to DMA to all of the addressable memory (ZONE_NORMAL). | ||||||
| 	 * carve out the portion of memory that is needed for these devices. | 	 * On architectures where this area covers the whole 32 bit address | ||||||
| 	 * The range is arch specific. | 	 * space ZONE_DMA32 is used. ZONE_DMA is left for the ones with smaller | ||||||
|  | 	 * DMA addressing constraints. This distinction is important as a 32bit | ||||||
|  | 	 * DMA mask is assumed when ZONE_DMA32 is defined. Some 64-bit | ||||||
|  | 	 * platforms may need both zones as they support peripherals with | ||||||
|  | 	 * different DMA addressing limitations. | ||||||
| 	 * | 	 * | ||||||
| 	 * Some examples | 	 * Some examples: | ||||||
| 	 * | 	 * | ||||||
| 	 * Architecture		Limit | 	 *  - i386 and x86_64 have a fixed 16M ZONE_DMA and ZONE_DMA32 for the | ||||||
| 	 * --------------------------- | 	 *    rest of the lower 4G. | ||||||
| 	 * parisc, ia64, sparc	<4G |  | ||||||
| 	 * s390, powerpc	<2G |  | ||||||
| 	 * arm			Various |  | ||||||
| 	 * alpha		Unlimited or 0-16MB. |  | ||||||
| 	 * | 	 * | ||||||
| 	 * i386, x86_64 and multiple other arches | 	 *  - arm only uses ZONE_DMA, the size, up to 4G, may vary depending on | ||||||
| 	 * 			<16M. | 	 *    the specific device. | ||||||
|  | 	 * | ||||||
|  | 	 *  - arm64 has a fixed 1G ZONE_DMA and ZONE_DMA32 for the rest of the | ||||||
|  | 	 *    lower 4G. | ||||||
|  | 	 * | ||||||
|  | 	 *  - powerpc only uses ZONE_DMA, the size, up to 2G, may vary | ||||||
|  | 	 *    depending on the specific device. | ||||||
|  | 	 * | ||||||
|  | 	 *  - s390 uses ZONE_DMA fixed to the lower 2G. | ||||||
|  | 	 * | ||||||
|  | 	 *  - ia64 and riscv only use ZONE_DMA32. | ||||||
|  | 	 * | ||||||
|  | 	 *  - parisc uses neither. | ||||||
| 	 */ | 	 */ | ||||||
|  | #ifdef CONFIG_ZONE_DMA | ||||||
| 	ZONE_DMA, | 	ZONE_DMA, | ||||||
| #endif | #endif | ||||||
| #ifdef CONFIG_ZONE_DMA32 | #ifdef CONFIG_ZONE_DMA32 | ||||||
| 	/*
 |  | ||||||
| 	 * x86_64 needs two ZONE_DMAs because it supports devices that are |  | ||||||
| 	 * only able to do DMA to the lower 16M but also 32 bit devices that |  | ||||||
| 	 * can only do DMA areas below 4G. |  | ||||||
| 	 */ |  | ||||||
| 	ZONE_DMA32, | 	ZONE_DMA32, | ||||||
| #endif | #endif | ||||||
| 	/*
 | 	/*
 | ||||||
|  |  | ||||||
|  | @ -7,6 +7,7 @@ | ||||||
| #ifndef __LINUX_PSCI_H | #ifndef __LINUX_PSCI_H | ||||||
| #define __LINUX_PSCI_H | #define __LINUX_PSCI_H | ||||||
| 
 | 
 | ||||||
|  | #include <linux/arm-smccc.h> | ||||||
| #include <linux/init.h> | #include <linux/init.h> | ||||||
| #include <linux/types.h> | #include <linux/types.h> | ||||||
| 
 | 
 | ||||||
|  | @ -18,12 +19,6 @@ bool psci_tos_resident_on(int cpu); | ||||||
| int psci_cpu_suspend_enter(u32 state); | int psci_cpu_suspend_enter(u32 state); | ||||||
| bool psci_power_state_is_valid(u32 state); | bool psci_power_state_is_valid(u32 state); | ||||||
| 
 | 
 | ||||||
| enum psci_conduit { |  | ||||||
| 	PSCI_CONDUIT_NONE, |  | ||||||
| 	PSCI_CONDUIT_SMC, |  | ||||||
| 	PSCI_CONDUIT_HVC, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| enum smccc_version { | enum smccc_version { | ||||||
| 	SMCCC_VERSION_1_0, | 	SMCCC_VERSION_1_0, | ||||||
| 	SMCCC_VERSION_1_1, | 	SMCCC_VERSION_1_1, | ||||||
|  | @ -38,7 +33,7 @@ struct psci_operations { | ||||||
| 	int (*affinity_info)(unsigned long target_affinity, | 	int (*affinity_info)(unsigned long target_affinity, | ||||||
| 			unsigned long lowest_affinity_level); | 			unsigned long lowest_affinity_level); | ||||||
| 	int (*migrate_info_type)(void); | 	int (*migrate_info_type)(void); | ||||||
| 	enum psci_conduit conduit; | 	enum arm_smccc_conduit conduit; | ||||||
| 	enum smccc_version smccc_version; | 	enum smccc_version smccc_version; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -10,6 +10,7 @@ | ||||||
| #include <linux/syscalls.h> | #include <linux/syscalls.h> | ||||||
| #include <linux/utime.h> | #include <linux/utime.h> | ||||||
| #include <linux/file.h> | #include <linux/file.h> | ||||||
|  | #include <linux/memblock.h> | ||||||
| 
 | 
 | ||||||
| static ssize_t __init xwrite(int fd, const char *p, size_t count) | static ssize_t __init xwrite(int fd, const char *p, size_t count) | ||||||
| { | { | ||||||
|  | @ -529,6 +530,13 @@ extern unsigned long __initramfs_size; | ||||||
| 
 | 
 | ||||||
| void __weak free_initrd_mem(unsigned long start, unsigned long end) | void __weak free_initrd_mem(unsigned long start, unsigned long end) | ||||||
| { | { | ||||||
|  | #ifdef CONFIG_ARCH_KEEP_MEMBLOCK | ||||||
|  | 	unsigned long aligned_start = ALIGN_DOWN(start, PAGE_SIZE); | ||||||
|  | 	unsigned long aligned_end = ALIGN(end, PAGE_SIZE); | ||||||
|  | 
 | ||||||
|  | 	memblock_free(__pa(aligned_start), aligned_end - aligned_start); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
| 	free_reserved_area((void *)start, (void *)end, POISON_FREE_INITMEM, | 	free_reserved_area((void *)start, (void *)end, POISON_FREE_INITMEM, | ||||||
| 			"initrd"); | 			"initrd"); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,12 +16,11 @@ | ||||||
| #include <linux/swiotlb.h> | #include <linux/swiotlb.h> | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Most architectures use ZONE_DMA for the first 16 Megabytes, but |  * Most architectures use ZONE_DMA for the first 16 Megabytes, but some use it | ||||||
|  * some use it for entirely different regions: |  * it for entirely different regions. In that case the arch code needs to | ||||||
|  |  * override the variable below for dma-direct to work properly. | ||||||
|  */ |  */ | ||||||
| #ifndef ARCH_ZONE_DMA_BITS | unsigned int zone_dma_bits __ro_after_init = 24; | ||||||
| #define ARCH_ZONE_DMA_BITS 24 |  | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| static void report_addr(struct device *dev, dma_addr_t dma_addr, size_t size) | static void report_addr(struct device *dev, dma_addr_t dma_addr, size_t size) | ||||||
| { | { | ||||||
|  | @ -69,7 +68,7 @@ static gfp_t __dma_direct_optimal_gfp_mask(struct device *dev, u64 dma_mask, | ||||||
| 	 * Note that GFP_DMA32 and GFP_DMA are no ops without the corresponding | 	 * Note that GFP_DMA32 and GFP_DMA are no ops without the corresponding | ||||||
| 	 * zones. | 	 * zones. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (*phys_mask <= DMA_BIT_MASK(ARCH_ZONE_DMA_BITS)) | 	if (*phys_mask <= DMA_BIT_MASK(zone_dma_bits)) | ||||||
| 		return GFP_DMA; | 		return GFP_DMA; | ||||||
| 	if (*phys_mask <= DMA_BIT_MASK(32)) | 	if (*phys_mask <= DMA_BIT_MASK(32)) | ||||||
| 		return GFP_DMA32; | 		return GFP_DMA32; | ||||||
|  | @ -395,7 +394,7 @@ int dma_direct_supported(struct device *dev, u64 mask) | ||||||
| 	u64 min_mask; | 	u64 min_mask; | ||||||
| 
 | 
 | ||||||
| 	if (IS_ENABLED(CONFIG_ZONE_DMA)) | 	if (IS_ENABLED(CONFIG_ZONE_DMA)) | ||||||
| 		min_mask = DMA_BIT_MASK(ARCH_ZONE_DMA_BITS); | 		min_mask = DMA_BIT_MASK(zone_dma_bits); | ||||||
| 	else | 	else | ||||||
| 		min_mask = DMA_BIT_MASK(32); | 		min_mask = DMA_BIT_MASK(32); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										104
									
								
								mm/memory.c
									
										
									
									
									
								
							
							
						
						
									
										104
									
								
								mm/memory.c
									
										
									
									
									
								
							|  | @ -118,6 +118,18 @@ int randomize_va_space __read_mostly = | ||||||
| 					2; | 					2; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | #ifndef arch_faults_on_old_pte | ||||||
|  | static inline bool arch_faults_on_old_pte(void) | ||||||
|  | { | ||||||
|  | 	/*
 | ||||||
|  | 	 * Those arches which don't have hw access flag feature need to | ||||||
|  | 	 * implement their own helper. By default, "true" means pagefault | ||||||
|  | 	 * will be hit on old pte. | ||||||
|  | 	 */ | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
| static int __init disable_randmaps(char *s) | static int __init disable_randmaps(char *s) | ||||||
| { | { | ||||||
| 	randomize_va_space = 0; | 	randomize_va_space = 0; | ||||||
|  | @ -2145,32 +2157,82 @@ static inline int pte_unmap_same(struct mm_struct *mm, pmd_t *pmd, | ||||||
| 	return same; | 	return same; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static inline void cow_user_page(struct page *dst, struct page *src, unsigned long va, struct vm_area_struct *vma) | static inline bool cow_user_page(struct page *dst, struct page *src, | ||||||
|  | 				 struct vm_fault *vmf) | ||||||
| { | { | ||||||
|  | 	bool ret; | ||||||
|  | 	void *kaddr; | ||||||
|  | 	void __user *uaddr; | ||||||
|  | 	bool force_mkyoung; | ||||||
|  | 	struct vm_area_struct *vma = vmf->vma; | ||||||
|  | 	struct mm_struct *mm = vma->vm_mm; | ||||||
|  | 	unsigned long addr = vmf->address; | ||||||
|  | 
 | ||||||
| 	debug_dma_assert_idle(src); | 	debug_dma_assert_idle(src); | ||||||
| 
 | 
 | ||||||
|  | 	if (likely(src)) { | ||||||
|  | 		copy_user_highpage(dst, src, addr, vma); | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If the source page was a PFN mapping, we don't have | 	 * If the source page was a PFN mapping, we don't have | ||||||
| 	 * a "struct page" for it. We do a best-effort copy by | 	 * a "struct page" for it. We do a best-effort copy by | ||||||
| 	 * just copying from the original user address. If that | 	 * just copying from the original user address. If that | ||||||
| 	 * fails, we just zero-fill it. Live with it. | 	 * fails, we just zero-fill it. Live with it. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (unlikely(!src)) { | 	kaddr = kmap_atomic(dst); | ||||||
| 		void *kaddr = kmap_atomic(dst); | 	uaddr = (void __user *)(addr & PAGE_MASK); | ||||||
| 		void __user *uaddr = (void __user *)(va & PAGE_MASK); |  | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * On architectures with software "accessed" bits, we would | ||||||
|  | 	 * take a double page fault, so mark it accessed here. | ||||||
|  | 	 */ | ||||||
|  | 	force_mkyoung = arch_faults_on_old_pte() && !pte_young(vmf->orig_pte); | ||||||
|  | 	if (force_mkyoung) { | ||||||
|  | 		pte_t entry; | ||||||
|  | 
 | ||||||
|  | 		vmf->pte = pte_offset_map_lock(mm, vmf->pmd, addr, &vmf->ptl); | ||||||
|  | 		if (!likely(pte_same(*vmf->pte, vmf->orig_pte))) { | ||||||
|  | 			/*
 | ||||||
|  | 			 * Other thread has already handled the fault | ||||||
|  | 			 * and we don't need to do anything. If it's | ||||||
|  | 			 * not the case, the fault will be triggered | ||||||
|  | 			 * again on the same address. | ||||||
|  | 			 */ | ||||||
|  | 			ret = false; | ||||||
|  | 			goto pte_unlock; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		entry = pte_mkyoung(vmf->orig_pte); | ||||||
|  | 		if (ptep_set_access_flags(vma, addr, vmf->pte, entry, 0)) | ||||||
|  | 			update_mmu_cache(vma, addr, vmf->pte); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * This really shouldn't fail, because the page is there | ||||||
|  | 	 * in the page tables. But it might just be unreadable, | ||||||
|  | 	 * in which case we just give up and fill the result with | ||||||
|  | 	 * zeroes. | ||||||
|  | 	 */ | ||||||
|  | 	if (__copy_from_user_inatomic(kaddr, uaddr, PAGE_SIZE)) { | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * This really shouldn't fail, because the page is there | 		 * Give a warn in case there can be some obscure | ||||||
| 		 * in the page tables. But it might just be unreadable, | 		 * use-case | ||||||
| 		 * in which case we just give up and fill the result with |  | ||||||
| 		 * zeroes. |  | ||||||
| 		 */ | 		 */ | ||||||
| 		if (__copy_from_user_inatomic(kaddr, uaddr, PAGE_SIZE)) | 		WARN_ON_ONCE(1); | ||||||
| 			clear_page(kaddr); | 		clear_page(kaddr); | ||||||
| 		kunmap_atomic(kaddr); | 	} | ||||||
| 		flush_dcache_page(dst); | 
 | ||||||
| 	} else | 	ret = true; | ||||||
| 		copy_user_highpage(dst, src, va, vma); | 
 | ||||||
|  | pte_unlock: | ||||||
|  | 	if (force_mkyoung) | ||||||
|  | 		pte_unmap_unlock(vmf->pte, vmf->ptl); | ||||||
|  | 	kunmap_atomic(kaddr); | ||||||
|  | 	flush_dcache_page(dst); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static gfp_t __get_fault_gfp_mask(struct vm_area_struct *vma) | static gfp_t __get_fault_gfp_mask(struct vm_area_struct *vma) | ||||||
|  | @ -2327,7 +2389,19 @@ static vm_fault_t wp_page_copy(struct vm_fault *vmf) | ||||||
| 				vmf->address); | 				vmf->address); | ||||||
| 		if (!new_page) | 		if (!new_page) | ||||||
| 			goto oom; | 			goto oom; | ||||||
| 		cow_user_page(new_page, old_page, vmf->address, vma); | 
 | ||||||
|  | 		if (!cow_user_page(new_page, old_page, vmf)) { | ||||||
|  | 			/*
 | ||||||
|  | 			 * COW failed, if the fault was solved by other, | ||||||
|  | 			 * it's fine. If not, userspace would re-fault on | ||||||
|  | 			 * the same address and we will handle the fault | ||||||
|  | 			 * from the second attempt. | ||||||
|  | 			 */ | ||||||
|  | 			put_page(new_page); | ||||||
|  | 			if (old_page) | ||||||
|  | 				put_page(old_page); | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (mem_cgroup_try_charge_delay(new_page, mm, GFP_KERNEL, &memcg, false)) | 	if (mem_cgroup_try_charge_delay(new_page, mm, GFP_KERNEL, &memcg, false)) | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| # SPDX-License-Identifier: GPL-2.0
 | # SPDX-License-Identifier: GPL-2.0
 | ||||||
| TARGETS = android | TARGETS = android | ||||||
|  | TARGETS += arm64 | ||||||
| TARGETS += bpf | TARGETS += bpf | ||||||
| TARGETS += breakpoints | TARGETS += breakpoints | ||||||
| TARGETS += capabilities | TARGETS += capabilities | ||||||
|  |  | ||||||
|  | @ -1,12 +1,66 @@ | ||||||
| # SPDX-License-Identifier: GPL-2.0
 | # SPDX-License-Identifier: GPL-2.0
 | ||||||
| 
 | 
 | ||||||
| # ARCH can be overridden by the user for cross compiling
 | # When ARCH not overridden for crosscompiling, lookup machine
 | ||||||
| ARCH ?= $(shell uname -m 2>/dev/null || echo not) | ARCH ?= $(shell uname -m 2>/dev/null || echo not) | ||||||
| 
 | 
 | ||||||
| ifneq (,$(filter $(ARCH),aarch64 arm64)) | ifneq (,$(filter $(ARCH),aarch64 arm64)) | ||||||
| CFLAGS += -I../../../../usr/include/ | ARM64_SUBTARGETS ?= tags signal | ||||||
| TEST_GEN_PROGS := tags_test | else | ||||||
| TEST_PROGS := run_tags_test.sh | ARM64_SUBTARGETS := | ||||||
| endif | endif | ||||||
| 
 | 
 | ||||||
| include ../lib.mk | CFLAGS := -Wall -O2 -g | ||||||
|  | 
 | ||||||
|  | # A proper top_srcdir is needed by KSFT(lib.mk)
 | ||||||
|  | top_srcdir = $(realpath ../../../../) | ||||||
|  | 
 | ||||||
|  | # Additional include paths needed by kselftest.h and local headers
 | ||||||
|  | CFLAGS += -I$(top_srcdir)/tools/testing/selftests/ | ||||||
|  | 
 | ||||||
|  | # Guessing where the Kernel headers could have been installed
 | ||||||
|  | # depending on ENV config
 | ||||||
|  | ifeq ($(KBUILD_OUTPUT),) | ||||||
|  | khdr_dir = $(top_srcdir)/usr/include | ||||||
|  | else | ||||||
|  | # the KSFT preferred location when KBUILD_OUTPUT is set
 | ||||||
|  | khdr_dir = $(KBUILD_OUTPUT)/kselftest/usr/include | ||||||
|  | endif | ||||||
|  | 
 | ||||||
|  | CFLAGS += -I$(khdr_dir) | ||||||
|  | 
 | ||||||
|  | export CFLAGS | ||||||
|  | export top_srcdir | ||||||
|  | 
 | ||||||
|  | all: | ||||||
|  | 	@for DIR in $(ARM64_SUBTARGETS); do				\
 | ||||||
|  | 		BUILD_TARGET=$(OUTPUT)/$$DIR;			\
 | ||||||
|  | 		mkdir -p $$BUILD_TARGET;			\
 | ||||||
|  | 		make OUTPUT=$$BUILD_TARGET -C $$DIR $@;		\
 | ||||||
|  | 	done | ||||||
|  | 
 | ||||||
|  | install: all | ||||||
|  | 	@for DIR in $(ARM64_SUBTARGETS); do				\
 | ||||||
|  | 		BUILD_TARGET=$(OUTPUT)/$$DIR;			\
 | ||||||
|  | 		make OUTPUT=$$BUILD_TARGET -C $$DIR $@;		\
 | ||||||
|  | 	done | ||||||
|  | 
 | ||||||
|  | run_tests: all | ||||||
|  | 	@for DIR in $(ARM64_SUBTARGETS); do				\
 | ||||||
|  | 		BUILD_TARGET=$(OUTPUT)/$$DIR;			\
 | ||||||
|  | 		make OUTPUT=$$BUILD_TARGET -C $$DIR $@;		\
 | ||||||
|  | 	done | ||||||
|  | 
 | ||||||
|  | # Avoid any output on non arm64 on emit_tests
 | ||||||
|  | emit_tests: all | ||||||
|  | 	@for DIR in $(ARM64_SUBTARGETS); do				\
 | ||||||
|  | 		BUILD_TARGET=$(OUTPUT)/$$DIR;			\
 | ||||||
|  | 		make OUTPUT=$$BUILD_TARGET -C $$DIR $@;		\
 | ||||||
|  | 	done | ||||||
|  | 
 | ||||||
|  | clean: | ||||||
|  | 	@for DIR in $(ARM64_SUBTARGETS); do				\
 | ||||||
|  | 		BUILD_TARGET=$(OUTPUT)/$$DIR;			\
 | ||||||
|  | 		make OUTPUT=$$BUILD_TARGET -C $$DIR $@;		\
 | ||||||
|  | 	done | ||||||
|  | 
 | ||||||
|  | .PHONY: all clean install run_tests emit_tests | ||||||
|  |  | ||||||
							
								
								
									
										25
									
								
								tools/testing/selftests/arm64/README
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								tools/testing/selftests/arm64/README
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | ||||||
|  | KSelfTest ARM64 | ||||||
|  | =============== | ||||||
|  | 
 | ||||||
|  | - These tests are arm64 specific and so not built or run but just skipped | ||||||
|  |   completely when env-variable ARCH is found to be different than 'arm64' | ||||||
|  |   and `uname -m` reports other than 'aarch64'. | ||||||
|  | 
 | ||||||
|  | - Holding true the above, ARM64 KSFT tests can be run within the KSelfTest | ||||||
|  |   framework using standard Linux top-level-makefile targets: | ||||||
|  | 
 | ||||||
|  |       $ make TARGETS=arm64 kselftest-clean | ||||||
|  |       $ make TARGETS=arm64 kselftest | ||||||
|  | 
 | ||||||
|  |       or | ||||||
|  | 
 | ||||||
|  |       $ make -C tools/testing/selftests TARGETS=arm64 \ | ||||||
|  | 		INSTALL_PATH=<your-installation-path> install | ||||||
|  | 
 | ||||||
|  |       or, alternatively, only specific arm64/ subtargets can be picked: | ||||||
|  | 
 | ||||||
|  |       $ make -C tools/testing/selftests TARGETS=arm64 ARM64_SUBTARGETS="tags signal" \ | ||||||
|  | 		INSTALL_PATH=<your-installation-path> install | ||||||
|  | 
 | ||||||
|  |    Further details on building and running KFST can be found in: | ||||||
|  |      Documentation/dev-tools/kselftest.rst | ||||||
							
								
								
									
										3
									
								
								tools/testing/selftests/arm64/signal/.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								tools/testing/selftests/arm64/signal/.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | ||||||
|  | mangle_* | ||||||
|  | fake_sigreturn_* | ||||||
|  | !*.[ch] | ||||||
							
								
								
									
										32
									
								
								tools/testing/selftests/arm64/signal/Makefile
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								tools/testing/selftests/arm64/signal/Makefile
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,32 @@ | ||||||
|  | # SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | # Copyright (C) 2019 ARM Limited
 | ||||||
|  | 
 | ||||||
|  | # Additional include paths needed by kselftest.h and local headers
 | ||||||
|  | CFLAGS += -D_GNU_SOURCE -std=gnu99 -I. | ||||||
|  | 
 | ||||||
|  | SRCS := $(filter-out testcases/testcases.c,$(wildcard testcases/*.c)) | ||||||
|  | PROGS := $(patsubst %.c,%,$(SRCS)) | ||||||
|  | 
 | ||||||
|  | # Generated binaries to be installed by top KSFT script
 | ||||||
|  | TEST_GEN_PROGS := $(notdir $(PROGS)) | ||||||
|  | 
 | ||||||
|  | # Get Kernel headers installed and use them.
 | ||||||
|  | KSFT_KHDR_INSTALL := 1 | ||||||
|  | 
 | ||||||
|  | # Including KSFT lib.mk here will also mangle the TEST_GEN_PROGS list
 | ||||||
|  | # to account for any OUTPUT target-dirs optionally provided by
 | ||||||
|  | # the toplevel makefile
 | ||||||
|  | include ../../lib.mk | ||||||
|  | 
 | ||||||
|  | $(TEST_GEN_PROGS): $(PROGS) | ||||||
|  | 	cp $(PROGS) $(OUTPUT)/ | ||||||
|  | 
 | ||||||
|  | clean: | ||||||
|  | 	$(CLEAN) | ||||||
|  | 	rm -f $(PROGS) | ||||||
|  | 
 | ||||||
|  | # Common test-unit targets to build common-layout test-cases executables
 | ||||||
|  | # Needs secondary expansion to properly include the testcase c-file in pre-reqs
 | ||||||
|  | .SECONDEXPANSION: | ||||||
|  | $(PROGS): test_signals.c test_signals_utils.c testcases/testcases.c signals.S $$@.c test_signals.h test_signals_utils.h testcases/testcases.h | ||||||
|  | 	$(CC) $(CFLAGS) $^ -o $@ | ||||||
							
								
								
									
										59
									
								
								tools/testing/selftests/arm64/signal/README
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								tools/testing/selftests/arm64/signal/README
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | ||||||
|  | KSelfTest arm64/signal/ | ||||||
|  | ======================= | ||||||
|  | 
 | ||||||
|  | Signals Tests | ||||||
|  | +++++++++++++ | ||||||
|  | 
 | ||||||
|  | - Tests are built around a common main compilation unit: such shared main | ||||||
|  |   enforces a standard sequence of operations needed to perform a single | ||||||
|  |   signal-test (setup/trigger/run/result/cleanup) | ||||||
|  | 
 | ||||||
|  | - The above mentioned ops are configurable on a test-by-test basis: each test | ||||||
|  |   is described (and configured) using the descriptor signals.h::struct tdescr | ||||||
|  | 
 | ||||||
|  | - Each signal testcase is compiled into its own executable: a separate | ||||||
|  |   executable is used for each test since many tests complete successfully | ||||||
|  |   by receiving some kind of fatal signal from the Kernel, so it's safer | ||||||
|  |   to run each test unit in its own standalone process, so as to start each | ||||||
|  |   test from a clean slate. | ||||||
|  | 
 | ||||||
|  | - New tests can be simply defined in testcases/ dir providing a proper struct | ||||||
|  |   tdescr overriding all the defaults we wish to change (as of now providing a | ||||||
|  |   custom run method is mandatory though) | ||||||
|  | 
 | ||||||
|  | - Signals' test-cases hereafter defined belong currently to two | ||||||
|  |   principal families: | ||||||
|  | 
 | ||||||
|  |   - 'mangle_' tests: a real signal (SIGUSR1) is raised and used as a trigger | ||||||
|  |     and then the test case code modifies the signal frame from inside the | ||||||
|  |     signal handler itself. | ||||||
|  | 
 | ||||||
|  |   - 'fake_sigreturn_' tests: a brand new custom artificial sigframe structure | ||||||
|  |     is placed on the stack and a sigreturn syscall is called to simulate a | ||||||
|  |     real signal return. This kind of tests does not use a trigger usually and | ||||||
|  |     they are just fired using some simple included assembly trampoline code. | ||||||
|  | 
 | ||||||
|  |  - Most of these tests are successfully passing if the process gets killed by | ||||||
|  |    some fatal signal: usually SIGSEGV or SIGBUS. Since while writing this | ||||||
|  |    kind of tests it is extremely easy in fact to end-up injecting other | ||||||
|  |    unrelated SEGV bugs in the testcases, it becomes extremely tricky to | ||||||
|  |    be really sure that the tests are really addressing what they are meant | ||||||
|  |    to address and they are not instead falling apart due to unplanned bugs | ||||||
|  |    in the test code. | ||||||
|  |    In order to alleviate the misery of the life of such test-developer, a few | ||||||
|  |    helpers are provided: | ||||||
|  | 
 | ||||||
|  |    - a couple of ASSERT_BAD/GOOD_CONTEXT() macros to easily parse a ucontext_t | ||||||
|  |      and verify if it is indeed GOOD or BAD (depending on what we were | ||||||
|  |      expecting), using the same logic/perspective as in the arm64 Kernel signals | ||||||
|  |      routines. | ||||||
|  | 
 | ||||||
|  |    - a sanity mechanism to be used in 'fake_sigreturn_'-alike tests: enabled by | ||||||
|  |      default it takes care to verify that the test-execution had at least | ||||||
|  |      successfully progressed up to the stage of triggering the fake sigreturn | ||||||
|  |      call. | ||||||
|  | 
 | ||||||
|  |   In both cases test results are expected in terms of: | ||||||
|  |    - some fatal signal sent by the Kernel to the test process | ||||||
|  |   or | ||||||
|  |   - analyzing some final regs state | ||||||
							
								
								
									
										64
									
								
								tools/testing/selftests/arm64/signal/signals.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								tools/testing/selftests/arm64/signal/signals.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | ||||||
|  | /* SPDX-License-Identifier: GPL-2.0 */ | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | 
 | ||||||
|  | #include <asm/unistd.h> | ||||||
|  | 
 | ||||||
|  | .section        .rodata, "a" | ||||||
|  | call_fmt: | ||||||
|  | 	.asciz "Calling sigreturn with fake sigframe sized:%zd at SP @%08lX\n"
 | ||||||
|  | 
 | ||||||
|  | .text | ||||||
|  | 
 | ||||||
|  | .globl fake_sigreturn
 | ||||||
|  | 
 | ||||||
|  | /*	fake_sigreturn	x0:&sigframe,  x1:sigframe_size,  x2:misalign_bytes */ | ||||||
|  | fake_sigreturn: | ||||||
|  | 	stp	x29, x30, [sp, #-16]! | ||||||
|  | 	mov	x29, sp | ||||||
|  | 
 | ||||||
|  | 	mov	x20, x0 | ||||||
|  | 	mov	x21, x1 | ||||||
|  | 	mov	x22, x2 | ||||||
|  | 
 | ||||||
|  | 	/* create space on the stack for fake sigframe 16 bytes-aligned */ | ||||||
|  | 	add	x0, x21, x22 | ||||||
|  | 	add	x0, x0, #15 | ||||||
|  | 	bic	x0, x0, #15 /* round_up(sigframe_size + misalign_bytes, 16) */ | ||||||
|  | 	sub	sp, sp, x0 | ||||||
|  | 	add	x23, sp, x22 /* new sigframe base with misaligment if any */ | ||||||
|  | 
 | ||||||
|  | 	ldr	x0, =call_fmt | ||||||
|  | 	mov	x1, x21 | ||||||
|  | 	mov	x2, x23 | ||||||
|  | 	bl	printf | ||||||
|  | 
 | ||||||
|  | 	/* memcpy the provided content, while still keeping SP aligned */ | ||||||
|  | 	mov	x0, x23 | ||||||
|  | 	mov	x1, x20 | ||||||
|  | 	mov	x2, x21 | ||||||
|  | 	bl	memcpy | ||||||
|  | 
 | ||||||
|  | 	/* | ||||||
|  | 	 * Here saving a last minute SP to current->token acts as a marker: | ||||||
|  | 	 * if we got here, we are successfully faking a sigreturn; in other
 | ||||||
|  | 	 * words we are sure no bad fatal signal has been raised till now | ||||||
|  | 	 * for unrelated reasons, so we should consider the possibly observed | ||||||
|  | 	 * fatal signal like SEGV coming from Kernel restore_sigframe() and | ||||||
|  | 	 * triggered as expected from our test-case. | ||||||
|  | 	 * For simplicity this assumes that current field 'token' is laid out | ||||||
|  | 	 * as first in struct tdescr | ||||||
|  | 	 */ | ||||||
|  | 	ldr	x0, current | ||||||
|  | 	str	x23, [x0] | ||||||
|  | 	/* finally move SP to misaligned address...if any requested */ | ||||||
|  | 	mov	sp, x23 | ||||||
|  | 
 | ||||||
|  | 	mov	x8, #__NR_rt_sigreturn | ||||||
|  | 	svc	#0 | ||||||
|  | 
 | ||||||
|  | 	/* | ||||||
|  | 	 * Above sigreturn should not return...looping here leads to a timeout | ||||||
|  | 	 * and ensure proper and clean test failure, instead of jumping around | ||||||
|  | 	 * on a potentially corrupted stack. | ||||||
|  | 	 */ | ||||||
|  | 	b	. | ||||||
							
								
								
									
										29
									
								
								tools/testing/selftests/arm64/signal/test_signals.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								tools/testing/selftests/arm64/signal/test_signals.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,29 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Generic test wrapper for arm64 signal tests. | ||||||
|  |  * | ||||||
|  |  * Each test provides its own tde struct tdescr descriptor to link with | ||||||
|  |  * this wrapper. Framework provides common helpers. | ||||||
|  |  */ | ||||||
|  | #include <kselftest.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals.h" | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | 
 | ||||||
|  | struct tdescr *current; | ||||||
|  | 
 | ||||||
|  | int main(int argc, char *argv[]) | ||||||
|  | { | ||||||
|  | 	current = &tde; | ||||||
|  | 
 | ||||||
|  | 	ksft_print_msg("%s :: %s\n", current->name, current->descr); | ||||||
|  | 	if (test_setup(current) && test_init(current)) { | ||||||
|  | 		test_run(current); | ||||||
|  | 		test_cleanup(current); | ||||||
|  | 	} | ||||||
|  | 	test_result(current); | ||||||
|  | 
 | ||||||
|  | 	return current->result; | ||||||
|  | } | ||||||
							
								
								
									
										100
									
								
								tools/testing/selftests/arm64/signal/test_signals.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								tools/testing/selftests/arm64/signal/test_signals.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,100 @@ | ||||||
|  | /* SPDX-License-Identifier: GPL-2.0 */ | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | 
 | ||||||
|  | #ifndef __TEST_SIGNALS_H__ | ||||||
|  | #define __TEST_SIGNALS_H__ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <stdbool.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Using ARCH specific and sanitized Kernel headers installed by KSFT | ||||||
|  |  * framework since we asked for it by setting flag KSFT_KHDR_INSTALL | ||||||
|  |  * in our Makefile. | ||||||
|  |  */ | ||||||
|  | #include <asm/ptrace.h> | ||||||
|  | #include <asm/hwcap.h> | ||||||
|  | 
 | ||||||
|  | #define __stringify_1(x...)	#x | ||||||
|  | #define __stringify(x...)	__stringify_1(x) | ||||||
|  | 
 | ||||||
|  | #define get_regval(regname, out)			\ | ||||||
|  | {							\ | ||||||
|  | 	asm volatile("mrs %0, " __stringify(regname)	\ | ||||||
|  | 	: "=r" (out)					\ | ||||||
|  | 	:						\ | ||||||
|  | 	: "memory");					\ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Feature flags used in tdescr.feats_required to specify | ||||||
|  |  * any feature by the test | ||||||
|  |  */ | ||||||
|  | enum { | ||||||
|  | 	FSSBS_BIT, | ||||||
|  | 	FMAX_END | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define FEAT_SSBS		(1UL << FSSBS_BIT) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * A descriptor used to describe and configure a test case. | ||||||
|  |  * Fields with a non-trivial meaning are described inline in the following. | ||||||
|  |  */ | ||||||
|  | struct tdescr { | ||||||
|  | 	/* KEEP THIS FIELD FIRST for easier lookup from assembly */ | ||||||
|  | 	void			*token; | ||||||
|  | 	/* when disabled token based sanity checking is skipped in handler */ | ||||||
|  | 	bool			sanity_disabled; | ||||||
|  | 	/* just a name for the test-case; manadatory field */ | ||||||
|  | 	char			*name; | ||||||
|  | 	char			*descr; | ||||||
|  | 	unsigned long		feats_required; | ||||||
|  | 	/* bitmask of effectively supported feats: populated at run-time */ | ||||||
|  | 	unsigned long		feats_supported; | ||||||
|  | 	bool			initialized; | ||||||
|  | 	unsigned int		minsigstksz; | ||||||
|  | 	/* signum used as a test trigger. Zero if no trigger-signal is used */ | ||||||
|  | 	int			sig_trig; | ||||||
|  | 	/*
 | ||||||
|  | 	 * signum considered as a successful test completion. | ||||||
|  | 	 * Zero when no signal is expected on success | ||||||
|  | 	 */ | ||||||
|  | 	int			sig_ok; | ||||||
|  | 	/* signum expected on unsupported CPU features. */ | ||||||
|  | 	int			sig_unsupp; | ||||||
|  | 	/* a timeout in second for test completion */ | ||||||
|  | 	unsigned int		timeout; | ||||||
|  | 	bool			triggered; | ||||||
|  | 	bool			pass; | ||||||
|  | 	unsigned int		result; | ||||||
|  | 	/* optional sa_flags for the installed handler */ | ||||||
|  | 	int			sa_flags; | ||||||
|  | 	ucontext_t		saved_uc; | ||||||
|  | 	/* used by get_current_ctx() */ | ||||||
|  | 	size_t			live_sz; | ||||||
|  | 	ucontext_t		*live_uc; | ||||||
|  | 	volatile sig_atomic_t	live_uc_valid; | ||||||
|  | 	/* optional test private data */ | ||||||
|  | 	void			*priv; | ||||||
|  | 
 | ||||||
|  | 	/* a custom setup: called alternatively to default_setup */ | ||||||
|  | 	int (*setup)(struct tdescr *td); | ||||||
|  | 	/* a custom init: called by default test init after test_setup */ | ||||||
|  | 	bool (*init)(struct tdescr *td); | ||||||
|  | 	/* a custom cleanup function called before test exits */ | ||||||
|  | 	void (*cleanup)(struct tdescr *td); | ||||||
|  | 	/* an optional function to be used as a trigger for starting test */ | ||||||
|  | 	int (*trigger)(struct tdescr *td); | ||||||
|  | 	/*
 | ||||||
|  | 	 * the actual test-core: invoked differently depending on the | ||||||
|  | 	 * presence of the trigger function above; this is mandatory | ||||||
|  | 	 */ | ||||||
|  | 	int (*run)(struct tdescr *td, siginfo_t *si, ucontext_t *uc); | ||||||
|  | 	/* an optional function for custom results' processing */ | ||||||
|  | 	void (*check_result)(struct tdescr *td); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern struct tdescr tde; | ||||||
|  | #endif | ||||||
							
								
								
									
										328
									
								
								tools/testing/selftests/arm64/signal/test_signals_utils.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										328
									
								
								tools/testing/selftests/arm64/signal/test_signals_utils.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,328 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | 
 | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <signal.h> | ||||||
|  | #include <string.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <assert.h> | ||||||
|  | #include <sys/auxv.h> | ||||||
|  | #include <linux/auxvec.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include <asm/unistd.h> | ||||||
|  | 
 | ||||||
|  | #include <kselftest.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals.h" | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases/testcases.h" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | extern struct tdescr *current; | ||||||
|  | 
 | ||||||
|  | static int sig_copyctx = SIGTRAP; | ||||||
|  | 
 | ||||||
|  | static char const *const feats_names[FMAX_END] = { | ||||||
|  | 	" SSBS ", | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define MAX_FEATS_SZ	128 | ||||||
|  | static char feats_string[MAX_FEATS_SZ]; | ||||||
|  | 
 | ||||||
|  | static inline char *feats_to_string(unsigned long feats) | ||||||
|  | { | ||||||
|  | 	size_t flen = MAX_FEATS_SZ - 1; | ||||||
|  | 
 | ||||||
|  | 	for (int i = 0; i < FMAX_END; i++) { | ||||||
|  | 		if (feats & (1UL << i)) { | ||||||
|  | 			size_t tlen = strlen(feats_names[i]); | ||||||
|  | 
 | ||||||
|  | 			assert(flen > tlen); | ||||||
|  | 			flen -= tlen; | ||||||
|  | 			strncat(feats_string, feats_names[i], flen); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return feats_string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void unblock_signal(int signum) | ||||||
|  | { | ||||||
|  | 	sigset_t sset; | ||||||
|  | 
 | ||||||
|  | 	sigemptyset(&sset); | ||||||
|  | 	sigaddset(&sset, signum); | ||||||
|  | 	sigprocmask(SIG_UNBLOCK, &sset, NULL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void default_result(struct tdescr *td, bool force_exit) | ||||||
|  | { | ||||||
|  | 	if (td->result == KSFT_SKIP) { | ||||||
|  | 		fprintf(stderr, "==>> completed. SKIP.\n"); | ||||||
|  | 	} else if (td->pass) { | ||||||
|  | 		fprintf(stderr, "==>> completed. PASS(1)\n"); | ||||||
|  | 		td->result = KSFT_PASS; | ||||||
|  | 	} else { | ||||||
|  | 		fprintf(stdout, "==>> completed. FAIL(0)\n"); | ||||||
|  | 		td->result = KSFT_FAIL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (force_exit) | ||||||
|  | 		exit(td->result); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * The following handle_signal_* helpers are used by main default_handler | ||||||
|  |  * and are meant to return true when signal is handled successfully: | ||||||
|  |  * when false is returned instead, it means that the signal was somehow | ||||||
|  |  * unexpected in that context and it was NOT handled; default_handler will | ||||||
|  |  * take care of such unexpected situations. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | static bool handle_signal_unsupported(struct tdescr *td, | ||||||
|  | 				      siginfo_t *si, void *uc) | ||||||
|  | { | ||||||
|  | 	if (feats_ok(td)) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	/* Mangling PC to avoid loops on original SIGILL */ | ||||||
|  | 	((ucontext_t *)uc)->uc_mcontext.pc += 4; | ||||||
|  | 
 | ||||||
|  | 	if (!td->initialized) { | ||||||
|  | 		fprintf(stderr, | ||||||
|  | 			"Got SIG_UNSUPP @test_init. Ignore.\n"); | ||||||
|  | 	} else { | ||||||
|  | 		fprintf(stderr, | ||||||
|  | 			"-- RX SIG_UNSUPP on unsupported feat...OK\n"); | ||||||
|  | 		td->pass = 1; | ||||||
|  | 		default_result(current, 1); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool handle_signal_trigger(struct tdescr *td, | ||||||
|  | 				  siginfo_t *si, void *uc) | ||||||
|  | { | ||||||
|  | 	td->triggered = 1; | ||||||
|  | 	/* ->run was asserted NON-NULL in test_setup() already */ | ||||||
|  | 	td->run(td, si, uc); | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool handle_signal_ok(struct tdescr *td, | ||||||
|  | 			     siginfo_t *si, void *uc) | ||||||
|  | { | ||||||
|  | 	/*
 | ||||||
|  | 	 * it's a bug in the test code when this assert fail: | ||||||
|  | 	 * if sig_trig was defined, it must have been used before getting here. | ||||||
|  | 	 */ | ||||||
|  | 	assert(!td->sig_trig || td->triggered); | ||||||
|  | 	fprintf(stderr, | ||||||
|  | 		"SIG_OK -- SP:0x%llX  si_addr@:%p  si_code:%d  token@:%p  offset:%ld\n", | ||||||
|  | 		((ucontext_t *)uc)->uc_mcontext.sp, | ||||||
|  | 		si->si_addr, si->si_code, td->token, td->token - si->si_addr); | ||||||
|  | 	/*
 | ||||||
|  | 	 * fake_sigreturn tests, which have sanity_enabled=1, set, at the very | ||||||
|  | 	 * last time, the token field to the SP address used to place the fake | ||||||
|  | 	 * sigframe: so token==0 means we never made it to the end, | ||||||
|  | 	 * segfaulting well-before, and the test is possibly broken. | ||||||
|  | 	 */ | ||||||
|  | 	if (!td->sanity_disabled && !td->token) { | ||||||
|  | 		fprintf(stdout, | ||||||
|  | 			"current->token ZEROED...test is probably broken!\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	/*
 | ||||||
|  | 	 * Trying to narrow down the SEGV to the ones generated by Kernel itself | ||||||
|  | 	 * via arm64_notify_segfault(). This is a best-effort check anyway, and | ||||||
|  | 	 * the si_code check may need to change if this aspect of the kernel | ||||||
|  | 	 * ABI changes. | ||||||
|  | 	 */ | ||||||
|  | 	if (td->sig_ok == SIGSEGV && si->si_code != SEGV_ACCERR) { | ||||||
|  | 		fprintf(stdout, | ||||||
|  | 			"si_code != SEGV_ACCERR...test is probably broken!\n"); | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 	td->pass = 1; | ||||||
|  | 	/*
 | ||||||
|  | 	 * Some tests can lead to SEGV loops: in such a case we want to | ||||||
|  | 	 * terminate immediately exiting straight away; some others are not | ||||||
|  | 	 * supposed to outlive the signal handler code, due to the content of | ||||||
|  | 	 * the fake sigframe which caused the signal itself. | ||||||
|  | 	 */ | ||||||
|  | 	default_result(current, 1); | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool handle_signal_copyctx(struct tdescr *td, | ||||||
|  | 				  siginfo_t *si, void *uc) | ||||||
|  | { | ||||||
|  | 	/* Mangling PC to avoid loops on original BRK instr */ | ||||||
|  | 	((ucontext_t *)uc)->uc_mcontext.pc += 4; | ||||||
|  | 	memcpy(td->live_uc, uc, td->live_sz); | ||||||
|  | 	ASSERT_GOOD_CONTEXT(td->live_uc); | ||||||
|  | 	td->live_uc_valid = 1; | ||||||
|  | 	fprintf(stderr, | ||||||
|  | 		"GOOD CONTEXT grabbed from sig_copyctx handler\n"); | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void default_handler(int signum, siginfo_t *si, void *uc) | ||||||
|  | { | ||||||
|  | 	if (current->sig_unsupp && signum == current->sig_unsupp && | ||||||
|  | 	    handle_signal_unsupported(current, si, uc)) { | ||||||
|  | 		fprintf(stderr, "Handled SIG_UNSUPP\n"); | ||||||
|  | 	} else if (current->sig_trig && signum == current->sig_trig && | ||||||
|  | 		   handle_signal_trigger(current, si, uc)) { | ||||||
|  | 		fprintf(stderr, "Handled SIG_TRIG\n"); | ||||||
|  | 	} else if (current->sig_ok && signum == current->sig_ok && | ||||||
|  | 		   handle_signal_ok(current, si, uc)) { | ||||||
|  | 		fprintf(stderr, "Handled SIG_OK\n"); | ||||||
|  | 	} else if (signum == sig_copyctx && current->live_uc && | ||||||
|  | 		   handle_signal_copyctx(current, si, uc)) { | ||||||
|  | 		fprintf(stderr, "Handled SIG_COPYCTX\n"); | ||||||
|  | 	} else { | ||||||
|  | 		if (signum == SIGALRM && current->timeout) { | ||||||
|  | 			fprintf(stderr, "-- Timeout !\n"); | ||||||
|  | 		} else { | ||||||
|  | 			fprintf(stderr, | ||||||
|  | 				"-- RX UNEXPECTED SIGNAL: %d\n", signum); | ||||||
|  | 		} | ||||||
|  | 		default_result(current, 1); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int default_setup(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	struct sigaction sa; | ||||||
|  | 
 | ||||||
|  | 	sa.sa_sigaction = default_handler; | ||||||
|  | 	sa.sa_flags = SA_SIGINFO | SA_RESTART; | ||||||
|  | 	sa.sa_flags |= td->sa_flags; | ||||||
|  | 	sigemptyset(&sa.sa_mask); | ||||||
|  | 	/* uncatchable signals naturally skipped ... */ | ||||||
|  | 	for (int sig = 1; sig < 32; sig++) | ||||||
|  | 		sigaction(sig, &sa, NULL); | ||||||
|  | 	/*
 | ||||||
|  | 	 * RT Signals default disposition is Term but they cannot be | ||||||
|  | 	 * generated by the Kernel in response to our tests; so just catch | ||||||
|  | 	 * them all and report them as UNEXPECTED signals. | ||||||
|  | 	 */ | ||||||
|  | 	for (int sig = SIGRTMIN; sig <= SIGRTMAX; sig++) | ||||||
|  | 		sigaction(sig, &sa, NULL); | ||||||
|  | 
 | ||||||
|  | 	/* just in case...unblock explicitly all we need */ | ||||||
|  | 	if (td->sig_trig) | ||||||
|  | 		unblock_signal(td->sig_trig); | ||||||
|  | 	if (td->sig_ok) | ||||||
|  | 		unblock_signal(td->sig_ok); | ||||||
|  | 	if (td->sig_unsupp) | ||||||
|  | 		unblock_signal(td->sig_unsupp); | ||||||
|  | 
 | ||||||
|  | 	if (td->timeout) { | ||||||
|  | 		unblock_signal(SIGALRM); | ||||||
|  | 		alarm(td->timeout); | ||||||
|  | 	} | ||||||
|  | 	fprintf(stderr, "Registered handlers for all signals.\n"); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int default_trigger(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	return !raise(td->sig_trig); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int test_init(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	if (td->sig_trig == sig_copyctx) { | ||||||
|  | 		fprintf(stdout, | ||||||
|  | 			"Signal %d is RESERVED, cannot be used as a trigger. Aborting\n", | ||||||
|  | 			sig_copyctx); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	/* just in case */ | ||||||
|  | 	unblock_signal(sig_copyctx); | ||||||
|  | 
 | ||||||
|  | 	td->minsigstksz = getauxval(AT_MINSIGSTKSZ); | ||||||
|  | 	if (!td->minsigstksz) | ||||||
|  | 		td->minsigstksz = MINSIGSTKSZ; | ||||||
|  | 	fprintf(stderr, "Detected MINSTKSIGSZ:%d\n", td->minsigstksz); | ||||||
|  | 
 | ||||||
|  | 	if (td->feats_required) { | ||||||
|  | 		td->feats_supported = 0; | ||||||
|  | 		/*
 | ||||||
|  | 		 * Checking for CPU required features using both the | ||||||
|  | 		 * auxval and the arm64 MRS Emulation to read sysregs. | ||||||
|  | 		 */ | ||||||
|  | 		if (getauxval(AT_HWCAP) & HWCAP_SSBS) | ||||||
|  | 			td->feats_supported |= FEAT_SSBS; | ||||||
|  | 		if (feats_ok(td)) | ||||||
|  | 			fprintf(stderr, | ||||||
|  | 				"Required Features: [%s] supported\n", | ||||||
|  | 				feats_to_string(td->feats_required & | ||||||
|  | 						td->feats_supported)); | ||||||
|  | 		else | ||||||
|  | 			fprintf(stderr, | ||||||
|  | 				"Required Features: [%s] NOT supported\n", | ||||||
|  | 				feats_to_string(td->feats_required & | ||||||
|  | 						~td->feats_supported)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Perform test specific additional initialization */ | ||||||
|  | 	if (td->init && !td->init(td)) { | ||||||
|  | 		fprintf(stderr, "FAILED Testcase initialization.\n"); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	td->initialized = 1; | ||||||
|  | 	fprintf(stderr, "Testcase initialized.\n"); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int test_setup(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	/* assert core invariants symptom of a rotten testcase */ | ||||||
|  | 	assert(current); | ||||||
|  | 	assert(td); | ||||||
|  | 	assert(td->name); | ||||||
|  | 	assert(td->run); | ||||||
|  | 
 | ||||||
|  | 	/* Default result is FAIL if test setup fails */ | ||||||
|  | 	td->result = KSFT_FAIL; | ||||||
|  | 	if (td->setup) | ||||||
|  | 		return td->setup(td); | ||||||
|  | 	else | ||||||
|  | 		return default_setup(td); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int test_run(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	if (td->sig_trig) { | ||||||
|  | 		if (td->trigger) | ||||||
|  | 			return td->trigger(td); | ||||||
|  | 		else | ||||||
|  | 			return default_trigger(td); | ||||||
|  | 	} else { | ||||||
|  | 		return td->run(td, NULL, NULL); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void test_result(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	if (td->initialized && td->result != KSFT_SKIP && td->check_result) | ||||||
|  | 		td->check_result(td); | ||||||
|  | 	default_result(td, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void test_cleanup(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	if (td->cleanup) | ||||||
|  | 		td->cleanup(td); | ||||||
|  | } | ||||||
							
								
								
									
										120
									
								
								tools/testing/selftests/arm64/signal/test_signals_utils.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								tools/testing/selftests/arm64/signal/test_signals_utils.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,120 @@ | ||||||
|  | /* SPDX-License-Identifier: GPL-2.0 */ | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | 
 | ||||||
|  | #ifndef __TEST_SIGNALS_UTILS_H__ | ||||||
|  | #define __TEST_SIGNALS_UTILS_H__ | ||||||
|  | 
 | ||||||
|  | #include <assert.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <string.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals.h" | ||||||
|  | 
 | ||||||
|  | int test_init(struct tdescr *td); | ||||||
|  | int test_setup(struct tdescr *td); | ||||||
|  | void test_cleanup(struct tdescr *td); | ||||||
|  | int test_run(struct tdescr *td); | ||||||
|  | void test_result(struct tdescr *td); | ||||||
|  | 
 | ||||||
|  | static inline bool feats_ok(struct tdescr *td) | ||||||
|  | { | ||||||
|  | 	return (td->feats_required & td->feats_supported) == td->feats_required; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Obtaining a valid and full-blown ucontext_t from userspace is tricky: | ||||||
|  |  * libc getcontext does() not save all the regs and messes with some of | ||||||
|  |  * them (pstate value in particular is not reliable). | ||||||
|  |  * | ||||||
|  |  * Here we use a service signal to grab the ucontext_t from inside a | ||||||
|  |  * dedicated signal handler, since there, it is populated by Kernel | ||||||
|  |  * itself in setup_sigframe(). The grabbed context is then stored and | ||||||
|  |  * made available in td->live_uc. | ||||||
|  |  * | ||||||
|  |  * As service-signal is used a SIGTRAP induced by a 'brk' instruction, | ||||||
|  |  * because here we have to avoid syscalls to trigger the signal since | ||||||
|  |  * they would cause any SVE sigframe content (if any) to be removed. | ||||||
|  |  * | ||||||
|  |  * Anyway this function really serves a dual purpose: | ||||||
|  |  * | ||||||
|  |  * 1. grab a valid sigcontext into td->live_uc for result analysis: in | ||||||
|  |  * such case it returns 1. | ||||||
|  |  * | ||||||
|  |  * 2. detect if, somehow, a previously grabbed live_uc context has been | ||||||
|  |  * used actively with a sigreturn: in such a case the execution would have | ||||||
|  |  * magically resumed in the middle of this function itself (seen_already==1): | ||||||
|  |  * in such a case return 0, since in fact we have not just simply grabbed | ||||||
|  |  * the context. | ||||||
|  |  * | ||||||
|  |  * This latter case is useful to detect when a fake_sigreturn test-case has | ||||||
|  |  * unexpectedly survived without hitting a SEGV. | ||||||
|  |  * | ||||||
|  |  * Note that the case of runtime dynamically sized sigframes (like in SVE | ||||||
|  |  * context) is still NOT addressed: sigframe size is supposed to be fixed | ||||||
|  |  * at sizeof(ucontext_t). | ||||||
|  |  */ | ||||||
|  | static __always_inline bool get_current_context(struct tdescr *td, | ||||||
|  | 						ucontext_t *dest_uc) | ||||||
|  | { | ||||||
|  | 	static volatile bool seen_already; | ||||||
|  | 
 | ||||||
|  | 	assert(td && dest_uc); | ||||||
|  | 	/* it's a genuine invocation..reinit */ | ||||||
|  | 	seen_already = 0; | ||||||
|  | 	td->live_uc_valid = 0; | ||||||
|  | 	td->live_sz = sizeof(*dest_uc); | ||||||
|  | 	memset(dest_uc, 0x00, td->live_sz); | ||||||
|  | 	td->live_uc = dest_uc; | ||||||
|  | 	/*
 | ||||||
|  | 	 * Grab ucontext_t triggering a SIGTRAP. | ||||||
|  | 	 * | ||||||
|  | 	 * Note that: | ||||||
|  | 	 * - live_uc_valid is declared volatile sig_atomic_t in | ||||||
|  | 	 *   struct tdescr since it will be changed inside the | ||||||
|  | 	 *   sig_copyctx handler | ||||||
|  | 	 * - the additional 'memory' clobber is there to avoid possible | ||||||
|  | 	 *   compiler's assumption on live_uc_valid and the content | ||||||
|  | 	 *   pointed by dest_uc, which are all changed inside the signal | ||||||
|  | 	 *   handler | ||||||
|  | 	 * - BRK causes a debug exception which is handled by the Kernel | ||||||
|  | 	 *   and finally causes the SIGTRAP signal to be delivered to this | ||||||
|  | 	 *   test thread. Since such delivery happens on the ret_to_user() | ||||||
|  | 	 *   /do_notify_resume() debug exception return-path, we are sure | ||||||
|  | 	 *   that the registered SIGTRAP handler has been run to completion | ||||||
|  | 	 *   before the execution path is restored here: as a consequence | ||||||
|  | 	 *   we can be sure that the volatile sig_atomic_t live_uc_valid | ||||||
|  | 	 *   carries a meaningful result. Being in a single thread context | ||||||
|  | 	 *   we'll also be sure that any access to memory modified by the | ||||||
|  | 	 *   handler (namely ucontext_t) will be visible once returned. | ||||||
|  | 	 * - note that since we are using a breakpoint instruction here | ||||||
|  | 	 *   to cause a SIGTRAP, the ucontext_t grabbed from the signal | ||||||
|  | 	 *   handler would naturally contain a PC pointing exactly to this | ||||||
|  | 	 *   BRK line, which means that, on return from the signal handler, | ||||||
|  | 	 *   or if we place the ucontext_t on the stack to fake a sigreturn, | ||||||
|  | 	 *   we'll end up in an infinite loop of BRK-SIGTRAP-handler. | ||||||
|  | 	 *   For this reason we take care to artificially move forward the | ||||||
|  | 	 *   PC to the next instruction while inside the signal handler. | ||||||
|  | 	 */ | ||||||
|  | 	asm volatile ("brk #666" | ||||||
|  | 		      : "+m" (*dest_uc) | ||||||
|  | 		      : | ||||||
|  | 		      : "memory"); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * If we get here with seen_already==1 it implies the td->live_uc | ||||||
|  | 	 * context has been used to get back here....this probably means | ||||||
|  | 	 * a test has failed to cause a SEGV...anyway live_uc does not | ||||||
|  | 	 * point to a just acquired copy of ucontext_t...so return 0 | ||||||
|  | 	 */ | ||||||
|  | 	if (seen_already) { | ||||||
|  | 		fprintf(stdout, | ||||||
|  | 			"Unexpected successful sigreturn detected: live_uc is stale !\n"); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	seen_already = 1; | ||||||
|  | 
 | ||||||
|  | 	return td->live_uc_valid; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int fake_sigreturn(void *sigframe, size_t sz, int misalign_bytes); | ||||||
|  | #endif | ||||||
|  | @ -0,0 +1,52 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack including a BAD Unknown magic | ||||||
|  |  * record: on sigreturn Kernel must spot this attempt and the test | ||||||
|  |  * case is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_bad_magic_run(struct tdescr *td, | ||||||
|  | 					siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	struct _aarch64_ctx *shead = GET_SF_RESV_HEAD(sf), *head; | ||||||
|  | 
 | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	/* need at least 2*HDR_SZ space: KSFT_BAD_MAGIC + terminator. */ | ||||||
|  | 	head = get_starting_head(shead, HDR_SZ * 2, GET_SF_RESV_SIZE(sf), NULL); | ||||||
|  | 	if (!head) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * use a well known NON existent bad magic...something | ||||||
|  | 	 * we should pretty sure won't be ever defined in Kernel | ||||||
|  | 	 */ | ||||||
|  | 	head->magic = KSFT_BAD_MAGIC; | ||||||
|  | 	head->size = HDR_SZ; | ||||||
|  | 	write_terminator_record(GET_RESV_NEXT_HEAD(head)); | ||||||
|  | 
 | ||||||
|  | 	ASSERT_BAD_CONTEXT(&sf.uc); | ||||||
|  | 	fake_sigreturn(&sf, sizeof(sf), 0); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_BAD_MAGIC", | ||||||
|  | 		.descr = "Trigger a sigreturn with a sigframe with a bad magic", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_bad_magic_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,77 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack including a bad record overflowing | ||||||
|  |  * the __reserved space: on sigreturn Kernel must spot this attempt and | ||||||
|  |  * the test case is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | #define MIN_SZ_ALIGN	16 | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_bad_size_run(struct tdescr *td, | ||||||
|  | 				       siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	size_t resv_sz, need_sz, offset; | ||||||
|  | 	struct _aarch64_ctx *shead = GET_SF_RESV_HEAD(sf), *head; | ||||||
|  | 
 | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	resv_sz = GET_SF_RESV_SIZE(sf); | ||||||
|  | 	/* at least HDR_SZ + bad sized esr_context needed */ | ||||||
|  | 	need_sz = sizeof(struct esr_context) + HDR_SZ; | ||||||
|  | 	head = get_starting_head(shead, need_sz, resv_sz, &offset); | ||||||
|  | 	if (!head) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Use an esr_context to build a fake header with a | ||||||
|  | 	 * size greater then the free __reserved area minus HDR_SZ; | ||||||
|  | 	 * using ESR_MAGIC here since it is not checked for size nor | ||||||
|  | 	 * is limited to one instance. | ||||||
|  | 	 * | ||||||
|  | 	 * At first inject an additional normal esr_context | ||||||
|  | 	 */ | ||||||
|  | 	head->magic = ESR_MAGIC; | ||||||
|  | 	head->size = sizeof(struct esr_context); | ||||||
|  | 	/* and terminate properly */ | ||||||
|  | 	write_terminator_record(GET_RESV_NEXT_HEAD(head)); | ||||||
|  | 	ASSERT_GOOD_CONTEXT(&sf.uc); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * now mess with fake esr_context size: leaving less space than | ||||||
|  | 	 * needed while keeping size value 16-aligned | ||||||
|  | 	 * | ||||||
|  | 	 * It must trigger a SEGV from Kernel on: | ||||||
|  | 	 * | ||||||
|  | 	 *	resv_sz - offset < sizeof(*head) | ||||||
|  | 	 */ | ||||||
|  | 	/* at first set the maximum good 16-aligned size */ | ||||||
|  | 	head->size = (resv_sz - offset - need_sz + MIN_SZ_ALIGN) & ~0xfUL; | ||||||
|  | 	/* plus a bit more of 16-aligned sized stuff */ | ||||||
|  | 	head->size += MIN_SZ_ALIGN; | ||||||
|  | 	/* and terminate properly */ | ||||||
|  | 	write_terminator_record(GET_RESV_NEXT_HEAD(head)); | ||||||
|  | 	ASSERT_BAD_CONTEXT(&sf.uc); | ||||||
|  | 	fake_sigreturn(&sf, sizeof(sf), 0); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_BAD_SIZE", | ||||||
|  | 		.descr = "Triggers a sigreturn with a overrun __reserved area", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_bad_size_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,46 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack including a badly sized terminator | ||||||
|  |  * record: on sigreturn Kernel must spot this attempt and the test case | ||||||
|  |  * is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_bad_size_for_magic0_run(struct tdescr *td, | ||||||
|  | 						  siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	struct _aarch64_ctx *shead = GET_SF_RESV_HEAD(sf), *head; | ||||||
|  | 
 | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	/* at least HDR_SZ for the badly sized terminator. */ | ||||||
|  | 	head = get_starting_head(shead, HDR_SZ, GET_SF_RESV_SIZE(sf), NULL); | ||||||
|  | 	if (!head) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	head->magic = 0; | ||||||
|  | 	head->size = HDR_SZ; | ||||||
|  | 	ASSERT_BAD_CONTEXT(&sf.uc); | ||||||
|  | 	fake_sigreturn(&sf, sizeof(sf), 0); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_BAD_SIZE_FOR_TERMINATOR", | ||||||
|  | 		.descr = "Trigger a sigreturn using non-zero size terminator", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_bad_size_for_magic0_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack including an additional FPSIMD | ||||||
|  |  * record: on sigreturn Kernel must spot this attempt and the test | ||||||
|  |  * case is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_duplicated_fpsimd_run(struct tdescr *td, | ||||||
|  | 						siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	struct _aarch64_ctx *shead = GET_SF_RESV_HEAD(sf), *head; | ||||||
|  | 
 | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	head = get_starting_head(shead, sizeof(struct fpsimd_context) + HDR_SZ, | ||||||
|  | 				 GET_SF_RESV_SIZE(sf), NULL); | ||||||
|  | 	if (!head) | ||||||
|  | 		return 0; | ||||||
|  | 
 | ||||||
|  | 	/* Add a spurious fpsimd_context */ | ||||||
|  | 	head->magic = FPSIMD_MAGIC; | ||||||
|  | 	head->size = sizeof(struct fpsimd_context); | ||||||
|  | 	/* and terminate */ | ||||||
|  | 	write_terminator_record(GET_RESV_NEXT_HEAD(head)); | ||||||
|  | 
 | ||||||
|  | 	ASSERT_BAD_CONTEXT(&sf.uc); | ||||||
|  | 	fake_sigreturn(&sf, sizeof(sf), 0); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_DUPLICATED_FPSIMD", | ||||||
|  | 		.descr = "Triggers a sigreturn including two fpsimd_context", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_duplicated_fpsimd_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,37 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack at a misaligned SP: on sigreturn | ||||||
|  |  * Kernel must spot this attempt and the test case is expected to be | ||||||
|  |  * terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_misaligned_run(struct tdescr *td, | ||||||
|  | 					 siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	/* Forcing sigframe on misaligned SP (16 + 3) */ | ||||||
|  | 	fake_sigreturn(&sf, sizeof(sf), 3); | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_MISALIGNED_SP", | ||||||
|  | 		.descr = "Triggers a sigreturn with a misaligned sigframe", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_misaligned_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Place a fake sigframe on the stack missing the mandatory FPSIMD | ||||||
|  |  * record: on sigreturn Kernel must spot this attempt and the test | ||||||
|  |  * case is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <signal.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe sf; | ||||||
|  | 
 | ||||||
|  | static int fake_sigreturn_missing_fpsimd_run(struct tdescr *td, | ||||||
|  | 					     siginfo_t *si, ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	size_t resv_sz, offset; | ||||||
|  | 	struct _aarch64_ctx *head = GET_SF_RESV_HEAD(sf); | ||||||
|  | 
 | ||||||
|  | 	/* just to fill the ucontext_t with something real */ | ||||||
|  | 	if (!get_current_context(td, &sf.uc)) | ||||||
|  | 		return 1; | ||||||
|  | 
 | ||||||
|  | 	resv_sz = GET_SF_RESV_SIZE(sf); | ||||||
|  | 	head = get_header(head, FPSIMD_MAGIC, resv_sz, &offset); | ||||||
|  | 	if (head && resv_sz - offset >= HDR_SZ) { | ||||||
|  | 		fprintf(stderr, "Mangling template header. Spare space:%zd\n", | ||||||
|  | 			resv_sz - offset); | ||||||
|  | 		/* Just overwrite fpsmid_context */ | ||||||
|  | 		write_terminator_record(head); | ||||||
|  | 
 | ||||||
|  | 		ASSERT_BAD_CONTEXT(&sf.uc); | ||||||
|  | 		fake_sigreturn(&sf, sizeof(sf), 0); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.name = "FAKE_SIGRETURN_MISSING_FPSIMD", | ||||||
|  | 		.descr = "Triggers a sigreturn with a missing fpsimd_context", | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.timeout = 3, | ||||||
|  | 		.run = fake_sigreturn_missing_fpsimd_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,31 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the execution state bit: this attempt must be spotted by Kernel and | ||||||
|  |  * the test case is expected to be terminated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | static int mangle_invalid_pstate_run(struct tdescr *td, siginfo_t *si, | ||||||
|  | 				     ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	ASSERT_GOOD_CONTEXT(uc); | ||||||
|  | 
 | ||||||
|  | 	/* This config should trigger a SIGSEGV by Kernel */ | ||||||
|  | 	uc->uc_mcontext.pstate ^= PSR_MODE32_BIT; | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.sanity_disabled = true, | ||||||
|  | 		.name = "MANGLE_PSTATE_INVALID_STATE_TOGGLE", | ||||||
|  | 		.descr = "Mangling uc_mcontext with INVALID STATE_TOGGLE", | ||||||
|  | 		.sig_trig = SIGUSR1, | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.run = mangle_invalid_pstate_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,35 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, mangling the | ||||||
|  |  * DAIF bits in an illegal manner: this attempt must be spotted by Kernel | ||||||
|  |  * and the test case is expected to be terminated via SEGV. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | static int mangle_invalid_pstate_run(struct tdescr *td, siginfo_t *si, | ||||||
|  | 				     ucontext_t *uc) | ||||||
|  | { | ||||||
|  | 	ASSERT_GOOD_CONTEXT(uc); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * This config should trigger a SIGSEGV by Kernel when it checks | ||||||
|  | 	 * the sigframe consistency in valid_user_regs() routine. | ||||||
|  | 	 */ | ||||||
|  | 	uc->uc_mcontext.pstate |= PSR_D_BIT | PSR_A_BIT | PSR_I_BIT | PSR_F_BIT; | ||||||
|  | 
 | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct tdescr tde = { | ||||||
|  | 		.sanity_disabled = true, | ||||||
|  | 		.name = "MANGLE_PSTATE_INVALID_DAIF_BITS", | ||||||
|  | 		.descr = "Mangling uc_mcontext with INVALID DAIF_BITS", | ||||||
|  | 		.sig_trig = SIGUSR1, | ||||||
|  | 		.sig_ok = SIGSEGV, | ||||||
|  | 		.run = mangle_invalid_pstate_run, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(1h); | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(1t); | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(2h); | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(2t); | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(3h); | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Try to mangle the ucontext from inside a signal handler, toggling | ||||||
|  |  * the mode bit to escalate exception level: this attempt must be spotted | ||||||
|  |  * by Kernel and the test case is expected to be termninated via SEGV. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_signals_utils.h" | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | #include "mangle_pstate_invalid_mode_template.h" | ||||||
|  | 
 | ||||||
|  | DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(3t); | ||||||
|  | @ -0,0 +1,28 @@ | ||||||
|  | /* SPDX-License-Identifier: GPL-2.0 */ | ||||||
|  | /*
 | ||||||
|  |  * Copyright (C) 2019 ARM Limited | ||||||
|  |  * | ||||||
|  |  * Utility macro to ease definition of testcases toggling mode EL | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define DEFINE_TESTCASE_MANGLE_PSTATE_INVALID_MODE(_mode)		\ | ||||||
|  | 									\ | ||||||
|  | static int mangle_invalid_pstate_run(struct tdescr *td, siginfo_t *si,	\ | ||||||
|  | 				     ucontext_t *uc)			\ | ||||||
|  | {									\ | ||||||
|  | 	ASSERT_GOOD_CONTEXT(uc);					\ | ||||||
|  | 									\ | ||||||
|  | 	uc->uc_mcontext.pstate &= ~PSR_MODE_MASK;			\ | ||||||
|  | 	uc->uc_mcontext.pstate |= PSR_MODE_EL ## _mode;			\ | ||||||
|  | 									\ | ||||||
|  | 	return 1;							\ | ||||||
|  | }									\ | ||||||
|  | 									\ | ||||||
|  | struct tdescr tde = {							\ | ||||||
|  | 		.sanity_disabled = true,				\ | ||||||
|  | 		.name = "MANGLE_PSTATE_INVALID_MODE_EL"#_mode,		\ | ||||||
|  | 		.descr = "Mangling uc_mcontext INVALID MODE EL"#_mode,	\ | ||||||
|  | 		.sig_trig = SIGUSR1,					\ | ||||||
|  | 		.sig_ok = SIGSEGV,					\ | ||||||
|  | 		.run = mangle_invalid_pstate_run,			\ | ||||||
|  | } | ||||||
							
								
								
									
										196
									
								
								tools/testing/selftests/arm64/signal/testcases/testcases.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								tools/testing/selftests/arm64/signal/testcases/testcases.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,196 @@ | ||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | #include "testcases.h" | ||||||
|  | 
 | ||||||
|  | struct _aarch64_ctx *get_header(struct _aarch64_ctx *head, uint32_t magic, | ||||||
|  | 				size_t resv_sz, size_t *offset) | ||||||
|  | { | ||||||
|  | 	size_t offs = 0; | ||||||
|  | 	struct _aarch64_ctx *found = NULL; | ||||||
|  | 
 | ||||||
|  | 	if (!head || resv_sz < HDR_SZ) | ||||||
|  | 		return found; | ||||||
|  | 
 | ||||||
|  | 	while (offs <= resv_sz - HDR_SZ && | ||||||
|  | 	       head->magic != magic && head->magic) { | ||||||
|  | 		offs += head->size; | ||||||
|  | 		head = GET_RESV_NEXT_HEAD(head); | ||||||
|  | 	} | ||||||
|  | 	if (head->magic == magic) { | ||||||
|  | 		found = head; | ||||||
|  | 		if (offset) | ||||||
|  | 			*offset = offs; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return found; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool validate_extra_context(struct extra_context *extra, char **err) | ||||||
|  | { | ||||||
|  | 	struct _aarch64_ctx *term; | ||||||
|  | 
 | ||||||
|  | 	if (!extra || !err) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	fprintf(stderr, "Validating EXTRA...\n"); | ||||||
|  | 	term = GET_RESV_NEXT_HEAD(extra); | ||||||
|  | 	if (!term || term->magic || term->size) { | ||||||
|  | 		*err = "Missing terminator after EXTRA context"; | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (extra->datap & 0x0fUL) | ||||||
|  | 		*err = "Extra DATAP misaligned"; | ||||||
|  | 	else if (extra->size & 0x0fUL) | ||||||
|  | 		*err = "Extra SIZE misaligned"; | ||||||
|  | 	else if (extra->datap != (uint64_t)term + sizeof(*term)) | ||||||
|  | 		*err = "Extra DATAP misplaced (not contiguos)"; | ||||||
|  | 	if (*err) | ||||||
|  | 		return false; | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool validate_reserved(ucontext_t *uc, size_t resv_sz, char **err) | ||||||
|  | { | ||||||
|  | 	bool terminated = false; | ||||||
|  | 	size_t offs = 0; | ||||||
|  | 	int flags = 0; | ||||||
|  | 	struct extra_context *extra = NULL; | ||||||
|  | 	struct _aarch64_ctx *head = | ||||||
|  | 		(struct _aarch64_ctx *)uc->uc_mcontext.__reserved; | ||||||
|  | 
 | ||||||
|  | 	if (!err) | ||||||
|  | 		return false; | ||||||
|  | 	/* Walk till the end terminator verifying __reserved contents */ | ||||||
|  | 	while (head && !terminated && offs < resv_sz) { | ||||||
|  | 		if ((uint64_t)head & 0x0fUL) { | ||||||
|  | 			*err = "Misaligned HEAD"; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		switch (head->magic) { | ||||||
|  | 		case 0: | ||||||
|  | 			if (head->size) | ||||||
|  | 				*err = "Bad size for terminator"; | ||||||
|  | 			else | ||||||
|  | 				terminated = true; | ||||||
|  | 			break; | ||||||
|  | 		case FPSIMD_MAGIC: | ||||||
|  | 			if (flags & FPSIMD_CTX) | ||||||
|  | 				*err = "Multiple FPSIMD_MAGIC"; | ||||||
|  | 			else if (head->size != | ||||||
|  | 				 sizeof(struct fpsimd_context)) | ||||||
|  | 				*err = "Bad size for fpsimd_context"; | ||||||
|  | 			flags |= FPSIMD_CTX; | ||||||
|  | 			break; | ||||||
|  | 		case ESR_MAGIC: | ||||||
|  | 			if (head->size != sizeof(struct esr_context)) | ||||||
|  | 				*err = "Bad size for esr_context"; | ||||||
|  | 			break; | ||||||
|  | 		case SVE_MAGIC: | ||||||
|  | 			if (flags & SVE_CTX) | ||||||
|  | 				*err = "Multiple SVE_MAGIC"; | ||||||
|  | 			else if (head->size != | ||||||
|  | 				 sizeof(struct sve_context)) | ||||||
|  | 				*err = "Bad size for sve_context"; | ||||||
|  | 			flags |= SVE_CTX; | ||||||
|  | 			break; | ||||||
|  | 		case EXTRA_MAGIC: | ||||||
|  | 			if (flags & EXTRA_CTX) | ||||||
|  | 				*err = "Multiple EXTRA_MAGIC"; | ||||||
|  | 			else if (head->size != | ||||||
|  | 				 sizeof(struct extra_context)) | ||||||
|  | 				*err = "Bad size for extra_context"; | ||||||
|  | 			flags |= EXTRA_CTX; | ||||||
|  | 			extra = (struct extra_context *)head; | ||||||
|  | 			break; | ||||||
|  | 		case KSFT_BAD_MAGIC: | ||||||
|  | 			/*
 | ||||||
|  | 			 * This is a BAD magic header defined | ||||||
|  | 			 * artificially by a testcase and surely | ||||||
|  | 			 * unknown to the Kernel parse_user_sigframe(). | ||||||
|  | 			 * It MUST cause a Kernel induced SEGV | ||||||
|  | 			 */ | ||||||
|  | 			*err = "BAD MAGIC !"; | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			/*
 | ||||||
|  | 			 * A still unknown Magic: potentially freshly added | ||||||
|  | 			 * to the Kernel code and still unknown to the | ||||||
|  | 			 * tests. | ||||||
|  | 			 */ | ||||||
|  | 			fprintf(stdout, | ||||||
|  | 				"SKIP Unknown MAGIC: 0x%X - Is KSFT arm64/signal up to date ?\n", | ||||||
|  | 				head->magic); | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (*err) | ||||||
|  | 			return false; | ||||||
|  | 
 | ||||||
|  | 		offs += head->size; | ||||||
|  | 		if (resv_sz < offs + sizeof(*head)) { | ||||||
|  | 			*err = "HEAD Overrun"; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (flags & EXTRA_CTX) | ||||||
|  | 			if (!validate_extra_context(extra, err)) | ||||||
|  | 				return false; | ||||||
|  | 
 | ||||||
|  | 		head = GET_RESV_NEXT_HEAD(head); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (terminated && !(flags & FPSIMD_CTX)) { | ||||||
|  | 		*err = "Missing FPSIMD"; | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This function walks through the records inside the provided reserved area | ||||||
|  |  * trying to find enough space to fit @need_sz bytes: if not enough space is | ||||||
|  |  * available and an extra_context record is present, it throws away the | ||||||
|  |  * extra_context record. | ||||||
|  |  * | ||||||
|  |  * It returns a pointer to a new header where it is possible to start storing | ||||||
|  |  * our need_sz bytes. | ||||||
|  |  * | ||||||
|  |  * @shead: points to the start of reserved area | ||||||
|  |  * @need_sz: needed bytes | ||||||
|  |  * @resv_sz: reserved area size in bytes | ||||||
|  |  * @offset: if not null, this will be filled with the offset of the return | ||||||
|  |  *	    head pointer from @shead | ||||||
|  |  * | ||||||
|  |  * @return: pointer to a new head where to start storing need_sz bytes, or | ||||||
|  |  *	    NULL if space could not be made available. | ||||||
|  |  */ | ||||||
|  | struct _aarch64_ctx *get_starting_head(struct _aarch64_ctx *shead, | ||||||
|  | 				       size_t need_sz, size_t resv_sz, | ||||||
|  | 				       size_t *offset) | ||||||
|  | { | ||||||
|  | 	size_t offs = 0; | ||||||
|  | 	struct _aarch64_ctx *head; | ||||||
|  | 
 | ||||||
|  | 	head = get_terminator(shead, resv_sz, &offs); | ||||||
|  | 	/* not found a terminator...no need to update offset if any */ | ||||||
|  | 	if (!head) | ||||||
|  | 		return head; | ||||||
|  | 	if (resv_sz - offs < need_sz) { | ||||||
|  | 		fprintf(stderr, "Low on space:%zd. Discarding extra_context.\n", | ||||||
|  | 			resv_sz - offs); | ||||||
|  | 		head = get_header(shead, EXTRA_MAGIC, resv_sz, &offs); | ||||||
|  | 		if (!head || resv_sz - offs < need_sz) { | ||||||
|  | 			fprintf(stderr, | ||||||
|  | 				"Failed to reclaim space on sigframe.\n"); | ||||||
|  | 			return NULL; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fprintf(stderr, "Available space:%zd\n", resv_sz - offs); | ||||||
|  | 	if (offset) | ||||||
|  | 		*offset = offs; | ||||||
|  | 	return head; | ||||||
|  | } | ||||||
							
								
								
									
										104
									
								
								tools/testing/selftests/arm64/signal/testcases/testcases.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								tools/testing/selftests/arm64/signal/testcases/testcases.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,104 @@ | ||||||
|  | /* SPDX-License-Identifier: GPL-2.0 */ | ||||||
|  | /* Copyright (C) 2019 ARM Limited */ | ||||||
|  | #ifndef __TESTCASES_H__ | ||||||
|  | #define __TESTCASES_H__ | ||||||
|  | 
 | ||||||
|  | #include <stddef.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdbool.h> | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <ucontext.h> | ||||||
|  | #include <signal.h> | ||||||
|  | 
 | ||||||
|  | /* Architecture specific sigframe definitions */ | ||||||
|  | #include <asm/sigcontext.h> | ||||||
|  | 
 | ||||||
|  | #define FPSIMD_CTX	(1 << 0) | ||||||
|  | #define SVE_CTX		(1 << 1) | ||||||
|  | #define EXTRA_CTX	(1 << 2) | ||||||
|  | 
 | ||||||
|  | #define KSFT_BAD_MAGIC	0xdeadbeef | ||||||
|  | 
 | ||||||
|  | #define HDR_SZ \ | ||||||
|  | 	sizeof(struct _aarch64_ctx) | ||||||
|  | 
 | ||||||
|  | #define GET_SF_RESV_HEAD(sf) \ | ||||||
|  | 	(struct _aarch64_ctx *)(&(sf).uc.uc_mcontext.__reserved) | ||||||
|  | 
 | ||||||
|  | #define GET_SF_RESV_SIZE(sf) \ | ||||||
|  | 	sizeof((sf).uc.uc_mcontext.__reserved) | ||||||
|  | 
 | ||||||
|  | #define GET_UCP_RESV_SIZE(ucp) \ | ||||||
|  | 	sizeof((ucp)->uc_mcontext.__reserved) | ||||||
|  | 
 | ||||||
|  | #define ASSERT_BAD_CONTEXT(uc) do {					\ | ||||||
|  | 	char *err = NULL;						\ | ||||||
|  | 	if (!validate_reserved((uc), GET_UCP_RESV_SIZE((uc)), &err)) {	\ | ||||||
|  | 		if (err)						\ | ||||||
|  | 			fprintf(stderr,					\ | ||||||
|  | 				"Using badly built context - ERR: %s\n",\ | ||||||
|  | 				err);					\ | ||||||
|  | 	} else {							\ | ||||||
|  | 		abort();						\ | ||||||
|  | 	}								\ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | #define ASSERT_GOOD_CONTEXT(uc) do {					 \ | ||||||
|  | 	char *err = NULL;						 \ | ||||||
|  | 	if (!validate_reserved((uc), GET_UCP_RESV_SIZE((uc)), &err)) {	 \ | ||||||
|  | 		if (err)						 \ | ||||||
|  | 			fprintf(stderr,					 \ | ||||||
|  | 				"Detected BAD context - ERR: %s\n", err);\ | ||||||
|  | 		abort();						 \ | ||||||
|  | 	} else {							 \ | ||||||
|  | 		fprintf(stderr, "uc context validated.\n");		 \ | ||||||
|  | 	}								 \ | ||||||
|  | } while (0) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * A simple record-walker for __reserved area: it walks through assuming | ||||||
|  |  * only to find a proper struct __aarch64_ctx header descriptor. | ||||||
|  |  * | ||||||
|  |  * Instead it makes no assumptions on the content and ordering of the | ||||||
|  |  * records, any needed bounds checking must be enforced by the caller | ||||||
|  |  * if wanted: this way can be used by caller on any maliciously built bad | ||||||
|  |  * contexts. | ||||||
|  |  * | ||||||
|  |  * head->size accounts both for payload and header _aarch64_ctx size ! | ||||||
|  |  */ | ||||||
|  | #define GET_RESV_NEXT_HEAD(h) \ | ||||||
|  | 	(struct _aarch64_ctx *)((char *)(h) + (h)->size) | ||||||
|  | 
 | ||||||
|  | struct fake_sigframe { | ||||||
|  | 	siginfo_t	info; | ||||||
|  | 	ucontext_t	uc; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | bool validate_reserved(ucontext_t *uc, size_t resv_sz, char **err); | ||||||
|  | 
 | ||||||
|  | bool validate_extra_context(struct extra_context *extra, char **err); | ||||||
|  | 
 | ||||||
|  | struct _aarch64_ctx *get_header(struct _aarch64_ctx *head, uint32_t magic, | ||||||
|  | 				size_t resv_sz, size_t *offset); | ||||||
|  | 
 | ||||||
|  | static inline struct _aarch64_ctx *get_terminator(struct _aarch64_ctx *head, | ||||||
|  | 						  size_t resv_sz, | ||||||
|  | 						  size_t *offset) | ||||||
|  | { | ||||||
|  | 	return get_header(head, 0, resv_sz, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void write_terminator_record(struct _aarch64_ctx *tail) | ||||||
|  | { | ||||||
|  | 	if (tail) { | ||||||
|  | 		tail->magic = 0; | ||||||
|  | 		tail->size = 0; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct _aarch64_ctx *get_starting_head(struct _aarch64_ctx *shead, | ||||||
|  | 				       size_t need_sz, size_t resv_sz, | ||||||
|  | 				       size_t *offset); | ||||||
|  | #endif | ||||||
							
								
								
									
										7
									
								
								tools/testing/selftests/arm64/tags/Makefile
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								tools/testing/selftests/arm64/tags/Makefile
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | ||||||
|  | # SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | 
 | ||||||
|  | CFLAGS += -I../../../../../usr/include/ | ||||||
|  | TEST_GEN_PROGS := tags_test | ||||||
|  | TEST_PROGS := run_tags_test.sh | ||||||
|  | 
 | ||||||
|  | include ../../lib.mk | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 Catalin Marinas
						Catalin Marinas