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 |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 20 | |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 21 | // Handler for post calls that look like a resume. |
| 22 | void VISIBLE16 |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 23 | handle_resume(void) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 24 | { |
Kevin O'Connor | a026308 | 2012-04-14 20:22:18 -0400 | [diff] [blame] | 25 | ASSERT16(); |
Kevin O'Connor | 8b7861c | 2013-09-15 02:29:06 -0400 | [diff] [blame] | 26 | int status = rtc_read(CMOS_RESET_CODE); |
| 27 | rtc_write(CMOS_RESET_CODE, 0); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 28 | dprintf(1, "In resume (status=%d)\n", status); |
| 29 | |
Kevin O'Connor | 9e4d41c | 2013-01-21 12:14:29 -0500 | [diff] [blame] | 30 | dma_setup(); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 31 | |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 32 | switch (status) { |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 33 | case 0x01 ... 0x04: |
| 34 | case 0x06 ... 0x09: |
| 35 | panic("Unimplemented shutdown status: %02x\n", status); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 36 | |
| 37 | case 0x05: |
| 38 | // flush keyboard (issue EOI) and jump via 40h:0067h |
Kevin O'Connor | aa7c234 | 2013-07-14 15:07:21 -0400 | [diff] [blame] | 39 | pic_eoi2(); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 40 | // NO BREAK |
| 41 | case 0x0a: |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 42 | #define BDA_JUMP (((struct bios_data_area_s *)0)->jump) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 43 | // resume execution by jump via 40h:0067h |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 44 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 45 | "movw %w1, %%ds\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 46 | "ljmpw *%0\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 47 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 48 | ); |
| 49 | break; |
| 50 | |
| 51 | case 0x0b: |
| 52 | // resume execution via IRET via 40h:0067h |
| 53 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 54 | "movw %w1, %%ds\n" |
| 55 | "lssw %0, %%sp\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 56 | "iretw\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 57 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 58 | ); |
| 59 | break; |
| 60 | |
| 61 | case 0x0c: |
| 62 | // resume execution via RETF via 40h:0067h |
| 63 | asm volatile( |
Kevin O'Connor | 590e715 | 2009-01-19 12:53:54 -0500 | [diff] [blame] | 64 | "movw %w1, %%ds\n" |
| 65 | "lssw %0, %%sp\n" |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 66 | "lretw\n" |
Kevin O'Connor | 9f98542 | 2009-09-09 11:34:39 -0400 | [diff] [blame] | 67 | : : "m"(BDA_JUMP), "r"(SEG_BDA) |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 68 | ); |
| 69 | break; |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 70 | |
| 71 | default: |
| 72 | break; |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 73 | } |
| 74 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 75 | // Not a 16bit resume - do remaining checks in 32bit mode |
| 76 | asm volatile( |
| 77 | "movw %w1, %%ss\n" |
| 78 | "movl %0, %%esp\n" |
| 79 | "movl $_cfunc32flat_handle_resume32, %%edx\n" |
| 80 | "jmp transition32\n" |
| 81 | : : "i"(BUILD_S3RESUME_STACK_ADDR), "r"(0), "a"(status) |
| 82 | ); |
Kevin O'Connor | 18e38b2 | 2008-12-10 20:40:13 -0500 | [diff] [blame] | 83 | } |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 84 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 85 | // Handle an S3 resume event |
| 86 | static void |
Kevin O'Connor | 1ca05b0 | 2010-01-03 17:43:37 -0500 | [diff] [blame] | 87 | s3_resume(void) |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 88 | { |
Kevin O'Connor | b24c574 | 2009-01-17 21:52:52 -0500 | [diff] [blame] | 89 | if (!CONFIG_S3_RESUME) |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 90 | return; |
Kevin O'Connor | b24c574 | 2009-01-17 21:52:52 -0500 | [diff] [blame] | 91 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 92 | u32 s3_resume_vector = find_resume_vector(); |
| 93 | if (!s3_resume_vector) { |
| 94 | dprintf(1, "No resume vector set!\n"); |
| 95 | return; |
| 96 | } |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 97 | |
Gleb Natapov | ecae9bf | 2012-01-24 14:33:42 +0200 | [diff] [blame] | 98 | pic_setup(); |
Kevin O'Connor | d83c87b | 2013-01-21 01:14:12 -0500 | [diff] [blame] | 99 | smm_setup(); |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 100 | |
Marcel Apfelbaum | 40d020f | 2014-01-15 14:20:06 +0200 | [diff] [blame] | 101 | pci_resume(); |
| 102 | |
Stefan Berger | b310dfa | 2015-03-23 14:22:16 -0400 | [diff] [blame] | 103 | /* resume TPM before we may measure option roms */ |
| 104 | tpm_s3_resume(); |
Kevin O'Connor | d83c87b | 2013-01-21 01:14:12 -0500 | [diff] [blame] | 105 | s3_resume_vga(); |
Kevin O'Connor | d282af7 | 2009-07-04 04:10:32 -0400 | [diff] [blame] | 106 | |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 107 | make_bios_readonly(); |
| 108 | |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 109 | // Invoke the resume vector. |
| 110 | struct bregs br; |
| 111 | memset(&br, 0, sizeof(br)); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 112 | dprintf(1, "Jump to resume vector (%x)\n", s3_resume_vector); |
| 113 | br.code = FLATPTR_TO_SEGOFF((void*)s3_resume_vector); |
Kevin O'Connor | c7ffbac | 2012-03-25 11:04:10 -0400 | [diff] [blame] | 114 | farcall16big(&br); |
Kevin O'Connor | 9967ab7 | 2008-12-18 21:57:33 -0500 | [diff] [blame] | 115 | } |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 116 | |
Kevin O'Connor | 400c66c | 2013-02-23 10:37:58 -0500 | [diff] [blame] | 117 | u8 HaveAttemptedReboot VARLOW; |
| 118 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 119 | // Attempt to invoke a hard-reboot. |
| 120 | static void |
| 121 | tryReboot(void) |
| 122 | { |
Kevin O'Connor | 400c66c | 2013-02-23 10:37:58 -0500 | [diff] [blame] | 123 | if (HaveAttemptedReboot) { |
| 124 | // Hard reboot has failed - try to shutdown machine. |
| 125 | dprintf(1, "Unable to hard-reboot machine - attempting shutdown.\n"); |
| 126 | apm_shutdown(); |
| 127 | } |
| 128 | HaveAttemptedReboot = 1; |
| 129 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 130 | dprintf(1, "Attempting a hard reboot\n"); |
| 131 | |
| 132 | // Setup for reset on qemu. |
Kevin O'Connor | 897fb11 | 2013-02-07 23:32:48 -0500 | [diff] [blame] | 133 | qemu_prep_reset(); |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 134 | |
David Woodhouse | d338eb9 | 2013-02-23 00:24:49 +0000 | [diff] [blame] | 135 | // Reboot using ACPI RESET_REG |
| 136 | acpi_reboot(); |
| 137 | |
Kevin O'Connor | 87b533b | 2011-07-10 22:35:07 -0400 | [diff] [blame] | 138 | // Try keyboard controller reboot. |
| 139 | i8042_reboot(); |
| 140 | |
| 141 | // Try PCI 0xcf9 reboot |
| 142 | pci_reboot(); |
| 143 | |
| 144 | // Try triple fault |
| 145 | asm volatile("int3"); |
| 146 | |
| 147 | panic("Could not reboot"); |
| 148 | } |
| 149 | |
| 150 | void VISIBLE32FLAT |
| 151 | handle_resume32(int status) |
| 152 | { |
| 153 | ASSERT32FLAT(); |
| 154 | dprintf(1, "In 32bit resume\n"); |
| 155 | |
| 156 | if (status == 0xfe) |
| 157 | s3_resume(); |
| 158 | |
| 159 | // Must be a soft reboot - invoke a hard reboot. |
| 160 | tryReboot(); |
| 161 | } |