0156-x86-entry-64-Move-the-IST-stacks-into-struct-cpu_ent.patch 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
  2. From: Andy Lutomirski <[email protected]>
  3. Date: Mon, 4 Dec 2017 15:07:26 +0100
  4. Subject: [PATCH] x86/entry/64: Move the IST stacks into struct cpu_entry_area
  5. MIME-Version: 1.0
  6. Content-Type: text/plain; charset=UTF-8
  7. Content-Transfer-Encoding: 8bit
  8. CVE-2017-5754
  9. The IST stacks are needed when an IST exception occurs and are accessed
  10. before any kernel code at all runs. Move them into struct cpu_entry_area.
  11. The IST stacks are unlike the rest of cpu_entry_area: they're used even for
  12. entries from kernel mode. This means that they should be set up before we
  13. load the final IDT. Move cpu_entry_area setup to trap_init() for the boot
  14. CPU and set it up for all possible CPUs at once in native_smp_prepare_cpus().
  15. Signed-off-by: Andy Lutomirski <[email protected]>
  16. Signed-off-by: Thomas Gleixner <[email protected]>
  17. Reviewed-by: Thomas Gleixner <[email protected]>
  18. Reviewed-by: Borislav Petkov <[email protected]>
  19. Cc: Boris Ostrovsky <[email protected]>
  20. Cc: Borislav Petkov <[email protected]>
  21. Cc: Borislav Petkov <[email protected]>
  22. Cc: Brian Gerst <[email protected]>
  23. Cc: Dave Hansen <[email protected]>
  24. Cc: Dave Hansen <[email protected]>
  25. Cc: David Laight <[email protected]>
  26. Cc: Denys Vlasenko <[email protected]>
  27. Cc: Eduardo Valentin <[email protected]>
  28. Cc: Greg KH <[email protected]>
  29. Cc: H. Peter Anvin <[email protected]>
  30. Cc: Josh Poimboeuf <[email protected]>
  31. Cc: Juergen Gross <[email protected]>
  32. Cc: Linus Torvalds <[email protected]>
  33. Cc: Peter Zijlstra <[email protected]>
  34. Cc: Rik van Riel <[email protected]>
  35. Cc: Will Deacon <[email protected]>
  36. Cc: [email protected]
  37. Cc: [email protected]
  38. Cc: [email protected]
  39. Cc: [email protected]
  40. Link: https://lkml.kernel.org/r/[email protected]
  41. Signed-off-by: Ingo Molnar <[email protected]>
  42. (backported from commit 40e7f949e0d9a33968ebde5d67f7e3a47c97742a)
  43. Signed-off-by: Andy Whitcroft <[email protected]>
  44. Signed-off-by: Kleber Sacilotto de Souza <[email protected]>
  45. (cherry picked from commit 88e7277709f2e7c023e66ff9ae158aeff4cf7c8f)
  46. Signed-off-by: Fabian Grünbichler <[email protected]>
  47. ---
  48. arch/x86/include/asm/fixmap.h | 12 +++++++
  49. arch/x86/kernel/cpu/common.c | 74 ++++++++++++++++++++++++-------------------
  50. arch/x86/kernel/traps.c | 3 ++
  51. 3 files changed, 57 insertions(+), 32 deletions(-)
  52. diff --git a/arch/x86/include/asm/fixmap.h b/arch/x86/include/asm/fixmap.h
  53. index 189d12d8afe0..953aed54cb5e 100644
  54. --- a/arch/x86/include/asm/fixmap.h
  55. +++ b/arch/x86/include/asm/fixmap.h
  56. @@ -63,10 +63,22 @@ struct cpu_entry_area {
  57. struct tss_struct tss;
  58. char entry_trampoline[PAGE_SIZE];
  59. +
  60. +#ifdef CONFIG_X86_64
  61. + /*
  62. + * Exception stacks used for IST entries.
  63. + *
  64. + * In the future, this should have a separate slot for each stack
  65. + * with guard pages between them.
  66. + */
  67. + char exception_stacks[(N_EXCEPTION_STACKS - 1) * EXCEPTION_STKSZ + DEBUG_STKSZ];
  68. +#endif
  69. };
  70. #define CPU_ENTRY_AREA_PAGES (sizeof(struct cpu_entry_area) / PAGE_SIZE)
  71. +extern void setup_cpu_entry_areas(void);
  72. +
  73. /*
  74. * Here we define all the compile-time 'special' virtual
  75. * addresses. The point is to have a constant address at
  76. diff --git a/arch/x86/kernel/cpu/common.c b/arch/x86/kernel/cpu/common.c
  77. index c2b2ee73b8a1..f487766855d3 100644
  78. --- a/arch/x86/kernel/cpu/common.c
  79. +++ b/arch/x86/kernel/cpu/common.c
  80. @@ -466,24 +466,36 @@ void load_percpu_segment(int cpu)
  81. load_stack_canary_segment();
  82. }
  83. -static void set_percpu_fixmap_pages(int fixmap_index, void *ptr,
  84. - int pages, pgprot_t prot)
  85. -{
  86. - int i;
  87. -
  88. - for (i = 0; i < pages; i++) {
  89. - __set_fixmap(fixmap_index - i,
  90. - per_cpu_ptr_to_phys(ptr + i * PAGE_SIZE), prot);
  91. - }
  92. -}
  93. -
  94. #ifdef CONFIG_X86_32
  95. /* The 32-bit entry code needs to find cpu_entry_area. */
  96. DEFINE_PER_CPU(struct cpu_entry_area *, cpu_entry_area);
  97. #endif
  98. +#ifdef CONFIG_X86_64
  99. +/*
  100. + * Special IST stacks which the CPU switches to when it calls
  101. + * an IST-marked descriptor entry. Up to 7 stacks (hardware
  102. + * limit), all of them are 4K, except the debug stack which
  103. + * is 8K.
  104. + */
  105. +static const unsigned int exception_stack_sizes[N_EXCEPTION_STACKS] = {
  106. + [0 ... N_EXCEPTION_STACKS - 1] = EXCEPTION_STKSZ,
  107. + [DEBUG_STACK - 1] = DEBUG_STKSZ
  108. +};
  109. +
  110. +static DEFINE_PER_CPU_PAGE_ALIGNED(char, exception_stacks
  111. + [(N_EXCEPTION_STACKS - 1) * EXCEPTION_STKSZ + DEBUG_STKSZ]);
  112. +#endif
  113. +
  114. +static void __init
  115. +set_percpu_fixmap_pages(int idx, void *ptr, int pages, pgprot_t prot)
  116. +{
  117. + for ( ; pages; pages--, idx--, ptr += PAGE_SIZE)
  118. + __set_fixmap(idx, per_cpu_ptr_to_phys(ptr), prot);
  119. +}
  120. +
  121. /* Setup the fixmap mappings only once per-processor */
  122. -static inline void setup_cpu_entry_area(int cpu)
  123. +static void __init setup_cpu_entry_area(int cpu)
  124. {
  125. #ifdef CONFIG_X86_64
  126. extern char _entry_trampoline[];
  127. @@ -532,15 +544,31 @@ static inline void setup_cpu_entry_area(int cpu)
  128. PAGE_KERNEL);
  129. #ifdef CONFIG_X86_32
  130. - this_cpu_write(cpu_entry_area, get_cpu_entry_area(cpu));
  131. + per_cpu(cpu_entry_area, cpu) = get_cpu_entry_area(cpu);
  132. #endif
  133. #ifdef CONFIG_X86_64
  134. + BUILD_BUG_ON(sizeof(exception_stacks) % PAGE_SIZE != 0);
  135. + BUILD_BUG_ON(sizeof(exception_stacks) !=
  136. + sizeof(((struct cpu_entry_area *)0)->exception_stacks));
  137. + set_percpu_fixmap_pages(get_cpu_entry_area_index(cpu, exception_stacks),
  138. + &per_cpu(exception_stacks, cpu),
  139. + sizeof(exception_stacks) / PAGE_SIZE,
  140. + PAGE_KERNEL);
  141. +
  142. __set_fixmap(get_cpu_entry_area_index(cpu, entry_trampoline),
  143. __pa_symbol(_entry_trampoline), PAGE_KERNEL_RX);
  144. #endif
  145. }
  146. +void __init setup_cpu_entry_areas(void)
  147. +{
  148. + unsigned int cpu;
  149. +
  150. + for_each_possible_cpu(cpu)
  151. + setup_cpu_entry_area(cpu);
  152. +}
  153. +
  154. /* Load the original GDT from the per-cpu structure */
  155. void load_direct_gdt(int cpu)
  156. {
  157. @@ -1386,20 +1414,6 @@ DEFINE_PER_CPU(unsigned int, irq_count) __visible = -1;
  158. DEFINE_PER_CPU(int, __preempt_count) = INIT_PREEMPT_COUNT;
  159. EXPORT_PER_CPU_SYMBOL(__preempt_count);
  160. -/*
  161. - * Special IST stacks which the CPU switches to when it calls
  162. - * an IST-marked descriptor entry. Up to 7 stacks (hardware
  163. - * limit), all of them are 4K, except the debug stack which
  164. - * is 8K.
  165. - */
  166. -static const unsigned int exception_stack_sizes[N_EXCEPTION_STACKS] = {
  167. - [0 ... N_EXCEPTION_STACKS - 1] = EXCEPTION_STKSZ,
  168. - [DEBUG_STACK - 1] = DEBUG_STKSZ
  169. -};
  170. -
  171. -static DEFINE_PER_CPU_PAGE_ALIGNED(char, exception_stacks
  172. - [(N_EXCEPTION_STACKS - 1) * EXCEPTION_STKSZ + DEBUG_STKSZ]);
  173. -
  174. /* May not be marked __init: used by software suspend */
  175. void syscall_init(void)
  176. {
  177. @@ -1608,7 +1622,7 @@ void cpu_init(void)
  178. * set up and load the per-CPU TSS
  179. */
  180. if (!oist->ist[0]) {
  181. - char *estacks = per_cpu(exception_stacks, cpu);
  182. + char *estacks = get_cpu_entry_area(cpu)->exception_stacks;
  183. for (v = 0; v < N_EXCEPTION_STACKS; v++) {
  184. estacks += exception_stack_sizes[v];
  185. @@ -1633,8 +1647,6 @@ void cpu_init(void)
  186. BUG_ON(me->mm);
  187. enter_lazy_tlb(&init_mm, me);
  188. - setup_cpu_entry_area(cpu);
  189. -
  190. /*
  191. * Initialize the TSS. sp0 points to the entry trampoline stack
  192. * regardless of what task is running.
  193. @@ -1693,8 +1705,6 @@ void cpu_init(void)
  194. BUG_ON(curr->mm);
  195. enter_lazy_tlb(&init_mm, curr);
  196. - setup_cpu_entry_area(cpu);
  197. -
  198. /*
  199. * Initialize the TSS. Don't bother initializing sp0, as the initial
  200. * task never enters user mode.
  201. diff --git a/arch/x86/kernel/traps.c b/arch/x86/kernel/traps.c
  202. index d9debdafe7a6..fd4d47e8672e 100644
  203. --- a/arch/x86/kernel/traps.c
  204. +++ b/arch/x86/kernel/traps.c
  205. @@ -992,6 +992,9 @@ void __init trap_init(void)
  206. {
  207. int i;
  208. + /* Init cpu_entry_area before IST entries are set up */
  209. + setup_cpu_entry_areas();
  210. +
  211. #ifdef CONFIG_EISA
  212. void __iomem *p = early_ioremap(0x0FFFD9, 4);
  213. --
  214. 2.14.2