mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-03 18:20:25 +02:00 
			
		
		
		
	ACPI processor hotplug: Delay acpi_processor_start() call for hotplugged cores
Delay the setting up of features (cpuidle, throttling by calling acpi_processor_start()) to the time when the hotplugged core got onlined the first time and got fully initialized. Signed-off-by: Thomas Renninger <trenn@suse.de> Signed-off-by: Len Brown <len.brown@intel.com>
This commit is contained in:
		
							parent
							
								
									54d5dcc45a
								
							
						
					
					
						commit
						99b7250844
					
				
					 3 changed files with 67 additions and 8 deletions
				
			
		| 
						 | 
					@ -84,7 +84,7 @@ static int acpi_processor_remove(struct acpi_device *device, int type);
 | 
				
			||||||
static void acpi_processor_notify(struct acpi_device *device, u32 event);
 | 
					static void acpi_processor_notify(struct acpi_device *device, u32 event);
 | 
				
			||||||
static acpi_status acpi_processor_hotadd_init(struct acpi_processor *pr);
 | 
					static acpi_status acpi_processor_hotadd_init(struct acpi_processor *pr);
 | 
				
			||||||
static int acpi_processor_handle_eject(struct acpi_processor *pr);
 | 
					static int acpi_processor_handle_eject(struct acpi_processor *pr);
 | 
				
			||||||
 | 
					static int acpi_processor_start(struct acpi_processor *pr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const struct acpi_device_id processor_device_ids[] = {
 | 
					static const struct acpi_device_id processor_device_ids[] = {
 | 
				
			||||||
	{ACPI_PROCESSOR_OBJECT_HID, 0},
 | 
						{ACPI_PROCESSOR_OBJECT_HID, 0},
 | 
				
			||||||
| 
						 | 
					@ -423,10 +423,29 @@ static int acpi_cpu_soft_notify(struct notifier_block *nfb,
 | 
				
			||||||
	struct acpi_processor *pr = per_cpu(processors, cpu);
 | 
						struct acpi_processor *pr = per_cpu(processors, cpu);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (action == CPU_ONLINE && pr) {
 | 
						if (action == CPU_ONLINE && pr) {
 | 
				
			||||||
		acpi_processor_ppc_has_changed(pr, 0);
 | 
							/* CPU got physically hotplugged and onlined the first time:
 | 
				
			||||||
		acpi_processor_hotplug(pr);
 | 
							 * Initialize missing things
 | 
				
			||||||
		acpi_processor_reevaluate_tstate(pr, action);
 | 
							 */
 | 
				
			||||||
		acpi_processor_tstate_has_changed(pr);
 | 
							if (pr->flags.need_hotplug_init) {
 | 
				
			||||||
 | 
								struct cpuidle_driver *idle_driver =
 | 
				
			||||||
 | 
									cpuidle_get_driver();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								printk(KERN_INFO "Will online and init hotplugged "
 | 
				
			||||||
 | 
								       "CPU: %d\n", pr->id);
 | 
				
			||||||
 | 
								WARN(acpi_processor_start(pr), "Failed to start CPU:"
 | 
				
			||||||
 | 
									" %d\n", pr->id);
 | 
				
			||||||
 | 
								pr->flags.need_hotplug_init = 0;
 | 
				
			||||||
 | 
								if (idle_driver && !strcmp(idle_driver->name,
 | 
				
			||||||
 | 
											   "intel_idle")) {
 | 
				
			||||||
 | 
									intel_idle_cpu_init(pr->id);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							/* Normal CPU soft online event */
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								acpi_processor_ppc_has_changed(pr, 0);
 | 
				
			||||||
 | 
								acpi_processor_cst_has_changed(pr);
 | 
				
			||||||
 | 
								acpi_processor_reevaluate_tstate(pr, action);
 | 
				
			||||||
 | 
								acpi_processor_tstate_has_changed(pr);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if (action == CPU_DEAD && pr) {
 | 
						if (action == CPU_DEAD && pr) {
 | 
				
			||||||
		/* invalidate the flag.throttling after one CPU is offline */
 | 
							/* invalidate the flag.throttling after one CPU is offline */
 | 
				
			||||||
| 
						 | 
					@ -440,7 +459,15 @@ static struct notifier_block acpi_cpu_notifier =
 | 
				
			||||||
	    .notifier_call = acpi_cpu_soft_notify,
 | 
						    .notifier_call = acpi_cpu_soft_notify,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int __cpuinit acpi_processor_start(struct acpi_processor *pr)
 | 
					/*
 | 
				
			||||||
 | 
					 * acpi_processor_start() is called by the cpu_hotplug_notifier func:
 | 
				
			||||||
 | 
					 * acpi_cpu_soft_notify(). Getting it __cpuinit{data} is difficult, the
 | 
				
			||||||
 | 
					 * root cause seem to be that acpi_processor_uninstall_hotplug_notify()
 | 
				
			||||||
 | 
					 * is in the module_exit (__exit) func. Allowing acpi_processor_start()
 | 
				
			||||||
 | 
					 * to not be in __cpuinit section, but being called from __cpuinit funcs
 | 
				
			||||||
 | 
					 * via __ref looks like the right thing to do here.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static __ref int acpi_processor_start(struct acpi_processor *pr)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct acpi_device *device = per_cpu(processor_device_array, pr->id);
 | 
						struct acpi_device *device = per_cpu(processor_device_array, pr->id);
 | 
				
			||||||
	int result = 0;
 | 
						int result = 0;
 | 
				
			||||||
| 
						 | 
					@ -491,7 +518,12 @@ static int __cpuinit acpi_processor_start(struct acpi_processor *pr)
 | 
				
			||||||
	return result;
 | 
						return result;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Do not put anything in here which needs the core to be online.
 | 
				
			||||||
 | 
					 * For example MSR access or setting up things which check for cpuinfo_x86
 | 
				
			||||||
 | 
					 * (cpu_data(cpu)) values, like CPU feature flags, family, model, etc.
 | 
				
			||||||
 | 
					 * Such things have to be put in and set up above in acpi_processor_start()
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
static int __cpuinit acpi_processor_add(struct acpi_device *device)
 | 
					static int __cpuinit acpi_processor_add(struct acpi_device *device)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct acpi_processor *pr = NULL;
 | 
						struct acpi_processor *pr = NULL;
 | 
				
			||||||
| 
						 | 
					@ -546,6 +578,21 @@ static int __cpuinit acpi_processor_add(struct acpi_device *device)
 | 
				
			||||||
		result = -EFAULT;
 | 
							result = -EFAULT;
 | 
				
			||||||
		goto err_free_cpumask;
 | 
							goto err_free_cpumask;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/*
 | 
				
			||||||
 | 
						 * Do not start hotplugged CPUs now, but when they
 | 
				
			||||||
 | 
						 * are onlined the first time
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						if (pr->flags.need_hotplug_init)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/*
 | 
				
			||||||
 | 
						 * Do not start hotplugged CPUs now, but when they
 | 
				
			||||||
 | 
						 * are onlined the first time
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						if (pr->flags.need_hotplug_init)
 | 
				
			||||||
 | 
							return 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	result = acpi_processor_start(pr);
 | 
						result = acpi_processor_start(pr);
 | 
				
			||||||
	if (result)
 | 
						if (result)
 | 
				
			||||||
		goto err_remove_sysfs;
 | 
							goto err_remove_sysfs;
 | 
				
			||||||
| 
						 | 
					@ -751,6 +798,17 @@ static acpi_status acpi_processor_hotadd_init(struct acpi_processor *pr)
 | 
				
			||||||
		return AE_ERROR;
 | 
							return AE_ERROR;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* CPU got hot-plugged, but cpu_data is not initialized yet
 | 
				
			||||||
 | 
						 * Set flag to delay cpu_idle/throttling initialization
 | 
				
			||||||
 | 
						 * in:
 | 
				
			||||||
 | 
						 * acpi_processor_add()
 | 
				
			||||||
 | 
						 *   acpi_processor_get_info()
 | 
				
			||||||
 | 
						 * and do it when the CPU gets online the first time
 | 
				
			||||||
 | 
						 * TBD: Cleanup above functions and try to do this more elegant.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						printk(KERN_INFO "CPU %d got hotplugged\n", pr->id);
 | 
				
			||||||
 | 
						pr->flags.need_hotplug_init = 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return AE_OK;
 | 
						return AE_OK;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -527,7 +527,7 @@ int intel_idle_cpu_init(int cpu)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					EXPORT_SYMBOL_GPL(intel_idle_cpu_init);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int __init intel_idle_init(void)
 | 
					static int __init intel_idle_init(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -195,6 +195,7 @@ struct acpi_processor_flags {
 | 
				
			||||||
	u8 has_cst:1;
 | 
						u8 has_cst:1;
 | 
				
			||||||
	u8 power_setup_done:1;
 | 
						u8 power_setup_done:1;
 | 
				
			||||||
	u8 bm_rld_set:1;
 | 
						u8 bm_rld_set:1;
 | 
				
			||||||
 | 
						u8 need_hotplug_init:1;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct acpi_processor {
 | 
					struct acpi_processor {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue