Pull x86 core updates from Thomas Gleixner: - Limit the hardcoded topology quirk for Hygon CPUs to those which have a model ID less than 4. The newer models have the topology CPUID leaf 0xB correctly implemented and are not affected. - Make SMT control more robust against enumeration failures SMT control was added to allow controlling SMT at boottime or runtime. The primary purpose was to provide a simple mechanism to disable SMT in the light of speculation attack vectors. It turned out that the code is sensible to enumeration failures and worked only by chance for XEN/PV. XEN/PV has no real APIC enumeration which means the primary thread mask is not set up correctly. By chance a XEN/PV boot ends up with smp_num_siblings == 2, which makes the hotplug control stay at its default value "enabled". So the mask is never evaluated. The ongoing rework of the topology evaluation caused XEN/PV to end up with smp_num_siblings == 1, which sets the SMT control to "not supported" and the empty primary thread mask causes the hotplug core to deny the bringup of the APS. Make the decision logic more robust and take 'not supported' and 'not implemented' into account for the decision whether a CPU should be booted or not. - Fake primary thread mask for XEN/PV Pretend that all XEN/PV vCPUs are primary threads, which makes the usage of the primary thread mask valid on XEN/PV. That is consistent with because all of the topology information on XEN/PV is fake or even non-existent. - Encapsulate topology information in cpuinfo_x86 Move the randomly scattered topology data into a separate data structure for readability and as a preparatory step for the topology evaluation overhaul. - Consolidate APIC ID data type to u32 It's fixed width hardware data and not randomly u16, int, unsigned long or whatever developers decided to use. - Cure the abuse of cpuinfo for persisting logical IDs. Per CPU cpuinfo is used to persist the logical package and die IDs. That's really not the right place simply because cpuinfo is subject to be reinitialized when a CPU goes through an offline/online cycle. Use separate per CPU data for the persisting to enable the further topology management rework. It will be removed once the new topology management is in place. - Provide a debug interface for inspecting topology information Useful in general and extremly helpful for validating the topology management rework in terms of correctness or "bug" compatibility. * tag 'x86-core-2023-10-29-v2' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: (23 commits) x86/apic, x86/hyperv: Use u32 in hv_snp_boot_ap() too x86/cpu: Provide debug interface x86/cpu/topology: Cure the abuse of cpuinfo for persisting logical ids x86/apic: Use u32 for wakeup_secondary_cpu[_64]() x86/apic: Use u32 for [gs]et_apic_id() x86/apic: Use u32 for phys_pkg_id() x86/apic: Use u32 for cpu_present_to_apicid() x86/apic: Use u32 for check_apicid_used() x86/apic: Use u32 for APIC IDs in global data x86/apic: Use BAD_APICID consistently x86/cpu: Move cpu_l[l2]c_id into topology info x86/cpu: Move logical package and die IDs into topology info x86/cpu: Remove pointless evaluation of x86_coreid_bits x86/cpu: Move cu_id into topology info x86/cpu: Move cpu_core_id into topology info hwmon: (fam15h_power) Use topology_core_id() scsi: lpfc: Use topology_core_id() x86/cpu: Move cpu_die_id into topology info x86/cpu: Move phys_proc_id into topology info x86/cpu: Encapsulate topology information in cpuinfo_x86 ...
		
			
				
	
	
		
			197 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0 */
 | |
| #ifndef _ASM_X86_SMP_H
 | |
| #define _ASM_X86_SMP_H
 | |
| #ifndef __ASSEMBLY__
 | |
| #include <linux/cpumask.h>
 | |
| 
 | |
| #include <asm/cpumask.h>
 | |
| #include <asm/current.h>
 | |
| #include <asm/thread_info.h>
 | |
| 
 | |
| extern int smp_num_siblings;
 | |
| extern unsigned int num_processors;
 | |
| 
 | |
| DECLARE_PER_CPU_READ_MOSTLY(cpumask_var_t, cpu_sibling_map);
 | |
| DECLARE_PER_CPU_READ_MOSTLY(cpumask_var_t, cpu_core_map);
 | |
| DECLARE_PER_CPU_READ_MOSTLY(cpumask_var_t, cpu_die_map);
 | |
