doublefault_32.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/mm.h>
  3. #include <linux/sched.h>
  4. #include <linux/sched/debug.h>
  5. #include <linux/init_task.h>
  6. #include <linux/fs.h>
  7. #include <linux/uaccess.h>
  8. #include <asm/processor.h>
  9. #include <asm/desc.h>
  10. #include <asm/traps.h>
  11. #include <asm/doublefault.h>
  12. #define ptr_ok(x) ((x) > PAGE_OFFSET && (x) < PAGE_OFFSET + MAXMEM)
  13. #define TSS(x) this_cpu_read(cpu_tss_rw.x86_tss.x)
  14. static void set_df_gdt_entry(unsigned int cpu);
  15. /*
  16. * Called by double_fault with CR0.TS and EFLAGS.NT cleared. The CPU thinks
  17. * we're running the doublefault task. Cannot return.
  18. */
  19. asmlinkage noinstr void __noreturn doublefault_shim(void)
  20. {
  21. unsigned long cr2;
  22. struct pt_regs regs;
  23. BUILD_BUG_ON(sizeof(struct doublefault_stack) != PAGE_SIZE);
  24. cr2 = native_read_cr2();
  25. /* Reset back to the normal kernel task. */
  26. force_reload_TR();
  27. set_df_gdt_entry(smp_processor_id());
  28. trace_hardirqs_off();
  29. /*
  30. * Fill in pt_regs. A downside of doing this in C is that the unwinder
  31. * won't see it (no ENCODE_FRAME_POINTER), so a nested stack dump
  32. * won't successfully unwind to the source of the double fault.
  33. * The main dump from exc_double_fault() is fine, though, since it
  34. * uses these regs directly.
  35. *
  36. * If anyone ever cares, this could be moved to asm.
  37. */
  38. regs.ss = TSS(ss);
  39. regs.__ssh = 0;
  40. regs.sp = TSS(sp);
  41. regs.flags = TSS(flags);
  42. regs.cs = TSS(cs);
  43. /* We won't go through the entry asm, so we can leave __csh as 0. */
  44. regs.__csh = 0;
  45. regs.ip = TSS(ip);
  46. regs.orig_ax = 0;
  47. regs.gs = TSS(gs);
  48. regs.__gsh = 0;
  49. regs.fs = TSS(fs);
  50. regs.__fsh = 0;
  51. regs.es = TSS(es);
  52. regs.__esh = 0;
  53. regs.ds = TSS(ds);
  54. regs.__dsh = 0;
  55. regs.ax = TSS(ax);
  56. regs.bp = TSS(bp);
  57. regs.di = TSS(di);
  58. regs.si = TSS(si);
  59. regs.dx = TSS(dx);
  60. regs.cx = TSS(cx);
  61. regs.bx = TSS(bx);
  62. exc_double_fault(&regs, 0, cr2);
  63. /*
  64. * x86_32 does not save the original CR3 anywhere on a task switch.
  65. * This means that, even if we wanted to return, we would need to find
  66. * some way to reconstruct CR3. We could make a credible guess based
  67. * on cpu_tlbstate, but that would be racy and would not account for
  68. * PTI.
  69. */
  70. panic("cannot return from double fault\n");
  71. }
  72. DEFINE_PER_CPU_PAGE_ALIGNED(struct doublefault_stack, doublefault_stack) = {
  73. .tss = {
  74. /*
  75. * No sp0 or ss0 -- we never run CPL != 0 with this TSS
  76. * active. sp is filled in later.
  77. */
  78. .ldt = 0,
  79. .io_bitmap_base = IO_BITMAP_OFFSET_INVALID,
  80. .ip = (unsigned long) asm_exc_double_fault,
  81. .flags = X86_EFLAGS_FIXED,
  82. .es = __USER_DS,
  83. .cs = __KERNEL_CS,
  84. .ss = __KERNEL_DS,
  85. .ds = __USER_DS,
  86. .fs = __KERNEL_PERCPU,
  87. .gs = 0,
  88. .__cr3 = __pa_nodebug(swapper_pg_dir),
  89. },
  90. };
  91. static void set_df_gdt_entry(unsigned int cpu)
  92. {
  93. /* Set up doublefault TSS pointer in the GDT */
  94. __set_tss_desc(cpu, GDT_ENTRY_DOUBLEFAULT_TSS,
  95. &get_cpu_entry_area(cpu)->doublefault_stack.tss);
  96. }
  97. void doublefault_init_cpu_tss(void)
  98. {
  99. unsigned int cpu = smp_processor_id();
  100. struct cpu_entry_area *cea = get_cpu_entry_area(cpu);
  101. /*
  102. * The linker isn't smart enough to initialize percpu variables that
  103. * point to other places in percpu space.
  104. */
  105. this_cpu_write(doublefault_stack.tss.sp,
  106. (unsigned long)&cea->doublefault_stack.stack +
  107. sizeof(doublefault_stack.stack));
  108. set_df_gdt_entry(cpu);
  109. }