Commit d659072f authored by Tao Ma's avatar Tao Ma Committed by Mark Fasheh

[PATCH 1/2] ocfs2: Add group extend for online resize

This patch adds the ability for a userspace program to request an extend of
last cluster group on an Ocfs2 file system. The request is made via ioctl,
OCFS2_IOC_GROUP_EXTEND. This is derived from EXT3_IOC_GROUP_EXTEND, but is
obviously Ocfs2 specific.

tunefs.ocfs2 would call this for an online-resize operation if the last
cluster group isn't full.
Signed-off-by: default avatarTao Ma <tao.ma@oracle.com>
Signed-off-by: default avatarMark Fasheh <mark.fasheh@oracle.com>
parent 7f68fc28
...@@ -21,6 +21,7 @@ ocfs2-objs := \ ...@@ -21,6 +21,7 @@ ocfs2-objs := \
localalloc.o \ localalloc.o \
mmap.o \ mmap.o \
namei.o \ namei.o \
resize.o \
slot_map.o \ slot_map.o \
suballoc.o \ suballoc.o \
super.o \ super.o \
......
...@@ -280,3 +280,64 @@ int ocfs2_read_blocks(struct ocfs2_super *osb, u64 block, int nr, ...@@ -280,3 +280,64 @@ int ocfs2_read_blocks(struct ocfs2_super *osb, u64 block, int nr,
mlog_exit(status); mlog_exit(status);
return status; return status;
} }
/* Check whether the blkno is the super block or one of the backups. */
static void ocfs2_check_super_or_backup(struct super_block *sb,
sector_t blkno)
{
int i;
u64 backup_blkno;
if (blkno == OCFS2_SUPER_BLOCK_BLKNO)
return;
for (i = 0; i < OCFS2_MAX_BACKUP_SUPERBLOCKS; i++) {
backup_blkno = ocfs2_backup_super_blkno(sb, i);
if (backup_blkno == blkno)
return;
}
BUG();
}
/*
* Write super block and backups doesn't need to collaborate with journal,
* so we don't need to lock ip_io_mutex and inode doesn't need to bea passed
* into this function.
*/
int ocfs2_write_super_or_backup(struct ocfs2_super *osb,
struct buffer_head *bh)
{
int ret = 0;
mlog_entry_void();
BUG_ON(buffer_jbd(bh));
ocfs2_check_super_or_backup(osb->sb, bh->b_blocknr);
if (ocfs2_is_hard_readonly(osb) || ocfs2_is_soft_readonly(osb)) {
ret = -EROFS;
goto out;
}
lock_buffer(bh);
set_buffer_uptodate(bh);
/* remove from dirty list before I/O. */
clear_buffer_dirty(bh);
get_bh(bh); /* for end_buffer_write_sync() */
bh->b_end_io = end_buffer_write_sync;
submit_bh(WRITE, bh);
wait_on_buffer(bh);
if (!buffer_uptodate(bh)) {
ret = -EIO;
brelse(bh);
}
out:
mlog_exit(ret);
return ret;
}
...@@ -47,6 +47,8 @@ int ocfs2_read_blocks(struct ocfs2_super *osb, ...@@ -47,6 +47,8 @@ int ocfs2_read_blocks(struct ocfs2_super *osb,
int flags, int flags,
struct inode *inode); struct inode *inode);
int ocfs2_write_super_or_backup(struct ocfs2_super *osb,
struct buffer_head *bh);
#define OCFS2_BH_CACHED 1 #define OCFS2_BH_CACHED 1
#define OCFS2_BH_READAHEAD 8 #define OCFS2_BH_READAHEAD 8
......
...@@ -20,6 +20,7 @@ ...@@ -20,6 +20,7 @@
#include "ocfs2_fs.h" #include "ocfs2_fs.h"
#include "ioctl.h" #include "ioctl.h"
#include "resize.h"
#include <linux/ext2_fs.h> #include <linux/ext2_fs.h>
...@@ -115,6 +116,7 @@ int ocfs2_ioctl(struct inode * inode, struct file * filp, ...@@ -115,6 +116,7 @@ int ocfs2_ioctl(struct inode * inode, struct file * filp,
unsigned int cmd, unsigned long arg) unsigned int cmd, unsigned long arg)
{ {
unsigned int flags; unsigned int flags;
int new_clusters;
int status; int status;
struct ocfs2_space_resv sr; struct ocfs2_space_resv sr;
...@@ -140,6 +142,11 @@ int ocfs2_ioctl(struct inode * inode, struct file * filp, ...@@ -140,6 +142,11 @@ int ocfs2_ioctl(struct inode * inode, struct file * filp,
return -EFAULT; return -EFAULT;
return ocfs2_change_file_space(filp, cmd, &sr); return ocfs2_change_file_space(filp, cmd, &sr);
case OCFS2_IOC_GROUP_EXTEND:
if (get_user(new_clusters, (int __user *)arg))
return -EFAULT;
return ocfs2_group_extend(inode, new_clusters);
default: default:
return -ENOTTY; return -ENOTTY;
} }
...@@ -162,6 +169,7 @@ long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg) ...@@ -162,6 +169,7 @@ long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg)
case OCFS2_IOC_RESVSP64: case OCFS2_IOC_RESVSP64:
case OCFS2_IOC_UNRESVSP: case OCFS2_IOC_UNRESVSP:
case OCFS2_IOC_UNRESVSP64: case OCFS2_IOC_UNRESVSP64:
case OCFS2_IOC_GROUP_EXTEND:
break; break;
default: default:
return -ENOIOCTLCMD; return -ENOIOCTLCMD;
......
...@@ -278,6 +278,9 @@ int ocfs2_journal_dirty_data(handle_t *handle, ...@@ -278,6 +278,9 @@ int ocfs2_journal_dirty_data(handle_t *handle,
/* simple file updates like chmod, etc. */ /* simple file updates like chmod, etc. */
#define OCFS2_INODE_UPDATE_CREDITS 1 #define OCFS2_INODE_UPDATE_CREDITS 1
/* group extend. inode update and last group update. */
#define OCFS2_GROUP_EXTEND_CREDITS (OCFS2_INODE_UPDATE_CREDITS + 1)
/* get one bit out of a suballocator: dinode + group descriptor + /* get one bit out of a suballocator: dinode + group descriptor +
* prev. group desc. if we relink. */ * prev. group desc. if we relink. */
#define OCFS2_SUBALLOC_ALLOC (3) #define OCFS2_SUBALLOC_ALLOC (3)
......
...@@ -231,6 +231,8 @@ struct ocfs2_space_resv { ...@@ -231,6 +231,8 @@ struct ocfs2_space_resv {
#define OCFS2_IOC_RESVSP64 _IOW ('X', 42, struct ocfs2_space_resv) #define OCFS2_IOC_RESVSP64 _IOW ('X', 42, struct ocfs2_space_resv)
#define OCFS2_IOC_UNRESVSP64 _IOW ('X', 43, struct ocfs2_space_resv) #define OCFS2_IOC_UNRESVSP64 _IOW ('X', 43, struct ocfs2_space_resv)
#define OCFS2_IOC_GROUP_EXTEND _IOW('o', 1, int)
/* /*
* Journal Flags (ocfs2_dinode.id1.journal1.i_flags) * Journal Flags (ocfs2_dinode.id1.journal1.i_flags)
*/ */
......
This diff is collapsed.
/* -*- mode: c; c-basic-offset: 8; -*-
* vim: noexpandtab sw=8 ts=8 sts=0:
*
* resize.h
*
* Function prototypes
*
* Copyright (C) 2007 Oracle. All rights reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 021110-1307, USA.
*/
#ifndef OCFS2_RESIZE_H
#define OCFS2_RESIZE_H
int ocfs2_group_extend(struct inode * inode, int new_clusters);
#endif /* OCFS2_RESIZE_H */
...@@ -101,8 +101,6 @@ static inline int ocfs2_block_group_reasonably_empty(struct ocfs2_group_desc *bg ...@@ -101,8 +101,6 @@ static inline int ocfs2_block_group_reasonably_empty(struct ocfs2_group_desc *bg
static inline u32 ocfs2_desc_bitmap_to_cluster_off(struct inode *inode, static inline u32 ocfs2_desc_bitmap_to_cluster_off(struct inode *inode,
u64 bg_blkno, u64 bg_blkno,
u16 bg_bit_off); u16 bg_bit_off);
static inline u64 ocfs2_which_cluster_group(struct inode *inode,
u32 cluster);
static inline void ocfs2_block_to_cluster_group(struct inode *inode, static inline void ocfs2_block_to_cluster_group(struct inode *inode,
u64 data_blkno, u64 data_blkno,
u64 *bg_blkno, u64 *bg_blkno,
...@@ -131,7 +129,7 @@ static u32 ocfs2_bits_per_group(struct ocfs2_chain_list *cl) ...@@ -131,7 +129,7 @@ static u32 ocfs2_bits_per_group(struct ocfs2_chain_list *cl)
} }
/* somewhat more expensive than our other checks, so use sparingly. */ /* somewhat more expensive than our other checks, so use sparingly. */
static int ocfs2_check_group_descriptor(struct super_block *sb, int ocfs2_check_group_descriptor(struct super_block *sb,
struct ocfs2_dinode *di, struct ocfs2_dinode *di,
struct ocfs2_group_desc *gd) struct ocfs2_group_desc *gd)
{ {
...@@ -1443,8 +1441,7 @@ static inline u32 ocfs2_desc_bitmap_to_cluster_off(struct inode *inode, ...@@ -1443,8 +1441,7 @@ static inline u32 ocfs2_desc_bitmap_to_cluster_off(struct inode *inode,
/* given a cluster offset, calculate which block group it belongs to /* given a cluster offset, calculate which block group it belongs to
* and return that block offset. */ * and return that block offset. */
static inline u64 ocfs2_which_cluster_group(struct inode *inode, u64 ocfs2_which_cluster_group(struct inode *inode, u32 cluster)
u32 cluster)
{ {
struct ocfs2_super *osb = OCFS2_SB(inode->i_sb); struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
u32 group_no; u32 group_no;
......
...@@ -147,4 +147,12 @@ static inline int ocfs2_is_cluster_bitmap(struct inode *inode) ...@@ -147,4 +147,12 @@ static inline int ocfs2_is_cluster_bitmap(struct inode *inode)
int ocfs2_reserve_cluster_bitmap_bits(struct ocfs2_super *osb, int ocfs2_reserve_cluster_bitmap_bits(struct ocfs2_super *osb,
struct ocfs2_alloc_context *ac); struct ocfs2_alloc_context *ac);
/* given a cluster offset, calculate which block group it belongs to
* and return that block offset. */
u64 ocfs2_which_cluster_group(struct inode *inode, u32 cluster);
/* somewhat more expensive than our other checks, so use sparingly. */
int ocfs2_check_group_descriptor(struct super_block *sb,
struct ocfs2_dinode *di,
struct ocfs2_group_desc *gd);
#endif /* _CHAINALLOC_H_ */ #endif /* _CHAINALLOC_H_ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment