public inbox for linux-acpi@vger.kernel.org
 help / color / mirror / Atom feed
* [PATCH]Make processor module unloading safe
@ 2005-09-06  8:08 Shaohua Li
  0 siblings, 0 replies; only message in thread
From: Shaohua Li @ 2005-09-06  8:08 UTC (permalink / raw)
  To: acpi-dev; +Cc: linux-JhLEnvuH02M, Venki, Zwane Mwaikambo, Len

Hi,
pm_idle isn’t a per-cpu data. In MP case, it is possible one CPU (eg.
the boot CPU) changed pm_idle, but other CPUs it’s not ready for it. The
ACPI module is such a case, CPU 0 changed pm_idle, which caused all CPUs
start using pm_idle, but other CPUs actually aren’t ready (their idle
information isn't initialized). The worse case is unloading ACPI module.
If non-boot CPU first freed their idle information, current code doesn’t
know it, and the CPUs possibly will use the freed memory again till boot
cpu restore pm_idle. This patch tries to fix the gaps.

Makes pm_idle a per-cpu data might be a better idea, but it’s platform
independent, so I didn't touch it but just fix the ACPI issue.

Thanks,
Shaohua


Signed-off-by: Shaohua Li<shaohua.li-ral2JQCrhuEAvxtiuMwx3w@public.gmane.org>
---

 linux-2.6.13-root/arch/i386/kernel/apm.c        |    2 -
 linux-2.6.13-root/arch/i386/kernel/process.c    |    4 +--
 linux-2.6.13-root/arch/ia64/kernel/process.c    |    4 +--
 linux-2.6.13-root/arch/x86_64/kernel/process.c  |    4 +--
 linux-2.6.13-root/drivers/acpi/processor_idle.c |   26 +++++++++++++++++++-----
 linux-2.6.13-root/include/asm-i386/system.h     |    3 +-
 linux-2.6.13-root/include/asm-ia64/system.h     |    3 +-
 linux-2.6.13-root/include/asm-x86_64/system.h   |    3 +-
 linux-2.6.13-root/kernel/sched.c                |    1 
 9 files changed, 35 insertions(+), 15 deletions(-)

diff -puN drivers/acpi/processor_idle.c~remove_cpu_idle_save drivers/acpi/processor_idle.c
--- linux-2.6.13/drivers/acpi/processor_idle.c~remove_cpu_idle_save	2005-09-05 09:30:34.595622176 +0800
+++ linux-2.6.13-root/drivers/acpi/processor_idle.c	2005-09-05 09:32:47.556409064 +0800
@@ -75,6 +75,8 @@ module_param(bm_history, uint, 0644);
                                 Power Management
    -------------------------------------------------------------------------- */
 
+static void acpi_processor_idle(void);
+static DEFINE_PER_CPU(void (*)(void), acpi_cpu_idle);
 /*
  * IBM ThinkPad R40e crashes mysteriously when going into C2 or C3.
  * For now disable this. Probably a bug somewhere else.
@@ -170,7 +172,7 @@ acpi_processor_power_activate (
 static atomic_t 	c3_cpu_count;
 
 
-static void acpi_processor_idle (void)
+static void __acpi_processor_idle (void)
 {
 	struct acpi_processor	*pr = NULL;
 	struct acpi_processor_cx *cx = NULL;
@@ -891,13 +893,13 @@ int acpi_processor_cst_has_changed (stru
 		return_VALUE(-ENODEV);
 
 	/* Fall back to the default idle loop */
-	pm_idle = pm_idle_save;
-	synchronize_sched();  /* Relies on interrupts forcing exit from idle. */
+	per_cpu(acpi_cpu_idle, pr->id) = NULL;
+	cpu_idle_wait(cpumask_of_cpu(pr->id));
 
 	pr->flags.power = 0;
 	result = acpi_processor_get_power_info(pr);
 	if ((pr->flags.power == 1) && (pr->flags.power_setup_done))
-		pm_idle = acpi_processor_idle;
+		per_cpu(acpi_cpu_idle, pr->id) = __acpi_processor_idle;
 
 	return_VALUE(result);
 }
@@ -983,6 +985,17 @@ static struct file_operations acpi_proce
 	.release	= single_release,
 };
 
