Re: must-fix list, v5

From: Andi Kleen (ak@suse.de)
Date: Thu May 29 2003 - 04:22:03 EST


Pavel Machek <pavel@xxxxxxx> writes:

> + kiov = (sg_iovec_t *) sgp->dxferp;
> + for (i = 0; i < sgp->iovec_count; i++) {
> + u32 iov_base32;
> + if (__get_user(iov_base32, &uiov->iov_base) ||
> + __get_user(kiov->iov_len, &uiov->iov_len))
> + return -EFAULT;
> + if (verify_area(VERIFY_WRITE, compat_ptr(iov_base32), kiov->iov_len))
> + return -EFAULT;
> + kiov->iov_base = compat_ptr(iov_base32);


This part won't work on sparc64 because it has separate address spaces
for user/kernel. I did it this way for x86-64 because I didn't realize
the sparc64 problem yet and it works fine there. For sparc64 it needs to be
converted to compat_alloc_user_space() (see net/compat.c for an example)
or to alloc/copy the payload again. The first is prefered.

Pavel you would need to fix this first, otherwise Ben Collins and DaveM
will be unhappy as soon as they want to burn a CD.

> + } else {
> + err = verify_area(VERIFY_WRITE, compat_ptr(dxferp32), sg_io64.dxfer_len);
> + if (err)
> + goto out;
> +
> + sg_io64.dxferp = compat_ptr(dxferp32);
> + }

Same here.

t_user(cdread.cdread_lba, &((struct cdrom_read32 *)arg)->cdread_lba);
> + err |= __get_user(addr, &((struct cdrom_read32 *)arg)->cdread_bufaddr);
> + err |= __get_user(cdread.cdread_buflen, &((struct cdrom_read32 *)arg)->cdread_buflen);
> + if (err)
> + return -EFAULT;
> + if (verify_area(VERIFY_WRITE, compat_ptr(addr), cdread.cdread_buflen))
> + return -EFAULT;
> + cdread.cdread_bufaddr = compat_ptr(addr);

Same here.

> + err |= __get_user(addr, &((struct cdrom_read_audio32 *)arg)->buf);
> + if (err)
> + return -EFAULT;
> +
> +
> + if (verify_area(VERIFY_WRITE, compat_ptr(addr), cdreadaudio.nframes*2352))
> + return -EFAULT;

And here.

err |= __get_user(addr, &((struct cdrom_generic_command32 *)arg)->buffer);
> + err |= __get_user(cgc.buflen, &((struct cdrom_generic_command32 *)arg)->buflen);
> + if (err)
> + return -EFAULT;
> + if (verify_area(VERIFY_WRITE, compat_ptr(addr), cgc.buflen))
> + return -EFAULT;
> + cgc.buffer = compat_ptr(addr);

And here

> + if (iobuf32.buffer == (compat_caddr_t) NULL || iobuf32.length == 0) {
> + iobuf.buffer = (void*)(unsigned long)iobuf32.buffer;
> + } else {
> + iobuf.buffer = compat_ptr(iobuf32.buffer);
> + if (verify_area(VERIFY_WRITE, iobuf.buffer, iobuf.length))
> + return -EINVAL;
> + }

And here.

> + if (sioc32.arg == (compat_caddr_t) NULL || sioc32.length == 0) {
> + sioc.arg = (void*)(unsigned long)sioc32.arg;
> + } else {
> + sioc.arg = compat_ptr(sioc32.arg);
> + if (verify_area(VERIFY_WRITE, sioc.arg, sioc32.length))
> + return -EFAULT;
> + }
> +
> + old_fs = get_fs(); set_fs (KERNEL_DS);
> + err = sys_ioctl (fd, cmd, (unsigned long)&sioc);

And here

> +
> + if (get_user(karg.start, &uarg->start) ||
> + get_user(karg.length, &uarg->length) ||
> + get_user(tmp, &uarg->ptr))
> + return -EFAULT;
> +
> + karg.ptr = compat_ptr(tmp);
> + if (verify_area(VERIFY_WRITE, karg.ptr, karg.length))
> + return -EFAULT;

And here.
> +
> + set_fs(KERNEL_DS);
> + if (MEMREADOOB32 == cmd)
> + ret = sys_ioctl(fd, MEMREADOOB, (unsigned long)&karg);
> + else if (MEMWRITEOOB32 == cmd)
> + ret = sys_ioctl(fd, MEMWRITEOOB, (unsigned long)&karg);
> + else
> + ret = -EINVAL;
> + set_fs(old_fs);

-Andi
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/