|
@@ -644,6 +644,36 @@ static void loop_reread_partitions(struct loop_device *lo,
|
|
|
__func__, lo->lo_number, lo->lo_file_name, rc);
|
|
|
}
|
|
|
|
|
|
+static inline int is_loop_device(struct file *file)
|
|
|
+{
|
|
|
+ struct inode *i = file->f_mapping->host;
|
|
|
+
|
|
|
+ return i && S_ISBLK(i->i_mode) && MAJOR(i->i_rdev) == LOOP_MAJOR;
|
|
|
+}
|
|
|
+
|
|
|
+static int loop_validate_file(struct file *file, struct block_device *bdev)
|
|
|
+{
|
|
|
+ struct inode *inode = file->f_mapping->host;
|
|
|
+ struct file *f = file;
|
|
|
+
|
|
|
+ /* Avoid recursion */
|
|
|
+ while (is_loop_device(f)) {
|
|
|
+ struct loop_device *l;
|
|
|
+
|
|
|
+ if (f->f_mapping->host->i_bdev == bdev)
|
|
|
+ return -EBADF;
|
|
|
+
|
|
|
+ l = f->f_mapping->host->i_bdev->bd_disk->private_data;
|
|
|
+ if (l->lo_state == Lo_unbound) {
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+ f = l->lo_backing_file;
|
|
|
+ }
|
|
|
+ if (!S_ISREG(inode->i_mode) && !S_ISBLK(inode->i_mode))
|
|
|
+ return -EINVAL;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
/*
|
|
|
* loop_change_fd switched the backing store of a loopback device to
|
|
|
* a new file. This is useful for operating system installers to free up
|
|
@@ -673,14 +703,15 @@ static int loop_change_fd(struct loop_device *lo, struct block_device *bdev,
|
|
|
if (!file)
|
|
|
goto out;
|
|
|
|
|
|
+ error = loop_validate_file(file, bdev);
|
|
|
+ if (error)
|
|
|
+ goto out_putf;
|
|
|
+
|
|
|
inode = file->f_mapping->host;
|
|
|
old_file = lo->lo_backing_file;
|
|
|
|
|
|
error = -EINVAL;
|
|
|
|
|
|
- if (!S_ISREG(inode->i_mode) && !S_ISBLK(inode->i_mode))
|
|
|
- goto out_putf;
|
|
|
-
|
|
|
/* size of the new backing store needs to be the same */
|
|
|
if (get_loop_size(lo, file) != get_loop_size(lo, old_file))
|
|
|
goto out_putf;
|
|
@@ -706,13 +737,6 @@ static int loop_change_fd(struct loop_device *lo, struct block_device *bdev,
|
|
|
return error;
|
|
|
}
|
|
|
|
|
|
-static inline int is_loop_device(struct file *file)
|
|
|
-{
|
|
|
- struct inode *i = file->f_mapping->host;
|
|
|
-
|
|
|
- return i && S_ISBLK(i->i_mode) && MAJOR(i->i_rdev) == LOOP_MAJOR;
|
|
|
-}
|
|
|
-
|
|
|
/* loop sysfs attributes */
|
|
|
|
|
|
static ssize_t loop_attr_show(struct device *dev, char *page,
|
|
@@ -878,7 +902,7 @@ static int loop_prepare_queue(struct loop_device *lo)
|
|
|
static int loop_set_fd(struct loop_device *lo, fmode_t mode,
|
|
|
struct block_device *bdev, unsigned int arg)
|
|
|
{
|
|
|
- struct file *file, *f;
|
|
|
+ struct file *file;
|
|
|
struct inode *inode;
|
|
|
struct address_space *mapping;
|
|
|
int lo_flags = 0;
|
|
@@ -897,29 +921,13 @@ static int loop_set_fd(struct loop_device *lo, fmode_t mode,
|
|
|
if (lo->lo_state != Lo_unbound)
|
|
|
goto out_putf;
|
|
|
|
|
|
- /* Avoid recursion */
|
|
|
- f = file;
|
|
|
- while (is_loop_device(f)) {
|
|
|
- struct loop_device *l;
|
|
|
-
|
|
|
- if (f->f_mapping->host->i_bdev == bdev)
|
|
|
- goto out_putf;
|
|
|
-
|
|
|
- l = f->f_mapping->host->i_bdev->bd_disk->private_data;
|
|
|
- if (l->lo_state == Lo_unbound) {
|
|
|
- error = -EINVAL;
|
|
|
- goto out_putf;
|
|
|
- }
|
|
|
- f = l->lo_backing_file;
|
|
|
- }
|
|
|
+ error = loop_validate_file(file, bdev);
|
|
|
+ if (error)
|
|
|
+ goto out_putf;
|
|
|
|
|
|
mapping = file->f_mapping;
|
|
|
inode = mapping->host;
|
|
|
|
|
|
- error = -EINVAL;
|
|
|
- if (!S_ISREG(inode->i_mode) && !S_ISBLK(inode->i_mode))
|
|
|
- goto out_putf;
|
|
|
-
|
|
|
if (!(file->f_mode & FMODE_WRITE) || !(mode & FMODE_WRITE) ||
|
|
|
!file->f_op->write_iter)
|
|
|
lo_flags |= LO_FLAGS_READ_ONLY;
|