| /* cpus sharing the last level cache: */
 | |
| DECLARE_PER_CPU_READ_MOSTLY(cpumask_var_t, cpu_llc_shared_map);
 | |
| DECLARE_PER_CPU_READ_MOSTLY(cpumask_var_t, cpu_l2c_shared_map);
 | |
| 
 | |
| DECLARE_EARLY_PER_CPU_READ_MOSTLY(u32, x86_cpu_to_apicid);
 | |
| DECLARE_EARLY_PER_CPU_READ_MOSTLY(u32, x86_cpu_to_acpiid);
 | |
| 
 | |
| struct task_struct;
 | |
| 
 | |
| struct smp_ops {
 | |
| 	void (*smp_prepare_boot_cpu)(void);
 | |
| 	void (*smp_prepare_cpus)(unsigned max_cpus);
 | |
| 	void (*smp_cpus_done)(unsigned max_cpus);
 | |
| 
 | |
| 	void (*stop_other_cpus)(int wait);
 | |
| 	void (*crash_stop_other_cpus)(void);
 | |
| 	void (*smp_send_reschedule)(int cpu);
 | |
| 
 | |
| 	void (*cleanup_dead_cpu)(unsigned cpu);
 | |
| 	void (*poll_sync_state)(void);
 | |
| 	int (*kick_ap_alive)(unsigned cpu, struct task_struct *tidle);
 | |
| 	int (*cpu_disable)(void);
 | |
| 	void (*cpu_die)(unsigned int cpu);
 | |
| 	void (*play_dead)(void);
 | |
| 
 | |
| 	void (*send_call_func_ipi)(const struct cpumask *mask);
 | |
| 	void (*send_call_func_single_ipi)(int cpu);
 | |
| };
 | |
| 
 | |
| /* Globals due to paravirt */
 | |
| extern void set_cpu_sibling_map(int cpu);
 | |
| 
 | |
| #ifdef CONFIG_SMP
 | |
| extern struct smp_ops smp_ops;
 | |
| 
 | |
| static inline void smp_send_stop(void)
 | |
| {
 | |
| 	smp_ops.stop_other_cpus(0);
 | |
| }
 | |
| 
 | |
| static inline void stop_other_cpus(void)
 | |
| {
 | |
| 	smp_ops.stop_other_cpus(1);
 | |
| }
 | |
| 
 | |
| static inline void smp_prepare_boot_cpu(void)
 | |
| {
 | |
| 	smp_ops.smp_prepare_boot_cpu();
 | |
| }
 | |
| 
 | |
| static inline void smp_prepare_cpus(unsigned int max_cpus)
 | |
| {
 | |
| 	smp_ops.smp_prepare_cpus(max_cpus);
 | |
| }
 | |
| 
 | |
| static inline void smp_cpus_done(unsigned int max_cpus)
 | |
| {
 | |
| 	smp_ops.smp_cpus_done(max_cpus);
 | |
| }
 | |
| 
 | |
| static inline int __cpu_disable(void)
 | |
| {
 | |
| 	return smp_ops.cpu_disable();
 | |
| }
 | |
| 
 | |
| static inline void __cpu_die(unsigned int cpu)
 | |
| {
 | |
| 	if (smp_ops.cpu_die)
 | |
| 		smp_ops.cpu_die(cpu);
 | |
| }
 | |
| 
 | |
| static inline void __noreturn play_dead(void)
 | |
| {
 | |
| 	smp_ops.play_dead();
 | |
| 	BUG();
 | |
| }
 | |
| 
 | |
| static inline void arch_smp_send_reschedule(int cpu)
 | |
| {
 | |
| 	smp_ops.smp_send_reschedule(cpu);
 | |
| }
 | |
| 
 | |
| static inline void arch_send_call_function_single_ipi(int cpu)
 | |
| {
 | |
| 	smp_ops.send_call_func_single_ipi(cpu);
 | |
| }
 | |
| 
 | |
| static inline void arch_send_call_function_ipi_mask(const struct cpumask *mask)
 | |
| {
 | |
| 	smp_ops.send_call_func_ipi(mask);
 | |
| }
 | |
| 
 | |
| void cpu_disable_common(void);
 | |
