/* * These are our native regset flavors.
*/ enum parisc_regset {
REGSET_GENERAL,
REGSET_FP
};
/* * Called by kernel/ptrace.c when detaching.. * * Make sure single step bits etc are not set.
*/ void ptrace_disable(struct task_struct *task)
{
clear_tsk_thread_flag(task, TIF_SINGLESTEP);
clear_tsk_thread_flag(task, TIF_BLOCKSTEP);
/* make sure the trap bits are not set */
pa_psw(task)->r = 0;
pa_psw(task)->t = 0;
pa_psw(task)->h = 0;
pa_psw(task)->l = 0;
}
/* * The following functions are called by ptrace_resume() when * enabling or disabling single/block tracing.
*/ void user_disable_single_step(struct task_struct *task)
{
ptrace_disable(task);
}
if (pa_psw(task)->n) { /* Nullified, just crank over the queue. */
task_regs(task)->iaoq[0] = task_regs(task)->iaoq[1];
task_regs(task)->iasq[0] = task_regs(task)->iasq[1];
task_regs(task)->iaoq[1] = task_regs(task)->iaoq[0] + 4;
pa_psw(task)->n = 0;
pa_psw(task)->x = 0;
pa_psw(task)->y = 0;
pa_psw(task)->z = 0;
pa_psw(task)->b = 0;
ptrace_disable(task); /* Don't wake up the task, but let the
parent know something happened. */
force_sig_fault_to_task(SIGTRAP, TRAP_TRACE,
(void __user *) (task_regs(task)->iaoq[0] & ~3),
task); /* notify_parent(task, SIGCHLD); */ return;
}
/* Enable recovery counter traps. The recovery counter * itself will be set to zero on a task switch. If the * task is suspended on a syscall then the syscall return * path will overwrite the recovery counter with a suitable * value such that it traps once back in user space. We * disable interrupts in the tasks PSW here also, to avoid * interrupts while the recovery counter is decrementing.
*/
pa_psw(task)->r = 1;
pa_psw(task)->t = 0;
pa_psw(task)->h = 0;
pa_psw(task)->l = 0;
}
/* Read the word at location addr in the USER area. For ptraced
processes, the kernel saves all regs on a syscall. */ case PTRACE_PEEKUSR: if ((addr & (sizeof(unsignedlong)-1)) ||
addr >= sizeof(struct pt_regs)) break;
tmp = *(unsignedlong *) ((char *) task_regs(child) + addr);
ret = put_user(tmp, datap); break;
/* Write the word at location addr in the USER area. This will need to change when the kernel no longer saves all regs on a syscall. FIXME. There is a problem at the moment in that r3-r18 are only saved if the process is ptraced on syscall entry, and even then those values are overwritten by actual register values on syscall
exit. */ case PTRACE_POKEUSR: /* Some register values written here may be ignored in * entry.S:syscall_restore_rfi; e.g. iaoq is written with * r31/r31+4, and not with the values in pt_regs.
*/ if (addr == PT_PSW) { /* Allow writing to Nullify, Divide-step-correction, * and carry/borrow bits. * BEWARE, if you set N, and then single step, it won't * stop on the nullified instruction.
*/
data &= USER_PSW_BITS;
task_regs(child)->gr[0] &= ~USER_PSW_BITS;
task_regs(child)->gr[0] |= data;
ret = 0; break;
}
case PTRACE_GETREGS: /* Get all gp regs from the child. */ return copy_regset_to_user(child,
task_user_regset_view(current),
REGSET_GENERAL,
0, user_regs_struct_size,
datap);
case PTRACE_SETREGS: /* Set all gp regs in the child. */ return copy_regset_from_user(child,
task_user_regset_view(current),
REGSET_GENERAL,
0, user_regs_struct_size,
datap);
case PTRACE_GETFPREGS: /* Get the child FPU state. */ return copy_regset_to_user(child,
task_user_regset_view(current),
REGSET_FP,
0, sizeof(struct user_fp_struct),
datap);
case PTRACE_SETFPREGS: /* Set the child FPU state. */ return copy_regset_from_user(child,
task_user_regset_view(current),
REGSET_FP,
0, sizeof(struct user_fp_struct),
datap);
default:
ret = ptrace_request(child, request, addr, data); break;
}
return ret;
}
#ifdef CONFIG_COMPAT
/* This function is needed to translate 32 bit pt_regs offsets in to * 64 bit pt_regs offsets. For example, a 32 bit gdb under a 64 bit kernel * will request offset 12 if it wants gr3, but the lower 32 bits of * the 64 bit kernels view of gr3 will be at offset 28 (3*8 + 4). * This code relies on a 32 bit pt_regs being comprised of 32 bit values * except for the fp registers which (a) are 64 bits, and (b) follow * the gr registers at the start of pt_regs. The 32 bit pt_regs should * be half the size of the 64 bit pt_regs, plus 32*4 to allow for fr[] * being 64 bit in both cases.
*/
/* Write the word at location addr in the USER area. This will need to change when the kernel no longer saves all regs on a syscall. FIXME. There is a problem at the moment in that r3-r18 are only saved if the process is ptraced on syscall entry, and even then those values are overwritten by actual register values on syscall
exit. */ case PTRACE_POKEUSR: /* Some register values written here may be ignored in * entry.S:syscall_restore_rfi; e.g. iaoq is written with * r31/r31+4, and not with the values in pt_regs.
*/ if (addr == PT_PSW) { /* Since PT_PSW==0, it is valid for 32 bit processes * under 64 bit kernels as well.
*/
ret = arch_ptrace(child, request, addr, data);
} else { if (addr & (sizeof(compat_uint_t)-1)) break;
addr = translate_usr_offset(addr); if (addr >= sizeof(struct pt_regs)) break; if (addr == PT_IAOQ0+4 || addr == PT_IAOQ1+4) {
data |= PRIV_USER; /* ensure userspace privilege */
} if (addr >= PT_FR0 && addr <= PT_FR31 + 4) { /* Special case, fp regs are 64 bits anyway */
*(__u32 *) ((char *) task_regs(child) + addr) = data;
ret = 0;
} elseif ((addr >= PT_GR1+4 && addr <= PT_GR31+4) ||
addr == PT_IAOQ0+4 || addr == PT_IAOQ1+4 ||
addr == PT_SAR+4) { /* Zero the top 32 bits */
*(__u32 *) ((char *) task_regs(child) + addr - 4) = 0;
*(__u32 *) ((char *) task_regs(child) + addr) = data;
ret = 0;
}
} break; case PTRACE_GETREGS: case PTRACE_SETREGS: case PTRACE_GETFPREGS: case PTRACE_SETFPREGS: return arch_ptrace(child, request, addr, data);
default:
ret = compat_ptrace_request(child, request, addr, data); break;
}
return ret;
} #endif
long do_syscall_trace_enter(struct pt_regs *regs)
{ if (test_thread_flag(TIF_SYSCALL_TRACE)) { int rc = ptrace_report_syscall_entry(regs);
/* * As tracesys_next does not set %r28 to -ENOSYS * when %r20 is set to -1, initialize it here.
*/
regs->gr[28] = -ENOSYS;
if (rc) { /* * A nonzero return code from * ptrace_report_syscall_entry() tells us * to prevent the syscall execution. Skip * the syscall call and the syscall restart handling. * * Note that the tracer may also just change * regs->gr[20] to an invalid syscall number, * that is handled by tracesys_next.
*/
regs->gr[20] = -1UL; return -1;
}
}
/* Do the secure computing check after ptrace. */ if (secure_computing() == -1) return -1;
#ifdef CONFIG_HAVE_SYSCALL_TRACEPOINTS if (unlikely(test_thread_flag(TIF_SYSCALL_TRACEPOINT)))
trace_sys_enter(regs, regs->gr[20]); #endif
staticunsignedlong get_reg(struct pt_regs *regs, int num)
{ switch (num) { case RI(gr[0]) ... RI(gr[31]): return regs->gr[num - RI(gr[0])]; case RI(sr[0]) ... RI(sr[7]): return regs->sr[num - RI(sr[0])]; case RI(iasq[0]): return regs->iasq[0]; case RI(iasq[1]): return regs->iasq[1]; case RI(iaoq[0]): return regs->iaoq[0]; case RI(iaoq[1]): return regs->iaoq[1]; case RI(sar): return regs->sar; case RI(iir): return regs->iir; case RI(isr): return regs->isr; case RI(ior): return regs->ior; case RI(ipsw): return regs->ipsw; case RI(cr27): return regs->cr27; case RI(cr0): return mfctl(0); case RI(cr24): return mfctl(24); case RI(cr25): return mfctl(25); case RI(cr26): return mfctl(26); case RI(cr28): return mfctl(28); case RI(cr29): return mfctl(29); case RI(cr30): return mfctl(30); case RI(cr31): return mfctl(31); case RI(cr8): return mfctl(8); case RI(cr9): return mfctl(9); case RI(cr12): return mfctl(12); case RI(cr13): return mfctl(13); case RI(cr10): return mfctl(10); case RI(cr15): return mfctl(15); default: return 0;
}
}
staticvoid set_reg(struct pt_regs *regs, int num, unsignedlong val)
{ switch (num) { case RI(gr[0]): /* * PSW is in gr[0]. * Allow writing to Nullify, Divide-step-correction, * and carry/borrow bits. * BEWARE, if you set N, and then single step, it won't * stop on the nullified instruction.
*/
val &= USER_PSW_BITS;
regs->gr[0] &= ~USER_PSW_BITS;
regs->gr[0] |= val; return; case RI(gr[1]) ... RI(gr[31]):
regs->gr[num - RI(gr[0])] = val; return; case RI(iaoq[0]): case RI(iaoq[1]): /* set 2 lowest bits to ensure userspace privilege: */
regs->iaoq[num - RI(iaoq[0])] = val | PRIV_USER; return; case RI(sar): regs->sar = val; return; default: return; #if 0 /* do not allow to change any of the following registers (yet) */ case RI(sr[0]) ... RI(sr[7]): return regs->sr[num - RI(sr[0])]; case RI(iasq[0]): return regs->iasq[0]; case RI(iasq[1]): return regs->iasq[1]; case RI(iir): return regs->iir; case RI(isr): return regs->isr; case RI(ior): return regs->ior; case RI(ipsw): return regs->ipsw; case RI(cr27): return regs->cr27; case cr0, cr24, cr25, cr26, cr27, cr28, cr29, cr30, cr31; case cr8, cr9, cr12, cr13, cr10, cr15; #endif
}
}
/** * regs_query_register_offset() - query register offset from its name * @name: the name of a register * * regs_query_register_offset() returns the offset of a register in struct * pt_regs from its name. If the name is invalid, this returns -EINVAL;
*/ int regs_query_register_offset(constchar *name)
{ conststruct pt_regs_offset *roff; for (roff = regoffset_table; roff->name != NULL; roff++) if (!strcmp(roff->name, name)) return roff->offset; return -EINVAL;
}
/** * regs_query_register_name() - query register name from its offset * @offset: the offset of a register in struct pt_regs. * * regs_query_register_name() returns the name of a register from its * offset in struct pt_regs. If the @offset is invalid, this returns NULL;
*/ constchar *regs_query_register_name(unsignedint offset)
{ conststruct pt_regs_offset *roff; for (roff = regoffset_table; roff->name != NULL; roff++) if (roff->offset == offset) return roff->name; return NULL;
}
/** * regs_within_kernel_stack() - check the address in the stack * @regs: pt_regs which contains kernel stack pointer. * @addr: address which is checked. * * regs_within_kernel_stack() checks @addr is within the kernel stack page(s). * If @addr is within the kernel stack, it returns true. If not, returns false.
*/ int regs_within_kernel_stack(struct pt_regs *regs, unsignedlong addr)
{ return ((addr & ~(THREAD_SIZE - 1)) ==
(kernel_stack_pointer(regs) & ~(THREAD_SIZE - 1)));
}
/** * regs_get_kernel_stack_nth() - get Nth entry of the stack * @regs: pt_regs which contains kernel stack pointer. * @n: stack entry number. * * regs_get_kernel_stack_nth() returns @n th entry of the kernel stack which * is specified by @regs. If the @n th entry is NOT in the kernel stack, * this returns 0.
*/ unsignedlong regs_get_kernel_stack_nth(struct pt_regs *regs, unsignedint n)
{ unsignedlong *addr = (unsignedlong *)kernel_stack_pointer(regs);
addr -= n;
if (!regs_within_kernel_stack(regs, (unsignedlong)addr)) return 0;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.