Commit 60e49ba3 authored by Stephen Rothwell's avatar Stephen Rothwell Committed by Linus Torvalds

[PATCH] 2.5.4-pre6 apm compile fix

Here is the patch against 2.5.4.  I have compiled this patch under
2.5.3, so it should still be OK.

This patch just resyncs the driver with 2.4.18-pre (which is what is
being testd by others).  The only outstanding known problem is some
very strange interaction with VMWARE.  But otherwise people seem
happy with the changes.

Original announcement to Dave Jones and Marcelo:

	Update a couple of email addresses
	Fix the idle handling (this is an improved version of the fix
		that Alan Cox has in his -ac tree)
	Notify user mode of suspend events before drivers (fix)
	Make the idling percentage boot time configurable
	Rename kapm-idled to kapmd

Credit to Andreas Steinmetz, Russell King, Thomas Hood and me.

More small updates to come.
--
Cheers,
Stephen Rothwell                    sfr@canb.auug.org.au
http://www.canb.auug.org.au/~sfr/
parent 74c01024
...@@ -39,6 +39,7 @@ ...@@ -39,6 +39,7 @@
* Feb 2000, Version 1.13 * Feb 2000, Version 1.13
* Nov 2000, Version 1.14 * Nov 2000, Version 1.14
* Oct 2001, Version 1.15 * Oct 2001, Version 1.15
* Jan 2002, Version 1.16
* *
* History: * History:
* 0.6b: first version in official kernel, Linux 1.3.46 * 0.6b: first version in official kernel, Linux 1.3.46
...@@ -85,7 +86,7 @@ ...@@ -85,7 +86,7 @@
* change APM_NOINTS to CONFIG_APM_ALLOW_INTS * change APM_NOINTS to CONFIG_APM_ALLOW_INTS
* remove dependency on CONFIG_PROC_FS * remove dependency on CONFIG_PROC_FS
* Stephen Rothwell * Stephen Rothwell
* 1.9: Fix small typo. <laslo@ilo.opole.pl> * 1.9: Fix small typo. <laslo@wodip.opole.pl>
* Try to cope with BIOS's that need to have all display * Try to cope with BIOS's that need to have all display
* devices blanked and not just the first one. * devices blanked and not just the first one.
* Ross Paterson <ross@soi.city.ac.uk> * Ross Paterson <ross@soi.city.ac.uk>
...@@ -164,8 +165,18 @@ ...@@ -164,8 +165,18 @@
* If an APM idle fails log it and idle sensibly * If an APM idle fails log it and idle sensibly
* 1.15: Don't queue events to clients who open the device O_WRONLY. * 1.15: Don't queue events to clients who open the device O_WRONLY.
* Don't expect replies from clients who open the device O_RDONLY. * Don't expect replies from clients who open the device O_RDONLY.
* (Idea from Thomas Hood <jdthood at yahoo.co.uk>) * (Idea from Thomas Hood <jdthood@mail.com>)
* Minor waitqueue cleanups.(John Fremlin <chief@bandits.org>) * Minor waitqueue cleanups. (John Fremlin <chief@bandits.org>)
* 1.16: Fix idle calling. (Andreas Steinmetz <ast@domdv.de> et al.)
* Notify listeners of standby or suspend events before notifying
* drivers. Return EBUSY to ioctl() if suspend is rejected.
* (Russell King <rmk@arm.linux.org.uk> and Thomas Hood)
* Ignore first resume after we generate our own resume event
* after a suspend (Thomas Hood <jdthood@mail.com>)
* Daemonize now gets rid of our controlling terminal (sfr).
* CONFIG_APM_CPU_IDLE now just affects the default value of
* idle_threshold (sfr).
* Change name of kernel apm daemon (as it no longer idles) (sfr).
* *
* APM 1.1 Reference: * APM 1.1 Reference:
* *
...@@ -238,6 +249,12 @@ extern int (*console_blank_hook)(int); ...@@ -238,6 +249,12 @@ extern int (*console_blank_hook)(int);
* [no-]power[-_]off power off on shutdown * [no-]power[-_]off power off on shutdown
* bounce[-_]interval=<n> number of ticks to ignore suspend * bounce[-_]interval=<n> number of ticks to ignore suspend
* bounces * bounces
* idle[-_]threshold=<n> System idle percentage above which to
* make APM BIOS idle calls. Set it to
* 100 to disable.
* idle[-_]period=<n> Period (in 1/100s of a second) over
* which the idle percentage is
* calculated.
*/ */
/* KNOWN PROBLEM MACHINES: /* KNOWN PROBLEM MACHINES:
...@@ -340,6 +357,16 @@ struct apm_user { ...@@ -340,6 +357,16 @@ struct apm_user {
*/ */
#define APM_BIOS_MAGIC 0x4101 #define APM_BIOS_MAGIC 0x4101
/*
* idle percentage above which bios idle calls are done
*/
#ifdef CONFIG_APM_CPU_IDLE
#define DEFAULT_IDLE_THRESHOLD 95
#else
#define DEFAULT_IDLE_THRESHOLD 100
#endif
#define DEFAULT_IDLE_PERIOD (100 / 3)
/* /*
* Local variables * Local variables
*/ */
...@@ -347,9 +374,10 @@ static struct { ...@@ -347,9 +374,10 @@ static struct {
unsigned long offset; unsigned long offset;
unsigned short segment; unsigned short segment;
} apm_bios_entry; } apm_bios_entry;
#ifdef CONFIG_APM_CPU_IDLE
static int clock_slowed; static int clock_slowed;
#endif static int idle_threshold = DEFAULT_IDLE_THRESHOLD;
static int idle_period = DEFAULT_IDLE_PERIOD;
static int set_pm_idle;
static int suspends_pending; static int suspends_pending;
static int standbys_pending; static int standbys_pending;
static int waiting_for_resume; static int waiting_for_resume;
...@@ -389,7 +417,7 @@ static DECLARE_WAIT_QUEUE_HEAD(apm_suspend_waitqueue); ...@@ -389,7 +417,7 @@ static DECLARE_WAIT_QUEUE_HEAD(apm_suspend_waitqueue);
static struct apm_user * user_list; static struct apm_user * user_list;
static spinlock_t user_list_lock = SPIN_LOCK_UNLOCKED; static spinlock_t user_list_lock = SPIN_LOCK_UNLOCKED;
static char driver_version[] = "1.15"; /* no spaces */ static char driver_version[] = "1.16"; /* no spaces */
/* /*
* APM event names taken from the APM 1.2 specification. These are * APM event names taken from the APM 1.2 specification. These are
...@@ -685,8 +713,6 @@ static int apm_set_power_state(u_short state) ...@@ -685,8 +713,6 @@ static int apm_set_power_state(u_short state)
return set_power_state(APM_DEVICE_ALL, state); return set_power_state(APM_DEVICE_ALL, state);
} }
#ifdef CONFIG_APM_CPU_IDLE
/** /**
* apm_do_idle - perform power saving * apm_do_idle - perform power saving
* *
...@@ -736,63 +762,89 @@ static void apm_do_busy(void) ...@@ -736,63 +762,89 @@ static void apm_do_busy(void)
} }
} }
#if 0
extern int hlt_counter;
/* /*
* If no process has been interested in this * If no process has really been interested in
* CPU for some time, we want to wake up the * the CPU for some time, we want to call BIOS
* power management thread - we probably want * power management - we probably want
* to conserve power. * to conserve power.
*/ */
#define HARD_IDLE_TIMEOUT (HZ/3) #define IDLE_CALC_LIMIT (HZ * 100)
#define IDLE_LEAKY_MAX 16
static void (*sys_idle)(void);
/* This should wake up kapmd and ask it to slow the CPU */ extern void default_idle(void);
#define powermanagement_idle() do { } while (0)
/** /**
* apm_cpu_idle - cpu idling for APM capable Linux * apm_cpu_idle - cpu idling for APM capable Linux
* *
* This is the idling function the kernel executes when APM is available. It * This is the idling function the kernel executes when APM is available. It
* tries to save processor time directly by using hlt instructions. A * tries to do BIOS powermanagement based on the average system idle time.
* separate apm thread tries to do the BIOS power management. * Furthermore it calls the system default idle routine.
*
* N.B. This is curently not used for kernels 2.4.x.
*/ */
static void apm_cpu_idle(void) static void apm_cpu_idle(void)
{ {
unsigned int start_idle; static int use_apm_idle = 0;
static unsigned int last_jiffies = 0;
static unsigned int last_stime = 0;
int apm_is_idle = 0;
unsigned int jiffies_since_last_check = jiffies - last_jiffies;
unsigned int t1;
recalc:
if (jiffies_since_last_check > IDLE_CALC_LIMIT) {
use_apm_idle = 0;
last_jiffies = jiffies;
last_stime = current->times.tms_stime;
} else if (jiffies_since_last_check > idle_period) {
unsigned int idle_percentage;
idle_percentage = current->times.tms_stime - last_stime;
idle_percentage *= 100;
idle_percentage /= jiffies_since_last_check;
use_apm_idle = (idle_percentage > idle_threshold);
last_jiffies = jiffies;
last_stime = current->times.tms_stime;
}
start_idle = jiffies; t1 = IDLE_LEAKY_MAX;
while (1) {
if (!need_resched()) { while (need_resched()) {
if (jiffies - start_idle < HARD_IDLE_TIMEOUT) { if (use_apm_idle) {
if (!current_cpu_data.hlt_works_ok) unsigned int t;
continue;
if (hlt_counter) t = jiffies;
switch (apm_do_idle()) {
case 0: apm_is_idle = 1;
if (t != jiffies) {
if (t1) {
t1 = IDLE_LEAKY_MAX;
continue;
}
} else if (t1) {
t1--;
continue; continue;
__cli(); }
if (!need_resched()) break;
safe_halt(); case 1: apm_is_idle = 1;
else break;
__sti();
continue;
} }
/*
* Ok, do some power management - we've been idle for too long
*/
powermanagement_idle();
} }
if (sys_idle)
schedule(); sys_idle();
check_pgt_cache(); else
start_idle = jiffies; default_idle();
jiffies_since_last_check = jiffies - last_jiffies;
if (jiffies_since_last_check > idle_period)
goto recalc;
} }
if (apm_is_idle)
apm_do_busy();
} }
#endif
#endif
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
static int apm_magic(void * unused) static int apm_magic(void * unused)
...@@ -1137,59 +1189,44 @@ static void reinit_timer(void) ...@@ -1137,59 +1189,44 @@ static void reinit_timer(void)
#endif #endif
} }
static int send_event(apm_event_t event) static int suspend(int vetoable)
{ {
switch (event) { int err;
case APM_SYS_SUSPEND: struct apm_user *as;
case APM_CRITICAL_SUSPEND:
case APM_USER_SUSPEND: if (pm_send_all(PM_SUSPEND, (void *)3)) {
/* map all suspends to ACPI D3 */ /* Vetoed */
if (pm_send_all(PM_SUSPEND, (void *)3)) { if (vetoable) {
if (event == APM_CRITICAL_SUSPEND) {
printk(KERN_CRIT
"apm: Critical suspend was vetoed, "
"expect armageddon\n" );
return 0;
}
if (apm_info.connection_version > 0x100) if (apm_info.connection_version > 0x100)
apm_set_power_state(APM_STATE_REJECT); apm_set_power_state(APM_STATE_REJECT);
return 0; err = -EBUSY;
waiting_for_resume = 0;
printk(KERN_WARNING "apm: suspend was vetoed.\n");
goto out;
} }
break; printk(KERN_CRIT "apm: suspend was vetoed, but suspending anyway.\n");
case APM_NORMAL_RESUME:
case APM_CRITICAL_RESUME:
/* map all resumes to ACPI D0 */
(void) pm_send_all(PM_RESUME, (void *)0);
break;
} }
return 1;
}
static int suspend(void)
{
int err;
struct apm_user *as;
get_time_diff(); get_time_diff();
cli(); cli();
err = apm_set_power_state(APM_STATE_SUSPEND); err = apm_set_power_state(APM_STATE_SUSPEND);
reinit_timer(); reinit_timer();
set_time(); set_time();
sti();
if (err == APM_NO_ERROR) if (err == APM_NO_ERROR)
err = APM_SUCCESS; err = APM_SUCCESS;
if (err != APM_SUCCESS) if (err != APM_SUCCESS)
apm_error("suspend", err); apm_error("suspend", err);
send_event(APM_NORMAL_RESUME); err = (err == APM_SUCCESS) ? 0 : -EIO;
sti(); pm_send_all(PM_RESUME, (void *)0);
queue_event(APM_NORMAL_RESUME, NULL); queue_event(APM_NORMAL_RESUME, NULL);
ignore_normal_resume = 1;
out:
spin_lock(&user_list_lock); spin_lock(&user_list_lock);
for (as = user_list; as != NULL; as = as->next) { for (as = user_list; as != NULL; as = as->next) {
as->suspend_wait = 0; as->suspend_wait = 0;
as->suspend_result = ((err == APM_SUCCESS) ? 0 : -EIO); as->suspend_result = err;
} }
spin_unlock(&user_list_lock); spin_unlock(&user_list_lock);
ignore_normal_resume = 1;
wake_up_interruptible(&apm_suspend_waitqueue); wake_up_interruptible(&apm_suspend_waitqueue);
return err; return err;
} }
...@@ -1198,6 +1235,7 @@ static void standby(void) ...@@ -1198,6 +1235,7 @@ static void standby(void)
{ {
int err; int err;
/* If needed, notify drivers here */
get_time_diff(); get_time_diff();
err = apm_set_power_state(APM_STATE_STANDBY); err = apm_set_power_state(APM_STATE_STANDBY);
if ((err != APM_SUCCESS) && (err != APM_NO_ERROR)) if ((err != APM_SUCCESS) && (err != APM_NO_ERROR))
...@@ -1241,17 +1279,13 @@ static void check_events(void) ...@@ -1241,17 +1279,13 @@ static void check_events(void)
if (ignore_bounce if (ignore_bounce
&& ((jiffies - last_resume) > bounce_interval)) && ((jiffies - last_resume) > bounce_interval))
ignore_bounce = 0; ignore_bounce = 0;
if (ignore_normal_resume && (event != APM_NORMAL_RESUME))
ignore_normal_resume = 0;
switch (event) { switch (event) {
case APM_SYS_STANDBY: case APM_SYS_STANDBY:
case APM_USER_STANDBY: case APM_USER_STANDBY:
if (send_event(event)) { queue_event(event, NULL);
queue_event(event, NULL); if (standbys_pending <= 0)
if (standbys_pending <= 0) standby();
standby();
}
break; break;
case APM_USER_SUSPEND: case APM_USER_SUSPEND:
...@@ -1276,12 +1310,10 @@ static void check_events(void) ...@@ -1276,12 +1310,10 @@ static void check_events(void)
*/ */
if (waiting_for_resume) if (waiting_for_resume)
return; return;
if (send_event(event)) { waiting_for_resume = 1;
queue_event(event, NULL); queue_event(event, NULL);
waiting_for_resume = 1; if (suspends_pending <= 0)
if (suspends_pending <= 0) (void) suspend(1);
(void) suspend();
}
break; break;
case APM_NORMAL_RESUME: case APM_NORMAL_RESUME:
...@@ -1293,16 +1325,17 @@ static void check_events(void) ...@@ -1293,16 +1325,17 @@ static void check_events(void)
if ((event != APM_NORMAL_RESUME) if ((event != APM_NORMAL_RESUME)
|| (ignore_normal_resume == 0)) { || (ignore_normal_resume == 0)) {
set_time(); set_time();
send_event(event); pm_send_all(PM_RESUME, (void *)0);
queue_event(event, NULL); queue_event(event, NULL);
} }
ignore_normal_resume = 0;
break; break;
case APM_CAPABILITY_CHANGE: case APM_CAPABILITY_CHANGE:
case APM_LOW_BATTERY: case APM_LOW_BATTERY:
case APM_POWER_STATUS_CHANGE: case APM_POWER_STATUS_CHANGE:
send_event(event);
queue_event(event, NULL); queue_event(event, NULL);
/* If needed, notify drivers here */
break; break;
case APM_UPDATE_TIME: case APM_UPDATE_TIME:
...@@ -1310,12 +1343,10 @@ static void check_events(void) ...@@ -1310,12 +1343,10 @@ static void check_events(void)
break; break;
case APM_CRITICAL_SUSPEND: case APM_CRITICAL_SUSPEND:
send_event(event);
/* /*
* We can only hope it worked - we are not allowed * We are not allowed to reject a critical suspend.
* to reject a critical suspend.
*/ */
(void) suspend(); (void) suspend(0);
break; break;
} }
} }
...@@ -1343,63 +1374,24 @@ static void apm_event_handler(void) ...@@ -1343,63 +1374,24 @@ static void apm_event_handler(void)
/* /*
* This is the APM thread main loop. * This is the APM thread main loop.
*
* Check whether we're the only running process to
* decide if we should just power down.
*
*/ */
#define system_idle() (nr_running == 1)
static void apm_mainloop(void) static void apm_mainloop(void)
{ {
int timeout = HZ;
DECLARE_WAITQUEUE(wait, current); DECLARE_WAITQUEUE(wait, current);
add_wait_queue(&apm_waitqueue, &wait); add_wait_queue(&apm_waitqueue, &wait);
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
for (;;) { for (;;) {
/* Nothing to do, just sleep for the timeout */ schedule_timeout(APM_CHECK_TIMEOUT);
timeout = 2 * timeout;
if (timeout > APM_CHECK_TIMEOUT)
timeout = APM_CHECK_TIMEOUT;
schedule_timeout(timeout);
if (exit_kapmd) if (exit_kapmd)
break; break;
/* /*
* Ok, check all events, check for idle (and mark us sleeping * Ok, check all events, check for idle (and mark us sleeping
* so as not to count towards the load average).. * so as not to count towards the load average)..
*/ */
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
apm_event_handler(); apm_event_handler();
#ifdef CONFIG_APM_CPU_IDLE
if (!system_idle())
continue;
/*
* If we can idle...
*/
if (apm_do_idle() != -1) {
unsigned long start = jiffies;
while ((!exit_kapmd) && system_idle()) {
if (apm_do_idle()) {
set_current_state(TASK_INTERRUPTIBLE);
/* APM needs us to snooze .. either
the BIOS call failed (-1) or it
slowed the clock (1). We sleep
until it talks to us again */
schedule_timeout(1);
}
if ((jiffies - start) > APM_CHECK_TIMEOUT) {
apm_event_handler();
start = jiffies;
}
}
apm_do_busy();
apm_event_handler();
timeout = 1;
}
#endif
} }
remove_wait_queue(&apm_waitqueue, &wait); remove_wait_queue(&apm_waitqueue, &wait);
} }
...@@ -1485,9 +1477,7 @@ static int do_ioctl(struct inode * inode, struct file *filp, ...@@ -1485,9 +1477,7 @@ static int do_ioctl(struct inode * inode, struct file *filp,
as->standbys_read--; as->standbys_read--;
as->standbys_pending--; as->standbys_pending--;
standbys_pending--; standbys_pending--;
} else if (!send_event(APM_USER_STANDBY)) } else
return -EAGAIN;
else
queue_event(APM_USER_STANDBY, as); queue_event(APM_USER_STANDBY, as);
if (standbys_pending <= 0) if (standbys_pending <= 0)
standby(); standby();
...@@ -1497,13 +1487,10 @@ static int do_ioctl(struct inode * inode, struct file *filp, ...@@ -1497,13 +1487,10 @@ static int do_ioctl(struct inode * inode, struct file *filp,
as->suspends_read--; as->suspends_read--;
as->suspends_pending--; as->suspends_pending--;
suspends_pending--; suspends_pending--;
} else if (!send_event(APM_USER_SUSPEND)) } else
return -EAGAIN;
else
queue_event(APM_USER_SUSPEND, as); queue_event(APM_USER_SUSPEND, as);
if (suspends_pending <= 0) { if (suspends_pending <= 0) {
if (suspend() != APM_SUCCESS) return suspend(1);
return -EIO;
} else { } else {
as->suspend_wait = 1; as->suspend_wait = 1;
wait_event_interruptible(apm_suspend_waitqueue, wait_event_interruptible(apm_suspend_waitqueue,
...@@ -1533,7 +1520,7 @@ static int do_release(struct inode * inode, struct file * filp) ...@@ -1533,7 +1520,7 @@ static int do_release(struct inode * inode, struct file * filp)
if (as->suspends_pending > 0) { if (as->suspends_pending > 0) {
suspends_pending -= as->suspends_pending; suspends_pending -= as->suspends_pending;
if (suspends_pending <= 0) if (suspends_pending <= 0)
(void) suspend(); (void) suspend(1);
} }
spin_lock(&user_list_lock); spin_lock(&user_list_lock);
if (user_list == as) if (user_list == as)
...@@ -1684,9 +1671,8 @@ static int apm(void *unused) ...@@ -1684,9 +1671,8 @@ static int apm(void *unused)
daemonize(); daemonize();
strcpy(current->comm, "kapm-idled"); strcpy(current->comm, "kapmd");
sigfillset(&current->blocked); sigfillset(&current->blocked);
current->tty = NULL; /* get rid of controlling tty */
if (apm_info.connection_version == 0) { if (apm_info.connection_version == 0) {
apm_info.connection_version = apm_info.bios.version; apm_info.connection_version = apm_info.bios.version;
...@@ -1805,7 +1791,14 @@ static int __init apm_setup(char *str) ...@@ -1805,7 +1791,14 @@ static int __init apm_setup(char *str)
if ((strncmp(str, "bounce-interval=", 16) == 0) || if ((strncmp(str, "bounce-interval=", 16) == 0) ||
(strncmp(str, "bounce_interval=", 16) == 0)) (strncmp(str, "bounce_interval=", 16) == 0))
bounce_interval = simple_strtol(str + 16, NULL, 0); bounce_interval = simple_strtol(str + 16, NULL, 0);
invert = (strncmp(str, "no-", 3) == 0); if ((strncmp(str, "idle-threshold=", 15) == 0) ||
(strncmp(str, "idle_threshold=", 15) == 0))
idle_threshold = simple_strtol(str + 15, NULL, 0);
if ((strncmp(str, "idle-period=", 12) == 0) ||
(strncmp(str, "idle_period=", 12) == 0))
idle_threshold = simple_strtol(str + 15, NULL, 0);
invert = (strncmp(str, "no-", 3) == 0) ||
(strncmp(str, "no_", 3) == 0);
if (invert) if (invert)
str += 3; str += 3;
if (strncmp(str, "debug", 5) == 0) if (strncmp(str, "debug", 5) == 0)
...@@ -1976,6 +1969,14 @@ static int __init apm_init(void) ...@@ -1976,6 +1969,14 @@ static int __init apm_init(void)
misc_register(&apm_device); misc_register(&apm_device);
if (HZ != 100)
idle_period = (idle_period * HZ) / 100;
if (idle_threshold < 100) {
sys_idle = pm_idle;
pm_idle = apm_cpu_idle;
set_pm_idle = 1;
}
return 0; return 0;
} }
...@@ -1983,6 +1984,8 @@ static void __exit apm_exit(void) ...@@ -1983,6 +1984,8 @@ static void __exit apm_exit(void)
{ {
int error; int error;
if (set_pm_idle)
pm_idle = sys_idle;
if (((apm_info.bios.flags & APM_BIOS_DISENGAGED) == 0) if (((apm_info.bios.flags & APM_BIOS_DISENGAGED) == 0)
&& (apm_info.connection_version > 0x0100)) { && (apm_info.connection_version > 0x0100)) {
error = apm_engage_power_management(APM_DEVICE_ALL, 0); error = apm_engage_power_management(APM_DEVICE_ALL, 0);
...@@ -2020,5 +2023,11 @@ MODULE_PARM_DESC(broken_psr, "BIOS has a broken GetPowerStatus call"); ...@@ -2020,5 +2023,11 @@ MODULE_PARM_DESC(broken_psr, "BIOS has a broken GetPowerStatus call");
MODULE_PARM(realmode_power_off, "i"); MODULE_PARM(realmode_power_off, "i");
MODULE_PARM_DESC(realmode_power_off, MODULE_PARM_DESC(realmode_power_off,
"Switch to real mode before powering off"); "Switch to real mode before powering off");
MODULE_PARM(idle_threshold, "i");
MODULE_PARM_DESC(idle_threshold,
"System idle percentage above which to make APM BIOS idle calls");
MODULE_PARM(idle_period, "i");
MODULE_PARM_DESC(idle_period,
"Period (in sec/100) over which to caculate the idle percentage");
EXPORT_NO_SYMBOLS; EXPORT_NO_SYMBOLS;
...@@ -32,9 +32,11 @@ ...@@ -32,9 +32,11 @@
extern void dump_thread(struct pt_regs *, struct user *); extern void dump_thread(struct pt_regs *, struct user *);
extern spinlock_t rtc_lock; extern spinlock_t rtc_lock;
#if defined(CONFIG_APM) || defined(CONFIG_APM_MODULE) #if defined(CONFIG_APM_MODULE)
extern void machine_real_restart(unsigned char *, int); extern void machine_real_restart(unsigned char *, int);
EXPORT_SYMBOL(machine_real_restart); EXPORT_SYMBOL(machine_real_restart);
extern void default_idle(void);
EXPORT_SYMBOL(default_idle);
#endif #endif
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
...@@ -93,7 +95,6 @@ EXPORT_SYMBOL_NOVERS(__get_user_4); ...@@ -93,7 +95,6 @@ EXPORT_SYMBOL_NOVERS(__get_user_4);
EXPORT_SYMBOL(strtok); EXPORT_SYMBOL(strtok);
EXPORT_SYMBOL(strpbrk); EXPORT_SYMBOL(strpbrk);
EXPORT_SYMBOL(simple_strtol);
EXPORT_SYMBOL(strstr); EXPORT_SYMBOL(strstr);
EXPORT_SYMBOL(strncpy_from_user); EXPORT_SYMBOL(strncpy_from_user);
......
...@@ -78,7 +78,7 @@ void enable_hlt(void) ...@@ -78,7 +78,7 @@ void enable_hlt(void)
* We use this if we don't have any better * We use this if we don't have any better
* idle routine.. * idle routine..
*/ */
static void default_idle(void) void default_idle(void)
{ {
if (current_cpu_data.hlt_works_ok && !hlt_counter) { if (current_cpu_data.hlt_works_ok && !hlt_counter) {
__cli(); __cli();
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment