Merge branch 'regset' (PTRACE_SETREGSET data leakage)
Merge PTRACE_SETREGSET leakage fixes from Dave Martin: "This series is the collection of fixes I proposed on this topic, that have not yet appeared upstream or in the stable branches, The issue can leak kernel stack, but doesn't appear to allow userspace to attack the kernel directly. The affected architectures are c6x, h8300, metag, mips and sparc. [ Mark Salter points out that c6x has no MMU or other mechanism to prevent userspace access to kernel code or data on c6x, but it doesn't hurt to clean that case up too. ] The bugs arise from use of user_regset_copyin(). Users of user_regset_copyin() can work in one of two ways: 1) Copy directly to thread_struct or equivalent. (This seems to be the design assumption of the regset API, and is the most common approach.) 2) Copy to a local variable and then transfer to thread_struct. (A significant minority of cases.) Buggy code typically involves approach 2" * emailed patches from Dave Martin <Dave.Martin@arm.com>: sparc/ptrace: Preserve previous registers for short regset write mips/ptrace: Preserve previous registers for short regset write metag/ptrace: Reject partial NT_METAG_RPIPE writes metag/ptrace: Provide default TXSTATUS for short NT_PRSTATUS metag/ptrace: Preserve previous registers for short regset write h8300/ptrace: Fix incorrect register transfer count c6x/ptrace: Remove useless PTRACE_SETREGSET implementation
This commit is contained in:
commit
72c33734b5
@ -70,46 +70,6 @@ static int gpr_get(struct task_struct *target,
|
|||||||
0, sizeof(*regs));
|
0, sizeof(*regs));
|
||||||
}
|
}
|
||||||
|
|
||||||
static int gpr_set(struct task_struct *target,
|
|
||||||
const struct user_regset *regset,
|
|
||||||
unsigned int pos, unsigned int count,
|
|
||||||
const void *kbuf, const void __user *ubuf)
|
|
||||||
{
|
|
||||||
int ret;
|
|
||||||
struct pt_regs *regs = task_pt_regs(target);
|
|
||||||
|
|
||||||
/* Don't copyin TSR or CSR */
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
|
||||||
®s,
|
|
||||||
0, PT_TSR * sizeof(long));
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = user_regset_copyin_ignore(&pos, &count, &kbuf, &ubuf,
|
|
||||||
PT_TSR * sizeof(long),
|
|
||||||
(PT_TSR + 1) * sizeof(long));
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
|
||||||
®s,
|
|
||||||
(PT_TSR + 1) * sizeof(long),
|
|
||||||
PT_CSR * sizeof(long));
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = user_regset_copyin_ignore(&pos, &count, &kbuf, &ubuf,
|
|
||||||
PT_CSR * sizeof(long),
|
|
||||||
(PT_CSR + 1) * sizeof(long));
|
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
|
||||||
®s,
|
|
||||||
(PT_CSR + 1) * sizeof(long), -1);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum c6x_regset {
|
enum c6x_regset {
|
||||||
REGSET_GPR,
|
REGSET_GPR,
|
||||||
};
|
};
|
||||||
@ -121,7 +81,6 @@ static const struct user_regset c6x_regsets[] = {
|
|||||||
.size = sizeof(u32),
|
.size = sizeof(u32),
|
||||||
.align = sizeof(u32),
|
.align = sizeof(u32),
|
||||||
.get = gpr_get,
|
.get = gpr_get,
|
||||||
.set = gpr_set
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -95,7 +95,8 @@ static int regs_get(struct task_struct *target,
|
|||||||
long *reg = (long *)®s;
|
long *reg = (long *)®s;
|
||||||
|
|
||||||
/* build user regs in buffer */
|
/* build user regs in buffer */
|
||||||
for (r = 0; r < ARRAY_SIZE(register_offset); r++)
|
BUILD_BUG_ON(sizeof(regs) % sizeof(long) != 0);
|
||||||
|
for (r = 0; r < sizeof(regs) / sizeof(long); r++)
|
||||||
*reg++ = h8300_get_reg(target, r);
|
*reg++ = h8300_get_reg(target, r);
|
||||||
|
|
||||||
return user_regset_copyout(&pos, &count, &kbuf, &ubuf,
|
return user_regset_copyout(&pos, &count, &kbuf, &ubuf,
|
||||||
@ -113,7 +114,8 @@ static int regs_set(struct task_struct *target,
|
|||||||
long *reg;
|
long *reg;
|
||||||
|
|
||||||
/* build user regs in buffer */
|
/* build user regs in buffer */
|
||||||
for (reg = (long *)®s, r = 0; r < ARRAY_SIZE(register_offset); r++)
|
BUILD_BUG_ON(sizeof(regs) % sizeof(long) != 0);
|
||||||
|
for (reg = (long *)®s, r = 0; r < sizeof(regs) / sizeof(long); r++)
|
||||||
*reg++ = h8300_get_reg(target, r);
|
*reg++ = h8300_get_reg(target, r);
|
||||||
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
||||||
@ -122,7 +124,7 @@ static int regs_set(struct task_struct *target,
|
|||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
/* write back to pt_regs */
|
/* write back to pt_regs */
|
||||||
for (reg = (long *)®s, r = 0; r < ARRAY_SIZE(register_offset); r++)
|
for (reg = (long *)®s, r = 0; r < sizeof(regs) / sizeof(long); r++)
|
||||||
h8300_put_reg(target, r, *reg++);
|
h8300_put_reg(target, r, *reg++);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,16 @@
|
|||||||
* user_regset definitions.
|
* user_regset definitions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
static unsigned long user_txstatus(const struct pt_regs *regs)
|
||||||
|
{
|
||||||
|
unsigned long data = (unsigned long)regs->ctx.Flags;
|
||||||
|
|
||||||
|
if (regs->ctx.SaveMask & TBICTX_CBUF_BIT)
|
||||||
|
data |= USER_GP_REGS_STATUS_CATCH_BIT;
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
int metag_gp_regs_copyout(const struct pt_regs *regs,
|
int metag_gp_regs_copyout(const struct pt_regs *regs,
|
||||||
unsigned int pos, unsigned int count,
|
unsigned int pos, unsigned int count,
|
||||||
void *kbuf, void __user *ubuf)
|
void *kbuf, void __user *ubuf)
|
||||||
@ -64,9 +74,7 @@ int metag_gp_regs_copyout(const struct pt_regs *regs,
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
/* TXSTATUS */
|
/* TXSTATUS */
|
||||||
data = (unsigned long)regs->ctx.Flags;
|
data = user_txstatus(regs);
|
||||||
if (regs->ctx.SaveMask & TBICTX_CBUF_BIT)
|
|
||||||
data |= USER_GP_REGS_STATUS_CATCH_BIT;
|
|
||||||
ret = user_regset_copyout(&pos, &count, &kbuf, &ubuf,
|
ret = user_regset_copyout(&pos, &count, &kbuf, &ubuf,
|
||||||
&data, 4*25, 4*26);
|
&data, 4*25, 4*26);
|
||||||
if (ret)
|
if (ret)
|
||||||
@ -121,6 +129,7 @@ int metag_gp_regs_copyin(struct pt_regs *regs,
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
/* TXSTATUS */
|
/* TXSTATUS */
|
||||||
|
data = user_txstatus(regs);
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
||||||
&data, 4*25, 4*26);
|
&data, 4*25, 4*26);
|
||||||
if (ret)
|
if (ret)
|
||||||
@ -246,6 +255,8 @@ int metag_rp_state_copyin(struct pt_regs *regs,
|
|||||||
unsigned long long *ptr;
|
unsigned long long *ptr;
|
||||||
int ret, i;
|
int ret, i;
|
||||||
|
|
||||||
|
if (count < 4*13)
|
||||||
|
return -EINVAL;
|
||||||
/* Read the entire pipeline before making any changes */
|
/* Read the entire pipeline before making any changes */
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
||||||
&rp, 0, 4*13);
|
&rp, 0, 4*13);
|
||||||
@ -305,7 +316,7 @@ static int metag_tls_set(struct task_struct *target,
|
|||||||
const void *kbuf, const void __user *ubuf)
|
const void *kbuf, const void __user *ubuf)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
void __user *tls;
|
void __user *tls = target->thread.tls_ptr;
|
||||||
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf, &tls, 0, -1);
|
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf, &tls, 0, -1);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -456,7 +456,8 @@ static int fpr_set(struct task_struct *target,
|
|||||||
&target->thread.fpu,
|
&target->thread.fpu,
|
||||||
0, sizeof(elf_fpregset_t));
|
0, sizeof(elf_fpregset_t));
|
||||||
|
|
||||||
for (i = 0; i < NUM_FPU_REGS; i++) {
|
BUILD_BUG_ON(sizeof(fpr_val) != sizeof(elf_fpreg_t));
|
||||||
|
for (i = 0; i < NUM_FPU_REGS && count >= sizeof(elf_fpreg_t); i++) {
|
||||||
err = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
err = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
||||||
&fpr_val, i * sizeof(elf_fpreg_t),
|
&fpr_val, i * sizeof(elf_fpreg_t),
|
||||||
(i + 1) * sizeof(elf_fpreg_t));
|
(i + 1) * sizeof(elf_fpreg_t));
|
||||||
|
@ -351,7 +351,7 @@ static int genregs64_set(struct task_struct *target,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
unsigned long y;
|
unsigned long y = regs->y;
|
||||||
|
|
||||||
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
ret = user_regset_copyin(&pos, &count, &kbuf, &ubuf,
|
||||||
&y,
|
&y,
|
||||||
|
Loading…
Reference in New Issue
Block a user