ftp://ftp.kernel.org/pub/linux/kernel/v2.6/linux-2.6.6.tar.bz2
[linux-2.6.git] / arch / um / kernel / tt / exec_kern.c
1 /* 
2  * Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
3  * Licensed under the GPL
4  */
5
6 #include "linux/kernel.h"
7 #include "linux/mm.h"
8 #include "asm/signal.h"
9 #include "asm/ptrace.h"
10 #include "asm/uaccess.h"
11 #include "asm/pgalloc.h"
12 #include "asm/tlbflush.h"
13 #include "user_util.h"
14 #include "kern_util.h"
15 #include "irq_user.h"
16 #include "time_user.h"
17 #include "mem_user.h"
18 #include "os.h"
19 #include "tlb.h"
20
21 static int exec_tramp(void *sig_stack)
22 {
23         init_new_thread_stack(sig_stack, NULL);
24         init_new_thread_signals(1);
25         os_stop_process(os_getpid());
26         return(0);
27 }
28
29 void flush_thread_tt(void)
30 {
31         unsigned long stack;
32         int new_pid;
33
34         stack = alloc_stack(0, 0);
35         if(stack == 0){
36                 printk(KERN_ERR 
37                        "flush_thread : failed to allocate temporary stack\n");
38                 do_exit(SIGKILL);
39         }
40                 
41         new_pid = start_fork_tramp((void *) current->thread.kernel_stack,
42                                    stack, 0, exec_tramp);
43         if(new_pid < 0){
44                 printk(KERN_ERR 
45                        "flush_thread : new thread failed, errno = %d\n",
46                        -new_pid);
47                 do_exit(SIGKILL);
48         }
49
50         if(current->thread_info->cpu == 0)
51                 forward_interrupts(new_pid);
52         current->thread.request.op = OP_EXEC;
53         current->thread.request.u.exec.pid = new_pid;
54         unprotect_stack((unsigned long) current->thread_info);
55         os_usr1_process(os_getpid());
56
57         enable_timer();
58         free_page(stack);
59         protect_memory(uml_reserved, high_physmem - uml_reserved, 1, 1, 0, 1);
60         task_protections((unsigned long) current->thread_info);
61         force_flush_all();
62         unblock_signals();
63 }
64
65 void start_thread_tt(struct pt_regs *regs, unsigned long eip, 
66                      unsigned long esp)
67 {
68         set_fs(USER_DS);
69         flush_tlb_mm(current->mm);
70         PT_REGS_IP(regs) = eip;
71         PT_REGS_SP(regs) = esp;
72         PT_FIX_EXEC_STACK(esp);
73 }
74
75 /*
76  * Overrides for Emacs so that we follow Linus's tabbing style.
77  * Emacs will notice this stuff at the end of the file and automatically
78  * adjust the settings for this buffer only.  This must remain at the end
79  * of the file.
80  * ---------------------------------------------------------------------------
81  * Local variables:
82  * c-file-style: "linux"
83  * End:
84  */