staging: zram: fix zram locking
Currently init_lock only prevents concurrent execution of zram_init_device() and zram_reset_device() but not zram_make_request() nor sysfs store functions. This patch changes init_lock into a rw_semaphore. A write lock is taken by init, reset and store functions, a read lock is taken by zram_make_request(). Also, avoids to release the lock before calling __zram_reset_device() for cleaning after a failed init, thus preventing any concurrent task to see an inconsistent state of zram. Signed-off-by: Jerome Marchand <jmarchan@redhat.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
a1c821ae6c
commit
0900beae17
3 changed files with 44 additions and 26 deletions
|
@ -560,27 +560,34 @@ static int zram_make_request(struct request_queue *queue, struct bio *bio)
|
||||||
{
|
{
|
||||||
struct zram *zram = queue->queuedata;
|
struct zram *zram = queue->queuedata;
|
||||||
|
|
||||||
|
if (unlikely(!zram->init_done) && zram_init_device(zram))
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
down_read(&zram->init_lock);
|
||||||
|
if (unlikely(!zram->init_done))
|
||||||
|
goto error_unlock;
|
||||||
|
|
||||||
if (!valid_io_request(zram, bio)) {
|
if (!valid_io_request(zram, bio)) {
|
||||||
zram_stat64_inc(zram, &zram->stats.invalid_io);
|
zram_stat64_inc(zram, &zram->stats.invalid_io);
|
||||||
bio_io_error(bio);
|
goto error_unlock;
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (unlikely(!zram->init_done) && zram_init_device(zram)) {
|
|
||||||
bio_io_error(bio);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
__zram_make_request(zram, bio, bio_data_dir(bio));
|
__zram_make_request(zram, bio, bio_data_dir(bio));
|
||||||
|
up_read(&zram->init_lock);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
error_unlock:
|
||||||
|
up_read(&zram->init_lock);
|
||||||
|
error:
|
||||||
|
bio_io_error(bio);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void zram_reset_device(struct zram *zram)
|
void __zram_reset_device(struct zram *zram)
|
||||||
{
|
{
|
||||||
size_t index;
|
size_t index;
|
||||||
|
|
||||||
mutex_lock(&zram->init_lock);
|
|
||||||
zram->init_done = 0;
|
zram->init_done = 0;
|
||||||
|
|
||||||
/* Free various per-device buffers */
|
/* Free various per-device buffers */
|
||||||
|
@ -617,7 +624,13 @@ void zram_reset_device(struct zram *zram)
|
||||||
memset(&zram->stats, 0, sizeof(zram->stats));
|
memset(&zram->stats, 0, sizeof(zram->stats));
|
||||||
|
|
||||||
zram->disksize = 0;
|
zram->disksize = 0;
|
||||||
mutex_unlock(&zram->init_lock);
|
}
|
||||||
|
|
||||||
|
void zram_reset_device(struct zram *zram)
|
||||||
|
{
|
||||||
|
down_write(&zram->init_lock);
|
||||||
|
__zram_reset_device(zram);
|
||||||
|
up_write(&zram->init_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
int zram_init_device(struct zram *zram)
|
int zram_init_device(struct zram *zram)
|
||||||
|
@ -625,10 +638,10 @@ int zram_init_device(struct zram *zram)
|
||||||
int ret;
|
int ret;
|
||||||
size_t num_pages;
|
size_t num_pages;
|
||||||
|
|
||||||
mutex_lock(&zram->init_lock);
|
down_write(&zram->init_lock);
|
||||||
|
|
||||||
if (zram->init_done) {
|
if (zram->init_done) {
|
||||||
mutex_unlock(&zram->init_lock);
|
up_write(&zram->init_lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -671,15 +684,14 @@ int zram_init_device(struct zram *zram)
|
||||||
}
|
}
|
||||||
|
|
||||||
zram->init_done = 1;
|
zram->init_done = 1;
|
||||||
mutex_unlock(&zram->init_lock);
|
up_write(&zram->init_lock);
|
||||||
|
|
||||||
pr_debug("Initialization done!\n");
|
pr_debug("Initialization done!\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fail:
|
fail:
|
||||||
mutex_unlock(&zram->init_lock);
|
__zram_reset_device(zram);
|
||||||
zram_reset_device(zram);
|
up_write(&zram->init_lock);
|
||||||
|
|
||||||
pr_err("Initialization failed: err=%d\n", ret);
|
pr_err("Initialization failed: err=%d\n", ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -703,7 +715,7 @@ static int create_device(struct zram *zram, int device_id)
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
init_rwsem(&zram->lock);
|
init_rwsem(&zram->lock);
|
||||||
mutex_init(&zram->init_lock);
|
init_rwsem(&zram->init_lock);
|
||||||
spin_lock_init(&zram->stat64_lock);
|
spin_lock_init(&zram->stat64_lock);
|
||||||
|
|
||||||
zram->queue = blk_alloc_queue(GFP_KERNEL);
|
zram->queue = blk_alloc_queue(GFP_KERNEL);
|
||||||
|
|
|
@ -112,8 +112,8 @@ struct zram {
|
||||||
struct request_queue *queue;
|
struct request_queue *queue;
|
||||||
struct gendisk *disk;
|
struct gendisk *disk;
|
||||||
int init_done;
|
int init_done;
|
||||||
/* Prevent concurrent execution of device init and reset */
|
/* Prevent concurrent execution of device init, reset and R/W request */
|
||||||
struct mutex init_lock;
|
struct rw_semaphore init_lock;
|
||||||
/*
|
/*
|
||||||
* This is the limit on amount of *uncompressed* worth of data
|
* This is the limit on amount of *uncompressed* worth of data
|
||||||
* we can store in a disk.
|
* we can store in a disk.
|
||||||
|
@ -130,6 +130,6 @@ extern struct attribute_group zram_disk_attr_group;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
extern int zram_init_device(struct zram *zram);
|
extern int zram_init_device(struct zram *zram);
|
||||||
extern void zram_reset_device(struct zram *zram);
|
extern void __zram_reset_device(struct zram *zram);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -55,19 +55,23 @@ static ssize_t disksize_store(struct device *dev,
|
||||||
struct device_attribute *attr, const char *buf, size_t len)
|
struct device_attribute *attr, const char *buf, size_t len)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
u64 disksize;
|
||||||
struct zram *zram = dev_to_zram(dev);
|
struct zram *zram = dev_to_zram(dev);
|
||||||
|
|
||||||
|
ret = strict_strtoull(buf, 10, &disksize);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
down_write(&zram->init_lock);
|
||||||
if (zram->init_done) {
|
if (zram->init_done) {
|
||||||
|
up_write(&zram->init_lock);
|
||||||
pr_info("Cannot change disksize for initialized device\n");
|
pr_info("Cannot change disksize for initialized device\n");
|
||||||
return -EBUSY;
|
return -EBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = strict_strtoull(buf, 10, &zram->disksize);
|
zram->disksize = PAGE_ALIGN(disksize);
|
||||||
if (ret)
|
|
||||||
return ret;
|
|
||||||
|
|
||||||
zram->disksize = PAGE_ALIGN(zram->disksize);
|
|
||||||
set_capacity(zram->disk, zram->disksize >> SECTOR_SHIFT);
|
set_capacity(zram->disk, zram->disksize >> SECTOR_SHIFT);
|
||||||
|
up_write(&zram->init_lock);
|
||||||
|
|
||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
@ -106,8 +110,10 @@ static ssize_t reset_store(struct device *dev,
|
||||||
if (bdev)
|
if (bdev)
|
||||||
fsync_bdev(bdev);
|
fsync_bdev(bdev);
|
||||||
|
|
||||||
|
down_write(&zram->init_lock);
|
||||||
if (zram->init_done)
|
if (zram->init_done)
|
||||||
zram_reset_device(zram);
|
__zram_reset_device(zram);
|
||||||
|
up_write(&zram->init_lock);
|
||||||
|
|
||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue