diff options
author | Jeff Dike <jdike@addtoit.com> | 2007-10-16 04:26:47 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-10-16 12:43:05 -0400 |
commit | 42daba316557e597a90a730f61c762602b7f0e0c (patch) | |
tree | 9b7e9da84cef7d2547c577f1cf94b408cf308619 /arch/um/os-Linux/sys-i386 | |
parent | 5c8aaceab88ac787c0a4038b29143c954c2a45e0 (diff) |
uml: stop saving process FP state
Throw out a lot of code dealing with saving and restoring floating-point
state. In skas mode, where processes run in a restoring floating-point state
on kernel entry and exit is pointless.
This eliminates most of arch/um/os-Linux/sys-{i386,x86_64}/registers.c. Most
of what remained is now arch-indpendent, and can be moved up to
arch/um/os-Linux/registers.c. Both arches need the jmp_buf accessor
get_thread_reg, and i386 needs {save,restore}_fp_regs because it cheats during
sigreturn by getting the fp state using ptrace rather than copying it out of
the process sigcontext.
After this, it turns out that arch/um/include/skas/mode-skas.h is almost
completely unneeded. The declarations in it are variables which either don't
exist or which don't have global scope. The one exception is
kill_off_processes_skas. If that's removed, this header can be deleted.
This uncovered a bug in user.h, which wasn't correctly making sure that a
size_t definition was available to both userspace and kernelspace files.
Signed-off-by: Jeff Dike <jdike@linux.intel.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'arch/um/os-Linux/sys-i386')
-rw-r--r-- | arch/um/os-Linux/sys-i386/registers.c | 108 |
1 files changed, 1 insertions, 107 deletions
diff --git a/arch/um/os-Linux/sys-i386/registers.c b/arch/um/os-Linux/sys-i386/registers.c index 84b44f9cd42a..f171204caa4e 100644 --- a/arch/um/os-Linux/sys-i386/registers.c +++ b/arch/um/os-Linux/sys-i386/registers.c | |||
@@ -4,30 +4,10 @@ | |||
4 | */ | 4 | */ |
5 | 5 | ||
6 | #include <errno.h> | 6 | #include <errno.h> |
7 | #include <string.h> | 7 | #include <sysdep/ptrace_user.h> |
8 | #include "sysdep/ptrace_user.h" | ||
9 | #include "sysdep/ptrace.h" | ||
10 | #include "uml-config.h" | ||
11 | #include "skas_ptregs.h" | ||
12 | #include "registers.h" | ||
13 | #include "longjmp.h" | 8 | #include "longjmp.h" |
14 | #include "user.h" | 9 | #include "user.h" |
15 | 10 | ||
16 | /* These are set once at boot time and not changed thereafter */ | ||
17 | |||
18 | static unsigned long exec_regs[MAX_REG_NR]; | ||
19 | static unsigned long exec_fp_regs[HOST_FP_SIZE]; | ||
20 | static unsigned long exec_fpx_regs[HOST_XFP_SIZE]; | ||
21 | static int have_fpx_regs = 1; | ||
22 | |||
23 | void init_thread_registers(union uml_pt_regs *to) | ||
24 | { | ||
25 | memcpy(to->skas.regs, exec_regs, sizeof(to->skas.regs)); | ||
26 | memcpy(to->skas.fp, exec_fp_regs, sizeof(to->skas.fp)); | ||
27 | if(have_fpx_regs) | ||
28 | memcpy(to->skas.xfp, exec_fpx_regs, sizeof(to->skas.xfp)); | ||
29 | } | ||
30 | |||
31 | /* XXX These need to use [GS]ETFPXREGS and copy_sc_{to,from}_user_skas needs | 11 | /* XXX These need to use [GS]ETFPXREGS and copy_sc_{to,from}_user_skas needs |
32 | * to pass in a sufficiently large buffer | 12 | * to pass in a sufficiently large buffer |
33 | */ | 13 | */ |
@@ -45,92 +25,6 @@ int restore_fp_registers(int pid, unsigned long *fp_regs) | |||
45 | return 0; | 25 | return 0; |
46 | } | 26 | } |
47 | 27 | ||
48 | static int move_registers(int pid, int int_op, union uml_pt_regs *regs, | ||
49 | int fp_op, unsigned long *fp_regs) | ||
50 | { | ||
51 | if(ptrace(int_op, pid, 0, regs->skas.regs) < 0) | ||
52 | return -errno; | ||
53 | |||
54 | if(ptrace(fp_op, pid, 0, fp_regs) < 0) | ||
55 | return -errno; | ||
56 | |||
57 | return 0; | ||
58 | } | ||
59 | |||
60 | void save_registers(int pid, union uml_pt_regs *regs) | ||
61 | { | ||
62 | unsigned long *fp_regs; | ||
63 | int err, fp_op; | ||
64 | |||
65 | if(have_fpx_regs){ | ||
66 | fp_op = PTRACE_GETFPXREGS; | ||
67 | fp_regs = regs->skas.xfp; | ||
68 | } | ||
69 | else { | ||
70 | fp_op = PTRACE_GETFPREGS; | ||
71 | fp_regs = regs->skas.fp; | ||
72 | } | ||
73 | |||
74 | err = move_registers(pid, PTRACE_GETREGS, regs, fp_op, fp_regs); | ||
75 | if(err) | ||
76 | panic("save_registers - saving registers failed, errno = %d\n", | ||
77 | -err); | ||
78 | } | ||
79 | |||
80 | void restore_registers(int pid, union uml_pt_regs *regs) | ||
81 | { | ||
82 | unsigned long *fp_regs; | ||
83 | int err, fp_op; | ||
84 | |||
85 | if(have_fpx_regs){ | ||
86 | fp_op = PTRACE_SETFPXREGS; | ||
87 | fp_regs = regs->skas.xfp; | ||
88 | } | ||
89 | else { | ||
90 | fp_op = PTRACE_SETFPREGS; | ||
91 | fp_regs = regs->skas.fp; | ||
92 | } | ||
93 | |||
94 | err = move_registers(pid, PTRACE_SETREGS, regs, fp_op, fp_regs); | ||
95 | if(err) | ||
96 | panic("restore_registers - saving registers failed, " | ||
97 | "errno = %d\n", -err); | ||
98 | } | ||
99 | |||
100 | void init_registers(int pid) | ||
101 | { | ||
102 | int err; | ||
103 | |||
104 | memset(exec_regs, 0, sizeof(exec_regs)); | ||
105 | err = ptrace(PTRACE_GETREGS, pid, 0, exec_regs); | ||
106 | if(err) | ||
107 | panic("check_ptrace : PTRACE_GETREGS failed, errno = %d", | ||
108 | errno); | ||
109 | |||
110 | errno = 0; | ||
111 | err = ptrace(PTRACE_GETFPXREGS, pid, 0, exec_fpx_regs); | ||
112 | if(!err) | ||
113 | return; | ||
114 | if(errno != EIO) | ||
115 | panic("check_ptrace : PTRACE_GETFPXREGS failed, errno = %d", | ||
116 | errno); | ||
117 | |||
118 | have_fpx_regs = 0; | ||
119 | |||
120 | err = ptrace(PTRACE_GETFPREGS, pid, 0, exec_fp_regs); | ||
121 | if(err) | ||
122 | panic("check_ptrace : PTRACE_GETFPREGS failed, errno = %d", | ||
123 | errno); | ||
124 | } | ||
125 | |||
126 | void get_safe_registers(unsigned long *regs, unsigned long *fp_regs) | ||
127 | { | ||
128 | memcpy(regs, exec_regs, sizeof(exec_regs)); | ||
129 | if(fp_regs != NULL) | ||
130 | memcpy(fp_regs, exec_fp_regs, | ||
131 | HOST_FP_SIZE * sizeof(unsigned long)); | ||
132 | } | ||
133 | |||
134 | unsigned long get_thread_reg(int reg, jmp_buf *buf) | 28 | unsigned long get_thread_reg(int reg, jmp_buf *buf) |
135 | { | 29 | { |
136 | switch(reg){ | 30 | switch(reg){ |