Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 1 | // Code for handling calls to "post" that are resume related. |
| 2 | // |
Kevin O'Connor | 1445843 | 2009-05-23 18:21:18 -0400 | [diff] [blame] | 3 | // Copyright (C) 2008,2009 Kevin O'Connor <kevin@koconnor.net> |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 4 | // |
Kevin O'Connor | b1b7c2a | 2009-01-15 20:52:58 -0500 | [diff] [blame] | 5 | // This file may be distributed under the terms of the GNU LGPLv3 license. |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 6 | |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 7 | #include "bregs.h" // struct bregs |
Kevin O'Connor | 392d2aa | 2013-09-15 00:14:28 -0400 | [diff] [blame] | 8 | #include "config.h" // CONFIG_* |
| 9 | #include "farptr.h" // FLATPTR_TO_SEGOFF |
Kevin O'Connor | 2d2fa31 | 2013-09-14 21:55:26 -0400 | [diff] [blame] | 10 | #include "hw/pci.h" // pci_reboot |
| 11 | #include "hw/pic.h" // pic_eoi2 |
| 12 | #include "hw/ps2port.h" // i8042_reboot |
Kevin O'Connor | 8b7861c | 2013-09-15 02:29:06 -0400 | [diff] [blame] | 13 | #include "hw/rtc.h" // rtc_read |
Kevin O'Connor | 2d2fa31 | 2013-09-14 21:55:26 -0400 | [diff] [blame] | 14 | #include "output.h" // dprintf |
Kevin O'Connor | 3df600b | 2013-09-14 19:28:55 -0400 | [diff] [blame] | 15 | #include "stacks.h" // farcall16big |
Kevin O'Connor | 392d2aa | 2013-09-15 00:14:28 -0400 | [diff] [blame] | 16 | #include "std/bda.h" // struct bios_data_area_s |
Kevin O'Connor | fa9c66a | 2013-09-14 19:10:40 -0400 | [diff] [blame] | 17 | #include "string.h" // memset |
Kevin O'Connor | 2d2fa31 | 2013-09-14 21:55:26 -0400 | [diff] [blame] | 18 | #include "util.h" // dma_setup |
Stefan Berger | b310dfa | 2015-03-23 14:22:16 -0400 | [diff] [blame] | 19 | #include "tcgbios.h" // tpm_s3_resume |
Ben Warren | 8f598a4 | 2017-02-20 19:56:19 -0800 | [diff] [blame] | 20 | #include "fw/romfile_loader.h" // romfile_fw_cfg_resume |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 21 | |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 22 | // Handler for post calls that look like a resume. |
| 23 | void VISIBLE16 |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 24 | handle_resume(void) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 25 | { |
Kevin O'Connor | a026308 | 2012-04-14 20:22:18 -0400 | [diff] [blame] | 26 | ASSERT16(); |
Kevin O'Connor | 8b7861c | 2013-09-15 02:29:06 -0400 | [diff] [blame] | 27 | int status = rtc_read(CMOS_RESET_CODE); |
| 28 | rtc_write(CMOS_RESET_CODE, 0); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 29 | dprintf(1, "In resume (status=%d)\n", status); |
| 30 | |
Kevin O'Connor | 9e4d41c | 2013-01-21 12:14:29 -0500 | [diff] [blame] | 31 | dma_setup(); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 32 | |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 33 | switch (status) { |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 34 | case 0x01 ... 0x04: |
| 35 | case 0x06 ... 0x09: |
| 36 | panic("Unimplemented shutdown status: %02x\n", status); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 37 | |
| 38 | case 0x05: |
| 39 | // flush keyboard (issue EOI) and jump via 40h:0067h |
Kevin O'Connor | aa7c234 | 2013-07-14 15:07:21 -0400 | [diff] [blame] | 40 | pic_eoi2(); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 41 | // NO BREAK |
| 42 | case 0x0a: |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 43 | #define BDA_JUMP (((struct bios_data_area_s *)0)->jump) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 44 | // resume execution by jump via 40h:0067h |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 45 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 46 | "movw %w1, %%ds\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 47 | "ljmpw *%0\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 48 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 49 | ); |
| 50 | break; |
| 51 | |
| 52 | case 0x0b: |
| 53 | // resume execution via IRET via 40h:0067h |
| 54 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 55 | "movw %w1, %%ds\n" |
| 56 | "lssw %0, %%sp\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 57 | "iretw\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 58 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 59 | ); |
| 60 | break; |
| 61 | |
| 62 | case 0x0c: |
| 63 | // resume execution via RETF via 40h:0067h |
| 64 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 65 | "movw %w1, %%ds\n" |
| 66 | "lssw %0, %%sp\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 67 | "lretw\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 68 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 69 | ); |
| 70 | break; |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 71 | |
| 72 | default: |
| 73 | break; |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 74 | } |
| 75 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 76 | // Not a 16bit resume - do remaining checks in 32bit mode |
| 77 | asm volatile( |
| 78 | "movw %w1, %%ss\n" |
| 79 | "movl %0, %%esp\n" |
| 80 | "movl $_cfunc32flat_handle_resume32, %%edx\n" |
| 81 | "jmp transition32\n" |
| 82 | : : "i"(BUILD_S3RESUME_STACK_ADDR), "r"(0), "a"(status) |
| 83 | ); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 84 | } |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 85 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 86 | // Handle an S3 resume event |
| 87 | static void |
Kevin O'Connor | 1ca05b0 | 2010-01-03 17:43:37 -0500 | [diff] [blame] | 88 | s3_resume(void) |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 89 | { |
Kevin O'Connor | b24c574 | 2009-01-17 21:52:52 -0500 | [diff] [blame] | 90 | if (!CONFIG_S3_RESUME) |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 91 | return; |
Kevin O'Connor | b24c574 | 2009-01-17 21:52:52 -0500 | [diff] [blame] | 92 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 93 | u32 s3_resume_vector = find_resume_vector(); |
| 94 | if (!s3_resume_vector) { |
| 95 | dprintf(1, "No resume vector set!\n"); |
| 96 | return; |
| 97 | } |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 98 | |
Gleb Natapov | ecae9bf | 2012-01-24 14:33:42 +0200 | [diff] [blame] | 99 | pic_setup(); |
Kevin O'Connor | d83c87b | 2013-01-21 01:14:12 -0500 | [diff] [blame] | 100 | smm_setup(); |
Paolo Bonzini | 54e3a88 | 2016-07-07 16:00:40 +0200 | [diff] [blame] | 101 | smp_resume(); |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 102 | |
Marcel Apfelbaum | 40d020f | 2014-01-15 14:20:06 +0200 | [diff] [blame] | 103 | pci_resume(); |
| 104 | |
Stefan Berger | b310dfa | 2015-03-23 14:22:16 -0400 | [diff] [blame] | 105 | /* resume TPM before we may measure option roms */ |
| 106 | tpm_s3_resume(); |
Kevin O'Connor | d83c87b | 2013-01-21 01:14:12 -0500 | [diff] [blame] | 107 | s3_resume_vga(); |
Kevin O'Connor | d282af7 | 2009-07-04 04:10:32 -0400 | [diff] [blame] | 108 | |
Ben Warren | 8f598a4 | 2017-02-20 19:56:19 -0800 | [diff] [blame] | 109 | /* Replay any fw_cfg entries that go back to the host */ |
| 110 | romfile_fw_cfg_resume(); |
| 111 | |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 112 | make_bios_readonly(); |
| 113 | |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 114 | // Invoke the resume vector. |
| 115 | struct bregs br; |
| 116 | memset(&br, 0, sizeof(br)); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 117 | dprintf(1, "Jump to resume vector (%x)\n", s3_resume_vector); |
| 118 | br.code = FLATPTR_TO_SEGOFF((void*)s3_resume_vector); |
Kevin O'Connor | c7ffbac | 2012-03-25 11:04:10 -0400 | [diff] [blame] | 119 | farcall16big(&br); |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 120 | } |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 121 | |
| 122 | // Attempt to invoke a hard-reboot. |
| 123 | static void |
| 124 | tryReboot(void) |
| 125 | { |
| 126 | dprintf(1, "Attempting a hard reboot\n"); |
| 127 | |
Kevin O'Connor | c68aff5 | 2017-03-03 10:48:45 -0500 | [diff] [blame] | 128 | // Use a QEMU specific reboot on QEMU |
| 129 | qemu_reboot(); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 130 | |
David Woodhouse | d338eb9 | 2013-02-23 00:24:49 +0000 | [diff] [blame] | 131 | // Reboot using ACPI RESET_REG |
| 132 | acpi_reboot(); |
| 133 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 134 | // Try keyboard controller reboot. |
| 135 | i8042_reboot(); |
| 136 | |
| 137 | // Try PCI 0xcf9 reboot |
| 138 | pci_reboot(); |
| 139 | |
| 140 | // Try triple fault |
| 141 | asm volatile("int3"); |
| 142 | |
| 143 | panic("Could not reboot"); |
| 144 | } |
| 145 | |
| 146 | void VISIBLE32FLAT |
| 147 | handle_resume32(int status) |
| 148 | { |
| 149 | ASSERT32FLAT(); |
| 150 | dprintf(1, "In 32bit resume\n"); |
| 151 | |
| 152 | if (status == 0xfe) |
| 153 | s3_resume(); |
| 154 | |
| 155 | // Must be a soft reboot - invoke a hard reboot. |
| 156 | tryReboot(); |
| 157 | } |