+static void acpi_processor_idle(void)
+{
+	int cpu = raw_smp_processor_id();
+	if (per_cpu(acpi_cpu_idle, cpu))
+		per_cpu(acpi_cpu_idle, cpu)();
+	else if (pm_idle_save)
+		pm_idle_save();
+	else
+		safe_halt();
+}
+
 int acpi_processor_power_init(struct acpi_processor *pr, struct acpi_device *device)
 {
 	acpi_status		status = 0;
@@ -1046,6 +1059,7 @@ int acpi_processor_power_init(struct acp
 	}
 
 	pr->flags.power_setup_done = 1;
+	per_cpu(acpi_cpu_idle, pr->id) = __acpi_processor_idle;
 
 	return_VALUE(0);
 }
@@ -1055,6 +1069,8 @@ int acpi_processor_power_exit(struct acp
 	ACPI_FUNCTION_TRACE("acpi_processor_power_exit");
 
 	pr->flags.power_setup_done = 0;
+	per_cpu(acpi_cpu_idle, pr->id) = NULL;
+	cpu_idle_wait(cpumask_of_cpu(pr->id));
 
 	if (acpi_device_dir(device))
 		remove_proc_entry(ACPI_PROCESSOR_FILE_POWER,acpi_device_dir(device));
@@ -1068,7 +1084,7 @@ int acpi_processor_power_exit(struct acp
 		 * (pm_idle), Wait for all processors to update cached/local
 		 * copies of pm_idle before proceeding.
 		 */
-		cpu_idle_wait();
+		cpu_idle_wait(cpu_online_map);
 	}
 
 	return_VALUE(0);
diff -puN arch/i386/kernel/process.c~remove_cpu_idle_save arch/i386/kernel/process.c
--- linux-2.6.13/arch/i386/kernel/process.c~remove_cpu_idle_save	2005-09-05 09:30:34.618618680 +0800
+++ linux-2.6.13-root/arch/i386/kernel/process.c	2005-09-05 09:30:34.669610928 +0800
@@ -207,7 +207,7 @@ void cpu_idle(void)
 	}
 }
 
-void cpu_idle_wait(void)
+void cpu_idle_wait(cpumask_t wait)
 {
 	unsigned int cpu, this_cpu = get_cpu();
 	cpumask_t map;
@@ -216,7 +216,7 @@ void cpu_idle_wait(void)
 	put_cpu();
 
 	cpus_clear(map);
-	for_each_online_cpu(cpu) {
+	for_each_cpu_mask(cpu, wait) {
 		per_cpu(cpu_idle_state, cpu) = 1;
 		cpu_set(cpu, map);
 	}
diff -puN arch/x86_64/kernel/process.c~remove_cpu_idle_save arch/x86_64/kernel/process.c
--- linux-2.6.13/arch/x86_64/kernel/process.c~remove_cpu_idle_save	2005-09-05 09:30:34.629617008 +0800
+++ linux-2.6.13-root/arch/x86_64/kernel/process.c	2005-09-05 09:30:34.670610776 +0800
@@ -128,7 +128,7 @@ static void poll_idle (void)
 	}
 }
 