| void native_smp_prepare_boot_cpu(void);
 | |
| void smp_prepare_cpus_common(void);
 | |
| void native_smp_prepare_cpus(unsigned int max_cpus);
 | |
| void calculate_max_logical_packages(void);
 | |
| void native_smp_cpus_done(unsigned int max_cpus);
 | |
| int common_cpu_up(unsigned int cpunum, struct task_struct *tidle);
 | |
| int native_kick_ap(unsigned int cpu, struct task_struct *tidle);
 | |
| int native_cpu_disable(void);
 | |
| void __noreturn hlt_play_dead(void);
 | |
| void native_play_dead(void);
 | |
| void play_dead_common(void);
 | |
| void wbinvd_on_cpu(int cpu);
 | |
| int wbinvd_on_all_cpus(void);
 | |
| 
 | |
| void smp_kick_mwait_play_dead(void);
 | |
| 
 | |
| void native_smp_send_reschedule(int cpu);
 | |
| void native_send_call_func_ipi(const struct cpumask *mask);
 | |
| void native_send_call_func_single_ipi(int cpu);
 | |
| 
 | |
| void smp_store_cpu_info(int id);
 | |
| 
 | |
| asmlinkage __visible void smp_reboot_interrupt(void);
 | |
| __visible void smp_reschedule_interrupt(struct pt_regs *regs);
 | |
| __visible void smp_call_function_interrupt(struct pt_regs *regs);
 | |
| __visible void smp_call_function_single_interrupt(struct pt_regs *r);
 | |
| 
 | |
| #define cpu_physical_id(cpu)	per_cpu(x86_cpu_to_apicid, cpu)
 | |
| #define cpu_acpi_id(cpu)	per_cpu(x86_cpu_to_acpiid, cpu)
 | |
| 
 | |
| /*
 | |
|  * This function is needed by all SMP systems. It must _always_ be valid
 | |
|  * from the initial startup.
 | |
|  */
 | |
| #define raw_smp_processor_id()  this_cpu_read(pcpu_hot.cpu_number)
 | |
| #define __smp_processor_id() __this_cpu_read(pcpu_hot.cpu_number)
 | |
| 
 | |
| #ifdef CONFIG_X86_32
 | |
| extern int safe_smp_processor_id(void);
 | |
| #else
 | |
| # define safe_smp_processor_id()	smp_processor_id()
 | |
| #endif
 | |
| 
 | |
| static inline struct cpumask *cpu_llc_shared_mask(int cpu)
 | |
| {
 | |
| 	return per_cpu(cpu_llc_shared_map, cpu);
 | |
| }
 | |
| 
 | |
| static inline struct cpumask *cpu_l2c_shared_mask(int cpu)
 | |
| {
 | |
| 	return per_cpu(cpu_l2c_shared_map, cpu);
 | |
| }
 | |
| 
 | |
| #else /* !CONFIG_SMP */
 | |
| #define wbinvd_on_cpu(cpu)     wbinvd()
 | |
| static inline int wbinvd_on_all_cpus(void)
 | |
| {
 | |
| 	wbinvd();
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static inline struct cpumask *cpu_llc_shared_mask(int cpu)
 | |
| {
 | |
| 	return (struct cpumask *)cpumask_of(0);
 | |
| }
 | |
| #endif /* CONFIG_SMP */
 | |
| 
 | |
| extern unsigned disabled_cpus;
 | |
| 
 | |
| #ifdef CONFIG_DEBUG_NMI_SELFTEST
 | |
| extern void nmi_selftest(void);
 | |
| #else
 | |
| #define nmi_selftest() do { } while (0)
 | |
| #endif
 | |
| 
 | |
| extern unsigned int smpboot_control;
 | |
| extern unsigned long apic_mmio_base;
 | |
| 
 | |
| #endif /* !__ASSEMBLY__ */
 | |
| 
 | |
| /* Control bits for startup_64 */
 | |
| #define STARTUP_READ_APICID	0x80000000
 | |
| 
 | |
| /* Top 8 bits are reserved for control */
 | |
| #define STARTUP_PARALLEL_MASK	0xFF000000
 | |
| 
 | |
| #endif /* _ASM_X86_SMP_H */
 |