Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 9421502b authored by Joern Engel's avatar Joern Engel
Browse files

[LogFS] Fix bdev erases

Erases for block devices were always just emulated by writing 0xff.
Some time back the write was removed and only the page cache was
changed to 0xff.  Superficialy a good idea with two problems:
1. Touching the page cache isn't necessary either.
2. However, writing out 0xff _is_ necessary for the journal.  As the
   journal is scanned linearly, an old non-overwritten commit entry
   can be used on next mount and cause havoc.

This should fix both aspects.
parent 5c564c2a
Loading
Loading
Loading
Loading
+76 −12
Original line number Original line Diff line number Diff line
@@ -167,27 +167,91 @@ static void bdev_writeseg(struct super_block *sb, u64 ofs, size_t len)
	generic_unplug_device(bdev_get_queue(logfs_super(sb)->s_bdev));
	generic_unplug_device(bdev_get_queue(logfs_super(sb)->s_bdev));
}
}


static int bdev_erase(struct super_block *sb, loff_t to, size_t len)

static void erase_end_io(struct bio *bio, int err) 
{ 
	const int uptodate = test_bit(BIO_UPTODATE, &bio->bi_flags); 
	struct super_block *sb = bio->bi_private; 
	struct logfs_super *super = logfs_super(sb); 

	BUG_ON(!uptodate); /* FIXME: Retry io or write elsewhere */ 
	BUG_ON(err); 
	BUG_ON(bio->bi_vcnt == 0); 
	bio_put(bio); 
	if (atomic_dec_and_test(&super->s_pending_writes))
		wake_up(&wq); 
} 

static int do_erase(struct super_block *sb, u64 ofs, pgoff_t index,
		size_t nr_pages)
{
	struct logfs_super *super = logfs_super(sb);
	struct bio *bio;
	struct request_queue *q = bdev_get_queue(sb->s_bdev);
	unsigned int max_pages = queue_max_hw_sectors(q) >> (PAGE_SHIFT - 9);
	int i;

	bio = bio_alloc(GFP_NOFS, max_pages);
	BUG_ON(!bio); /* FIXME: handle this */

	for (i = 0; i < nr_pages; i++) {
		if (i >= max_pages) {
			/* Block layer cannot split bios :( */
			bio->bi_vcnt = i;
			bio->bi_idx = 0;
			bio->bi_size = i * PAGE_SIZE;
			bio->bi_bdev = super->s_bdev;
			bio->bi_sector = ofs >> 9;
			bio->bi_private = sb;
			bio->bi_end_io = erase_end_io;
			atomic_inc(&super->s_pending_writes);
			submit_bio(WRITE, bio);

			ofs += i * PAGE_SIZE;
			index += i;
			nr_pages -= i;
			i = 0;

			bio = bio_alloc(GFP_NOFS, max_pages);
			BUG_ON(!bio);
		}
		bio->bi_io_vec[i].bv_page = super->s_erase_page;
		bio->bi_io_vec[i].bv_len = PAGE_SIZE;
		bio->bi_io_vec[i].bv_offset = 0;
	}
	bio->bi_vcnt = nr_pages;
	bio->bi_idx = 0;
	bio->bi_size = nr_pages * PAGE_SIZE;
	bio->bi_bdev = super->s_bdev;
	bio->bi_sector = ofs >> 9;
	bio->bi_private = sb;
	bio->bi_end_io = erase_end_io;
	atomic_inc(&super->s_pending_writes);
	submit_bio(WRITE, bio);
	return 0;
}

static int bdev_erase(struct super_block *sb, loff_t to, size_t len,
		int ensure_write)
{
{
	struct logfs_super *super = logfs_super(sb);
	struct logfs_super *super = logfs_super(sb);
	struct address_space *mapping = super->s_mapping_inode->i_mapping;
	struct page *page;
	pgoff_t index = to >> PAGE_SHIFT;
	int i, nr_pages = len >> PAGE_SHIFT;


	BUG_ON(to & (PAGE_SIZE - 1));
	BUG_ON(to & (PAGE_SIZE - 1));
	BUG_ON(len & (PAGE_SIZE - 1));
	BUG_ON(len & (PAGE_SIZE - 1));


	if (logfs_super(sb)->s_flags & LOGFS_SB_FLAG_RO)
	if (super->s_flags & LOGFS_SB_FLAG_RO)
		return -EROFS;
		return -EROFS;


	for (i = 0; i < nr_pages; i++) {
	if (ensure_write) {
		page = find_get_page(mapping, index + i);
		/*
		if (page) {
		 * Object store doesn't care whether erases happen or not.
			memset(page_address(page), 0xFF, PAGE_SIZE);
		 * But for the journal they are required.  Otherwise a scan
			page_cache_release(page);
		 * can find an old commit entry and assume it is the current
		}
		 * one, travelling back in time.
		 */
		do_erase(sb, to, to >> PAGE_SHIFT, len >> PAGE_SHIFT);
	}
	}

	return 0;
	return 0;
}
}