-void cpu_idle_wait(void)
+void cpu_idle_wait(cpumask_t wait)
 {
 	unsigned int cpu, this_cpu = get_cpu();
 	cpumask_t map;
@@ -137,7 +137,7 @@ void cpu_idle_wait(void)
 	put_cpu();
 
 	cpus_clear(map);
-	for_each_online_cpu(cpu) {
+	for_each_cpu_mask(cpu, wait) {
 		per_cpu(cpu_idle_state, cpu) = 1;
 		cpu_set(cpu, map);
 	}
diff -puN arch/ia64/kernel/process.c~remove_cpu_idle_save arch/ia64/kernel/process.c
--- linux-2.6.13/arch/ia64/kernel/process.c~remove_cpu_idle_save	2005-09-05 09:30:34.637615792 +0800
+++ linux-2.6.13-root/arch/ia64/kernel/process.c	2005-09-05 09:30:34.670610776 +0800
@@ -231,7 +231,7 @@ static inline void play_dead(void)
 }
 #endif /* CONFIG_HOTPLUG_CPU */
 
-void cpu_idle_wait(void)
+void cpu_idle_wait(cpumask_t wait)
 {
 	unsigned int cpu, this_cpu = get_cpu();
 	cpumask_t map;
@@ -240,7 +240,7 @@ void cpu_idle_wait(void)
 	put_cpu();
 
 	cpus_clear(map);
-	for_each_online_cpu(cpu) {
+	for_each_cpu_mask(cpu, wait) {
 		per_cpu(cpu_idle_state, cpu) = 1;
 		cpu_set(cpu, map);
 	}
diff -puN include/asm-x86_64/system.h~remove_cpu_idle_save include/asm-x86_64/system.h
--- linux-2.6.13/include/asm-x86_64/system.h~remove_cpu_idle_save	2005-09-05 09:30:34.638615640 +0800
+++ linux-2.6.13-root/include/asm-x86_64/system.h	2005-09-05 09:30:34.671610624 +0800
@@ -326,7 +326,8 @@ static inline unsigned long __cmpxchg(vo
 /* For spinlocks etc */
 #define local_irq_save(x) 	do { warn_if_not_ulong(x); __asm__ __volatile__("# local_irq_save \n\t pushfq ; popq %0 ; cli":"=g" (x): /* no input */ :"memory"); } while (0)
 
-void cpu_idle_wait(void);
+#include <linux/cpumask.h>
+void cpu_idle_wait(cpumask_t wait);
 
 /*
  * disable hlt during certain critical i/o operations
diff -puN include/asm-i386/system.h~remove_cpu_idle_save include/asm-i386/system.h
--- linux-2.6.13/include/asm-i386/system.h~remove_cpu_idle_save	2005-09-05 09:30:34.646614424 +0800
+++ linux-2.6.13-root/include/asm-i386/system.h	2005-09-05 09:30:34.671610624 +0800
@@ -466,7 +466,8 @@ void disable_hlt(void);
 void enable_hlt(void);
 
 extern int es7000_plat;
-void cpu_idle_wait(void);
+#include <linux/cpumask.h>
+void cpu_idle_wait(cpumask_t wait);
 
 extern unsigned long arch_align_stack(unsigned long sp);
 
diff -puN include/asm-ia64/system.h~remove_cpu_idle_save include/asm-ia64/system.h
--- linux-2.6.13/include/asm-ia64/system.h~remove_cpu_idle_save	2005-09-05 09:30:34.647614272 +0800
+++ linux-2.6.13-root/include/asm-ia64/system.h	2005-09-05 09:30:34.671610624 +0800
@@ -276,7 +276,8 @@ extern void ia64_load_extra (struct task
 
 #define ia64_platform_is(x) (strcmp(x, platform_name) == 0)
 
-void cpu_idle_wait(void);
+#include <linux/cpumask.h>
+void cpu_idle_wait(cpumask_t wait);
 
 #define arch_align_stack(x) (x)
 
diff -puN arch/i386/kernel/apm.c~remove_cpu_idle_save arch/i386/kernel/apm.c
--- linux-2.6.13/arch/i386/kernel/apm.c~remove_cpu_idle_save	2005-09-05 09:30:34.655613056 +0800
+++ linux-2.6.13-root/arch/i386/kernel/apm.c	2005-09-05 09:30:34.673610320 +0800
@@ -2370,7 +2370,7 @@ static void __exit apm_exit(void)
 		 * (pm_idle), Wait for all processors to update cached/local
 		 * copies of pm_idle before proceeding.
 		 */
-		cpu_idle_wait();
+		cpu_idle_wait(cpu_online_map);
 	}
 	if (((apm_info.bios.flags & APM_BIOS_DISENGAGED) == 0)
 	    && (apm_info.connection_version > 0x0100)) {
diff -puN kernel/sched.c~remove_cpu_idle_save kernel/sched.c
--- linux-2.6.13/kernel/sched.c~remove_cpu_idle_save	2005-09-05 09:30:34.663611840 +0800
+++ linux-2.6.13-root/kernel/sched.c	2005-09-05 09:30:34.675610016 +0800
@@ -3771,6 +3771,7 @@ EXPORT_SYMBOL(cpu_present_map);
 
 #ifndef CONFIG_SMP
 cpumask_t cpu_online_map = CPU_MASK_ALL;
+EXPORT_SYMBOL(cpu_online_map);
 cpumask_t cpu_possible_map = CPU_MASK_ALL;
 #endif
 
_




-------------------------------------------------------
SF.Net email is Sponsored by the Better Software Conference & EXPO
September 19-22, 2005 * San Francisco, CA * Development Lifecycle Practices
Agile & Plan-Driven Development * Managing Projects & Teams * Testing & QA
Security * Process Improvement & Measurement * http://www.sqe.com/bsce5sf

^ permalink raw reply	[flat|nested] only message in thread

only message in thread, other threads:[~2005-09-06  8:08 UTC | newest]

Thread overview: (only message) (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2005-09-06  8:08 [PATCH]Make processor module unloading safe Shaohua Li

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox