aboutsummaryrefslogtreecommitdiffstats
path: root/arch/s390/kernel/ptrace.c
diff options
context:
space:
mode:
authorMartin Schwidefsky <schwidefsky@de.ibm.com>2011-10-30 10:16:51 -0400
committerMartin Schwidefsky <schwidefsky@de.ibm.com>2011-10-30 10:16:43 -0400
commitd4e81b35b882d96f059afdb0f98e5b6025973b09 (patch)
tree5485be2f8bb0e31d153bf2b7a8bfa3bcabc457ac /arch/s390/kernel/ptrace.c
parentb50511e41aa51a89b4176784a670582424bc7db6 (diff)
[S390] allow all addressing modes
The user space program can change its addressing mode between the 24-bit, 31-bit and the 64-bit mode if the kernel is 64 bit. Currently the kernel always forces the standard amode on signal delivery and signal return and on ptrace: 64-bit for a 64-bit process, 31-bit for a compat process and 31-bit kernels. Change the signal and ptrace code to allow the full range of addressing modes. Signal handlers are run in the standard addressing mode for the process. One caveat is that even an 31-bit compat process can switch to the 64-bit mode. The next signal will switch back into the 31-bit mode and there is no room in the 31-bit compat signal frame to store the information that the program came from the 64-bit mode. Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Diffstat (limited to 'arch/s390/kernel/ptrace.c')
-rw-r--r--arch/s390/kernel/ptrace.c25
1 files changed, 10 insertions, 15 deletions
diff --git a/arch/s390/kernel/ptrace.c b/arch/s390/kernel/ptrace.c
index 41e20763886e..450931a45b68 100644
--- a/arch/s390/kernel/ptrace.c
+++ b/arch/s390/kernel/ptrace.c
@@ -170,8 +170,7 @@ static unsigned long __peek_user(struct task_struct *child, addr_t addr)
170 tmp = *(addr_t *)((addr_t) &task_pt_regs(child)->psw + addr); 170 tmp = *(addr_t *)((addr_t) &task_pt_regs(child)->psw + addr);
171 if (addr == (addr_t) &dummy->regs.psw.mask) 171 if (addr == (addr_t) &dummy->regs.psw.mask)
172 /* Return a clean psw mask. */ 172 /* Return a clean psw mask. */
173 tmp = psw_user_bits | (tmp & PSW_MASK_USER) | 173 tmp = psw_user_bits | (tmp & PSW_MASK_USER);
174 PSW_MASK_EA | PSW_MASK_BA;
175 174
176 } else if (addr < (addr_t) &dummy->regs.orig_gpr2) { 175 } else if (addr < (addr_t) &dummy->regs.orig_gpr2) {
177 /* 176 /*
@@ -286,26 +285,17 @@ static inline void __poke_user_per(struct task_struct *child,
286static int __poke_user(struct task_struct *child, addr_t addr, addr_t data) 285static int __poke_user(struct task_struct *child, addr_t addr, addr_t data)
287{ 286{
288 struct user *dummy = NULL; 287 struct user *dummy = NULL;
289 addr_t offset, tmp; 288 addr_t offset;
290 289
291 if (addr < (addr_t) &dummy->regs.acrs) { 290 if (addr < (addr_t) &dummy->regs.acrs) {
292 /* 291 /*
293 * psw and gprs are stored on the stack 292 * psw and gprs are stored on the stack
294 */ 293 */
295 tmp = (data & ~PSW_MASK_USER) ^ psw_user_bits;
296 if (addr == (addr_t) &dummy->regs.psw.mask && 294 if (addr == (addr_t) &dummy->regs.psw.mask &&
297#ifdef CONFIG_COMPAT 295 ((data & ~PSW_MASK_USER) != psw_user_bits ||
298 tmp != PSW_MASK_BA && 296 ((data & PSW_MASK_EA) && !(data & PSW_MASK_BA))))
299#endif
300 tmp != (PSW_MASK_EA | PSW_MASK_BA))
301 /* Invalid psw mask. */ 297 /* Invalid psw mask. */
302 return -EINVAL; 298 return -EINVAL;
303#ifndef CONFIG_64BIT
304 if (addr == (addr_t) &dummy->regs.psw.addr)
305 /* I'd like to reject addresses without the
306 high order bit but older gdb's rely on it */
307 data |= PSW_ADDR_AMODE;
308#endif
309 if (addr == (addr_t) &dummy->regs.psw.addr) 299 if (addr == (addr_t) &dummy->regs.psw.addr)
310 /* 300 /*
311 * The debugger changed the instruction address, 301 * The debugger changed the instruction address,
@@ -517,7 +507,8 @@ static u32 __peek_user_compat(struct task_struct *child, addr_t addr)
517 tmp = psw32_user_bits | (tmp & PSW32_MASK_USER); 507 tmp = psw32_user_bits | (tmp & PSW32_MASK_USER);
518 } else if (addr == (addr_t) &dummy32->regs.psw.addr) { 508 } else if (addr == (addr_t) &dummy32->regs.psw.addr) {
519 /* Fake a 31 bit psw address. */ 509 /* Fake a 31 bit psw address. */
520 tmp = (__u32) regs->psw.addr | PSW32_ADDR_AMODE; 510 tmp = (__u32) regs->psw.addr |
511 (__u32)(regs->psw.mask & PSW_MASK_BA);
521 } else { 512 } else {
522 /* gpr 0-15 */ 513 /* gpr 0-15 */
523 tmp = *(__u32 *)((addr_t) &regs->psw + addr*2 + 4); 514 tmp = *(__u32 *)((addr_t) &regs->psw + addr*2 + 4);
@@ -615,10 +606,14 @@ static int __poke_user_compat(struct task_struct *child,
615 /* Invalid psw mask. */ 606 /* Invalid psw mask. */
616 return -EINVAL; 607 return -EINVAL;
617 regs->psw.mask = (regs->psw.mask & ~PSW_MASK_USER) | 608 regs->psw.mask = (regs->psw.mask & ~PSW_MASK_USER) |
609 (regs->psw.mask & PSW_MASK_BA) |
618 (__u64)(tmp & PSW32_MASK_USER) << 32; 610 (__u64)(tmp & PSW32_MASK_USER) << 32;
619 } else if (addr == (addr_t) &dummy32->regs.psw.addr) { 611 } else if (addr == (addr_t) &dummy32->regs.psw.addr) {
620 /* Build a 64 bit psw address from 31 bit address. */ 612 /* Build a 64 bit psw address from 31 bit address. */
621 regs->psw.addr = (__u64) tmp & PSW32_ADDR_INSN; 613 regs->psw.addr = (__u64) tmp & PSW32_ADDR_INSN;
614 /* Transfer 31 bit amode bit to psw mask. */
615 regs->psw.mask = (regs->psw.mask & ~PSW_MASK_BA) |
616 (__u64)(tmp & PSW32_ADDR_AMODE);
622 /* 617 /*
623 * The debugger changed the instruction address, 618 * The debugger changed the instruction address,
624 * reset system call restart, see signal.c:do_signal 619 * reset system call restart, see signal.c:do_signal