+2 −1
Original line number Original line Diff line number Diff line
@@ -83,7 +83,8 @@ static int mtd_erase_mapping(struct super_block *sb, loff_t ofs, size_t len)
	return 0;
	return 0;
}
}


static int mtd_erase(struct super_block *sb, loff_t ofs, size_t len)
static int mtd_erase(struct super_block *sb, loff_t ofs, size_t len,
		int ensure_write)
{
{
	struct mtd_info *mtd = logfs_super(sb)->s_mtd;
	struct mtd_info *mtd = logfs_super(sb)->s_mtd;
	struct erase_info ei;
	struct erase_info ei;
+1 −1
Original line number Original line Diff line number Diff line
@@ -392,7 +392,7 @@ static int journal_erase_segment(struct logfs_area *area)
	u64 ofs;
	u64 ofs;
	int err;
	int err;


	err = logfs_erase_segment(sb, area->a_segno);
	err = logfs_erase_segment(sb, area->a_segno, 1);
	if (err)
	if (err)
		return err;
		return err;


+4 −2
Original line number Original line Diff line number Diff line
@@ -151,7 +151,8 @@ struct logfs_device_ops {
	int (*write_sb)(struct super_block *sb, struct page *page);
	int (*write_sb)(struct super_block *sb, struct page *page);
	int (*readpage)(void *_sb, struct page *page);
	int (*readpage)(void *_sb, struct page *page);
	void (*writeseg)(struct super_block *sb, u64 ofs, size_t len);
	void (*writeseg)(struct super_block *sb, u64 ofs, size_t len);
	int (*erase)(struct super_block *sb, loff_t ofs, size_t len);
	int (*erase)(struct super_block *sb, loff_t ofs, size_t len,
			int ensure_write);
	void (*sync)(struct super_block *sb);
	void (*sync)(struct super_block *sb);
	void (*put_device)(struct super_block *sb);
	void (*put_device)(struct super_block *sb);
};
};
@@ -327,6 +328,7 @@ struct logfs_super {
	u64	 s_feature_compat;
	u64	 s_feature_compat;
	u64	 s_feature_flags;
	u64	 s_feature_flags;
	u64	 s_sb_ofs[2];
	u64	 s_sb_ofs[2];
	struct page *s_erase_page;		/* for dev_bdev.c */
	/* alias.c fields */
	/* alias.c fields */
	struct btree_head32 s_segment_alias;	/* remapped segments */
	struct btree_head32 s_segment_alias;	/* remapped segments */
	int	 s_no_object_aliases;
	int	 s_no_object_aliases;
@@ -572,7 +574,7 @@ int get_page_reserve(struct inode *inode, struct page *page);
extern struct logfs_block_ops indirect_block_ops;
extern struct logfs_block_ops indirect_block_ops;


/* segment.c */
/* segment.c */
int logfs_erase_segment(struct super_block *sb, u32 ofs);
int logfs_erase_segment(struct super_block *sb, u32 ofs, int ensure_erase);
int wbuf_read(struct super_block *sb, u64 ofs, size_t len, void *buf);
int wbuf_read(struct super_block *sb, u64 ofs, size_t len, void *buf);
int logfs_segment_read(struct inode *inode, struct page *page, u64 ofs, u64 bix,
int logfs_segment_read(struct inode *inode, struct page *page, u64 ofs, u64 bix,
		level_t level);
		level_t level);
+3 −3
Original line number Original line Diff line number Diff line
@@ -25,14 +25,14 @@ static int logfs_mark_segment_bad(struct super_block *sb, u32 segno)
	return 0;
	return 0;
}
}


int logfs_erase_segment(struct super_block *sb, u32 segno)
int logfs_erase_segment(struct super_block *sb, u32 segno, int ensure_erase)
{
{
	struct logfs_super *super = logfs_super(sb);
	struct logfs_super *super = logfs_super(sb);


	super->s_gec++;
	super->s_gec++;


	return super->s_devops->erase(sb, (u64)segno << super->s_segshift,
	return super->s_devops->erase(sb, (u64)segno << super->s_segshift,
			super->s_segsize);
			super->s_segsize, ensure_erase);
}
}


static s64 logfs_get_free_bytes(struct logfs_area *area, size_t bytes)
static s64 logfs_get_free_bytes(struct logfs_area *area, size_t bytes)
@@ -798,7 +798,7 @@ static int ostore_erase_segment(struct logfs_area *area)
	u64 ofs;
	u64 ofs;
	int err;
	int err;


	err = logfs_erase_segment(sb, area->a_segno);
	err = logfs_erase_segment(sb, area->a_segno, 0);
	if (err)
	if (err)
		return err;
		return err;


Loading