The APM now successfully shuts down my IBM ThinkPad T40. It's still disabled
for more testing on other machines. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@16136 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
fd9772def6
commit
5da68569d0
@ -52,6 +52,7 @@ typedef struct apm_info {
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
status_t apm_shutdown(void);
|
||||
status_t apm_init(struct kernel_args *args);
|
||||
|
||||
#ifdef __cplusplus
|
||||
|
@ -18,12 +18,24 @@
|
||||
# define TRACE(x) ;
|
||||
#endif
|
||||
|
||||
#define APM_ERROR_DISABLED 0x01
|
||||
#define APM_ERROR_DISCONNECTED 0x03
|
||||
#define APM_ERROR_UNKNOWN_DEVICE 0x09
|
||||
#define APM_ERROR_OUT_OF_RANGE 0x0a
|
||||
#define APM_ERROR_DISENGAGED 0x0b
|
||||
#define APM_ERROR_NOT_SUPPORTED 0x0c
|
||||
#define APM_ERROR_RESUME_TIMER_DISABLED 0x0d
|
||||
#define APM_ERROR_UNABLE_TO_ENTER_STATE 0x60
|
||||
#define APM_ERROR_NO_EVENTS_PENDING 0x80
|
||||
#define APM_ERROR_APM_NOT_PRESENT 0x86
|
||||
|
||||
#define CARRY_FLAG 0x01
|
||||
|
||||
extern segment_descriptor *gGDT;
|
||||
extern void *gDmaAddress;
|
||||
extern addr_t gBiosBase;
|
||||
|
||||
static bool sAPMEnabled = false;
|
||||
static struct {
|
||||
uint32 offset;
|
||||
uint16 segment;
|
||||
@ -41,6 +53,39 @@ struct bios_regs {
|
||||
};
|
||||
|
||||
|
||||
#ifdef TRACE_APM
|
||||
static const char *
|
||||
apm_error(uint32 error)
|
||||
{
|
||||
switch (error >> 8) {
|
||||
case APM_ERROR_DISABLED:
|
||||
return "Power Management disabled";
|
||||
case APM_ERROR_DISCONNECTED:
|
||||
return "Interface disconnected";
|
||||
case APM_ERROR_UNKNOWN_DEVICE:
|
||||
return "Unrecognized device ID";
|
||||
case APM_ERROR_OUT_OF_RANGE:
|
||||
return "Parameter value out of range";
|
||||
case APM_ERROR_DISENGAGED:
|
||||
return "Interface not engaged";
|
||||
case APM_ERROR_NOT_SUPPORTED:
|
||||
return "Function not supported";
|
||||
case APM_ERROR_RESUME_TIMER_DISABLED:
|
||||
return "Resume timer disabled";
|
||||
case APM_ERROR_UNABLE_TO_ENTER_STATE:
|
||||
return "Unable to enter requested state";
|
||||
case APM_ERROR_NO_EVENTS_PENDING:
|
||||
return "No power management events pending";
|
||||
case APM_ERROR_APM_NOT_PRESENT:
|
||||
return "APM not present";
|
||||
|
||||
default:
|
||||
return "Unknown error";
|
||||
}
|
||||
}
|
||||
#endif // TRACE_APM
|
||||
|
||||
|
||||
static status_t
|
||||
call_apm_bios(bios_regs *regs)
|
||||
{
|
||||
@ -88,7 +133,12 @@ apm_set_state(uint16 device, uint16 state)
|
||||
regs.ebx = device;
|
||||
regs.ecx = state;
|
||||
|
||||
return call_apm_bios(®s);
|
||||
status_t status = call_apm_bios(®s);
|
||||
if (status == B_OK)
|
||||
return B_OK;
|
||||
|
||||
TRACE(("apm_set_state() error: %s\n", apm_error(regs.eax)));
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
@ -98,17 +148,19 @@ apm_enable_power_management(uint16 device, bool enable)
|
||||
bios_regs regs;
|
||||
regs.eax = BIOS_APM_ENABLE;
|
||||
regs.ebx = device;
|
||||
regs.ecx = enable ? 0x01 : 0x00;
|
||||
|
||||
return call_apm_bios(®s);
|
||||
}
|
||||
|
||||
|
||||
static status_t
|
||||
apm_engage_power_management(uint16 device, bool enable)
|
||||
apm_engage_power_management(uint16 device, bool engage)
|
||||
{
|
||||
bios_regs regs;
|
||||
regs.eax = BIOS_APM_ENGAGE;
|
||||
regs.ebx = device;
|
||||
regs.ecx = engage ? 0x01 : 0x00;
|
||||
|
||||
return call_apm_bios(®s);
|
||||
}
|
||||
@ -143,6 +195,19 @@ apm_daemon(void *arg, int iteration)
|
||||
}
|
||||
|
||||
|
||||
// #pragma mark -
|
||||
|
||||
|
||||
status_t
|
||||
apm_shutdown(void)
|
||||
{
|
||||
if (!sAPMEnabled)
|
||||
return B_NOT_SUPPORTED;
|
||||
|
||||
return apm_set_state(APM_ALL_DEVICES, APM_POWER_STATE_OFF);
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
apm_init(kernel_args *args)
|
||||
{
|
||||
@ -189,9 +254,17 @@ apm_init(kernel_args *args)
|
||||
gGDT[APM_CODE16_SEGMENT >> 3].d_b = 0;
|
||||
// 16-bit segment
|
||||
|
||||
set_segment_descriptor(&gGDT[APM_DATA_SEGMENT >> 3],
|
||||
gBiosBase + (info.data_segment_base << 4) - 0xe0000, info.data_segment_length,
|
||||
DT_DATA_WRITEABLE, DPL_KERNEL);
|
||||
if ((info.data_segment_base << 4) < 0xe0000) {
|
||||
// use the BIOS data segment as data segment for APM
|
||||
set_segment_descriptor(&gGDT[APM_DATA_SEGMENT >> 3],
|
||||
(addr_t)gDmaAddress + (info.data_segment_base << 4), info.data_segment_length,
|
||||
DT_DATA_WRITEABLE, DPL_KERNEL);
|
||||
} else {
|
||||
// use the BIOS area as data segment
|
||||
set_segment_descriptor(&gGDT[APM_DATA_SEGMENT >> 3],
|
||||
gBiosBase + (info.data_segment_base << 4) - 0xe0000, info.data_segment_length,
|
||||
DT_DATA_WRITEABLE, DPL_KERNEL);
|
||||
}
|
||||
|
||||
// setup APM entry point
|
||||
|
||||
@ -208,9 +281,7 @@ apm_init(kernel_args *args)
|
||||
register_kernel_daemon(apm_daemon, NULL, 10);
|
||||
// run the daemon once every second
|
||||
|
||||
//apm_set_state(APM_ALL_DEVICES, APM_POWER_STATE_OFF);
|
||||
panic("done");
|
||||
|
||||
sAPMEnabled = true;
|
||||
return B_OK;
|
||||
}
|
||||
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright 2002-2005, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
|
||||
* Copyright 2002-2006, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
|
||||
* Distributed under the terms of the MIT License.
|
||||
*
|
||||
* Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
|
||||
@ -450,18 +450,25 @@ error:
|
||||
|
||||
|
||||
status_t
|
||||
arch_cpu_shutdown(bool _reboot)
|
||||
arch_cpu_shutdown(bool rebootSystem)
|
||||
{
|
||||
// ToDo: support real shutdown
|
||||
if (!_reboot)
|
||||
return B_NOT_SUPPORTED;
|
||||
cpu_status state = disable_interrupts();
|
||||
|
||||
if (!rebootSystem) {
|
||||
status_t status = apm_shutdown();
|
||||
|
||||
restore_interrupts(state);
|
||||
return status;
|
||||
}
|
||||
|
||||
// try to reset the system using the keyboard controller
|
||||
out8(0xfe, 0x64);
|
||||
|
||||
// if that didn't help, try it this way
|
||||
reboot();
|
||||
return B_OK;
|
||||
|
||||
restore_interrupts(state);
|
||||
return B_ERROR;
|
||||
}
|
||||
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user