2005-06-22 02:17:14 +02:00
|
|
|
/*
|
|
|
|
* bitmap.c two-level bitmap (C) Peter T. Breuer (ptb@ot.uc3m.es) 2003
|
|
|
|
*
|
|
|
|
* bitmap_create - sets up the bitmap structure
|
|
|
|
* bitmap_destroy - destroys the bitmap structure
|
|
|
|
*
|
|
|
|
* additions, Copyright (C) 2003-2004, Paul Clements, SteelEye Technology, Inc.:
|
|
|
|
* - added disk storage for bitmap
|
|
|
|
* - changes to allow various bitmap chunk sizes
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Still to do:
|
|
|
|
*
|
|
|
|
* flush after percent set rather than just time based. (maybe both).
|
|
|
|
* wait if count gets too high, wake when it drops to half.
|
|
|
|
*/
|
|
|
|
|
2009-03-31 05:33:13 +02:00
|
|
|
#include <linux/blkdev.h>
|
2005-06-22 02:17:14 +02:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/timer.h>
|
|
|
|
#include <linux/sched.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/file.h>
|
|
|
|
#include <linux/mount.h>
|
|
|
|
#include <linux/buffer_head.h>
|
2009-03-31 05:33:13 +02:00
|
|
|
#include "md.h"
|
2009-03-31 05:27:03 +02:00
|
|
|
#include "bitmap.h"
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* debug macros */
|
|
|
|
|
|
|
|
#define DEBUG 0
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
/* these are for debugging purposes only! */
|
|
|
|
|
|
|
|
/* define one and only one of these */
|
|
|
|
#define INJECT_FAULTS_1 0 /* cause bitmap_alloc_page to fail always */
|
|
|
|
#define INJECT_FAULTS_2 0 /* cause bitmap file to be kicked when first bit set*/
|
|
|
|
#define INJECT_FAULTS_3 0 /* treat bitmap file as kicked at init time */
|
|
|
|
#define INJECT_FAULTS_4 0 /* undef */
|
|
|
|
#define INJECT_FAULTS_5 0 /* undef */
|
|
|
|
#define INJECT_FAULTS_6 0
|
|
|
|
|
|
|
|
/* if these are defined, the driver will fail! debug only */
|
|
|
|
#define INJECT_FATAL_FAULT_1 0 /* fail kmalloc, causing bitmap_create to fail */
|
|
|
|
#define INJECT_FATAL_FAULT_2 0 /* undef */
|
|
|
|
#define INJECT_FATAL_FAULT_3 0 /* undef */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//#define DPRINTK PRINTK /* set this NULL to avoid verbose debug output */
|
|
|
|
#define DPRINTK(x...) do { } while(0)
|
|
|
|
|
|
|
|
#ifndef PRINTK
|
|
|
|
# if DEBUG > 0
|
|
|
|
# define PRINTK(x...) printk(KERN_DEBUG x)
|
|
|
|
# else
|
|
|
|
# define PRINTK(x...)
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static inline char * bmname(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
return bitmap->mddev ? mdname(bitmap->mddev) : "mdX";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* just a placeholder - calls kmalloc for bitmap pages
|
|
|
|
*/
|
|
|
|
static unsigned char *bitmap_alloc_page(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
unsigned char *page;
|
|
|
|
|
2005-07-27 20:45:17 +02:00
|
|
|
#ifdef INJECT_FAULTS_1
|
2005-06-22 02:17:14 +02:00
|
|
|
page = NULL;
|
|
|
|
#else
|
|
|
|
page = kmalloc(PAGE_SIZE, GFP_NOIO);
|
|
|
|
#endif
|
|
|
|
if (!page)
|
|
|
|
printk("%s: bitmap_alloc_page FAILED\n", bmname(bitmap));
|
|
|
|
else
|
2005-06-22 02:17:27 +02:00
|
|
|
PRINTK("%s: bitmap_alloc_page: allocated page at %p\n",
|
2005-06-22 02:17:14 +02:00
|
|
|
bmname(bitmap), page);
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* for now just a placeholder -- just calls kfree for bitmap pages
|
|
|
|
*/
|
|
|
|
static void bitmap_free_page(struct bitmap *bitmap, unsigned char *page)
|
|
|
|
{
|
|
|
|
PRINTK("%s: bitmap_free_page: free page %p\n", bmname(bitmap), page);
|
|
|
|
kfree(page);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* check a page and, if necessary, allocate it (or hijack it if the alloc fails)
|
|
|
|
*
|
|
|
|
* 1) check to see if this page is allocated, if it's not then try to alloc
|
|
|
|
* 2) if the alloc fails, set the page's hijacked flag so we'll use the
|
|
|
|
* page pointer directly as a counter
|
|
|
|
*
|
|
|
|
* if we find our page, we increment the page's refcount so that it stays
|
|
|
|
* allocated while we're using it
|
|
|
|
*/
|
|
|
|
static int bitmap_checkpage(struct bitmap *bitmap, unsigned long page, int create)
|
2009-09-23 10:06:44 +02:00
|
|
|
__releases(bitmap->lock)
|
|
|
|
__acquires(bitmap->lock)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
unsigned char *mappage;
|
|
|
|
|
|
|
|
if (page >= bitmap->pages) {
|
2009-03-31 05:27:02 +02:00
|
|
|
/* This can happen if bitmap_start_sync goes beyond
|
|
|
|
* End-of-device while looking for a whole page.
|
|
|
|
* It is harmless.
|
|
|
|
*/
|
2005-06-22 02:17:14 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (bitmap->bp[page].hijacked) /* it's hijacked, don't try to alloc */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (bitmap->bp[page].map) /* page is already allocated, just return */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!create)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
|
|
|
|
|
|
|
/* this page has not been allocated yet */
|
|
|
|
|
|
|
|
if ((mappage = bitmap_alloc_page(bitmap)) == NULL) {
|
|
|
|
PRINTK("%s: bitmap map page allocation failed, hijacking\n",
|
|
|
|
bmname(bitmap));
|
|
|
|
/* failed - set the hijacked flag so that we can use the
|
|
|
|
* pointer as a counter */
|
|
|
|
spin_lock_irq(&bitmap->lock);
|
|
|
|
if (!bitmap->bp[page].map)
|
|
|
|
bitmap->bp[page].hijacked = 1;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* got a page */
|
|
|
|
|
|
|
|
spin_lock_irq(&bitmap->lock);
|
|
|
|
|
|
|
|
/* recheck the page */
|
|
|
|
|
|
|
|
if (bitmap->bp[page].map || bitmap->bp[page].hijacked) {
|
|
|
|
/* somebody beat us to getting the page */
|
|
|
|
bitmap_free_page(bitmap, mappage);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no page was in place and we have one, so install it */
|
|
|
|
|
|
|
|
memset(mappage, 0, PAGE_SIZE);
|
|
|
|
bitmap->bp[page].map = mappage;
|
|
|
|
bitmap->missing_pages--;
|
|
|
|
out:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* if page is completely empty, put it back on the free list, or dealloc it */
|
|
|
|
/* if page was hijacked, unmark the flag so it might get alloced next time */
|
|
|
|
/* Note: lock should be held when calling this */
|
2006-01-14 22:20:43 +01:00
|
|
|
static void bitmap_checkfree(struct bitmap *bitmap, unsigned long page)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
if (bitmap->bp[page].count) /* page is still busy */
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* page is no longer in use, it can be released */
|
|
|
|
|
|
|
|
if (bitmap->bp[page].hijacked) { /* page was hijacked, undo this now */
|
|
|
|
bitmap->bp[page].hijacked = 0;
|
|
|
|
bitmap->bp[page].map = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* normal case, free the page */
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* actually ... let's not. We will probably need the page again exactly when
|
|
|
|
* memory is tight and we are flusing to disk
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
ptr = bitmap->bp[page].map;
|
|
|
|
bitmap->bp[page].map = NULL;
|
|
|
|
bitmap->missing_pages++;
|
|
|
|
bitmap_free_page(bitmap, ptr);
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bitmap file handling - read and write the bitmap file and its superblock
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* basic page I/O operations
|
|
|
|
*/
|
|
|
|
|
2005-06-22 02:17:27 +02:00
|
|
|
/* IO operations when bitmap is stored near all superblocks */
|
2009-12-14 02:49:54 +01:00
|
|
|
static struct page *read_sb_page(mddev_t *mddev, loff_t offset,
|
2008-12-19 06:25:01 +01:00
|
|
|
struct page *page,
|
|
|
|
unsigned long index, int size)
|
2005-06-22 02:17:27 +02:00
|
|
|
{
|
|
|
|
/* choose a good rdev and read the page from there */
|
|
|
|
|
|
|
|
mdk_rdev_t *rdev;
|
|
|
|
sector_t target;
|
|
|
|
|
2008-12-19 06:25:01 +01:00
|
|
|
if (!page)
|
|
|
|
page = alloc_page(GFP_KERNEL);
|
2005-06-22 02:17:27 +02:00
|
|
|
if (!page)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
2009-01-08 22:31:08 +01:00
|
|
|
list_for_each_entry(rdev, &mddev->disks, same_set) {
|
2005-11-09 06:39:31 +01:00
|
|
|
if (! test_bit(In_sync, &rdev->flags)
|
|
|
|
|| test_bit(Faulty, &rdev->flags))
|
2005-09-10 01:23:52 +02:00
|
|
|
continue;
|
|
|
|
|
2008-07-11 14:02:23 +02:00
|
|
|
target = rdev->sb_start + offset + index * (PAGE_SIZE/512);
|
2005-06-22 02:17:27 +02:00
|
|
|
|
2008-12-19 06:25:01 +01:00
|
|
|
if (sync_page_io(rdev->bdev, target,
|
2009-05-22 23:17:49 +02:00
|
|
|
roundup(size, bdev_logical_block_size(rdev->bdev)),
|
2008-12-19 06:25:01 +01:00
|
|
|
page, READ)) {
|
2005-09-10 01:23:52 +02:00
|
|
|
page->index = index;
|
2006-06-26 09:27:49 +02:00
|
|
|
attach_page_buffers(page, NULL); /* so that free_buffer will
|
|
|
|
* quietly no-op */
|
2005-09-10 01:23:52 +02:00
|
|
|
return page;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ERR_PTR(-EIO);
|
2005-06-22 02:17:27 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-09-01 04:48:13 +02:00
|
|
|
static mdk_rdev_t *next_active_rdev(mdk_rdev_t *rdev, mddev_t *mddev)
|
|
|
|
{
|
|
|
|
/* Iterate the disks of an mddev, using rcu to protect access to the
|
|
|
|
* linked list, and raising the refcount of devices we return to ensure
|
|
|
|
* they don't disappear while in use.
|
|
|
|
* As devices are only added or removed when raid_disk is < 0 and
|
|
|
|
* nr_pending is 0 and In_sync is clear, the entries we return will
|
|
|
|
* still be in the same position on the list when we re-enter
|
|
|
|
* list_for_each_continue_rcu.
|
|
|
|
*/
|
|
|
|
struct list_head *pos;
|
|
|
|
rcu_read_lock();
|
|
|
|
if (rdev == NULL)
|
|
|
|
/* start at the beginning */
|
|
|
|
pos = &mddev->disks;
|
|
|
|
else {
|
|
|
|
/* release the previous rdev and start from there. */
|
|
|
|
rdev_dec_pending(rdev, mddev);
|
|
|
|
pos = &rdev->same_set;
|
|
|
|
}
|
|
|
|
list_for_each_continue_rcu(pos, &mddev->disks) {
|
|
|
|
rdev = list_entry(pos, mdk_rdev_t, same_set);
|
|
|
|
if (rdev->raid_disk >= 0 &&
|
|
|
|
!test_bit(Faulty, &rdev->flags)) {
|
|
|
|
/* this is a usable devices */
|
|
|
|
atomic_inc(&rdev->nr_pending);
|
|
|
|
rcu_read_unlock();
|
|
|
|
return rdev;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-05-23 22:58:10 +02:00
|
|
|
static int write_sb_page(struct bitmap *bitmap, struct page *page, int wait)
|
2005-06-22 02:17:27 +02:00
|
|
|
{
|
2008-09-01 04:48:13 +02:00
|
|
|
mdk_rdev_t *rdev = NULL;
|
2007-05-23 22:58:10 +02:00
|
|
|
mddev_t *mddev = bitmap->mddev;
|
2005-06-22 02:17:27 +02:00
|
|
|
|
2008-09-01 04:48:13 +02:00
|
|
|
while ((rdev = next_active_rdev(rdev, mddev)) != NULL) {
|
2007-05-23 22:58:10 +02:00
|
|
|
int size = PAGE_SIZE;
|
2009-12-14 02:49:54 +01:00
|
|
|
loff_t offset = mddev->bitmap_info.offset;
|
2007-05-23 22:58:10 +02:00
|
|
|
if (page->index == bitmap->file_pages-1)
|
|
|
|
size = roundup(bitmap->last_page_size,
|
2009-05-22 23:17:49 +02:00
|
|
|
bdev_logical_block_size(rdev->bdev));
|
2007-07-17 13:06:12 +02:00
|
|
|
/* Just make sure we aren't corrupting data or
|
|
|
|
* metadata
|
|
|
|
*/
|
2009-12-14 02:49:54 +01:00
|
|
|
if (mddev->external) {
|
|
|
|
/* Bitmap could be anywhere. */
|
|
|
|
if (rdev->sb_start + offset + (page->index *(PAGE_SIZE/512)) >
|
|
|
|
rdev->data_offset &&
|
|
|
|
rdev->sb_start + offset <
|
|
|
|
rdev->data_offset + mddev->dev_sectors +
|
|
|
|
(PAGE_SIZE/512))
|
|
|
|
goto bad_alignment;
|
|
|
|
} else if (offset < 0) {
|
2007-07-17 13:06:12 +02:00
|
|
|
/* DATA BITMAP METADATA */
|
2009-12-14 02:49:53 +01:00
|
|
|
if (offset
|
2007-10-23 05:45:11 +02:00
|
|
|
+ (long)(page->index * (PAGE_SIZE/512))
|
2007-07-17 13:06:12 +02:00
|
|
|
+ size/512 > 0)
|
|
|
|
/* bitmap runs in to metadata */
|
2008-07-21 09:05:25 +02:00
|
|
|
goto bad_alignment;
|
2009-03-31 05:33:13 +02:00
|
|
|
if (rdev->data_offset + mddev->dev_sectors
|
2009-12-14 02:49:53 +01:00
|
|
|
> rdev->sb_start + offset)
|
2007-07-17 13:06:12 +02:00
|
|
|
/* data runs in to bitmap */
|
2008-07-21 09:05:25 +02:00
|
|
|
goto bad_alignment;
|
2008-07-11 14:02:23 +02:00
|
|
|
} else if (rdev->sb_start < rdev->data_offset) {
|
2007-07-17 13:06:12 +02:00
|
|
|
/* METADATA BITMAP DATA */
|
2008-07-11 14:02:23 +02:00
|
|
|
if (rdev->sb_start
|
2009-12-14 02:49:53 +01:00
|
|
|
+ offset
|
2007-07-17 13:06:12 +02:00
|
|
|
+ page->index*(PAGE_SIZE/512) + size/512
|
|
|
|
> rdev->data_offset)
|
|
|
|
/* bitmap runs in to data */
|
2008-07-21 09:05:25 +02:00
|
|
|
goto bad_alignment;
|
2007-07-17 13:06:12 +02:00
|
|
|
} else {
|
|
|
|
/* DATA METADATA BITMAP - no problems */
|
|
|
|
}
|
2005-06-22 02:17:27 +02:00
|
|
|
md_super_write(mddev, rdev,
|
2009-12-14 02:49:53 +01:00
|
|
|
rdev->sb_start + offset
|
2005-06-22 02:17:27 +02:00
|
|
|
+ page->index * (PAGE_SIZE/512),
|
2007-05-23 22:58:10 +02:00
|
|
|
size,
|
2005-06-22 02:17:27 +02:00
|
|
|
page);
|
2008-09-01 04:48:13 +02:00
|
|
|
}
|
2005-06-22 02:17:27 +02:00
|
|
|
|
|
|
|
if (wait)
|
[PATCH] md: support BIO_RW_BARRIER for md/raid1
We can only accept BARRIER requests if all slaves handle
barriers, and that can, of course, change with time....
So we keep track of whether the whole array seems safe for barriers,
and also whether each individual rdev handles barriers.
We initially assumes barriers are OK.
When writing the superblock we try a barrier, and if that fails, we flag
things for no-barriers. This will usually clear the flags fairly quickly.
If writing the superblock finds that BIO_RW_BARRIER is -ENOTSUPP, we need to
resubmit, so introduce function "md_super_wait" which waits for requests to
finish, and retries ENOTSUPP requests without the barrier flag.
When writing the real raid1, write requests which were BIO_RW_BARRIER but
which aresn't supported need to be retried. So raid1d is enhanced to do this,
and when any bio write completes (i.e. no retry needed) we remove it from the
r1bio, so that devices needing retry are easy to find.
We should hardly ever get -ENOTSUPP errors when writing data to the raid.
It should only happen if:
1/ the device used to support BARRIER, but now doesn't. Few devices
change like this, though raid1 can!
or
2/ the array has no persistent superblock, so there was no opportunity to
pre-test for barriers when writing the superblock.
Signed-off-by: Neil Brown <neilb@cse.unsw.edu.au>
Signed-off-by: Neil Brown <neilb@suse.de>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-11-09 06:39:34 +01:00
|
|
|
md_super_wait(mddev);
|
2005-06-22 02:17:27 +02:00
|
|
|
return 0;
|
2008-07-21 09:05:25 +02:00
|
|
|
|
|
|
|
bad_alignment:
|
|
|
|
return -EINVAL;
|
2005-06-22 02:17:27 +02:00
|
|
|
}
|
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
static void bitmap_file_kick(struct bitmap *bitmap);
|
2005-06-22 02:17:14 +02:00
|
|
|
/*
|
2005-06-22 02:17:27 +02:00
|
|
|
* write out a page to a file
|
2005-06-22 02:17:14 +02:00
|
|
|
*/
|
2007-07-17 13:06:13 +02:00
|
|
|
static void write_page(struct bitmap *bitmap, struct page *page, int wait)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2006-06-26 09:27:48 +02:00
|
|
|
struct buffer_head *bh;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2007-07-17 13:06:12 +02:00
|
|
|
if (bitmap->file == NULL) {
|
|
|
|
switch (write_sb_page(bitmap, page, wait)) {
|
|
|
|
case -EINVAL:
|
|
|
|
bitmap->flags |= BITMAP_WRITE_ERROR;
|
|
|
|
}
|
2007-07-17 13:06:13 +02:00
|
|
|
} else {
|
2005-06-22 02:17:27 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
bh = page_buffers(page);
|
2006-01-06 09:20:45 +01:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
while (bh && bh->b_blocknr) {
|
|
|
|
atomic_inc(&bitmap->pending_writes);
|
|
|
|
set_buffer_locked(bh);
|
|
|
|
set_buffer_mapped(bh);
|
|
|
|
submit_bh(WRITE, bh);
|
|
|
|
bh = bh->b_this_page;
|
|
|
|
}
|
2006-06-26 09:27:48 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
if (wait) {
|
|
|
|
wait_event(bitmap->write_wait,
|
|
|
|
atomic_read(&bitmap->pending_writes)==0);
|
|
|
|
}
|
2005-06-22 02:17:29 +02:00
|
|
|
}
|
2007-07-17 13:06:13 +02:00
|
|
|
if (bitmap->flags & BITMAP_WRITE_ERROR)
|
|
|
|
bitmap_file_kick(bitmap);
|
2006-06-26 09:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void end_bitmap_write(struct buffer_head *bh, int uptodate)
|
|
|
|
{
|
|
|
|
struct bitmap *bitmap = bh->b_private;
|
|
|
|
unsigned long flags;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
if (!uptodate) {
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
bitmap->flags |= BITMAP_WRITE_ERROR;
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2006-06-26 09:27:48 +02:00
|
|
|
if (atomic_dec_and_test(&bitmap->pending_writes))
|
|
|
|
wake_up(&bitmap->write_wait);
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
/* copied from buffer.c */
|
|
|
|
static void
|
|
|
|
__clear_page_buffers(struct page *page)
|
|
|
|
{
|
|
|
|
ClearPagePrivate(page);
|
|
|
|
set_page_private(page, 0);
|
|
|
|
page_cache_release(page);
|
|
|
|
}
|
|
|
|
static void free_buffers(struct page *page)
|
|
|
|
{
|
|
|
|
struct buffer_head *bh = page_buffers(page);
|
2005-06-22 02:17:21 +02:00
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
while (bh) {
|
|
|
|
struct buffer_head *next = bh->b_this_page;
|
|
|
|
free_buffer_head(bh);
|
|
|
|
bh = next;
|
2005-06-22 02:17:21 +02:00
|
|
|
}
|
2006-06-26 09:27:48 +02:00
|
|
|
__clear_page_buffers(page);
|
|
|
|
put_page(page);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
/* read a page from a file.
|
|
|
|
* We both read the page, and attach buffers to the page to record the
|
|
|
|
* address of each block (using bmap). These addresses will be used
|
|
|
|
* to write the block later, completely bypassing the filesystem.
|
|
|
|
* This usage is similar to how swap files are handled, and allows us
|
|
|
|
* to write to a file with no concerns of memory allocation failing.
|
|
|
|
*/
|
2005-06-22 02:17:14 +02:00
|
|
|
static struct page *read_page(struct file *file, unsigned long index,
|
2006-06-26 09:27:48 +02:00
|
|
|
struct bitmap *bitmap,
|
|
|
|
unsigned long count)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
struct page *page = NULL;
|
2006-12-08 11:37:19 +01:00
|
|
|
struct inode *inode = file->f_path.dentry->d_inode;
|
2006-06-26 09:27:48 +02:00
|
|
|
struct buffer_head *bh;
|
|
|
|
sector_t block;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-01-06 09:20:31 +01:00
|
|
|
PRINTK("read bitmap file (%dB @ %Lu)\n", (int)PAGE_SIZE,
|
|
|
|
(unsigned long long)index << PAGE_SHIFT);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
page = alloc_page(GFP_KERNEL);
|
|
|
|
if (!page)
|
|
|
|
page = ERR_PTR(-ENOMEM);
|
2005-06-22 02:17:14 +02:00
|
|
|
if (IS_ERR(page))
|
|
|
|
goto out;
|
2006-06-26 09:27:48 +02:00
|
|
|
|
|
|
|
bh = alloc_page_buffers(page, 1<<inode->i_blkbits, 0);
|
|
|
|
if (!bh) {
|
2006-01-06 09:20:31 +01:00
|
|
|
put_page(page);
|
2006-06-26 09:27:48 +02:00
|
|
|
page = ERR_PTR(-ENOMEM);
|
2005-06-22 02:17:14 +02:00
|
|
|
goto out;
|
|
|
|
}
|
2006-06-26 09:27:48 +02:00
|
|
|
attach_page_buffers(page, bh);
|
|
|
|
block = index << (PAGE_SHIFT - inode->i_blkbits);
|
|
|
|
while (bh) {
|
|
|
|
if (count == 0)
|
|
|
|
bh->b_blocknr = 0;
|
|
|
|
else {
|
|
|
|
bh->b_blocknr = bmap(inode, block);
|
|
|
|
if (bh->b_blocknr == 0) {
|
|
|
|
/* Cannot use this file! */
|
|
|
|
free_buffers(page);
|
|
|
|
page = ERR_PTR(-EINVAL);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
bh->b_bdev = inode->i_sb->s_bdev;
|
|
|
|
if (count < (1<<inode->i_blkbits))
|
|
|
|
count = 0;
|
|
|
|
else
|
|
|
|
count -= (1<<inode->i_blkbits);
|
|
|
|
|
|
|
|
bh->b_end_io = end_bitmap_write;
|
|
|
|
bh->b_private = bitmap;
|
2006-06-26 09:27:49 +02:00
|
|
|
atomic_inc(&bitmap->pending_writes);
|
|
|
|
set_buffer_locked(bh);
|
|
|
|
set_buffer_mapped(bh);
|
|
|
|
submit_bh(READ, bh);
|
2006-06-26 09:27:48 +02:00
|
|
|
}
|
|
|
|
block++;
|
|
|
|
bh = bh->b_this_page;
|
|
|
|
}
|
|
|
|
page->index = index;
|
2006-06-26 09:27:49 +02:00
|
|
|
|
|
|
|
wait_event(bitmap->write_wait,
|
|
|
|
atomic_read(&bitmap->pending_writes)==0);
|
|
|
|
if (bitmap->flags & BITMAP_WRITE_ERROR) {
|
|
|
|
free_buffers(page);
|
|
|
|
page = ERR_PTR(-EIO);
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
out:
|
|
|
|
if (IS_ERR(page))
|
|
|
|
printk(KERN_ALERT "md: bitmap read error: (%dB @ %Lu): %ld\n",
|
2006-01-06 09:20:31 +01:00
|
|
|
(int)PAGE_SIZE,
|
|
|
|
(unsigned long long)index << PAGE_SHIFT,
|
2005-06-22 02:17:14 +02:00
|
|
|
PTR_ERR(page));
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bitmap file superblock operations
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* update the event counter and sync the superblock to disk */
|
2007-07-17 13:06:13 +02:00
|
|
|
void bitmap_update_sb(struct bitmap *bitmap)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
bitmap_super_t *sb;
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
if (!bitmap || !bitmap->mddev) /* no bitmap for this array */
|
2007-07-17 13:06:13 +02:00
|
|
|
return;
|
2009-12-14 02:49:56 +01:00
|
|
|
if (bitmap->mddev->bitmap_info.external)
|
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
if (!bitmap->sb_page) { /* no superblock */
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2007-07-17 13:06:13 +02:00
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2010-03-08 06:02:40 +01:00
|
|
|
sb = kmap_atomic(bitmap->sb_page, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
sb->events = cpu_to_le64(bitmap->mddev->events);
|
2008-06-28 00:31:22 +02:00
|
|
|
if (bitmap->mddev->events < bitmap->events_cleared) {
|
|
|
|
/* rocking back to read-only */
|
|
|
|
bitmap->events_cleared = bitmap->mddev->events;
|
|
|
|
sb->events_cleared = cpu_to_le64(bitmap->events_cleared);
|
|
|
|
}
|
2009-12-14 02:49:55 +01:00
|
|
|
/* Just in case these have been changed via sysfs: */
|
|
|
|
sb->daemon_sleep = cpu_to_le32(bitmap->mddev->bitmap_info.daemon_sleep/HZ);
|
|
|
|
sb->write_behind = cpu_to_le32(bitmap->mddev->bitmap_info.max_write_behind);
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(sb, KM_USER0);
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, bitmap->sb_page, 1);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* print out the bitmap file superblock */
|
|
|
|
void bitmap_print_sb(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
bitmap_super_t *sb;
|
|
|
|
|
|
|
|
if (!bitmap || !bitmap->sb_page)
|
|
|
|
return;
|
2010-03-08 06:02:40 +01:00
|
|
|
sb = kmap_atomic(bitmap->sb_page, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
printk(KERN_DEBUG "%s: bitmap file superblock:\n", bmname(bitmap));
|
2005-06-22 02:17:20 +02:00
|
|
|
printk(KERN_DEBUG " magic: %08x\n", le32_to_cpu(sb->magic));
|
|
|
|
printk(KERN_DEBUG " version: %d\n", le32_to_cpu(sb->version));
|
|
|
|
printk(KERN_DEBUG " uuid: %08x.%08x.%08x.%08x\n",
|
2005-06-22 02:17:14 +02:00
|
|
|
*(__u32 *)(sb->uuid+0),
|
|
|
|
*(__u32 *)(sb->uuid+4),
|
|
|
|
*(__u32 *)(sb->uuid+8),
|
|
|
|
*(__u32 *)(sb->uuid+12));
|
2005-06-22 02:17:20 +02:00
|
|
|
printk(KERN_DEBUG " events: %llu\n",
|
2005-06-22 02:17:14 +02:00
|
|
|
(unsigned long long) le64_to_cpu(sb->events));
|
2005-06-22 02:17:20 +02:00
|
|
|
printk(KERN_DEBUG "events cleared: %llu\n",
|
2005-06-22 02:17:14 +02:00
|
|
|
(unsigned long long) le64_to_cpu(sb->events_cleared));
|
2005-06-22 02:17:20 +02:00
|
|
|
printk(KERN_DEBUG " state: %08x\n", le32_to_cpu(sb->state));
|
|
|
|
printk(KERN_DEBUG " chunksize: %d B\n", le32_to_cpu(sb->chunksize));
|
|
|
|
printk(KERN_DEBUG " daemon sleep: %ds\n", le32_to_cpu(sb->daemon_sleep));
|
|
|
|
printk(KERN_DEBUG " sync size: %llu KB\n",
|
|
|
|
(unsigned long long)le64_to_cpu(sb->sync_size)/2);
|
2005-09-10 01:23:47 +02:00
|
|
|
printk(KERN_DEBUG "max write behind: %d\n", le32_to_cpu(sb->write_behind));
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(sb, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* read the superblock from the bitmap file and initialize some bitmap fields */
|
|
|
|
static int bitmap_read_sb(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
char *reason = NULL;
|
|
|
|
bitmap_super_t *sb;
|
2005-09-10 01:23:47 +02:00
|
|
|
unsigned long chunksize, daemon_sleep, write_behind;
|
2005-06-22 02:17:14 +02:00
|
|
|
unsigned long long events;
|
|
|
|
int err = -EINVAL;
|
|
|
|
|
|
|
|
/* page 0 is the superblock, read it... */
|
2007-01-26 09:57:03 +01:00
|
|
|
if (bitmap->file) {
|
|
|
|
loff_t isize = i_size_read(bitmap->file->f_mapping->host);
|
|
|
|
int bytes = isize > PAGE_SIZE ? PAGE_SIZE : isize;
|
|
|
|
|
|
|
|
bitmap->sb_page = read_page(bitmap->file, 0, bitmap, bytes);
|
|
|
|
} else {
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->sb_page = read_sb_page(bitmap->mddev,
|
|
|
|
bitmap->mddev->bitmap_info.offset,
|
2008-12-19 06:25:01 +01:00
|
|
|
NULL,
|
|
|
|
0, sizeof(bitmap_super_t));
|
2005-06-22 02:17:27 +02:00
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
if (IS_ERR(bitmap->sb_page)) {
|
|
|
|
err = PTR_ERR(bitmap->sb_page);
|
|
|
|
bitmap->sb_page = NULL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2010-03-08 06:02:40 +01:00
|
|
|
sb = kmap_atomic(bitmap->sb_page, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
chunksize = le32_to_cpu(sb->chunksize);
|
2009-12-14 02:49:53 +01:00
|
|
|
daemon_sleep = le32_to_cpu(sb->daemon_sleep) * HZ;
|
2005-09-10 01:23:47 +02:00
|
|
|
write_behind = le32_to_cpu(sb->write_behind);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* verify that the bitmap-specific fields are valid */
|
|
|
|
if (sb->magic != cpu_to_le32(BITMAP_MAGIC))
|
|
|
|
reason = "bad magic";
|
2005-11-09 06:39:32 +01:00
|
|
|
else if (le32_to_cpu(sb->version) < BITMAP_MAJOR_LO ||
|
|
|
|
le32_to_cpu(sb->version) > BITMAP_MAJOR_HI)
|
2005-06-22 02:17:14 +02:00
|
|
|
reason = "unrecognized superblock version";
|
2009-03-31 05:27:02 +02:00
|
|
|
else if (chunksize < 512)
|
2006-01-06 09:20:39 +01:00
|
|
|
reason = "bitmap chunksize too small";
|
2005-06-22 02:17:14 +02:00
|
|
|
else if ((1 << ffz(~chunksize)) != chunksize)
|
|
|
|
reason = "bitmap chunksize not a power of 2";
|
2009-12-14 02:49:53 +01:00
|
|
|
else if (daemon_sleep < 1 || daemon_sleep > MAX_SCHEDULE_TIMEOUT)
|
2006-01-06 09:20:39 +01:00
|
|
|
reason = "daemon sleep period out of range";
|
2005-09-10 01:23:47 +02:00
|
|
|
else if (write_behind > COUNTER_MAX)
|
|
|
|
reason = "write-behind limit out of range (0 - 16383)";
|
2005-06-22 02:17:14 +02:00
|
|
|
if (reason) {
|
|
|
|
printk(KERN_INFO "%s: invalid bitmap file superblock: %s\n",
|
|
|
|
bmname(bitmap), reason);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* keep the array size field of the bitmap superblock up to date */
|
|
|
|
sb->sync_size = cpu_to_le64(bitmap->mddev->resync_max_sectors);
|
|
|
|
|
|
|
|
if (!bitmap->mddev->persistent)
|
|
|
|
goto success;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* if we have a persistent array superblock, compare the
|
|
|
|
* bitmap's UUID and event counter to the mddev's
|
|
|
|
*/
|
|
|
|
if (memcmp(sb->uuid, bitmap->mddev->uuid, 16)) {
|
|
|
|
printk(KERN_INFO "%s: bitmap superblock UUID mismatch\n",
|
|
|
|
bmname(bitmap));
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
events = le64_to_cpu(sb->events);
|
|
|
|
if (events < bitmap->mddev->events) {
|
|
|
|
printk(KERN_INFO "%s: bitmap file is out of date (%llu < %llu) "
|
|
|
|
"-- forcing full recovery\n", bmname(bitmap), events,
|
|
|
|
(unsigned long long) bitmap->mddev->events);
|
2006-10-21 19:24:09 +02:00
|
|
|
sb->state |= cpu_to_le32(BITMAP_STALE);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
success:
|
|
|
|
/* assign fields using values from superblock */
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->mddev->bitmap_info.chunksize = chunksize;
|
|
|
|
bitmap->mddev->bitmap_info.daemon_sleep = daemon_sleep;
|
|
|
|
bitmap->mddev->bitmap_info.max_write_behind = write_behind;
|
2006-10-21 19:24:09 +02:00
|
|
|
bitmap->flags |= le32_to_cpu(sb->state);
|
2005-11-09 06:39:32 +01:00
|
|
|
if (le32_to_cpu(sb->version) == BITMAP_MAJOR_HOSTENDIAN)
|
|
|
|
bitmap->flags |= BITMAP_HOSTENDIAN;
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->events_cleared = le64_to_cpu(sb->events_cleared);
|
2006-10-21 19:24:09 +02:00
|
|
|
if (sb->state & cpu_to_le32(BITMAP_STALE))
|
2005-09-10 01:23:44 +02:00
|
|
|
bitmap->events_cleared = bitmap->mddev->events;
|
2005-06-22 02:17:14 +02:00
|
|
|
err = 0;
|
|
|
|
out:
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(sb, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
if (err)
|
|
|
|
bitmap_print_sb(bitmap);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum bitmap_mask_op {
|
|
|
|
MASK_SET,
|
|
|
|
MASK_UNSET
|
|
|
|
};
|
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
/* record the state of the bitmap in the superblock. Return the old value */
|
|
|
|
static int bitmap_mask_state(struct bitmap *bitmap, enum bitmap_state bits,
|
|
|
|
enum bitmap_mask_op op)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
bitmap_super_t *sb;
|
|
|
|
unsigned long flags;
|
2007-07-17 13:06:13 +02:00
|
|
|
int old;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
2006-03-25 12:07:51 +01:00
|
|
|
if (!bitmap->sb_page) { /* can't set the state */
|
2005-06-22 02:17:14 +02:00
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2007-07-17 13:06:13 +02:00
|
|
|
return 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2010-03-08 06:02:40 +01:00
|
|
|
sb = kmap_atomic(bitmap->sb_page, KM_USER0);
|
2007-07-17 13:06:13 +02:00
|
|
|
old = le32_to_cpu(sb->state) & bits;
|
2005-06-22 02:17:14 +02:00
|
|
|
switch (op) {
|
2006-10-21 19:24:09 +02:00
|
|
|
case MASK_SET: sb->state |= cpu_to_le32(bits);
|
2005-06-22 02:17:14 +02:00
|
|
|
break;
|
2006-10-21 19:24:09 +02:00
|
|
|
case MASK_UNSET: sb->state &= cpu_to_le32(~bits);
|
2005-06-22 02:17:14 +02:00
|
|
|
break;
|
|
|
|
default: BUG();
|
|
|
|
}
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(sb, KM_USER0);
|
2007-07-17 13:06:13 +02:00
|
|
|
return old;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* general bitmap file operations
|
|
|
|
*/
|
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
/*
|
|
|
|
* on-disk bitmap:
|
|
|
|
*
|
|
|
|
* Use one bit per "chunk" (block set). We do the disk I/O on the bitmap
|
|
|
|
* file a page at a time. There's a superblock at the start of the file.
|
|
|
|
*/
|
2005-06-22 02:17:14 +02:00
|
|
|
/* calculate the index of the page that contains this bit */
|
2009-12-14 02:49:56 +01:00
|
|
|
static inline unsigned long file_page_index(struct bitmap *bitmap, unsigned long chunk)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2009-12-14 02:49:56 +01:00
|
|
|
if (!bitmap->mddev->bitmap_info.external)
|
|
|
|
chunk += sizeof(bitmap_super_t) << 3;
|
|
|
|
return chunk >> PAGE_BIT_SHIFT;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* calculate the (bit) offset of this bit within a page */
|
2009-12-14 02:49:56 +01:00
|
|
|
static inline unsigned long file_page_offset(struct bitmap *bitmap, unsigned long chunk)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2009-12-14 02:49:56 +01:00
|
|
|
if (!bitmap->mddev->bitmap_info.external)
|
|
|
|
chunk += sizeof(bitmap_super_t) << 3;
|
|
|
|
return chunk & (PAGE_BITS - 1);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* return a pointer to the page in the filemap that contains the given bit
|
|
|
|
*
|
|
|
|
* this lookup is complicated by the fact that the bitmap sb might be exactly
|
|
|
|
* 1 page (e.g., x86) or less than 1 page -- so the bitmap might start on page
|
|
|
|
* 0 or page 1
|
|
|
|
*/
|
|
|
|
static inline struct page *filemap_get_page(struct bitmap *bitmap,
|
|
|
|
unsigned long chunk)
|
|
|
|
{
|
2009-12-14 02:49:56 +01:00
|
|
|
if (file_page_index(bitmap, chunk) >= bitmap->file_pages) return NULL;
|
|
|
|
return bitmap->filemap[file_page_index(bitmap, chunk)
|
|
|
|
- file_page_index(bitmap, 0)];
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void bitmap_file_unmap(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
struct page **map, *sb_page;
|
|
|
|
unsigned long *attr;
|
|
|
|
int pages;
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
map = bitmap->filemap;
|
|
|
|
bitmap->filemap = NULL;
|
|
|
|
attr = bitmap->filemap_attr;
|
|
|
|
bitmap->filemap_attr = NULL;
|
|
|
|
pages = bitmap->file_pages;
|
|
|
|
bitmap->file_pages = 0;
|
|
|
|
sb_page = bitmap->sb_page;
|
|
|
|
bitmap->sb_page = NULL;
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
|
|
|
|
while (pages--)
|
2009-12-14 02:49:56 +01:00
|
|
|
if (map[pages] != sb_page) /* 0 is sb_page, release it below */
|
2006-06-26 09:27:48 +02:00
|
|
|
free_buffers(map[pages]);
|
2005-06-22 02:17:14 +02:00
|
|
|
kfree(map);
|
|
|
|
kfree(attr);
|
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
if (sb_page)
|
|
|
|
free_buffers(sb_page);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void bitmap_file_put(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
struct file *file;
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
file = bitmap->file;
|
|
|
|
bitmap->file = NULL;
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
if (file)
|
|
|
|
wait_event(bitmap->write_wait,
|
|
|
|
atomic_read(&bitmap->pending_writes)==0);
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap_file_unmap(bitmap);
|
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
if (file) {
|
2006-12-08 11:37:19 +01:00
|
|
|
struct inode *inode = file->f_path.dentry->d_inode;
|
2007-02-10 10:45:39 +01:00
|
|
|
invalidate_mapping_pages(inode->i_mapping, 0, -1);
|
2005-06-22 02:17:14 +02:00
|
|
|
fput(file);
|
2006-06-26 09:27:48 +02:00
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bitmap_file_kick - if an error occurs while manipulating the bitmap file
|
|
|
|
* then it is no longer reliable, so we stop using it and we mark the file
|
|
|
|
* as failed in the superblock
|
|
|
|
*/
|
|
|
|
static void bitmap_file_kick(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
char *path, *ptr = NULL;
|
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
if (bitmap_mask_state(bitmap, BITMAP_STALE, MASK_SET) == 0) {
|
|
|
|
bitmap_update_sb(bitmap);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
if (bitmap->file) {
|
|
|
|
path = kmalloc(PAGE_SIZE, GFP_KERNEL);
|
|
|
|
if (path)
|
2008-05-23 22:04:34 +02:00
|
|
|
ptr = d_path(&bitmap->file->f_path, path,
|
|
|
|
PAGE_SIZE);
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
printk(KERN_ALERT
|
|
|
|
"%s: kicking failed bitmap file %s from array!\n",
|
2008-05-23 22:04:34 +02:00
|
|
|
bmname(bitmap), IS_ERR(ptr) ? "" : ptr);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
kfree(path);
|
|
|
|
} else
|
|
|
|
printk(KERN_ALERT
|
|
|
|
"%s: disabling internal bitmap due to errors\n",
|
|
|
|
bmname(bitmap));
|
2005-06-22 02:17:27 +02:00
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
bitmap_file_put(bitmap);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum bitmap_page_attr {
|
2006-06-26 09:27:45 +02:00
|
|
|
BITMAP_PAGE_DIRTY = 0, // there are set bits that need to be synced
|
|
|
|
BITMAP_PAGE_CLEAN = 1, // there are bits that might need to be cleared
|
|
|
|
BITMAP_PAGE_NEEDWRITE=2, // there are cleared bits that need to be synced
|
2005-06-22 02:17:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline void set_page_attr(struct bitmap *bitmap, struct page *page,
|
|
|
|
enum bitmap_page_attr attr)
|
|
|
|
{
|
2006-06-26 09:27:45 +02:00
|
|
|
__set_bit((page->index<<2) + attr, bitmap->filemap_attr);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void clear_page_attr(struct bitmap *bitmap, struct page *page,
|
|
|
|
enum bitmap_page_attr attr)
|
|
|
|
{
|
2006-06-26 09:27:45 +02:00
|
|
|
__clear_bit((page->index<<2) + attr, bitmap->filemap_attr);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2006-06-26 09:27:45 +02:00
|
|
|
static inline unsigned long test_page_attr(struct bitmap *bitmap, struct page *page,
|
|
|
|
enum bitmap_page_attr attr)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2006-06-26 09:27:45 +02:00
|
|
|
return test_bit((page->index<<2) + attr, bitmap->filemap_attr);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bitmap_file_set_bit -- called before performing a write to the md device
|
|
|
|
* to set (and eventually sync) a particular bit in the bitmap file
|
|
|
|
*
|
|
|
|
* we set the bit immediately, then we record the page number so that
|
|
|
|
* when an unplug occurs, we can flush the dirty pages out to disk
|
|
|
|
*/
|
|
|
|
static void bitmap_file_set_bit(struct bitmap *bitmap, sector_t block)
|
|
|
|
{
|
|
|
|
unsigned long bit;
|
|
|
|
struct page *page;
|
|
|
|
void *kaddr;
|
|
|
|
unsigned long chunk = block >> CHUNK_BLOCK_SHIFT(bitmap);
|
|
|
|
|
2005-06-22 02:17:27 +02:00
|
|
|
if (!bitmap->filemap) {
|
2005-06-22 02:17:14 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
page = filemap_get_page(bitmap, chunk);
|
2006-10-03 10:15:49 +02:00
|
|
|
if (!page) return;
|
2009-12-14 02:49:56 +01:00
|
|
|
bit = file_page_offset(bitmap, chunk);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* set the bit */
|
|
|
|
kaddr = kmap_atomic(page, KM_USER0);
|
2005-11-09 06:39:32 +01:00
|
|
|
if (bitmap->flags & BITMAP_HOSTENDIAN)
|
|
|
|
set_bit(bit, kaddr);
|
|
|
|
else
|
|
|
|
ext2_set_bit(bit, kaddr);
|
2005-06-22 02:17:14 +02:00
|
|
|
kunmap_atomic(kaddr, KM_USER0);
|
|
|
|
PRINTK("set file bit %lu page %lu\n", bit, page->index);
|
|
|
|
|
|
|
|
/* record page number so it gets flushed to disk when unplug occurs */
|
|
|
|
set_page_attr(bitmap, page, BITMAP_PAGE_DIRTY);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this gets called when the md device is ready to unplug its underlying
|
|
|
|
* (slave) device queues -- before we let any writes go down, we need to
|
|
|
|
* sync the dirty pages of the bitmap file to disk */
|
2007-07-17 13:06:13 +02:00
|
|
|
void bitmap_unplug(struct bitmap *bitmap)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2006-06-26 09:27:45 +02:00
|
|
|
unsigned long i, flags;
|
|
|
|
int dirty, need_write;
|
2005-06-22 02:17:14 +02:00
|
|
|
struct page *page;
|
|
|
|
int wait = 0;
|
|
|
|
|
|
|
|
if (!bitmap)
|
2007-07-17 13:06:13 +02:00
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* look at each page to see if there are any set bits that need to be
|
|
|
|
* flushed out to disk */
|
|
|
|
for (i = 0; i < bitmap->file_pages; i++) {
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
2005-06-22 02:17:27 +02:00
|
|
|
if (!bitmap->filemap) {
|
2005-06-22 02:17:14 +02:00
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2007-07-17 13:06:13 +02:00
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
page = bitmap->filemap[i];
|
2006-06-26 09:27:45 +02:00
|
|
|
dirty = test_page_attr(bitmap, page, BITMAP_PAGE_DIRTY);
|
|
|
|
need_write = test_page_attr(bitmap, page, BITMAP_PAGE_NEEDWRITE);
|
2005-06-22 02:17:14 +02:00
|
|
|
clear_page_attr(bitmap, page, BITMAP_PAGE_DIRTY);
|
|
|
|
clear_page_attr(bitmap, page, BITMAP_PAGE_NEEDWRITE);
|
2006-06-26 09:27:45 +02:00
|
|
|
if (dirty)
|
2005-06-22 02:17:14 +02:00
|
|
|
wait = 1;
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
|
2006-06-26 09:27:48 +02:00
|
|
|
if (dirty | need_write)
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, page, 0);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
if (wait) { /* if any writes were performed, we need to wait on them */
|
2006-06-26 09:27:44 +02:00
|
|
|
if (bitmap->file)
|
2006-06-26 09:27:48 +02:00
|
|
|
wait_event(bitmap->write_wait,
|
|
|
|
atomic_read(&bitmap->pending_writes)==0);
|
2006-06-26 09:27:44 +02:00
|
|
|
else
|
[PATCH] md: support BIO_RW_BARRIER for md/raid1
We can only accept BARRIER requests if all slaves handle
barriers, and that can, of course, change with time....
So we keep track of whether the whole array seems safe for barriers,
and also whether each individual rdev handles barriers.
We initially assumes barriers are OK.
When writing the superblock we try a barrier, and if that fails, we flag
things for no-barriers. This will usually clear the flags fairly quickly.
If writing the superblock finds that BIO_RW_BARRIER is -ENOTSUPP, we need to
resubmit, so introduce function "md_super_wait" which waits for requests to
finish, and retries ENOTSUPP requests without the barrier flag.
When writing the real raid1, write requests which were BIO_RW_BARRIER but
which aresn't supported need to be retried. So raid1d is enhanced to do this,
and when any bio write completes (i.e. no retry needed) we remove it from the
r1bio, so that devices needing retry are easy to find.
We should hardly ever get -ENOTSUPP errors when writing data to the raid.
It should only happen if:
1/ the device used to support BARRIER, but now doesn't. Few devices
change like this, though raid1 can!
or
2/ the array has no persistent superblock, so there was no opportunity to
pre-test for barriers when writing the superblock.
Signed-off-by: Neil Brown <neilb@cse.unsw.edu.au>
Signed-off-by: Neil Brown <neilb@suse.de>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-11-09 06:39:34 +01:00
|
|
|
md_super_wait(bitmap->mddev);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2006-06-26 09:27:48 +02:00
|
|
|
if (bitmap->flags & BITMAP_WRITE_ERROR)
|
|
|
|
bitmap_file_kick(bitmap);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2005-09-10 01:23:44 +02:00
|
|
|
static void bitmap_set_memory_bits(struct bitmap *bitmap, sector_t offset, int needed);
|
2005-06-22 02:17:14 +02:00
|
|
|
/* * bitmap_init_from_disk -- called at bitmap_create time to initialize
|
|
|
|
* the in-memory bitmap from the on-disk bitmap -- also, sets up the
|
|
|
|
* memory mapping of the bitmap file
|
|
|
|
* Special cases:
|
|
|
|
* if there's no bitmap file, or if the bitmap file had been
|
|
|
|
* previously kicked from the array, we mark all the bits as
|
|
|
|
* 1's in order to cause a full resync.
|
2005-09-10 01:23:44 +02:00
|
|
|
*
|
|
|
|
* We ignore all bits for sectors that end earlier than 'start'.
|
|
|
|
* This is used when reading an out-of-date bitmap...
|
2005-06-22 02:17:14 +02:00
|
|
|
*/
|
2005-09-10 01:23:44 +02:00
|
|
|
static int bitmap_init_from_disk(struct bitmap *bitmap, sector_t start)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
unsigned long i, chunks, index, oldindex, bit;
|
|
|
|
struct page *page = NULL, *oldpage = NULL;
|
|
|
|
unsigned long num_pages, bit_cnt = 0;
|
|
|
|
struct file *file;
|
2006-06-26 09:27:48 +02:00
|
|
|
unsigned long bytes, offset;
|
2005-06-22 02:17:14 +02:00
|
|
|
int outofdate;
|
|
|
|
int ret = -ENOSPC;
|
2006-01-06 09:20:34 +01:00
|
|
|
void *paddr;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
chunks = bitmap->chunks;
|
|
|
|
file = bitmap->file;
|
|
|
|
|
2009-12-14 02:49:53 +01:00
|
|
|
BUG_ON(!file && !bitmap->mddev->bitmap_info.offset);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2005-07-27 20:45:17 +02:00
|
|
|
#ifdef INJECT_FAULTS_3
|
2005-06-22 02:17:14 +02:00
|
|
|
outofdate = 1;
|
|
|
|
#else
|
|
|
|
outofdate = bitmap->flags & BITMAP_STALE;
|
|
|
|
#endif
|
|
|
|
if (outofdate)
|
|
|
|
printk(KERN_INFO "%s: bitmap file is out of date, doing full "
|
|
|
|
"recovery\n", bmname(bitmap));
|
|
|
|
|
|
|
|
bytes = (chunks + 7) / 8;
|
2009-12-14 02:49:56 +01:00
|
|
|
if (!bitmap->mddev->bitmap_info.external)
|
|
|
|
bytes += sizeof(bitmap_super_t);
|
2005-06-22 02:17:17 +02:00
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
|
|
|
|
num_pages = (bytes + PAGE_SIZE - 1) / PAGE_SIZE;
|
2005-06-22 02:17:17 +02:00
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
if (file && i_size_read(file->f_mapping->host) < bytes) {
|
2005-06-22 02:17:14 +02:00
|
|
|
printk(KERN_INFO "%s: bitmap file too short %lu < %lu\n",
|
|
|
|
bmname(bitmap),
|
|
|
|
(unsigned long) i_size_read(file->f_mapping->host),
|
2009-12-14 02:49:56 +01:00
|
|
|
bytes);
|
2007-07-17 13:06:13 +02:00
|
|
|
goto err;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2005-06-22 02:17:17 +02:00
|
|
|
|
|
|
|
ret = -ENOMEM;
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->filemap = kmalloc(sizeof(struct page *) * num_pages, GFP_KERNEL);
|
2005-06-22 02:17:17 +02:00
|
|
|
if (!bitmap->filemap)
|
2007-07-17 13:06:13 +02:00
|
|
|
goto err;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-06-26 09:27:45 +02:00
|
|
|
/* We need 4 bits per page, rounded up to a multiple of sizeof(unsigned long) */
|
|
|
|
bitmap->filemap_attr = kzalloc(
|
2007-04-12 08:28:44 +02:00
|
|
|
roundup( DIV_ROUND_UP(num_pages*4, 8), sizeof(unsigned long)),
|
2006-06-26 09:27:45 +02:00
|
|
|
GFP_KERNEL);
|
2005-06-22 02:17:17 +02:00
|
|
|
if (!bitmap->filemap_attr)
|
2007-07-17 13:06:13 +02:00
|
|
|
goto err;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
oldindex = ~0L;
|
|
|
|
|
|
|
|
for (i = 0; i < chunks; i++) {
|
2005-11-09 06:39:32 +01:00
|
|
|
int b;
|
2009-12-14 02:49:56 +01:00
|
|
|
index = file_page_index(bitmap, i);
|
|
|
|
bit = file_page_offset(bitmap, i);
|
2005-06-22 02:17:14 +02:00
|
|
|
if (index != oldindex) { /* this is a new page, read it in */
|
2006-06-26 09:27:48 +02:00
|
|
|
int count;
|
2005-06-22 02:17:14 +02:00
|
|
|
/* unmap the old page, we're done with it */
|
2006-06-26 09:27:48 +02:00
|
|
|
if (index == num_pages-1)
|
2009-12-14 02:49:56 +01:00
|
|
|
count = bytes - index * PAGE_SIZE;
|
2006-06-26 09:27:48 +02:00
|
|
|
else
|
|
|
|
count = PAGE_SIZE;
|
2009-12-14 02:49:56 +01:00
|
|
|
if (index == 0 && bitmap->sb_page) {
|
2005-06-22 02:17:14 +02:00
|
|
|
/*
|
|
|
|
* if we're here then the superblock page
|
|
|
|
* contains some bits (PAGE_SIZE != sizeof sb)
|
|
|
|
* we've already read it in, so just use it
|
|
|
|
*/
|
|
|
|
page = bitmap->sb_page;
|
|
|
|
offset = sizeof(bitmap_super_t);
|
2009-01-08 22:31:05 +01:00
|
|
|
if (!file)
|
|
|
|
read_sb_page(bitmap->mddev,
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->mddev->bitmap_info.offset,
|
2009-01-08 22:31:05 +01:00
|
|
|
page,
|
|
|
|
index, count);
|
2005-06-22 02:17:27 +02:00
|
|
|
} else if (file) {
|
2006-06-26 09:27:48 +02:00
|
|
|
page = read_page(file, index, bitmap, count);
|
2005-06-22 02:17:27 +02:00
|
|
|
offset = 0;
|
|
|
|
} else {
|
2009-12-14 02:49:53 +01:00
|
|
|
page = read_sb_page(bitmap->mddev,
|
|
|
|
bitmap->mddev->bitmap_info.offset,
|
2008-12-19 06:25:01 +01:00
|
|
|
NULL,
|
|
|
|
index, count);
|
2005-06-22 02:17:14 +02:00
|
|
|
offset = 0;
|
|
|
|
}
|
2005-06-22 02:17:27 +02:00
|
|
|
if (IS_ERR(page)) { /* read error */
|
|
|
|
ret = PTR_ERR(page);
|
2007-07-17 13:06:13 +02:00
|
|
|
goto err;
|
2005-06-22 02:17:27 +02:00
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
oldindex = index;
|
|
|
|
oldpage = page;
|
|
|
|
|
2009-05-07 04:47:19 +02:00
|
|
|
bitmap->filemap[bitmap->file_pages++] = page;
|
|
|
|
bitmap->last_page_size = count;
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
if (outofdate) {
|
|
|
|
/*
|
|
|
|
* if bitmap is out of date, dirty the
|
|
|
|
* whole page and write it out
|
|
|
|
*/
|
2006-01-06 09:20:34 +01:00
|
|
|
paddr = kmap_atomic(page, KM_USER0);
|
|
|
|
memset(paddr + offset, 0xff,
|
2005-09-10 01:23:44 +02:00
|
|
|
PAGE_SIZE - offset);
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(paddr, KM_USER0);
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, page, 1);
|
|
|
|
|
|
|
|
ret = -EIO;
|
2009-05-07 04:47:19 +02:00
|
|
|
if (bitmap->flags & BITMAP_WRITE_ERROR)
|
2007-07-17 13:06:13 +02:00
|
|
|
goto err;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
}
|
2006-01-06 09:20:34 +01:00
|
|
|
paddr = kmap_atomic(page, KM_USER0);
|
2005-11-09 06:39:32 +01:00
|
|
|
if (bitmap->flags & BITMAP_HOSTENDIAN)
|
2006-01-06 09:20:34 +01:00
|
|
|
b = test_bit(bit, paddr);
|
2005-11-09 06:39:32 +01:00
|
|
|
else
|
2006-01-06 09:20:34 +01:00
|
|
|
b = ext2_test_bit(bit, paddr);
|
|
|
|
kunmap_atomic(paddr, KM_USER0);
|
2005-11-09 06:39:32 +01:00
|
|
|
if (b) {
|
2005-06-22 02:17:14 +02:00
|
|
|
/* if the disk bit is set, set the memory bit */
|
2009-05-07 04:49:06 +02:00
|
|
|
int needed = ((sector_t)(i+1) << (CHUNK_BLOCK_SHIFT(bitmap))
|
|
|
|
>= start);
|
|
|
|
bitmap_set_memory_bits(bitmap,
|
|
|
|
(sector_t)i << CHUNK_BLOCK_SHIFT(bitmap),
|
|
|
|
needed);
|
2005-06-22 02:17:14 +02:00
|
|
|
bit_cnt++;
|
2005-09-10 01:23:44 +02:00
|
|
|
set_page_attr(bitmap, page, BITMAP_PAGE_CLEAN);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* everything went OK */
|
|
|
|
ret = 0;
|
|
|
|
bitmap_mask_state(bitmap, BITMAP_STALE, MASK_UNSET);
|
|
|
|
|
|
|
|
if (bit_cnt) { /* Kick recovery if any bits were set */
|
|
|
|
set_bit(MD_RECOVERY_NEEDED, &bitmap->mddev->recovery);
|
|
|
|
md_wakeup_thread(bitmap->mddev->thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
printk(KERN_INFO "%s: bitmap initialized from disk: "
|
2007-07-17 13:06:13 +02:00
|
|
|
"read %lu/%lu pages, set %lu bits\n",
|
|
|
|
bmname(bitmap), bitmap->file_pages, num_pages, bit_cnt);
|
|
|
|
|
|
|
|
return 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
err:
|
|
|
|
printk(KERN_INFO "%s: bitmap initialisation failed: %d\n",
|
|
|
|
bmname(bitmap), ret);
|
2005-06-22 02:17:14 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:27 +02:00
|
|
|
void bitmap_write_all(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
/* We don't actually write all bitmap blocks here,
|
|
|
|
* just flag them as needing to be written
|
|
|
|
*/
|
2006-06-26 09:27:45 +02:00
|
|
|
int i;
|
2005-06-22 02:17:27 +02:00
|
|
|
|
2006-06-26 09:27:45 +02:00
|
|
|
for (i=0; i < bitmap->file_pages; i++)
|
|
|
|
set_page_attr(bitmap, bitmap->filemap[i],
|
|
|
|
BITMAP_PAGE_NEEDWRITE);
|
2005-06-22 02:17:27 +02:00
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
static void bitmap_count_page(struct bitmap *bitmap, sector_t offset, int inc)
|
|
|
|
{
|
|
|
|
sector_t chunk = offset >> CHUNK_BLOCK_SHIFT(bitmap);
|
|
|
|
unsigned long page = chunk >> PAGE_COUNTER_SHIFT;
|
|
|
|
bitmap->bp[page].count += inc;
|
|
|
|
/*
|
|
|
|
if (page == 0) printk("count page 0, offset %llu: %d gives %d\n",
|
|
|
|
(unsigned long long)offset, inc, bitmap->bp[page].count);
|
|
|
|
*/
|
|
|
|
bitmap_checkfree(bitmap, page);
|
|
|
|
}
|
|
|
|
static bitmap_counter_t *bitmap_get_counter(struct bitmap *bitmap,
|
|
|
|
sector_t offset, int *blocks,
|
|
|
|
int create);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* bitmap daemon -- periodically wakes up to clean bits and flush pages
|
|
|
|
* out to disk
|
|
|
|
*/
|
|
|
|
|
2009-12-14 02:49:46 +01:00
|
|
|
void bitmap_daemon_work(mddev_t *mddev)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
2009-12-14 02:49:46 +01:00
|
|
|
struct bitmap *bitmap;
|
2005-06-22 02:17:22 +02:00
|
|
|
unsigned long j;
|
2005-06-22 02:17:14 +02:00
|
|
|
unsigned long flags;
|
|
|
|
struct page *page = NULL, *lastpage = NULL;
|
|
|
|
int blocks;
|
2006-01-06 09:20:34 +01:00
|
|
|
void *paddr;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2009-12-14 02:49:46 +01:00
|
|
|
/* Use a mutex to guard daemon_work against
|
|
|
|
* bitmap_destroy.
|
|
|
|
*/
|
2009-12-14 02:49:52 +01:00
|
|
|
mutex_lock(&mddev->bitmap_info.mutex);
|
2009-12-14 02:49:46 +01:00
|
|
|
bitmap = mddev->bitmap;
|
|
|
|
if (bitmap == NULL) {
|
2009-12-14 02:49:52 +01:00
|
|
|
mutex_unlock(&mddev->bitmap_info.mutex);
|
2007-07-17 13:06:13 +02:00
|
|
|
return;
|
2009-12-14 02:49:46 +01:00
|
|
|
}
|
2009-12-14 02:49:53 +01:00
|
|
|
if (time_before(jiffies, bitmap->daemon_lastrun
|
2009-12-14 02:49:53 +01:00
|
|
|
+ bitmap->mddev->bitmap_info.daemon_sleep))
|
2008-03-10 19:43:48 +01:00
|
|
|
goto done;
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->daemon_lastrun = jiffies;
|
2008-03-04 23:29:30 +01:00
|
|
|
if (bitmap->allclean) {
|
|
|
|
bitmap->mddev->thread->timeout = MAX_SCHEDULE_TIMEOUT;
|
2009-12-14 02:49:46 +01:00
|
|
|
goto done;
|
2008-03-04 23:29:30 +01:00
|
|
|
}
|
|
|
|
bitmap->allclean = 1;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2009-05-26 01:41:17 +02:00
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
2005-06-22 02:17:14 +02:00
|
|
|
for (j = 0; j < bitmap->chunks; j++) {
|
|
|
|
bitmap_counter_t *bmc;
|
2009-05-26 01:41:17 +02:00
|
|
|
if (!bitmap->filemap)
|
2005-06-22 02:17:14 +02:00
|
|
|
/* error or shutdown */
|
|
|
|
break;
|
|
|
|
|
|
|
|
page = filemap_get_page(bitmap, j);
|
|
|
|
|
|
|
|
if (page != lastpage) {
|
2005-06-22 02:17:22 +02:00
|
|
|
/* skip this page unless it's marked as needing cleaning */
|
2006-06-26 09:27:45 +02:00
|
|
|
if (!test_page_attr(bitmap, page, BITMAP_PAGE_CLEAN)) {
|
|
|
|
int need_write = test_page_attr(bitmap, page,
|
|
|
|
BITMAP_PAGE_NEEDWRITE);
|
2006-06-26 09:27:46 +02:00
|
|
|
if (need_write)
|
2005-06-22 02:17:22 +02:00
|
|
|
clear_page_attr(bitmap, page, BITMAP_PAGE_NEEDWRITE);
|
2006-06-26 09:27:46 +02:00
|
|
|
|
2005-06-22 02:17:22 +02:00
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2008-03-04 23:29:30 +01:00
|
|
|
if (need_write) {
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, page, 0);
|
2008-03-04 23:29:30 +01:00
|
|
|
bitmap->allclean = 0;
|
|
|
|
}
|
2009-05-26 01:41:17 +02:00
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
j |= (PAGE_BITS - 1);
|
2005-06-22 02:17:22 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
/* grab the new page, sync and release the old */
|
|
|
|
if (lastpage != NULL) {
|
2006-06-26 09:27:45 +02:00
|
|
|
if (test_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE)) {
|
2005-06-22 02:17:14 +02:00
|
|
|
clear_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE);
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, lastpage, 0);
|
2005-06-22 02:17:14 +02:00
|
|
|
} else {
|
|
|
|
set_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE);
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
lastpage = page;
|
2008-06-28 00:31:22 +02:00
|
|
|
|
|
|
|
/* We are possibly going to clear some bits, so make
|
|
|
|
* sure that events_cleared is up-to-date.
|
|
|
|
*/
|
2009-12-14 02:49:56 +01:00
|
|
|
if (bitmap->need_sync &&
|
|
|
|
bitmap->mddev->bitmap_info.external == 0) {
|
2008-06-28 00:31:22 +02:00
|
|
|
bitmap_super_t *sb;
|
|
|
|
bitmap->need_sync = 0;
|
|
|
|
sb = kmap_atomic(bitmap->sb_page, KM_USER0);
|
|
|
|
sb->events_cleared =
|
|
|
|
cpu_to_le64(bitmap->events_cleared);
|
|
|
|
kunmap_atomic(sb, KM_USER0);
|
|
|
|
write_page(bitmap, bitmap->sb_page, 1);
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
2009-12-14 02:49:56 +01:00
|
|
|
if (!bitmap->need_sync)
|
|
|
|
clear_page_attr(bitmap, page, BITMAP_PAGE_CLEAN);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2009-05-07 04:49:06 +02:00
|
|
|
bmc = bitmap_get_counter(bitmap,
|
|
|
|
(sector_t)j << CHUNK_BLOCK_SHIFT(bitmap),
|
|
|
|
&blocks, 0);
|
2005-06-22 02:17:14 +02:00
|
|
|
if (bmc) {
|
|
|
|
/*
|
|
|
|
if (j < 100) printk("bitmap: j=%lu, *bmc = 0x%x\n", j, *bmc);
|
|
|
|
*/
|
2008-03-04 23:29:30 +01:00
|
|
|
if (*bmc)
|
|
|
|
bitmap->allclean = 0;
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
if (*bmc == 2) {
|
|
|
|
*bmc=1; /* maybe clear the bit next time */
|
|
|
|
set_page_attr(bitmap, page, BITMAP_PAGE_CLEAN);
|
2009-12-14 02:49:56 +01:00
|
|
|
} else if (*bmc == 1 && !bitmap->need_sync) {
|
2005-06-22 02:17:14 +02:00
|
|
|
/* we can clear the bit */
|
|
|
|
*bmc = 0;
|
2009-05-07 04:49:06 +02:00
|
|
|
bitmap_count_page(bitmap,
|
|
|
|
(sector_t)j << CHUNK_BLOCK_SHIFT(bitmap),
|
2005-06-22 02:17:14 +02:00
|
|
|
-1);
|
|
|
|
|
|
|
|
/* clear the bit */
|
2006-01-06 09:20:34 +01:00
|
|
|
paddr = kmap_atomic(page, KM_USER0);
|
2005-11-09 06:39:32 +01:00
|
|
|
if (bitmap->flags & BITMAP_HOSTENDIAN)
|
2009-12-14 02:49:56 +01:00
|
|
|
clear_bit(file_page_offset(bitmap, j),
|
|
|
|
paddr);
|
2005-11-09 06:39:32 +01:00
|
|
|
else
|
2009-12-14 02:49:56 +01:00
|
|
|
ext2_clear_bit(file_page_offset(bitmap, j),
|
|
|
|
paddr);
|
2006-01-06 09:20:34 +01:00
|
|
|
kunmap_atomic(paddr, KM_USER0);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2009-05-26 01:41:17 +02:00
|
|
|
} else
|
|
|
|
j |= PAGE_COUNTER_MASK;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2009-05-26 01:41:17 +02:00
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* now sync the final page */
|
|
|
|
if (lastpage != NULL) {
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
2006-06-26 09:27:45 +02:00
|
|
|
if (test_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE)) {
|
2005-06-22 02:17:14 +02:00
|
|
|
clear_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE);
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2007-07-17 13:06:13 +02:00
|
|
|
write_page(bitmap, lastpage, 0);
|
2005-06-22 02:17:14 +02:00
|
|
|
} else {
|
|
|
|
set_page_attr(bitmap, lastpage, BITMAP_PAGE_NEEDWRITE);
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-10 19:43:48 +01:00
|
|
|
done:
|
2008-03-04 23:29:30 +01:00
|
|
|
if (bitmap->allclean == 0)
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->mddev->thread->timeout =
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->mddev->bitmap_info.daemon_sleep;
|
2009-12-14 02:49:52 +01:00
|
|
|
mutex_unlock(&mddev->bitmap_info.mutex);
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static bitmap_counter_t *bitmap_get_counter(struct bitmap *bitmap,
|
|
|
|
sector_t offset, int *blocks,
|
|
|
|
int create)
|
2009-09-23 10:06:44 +02:00
|
|
|
__releases(bitmap->lock)
|
|
|
|
__acquires(bitmap->lock)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
/* If 'create', we might release the lock and reclaim it.
|
|
|
|
* The lock must have been taken with interrupts enabled.
|
|
|
|
* If !create, we don't release the lock.
|
|
|
|
*/
|
|
|
|
sector_t chunk = offset >> CHUNK_BLOCK_SHIFT(bitmap);
|
|
|
|
unsigned long page = chunk >> PAGE_COUNTER_SHIFT;
|
|
|
|
unsigned long pageoff = (chunk & PAGE_COUNTER_MASK) << COUNTER_BYTE_SHIFT;
|
|
|
|
sector_t csize;
|
|
|
|
|
|
|
|
if (bitmap_checkpage(bitmap, page, create) < 0) {
|
|
|
|
csize = ((sector_t)1) << (CHUNK_BLOCK_SHIFT(bitmap));
|
|
|
|
*blocks = csize - (offset & (csize- 1));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
/* now locked ... */
|
|
|
|
|
|
|
|
if (bitmap->bp[page].hijacked) { /* hijacked pointer */
|
|
|
|
/* should we use the first or second counter field
|
|
|
|
* of the hijacked pointer? */
|
|
|
|
int hi = (pageoff > PAGE_COUNTER_MASK);
|
|
|
|
csize = ((sector_t)1) << (CHUNK_BLOCK_SHIFT(bitmap) +
|
|
|
|
PAGE_COUNTER_SHIFT - 1);
|
|
|
|
*blocks = csize - (offset & (csize- 1));
|
|
|
|
return &((bitmap_counter_t *)
|
|
|
|
&bitmap->bp[page].map)[hi];
|
|
|
|
} else { /* page is allocated */
|
|
|
|
csize = ((sector_t)1) << (CHUNK_BLOCK_SHIFT(bitmap));
|
|
|
|
*blocks = csize - (offset & (csize- 1));
|
|
|
|
return (bitmap_counter_t *)
|
|
|
|
&(bitmap->bp[page].map[pageoff]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-09-10 01:23:47 +02:00
|
|
|
int bitmap_startwrite(struct bitmap *bitmap, sector_t offset, unsigned long sectors, int behind)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
if (!bitmap) return 0;
|
2005-09-10 01:23:47 +02:00
|
|
|
|
|
|
|
if (behind) {
|
2010-03-08 06:02:37 +01:00
|
|
|
int bw;
|
2005-09-10 01:23:47 +02:00
|
|
|
atomic_inc(&bitmap->behind_writes);
|
2010-03-08 06:02:37 +01:00
|
|
|
bw = atomic_read(&bitmap->behind_writes);
|
|
|
|
if (bw > bitmap->behind_writes_used)
|
|
|
|
bitmap->behind_writes_used = bw;
|
|
|
|
|
2005-09-10 01:23:47 +02:00
|
|
|
PRINTK(KERN_DEBUG "inc write-behind count %d/%d\n",
|
2010-03-08 06:02:37 +01:00
|
|
|
bw, bitmap->max_write_behind);
|
2005-09-10 01:23:47 +02:00
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
while (sectors) {
|
|
|
|
int blocks;
|
|
|
|
bitmap_counter_t *bmc;
|
|
|
|
|
|
|
|
spin_lock_irq(&bitmap->lock);
|
|
|
|
bmc = bitmap_get_counter(bitmap, offset, &blocks, 1);
|
|
|
|
if (!bmc) {
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-02-08 23:20:37 +01:00
|
|
|
if (unlikely((*bmc & COUNTER_MAX) == COUNTER_MAX)) {
|
|
|
|
DEFINE_WAIT(__wait);
|
|
|
|
/* note that it is safe to do the prepare_to_wait
|
|
|
|
* after the test as long as we do it before dropping
|
|
|
|
* the spinlock.
|
|
|
|
*/
|
|
|
|
prepare_to_wait(&bitmap->overflow_wait, &__wait,
|
|
|
|
TASK_UNINTERRUPTIBLE);
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
2007-11-07 20:26:56 +01:00
|
|
|
blk_unplug(bitmap->mddev->queue);
|
2007-02-08 23:20:37 +01:00
|
|
|
schedule();
|
|
|
|
finish_wait(&bitmap->overflow_wait, &__wait);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
switch(*bmc) {
|
|
|
|
case 0:
|
|
|
|
bitmap_file_set_bit(bitmap, offset);
|
|
|
|
bitmap_count_page(bitmap,offset, 1);
|
2008-08-01 20:32:31 +02:00
|
|
|
blk_plug_device_unlocked(bitmap->mddev->queue);
|
2005-06-22 02:17:14 +02:00
|
|
|
/* fall through */
|
|
|
|
case 1:
|
|
|
|
*bmc = 2;
|
|
|
|
}
|
2007-02-08 23:20:37 +01:00
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
(*bmc)++;
|
|
|
|
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
|
|
|
|
|
|
|
offset += blocks;
|
|
|
|
if (sectors > blocks)
|
|
|
|
sectors -= blocks;
|
|
|
|
else sectors = 0;
|
|
|
|
}
|
2008-03-04 23:29:30 +01:00
|
|
|
bitmap->allclean = 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void bitmap_endwrite(struct bitmap *bitmap, sector_t offset, unsigned long sectors,
|
2005-09-10 01:23:47 +02:00
|
|
|
int success, int behind)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
if (!bitmap) return;
|
2005-09-10 01:23:47 +02:00
|
|
|
if (behind) {
|
2010-03-31 02:21:44 +02:00
|
|
|
if (atomic_dec_and_test(&bitmap->behind_writes))
|
|
|
|
wake_up(&bitmap->behind_wait);
|
2005-09-10 01:23:47 +02:00
|
|
|
PRINTK(KERN_DEBUG "dec write-behind count %d/%d\n",
|
|
|
|
atomic_read(&bitmap->behind_writes), bitmap->max_write_behind);
|
|
|
|
}
|
2009-03-31 05:27:02 +02:00
|
|
|
if (bitmap->mddev->degraded)
|
|
|
|
/* Never clear bits or update events_cleared when degraded */
|
|
|
|
success = 0;
|
2005-09-10 01:23:47 +02:00
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
while (sectors) {
|
|
|
|
int blocks;
|
|
|
|
unsigned long flags;
|
|
|
|
bitmap_counter_t *bmc;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
bmc = bitmap_get_counter(bitmap, offset, &blocks, 0);
|
|
|
|
if (!bmc) {
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-06-28 00:31:22 +02:00
|
|
|
if (success &&
|
|
|
|
bitmap->events_cleared < bitmap->mddev->events) {
|
|
|
|
bitmap->events_cleared = bitmap->mddev->events;
|
|
|
|
bitmap->need_sync = 1;
|
2009-12-14 02:49:56 +01:00
|
|
|
sysfs_notify_dirent(bitmap->sysfs_can_clear);
|
2008-06-28 00:31:22 +02:00
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
if (!success && ! (*bmc & NEEDED_MASK))
|
|
|
|
*bmc |= NEEDED_MASK;
|
|
|
|
|
2007-02-08 23:20:37 +01:00
|
|
|
if ((*bmc & COUNTER_MAX) == COUNTER_MAX)
|
|
|
|
wake_up(&bitmap->overflow_wait);
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
(*bmc)--;
|
|
|
|
if (*bmc <= 2) {
|
|
|
|
set_page_attr(bitmap,
|
|
|
|
filemap_get_page(bitmap, offset >> CHUNK_BLOCK_SHIFT(bitmap)),
|
|
|
|
BITMAP_PAGE_CLEAN);
|
|
|
|
}
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
|
|
|
offset += blocks;
|
|
|
|
if (sectors > blocks)
|
|
|
|
sectors -= blocks;
|
|
|
|
else sectors = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-31 05:27:02 +02:00
|
|
|
static int __bitmap_start_sync(struct bitmap *bitmap, sector_t offset, int *blocks,
|
|
|
|
int degraded)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
bitmap_counter_t *bmc;
|
|
|
|
int rv;
|
|
|
|
if (bitmap == NULL) {/* FIXME or bitmap set as 'failed' */
|
|
|
|
*blocks = 1024;
|
|
|
|
return 1; /* always resync if no bitmap */
|
|
|
|
}
|
|
|
|
spin_lock_irq(&bitmap->lock);
|
|
|
|
bmc = bitmap_get_counter(bitmap, offset, blocks, 0);
|
|
|
|
rv = 0;
|
|
|
|
if (bmc) {
|
|
|
|
/* locked */
|
|
|
|
if (RESYNC(*bmc))
|
|
|
|
rv = 1;
|
|
|
|
else if (NEEDED(*bmc)) {
|
|
|
|
rv = 1;
|
2005-07-15 12:56:35 +02:00
|
|
|
if (!degraded) { /* don't set/clear bits if degraded */
|
|
|
|
*bmc |= RESYNC_MASK;
|
|
|
|
*bmc &= ~NEEDED_MASK;
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
2008-03-04 23:29:30 +01:00
|
|
|
bitmap->allclean = 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2009-03-31 05:27:02 +02:00
|
|
|
int bitmap_start_sync(struct bitmap *bitmap, sector_t offset, int *blocks,
|
|
|
|
int degraded)
|
|
|
|
{
|
|
|
|
/* bitmap_start_sync must always report on multiples of whole
|
|
|
|
* pages, otherwise resync (which is very PAGE_SIZE based) will
|
|
|
|
* get confused.
|
|
|
|
* So call __bitmap_start_sync repeatedly (if needed) until
|
|
|
|
* At least PAGE_SIZE>>9 blocks are covered.
|
|
|
|
* Return the 'or' of the result.
|
|
|
|
*/
|
|
|
|
int rv = 0;
|
|
|
|
int blocks1;
|
|
|
|
|
|
|
|
*blocks = 0;
|
|
|
|
while (*blocks < (PAGE_SIZE>>9)) {
|
|
|
|
rv |= __bitmap_start_sync(bitmap, offset,
|
|
|
|
&blocks1, degraded);
|
|
|
|
offset += blocks1;
|
|
|
|
*blocks += blocks1;
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
void bitmap_end_sync(struct bitmap *bitmap, sector_t offset, int *blocks, int aborted)
|
|
|
|
{
|
|
|
|
bitmap_counter_t *bmc;
|
|
|
|
unsigned long flags;
|
|
|
|
/*
|
|
|
|
if (offset == 0) printk("bitmap_end_sync 0 (%d)\n", aborted);
|
|
|
|
*/ if (bitmap == NULL) {
|
|
|
|
*blocks = 1024;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
spin_lock_irqsave(&bitmap->lock, flags);
|
|
|
|
bmc = bitmap_get_counter(bitmap, offset, blocks, 0);
|
|
|
|
if (bmc == NULL)
|
|
|
|
goto unlock;
|
|
|
|
/* locked */
|
|
|
|
/*
|
|
|
|
if (offset == 0) printk("bitmap_end sync found 0x%x, blocks %d\n", *bmc, *blocks);
|
|
|
|
*/
|
|
|
|
if (RESYNC(*bmc)) {
|
|
|
|
*bmc &= ~RESYNC_MASK;
|
|
|
|
|
|
|
|
if (!NEEDED(*bmc) && aborted)
|
|
|
|
*bmc |= NEEDED_MASK;
|
|
|
|
else {
|
|
|
|
if (*bmc <= 2) {
|
|
|
|
set_page_attr(bitmap,
|
|
|
|
filemap_get_page(bitmap, offset >> CHUNK_BLOCK_SHIFT(bitmap)),
|
|
|
|
BITMAP_PAGE_CLEAN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
unlock:
|
|
|
|
spin_unlock_irqrestore(&bitmap->lock, flags);
|
2008-03-04 23:29:30 +01:00
|
|
|
bitmap->allclean = 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void bitmap_close_sync(struct bitmap *bitmap)
|
|
|
|
{
|
|
|
|
/* Sync has finished, and any bitmap chunks that weren't synced
|
|
|
|
* properly have been aborted. It remains to us to clear the
|
|
|
|
* RESYNC bit wherever it is still on
|
|
|
|
*/
|
|
|
|
sector_t sector = 0;
|
|
|
|
int blocks;
|
2008-02-06 10:39:50 +01:00
|
|
|
if (!bitmap)
|
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
while (sector < bitmap->mddev->resync_max_sectors) {
|
|
|
|
bitmap_end_sync(bitmap, sector, &blocks, 0);
|
2008-02-06 10:39:50 +01:00
|
|
|
sector += blocks;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bitmap_cond_end_sync(struct bitmap *bitmap, sector_t sector)
|
|
|
|
{
|
|
|
|
sector_t s = 0;
|
|
|
|
int blocks;
|
|
|
|
|
|
|
|
if (!bitmap)
|
|
|
|
return;
|
|
|
|
if (sector == 0) {
|
|
|
|
bitmap->last_end_sync = jiffies;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (time_before(jiffies, (bitmap->last_end_sync
|
2009-12-14 02:49:53 +01:00
|
|
|
+ bitmap->mddev->bitmap_info.daemon_sleep)))
|
2008-02-06 10:39:50 +01:00
|
|
|
return;
|
|
|
|
wait_event(bitmap->mddev->recovery_wait,
|
|
|
|
atomic_read(&bitmap->mddev->recovery_active) == 0);
|
|
|
|
|
2009-03-31 05:33:13 +02:00
|
|
|
bitmap->mddev->curr_resync_completed = bitmap->mddev->curr_resync;
|
|
|
|
set_bit(MD_CHANGE_CLEAN, &bitmap->mddev->flags);
|
2008-02-06 10:39:50 +01:00
|
|
|
sector &= ~((1ULL << CHUNK_BLOCK_SHIFT(bitmap)) - 1);
|
|
|
|
s = 0;
|
|
|
|
while (s < sector && s < bitmap->mddev->resync_max_sectors) {
|
|
|
|
bitmap_end_sync(bitmap, s, &blocks, 0);
|
|
|
|
s += blocks;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2008-02-06 10:39:50 +01:00
|
|
|
bitmap->last_end_sync = jiffies;
|
2009-04-14 08:28:34 +02:00
|
|
|
sysfs_notify(&bitmap->mddev->kobj, NULL, "sync_completed");
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2005-09-10 01:23:44 +02:00
|
|
|
static void bitmap_set_memory_bits(struct bitmap *bitmap, sector_t offset, int needed)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
/* For each chunk covered by any of these sectors, set the
|
2005-08-04 21:53:33 +02:00
|
|
|
* counter to 1 and set resync_needed. They should all
|
2005-06-22 02:17:14 +02:00
|
|
|
* be 0 at this point
|
|
|
|
*/
|
2005-08-04 21:53:33 +02:00
|
|
|
|
|
|
|
int secs;
|
|
|
|
bitmap_counter_t *bmc;
|
|
|
|
spin_lock_irq(&bitmap->lock);
|
|
|
|
bmc = bitmap_get_counter(bitmap, offset, &secs, 1);
|
|
|
|
if (!bmc) {
|
2005-06-22 02:17:14 +02:00
|
|
|
spin_unlock_irq(&bitmap->lock);
|
2005-08-04 21:53:33 +02:00
|
|
|
return;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
2005-08-04 21:53:33 +02:00
|
|
|
if (! *bmc) {
|
|
|
|
struct page *page;
|
2005-09-10 01:23:44 +02:00
|
|
|
*bmc = 1 | (needed?NEEDED_MASK:0);
|
2005-08-04 21:53:33 +02:00
|
|
|
bitmap_count_page(bitmap, offset, 1);
|
|
|
|
page = filemap_get_page(bitmap, offset >> CHUNK_BLOCK_SHIFT(bitmap));
|
|
|
|
set_page_attr(bitmap, page, BITMAP_PAGE_CLEAN);
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&bitmap->lock);
|
2008-03-04 23:29:30 +01:00
|
|
|
bitmap->allclean = 0;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2006-10-03 10:15:49 +02:00
|
|
|
/* dirty the memory and file bits for bitmap chunks "s" to "e" */
|
|
|
|
void bitmap_dirty_bits(struct bitmap *bitmap, unsigned long s, unsigned long e)
|
|
|
|
{
|
|
|
|
unsigned long chunk;
|
|
|
|
|
|
|
|
for (chunk = s; chunk <= e; chunk++) {
|
2009-05-07 04:49:06 +02:00
|
|
|
sector_t sec = (sector_t)chunk << CHUNK_BLOCK_SHIFT(bitmap);
|
2006-10-03 10:15:49 +02:00
|
|
|
bitmap_set_memory_bits(bitmap, sec, 1);
|
|
|
|
bitmap_file_set_bit(bitmap, sec);
|
2009-12-14 02:49:56 +01:00
|
|
|
if (sec < bitmap->mddev->recovery_cp)
|
|
|
|
/* We are asserting that the array is dirty,
|
|
|
|
* so move the recovery_cp address back so
|
|
|
|
* that it is obvious that it is dirty
|
|
|
|
*/
|
|
|
|
bitmap->mddev->recovery_cp = sec;
|
2006-10-03 10:15:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-08-04 21:53:35 +02:00
|
|
|
/*
|
|
|
|
* flush out any pending updates
|
|
|
|
*/
|
|
|
|
void bitmap_flush(mddev_t *mddev)
|
|
|
|
{
|
|
|
|
struct bitmap *bitmap = mddev->bitmap;
|
2009-12-14 02:49:53 +01:00
|
|
|
long sleep;
|
2005-08-04 21:53:35 +02:00
|
|
|
|
|
|
|
if (!bitmap) /* there was no bitmap */
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* run the daemon_work three time to ensure everything is flushed
|
|
|
|
* that can be
|
|
|
|
*/
|
2009-12-14 02:49:53 +01:00
|
|
|
sleep = mddev->bitmap_info.daemon_sleep * 2;
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->daemon_lastrun -= sleep;
|
2009-12-14 02:49:46 +01:00
|
|
|
bitmap_daemon_work(mddev);
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->daemon_lastrun -= sleep;
|
2009-12-14 02:49:46 +01:00
|
|
|
bitmap_daemon_work(mddev);
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->daemon_lastrun -= sleep;
|
2009-12-14 02:49:46 +01:00
|
|
|
bitmap_daemon_work(mddev);
|
2005-08-04 21:53:35 +02:00
|
|
|
bitmap_update_sb(bitmap);
|
|
|
|
}
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
/*
|
|
|
|
* free memory that was allocated
|
|
|
|
*/
|
2005-09-10 01:23:50 +02:00
|
|
|
static void bitmap_free(struct bitmap *bitmap)
|
2005-06-22 02:17:14 +02:00
|
|
|
{
|
|
|
|
unsigned long k, pages;
|
|
|
|
struct bitmap_page *bp;
|
|
|
|
|
|
|
|
if (!bitmap) /* there was no bitmap */
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* release the bitmap file and kill the daemon */
|
|
|
|
bitmap_file_put(bitmap);
|
|
|
|
|
|
|
|
bp = bitmap->bp;
|
|
|
|
pages = bitmap->pages;
|
|
|
|
|
|
|
|
/* free all allocated memory */
|
|
|
|
|
|
|
|
if (bp) /* deallocate the page memory */
|
|
|
|
for (k = 0; k < pages; k++)
|
|
|
|
if (bp[k].map && !bp[k].hijacked)
|
|
|
|
kfree(bp[k].map);
|
|
|
|
kfree(bp);
|
|
|
|
kfree(bitmap);
|
|
|
|
}
|
2009-12-14 02:49:46 +01:00
|
|
|
|
2005-09-10 01:23:50 +02:00
|
|
|
void bitmap_destroy(mddev_t *mddev)
|
|
|
|
{
|
|
|
|
struct bitmap *bitmap = mddev->bitmap;
|
|
|
|
|
|
|
|
if (!bitmap) /* there was no bitmap */
|
|
|
|
return;
|
|
|
|
|
2009-12-14 02:49:52 +01:00
|
|
|
mutex_lock(&mddev->bitmap_info.mutex);
|
2005-09-10 01:23:50 +02:00
|
|
|
mddev->bitmap = NULL; /* disconnect from the md device */
|
2009-12-14 02:49:52 +01:00
|
|
|
mutex_unlock(&mddev->bitmap_info.mutex);
|
2006-01-06 09:20:16 +01:00
|
|
|
if (mddev->thread)
|
|
|
|
mddev->thread->timeout = MAX_SCHEDULE_TIMEOUT;
|
2005-09-10 01:23:50 +02:00
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
if (bitmap->sysfs_can_clear)
|
|
|
|
sysfs_put(bitmap->sysfs_can_clear);
|
|
|
|
|
2005-09-10 01:23:50 +02:00
|
|
|
bitmap_free(bitmap);
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* initialize the bitmap structure
|
|
|
|
* if this returns an error, bitmap_destroy must be called to do clean up
|
|
|
|
*/
|
|
|
|
int bitmap_create(mddev_t *mddev)
|
|
|
|
{
|
|
|
|
struct bitmap *bitmap;
|
2009-04-20 03:50:24 +02:00
|
|
|
sector_t blocks = mddev->resync_max_sectors;
|
2005-06-22 02:17:14 +02:00
|
|
|
unsigned long chunks;
|
|
|
|
unsigned long pages;
|
2009-12-14 02:49:52 +01:00
|
|
|
struct file *file = mddev->bitmap_info.file;
|
2005-06-22 02:17:14 +02:00
|
|
|
int err;
|
2005-09-10 01:23:44 +02:00
|
|
|
sector_t start;
|
2009-12-14 02:49:56 +01:00
|
|
|
struct sysfs_dirent *bm;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2006-10-11 10:22:26 +02:00
|
|
|
BUILD_BUG_ON(sizeof(bitmap_super_t) != 256);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2009-12-14 02:49:52 +01:00
|
|
|
if (!file && !mddev->bitmap_info.offset) /* bitmap disabled, nothing to do */
|
2005-06-22 02:17:14 +02:00
|
|
|
return 0;
|
|
|
|
|
2009-12-14 02:49:52 +01:00
|
|
|
BUG_ON(file && mddev->bitmap_info.offset);
|
2005-06-22 02:17:27 +02:00
|
|
|
|
2006-01-06 09:20:32 +01:00
|
|
|
bitmap = kzalloc(sizeof(*bitmap), GFP_KERNEL);
|
2005-06-22 02:17:14 +02:00
|
|
|
if (!bitmap)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
spin_lock_init(&bitmap->lock);
|
2006-06-26 09:27:49 +02:00
|
|
|
atomic_set(&bitmap->pending_writes, 0);
|
|
|
|
init_waitqueue_head(&bitmap->write_wait);
|
2007-02-08 23:20:37 +01:00
|
|
|
init_waitqueue_head(&bitmap->overflow_wait);
|
2010-03-31 02:21:44 +02:00
|
|
|
init_waitqueue_head(&bitmap->behind_wait);
|
2006-06-26 09:27:49 +02:00
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->mddev = mddev;
|
|
|
|
|
2010-03-30 20:31:26 +02:00
|
|
|
bm = sysfs_get_dirent(mddev->kobj.sd, NULL, "bitmap");
|
2009-12-14 02:49:56 +01:00
|
|
|
if (bm) {
|
2010-03-30 20:31:26 +02:00
|
|
|
bitmap->sysfs_can_clear = sysfs_get_dirent(bm, NULL, "can_clear");
|
2009-12-14 02:49:56 +01:00
|
|
|
sysfs_put(bm);
|
|
|
|
} else
|
|
|
|
bitmap->sysfs_can_clear = NULL;
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->file = file;
|
2006-06-26 09:27:49 +02:00
|
|
|
if (file) {
|
|
|
|
get_file(file);
|
2009-10-16 06:56:01 +02:00
|
|
|
/* As future accesses to this file will use bmap,
|
|
|
|
* and bypass the page cache, we must sync the file
|
|
|
|
* first.
|
|
|
|
*/
|
2010-03-22 17:32:25 +01:00
|
|
|
vfs_fsync(file, 1);
|
2006-06-26 09:27:49 +02:00
|
|
|
}
|
2009-12-14 02:49:53 +01:00
|
|
|
/* read superblock from bitmap file (this sets mddev->bitmap_info.chunksize) */
|
2009-12-14 02:49:56 +01:00
|
|
|
if (!mddev->bitmap_info.external)
|
|
|
|
err = bitmap_read_sb(bitmap);
|
|
|
|
else {
|
|
|
|
err = 0;
|
|
|
|
if (mddev->bitmap_info.chunksize == 0 ||
|
|
|
|
mddev->bitmap_info.daemon_sleep == 0)
|
|
|
|
/* chunksize and time_base need to be
|
|
|
|
* set first. */
|
|
|
|
err = -EINVAL;
|
|
|
|
}
|
2005-06-22 02:17:14 +02:00
|
|
|
if (err)
|
2005-09-10 01:23:50 +02:00
|
|
|
goto error;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
bitmap->daemon_lastrun = jiffies;
|
2009-12-14 02:49:53 +01:00
|
|
|
bitmap->chunkshift = ffz(~mddev->bitmap_info.chunksize);
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* now that chunksize and chunkshift are set, we can use these macros */
|
2009-04-20 03:50:24 +02:00
|
|
|
chunks = (blocks + CHUNK_BLOCK_RATIO(bitmap) - 1) >>
|
|
|
|
CHUNK_BLOCK_SHIFT(bitmap);
|
2005-06-22 02:17:14 +02:00
|
|
|
pages = (chunks + PAGE_COUNTER_RATIO - 1) / PAGE_COUNTER_RATIO;
|
|
|
|
|
|
|
|
BUG_ON(!pages);
|
|
|
|
|
|
|
|
bitmap->chunks = chunks;
|
|
|
|
bitmap->pages = pages;
|
|
|
|
bitmap->missing_pages = pages;
|
|
|
|
bitmap->counter_bits = COUNTER_BITS;
|
|
|
|
|
|
|
|
bitmap->syncchunk = ~0UL;
|
|
|
|
|
2005-07-27 20:45:17 +02:00
|
|
|
#ifdef INJECT_FATAL_FAULT_1
|
2005-06-22 02:17:14 +02:00
|
|
|
bitmap->bp = NULL;
|
|
|
|
#else
|
2006-01-06 09:20:32 +01:00
|
|
|
bitmap->bp = kzalloc(pages * sizeof(*bitmap->bp), GFP_KERNEL);
|
2005-06-22 02:17:14 +02:00
|
|
|
#endif
|
2005-09-10 01:23:50 +02:00
|
|
|
err = -ENOMEM;
|
2005-06-22 02:17:14 +02:00
|
|
|
if (!bitmap->bp)
|
2005-09-10 01:23:50 +02:00
|
|
|
goto error;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
/* now that we have some pages available, initialize the in-memory
|
|
|
|
* bitmap from the on-disk bitmap */
|
2005-09-10 01:23:44 +02:00
|
|
|
start = 0;
|
|
|
|
if (mddev->degraded == 0
|
|
|
|
|| bitmap->events_cleared == mddev->events)
|
|
|
|
/* no need to keep dirty bits to optimise a re-add of a missing device */
|
|
|
|
start = mddev->recovery_cp;
|
|
|
|
err = bitmap_init_from_disk(bitmap, start);
|
2005-08-04 21:53:33 +02:00
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
if (err)
|
2005-09-10 01:23:50 +02:00
|
|
|
goto error;
|
2005-06-22 02:17:14 +02:00
|
|
|
|
|
|
|
printk(KERN_INFO "created bitmap (%lu pages) for device %s\n",
|
|
|
|
pages, bmname(bitmap));
|
|
|
|
|
2005-09-10 01:23:50 +02:00
|
|
|
mddev->bitmap = bitmap;
|
|
|
|
|
2009-12-14 02:49:53 +01:00
|
|
|
mddev->thread->timeout = mddev->bitmap_info.daemon_sleep;
|
2009-12-14 02:49:54 +01:00
|
|
|
md_wakeup_thread(mddev->thread);
|
2006-01-06 09:20:16 +01:00
|
|
|
|
2007-07-17 13:06:13 +02:00
|
|
|
bitmap_update_sb(bitmap);
|
|
|
|
|
|
|
|
return (bitmap->flags & BITMAP_WRITE_ERROR) ? -EIO : 0;
|
2005-09-10 01:23:50 +02:00
|
|
|
|
|
|
|
error:
|
|
|
|
bitmap_free(bitmap);
|
|
|
|
return err;
|
2005-06-22 02:17:14 +02:00
|
|
|
}
|
|
|
|
|
2009-12-14 02:49:55 +01:00
|
|
|
static ssize_t
|
|
|
|
location_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
ssize_t len;
|
|
|
|
if (mddev->bitmap_info.file) {
|
|
|
|
len = sprintf(page, "file");
|
|
|
|
} else if (mddev->bitmap_info.offset) {
|
|
|
|
len = sprintf(page, "%+lld", (long long)mddev->bitmap_info.offset);
|
|
|
|
} else
|
|
|
|
len = sprintf(page, "none");
|
|
|
|
len += sprintf(page+len, "\n");
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
location_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (mddev->pers) {
|
|
|
|
if (!mddev->pers->quiesce)
|
|
|
|
return -EBUSY;
|
|
|
|
if (mddev->recovery || mddev->sync_thread)
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mddev->bitmap || mddev->bitmap_info.file ||
|
|
|
|
mddev->bitmap_info.offset) {
|
|
|
|
/* bitmap already configured. Only option is to clear it */
|
|
|
|
if (strncmp(buf, "none", 4) != 0)
|
|
|
|
return -EBUSY;
|
|
|
|
if (mddev->pers) {
|
|
|
|
mddev->pers->quiesce(mddev, 1);
|
|
|
|
bitmap_destroy(mddev);
|
|
|
|
mddev->pers->quiesce(mddev, 0);
|
|
|
|
}
|
|
|
|
mddev->bitmap_info.offset = 0;
|
|
|
|
if (mddev->bitmap_info.file) {
|
|
|
|
struct file *f = mddev->bitmap_info.file;
|
|
|
|
mddev->bitmap_info.file = NULL;
|
|
|
|
restore_bitmap_write_access(f);
|
|
|
|
fput(f);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* No bitmap, OK to set a location */
|
|
|
|
long long offset;
|
|
|
|
if (strncmp(buf, "none", 4) == 0)
|
|
|
|
/* nothing to be done */;
|
|
|
|
else if (strncmp(buf, "file:", 5) == 0) {
|
|
|
|
/* Not supported yet */
|
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
|
|
|
int rv;
|
|
|
|
if (buf[0] == '+')
|
|
|
|
rv = strict_strtoll(buf+1, 10, &offset);
|
|
|
|
else
|
|
|
|
rv = strict_strtoll(buf, 10, &offset);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
if (offset == 0)
|
|
|
|
return -EINVAL;
|
2009-12-14 02:49:56 +01:00
|
|
|
if (mddev->bitmap_info.external == 0 &&
|
|
|
|
mddev->major_version == 0 &&
|
2009-12-14 02:49:55 +01:00
|
|
|
offset != mddev->bitmap_info.default_offset)
|
|
|
|
return -EINVAL;
|
|
|
|
mddev->bitmap_info.offset = offset;
|
|
|
|
if (mddev->pers) {
|
|
|
|
mddev->pers->quiesce(mddev, 1);
|
|
|
|
rv = bitmap_create(mddev);
|
|
|
|
if (rv) {
|
|
|
|
bitmap_destroy(mddev);
|
|
|
|
mddev->bitmap_info.offset = 0;
|
|
|
|
}
|
|
|
|
mddev->pers->quiesce(mddev, 0);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!mddev->external) {
|
|
|
|
/* Ensure new bitmap info is stored in
|
|
|
|
* metadata promptly.
|
|
|
|
*/
|
|
|
|
set_bit(MD_CHANGE_DEVS, &mddev->flags);
|
|
|
|
md_wakeup_thread(mddev->thread);
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_location =
|
|
|
|
__ATTR(location, S_IRUGO|S_IWUSR, location_show, location_store);
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
timeout_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
ssize_t len;
|
|
|
|
unsigned long secs = mddev->bitmap_info.daemon_sleep / HZ;
|
|
|
|
unsigned long jifs = mddev->bitmap_info.daemon_sleep % HZ;
|
|
|
|
|
|
|
|
len = sprintf(page, "%lu", secs);
|
|
|
|
if (jifs)
|
|
|
|
len += sprintf(page+len, ".%03u", jiffies_to_msecs(jifs));
|
|
|
|
len += sprintf(page+len, "\n");
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
timeout_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
/* timeout can be set at any time */
|
|
|
|
unsigned long timeout;
|
|
|
|
int rv = strict_strtoul_scaled(buf, &timeout, 4);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
/* just to make sure we don't overflow... */
|
|
|
|
if (timeout >= LONG_MAX / HZ)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
timeout = timeout * HZ / 10000;
|
|
|
|
|
|
|
|
if (timeout >= MAX_SCHEDULE_TIMEOUT)
|
|
|
|
timeout = MAX_SCHEDULE_TIMEOUT-1;
|
|
|
|
if (timeout < 1)
|
|
|
|
timeout = 1;
|
|
|
|
mddev->bitmap_info.daemon_sleep = timeout;
|
|
|
|
if (mddev->thread) {
|
|
|
|
/* if thread->timeout is MAX_SCHEDULE_TIMEOUT, then
|
|
|
|
* the bitmap is all clean and we don't need to
|
|
|
|
* adjust the timeout right now
|
|
|
|
*/
|
|
|
|
if (mddev->thread->timeout < MAX_SCHEDULE_TIMEOUT) {
|
|
|
|
mddev->thread->timeout = timeout;
|
|
|
|
md_wakeup_thread(mddev->thread);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_timeout =
|
|
|
|
__ATTR(time_base, S_IRUGO|S_IWUSR, timeout_show, timeout_store);
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
backlog_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
return sprintf(page, "%lu\n", mddev->bitmap_info.max_write_behind);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
backlog_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
unsigned long backlog;
|
|
|
|
int rv = strict_strtoul(buf, 10, &backlog);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
if (backlog > COUNTER_MAX)
|
|
|
|
return -EINVAL;
|
|
|
|
mddev->bitmap_info.max_write_behind = backlog;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_backlog =
|
|
|
|
__ATTR(backlog, S_IRUGO|S_IWUSR, backlog_show, backlog_store);
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
chunksize_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
return sprintf(page, "%lu\n", mddev->bitmap_info.chunksize);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
chunksize_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
/* Can only be changed when no bitmap is active */
|
|
|
|
int rv;
|
|
|
|
unsigned long csize;
|
|
|
|
if (mddev->bitmap)
|
|
|
|
return -EBUSY;
|
|
|
|
rv = strict_strtoul(buf, 10, &csize);
|
|
|
|
if (rv)
|
|
|
|
return rv;
|
|
|
|
if (csize < 512 ||
|
|
|
|
!is_power_of_2(csize))
|
|
|
|
return -EINVAL;
|
|
|
|
mddev->bitmap_info.chunksize = csize;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_chunksize =
|
|
|
|
__ATTR(chunksize, S_IRUGO|S_IWUSR, chunksize_show, chunksize_store);
|
|
|
|
|
2009-12-14 02:49:56 +01:00
|
|
|
static ssize_t metadata_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
return sprintf(page, "%s\n", (mddev->bitmap_info.external
|
|
|
|
? "external" : "internal"));
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t metadata_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
if (mddev->bitmap ||
|
|
|
|
mddev->bitmap_info.file ||
|
|
|
|
mddev->bitmap_info.offset)
|
|
|
|
return -EBUSY;
|
|
|
|
if (strncmp(buf, "external", 8) == 0)
|
|
|
|
mddev->bitmap_info.external = 1;
|
|
|
|
else if (strncmp(buf, "internal", 8) == 0)
|
|
|
|
mddev->bitmap_info.external = 0;
|
|
|
|
else
|
|
|
|
return -EINVAL;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_metadata =
|
|
|
|
__ATTR(metadata, S_IRUGO|S_IWUSR, metadata_show, metadata_store);
|
|
|
|
|
|
|
|
static ssize_t can_clear_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
if (mddev->bitmap)
|
|
|
|
len = sprintf(page, "%s\n", (mddev->bitmap->need_sync ?
|
|
|
|
"false" : "true"));
|
|
|
|
else
|
|
|
|
len = sprintf(page, "\n");
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t can_clear_store(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
if (mddev->bitmap == NULL)
|
|
|
|
return -ENOENT;
|
|
|
|
if (strncmp(buf, "false", 5) == 0)
|
|
|
|
mddev->bitmap->need_sync = 1;
|
|
|
|
else if (strncmp(buf, "true", 4) == 0) {
|
|
|
|
if (mddev->degraded)
|
|
|
|
return -EBUSY;
|
|
|
|
mddev->bitmap->need_sync = 0;
|
|
|
|
} else
|
|
|
|
return -EINVAL;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry bitmap_can_clear =
|
|
|
|
__ATTR(can_clear, S_IRUGO|S_IWUSR, can_clear_show, can_clear_store);
|
|
|
|
|
2010-03-08 06:02:37 +01:00
|
|
|
static ssize_t
|
|
|
|
behind_writes_used_show(mddev_t *mddev, char *page)
|
|
|
|
{
|
|
|
|
if (mddev->bitmap == NULL)
|
|
|
|
return sprintf(page, "0\n");
|
|
|
|
return sprintf(page, "%lu\n",
|
|
|
|
mddev->bitmap->behind_writes_used);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t
|
|
|
|
behind_writes_used_reset(mddev_t *mddev, const char *buf, size_t len)
|
|
|
|
{
|
|
|
|
if (mddev->bitmap)
|
|
|
|
mddev->bitmap->behind_writes_used = 0;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct md_sysfs_entry max_backlog_used =
|
|
|
|
__ATTR(max_backlog_used, S_IRUGO | S_IWUSR,
|
|
|
|
behind_writes_used_show, behind_writes_used_reset);
|
|
|
|
|
2009-12-14 02:49:55 +01:00
|
|
|
static struct attribute *md_bitmap_attrs[] = {
|
|
|
|
&bitmap_location.attr,
|
|
|
|
&bitmap_timeout.attr,
|
|
|
|
&bitmap_backlog.attr,
|
|
|
|
&bitmap_chunksize.attr,
|
2009-12-14 02:49:56 +01:00
|
|
|
&bitmap_metadata.attr,
|
|
|
|
&bitmap_can_clear.attr,
|
2010-03-08 06:02:37 +01:00
|
|
|
&max_backlog_used.attr,
|
2009-12-14 02:49:55 +01:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
struct attribute_group md_bitmap_group = {
|
|
|
|
.name = "bitmap",
|
|
|
|
.attrs = md_bitmap_attrs,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2005-06-22 02:17:14 +02:00
|
|
|
/* the bitmap API -- for raid personalities */
|
|
|
|
EXPORT_SYMBOL(bitmap_startwrite);
|
|
|
|
EXPORT_SYMBOL(bitmap_endwrite);
|
|
|
|
EXPORT_SYMBOL(bitmap_start_sync);
|
|
|
|
EXPORT_SYMBOL(bitmap_end_sync);
|
|
|
|
EXPORT_SYMBOL(bitmap_unplug);
|
|
|
|
EXPORT_SYMBOL(bitmap_close_sync);
|
2008-02-06 10:39:50 +01:00
|
|
|
EXPORT_SYMBOL(bitmap_cond_end_sync);
|