2008-01-30 19:32:54 +07:00
|
|
|
/*
|
|
|
|
* sleep.c - x86-specific ACPI sleep support.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2001-2003 Patrick Mochel
|
2010-07-18 19:27:13 +07:00
|
|
|
* Copyright (C) 2001-2003 Pavel Machek <pavel@ucw.cz>
|
2008-01-30 19:32:54 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/acpi.h>
|
|
|
|
#include <linux/bootmem.h>
|
|
|
|
#include <linux/dmi.h>
|
|
|
|
#include <linux/cpumask.h>
|
2008-07-18 01:29:24 +07:00
|
|
|
#include <asm/segment.h>
|
2008-10-17 06:26:27 +07:00
|
|
|
#include <asm/desc.h>
|
2008-01-30 19:32:54 +07:00
|
|
|
|
2010-08-28 20:58:33 +07:00
|
|
|
#ifdef CONFIG_X86_32
|
|
|
|
#include <asm/pgtable.h>
|
|
|
|
#include <asm/pgtable_32.h>
|
|
|
|
#endif
|
|
|
|
|
2008-04-11 04:28:10 +07:00
|
|
|
#include "realmode/wakeup.h"
|
|
|
|
#include "sleep.h"
|
2008-01-30 19:32:54 +07:00
|
|
|
|
2008-02-23 05:11:39 +07:00
|
|
|
unsigned long acpi_wakeup_address;
|
2008-01-30 19:32:54 +07:00
|
|
|
unsigned long acpi_realmode_flags;
|
|
|
|
|
2008-04-11 04:28:10 +07:00
|
|
|
/* address in low memory of the wakeup routine. */
|
|
|
|
static unsigned long acpi_realmode;
|
|
|
|
|
2008-08-04 00:25:48 +07:00
|
|
|
#if defined(CONFIG_SMP) && defined(CONFIG_64BIT)
|
2008-10-09 23:56:21 +07:00
|
|
|
static char temp_stack[4096];
|
2008-04-11 04:28:10 +07:00
|
|
|
#endif
|
2008-01-30 19:32:54 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* acpi_save_state_mem - save kernel state
|
|
|
|
*
|
|
|
|
* Create an identity mapped page table and copy the wakeup routine to
|
|
|
|
* low memory.
|
2008-04-11 04:28:10 +07:00
|
|
|
*
|
|
|
|
* Note that this is too late to change acpi_wakeup_address.
|
2008-01-30 19:32:54 +07:00
|
|
|
*/
|
|
|
|
int acpi_save_state_mem(void)
|
|
|
|
{
|
2008-04-11 04:28:10 +07:00
|
|
|
struct wakeup_header *header;
|
|
|
|
|
|
|
|
if (!acpi_realmode) {
|
|
|
|
printk(KERN_ERR "Could not allocate memory during boot, "
|
|
|
|
"S3 disabled\n");
|
2008-01-30 19:32:54 +07:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2008-04-11 04:28:10 +07:00
|
|
|
memcpy((void *)acpi_realmode, &wakeup_code_start, WAKEUP_SIZE);
|
|
|
|
|
|
|
|
header = (struct wakeup_header *)(acpi_realmode + HEADER_OFFSET);
|
|
|
|
if (header->signature != 0x51ee1111) {
|
|
|
|
printk(KERN_ERR "wakeup header does not match\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
header->video_mode = saved_video_mode;
|
|
|
|
|
2008-06-25 04:03:48 +07:00
|
|
|
header->wakeup_jmp_seg = acpi_wakeup_address >> 4;
|
2008-07-15 01:44:26 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up the wakeup GDT. We set these up as Big Real Mode,
|
|
|
|
* that is, with limits set to 4 GB. At least the Lenovo
|
|
|
|
* Thinkpad X61 is known to need this for the video BIOS
|
|
|
|
* initialization quirk to work; this is likely to also
|
|
|
|
* be the case for other laptops or integrated video devices.
|
|
|
|
*/
|
|
|
|
|
2008-06-25 04:03:48 +07:00
|
|
|
/* GDT[0]: GDT self-pointer */
|
|
|
|
header->wakeup_gdt[0] =
|
|
|
|
(u64)(sizeof(header->wakeup_gdt) - 1) +
|
|
|
|
((u64)(acpi_wakeup_address +
|
|
|
|
((char *)&header->wakeup_gdt - (char *)acpi_realmode))
|
|
|
|
<< 16);
|
2008-07-15 01:44:26 +07:00
|
|
|
/* GDT[1]: big real mode-like code segment */
|
x86, suspend, acpi: enter Big Real Mode
The explanation for recent video BIOS suspend quirk failures is that
the VESA BIOS expects to be entered in Big Real Mode (*.limit = 0xffffffff)
instead of ordinary Real Mode (*.limit = 0xffff).
This patch changes the segment descriptors to Big Real Mode instead.
The segment descriptor registers (what Intel calls "segment cache") is
always active. The only thing that changes based on CR0.PE is how it is
*loaded* and the interpretation of the CS flags.
The segment descriptor registers contain of the following sub-registers:
selector (the "visible" part), base, limit and flags. In protected mode
or long mode, they are loaded from descriptors (or fs.base or gs.base can
be manipulated directly in long mode.) In real mode, the only thing
changed by a segment register load is the selector and the base, where the
base <- selector << 4. In particular, *the limit and the flags are not
changed*.
As far as the handling of the CS flags: a code segment cannot be writable
in protected mode, whereas it is "just another segment" in real mode, so
there is some kind of quirk that kicks in for this when CR0.PE <- 0. I'm
not sure if this is accomplished by actually changing the cs.flags register
or just changing the interpretation; it might be something that is
CPU-specific. In particular, the Transmeta CPUs had an explicit "CS is
writable if you're in real mode" override, so even if you had loaded CS
with an execute-only segment it'd be writable (but not readable!) on return
to real mode. I'm not at all sure if that is how other CPUs behave.
Signed-off-by: "H. Peter Anvin" <hpa@zytor.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-14 11:18:02 +07:00
|
|
|
header->wakeup_gdt[1] =
|
|
|
|
GDT_ENTRY(0x809b, acpi_wakeup_address, 0xfffff);
|
2008-07-15 01:44:26 +07:00
|
|
|
/* GDT[2]: big real mode-like data segment */
|
x86, suspend, acpi: enter Big Real Mode
The explanation for recent video BIOS suspend quirk failures is that
the VESA BIOS expects to be entered in Big Real Mode (*.limit = 0xffffffff)
instead of ordinary Real Mode (*.limit = 0xffff).
This patch changes the segment descriptors to Big Real Mode instead.
The segment descriptor registers (what Intel calls "segment cache") is
always active. The only thing that changes based on CR0.PE is how it is
*loaded* and the interpretation of the CS flags.
The segment descriptor registers contain of the following sub-registers:
selector (the "visible" part), base, limit and flags. In protected mode
or long mode, they are loaded from descriptors (or fs.base or gs.base can
be manipulated directly in long mode.) In real mode, the only thing
changed by a segment register load is the selector and the base, where the
base <- selector << 4. In particular, *the limit and the flags are not
changed*.
As far as the handling of the CS flags: a code segment cannot be writable
in protected mode, whereas it is "just another segment" in real mode, so
there is some kind of quirk that kicks in for this when CR0.PE <- 0. I'm
not sure if this is accomplished by actually changing the cs.flags register
or just changing the interpretation; it might be something that is
CPU-specific. In particular, the Transmeta CPUs had an explicit "CS is
writable if you're in real mode" override, so even if you had loaded CS
with an execute-only segment it'd be writable (but not readable!) on return
to real mode. I'm not at all sure if that is how other CPUs behave.
Signed-off-by: "H. Peter Anvin" <hpa@zytor.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2008-07-14 11:18:02 +07:00
|
|
|
header->wakeup_gdt[2] =
|
|
|
|
GDT_ENTRY(0x8093, acpi_wakeup_address, 0xfffff);
|
2008-06-25 04:03:48 +07:00
|
|
|
|
2008-04-11 04:28:10 +07:00
|
|
|
#ifndef CONFIG_64BIT
|
|
|
|
store_gdt((struct desc_ptr *)&header->pmode_gdt);
|
|
|
|
|
2009-11-14 06:28:14 +07:00
|
|
|
if (rdmsr_safe(MSR_EFER, &header->pmode_efer_low,
|
|
|
|
&header->pmode_efer_high))
|
|
|
|
header->pmode_efer_low = header->pmode_efer_high = 0;
|
2008-04-11 04:28:10 +07:00
|
|
|
#endif /* !CONFIG_64BIT */
|
|
|
|
|
|
|
|
header->pmode_cr0 = read_cr0();
|
2008-08-18 11:03:40 +07:00
|
|
|
header->pmode_cr4 = read_cr4_safe();
|
2008-04-11 04:28:10 +07:00
|
|
|
header->realmode_flags = acpi_realmode_flags;
|
|
|
|
header->real_magic = 0x12345678;
|
|
|
|
|
|
|
|
#ifndef CONFIG_64BIT
|
|
|
|
header->pmode_entry = (u32)&wakeup_pmode_return;
|
2010-08-28 20:58:33 +07:00
|
|
|
header->pmode_cr3 = (u32)__pa(&initial_page_table);
|
2008-04-11 04:28:10 +07:00
|
|
|
saved_magic = 0x12345678;
|
|
|
|
#else /* CONFIG_64BIT */
|
|
|
|
header->trampoline_segment = setup_trampoline() >> 4;
|
2008-06-14 01:31:54 +07:00
|
|
|
#ifdef CONFIG_SMP
|
2008-10-10 00:41:50 +07:00
|
|
|
stack_start.sp = temp_stack + sizeof(temp_stack);
|
2008-10-17 06:26:27 +07:00
|
|
|
early_gdt_descr.address =
|
|
|
|
(unsigned long)get_cpu_gdt_table(smp_processor_id());
|
2009-01-13 18:41:35 +07:00
|
|
|
initial_gs = per_cpu_offset(smp_processor_id());
|
2008-06-14 01:31:54 +07:00
|
|
|
#endif
|
2008-04-11 04:28:10 +07:00
|
|
|
initial_code = (unsigned long)wakeup_long64;
|
2009-04-18 18:44:57 +07:00
|
|
|
saved_magic = 0x123456789abcdef0L;
|
2008-04-11 04:28:10 +07:00
|
|
|
#endif /* CONFIG_64BIT */
|
2008-01-30 19:32:54 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* acpi_restore_state - undo effects of acpi_save_state_mem
|
|
|
|
*/
|
|
|
|
void acpi_restore_state_mem(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2009-11-11 09:27:23 +07:00
|
|
|
* acpi_reserve_wakeup_memory - do _very_ early ACPI initialisation
|
2008-01-30 19:32:54 +07:00
|
|
|
*
|
|
|
|
* We allocate a page from the first 1MB of memory for the wakeup
|
|
|
|
* routine for when we come back from a sleep state. The
|
|
|
|
* runtime allocator allows specification of <16MB pages, but not
|
|
|
|
* <1MB pages.
|
|
|
|
*/
|
2009-11-11 09:27:23 +07:00
|
|
|
void __init acpi_reserve_wakeup_memory(void)
|
2008-01-30 19:32:54 +07:00
|
|
|
{
|
2009-11-11 09:27:23 +07:00
|
|
|
unsigned long mem;
|
|
|
|
|
2008-04-11 04:28:10 +07:00
|
|
|
if ((&wakeup_code_end - &wakeup_code_start) > WAKEUP_SIZE) {
|
2008-01-30 19:32:54 +07:00
|
|
|
printk(KERN_ERR
|
|
|
|
"ACPI: Wakeup code way too big, S3 disabled.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-11-11 09:27:23 +07:00
|
|
|
mem = find_e820_area(0, 1<<20, WAKEUP_SIZE, PAGE_SIZE);
|
2008-04-11 04:28:10 +07:00
|
|
|
|
2009-11-11 09:27:23 +07:00
|
|
|
if (mem == -1L) {
|
2008-01-30 19:32:54 +07:00
|
|
|
printk(KERN_ERR "ACPI: Cannot allocate lowmem, S3 disabled.\n");
|
2008-04-11 04:28:10 +07:00
|
|
|
return;
|
|
|
|
}
|
2009-11-11 09:27:23 +07:00
|
|
|
acpi_realmode = (unsigned long) phys_to_virt(mem);
|
|
|
|
acpi_wakeup_address = mem;
|
|
|
|
reserve_early(mem, mem + WAKEUP_SIZE, "ACPI WAKEUP");
|
2008-01-30 19:32:54 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int __init acpi_sleep_setup(char *str)
|
|
|
|
{
|
|
|
|
while ((str != NULL) && (*str != '\0')) {
|
|
|
|
if (strncmp(str, "s3_bios", 7) == 0)
|
|
|
|
acpi_realmode_flags |= 1;
|
|
|
|
if (strncmp(str, "s3_mode", 7) == 0)
|
|
|
|
acpi_realmode_flags |= 2;
|
|
|
|
if (strncmp(str, "s3_beep", 7) == 0)
|
|
|
|
acpi_realmode_flags |= 4;
|
2008-07-24 11:28:41 +07:00
|
|
|
#ifdef CONFIG_HIBERNATION
|
|
|
|
if (strncmp(str, "s4_nohwsig", 10) == 0)
|
|
|
|
acpi_no_s4_hw_signature();
|
2010-07-24 03:59:09 +07:00
|
|
|
if (strncmp(str, "s4_nonvs", 8) == 0) {
|
|
|
|
pr_warning("ACPI: acpi_sleep=s4_nonvs is deprecated, "
|
|
|
|
"please use acpi_sleep=nonvs instead");
|
|
|
|
acpi_nvs_nosave();
|
|
|
|
}
|
2008-07-24 11:28:41 +07:00
|
|
|
#endif
|
2010-07-24 03:59:09 +07:00
|
|
|
if (strncmp(str, "nonvs", 5) == 0)
|
|
|
|
acpi_nvs_nosave();
|
2008-06-13 04:24:06 +07:00
|
|
|
if (strncmp(str, "old_ordering", 12) == 0)
|
|
|
|
acpi_old_suspend_ordering();
|
2008-01-30 19:32:54 +07:00
|
|
|
str = strchr(str, ',');
|
|
|
|
if (str != NULL)
|
|
|
|
str += strspn(str, ", \t");
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
__setup("acpi_sleep=", acpi_sleep_setup);
|