mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-02-10 23:25:58 +07:00
![Linus Torvalds](/assets/img/avatar_default.png)
Pull x86 iopl updates from Ingo Molnar: "This implements a nice simplification of the iopl and ioperm code that Thomas Gleixner discovered: we can implement the IO privilege features of the iopl system call by using the IO permission bitmap in permissive mode, while trapping CLI/STI/POPF/PUSHF uses in user-space if they change the interrupt flag. This implements that feature, with testing facilities and related cleanups" [ "Simplification" may be an over-statement. The main goal is to avoid the cli/sti of iopl by effectively implementing the IO port access parts of iopl in terms of ioperm. This may end up not workign well in case people actually depend on cli/sti being available, or if there are mixed uses of iopl and ioperm. We will see.. - Linus ] * 'x86-iopl-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: (22 commits) x86/ioperm: Fix use of deprecated config option x86/entry/32: Clarify register saving in __switch_to_asm() selftests/x86/iopl: Extend test to cover IOPL emulation x86/ioperm: Extend IOPL config to control ioperm() as well x86/iopl: Remove legacy IOPL option x86/iopl: Restrict iopl() permission scope x86/iopl: Fixup misleading comment selftests/x86/ioperm: Extend testing so the shared bitmap is exercised x86/ioperm: Share I/O bitmap if identical x86/ioperm: Remove bitmap if all permissions dropped x86/ioperm: Move TSS bitmap update to exit to user work x86/ioperm: Add bitmap sequence number x86/ioperm: Move iobitmap data into a struct x86/tss: Move I/O bitmap data into a seperate struct x86/io: Speedup schedule out of I/O bitmap user x86/ioperm: Avoid bitmap allocation if no permissions are set x86/ioperm: Simplify first ioperm() invocation logic x86/iopl: Cleanup include maze x86/tss: Fix and move VMX BUILD_BUG_ON() x86/cpu: Unify cpu_init() ...
87 lines
2.1 KiB
C
87 lines
2.1 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <linux/mm.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/sched/debug.h>
|
|
#include <linux/init_task.h>
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/uaccess.h>
|
|
#include <asm/pgtable.h>
|
|
#include <asm/processor.h>
|
|
#include <asm/desc.h>
|
|
|
|
#ifdef CONFIG_X86_32
|
|
|
|
#define DOUBLEFAULT_STACKSIZE (1024)
|
|
static unsigned long doublefault_stack[DOUBLEFAULT_STACKSIZE];
|
|
#define STACK_START (unsigned long)(doublefault_stack+DOUBLEFAULT_STACKSIZE)
|
|
|
|
#define ptr_ok(x) ((x) > PAGE_OFFSET && (x) < PAGE_OFFSET + MAXMEM)
|
|
|
|
static void doublefault_fn(void)
|
|
{
|
|
struct desc_ptr gdt_desc = {0, 0};
|
|
unsigned long gdt, tss;
|
|
|
|
native_store_gdt(&gdt_desc);
|
|
gdt = gdt_desc.address;
|
|
|
|
printk(KERN_EMERG "PANIC: double fault, gdt at %08lx [%d bytes]\n", gdt, gdt_desc.size);
|
|
|
|
if (ptr_ok(gdt)) {
|
|
gdt += GDT_ENTRY_TSS << 3;
|
|
tss = get_desc_base((struct desc_struct *)gdt);
|
|
printk(KERN_EMERG "double fault, tss at %08lx\n", tss);
|
|
|
|
if (ptr_ok(tss)) {
|
|
struct x86_hw_tss *t = (struct x86_hw_tss *)tss;
|
|
|
|
printk(KERN_EMERG "eip = %08lx, esp = %08lx\n",
|
|
t->ip, t->sp);
|
|
|
|
printk(KERN_EMERG "eax = %08lx, ebx = %08lx, ecx = %08lx, edx = %08lx\n",
|
|
t->ax, t->bx, t->cx, t->dx);
|
|
printk(KERN_EMERG "esi = %08lx, edi = %08lx\n",
|
|
t->si, t->di);
|
|
}
|
|
}
|
|
|
|
for (;;)
|
|
cpu_relax();
|
|
}
|
|
|
|
struct x86_hw_tss doublefault_tss __cacheline_aligned = {
|
|
.sp0 = STACK_START,
|
|
.ss0 = __KERNEL_DS,
|
|
.ldt = 0,
|
|
.io_bitmap_base = IO_BITMAP_OFFSET_INVALID,
|
|
|
|
.ip = (unsigned long) doublefault_fn,
|
|
/* 0x2 bit is always set */
|
|
.flags = X86_EFLAGS_SF | 0x2,
|
|
.sp = STACK_START,
|
|
.es = __USER_DS,
|
|
.cs = __KERNEL_CS,
|
|
.ss = __KERNEL_DS,
|
|
.ds = __USER_DS,
|
|
.fs = __KERNEL_PERCPU,
|
|
#ifndef CONFIG_X86_32_LAZY_GS
|
|
.gs = __KERNEL_STACK_CANARY,
|
|
#endif
|
|
|
|
.__cr3 = __pa_nodebug(swapper_pg_dir),
|
|
};
|
|
|
|
/* dummy for do_double_fault() call */
|
|
void df_debug(struct pt_regs *regs, long error_code) {}
|
|
|
|
#else /* !CONFIG_X86_32 */
|
|
|
|
void df_debug(struct pt_regs *regs, long error_code)
|
|
{
|
|
pr_emerg("PANIC: double fault, error_code: 0x%lx\n", error_code);
|
|
show_regs(regs);
|
|
panic("Machine halted.");
|
|
}
|
|
#endif
|