Release 4.7 arch/x86/kernel/process_32.c
/*
* Copyright (C) 1995 Linus Torvalds
*
* Pentium III FXSR, SSE support
* Gareth Hughes <gareth@valinux.com>, May 2000
*/
/*
* This file handles the architecture-dependent parts of process handling..
*/
#include <linux/cpu.h>
#include <linux/errno.h>
#include <linux/sched.h>
#include <linux/fs.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/elfcore.h>
#include <linux/smp.h>
#include <linux/stddef.h>
#include <linux/slab.h>
#include <linux/vmalloc.h>
#include <linux/user.h>
#include <linux/interrupt.h>
#include <linux/delay.h>
#include <linux/reboot.h>
#include <linux/mc146818rtc.h>
#include <linux/module.h>
#include <linux/kallsyms.h>
#include <linux/ptrace.h>
#include <linux/personality.h>
#include <linux/percpu.h>
#include <linux/prctl.h>
#include <linux/ftrace.h>
#include <linux/uaccess.h>
#include <linux/io.h>
#include <linux/kdebug.h>
#include <asm/pgtable.h>
#include <asm/ldt.h>
#include <asm/processor.h>
#include <asm/fpu/internal.h>
#include <asm/desc.h>
#ifdef CONFIG_MATH_EMULATION
#include <asm/math_emu.h>
#endif
#include <linux/err.h>
#include <asm/tlbflush.h>
#include <asm/cpu.h>
#include <asm/idle.h>
#include <asm/syscalls.h>
#include <asm/debugreg.h>
#include <asm/switch_to.h>
#include <asm/vm86.h>
asmlinkage void ret_from_fork(void) __asm__("ret_from_fork");
asmlinkage void ret_from_kernel_thread(void) __asm__("ret_from_kernel_thread");
/*
* Return saved PC of a blocked thread.
*/
unsigned long thread_saved_pc(struct task_struct *tsk)
{
return ((unsigned long *)tsk->thread.sp)[3];
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| jens axboe | jens axboe | 27 | 96.43% | 1 | 50.00% |
| h. peter anvin | h. peter anvin | 1 | 3.57% | 1 | 50.00% |
| Total | 28 | 100.00% | 2 | 100.00% |
void __show_regs(struct pt_regs *regs, int all)
{
unsigned long cr0 = 0L, cr2 = 0L, cr3 = 0L, cr4 = 0L;
unsigned long d0, d1, d2, d3, d6, d7;
unsigned long sp;
unsigned short ss, gs;
if (user_mode(regs)) {
sp = regs->sp;
ss = regs->ss & 0xffff;
gs = get_user_gs(regs);
} else {
sp = kernel_stack_pointer(regs);
savesegment(ss, ss);
savesegment(gs, gs);
}
printk(KERN_DEFAULT "EIP: %04x:[<%08lx>] EFLAGS: %08lx CPU: %d\n",
(u16)regs->cs, regs->ip, regs->flags,
smp_processor_id());
print_symbol("EIP is at %s\n", regs->ip);
printk(KERN_DEFAULT "EAX: %08lx EBX: %08lx ECX: %08lx EDX: %08lx\n",
regs->ax, regs->bx, regs->cx, regs->dx);
printk(KERN_DEFAULT "ESI: %08lx EDI: %08lx EBP: %08lx ESP: %08lx\n",
regs->si, regs->di, regs->bp, sp);
printk(KERN_DEFAULT " DS: %04x ES: %04x FS: %04x GS: %04x SS: %04x\n",
(u16)regs->ds, (u16)regs->es, (u16)regs->fs, gs, ss);
if (!all)
return;
cr0 = read_cr0();
cr2 = read_cr2();
cr3 = read_cr3();
cr4 = __read_cr4_safe();
printk(KERN_DEFAULT "CR0: %08lx CR2: %08lx CR3: %08lx CR4: %08lx\n",
cr0, cr2, cr3, cr4);
get_debugreg(d0, 0);
get_debugreg(d1, 1);
get_debugreg(d2, 2);
get_debugreg(d3, 3);
get_debugreg(d6, 6);
get_debugreg(d7, 7);
/* Only print out debug registers if they are in their non-default state. */
if ((d0 == 0) && (d1 == 0) && (d2 == 0) && (d3 == 0) &&
(d6 == DR6_RESERVED) && (d7 == 0x400))
return;
printk(KERN_DEFAULT "DR0: %08lx DR1: %08lx DR2: %08lx DR3: %08lx\n",
d0, d1, d2, d3);
printk(KERN_DEFAULT "DR6: %08lx DR7: %08lx\n",
d6, d7);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| pre-git | pre-git | 84 | 23.27% | 3 | 15.79% |
| pavel emelianov | pavel emelianov | 81 | 22.44% | 1 | 5.26% |
| alan stern | alan stern | 65 | 18.01% | 1 | 5.26% |
| dave jones | dave jones | 54 | 14.96% | 1 | 5.26% |
| h. peter anvin | h. peter anvin | 21 | 5.82% | 2 | 10.53% |
| zachary amsden | zachary amsden | 19 | 5.26% | 1 | 5.26% |
| harvey harrison | harvey harrison | 12 | 3.32% | 1 | 5.26% |
| pekka j enberg | pekka j enberg | 7 | 1.94% | 2 | 10.53% |
| tejun heo | tejun heo | 4 | 1.11% | 1 | 5.26% |
| ingo molnar | ingo molnar | 4 | 1.11% | 1 | 5.26% |
| linus torvalds | linus torvalds | 4 | 1.11% | 1 | 5.26% |
| jeremy fitzhardinge | jeremy fitzhardinge | 2 | 0.55% | 1 | 5.26% |
| andy lutomirski | andy lutomirski | 2 | 0.55% | 2 | 10.53% |
| vincent hanquez | vincent hanquez | 2 | 0.55% | 1 | 5.26% |
| Total | 361 | 100.00% | 19 | 100.00% |
void release_thread(struct task_struct *dead_task)
{
BUG_ON(dead_task->mm);
release_vm86_irqs(dead_task);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| pre-git | pre-git | 13 | 59.09% | 4 | 50.00% |
| william lee irwin iii | william lee irwin iii | 4 | 18.18% | 1 | 12.50% |
| dave jones | dave jones | 3 | 13.64% | 1 | 12.50% |
| ingo molnar | ingo molnar | 1 | 4.55% | 1 | 12.50% |
| zachary amsden | zachary amsden | 1 | 4.55% | 1 | 12.50% |
| Total | 22 | 100.00% | 8 | 100.00% |
int copy_thread_tls(unsigned long clone_flags, unsigned long sp,
unsigned long arg, struct task_struct *p, unsigned long tls)
{
struct pt_regs *childregs = task_pt_regs(p);
struct task_struct *tsk;
int err;
p->thread.sp = (unsigned long) childregs;
p->thread.sp0 = (unsigned long) (childregs+1);
memset(p->thread.ptrace_bps, 0, sizeof(p->thread.ptrace_bps));
if (unlikely(p->flags & PF_KTHREAD)) {
/* kernel thread */
memset(childregs, 0, sizeof(struct pt_regs));
p->thread.ip = (unsigned long) ret_from_kernel_thread;
task_user_gs(p) = __KERNEL_STACK_CANARY;
childregs->ds = __USER_DS;
childregs->es = __USER_DS;
childregs->fs = __KERNEL_PERCPU;
childregs->bx = sp; /* function */
childregs->bp = arg;
childregs->orig_ax = -1;
childregs->cs = __KERNEL_CS | get_kernel_rpl();
childregs->flags = X86_EFLAGS_IF | X86_EFLAGS_FIXED;
p->thread.io_bitmap_ptr = NULL;
return 0;
}
*childregs = *current_pt_regs();
childregs->ax = 0;
if (sp)
childregs->sp = sp;
p->thread.ip = (unsigned long) ret_from_fork;
task_user_gs(p) = get_user_gs(current_pt_regs());
p->thread.io_bitmap_ptr = NULL;
tsk = current;
err = -ENOMEM;
if (unlikely(test_tsk_thread_flag(tsk, TIF_IO_BITMAP))) {
p->thread.io_bitmap_ptr = kmemdup(tsk->thread.io_bitmap_ptr,
IO_BITMAP_BYTES, GFP_KERNEL);
if (!p->thread.io_bitmap_ptr) {
p->thread.io_bitmap_max = 0;
return -ENOMEM;
}
set_tsk_thread_flag(p, TIF_IO_BITMAP);
}
err = 0;
/*
* Set a new TLS for the child thread?
*/
if (clone_flags & CLONE_SETTLS)
err = do_set_thread_area(p, -1,
(struct user_desc __user *)tls, 0);
if (err && p->thread.io_bitmap_ptr) {
kfree(p->thread.io_bitmap_ptr);
p->thread.io_bitmap_max = 0;
}
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| al viro | al viro | 147 | 37.40% | 2 | 7.14% |
| ingo molnar | ingo molnar | 65 | 16.54% | 3 | 10.71% |
| pre-git | pre-git | 57 | 14.50% | 5 | 17.86% |
| andrew morton | andrew morton | 28 | 7.12% | 3 | 10.71% |
| oleg nesterov | oleg nesterov | 20 | 5.09% | 1 | 3.57% |
| roland mcgrath | roland mcgrath | 12 | 3.05% | 1 | 3.57% |
| stephane eranian | stephane eranian | 12 | 3.05% | 1 | 3.57% |
| k.prasad | k.prasad | 11 | 2.80% | 1 | 3.57% |
| benjamin lahaise | benjamin lahaise | 9 | 2.29% | 1 | 3.57% |
| alexey dobriyan | alexey dobriyan | 7 | 1.78% | 1 | 3.57% |
| josh triplett | josh triplett | 6 | 1.53% | 1 | 3.57% |
| tejun heo | tejun heo | 5 | 1.27% | 1 | 3.57% |
| h. peter anvin | h. peter anvin | 4 | 1.02% | 3 | 10.71% |
| vineet gupta | vineet gupta | 4 | 1.02% | 1 | 3.57% |
| albert cahalan | albert cahalan | 4 | 1.02% | 1 | 3.57% |
| alexander nyberg | alexander nyberg | 1 | 0.25% | 1 | 3.57% |
| linus torvalds | linus torvalds | 1 | 0.25% | 1 | 3.57% |
| Total | 393 | 100.00% | 28 | 100.00% |
void
start_thread(struct pt_regs *regs, unsigned long new_ip, unsigned long new_sp)
{
set_user_gs(regs, 0);
regs->fs = 0;
regs->ds = __USER_DS;
regs->es = __USER_DS;
regs->ss = __USER_DS;
regs->cs = __USER_CS;
regs->ip = new_ip;
regs->sp = new_sp;
regs->flags = X86_EFLAGS_IF;
force_iret();
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| ingo molnar | ingo molnar | 60 | 78.95% | 1 | 20.00% |
| al viro | al viro | 7 | 9.21% | 2 | 40.00% |
| tejun heo | tejun heo | 7 | 9.21% | 1 | 20.00% |
| brian gerst | brian gerst | 2 | 2.63% | 1 | 20.00% |
| Total | 76 | 100.00% | 5 | 100.00% |
EXPORT_SYMBOL_GPL(start_thread);
/*
* switch_to(x,y) should switch tasks from x to y.
*
* We fsave/fwait so that an exception goes off at the right time
* (as a call from the fsave or fwait in effect) rather than to
* the wrong process. Lazy FP saving no longer makes any sense
* with modern CPU's, and this simplifies a lot of things (SMP
* and UP become the same).
*
* NOTE! We used to use the x86 hardware context switching. The
* reason for not using it any more becomes apparent when you
* try to recover gracefully from saved state that is no longer
* valid (stale segment register values in particular). With the
* hardware task-switch, there is no way to fix up bad state in
* a reasonable manner.
*
* The fact that Intel documents the hardware task-switching to
* be slow is a fairly red herring - this code is not noticeably
* faster. However, there _is_ some room for improvement here,
* so the performance issues may eventually be a valid point.
* More important, however, is the fact that this allows us much
* more flexibility.
*
* The return value (in %ax) will be the "prev" task after
* the task-switch, and shows up in ret_from_fork in entry.S,
* for example.
*/
__visible __notrace_funcgraph struct task_struct *
__switch_to(struct task_struct *prev_p, struct task_struct *next_p)
{
struct thread_struct *prev = &prev_p->thread,
*next = &next_p->thread;
struct fpu *prev_fpu = &prev->fpu;
struct fpu *next_fpu = &next->fpu;
int cpu = smp_processor_id();
struct tss_struct *tss = &per_cpu(cpu_tss, cpu);
fpu_switch_t fpu_switch;
/* never put a printk in __switch_to... printk() calls wake_up*() indirectly */
fpu_switch = switch_fpu_prepare(prev_fpu, next_fpu, cpu);
/*
* Save away %gs. No need to save %fs, as it was saved on the
* stack on entry. No need to save %es and %ds, as those are
* always kernel segments while inside the kernel. Doing this
* before setting the new TLS descriptors avoids the situation
* where we temporarily have non-reloadable segments in %fs
* and %gs. This could be an issue if the NMI handler ever
* used %fs or %gs (it does not today), or if the kernel is
* running inside of a hypervisor layer.
*/
lazy_save_gs(prev->gs);
/*
* Load the per-thread Thread-Local Storage descriptor.
*/
load_TLS(next, cpu);
/*
* Restore IOPL if needed. In normal use, the flags restore
* in the switch assembly will handle this. But if the kernel
* is running virtualized at a non-zero CPL, the popf will
* not restore flags, so it must be done in a separate step.
*/
if (get_kernel_rpl() && unlikely(prev->iopl != next->iopl))
set_iopl_mask(next->iopl);
/*
* Now maybe handle debug registers and/or IO bitmaps
*/
if (unlikely(task_thread_info(prev_p)->flags & _TIF_WORK_CTXSW_PREV ||
task_thread_info(next_p)->flags & _TIF_WORK_CTXSW_NEXT))
__switch_to_xtra(prev_p, next_p, tss);
/*
* Leave lazy mode, flushing any hypercalls made here.
* This must be done before restoring TLS segments so
* the GDT and LDT are properly updated, and must be
* done before fpu__restore(), so the TS bit is up
* to date.
*/
arch_end_context_switch(next_p);
/*
* Reload esp0 and cpu_current_top_of_stack. This changes
* current_thread_info().
*/
load_sp0(tss, next);
this_cpu_write(cpu_current_top_of_stack,
(unsigned long)task_stack_page(next_p) +
THREAD_SIZE);
/*
* Restore %gs if needed (which is common)
*/
if (prev->gs | next->gs)
lazy_load_gs(next->gs);
switch_fpu_finish(next_fpu, fpu_switch);
this_cpu_write(current_task, next_p);
return prev_p;
}
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp |
| pre-git | pre-git | 238 | 18.96% | 26 | 21.14% |
| ingo molnar | ingo molnar | 171 | 13.63% | 12 | 9.76% |
| al viro | al viro | 165 | 13.15% | 4 | 3.25% |
| pavel emelianov | pavel emelianov | 81 | 6.45% | 1 | 0.81% |
| zachary amsden | zachary amsden | 71 | 5.66% | 5 | 4.07% |
| alan stern | alan stern | 65 | 5.18% | 1 | 0.81% |
| dave jones | dave jones | 62 | 4.94% | 4 | 3.25% |
| jeremy fitzhardinge | jeremy fitzhardinge | 35 | 2.79% | 6 | 4.88% |
| jens axboe | jens axboe | 28 | 2.23% | 1 | 0.81% |
| andi kleen | andi kleen | 28 | 2.23% | 2 | 1.63% |
| andrew morton | andrew morton | 28 | 2.23% | 3 | 2.44% |
| h. peter anvin | h. peter anvin | 26 | 2.07% | 4 | 3.25% |
| andy lutomirski | andy lutomirski | 26 | 2.07% | 5 | 4.07% |
| oleg nesterov | oleg nesterov | 25 | 1.99% | 2 | 1.63% |
| linus torvalds | linus torvalds | 25 | 1.99% | 8 | 6.50% |
| tejun heo | tejun heo | 18 | 1.43% | 2 | 1.63% |
| stephane eranian | stephane eranian | 18 | 1.43% | 1 | 0.81% |
| andrea arcangeli | andrea arcangeli | 17 | 1.35% | 2 | 1.63% |
| k.prasad | k.prasad | 12 | 0.96% | 1 | 0.81% |
| harvey harrison | harvey harrison | 12 | 0.96% | 1 | 0.81% |
| roland mcgrath | roland mcgrath | 12 | 0.96% | 1 | 0.81% |
| zwane mwaikambo | zwane mwaikambo | 9 | 0.72% | 1 | 0.81% |
| jaswinder singh rajput | jaswinder singh rajput | 9 | 0.72% | 3 | 2.44% |
| benjamin lahaise | benjamin lahaise | 9 | 0.72% | 1 | 0.81% |
| alexey dobriyan | alexey dobriyan | 7 | 0.56% | 1 | 0.81% |
| pekka j enberg | pekka j enberg | 7 | 0.56% | 2 | 1.63% |
| josh triplett | josh triplett | 6 | 0.48% | 1 | 0.81% |
| brian gerst | brian gerst | 5 | 0.40% | 2 | 1.63% |
| albert cahalan | albert cahalan | 4 | 0.32% | 1 | 0.81% |
| vineet gupta | vineet gupta | 4 | 0.32% | 1 | 0.81% |
| william lee irwin iii | william lee irwin iii | 4 | 0.32% | 1 | 0.81% |
| vincent hanquez | vincent hanquez | 3 | 0.24% | 2 | 1.63% |
| rusty russell | rusty russell | 3 | 0.24% | 1 | 0.81% |
| erik bosman | erik bosman | 3 | 0.24% | 1 | 0.81% |
| daniel jacobowitz | daniel jacobowitz | 3 | 0.24% | 1 | 0.81% |
| frederic weisbecker | frederic weisbecker | 3 | 0.24% | 1 | 0.81% |
| don zickus | don zickus | 2 | 0.16% | 1 | 0.81% |
| thomas gleixner | thomas gleixner | 2 | 0.16% | 1 | 0.81% |
| david howells | david howells | 2 | 0.16% | 2 | 1.63% |
| alexander nyberg | alexander nyberg | 1 | 0.08% | 1 | 0.81% |
| denys vlasenko | denys vlasenko | 1 | 0.08% | 1 | 0.81% |
| kamalesh babulal | kamalesh babulal | 1 | 0.08% | 1 | 0.81% |
| marc dionne | marc dionne | 1 | 0.08% | 1 | 0.81% |
| shai fultheim | shai fultheim | 1 | 0.08% | 1 | 0.81% |
| chuck ebbert | chuck ebbert | 1 | 0.08% | 1 | 0.81% |
| alex shi | alex shi | 1 | 0.08% | 1 | 0.81% |
| Total | 1255 | 100.00% | 123 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.