Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the coreboot project. |
| 3 | * |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 4 | * Copyright (C) 2008-2010 coresystems GmbH |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 5 | * |
| 6 | * This program is free software; you can redistribute it and/or |
| 7 | * modify it under the terms of the GNU General Public License as |
| 8 | * published by the Free Software Foundation; version 2 of |
| 9 | * the License. |
| 10 | * |
| 11 | * This program is distributed in the hope that it will be useful, |
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 14 | * GNU General Public License for more details. |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 15 | */ |
| 16 | |
Stefan Reinauer | 4da810b | 2009-07-21 21:41:42 +0000 | [diff] [blame] | 17 | // Make sure no stage 2 code is included: |
Myles Watson | 1d6d45e | 2009-11-06 17:02:51 +0000 | [diff] [blame] | 18 | #define __PRE_RAM__ |
Stefan Reinauer | 4da810b | 2009-07-21 21:41:42 +0000 | [diff] [blame] | 19 | |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 20 | // FIXME: Is this piece of code southbridge specific, or |
Stefan Reinauer | 4da810b | 2009-07-21 21:41:42 +0000 | [diff] [blame] | 21 | // can it be cleaned up so this include is not required? |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 22 | // It's needed right now because we get our DEFAULT_PMBASE from |
Stefan Reinauer | 5f5436f | 2010-04-25 20:42:02 +0000 | [diff] [blame] | 23 | // here. |
Stefan Reinauer | 582748f | 2011-04-19 01:18:54 +0000 | [diff] [blame] | 24 | #if CONFIG_SOUTHBRIDGE_INTEL_I82801GX |
Stefan Reinauer | 4da810b | 2009-07-21 21:41:42 +0000 | [diff] [blame] | 25 | #include "../../../southbridge/intel/i82801gx/i82801gx.h" |
Stefan Reinauer | 582748f | 2011-04-19 01:18:54 +0000 | [diff] [blame] | 26 | #elif CONFIG_SOUTHBRIDGE_INTEL_I82801DX |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 27 | #include "../../../southbridge/intel/i82801dx/i82801dx.h" |
Stefan Reinauer | 4bab6e7 | 2016-05-03 15:53:33 -0700 | [diff] [blame] | 28 | #elif CONFIG_SOC_INTEL_SCH |
| 29 | #include "../../../soc/intel/sch/sch.h" |
Patrick Georgi | e72a8a3 | 2012-11-06 11:05:09 +0100 | [diff] [blame] | 30 | #elif CONFIG_SOUTHBRIDGE_INTEL_I82801IX |
| 31 | #include "../../../southbridge/intel/i82801ix/i82801ix.h" |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 32 | #else |
| 33 | #error "Southbridge needs SMM handler support." |
| 34 | #endif |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 35 | |
Stefan Reinauer | 3aa067f | 2012-04-02 13:24:04 -0700 | [diff] [blame] | 36 | #if CONFIG_SMM_TSEG |
Vladimir Serbinenko | 7f46420 | 2015-05-28 21:03:51 +0200 | [diff] [blame] | 37 | #error "Don't use this file with TSEG." |
Stefan Reinauer | 3aa067f | 2012-04-02 13:24:04 -0700 | [diff] [blame] | 38 | |
| 39 | #endif /* CONFIG_SMM_TSEG */ |
| 40 | |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 41 | #define LAPIC_ID 0xfee00020 |
| 42 | |
| 43 | .global smm_relocation_start |
| 44 | .global smm_relocation_end |
| 45 | |
| 46 | /* initially SMM is some sort of real mode. */ |
| 47 | .code16 |
| 48 | |
| 49 | /** |
Stefan Reinauer | 8c5b58e | 2012-04-04 10:38:05 -0700 | [diff] [blame] | 50 | * When starting up, x86 CPUs have their SMBASE set to 0x30000. However, |
| 51 | * this is not a good place for the SMM handler to live, so it needs to |
| 52 | * be relocated. |
| 53 | * Traditionally SMM handlers used to live in the A segment (0xa0000). |
| 54 | * With growing SMM handlers, more CPU cores, etc. CPU vendors started |
| 55 | * allowing to relocate the handler to the end of physical memory, which |
| 56 | * they refer to as TSEG. |
| 57 | * This trampoline code relocates SMBASE to base address - ( lapicid * 0x400 ) |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 58 | * |
| 59 | * Why 0x400? It is a safe value to cover the save state area per CPU. On |
| 60 | * current AMD CPUs this area is _documented_ to be 0x200 bytes. On Intel |
| 61 | * Core 2 CPUs the _documented_ parts of the save state area is 48 bytes |
| 62 | * bigger, effectively sizing our data structures 0x300 bytes. |
| 63 | * |
Stefan Reinauer | 8c5b58e | 2012-04-04 10:38:05 -0700 | [diff] [blame] | 64 | * Example (with SMM handler living at 0xa0000): |
| 65 | * |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 66 | * LAPICID SMBASE SMM Entry SAVE STATE |
| 67 | * 0 0xa0000 0xa8000 0xafd00 |
| 68 | * 1 0x9fc00 0xa7c00 0xaf900 |
| 69 | * 2 0x9f800 0xa7800 0xaf500 |
| 70 | * 3 0x9f400 0xa7400 0xaf100 |
| 71 | * 4 0x9f000 0xa7000 0xaed00 |
| 72 | * 5 0x9ec00 0xa6c00 0xae900 |
| 73 | * 6 0x9e800 0xa6800 0xae500 |
| 74 | * 7 0x9e400 0xa6400 0xae100 |
| 75 | * 8 0x9e000 0xa6000 0xadd00 |
| 76 | * 9 0x9dc00 0xa5c00 0xad900 |
| 77 | * 10 0x9d800 0xa5800 0xad500 |
| 78 | * 11 0x9d400 0xa5400 0xad100 |
| 79 | * 12 0x9d000 0xa5000 0xacd00 |
| 80 | * 13 0x9cc00 0xa4c00 0xac900 |
| 81 | * 14 0x9c800 0xa4800 0xac500 |
| 82 | * 15 0x9c400 0xa4400 0xac100 |
| 83 | * . . . . |
| 84 | * . . . . |
| 85 | * . . . . |
| 86 | * 31 0x98400 0xa0400 0xa8100 |
| 87 | * |
| 88 | * With 32 cores, the SMM handler would need to fit between |
| 89 | * 0xa0000-0xa0400 and the stub plus stack would need to go |
| 90 | * at 0xa8000-0xa8100 (example for core 0). That is not enough. |
| 91 | * |
Elyes HAOUAS | d82be92 | 2016-07-28 18:58:27 +0200 | [diff] [blame] | 92 | * This means we're basically limited to 16 CPU cores before |
Stefan Reinauer | 8c5b58e | 2012-04-04 10:38:05 -0700 | [diff] [blame] | 93 | * we need to move the SMM handler to TSEG. |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 94 | * |
| 95 | * Note: Some versions of Pentium M need their SMBASE aligned to 32k. |
| 96 | * On those the above only works for up to 2 cores. But for now we only |
| 97 | * care fore Core (2) Duo/Solo |
| 98 | * |
| 99 | */ |
| 100 | |
| 101 | smm_relocation_start: |
| 102 | /* Check revision to see if AMD64 style SMM_BASE |
| 103 | * Intel Core Solo/Duo: 0x30007 |
| 104 | * Intel Core2 Solo/Duo: 0x30100 |
Stefan Reinauer | 3aa067f | 2012-04-02 13:24:04 -0700 | [diff] [blame] | 105 | * Intel SandyBridge: 0x30101 |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 106 | * AMD64: 0x3XX64 |
| 107 | * This check does not make much sense, unless someone ports |
| 108 | * SMI handling to AMD64 CPUs. |
| 109 | */ |
| 110 | |
| 111 | mov $0x38000 + 0x7efc, %ebx |
| 112 | addr32 mov (%ebx), %al |
| 113 | cmp $0x64, %al |
| 114 | je 1f |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 115 | |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 116 | mov $0x38000 + 0x7ef8, %ebx |
| 117 | jmp smm_relocate |
| 118 | 1: |
| 119 | mov $0x38000 + 0x7f00, %ebx |
| 120 | |
| 121 | smm_relocate: |
| 122 | /* Get this CPU's LAPIC ID */ |
| 123 | movl $LAPIC_ID, %esi |
| 124 | addr32 movl (%esi), %ecx |
| 125 | shr $24, %ecx |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 126 | |
| 127 | /* calculate offset by multiplying the |
Elyes HAOUAS | d6e9686 | 2016-08-21 10:12:15 +0200 | [diff] [blame] | 128 | * APIC ID by 1024 (0x400) |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 129 | */ |
| 130 | movl %ecx, %edx |
| 131 | shl $10, %edx |
| 132 | |
| 133 | movl $0xa0000, %eax |
| 134 | subl %edx, %eax /* subtract offset, see above */ |
| 135 | |
| 136 | addr32 movl %eax, (%ebx) |
| 137 | |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 138 | /* The next section of code is potentially southbridge specific */ |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 139 | |
| 140 | /* Clear SMI status */ |
| 141 | movw $(DEFAULT_PMBASE + 0x34), %dx |
| 142 | inw %dx, %ax |
| 143 | outw %ax, %dx |
| 144 | |
| 145 | /* Clear PM1 status */ |
| 146 | movw $(DEFAULT_PMBASE + 0x00), %dx |
| 147 | inw %dx, %ax |
| 148 | outw %ax, %dx |
| 149 | |
| 150 | /* Set EOS bit so other SMIs can occur */ |
| 151 | movw $(DEFAULT_PMBASE + 0x30), %dx |
| 152 | inl %dx, %eax |
| 153 | orl $(1 << 1), %eax |
| 154 | outl %eax, %dx |
| 155 | |
Stefan Reinauer | bc0f7a6 | 2010-08-01 15:41:14 +0000 | [diff] [blame] | 156 | /* End of southbridge specific section. */ |
| 157 | |
Stefan Reinauer | 582748f | 2011-04-19 01:18:54 +0000 | [diff] [blame] | 158 | #if CONFIG_DEBUG_SMM_RELOCATION |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 159 | /* print [SMM-x] so we can determine if CPUx went to SMM */ |
Stefan Reinauer | 0867062 | 2009-06-30 15:17:49 +0000 | [diff] [blame] | 160 | movw $CONFIG_TTYS0_BASE, %dx |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 161 | mov $'[', %al |
| 162 | outb %al, %dx |
| 163 | mov $'S', %al |
| 164 | outb %al, %dx |
| 165 | mov $'M', %al |
| 166 | outb %al, %dx |
| 167 | outb %al, %dx |
| 168 | movb $'-', %al |
| 169 | outb %al, %dx |
Elyes HAOUAS | d82be92 | 2016-07-28 18:58:27 +0200 | [diff] [blame] | 170 | /* calculate ascii of CPU number. More than 9 cores? -> FIXME */ |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 171 | movb %cl, %al |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 172 | addb $'0', %al |
Stefan Reinauer | debb11f | 2008-10-29 04:46:52 +0000 | [diff] [blame] | 173 | outb %al, %dx |
| 174 | mov $']', %al |
| 175 | outb %al, %dx |
| 176 | mov $'\r', %al |
| 177 | outb %al, %dx |
| 178 | mov $'\n', %al |
| 179 | outb %al, %dx |
| 180 | #endif |
| 181 | |
| 182 | /* That's it. return */ |
| 183 | rsm |
| 184 | smm_relocation_end: |