2  * Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
 
   3  * Licensed under the GPL
 
   6 #include "linux/sched.h"
 
   7 #include "linux/slab.h"
 
   8 #include "linux/ptrace.h"
 
   9 #include "linux/proc_fs.h"
 
  10 #include "linux/file.h"
 
  11 #include "linux/errno.h"
 
  12 #include "linux/init.h"
 
  13 #include "asm/uaccess.h"
 
  14 #include "asm/atomic.h"
 
  15 #include "kern_util.h"
 
  16 #include "time_user.h"
 
  17 #include "signal_user.h"
 
  20 #include "user_util.h"
 
  25 #include "registers.h"
 
  27 void switch_to_skas(void *prev, void *next)
 
  29         struct task_struct *from, *to;
 
  34         /* XXX need to check runqueues[cpu].idle */
 
  38         switch_threads(&from->thread.mode.skas.switch_buf, 
 
  39                        to->thread.mode.skas.switch_buf);
 
  45 extern void schedule_tail(struct task_struct *prev);
 
  47 void new_thread_handler(int sig)
 
  52         fn = current->thread.request.u.thread.proc;
 
  53         arg = current->thread.request.u.thread.arg;
 
  54         change_sig(SIGUSR1, 1);
 
  55         thread_wait(¤t->thread.mode.skas.switch_buf, 
 
  56                     current->thread.mode.skas.fork_buf);
 
  58         if(current->thread.prev_sched != NULL)
 
  59                 schedule_tail(current->thread.prev_sched);
 
  60         current->thread.prev_sched = NULL;
 
  62         /* The return value is 1 if the kernel thread execs a process,
 
  65         n = run_kernel_thread(fn, arg, ¤t->thread.exec_buf);
 
  67                 /* Handle any immediate reschedules or signals */
 
  69                 userspace(¤t->thread.regs.regs);
 
  74 void new_thread_proc(void *stack, void (*handler)(int sig))
 
  76         init_new_thread_stack(stack, handler);
 
  77         os_usr1_process(os_getpid());
 
  80 void release_thread_skas(struct task_struct *task)
 
  84 void fork_handler(int sig)
 
  86         change_sig(SIGUSR1, 1);
 
  87         thread_wait(¤t->thread.mode.skas.switch_buf, 
 
  88                     current->thread.mode.skas.fork_buf);
 
  91         if(current->thread.prev_sched == NULL)
 
  94         schedule_tail(current->thread.prev_sched);
 
  95         current->thread.prev_sched = NULL;
 
  97         /* Handle any immediate reschedules or signals */
 
  99         userspace(¤t->thread.regs.regs);
 
 102 int copy_thread_skas(int nr, unsigned long clone_flags, unsigned long sp,
 
 103                      unsigned long stack_top, struct task_struct * p, 
 
 104                      struct pt_regs *regs)
 
 106         void (*handler)(int);
 
 108         if(current->thread.forking){
 
 109                 memcpy(&p->thread.regs.regs.skas, ®s->regs.skas,
 
 110                        sizeof(p->thread.regs.regs.skas));
 
 111                 REGS_SET_SYSCALL_RETURN(p->thread.regs.regs.skas.regs, 0);
 
 112                 if(sp != 0) REGS_SP(p->thread.regs.regs.skas.regs) = sp;
 
 114                 handler = fork_handler;
 
 117                 init_thread_registers(&p->thread.regs.regs);
 
 118                 p->thread.request.u.thread = current->thread.request.u.thread;
 
 119                 handler = new_thread_handler;
 
 122         new_thread(p->thread_info, &p->thread.mode.skas.switch_buf,
 
 123                    &p->thread.mode.skas.fork_buf, handler);
 
 127 extern void map_stub_pages(int fd, unsigned long code,
 
 128                            unsigned long data, unsigned long stack);
 
 129 int new_mm(int from, unsigned long stack)
 
 131         struct proc_mm_op copy;
 
 134         fd = os_open_file("/proc/mm", of_cloexec(of_write(OPENFLAGS())), 0);
 
 139                 copy = ((struct proc_mm_op) { .op       = MM_COPY_SEGMENTS,
 
 141                                               { .copy_segments  = from } } );
 
 142                 n = os_write_file(fd, ©, sizeof(copy));
 
 143                 if(n != sizeof(copy))
 
 144                         printk("new_mm : /proc/mm copy_segments failed, "
 
 148         if(!ptrace_faultinfo)
 
 149                 map_stub_pages(fd, CONFIG_STUB_CODE, CONFIG_STUB_DATA, stack);
 
 154 void init_idle_skas(void)
 
 156         cpu_tasks[current_thread->cpu].pid = os_getpid();
 
 160 extern void start_kernel(void);
 
 162 static int start_kernel_proc(void *unused)
 
 169         cpu_tasks[0].pid = pid;
 
 170         cpu_tasks[0].task = current;
 
 172         cpu_online_map = cpumask_of_cpu(0);
 
 178 extern int userspace_pid[];
 
 180 int start_uml_skas(void)
 
 183                 userspace_pid[0] = start_userspace(0);
 
 185         init_new_thread_signals(1);
 
 187         init_task.thread.request.u.thread.proc = start_kernel_proc;
 
 188         init_task.thread.request.u.thread.arg = NULL;
 
 189         return(start_idle_thread(init_task.thread_info,
 
 190                                  &init_task.thread.mode.skas.switch_buf,
 
 191                                  &init_task.thread.mode.skas.fork_buf));
 
 194 int external_pid_skas(struct task_struct *task)
 
 196 #warning Need to look up userspace_pid by cpu
 
 197         return(userspace_pid[0]);
 
 200 int thread_pid_skas(struct task_struct *task)
 
 202 #warning Need to look up userspace_pid by cpu
 
 203         return(userspace_pid[0]);
 
 206 void kill_off_processes_skas(void)
 
 209 #warning need to loop over userspace_pids in kill_off_processes_skas
 
 210                 os_kill_ptraced_process(userspace_pid[0], 1);
 
 212                 struct task_struct *p;
 
 220                         pid = p->mm->context.skas.id.u.pid;
 
 221                         os_kill_ptraced_process(pid, 1);
 
 226 unsigned long current_stub_stack(void)
 
 228         if(current->mm == NULL)
 
 231         return(current->mm->context.skas.id.stack);