hc
2024-12-19 9370bb92b2d16684ee45cf24e879c93c509162da
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/*
 * Copyright (C) 2017, Bin Meng <bmeng.cn@gmail.com>
 *
 * SPDX-License-Identifier:    GPL-2.0+
 */
 
#include <common.h>
#include <asm/acpi_s3.h>
#include <asm/acpi_table.h>
#include <asm/post.h>
#include <linux/linkage.h>
 
DECLARE_GLOBAL_DATA_PTR;
 
static void asmlinkage (*acpi_do_wakeup)(void *vector) = (void *)WAKEUP_BASE;
 
static void acpi_jump_to_wakeup(void *vector)
{
   /* Copy wakeup trampoline in place */
   memcpy((void *)WAKEUP_BASE, __wakeup, __wakeup_size);
 
   printf("Jumping to OS waking vector %p\n", vector);
   acpi_do_wakeup(vector);
}
 
void acpi_resume(struct acpi_fadt *fadt)
{
   void *wake_vec;
 
   /* Turn on ACPI mode for S3 */
   enter_acpi_mode(fadt->pm1a_cnt_blk);
 
   wake_vec = acpi_find_wakeup_vector(fadt);
 
   /*
    * Restore the memory content starting from address 0x1000 which is
    * used for the real mode interrupt handler stubs.
    */
   memcpy((void *)0x1000, (const void *)gd->arch.backup_mem,
          S3_RESERVE_SIZE);
 
   post_code(POST_OS_RESUME);
   acpi_jump_to_wakeup(wake_vec);
}
 
int acpi_s3_reserve(void)
{
   /* adjust stack pointer for ACPI S3 resume backup memory */
   gd->start_addr_sp -= S3_RESERVE_SIZE;
   gd->arch.backup_mem = gd->start_addr_sp;
 
   gd->start_addr_sp &= ~0xf;
 
   /*
    * U-Boot sets up the real mode interrupt handler stubs starting from
    * address 0x1000. In most cases, the first 640K (0x00000 - 0x9ffff)
    * system memory is reported as system RAM in E820 table to the OS.
    * (see install_e820_map() implementation for each platform). So OS
    * can use these memories whatever it wants.
    *
    * If U-Boot is in an S3 resume path, care must be taken not to corrupt
    * these memorie otherwise OS data gets lost. Testing shows that, on
    * Microsoft Windows 10 on Intel Baytrail its wake up vector happens to
    * be installed at the same address 0x1000. While on Linux its wake up
    * vector does not overlap this memory range, but after resume kernel
    * checks low memory range per config option CONFIG_X86_RESERVE_LOW
    * which is 64K by default to see whether a memory corruption occurs
    * during the suspend/resume (it's harmless, but warnings are shown
    * in the kernel dmesg logs).
    *
    * We cannot simply mark the these memory as reserved in E820 table
    * because such configuration makes GRUB complain: unable to allocate
    * real mode page. Hence we choose to back up these memories to the
    * place where we reserved on our stack for our S3 resume work.
    * Before jumping to OS wake up vector, we need restore the original
    * content there (see acpi_resume() above).
    */
   if (gd->arch.prev_sleep_state == ACPI_S3)
       memcpy((void *)gd->arch.backup_mem, (const void *)0x1000,
              S3_RESERVE_SIZE);
 
   return 0;
}