diff options
Diffstat (limited to 'block/compat_ioctl.c')
-rw-r--r-- | block/compat_ioctl.c | 101 |
1 files changed, 49 insertions, 52 deletions
diff --git a/block/compat_ioctl.c b/block/compat_ioctl.c index 504344d29022..5b3db0640d87 100644 --- a/block/compat_ioctl.c +++ b/block/compat_ioctl.c | |||
@@ -688,12 +688,40 @@ static int compat_blkdev_driver_ioctl(struct block_device *bdev, fmode_t mode, | |||
688 | return __blkdev_driver_ioctl(bdev, mode, cmd, arg); | 688 | return __blkdev_driver_ioctl(bdev, mode, cmd, arg); |
689 | } | 689 | } |
690 | 690 | ||
691 | static int compat_blkdev_locked_ioctl(struct block_device *bdev, | 691 | /* Most of the generic ioctls are handled in the normal fallback path. |
692 | unsigned cmd, unsigned long arg) | 692 | This assumes the blkdev's low level compat_ioctl always returns |
693 | ENOIOCTLCMD for unknown ioctls. */ | ||
694 | long compat_blkdev_ioctl(struct file *file, unsigned cmd, unsigned long arg) | ||
693 | { | 695 | { |
696 | int ret = -ENOIOCTLCMD; | ||
697 | struct inode *inode = file->f_mapping->host; | ||
698 | struct block_device *bdev = inode->i_bdev; | ||
699 | struct gendisk *disk = bdev->bd_disk; | ||
700 | fmode_t mode = file->f_mode; | ||
694 | struct backing_dev_info *bdi; | 701 | struct backing_dev_info *bdi; |
702 | loff_t size; | ||
703 | |||
704 | if (file->f_flags & O_NDELAY) | ||
705 | mode |= FMODE_NDELAY_NOW; | ||
695 | 706 | ||
696 | switch (cmd) { | 707 | switch (cmd) { |
708 | case HDIO_GETGEO: | ||
709 | return compat_hdio_getgeo(disk, bdev, compat_ptr(arg)); | ||
710 | case BLKFLSBUF: | ||
711 | case BLKROSET: | ||
712 | case BLKDISCARD: | ||
713 | /* | ||
714 | * the ones below are implemented in blkdev_locked_ioctl, | ||
715 | * but we call blkdev_ioctl, which gets the lock for us | ||
716 | */ | ||
717 | case BLKRRPART: | ||
718 | return blkdev_ioctl(inode, file, cmd, | ||
719 | (unsigned long)compat_ptr(arg)); | ||
720 | case BLKBSZSET_32: | ||
721 | return blkdev_ioctl(inode, file, BLKBSZSET, | ||
722 | (unsigned long)compat_ptr(arg)); | ||
723 | case BLKPG: | ||
724 | return compat_blkpg_ioctl(inode, file, cmd, compat_ptr(arg)); | ||
697 | case BLKRAGET: | 725 | case BLKRAGET: |
698 | case BLKFRAGET: | 726 | case BLKFRAGET: |
699 | if (!arg) | 727 | if (!arg) |
@@ -719,67 +747,36 @@ static int compat_blkdev_locked_ioctl(struct block_device *bdev, | |||
719 | bdi = blk_get_backing_dev_info(bdev); | 747 | bdi = blk_get_backing_dev_info(bdev); |
720 | if (bdi == NULL) | 748 | if (bdi == NULL) |
721 | return -ENOTTY; | 749 | return -ENOTTY; |
750 | lock_kernel(); | ||
722 | bdi->ra_pages = (arg * 512) / PAGE_CACHE_SIZE; | 751 | bdi->ra_pages = (arg * 512) / PAGE_CACHE_SIZE; |
752 | unlock_kernel(); | ||
723 | return 0; | 753 | return 0; |
724 | case BLKGETSIZE: | 754 | case BLKGETSIZE: |
725 | if ((bdev->bd_inode->i_size >> 9) > ~0UL) | 755 | size = bdev->bd_inode->i_size; |
756 | if ((size >> 9) > ~0UL) | ||
726 | return -EFBIG; | 757 | return -EFBIG; |
727 | return compat_put_ulong(arg, bdev->bd_inode->i_size >> 9); | 758 | return compat_put_ulong(arg, size >> 9); |
728 | 759 | ||
729 | case BLKGETSIZE64_32: | 760 | case BLKGETSIZE64_32: |
730 | return compat_put_u64(arg, bdev->bd_inode->i_size); | 761 | return compat_put_u64(arg, bdev->bd_inode->i_size); |
731 | 762 | ||
732 | case BLKTRACESETUP32: | 763 | case BLKTRACESETUP32: |
733 | return compat_blk_trace_setup(bdev, compat_ptr(arg)); | 764 | lock_kernel(); |
765 | ret = compat_blk_trace_setup(bdev, compat_ptr(arg)); | ||
766 | unlock_kernel(); | ||
767 | return ret; | ||
734 | case BLKTRACESTART: /* compatible */ | 768 | case BLKTRACESTART: /* compatible */ |
735 | case BLKTRACESTOP: /* compatible */ | 769 | case BLKTRACESTOP: /* compatible */ |
736 | case BLKTRACETEARDOWN: /* compatible */ | 770 | case BLKTRACETEARDOWN: /* compatible */ |
737 | return blk_trace_ioctl(bdev, cmd, compat_ptr(arg)); | 771 | lock_kernel(); |
738 | } | 772 | ret = blk_trace_ioctl(bdev, cmd, compat_ptr(arg)); |
739 | return -ENOIOCTLCMD; | 773 | unlock_kernel(); |
740 | } | ||
741 | |||
742 | /* Most of the generic ioctls are handled in the normal fallback path. | ||
743 | This assumes the blkdev's low level compat_ioctl always returns | ||
744 | ENOIOCTLCMD for unknown ioctls. */ | ||
745 | long compat_blkdev_ioctl(struct file *file, unsigned cmd, unsigned long arg) | ||
746 | { | ||
747 | int ret = -ENOIOCTLCMD; | ||
748 | struct inode *inode = file->f_mapping->host; | ||
749 | struct block_device *bdev = inode->i_bdev; | ||
750 | struct gendisk *disk = bdev->bd_disk; | ||
751 | fmode_t mode = file->f_mode; | ||
752 | if (file->f_flags & O_NDELAY) | ||
753 | mode |= FMODE_NDELAY_NOW; | ||
754 | |||
755 | switch (cmd) { | ||
756 | case HDIO_GETGEO: | ||
757 | return compat_hdio_getgeo(disk, bdev, compat_ptr(arg)); | ||
758 | case BLKFLSBUF: | ||
759 | case BLKROSET: | ||
760 | case BLKDISCARD: | ||
761 | /* | ||
762 | * the ones below are implemented in blkdev_locked_ioctl, | ||
763 | * but we call blkdev_ioctl, which gets the lock for us | ||
764 | */ | ||
765 | case BLKRRPART: | ||
766 | return blkdev_ioctl(inode, file, cmd, | ||
767 | (unsigned long)compat_ptr(arg)); | ||
768 | case BLKBSZSET_32: | ||
769 | return blkdev_ioctl(inode, file, BLKBSZSET, | ||
770 | (unsigned long)compat_ptr(arg)); | ||
771 | case BLKPG: | ||
772 | return compat_blkpg_ioctl(inode, file, cmd, compat_ptr(arg)); | ||
773 | } | ||
774 | |||
775 | lock_kernel(); | ||
776 | ret = compat_blkdev_locked_ioctl(bdev, cmd, arg); | ||
777 | unlock_kernel(); | ||
778 | if (ret == -ENOIOCTLCMD && disk->fops->compat_ioctl) | ||
779 | ret = disk->fops->compat_ioctl(bdev, mode, cmd, arg); | ||
780 | |||
781 | if (ret != -ENOIOCTLCMD) | ||
782 | return ret; | 774 | return ret; |
783 | 775 | default: | |
784 | return compat_blkdev_driver_ioctl(bdev, mode, cmd, arg); | 776 | if (disk->fops->compat_ioctl) |
777 | ret = disk->fops->compat_ioctl(bdev, mode, cmd, arg); | ||
778 | if (ret == -ENOIOCTLCMD) | ||
779 | ret = compat_blkdev_driver_ioctl(bdev, mode, cmd, arg); | ||
780 | return ret; | ||
781 | } | ||
785 | } | 782 | } |