Commit a16539ba authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] ramdisk fixes

- Remove the ramdisk special-case in fs-writeback.c - it will soon be
  unneeded.

- Fix rd_ioctl() to return -ENOTTY on invalid ioctl types, not -EINVAL.

- Make ramdisk Kconfig friendlier.
parent c8dcfa85
...@@ -321,12 +321,13 @@ config BLK_DEV_RAM ...@@ -321,12 +321,13 @@ config BLK_DEV_RAM
thus say N here. thus say N here.
config BLK_DEV_RAM_SIZE config BLK_DEV_RAM_SIZE
int "Default RAM disk size" int "Default RAM disk size (kbytes)"
depends on BLK_DEV_RAM depends on BLK_DEV_RAM
default "4096" default "4096"
help help
The default value is 4096. Only change this if you know what are The default value is 4096 kilobytes. Only change this if you know
you doing. If you are using IBM S/390, then set this to 8192. what are you doing. If you are using IBM S/390, then set this to
8192.
config BLK_DEV_INITRD config BLK_DEV_INITRD
bool "Initial RAM disk (initrd) support" bool "Initial RAM disk (initrd) support"
......
...@@ -57,6 +57,8 @@ ...@@ -57,6 +57,8 @@
#include <linux/buffer_head.h> /* for invalidate_bdev() */ #include <linux/buffer_head.h> /* for invalidate_bdev() */
#include <linux/backing-dev.h> #include <linux/backing-dev.h>
#include <linux/blkpg.h> #include <linux/blkpg.h>
#include <linux/writeback.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
/* The RAM disk size is now a parameter */ /* The RAM disk size is now a parameter */
...@@ -119,13 +121,39 @@ static int ramdisk_prepare_write(struct file *file, struct page *page, ...@@ -119,13 +121,39 @@ static int ramdisk_prepare_write(struct file *file, struct page *page,
kunmap_atomic(kaddr, KM_USER0); kunmap_atomic(kaddr, KM_USER0);
SetPageUptodate(page); SetPageUptodate(page);
} }
SetPageDirty(page);
return 0; return 0;
} }
static int ramdisk_commit_write(struct file *file, struct page *page, static int ramdisk_commit_write(struct file *file, struct page *page,
unsigned offset, unsigned to) unsigned offset, unsigned to)
{ {
set_page_dirty(page);
return 0;
}
/*
* ->writepage to the the blockdev's mapping has to redirty the page so that the
* VM doesn't go and steal it. We return WRITEPAGE_ACTIVATE so that the VM
* won't try to (pointlessly) write the page again for a while.
*
* Really, these pages should not be on the LRU at all.
*/
static int ramdisk_writepage(struct page *page, struct writeback_control *wbc)
{
redirty_page_for_writepage(wbc, page);
if (wbc->for_reclaim)
return WRITEPAGE_ACTIVATE;
unlock_page(page);
return 0;
}
/*
* ramdisk blockdev pages have their own ->set_page_dirty() because we don't
* want them to contribute to dirty memory accounting.
*/
static int ramdisk_set_page_dirty(struct page *page)
{
SetPageDirty(page);
return 0; return 0;
} }
...@@ -133,6 +161,8 @@ static struct address_space_operations ramdisk_aops = { ...@@ -133,6 +161,8 @@ static struct address_space_operations ramdisk_aops = {
.readpage = ramdisk_readpage, .readpage = ramdisk_readpage,
.prepare_write = ramdisk_prepare_write, .prepare_write = ramdisk_prepare_write,
.commit_write = ramdisk_commit_write, .commit_write = ramdisk_commit_write,
.writepage = ramdisk_writepage,
.set_page_dirty = ramdisk_set_page_dirty,
}; };
static int rd_blkdev_pagecache_IO(int rw, struct bio_vec *vec, sector_t sector, static int rd_blkdev_pagecache_IO(int rw, struct bio_vec *vec, sector_t sector,
...@@ -195,11 +225,11 @@ static int rd_blkdev_pagecache_IO(int rw, struct bio_vec *vec, sector_t sector, ...@@ -195,11 +225,11 @@ static int rd_blkdev_pagecache_IO(int rw, struct bio_vec *vec, sector_t sector,
if (rw == READ) { if (rw == READ) {
flush_dcache_page(vec->bv_page); flush_dcache_page(vec->bv_page);
} else { } else {
SetPageDirty(page); set_page_dirty(page);
} }
if (unlock) if (unlock)
unlock_page(page); unlock_page(page);
__free_page(page); put_page(page);
} while (size); } while (size);
out: out:
...@@ -251,7 +281,7 @@ static int rd_ioctl(struct inode *inode, struct file *file, ...@@ -251,7 +281,7 @@ static int rd_ioctl(struct inode *inode, struct file *file,
struct block_device *bdev = inode->i_bdev; struct block_device *bdev = inode->i_bdev;
if (cmd != BLKFLSBUF) if (cmd != BLKFLSBUF)
return -EINVAL; return -ENOTTY;
/* /*
* special: we want to release the ramdisk memory, it's not like with * special: we want to release the ramdisk memory, it's not like with
...@@ -268,6 +298,17 @@ static int rd_ioctl(struct inode *inode, struct file *file, ...@@ -268,6 +298,17 @@ static int rd_ioctl(struct inode *inode, struct file *file,
return error; return error;
} }
/*
* The backing_dev_info is shared between files which are backed by the ramdisk
* inode and by the ramdisk inode itself. This is a bit unfortunate because
* they really want separate semantics. The files *do* want full writeback
* and dirty-memory accounting treatment, whereas the ramdisk blockdev mapping
* wants neither.
*
* So we make things look like a regular blockdev and the cheat in various ways
* in the ramdisk inode's a_ops.
*/
static struct backing_dev_info rd_backing_dev_info = { static struct backing_dev_info rd_backing_dev_info = {
.ra_pages = 0, /* No readahead */ .ra_pages = 0, /* No readahead */
.memory_backed = 1, /* Does not contribute to dirty memory */ .memory_backed = 1, /* Does not contribute to dirty memory */
......
...@@ -310,11 +310,6 @@ sync_sb_inodes(struct super_block *sb, struct writeback_control *wbc) ...@@ -310,11 +310,6 @@ sync_sb_inodes(struct super_block *sb, struct writeback_control *wbc)
list_move(&inode->i_list, &sb->s_dirty); list_move(&inode->i_list, &sb->s_dirty);
continue; continue;
} }
/*
* Assume that all inodes on this superblock are memory
* backed. Skip the superblock.
*/
break;
} }
if (wbc->nonblocking && bdi_write_congested(bdi)) { if (wbc->nonblocking && bdi_write_congested(bdi)) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment