mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-07 22:42:04 +00:00
x86/32: Remove lazy GS macros
GS is always a user segment now. Signed-off-by: Brian Gerst <brgerst@gmail.com> Signed-off-by: Borislav Petkov <bp@suse.de> Reviewed-by: Thomas Gleixner <tglx@linutronix.de> Acked-by: Andy Lutomirski <luto@kernel.org> Link: https://lore.kernel.org/r/20220325153953.162643-4-brgerst@gmail.com
This commit is contained in:
parent
9554e908fb
commit
3a24a60854
@ -141,7 +141,7 @@ do { \
|
|||||||
#ifdef CONFIG_X86_32
|
#ifdef CONFIG_X86_32
|
||||||
#define deactivate_mm(tsk, mm) \
|
#define deactivate_mm(tsk, mm) \
|
||||||
do { \
|
do { \
|
||||||
lazy_load_gs(0); \
|
loadsegment(gs, 0); \
|
||||||
} while (0)
|
} while (0)
|
||||||
#else
|
#else
|
||||||
#define deactivate_mm(tsk, mm) \
|
#define deactivate_mm(tsk, mm) \
|
||||||
|
@ -354,11 +354,6 @@ static inline void __loadsegment_fs(unsigned short value)
|
|||||||
* x86-32 user GS accessors. This is ugly and could do with some cleaning up.
|
* x86-32 user GS accessors. This is ugly and could do with some cleaning up.
|
||||||
*/
|
*/
|
||||||
#ifdef CONFIG_X86_32
|
#ifdef CONFIG_X86_32
|
||||||
# define get_user_gs(regs) (u16)({ unsigned long v; savesegment(gs, v); v; })
|
|
||||||
# define set_user_gs(regs, v) loadsegment(gs, (unsigned long)(v))
|
|
||||||
# define task_user_gs(tsk) ((tsk)->thread.gs)
|
|
||||||
# define lazy_save_gs(v) savesegment(gs, (v))
|
|
||||||
# define lazy_load_gs(v) loadsegment(gs, (v))
|
|
||||||
# define load_gs_index(v) loadsegment(gs, (v))
|
# define load_gs_index(v) loadsegment(gs, (v))
|
||||||
#endif /* X86_32 */
|
#endif /* X86_32 */
|
||||||
|
|
||||||
|
@ -160,6 +160,7 @@ int copy_thread(unsigned long clone_flags, unsigned long sp, unsigned long arg,
|
|||||||
savesegment(ds, p->thread.ds);
|
savesegment(ds, p->thread.ds);
|
||||||
#else
|
#else
|
||||||
p->thread.sp0 = (unsigned long) (childregs + 1);
|
p->thread.sp0 = (unsigned long) (childregs + 1);
|
||||||
|
savesegment(gs, p->thread.gs);
|
||||||
/*
|
/*
|
||||||
* Clear all status flags including IF and set fixed bit. 64bit
|
* Clear all status flags including IF and set fixed bit. 64bit
|
||||||
* does not have this initialization as the frame does not contain
|
* does not have this initialization as the frame does not contain
|
||||||
@ -191,10 +192,6 @@ int copy_thread(unsigned long clone_flags, unsigned long sp, unsigned long arg,
|
|||||||
if (sp)
|
if (sp)
|
||||||
childregs->sp = sp;
|
childregs->sp = sp;
|
||||||
|
|
||||||
#ifdef CONFIG_X86_32
|
|
||||||
task_user_gs(p) = get_user_gs(current_pt_regs());
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (unlikely(p->flags & PF_IO_WORKER)) {
|
if (unlikely(p->flags & PF_IO_WORKER)) {
|
||||||
/*
|
/*
|
||||||
* An IO thread is a user space thread, but it doesn't
|
* An IO thread is a user space thread, but it doesn't
|
||||||
|
@ -63,10 +63,7 @@ void __show_regs(struct pt_regs *regs, enum show_regs_mode mode,
|
|||||||
unsigned long d0, d1, d2, d3, d6, d7;
|
unsigned long d0, d1, d2, d3, d6, d7;
|
||||||
unsigned short gs;
|
unsigned short gs;
|
||||||
|
|
||||||
if (user_mode(regs))
|
savesegment(gs, gs);
|
||||||
gs = get_user_gs(regs);
|
|
||||||
else
|
|
||||||
savesegment(gs, gs);
|
|
||||||
|
|
||||||
show_ip(regs, log_lvl);
|
show_ip(regs, log_lvl);
|
||||||
|
|
||||||
@ -114,7 +111,7 @@ void release_thread(struct task_struct *dead_task)
|
|||||||
void
|
void
|
||||||
start_thread(struct pt_regs *regs, unsigned long new_ip, unsigned long new_sp)
|
start_thread(struct pt_regs *regs, unsigned long new_ip, unsigned long new_sp)
|
||||||
{
|
{
|
||||||
set_user_gs(regs, 0);
|
loadsegment(gs, 0);
|
||||||
regs->fs = 0;
|
regs->fs = 0;
|
||||||
regs->ds = __USER_DS;
|
regs->ds = __USER_DS;
|
||||||
regs->es = __USER_DS;
|
regs->es = __USER_DS;
|
||||||
@ -177,7 +174,7 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
|
|||||||
* used %fs or %gs (it does not today), or if the kernel is
|
* used %fs or %gs (it does not today), or if the kernel is
|
||||||
* running inside of a hypervisor layer.
|
* running inside of a hypervisor layer.
|
||||||
*/
|
*/
|
||||||
lazy_save_gs(prev->gs);
|
savesegment(gs, prev->gs);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Load the per-thread Thread-Local Storage descriptor.
|
* Load the per-thread Thread-Local Storage descriptor.
|
||||||
@ -208,7 +205,7 @@ __switch_to(struct task_struct *prev_p, struct task_struct *next_p)
|
|||||||
* Restore %gs if needed (which is common)
|
* Restore %gs if needed (which is common)
|
||||||
*/
|
*/
|
||||||
if (prev->gs | next->gs)
|
if (prev->gs | next->gs)
|
||||||
lazy_load_gs(next->gs);
|
loadsegment(gs, next->gs);
|
||||||
|
|
||||||
this_cpu_write(current_task, next_p);
|
this_cpu_write(current_task, next_p);
|
||||||
|
|
||||||
|
@ -170,9 +170,9 @@ static u16 get_segment_reg(struct task_struct *task, unsigned long offset)
|
|||||||
retval = *pt_regs_access(task_pt_regs(task), offset);
|
retval = *pt_regs_access(task_pt_regs(task), offset);
|
||||||
else {
|
else {
|
||||||
if (task == current)
|
if (task == current)
|
||||||
retval = get_user_gs(task_pt_regs(task));
|
savesegment(gs, retval);
|
||||||
else
|
else
|
||||||
retval = task_user_gs(task);
|
retval = task->thread.gs;
|
||||||
}
|
}
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
@ -210,7 +210,7 @@ static int set_segment_reg(struct task_struct *task,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case offsetof(struct user_regs_struct, gs):
|
case offsetof(struct user_regs_struct, gs):
|
||||||
task_user_gs(task) = value;
|
task->thread.gs = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -93,7 +93,7 @@ static bool restore_sigcontext(struct pt_regs *regs,
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
#ifdef CONFIG_X86_32
|
#ifdef CONFIG_X86_32
|
||||||
set_user_gs(regs, sc.gs);
|
loadsegment(gs, sc.gs);
|
||||||
regs->fs = sc.fs;
|
regs->fs = sc.fs;
|
||||||
regs->es = sc.es;
|
regs->es = sc.es;
|
||||||
regs->ds = sc.ds;
|
regs->ds = sc.ds;
|
||||||
@ -146,8 +146,10 @@ __unsafe_setup_sigcontext(struct sigcontext __user *sc, void __user *fpstate,
|
|||||||
struct pt_regs *regs, unsigned long mask)
|
struct pt_regs *regs, unsigned long mask)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_X86_32
|
#ifdef CONFIG_X86_32
|
||||||
unsafe_put_user(get_user_gs(regs),
|
unsigned int gs;
|
||||||
(unsigned int __user *)&sc->gs, Efault);
|
savesegment(gs, gs);
|
||||||
|
|
||||||
|
unsafe_put_user(gs, (unsigned int __user *)&sc->gs, Efault);
|
||||||
unsafe_put_user(regs->fs, (unsigned int __user *)&sc->fs, Efault);
|
unsafe_put_user(regs->fs, (unsigned int __user *)&sc->fs, Efault);
|
||||||
unsafe_put_user(regs->es, (unsigned int __user *)&sc->es, Efault);
|
unsafe_put_user(regs->es, (unsigned int __user *)&sc->es, Efault);
|
||||||
unsafe_put_user(regs->ds, (unsigned int __user *)&sc->ds, Efault);
|
unsafe_put_user(regs->ds, (unsigned int __user *)&sc->ds, Efault);
|
||||||
|
@ -151,7 +151,7 @@ void save_v86_state(struct kernel_vm86_regs *regs, int retval)
|
|||||||
|
|
||||||
memcpy(®s->pt, &vm86->regs32, sizeof(struct pt_regs));
|
memcpy(®s->pt, &vm86->regs32, sizeof(struct pt_regs));
|
||||||
|
|
||||||
lazy_load_gs(vm86->regs32.gs);
|
loadsegment(gs, vm86->regs32.gs);
|
||||||
|
|
||||||
regs->pt.ax = retval;
|
regs->pt.ax = retval;
|
||||||
return;
|
return;
|
||||||
@ -325,7 +325,7 @@ static long do_sys_vm86(struct vm86plus_struct __user *user_vm86, bool plus)
|
|||||||
* Save old state
|
* Save old state
|
||||||
*/
|
*/
|
||||||
vm86->saved_sp0 = tsk->thread.sp0;
|
vm86->saved_sp0 = tsk->thread.sp0;
|
||||||
lazy_save_gs(vm86->regs32.gs);
|
savesegment(gs, vm86->regs32.gs);
|
||||||
|
|
||||||
/* make room for real-mode segments */
|
/* make room for real-mode segments */
|
||||||
preempt_disable();
|
preempt_disable();
|
||||||
|
@ -342,9 +342,9 @@ static int resolve_seg_reg(struct insn *insn, struct pt_regs *regs, int regoff)
|
|||||||
*/
|
*/
|
||||||
static short get_segment_selector(struct pt_regs *regs, int seg_reg_idx)
|
static short get_segment_selector(struct pt_regs *regs, int seg_reg_idx)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_X86_64
|
|
||||||
unsigned short sel;
|
unsigned short sel;
|
||||||
|
|
||||||
|
#ifdef CONFIG_X86_64
|
||||||
switch (seg_reg_idx) {
|
switch (seg_reg_idx) {
|
||||||
case INAT_SEG_REG_IGNORE:
|
case INAT_SEG_REG_IGNORE:
|
||||||
return 0;
|
return 0;
|
||||||
@ -402,7 +402,8 @@ static short get_segment_selector(struct pt_regs *regs, int seg_reg_idx)
|
|||||||
case INAT_SEG_REG_FS:
|
case INAT_SEG_REG_FS:
|
||||||
return (unsigned short)(regs->fs & 0xffff);
|
return (unsigned short)(regs->fs & 0xffff);
|
||||||
case INAT_SEG_REG_GS:
|
case INAT_SEG_REG_GS:
|
||||||
return get_user_gs(regs);
|
savesegment(gs, sel);
|
||||||
|
return sel;
|
||||||
case INAT_SEG_REG_IGNORE:
|
case INAT_SEG_REG_IGNORE:
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -153,7 +153,7 @@ static long pm_address(u_char FPU_modrm, u_char segment,
|
|||||||
switch (segment) {
|
switch (segment) {
|
||||||
case PREFIX_GS_ - 1:
|
case PREFIX_GS_ - 1:
|
||||||
/* user gs handling can be lazy, use special accessors */
|
/* user gs handling can be lazy, use special accessors */
|
||||||
addr->selector = get_user_gs(FPU_info->regs);
|
savesegment(gs, addr->selector);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
addr->selector = PM_REG_(segment);
|
addr->selector = PM_REG_(segment);
|
||||||
|
Loading…
Reference in New Issue
Block a user