1
2
3
4
5
6
7
8#include <linux/sched.h>
9#include <linux/uaccess.h>
10#include <asm/prctl.h>
11#include <os.h>
12
13long arch_prctl(struct task_struct *task, int code, unsigned long __user *addr)
14{
15 unsigned long *ptr = addr, tmp;
16 long ret;
17 int pid = task->mm->context.id.u.pid;
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32 switch (code) {
33 case ARCH_SET_FS:
34 case ARCH_SET_GS:
35 ret = restore_registers(pid, ¤t->thread.regs.regs);
36 if (ret)
37 return ret;
38 break;
39 case ARCH_GET_FS:
40 case ARCH_GET_GS:
41
42
43
44
45
46
47
48
49 ptr = &tmp;
50 }
51
52 ret = os_arch_prctl(pid, code, ptr);
53 if (ret)
54 return ret;
55
56 switch (code) {
57 case ARCH_SET_FS:
58 current->thread.arch.fs = (unsigned long) ptr;
59 ret = save_registers(pid, ¤t->thread.regs.regs);
60 break;
61 case ARCH_SET_GS:
62 ret = save_registers(pid, ¤t->thread.regs.regs);
63 break;
64 case ARCH_GET_FS:
65 ret = put_user(tmp, addr);
66 break;
67 case ARCH_GET_GS:
68 ret = put_user(tmp, addr);
69 break;
70 }
71
72 return ret;
73}
74
75long sys_arch_prctl(int code, unsigned long addr)
76{
77 return arch_prctl(current, code, (unsigned long __user *) addr);
78}
79
80void arch_switch_to(struct task_struct *to)
81{
82 if ((to->thread.arch.fs == 0) || (to->mm == NULL))
83 return;
84
85 arch_prctl(to, ARCH_SET_FS, (void __user *) to->thread.arch.fs);
86}
87