Merge branch 'upstream-linus' of git://oss.oracle.com/home/sourcebo/git/ocfs2
This commit is contained in:
commit
d1ffa5669c
@ -320,6 +320,7 @@ static struct config_item_type simple_children_type = {
|
||||
.ct_item_ops = &simple_children_item_ops,
|
||||
.ct_group_ops = &simple_children_group_ops,
|
||||
.ct_attrs = simple_children_attrs,
|
||||
.ct_owner = THIS_MODULE,
|
||||
};
|
||||
|
||||
static struct configfs_subsystem simple_children_subsys = {
|
||||
@ -403,6 +404,7 @@ static struct config_item_type group_children_type = {
|
||||
.ct_item_ops = &group_children_item_ops,
|
||||
.ct_group_ops = &group_children_group_ops,
|
||||
.ct_attrs = group_children_attrs,
|
||||
.ct_owner = THIS_MODULE,
|
||||
};
|
||||
|
||||
static struct configfs_subsystem group_children_subsys = {
|
||||
|
@ -35,6 +35,7 @@ Features which OCFS2 does not support yet:
|
||||
be cluster coherent.
|
||||
- quotas
|
||||
- cluster aware flock
|
||||
- cluster aware lockf
|
||||
- Directory change notification (F_NOTIFY)
|
||||
- Distributed Caching (F_SETLEASE/F_GETLEASE/break_lease)
|
||||
- POSIX ACLs
|
||||
|
@ -557,7 +557,8 @@ S: Supported
|
||||
|
||||
CONFIGFS
|
||||
P: Joel Becker
|
||||
M: Joel Becker <joel.becker@oracle.com>
|
||||
M: joel.becker@oracle.com
|
||||
L: linux-kernel@vger.kernel.org
|
||||
S: Supported
|
||||
|
||||
CIRRUS LOGIC GENERIC FBDEV DRIVER
|
||||
|
@ -883,8 +883,6 @@ config CONFIGFS_FS
|
||||
Both sysfs and configfs can and should exist together on the
|
||||
same system. One is not a replacement for the other.
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
endmenu
|
||||
|
||||
menu "Miscellaneous filesystems"
|
||||
|
@ -36,6 +36,7 @@ struct configfs_dirent {
|
||||
int s_type;
|
||||
umode_t s_mode;
|
||||
struct dentry * s_dentry;
|
||||
struct iattr * s_iattr;
|
||||
};
|
||||
|
||||
#define CONFIGFS_ROOT 0x0001
|
||||
@ -48,10 +49,11 @@ struct configfs_dirent {
|
||||
#define CONFIGFS_NOT_PINNED (CONFIGFS_ITEM_ATTR)
|
||||
|
||||
extern struct vfsmount * configfs_mount;
|
||||
extern kmem_cache_t *configfs_dir_cachep;
|
||||
|
||||
extern int configfs_is_root(struct config_item *item);
|
||||
|
||||
extern struct inode * configfs_new_inode(mode_t mode);
|
||||
extern struct inode * configfs_new_inode(mode_t mode, struct configfs_dirent *);
|
||||
extern int configfs_create(struct dentry *, int mode, int (*init)(struct inode *));
|
||||
|
||||
extern int configfs_create_file(struct config_item *, const struct configfs_attribute *);
|
||||
@ -63,6 +65,7 @@ extern void configfs_hash_and_remove(struct dentry * dir, const char * name);
|
||||
|
||||
extern const unsigned char * configfs_get_name(struct configfs_dirent *sd);
|
||||
extern void configfs_drop_dentry(struct configfs_dirent *sd, struct dentry *parent);
|
||||
extern int configfs_setattr(struct dentry *dentry, struct iattr *iattr);
|
||||
|
||||
extern int configfs_pin_fs(void);
|
||||
extern void configfs_release_fs(void);
|
||||
@ -120,8 +123,10 @@ static inline struct config_item *configfs_get_config_item(struct dentry *dentry
|
||||
|
||||
static inline void release_configfs_dirent(struct configfs_dirent * sd)
|
||||
{
|
||||
if (!(sd->s_type & CONFIGFS_ROOT))
|
||||
kfree(sd);
|
||||
if (!(sd->s_type & CONFIGFS_ROOT)) {
|
||||
kfree(sd->s_iattr);
|
||||
kmem_cache_free(configfs_dir_cachep, sd);
|
||||
}
|
||||
}
|
||||
|
||||
static inline struct configfs_dirent * configfs_get(struct configfs_dirent * sd)
|
||||
|
@ -72,7 +72,7 @@ static struct configfs_dirent *configfs_new_dirent(struct configfs_dirent * pare
|
||||
{
|
||||
struct configfs_dirent * sd;
|
||||
|
||||
sd = kmalloc(sizeof(*sd), GFP_KERNEL);
|
||||
sd = kmem_cache_alloc(configfs_dir_cachep, GFP_KERNEL);
|
||||
if (!sd)
|
||||
return NULL;
|
||||
|
||||
@ -136,13 +136,19 @@ static int create_dir(struct config_item * k, struct dentry * p,
|
||||
int error;
|
||||
umode_t mode = S_IFDIR| S_IRWXU | S_IRUGO | S_IXUGO;
|
||||
|
||||
error = configfs_create(d, mode, init_dir);
|
||||
error = configfs_make_dirent(p->d_fsdata, d, k, mode,
|
||||
CONFIGFS_DIR);
|
||||
if (!error) {
|
||||
error = configfs_make_dirent(p->d_fsdata, d, k, mode,
|
||||
CONFIGFS_DIR);
|
||||
error = configfs_create(d, mode, init_dir);
|
||||
if (!error) {
|
||||
p->d_inode->i_nlink++;
|
||||
(d)->d_op = &configfs_dentry_ops;
|
||||
} else {
|
||||
struct configfs_dirent *sd = d->d_fsdata;
|
||||
if (sd) {
|
||||
list_del_init(&sd->s_sibling);
|
||||
configfs_put(sd);
|
||||
}
|
||||
}
|
||||
}
|
||||
return error;
|
||||
@ -182,12 +188,19 @@ int configfs_create_link(struct configfs_symlink *sl,
|
||||
int err = 0;
|
||||
umode_t mode = S_IFLNK | S_IRWXUGO;
|
||||
|
||||
err = configfs_create(dentry, mode, init_symlink);
|
||||
err = configfs_make_dirent(parent->d_fsdata, dentry, sl, mode,
|
||||
CONFIGFS_ITEM_LINK);
|
||||
if (!err) {
|
||||
err = configfs_make_dirent(parent->d_fsdata, dentry, sl,
|
||||
mode, CONFIGFS_ITEM_LINK);
|
||||
err = configfs_create(dentry, mode, init_symlink);
|
||||
if (!err)
|
||||
dentry->d_op = &configfs_dentry_ops;
|
||||
else {
|
||||
struct configfs_dirent *sd = dentry->d_fsdata;
|
||||
if (sd) {
|
||||
list_del_init(&sd->s_sibling);
|
||||
configfs_put(sd);
|
||||
}
|
||||
}
|
||||
}
|
||||
return err;
|
||||
}
|
||||
@ -241,13 +254,15 @@ static int configfs_attach_attr(struct configfs_dirent * sd, struct dentry * den
|
||||
struct configfs_attribute * attr = sd->s_element;
|
||||
int error;
|
||||
|
||||
error = configfs_create(dentry, (attr->ca_mode & S_IALLUGO) | S_IFREG, init_file);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
dentry->d_op = &configfs_dentry_ops;
|
||||
dentry->d_fsdata = configfs_get(sd);
|
||||
sd->s_dentry = dentry;
|
||||
error = configfs_create(dentry, (attr->ca_mode & S_IALLUGO) | S_IFREG, init_file);
|
||||
if (error) {
|
||||
configfs_put(sd);
|
||||
return error;
|
||||
}
|
||||
|
||||
dentry->d_op = &configfs_dentry_ops;
|
||||
d_rehash(dentry);
|
||||
|
||||
return 0;
|
||||
@ -839,6 +854,7 @@ struct inode_operations configfs_dir_inode_operations = {
|
||||
.symlink = configfs_symlink,
|
||||
.unlink = configfs_unlink,
|
||||
.lookup = configfs_lookup,
|
||||
.setattr = configfs_setattr,
|
||||
};
|
||||
|
||||
#if 0
|
||||
|
@ -26,7 +26,6 @@
|
||||
|
||||
#include <linux/fs.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/dnotify.h>
|
||||
#include <linux/slab.h>
|
||||
#include <asm/uaccess.h>
|
||||
#include <asm/semaphore.h>
|
||||
@ -150,7 +149,7 @@ out:
|
||||
/**
|
||||
* fill_write_buffer - copy buffer from userspace.
|
||||
* @buffer: data buffer for file.
|
||||
* @userbuf: data from user.
|
||||
* @buf: data from user.
|
||||
* @count: number of bytes in @userbuf.
|
||||
*
|
||||
* Allocate @buffer->page if it hasn't been already, then
|
||||
@ -177,8 +176,9 @@ fill_write_buffer(struct configfs_buffer * buffer, const char __user * buf, size
|
||||
|
||||
/**
|
||||
* flush_write_buffer - push buffer to config_item.
|
||||
* @file: file pointer.
|
||||
* @dentry: dentry to the attribute
|
||||
* @buffer: data buffer for file.
|
||||
* @count: number of bytes
|
||||
*
|
||||
* Get the correct pointers for the config_item and the attribute we're
|
||||
* dealing with, then call the store() method for the attribute,
|
||||
@ -217,15 +217,16 @@ static ssize_t
|
||||
configfs_write_file(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
|
||||
{
|
||||
struct configfs_buffer * buffer = file->private_data;
|
||||
ssize_t len;
|
||||
|
||||
down(&buffer->sem);
|
||||
count = fill_write_buffer(buffer,buf,count);
|
||||
if (count > 0)
|
||||
count = flush_write_buffer(file->f_dentry,buffer,count);
|
||||
if (count > 0)
|
||||
*ppos += count;
|
||||
len = fill_write_buffer(buffer, buf, count);
|
||||
if (len > 0)
|
||||
len = flush_write_buffer(file->f_dentry, buffer, count);
|
||||
if (len > 0)
|
||||
*ppos += len;
|
||||
up(&buffer->sem);
|
||||
return count;
|
||||
return len;
|
||||
}
|
||||
|
||||
static int check_perm(struct inode * inode, struct file * file)
|
||||
|
@ -31,6 +31,7 @@
|
||||
#include <linux/pagemap.h>
|
||||
#include <linux/namei.h>
|
||||
#include <linux/backing-dev.h>
|
||||
#include <linux/capability.h>
|
||||
|
||||
#include <linux/configfs.h>
|
||||
#include "configfs_internal.h"
|
||||
@ -48,18 +49,107 @@ static struct backing_dev_info configfs_backing_dev_info = {
|
||||
.capabilities = BDI_CAP_NO_ACCT_DIRTY | BDI_CAP_NO_WRITEBACK,
|
||||
};
|
||||
|
||||
struct inode * configfs_new_inode(mode_t mode)
|
||||
static struct inode_operations configfs_inode_operations ={
|
||||
.setattr = configfs_setattr,
|
||||
};
|
||||
|
||||
int configfs_setattr(struct dentry * dentry, struct iattr * iattr)
|
||||
{
|
||||
struct inode * inode = dentry->d_inode;
|
||||
struct configfs_dirent * sd = dentry->d_fsdata;
|
||||
struct iattr * sd_iattr;
|
||||
unsigned int ia_valid = iattr->ia_valid;
|
||||
int error;
|
||||
|
||||
if (!sd)
|
||||
return -EINVAL;
|
||||
|
||||
sd_iattr = sd->s_iattr;
|
||||
|
||||
error = inode_change_ok(inode, iattr);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
error = inode_setattr(inode, iattr);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
if (!sd_iattr) {
|
||||
/* setting attributes for the first time, allocate now */
|
||||
sd_iattr = kmalloc(sizeof(struct iattr), GFP_KERNEL);
|
||||
if (!sd_iattr)
|
||||
return -ENOMEM;
|
||||
/* assign default attributes */
|
||||
memset(sd_iattr, 0, sizeof(struct iattr));
|
||||
sd_iattr->ia_mode = sd->s_mode;
|
||||
sd_iattr->ia_uid = 0;
|
||||
sd_iattr->ia_gid = 0;
|
||||
sd_iattr->ia_atime = sd_iattr->ia_mtime = sd_iattr->ia_ctime = CURRENT_TIME;
|
||||
sd->s_iattr = sd_iattr;
|
||||
}
|
||||
|
||||
/* attributes were changed atleast once in past */
|
||||
|
||||
if (ia_valid & ATTR_UID)
|
||||
sd_iattr->ia_uid = iattr->ia_uid;
|
||||
if (ia_valid & ATTR_GID)
|
||||
sd_iattr->ia_gid = iattr->ia_gid;
|
||||
if (ia_valid & ATTR_ATIME)
|
||||
sd_iattr->ia_atime = timespec_trunc(iattr->ia_atime,
|
||||
inode->i_sb->s_time_gran);
|
||||
if (ia_valid & ATTR_MTIME)
|
||||
sd_iattr->ia_mtime = timespec_trunc(iattr->ia_mtime,
|
||||
inode->i_sb->s_time_gran);
|
||||
if (ia_valid & ATTR_CTIME)
|
||||
sd_iattr->ia_ctime = timespec_trunc(iattr->ia_ctime,
|
||||
inode->i_sb->s_time_gran);
|
||||
if (ia_valid & ATTR_MODE) {
|
||||
umode_t mode = iattr->ia_mode;
|
||||
|
||||
if (!in_group_p(inode->i_gid) && !capable(CAP_FSETID))
|
||||
mode &= ~S_ISGID;
|
||||
sd_iattr->ia_mode = sd->s_mode = mode;
|
||||
}
|
||||
|
||||
return error;
|
||||
}
|
||||
|
||||
static inline void set_default_inode_attr(struct inode * inode, mode_t mode)
|
||||
{
|
||||
inode->i_mode = mode;
|
||||
inode->i_uid = 0;
|
||||
inode->i_gid = 0;
|
||||
inode->i_atime = inode->i_mtime = inode->i_ctime = CURRENT_TIME;
|
||||
}
|
||||
|
||||
static inline void set_inode_attr(struct inode * inode, struct iattr * iattr)
|
||||
{
|
||||
inode->i_mode = iattr->ia_mode;
|
||||
inode->i_uid = iattr->ia_uid;
|
||||
inode->i_gid = iattr->ia_gid;
|
||||
inode->i_atime = iattr->ia_atime;
|
||||
inode->i_mtime = iattr->ia_mtime;
|
||||
inode->i_ctime = iattr->ia_ctime;
|
||||
}
|
||||
|
||||
struct inode * configfs_new_inode(mode_t mode, struct configfs_dirent * sd)
|
||||
{
|
||||
struct inode * inode = new_inode(configfs_sb);
|
||||
if (inode) {
|
||||
inode->i_mode = mode;
|
||||
inode->i_uid = 0;
|
||||
inode->i_gid = 0;
|
||||
inode->i_blksize = PAGE_CACHE_SIZE;
|
||||
inode->i_blocks = 0;
|
||||
inode->i_atime = inode->i_mtime = inode->i_ctime = CURRENT_TIME;
|
||||
inode->i_mapping->a_ops = &configfs_aops;
|
||||
inode->i_mapping->backing_dev_info = &configfs_backing_dev_info;
|
||||
inode->i_op = &configfs_inode_operations;
|
||||
|
||||
if (sd->s_iattr) {
|
||||
/* sysfs_dirent has non-default attributes
|
||||
* get them for the new inode from persistent copy
|
||||
* in sysfs_dirent
|
||||
*/
|
||||
set_inode_attr(inode, sd->s_iattr);
|
||||
} else
|
||||
set_default_inode_attr(inode, mode);
|
||||
}
|
||||
return inode;
|
||||
}
|
||||
@ -70,7 +160,8 @@ int configfs_create(struct dentry * dentry, int mode, int (*init)(struct inode *
|
||||
struct inode * inode = NULL;
|
||||
if (dentry) {
|
||||
if (!dentry->d_inode) {
|
||||
if ((inode = configfs_new_inode(mode))) {
|
||||
struct configfs_dirent *sd = dentry->d_fsdata;
|
||||
if ((inode = configfs_new_inode(mode, sd))) {
|
||||
if (dentry->d_parent && dentry->d_parent->d_inode) {
|
||||
struct inode *p_inode = dentry->d_parent->d_inode;
|
||||
p_inode->i_mtime = p_inode->i_ctime = CURRENT_TIME;
|
||||
@ -103,10 +194,9 @@ int configfs_create(struct dentry * dentry, int mode, int (*init)(struct inode *
|
||||
*/
|
||||
const unsigned char * configfs_get_name(struct configfs_dirent *sd)
|
||||
{
|
||||
struct attribute * attr;
|
||||
struct configfs_attribute *attr;
|
||||
|
||||
if (!sd || !sd->s_element)
|
||||
BUG();
|
||||
BUG_ON(!sd || !sd->s_element);
|
||||
|
||||
/* These always have a dentry, so use that */
|
||||
if (sd->s_type & (CONFIGFS_DIR | CONFIGFS_ITEM_LINK))
|
||||
@ -114,7 +204,7 @@ const unsigned char * configfs_get_name(struct configfs_dirent *sd)
|
||||
|
||||
if (sd->s_type & CONFIGFS_ITEM_ATTR) {
|
||||
attr = sd->s_element;
|
||||
return attr->name;
|
||||
return attr->ca_name;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
@ -130,13 +220,17 @@ void configfs_drop_dentry(struct configfs_dirent * sd, struct dentry * parent)
|
||||
|
||||
if (dentry) {
|
||||
spin_lock(&dcache_lock);
|
||||
spin_lock(&dentry->d_lock);
|
||||
if (!(d_unhashed(dentry) && dentry->d_inode)) {
|
||||
dget_locked(dentry);
|
||||
__d_drop(dentry);
|
||||
spin_unlock(&dentry->d_lock);
|
||||
spin_unlock(&dcache_lock);
|
||||
simple_unlink(parent->d_inode, dentry);
|
||||
} else
|
||||
} else {
|
||||
spin_unlock(&dentry->d_lock);
|
||||
spin_unlock(&dcache_lock);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -145,6 +239,10 @@ void configfs_hash_and_remove(struct dentry * dir, const char * name)
|
||||
struct configfs_dirent * sd;
|
||||
struct configfs_dirent * parent_sd = dir->d_fsdata;
|
||||
|
||||
if (dir->d_inode == NULL)
|
||||
/* no inode means this hasn't been made visible yet */
|
||||
return;
|
||||
|
||||
mutex_lock(&dir->d_inode->i_mutex);
|
||||
list_for_each_entry(sd, &parent_sd->s_children, s_sibling) {
|
||||
if (!sd->s_element)
|
||||
|
@ -38,6 +38,7 @@
|
||||
|
||||
struct vfsmount * configfs_mount = NULL;
|
||||
struct super_block * configfs_sb = NULL;
|
||||
kmem_cache_t *configfs_dir_cachep;
|
||||
static int configfs_mnt_count = 0;
|
||||
|
||||
static struct super_operations configfs_ops = {
|
||||
@ -62,6 +63,7 @@ static struct configfs_dirent configfs_root = {
|
||||
.s_children = LIST_HEAD_INIT(configfs_root.s_children),
|
||||
.s_element = &configfs_root_group.cg_item,
|
||||
.s_type = CONFIGFS_ROOT,
|
||||
.s_iattr = NULL,
|
||||
};
|
||||
|
||||
static int configfs_fill_super(struct super_block *sb, void *data, int silent)
|
||||
@ -73,9 +75,11 @@ static int configfs_fill_super(struct super_block *sb, void *data, int silent)
|
||||
sb->s_blocksize_bits = PAGE_CACHE_SHIFT;
|
||||
sb->s_magic = CONFIGFS_MAGIC;
|
||||
sb->s_op = &configfs_ops;
|
||||
sb->s_time_gran = 1;
|
||||
configfs_sb = sb;
|
||||
|
||||
inode = configfs_new_inode(S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO);
|
||||
inode = configfs_new_inode(S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO,
|
||||
&configfs_root);
|
||||
if (inode) {
|
||||
inode->i_op = &configfs_dir_inode_operations;
|
||||
inode->i_fop = &configfs_dir_operations;
|
||||
@ -128,19 +132,31 @@ static decl_subsys(config, NULL, NULL);
|
||||
|
||||
static int __init configfs_init(void)
|
||||
{
|
||||
int err;
|
||||
int err = -ENOMEM;
|
||||
|
||||
configfs_dir_cachep = kmem_cache_create("configfs_dir_cache",
|
||||
sizeof(struct configfs_dirent),
|
||||
0, 0, NULL, NULL);
|
||||
if (!configfs_dir_cachep)
|
||||
goto out;
|
||||
|
||||
kset_set_kset_s(&config_subsys, kernel_subsys);
|
||||
err = subsystem_register(&config_subsys);
|
||||
if (err)
|
||||
return err;
|
||||
if (err) {
|
||||
kmem_cache_destroy(configfs_dir_cachep);
|
||||
configfs_dir_cachep = NULL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = register_filesystem(&configfs_fs_type);
|
||||
if (err) {
|
||||
printk(KERN_ERR "configfs: Unable to register filesystem!\n");
|
||||
subsystem_unregister(&config_subsys);
|
||||
kmem_cache_destroy(configfs_dir_cachep);
|
||||
configfs_dir_cachep = NULL;
|
||||
}
|
||||
|
||||
out:
|
||||
return err;
|
||||
}
|
||||
|
||||
@ -148,11 +164,13 @@ static void __exit configfs_exit(void)
|
||||
{
|
||||
unregister_filesystem(&configfs_fs_type);
|
||||
subsystem_unregister(&config_subsys);
|
||||
kmem_cache_destroy(configfs_dir_cachep);
|
||||
configfs_dir_cachep = NULL;
|
||||
}
|
||||
|
||||
MODULE_AUTHOR("Oracle");
|
||||
MODULE_LICENSE("GPL");
|
||||
MODULE_VERSION("0.0.1");
|
||||
MODULE_VERSION("0.0.2");
|
||||
MODULE_DESCRIPTION("Simple RAM filesystem for user driven kernel subsystem configuration.");
|
||||
|
||||
module_init(configfs_init);
|
||||
|
@ -162,8 +162,7 @@ int configfs_unlink(struct inode *dir, struct dentry *dentry)
|
||||
if (!(sd->s_type & CONFIGFS_ITEM_LINK))
|
||||
goto out;
|
||||
|
||||
if (dentry->d_parent == configfs_sb->s_root)
|
||||
BUG();
|
||||
BUG_ON(dentry->d_parent == configfs_sb->s_root);
|
||||
|
||||
sl = sd->s_element;
|
||||
|
||||
@ -277,5 +276,6 @@ struct inode_operations configfs_symlink_inode_operations = {
|
||||
.follow_link = configfs_follow_link,
|
||||
.readlink = generic_readlink,
|
||||
.put_link = configfs_put_link,
|
||||
.setattr = configfs_setattr,
|
||||
};
|
||||
|
||||
|
@ -58,7 +58,7 @@ int ocfs2_write_block(struct ocfs2_super *osb, struct buffer_head *bh,
|
||||
goto out;
|
||||
}
|
||||
|
||||
down(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_lock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
|
||||
lock_buffer(bh);
|
||||
set_buffer_uptodate(bh);
|
||||
@ -82,7 +82,7 @@ int ocfs2_write_block(struct ocfs2_super *osb, struct buffer_head *bh,
|
||||
brelse(bh);
|
||||
}
|
||||
|
||||
up(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_unlock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
out:
|
||||
mlog_exit(ret);
|
||||
return ret;
|
||||
@ -125,13 +125,13 @@ int ocfs2_read_blocks(struct ocfs2_super *osb, u64 block, int nr,
|
||||
flags &= ~OCFS2_BH_CACHED;
|
||||
|
||||
if (inode)
|
||||
down(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_lock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
for (i = 0 ; i < nr ; i++) {
|
||||
if (bhs[i] == NULL) {
|
||||
bhs[i] = sb_getblk(sb, block++);
|
||||
if (bhs[i] == NULL) {
|
||||
if (inode)
|
||||
up(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_unlock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
status = -EIO;
|
||||
mlog_errno(status);
|
||||
goto bail;
|
||||
@ -220,7 +220,7 @@ int ocfs2_read_blocks(struct ocfs2_super *osb, u64 block, int nr,
|
||||
ocfs2_set_buffer_uptodate(inode, bh);
|
||||
}
|
||||
if (inode)
|
||||
up(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_unlock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
|
||||
mlog(ML_BH_IO, "block=(%"MLFu64"), nr=(%d), cached=%s\n", block, nr,
|
||||
(!(flags & OCFS2_BH_CACHED) || ignore_cache) ? "no" : "yes");
|
||||
|
@ -917,8 +917,9 @@ static int o2hb_thread(void *data)
|
||||
elapsed_msec = o2hb_elapsed_msecs(&before_hb, &after_hb);
|
||||
|
||||
mlog(0, "start = %lu.%lu, end = %lu.%lu, msec = %u\n",
|
||||
before_hb.tv_sec, before_hb.tv_usec,
|
||||
after_hb.tv_sec, after_hb.tv_usec, elapsed_msec);
|
||||
before_hb.tv_sec, (unsigned long) before_hb.tv_usec,
|
||||
after_hb.tv_sec, (unsigned long) after_hb.tv_usec,
|
||||
elapsed_msec);
|
||||
|
||||
if (elapsed_msec < reg->hr_timeout_ms) {
|
||||
/* the kthread api has blocked signals for us so no
|
||||
|
@ -1285,14 +1285,16 @@ static void o2net_idle_timer(unsigned long data)
|
||||
mlog(ML_NOTICE, "here are some times that might help debug the "
|
||||
"situation: (tmr %ld.%ld now %ld.%ld dr %ld.%ld adv "
|
||||
"%ld.%ld:%ld.%ld func (%08x:%u) %ld.%ld:%ld.%ld)\n",
|
||||
sc->sc_tv_timer.tv_sec, sc->sc_tv_timer.tv_usec,
|
||||
now.tv_sec, now.tv_usec,
|
||||
sc->sc_tv_data_ready.tv_sec, sc->sc_tv_data_ready.tv_usec,
|
||||
sc->sc_tv_advance_start.tv_sec, sc->sc_tv_advance_start.tv_usec,
|
||||
sc->sc_tv_advance_stop.tv_sec, sc->sc_tv_advance_stop.tv_usec,
|
||||
sc->sc_tv_timer.tv_sec, (long) sc->sc_tv_timer.tv_usec,
|
||||
now.tv_sec, (long) now.tv_usec,
|
||||
sc->sc_tv_data_ready.tv_sec, (long) sc->sc_tv_data_ready.tv_usec,
|
||||
sc->sc_tv_advance_start.tv_sec,
|
||||
(long) sc->sc_tv_advance_start.tv_usec,
|
||||
sc->sc_tv_advance_stop.tv_sec,
|
||||
(long) sc->sc_tv_advance_stop.tv_usec,
|
||||
sc->sc_msg_key, sc->sc_msg_type,
|
||||
sc->sc_tv_func_start.tv_sec, sc->sc_tv_func_start.tv_usec,
|
||||
sc->sc_tv_func_stop.tv_sec, sc->sc_tv_func_stop.tv_usec);
|
||||
sc->sc_tv_func_start.tv_sec, (long) sc->sc_tv_func_start.tv_usec,
|
||||
sc->sc_tv_func_stop.tv_sec, (long) sc->sc_tv_func_stop.tv_usec);
|
||||
|
||||
o2net_sc_queue_work(sc, &sc->sc_shutdown_work);
|
||||
}
|
||||
|
@ -657,6 +657,7 @@ void dlm_complete_thread(struct dlm_ctxt *dlm);
|
||||
int dlm_launch_recovery_thread(struct dlm_ctxt *dlm);
|
||||
void dlm_complete_recovery_thread(struct dlm_ctxt *dlm);
|
||||
void dlm_wait_for_recovery(struct dlm_ctxt *dlm);
|
||||
int dlm_is_node_dead(struct dlm_ctxt *dlm, u8 node);
|
||||
|
||||
void dlm_put(struct dlm_ctxt *dlm);
|
||||
struct dlm_ctxt *dlm_grab(struct dlm_ctxt *dlm);
|
||||
|
@ -573,8 +573,11 @@ static int dlm_query_join_handler(struct o2net_msg *msg, u32 len, void *data)
|
||||
spin_lock(&dlm_domain_lock);
|
||||
dlm = __dlm_lookup_domain_full(query->domain, query->name_len);
|
||||
/* Once the dlm ctxt is marked as leaving then we don't want
|
||||
* to be put in someone's domain map. */
|
||||
* to be put in someone's domain map.
|
||||
* Also, explicitly disallow joining at certain troublesome
|
||||
* times (ie. during recovery). */
|
||||
if (dlm && dlm->dlm_state != DLM_CTXT_LEAVING) {
|
||||
int bit = query->node_idx;
|
||||
spin_lock(&dlm->spinlock);
|
||||
|
||||
if (dlm->dlm_state == DLM_CTXT_NEW &&
|
||||
@ -586,6 +589,19 @@ static int dlm_query_join_handler(struct o2net_msg *msg, u32 len, void *data)
|
||||
} else if (dlm->joining_node != DLM_LOCK_RES_OWNER_UNKNOWN) {
|
||||
/* Disallow parallel joins. */
|
||||
response = JOIN_DISALLOW;
|
||||
} else if (dlm->reco.state & DLM_RECO_STATE_ACTIVE) {
|
||||
mlog(ML_NOTICE, "node %u trying to join, but recovery "
|
||||
"is ongoing.\n", bit);
|
||||
response = JOIN_DISALLOW;
|
||||
} else if (test_bit(bit, dlm->recovery_map)) {
|
||||
mlog(ML_NOTICE, "node %u trying to join, but it "
|
||||
"still needs recovery.\n", bit);
|
||||
response = JOIN_DISALLOW;
|
||||
} else if (test_bit(bit, dlm->domain_map)) {
|
||||
mlog(ML_NOTICE, "node %u trying to join, but it "
|
||||
"is still in the domain! needs recovery?\n",
|
||||
bit);
|
||||
response = JOIN_DISALLOW;
|
||||
} else {
|
||||
/* Alright we're fully a part of this domain
|
||||
* so we keep some state as to who's joining
|
||||
|
@ -1050,17 +1050,10 @@ static int dlm_restart_lock_mastery(struct dlm_ctxt *dlm,
|
||||
node = dlm_bitmap_diff_iter_next(&bdi, &sc);
|
||||
while (node >= 0) {
|
||||
if (sc == NODE_UP) {
|
||||
/* a node came up. easy. might not even need
|
||||
* to talk to it if its node number is higher
|
||||
* or if we are already blocked. */
|
||||
mlog(0, "node up! %d\n", node);
|
||||
if (blocked)
|
||||
goto next;
|
||||
|
||||
if (node > dlm->node_num) {
|
||||
mlog(0, "node > this node. skipping.\n");
|
||||
goto next;
|
||||
}
|
||||
/* a node came up. clear any old vote from
|
||||
* the response map and set it in the vote map
|
||||
* then restart the mastery. */
|
||||
mlog(ML_NOTICE, "node %d up while restarting\n", node);
|
||||
|
||||
/* redo the master request, but only for the new node */
|
||||
mlog(0, "sending request to new node\n");
|
||||
@ -2005,6 +1998,15 @@ fail:
|
||||
break;
|
||||
|
||||
mlog(0, "timed out during migration\n");
|
||||
/* avoid hang during shutdown when migrating lockres
|
||||
* to a node which also goes down */
|
||||
if (dlm_is_node_dead(dlm, target)) {
|
||||
mlog(0, "%s:%.*s: expected migration target %u "
|
||||
"is no longer up. restarting.\n",
|
||||
dlm->name, res->lockname.len,
|
||||
res->lockname.name, target);
|
||||
ret = -ERESTARTSYS;
|
||||
}
|
||||
}
|
||||
if (ret == -ERESTARTSYS) {
|
||||
/* migration failed, detach and clean up mle */
|
||||
|
@ -39,6 +39,7 @@
|
||||
#include <linux/inet.h>
|
||||
#include <linux/timer.h>
|
||||
#include <linux/kthread.h>
|
||||
#include <linux/delay.h>
|
||||
|
||||
|
||||
#include "cluster/heartbeat.h"
|
||||
@ -256,6 +257,27 @@ static int dlm_recovery_thread(void *data)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* returns true when the recovery master has contacted us */
|
||||
static int dlm_reco_master_ready(struct dlm_ctxt *dlm)
|
||||
{
|
||||
int ready;
|
||||
spin_lock(&dlm->spinlock);
|
||||
ready = (dlm->reco.new_master != O2NM_INVALID_NODE_NUM);
|
||||
spin_unlock(&dlm->spinlock);
|
||||
return ready;
|
||||
}
|
||||
|
||||
/* returns true if node is no longer in the domain
|
||||
* could be dead or just not joined */
|
||||
int dlm_is_node_dead(struct dlm_ctxt *dlm, u8 node)
|
||||
{
|
||||
int dead;
|
||||
spin_lock(&dlm->spinlock);
|
||||
dead = test_bit(node, dlm->domain_map);
|
||||
spin_unlock(&dlm->spinlock);
|
||||
return dead;
|
||||
}
|
||||
|
||||
/* callers of the top-level api calls (dlmlock/dlmunlock) should
|
||||
* block on the dlm->reco.event when recovery is in progress.
|
||||
* the dlm recovery thread will set this state when it begins
|
||||
@ -297,6 +319,7 @@ static void dlm_end_recovery(struct dlm_ctxt *dlm)
|
||||
static int dlm_do_recovery(struct dlm_ctxt *dlm)
|
||||
{
|
||||
int status = 0;
|
||||
int ret;
|
||||
|
||||
spin_lock(&dlm->spinlock);
|
||||
|
||||
@ -343,10 +366,13 @@ static int dlm_do_recovery(struct dlm_ctxt *dlm)
|
||||
goto master_here;
|
||||
|
||||
if (dlm->reco.new_master == O2NM_INVALID_NODE_NUM) {
|
||||
/* choose a new master */
|
||||
if (!dlm_pick_recovery_master(dlm)) {
|
||||
/* choose a new master, returns 0 if this node
|
||||
* is the master, -EEXIST if it's another node.
|
||||
* this does not return until a new master is chosen
|
||||
* or recovery completes entirely. */
|
||||
ret = dlm_pick_recovery_master(dlm);
|
||||
if (!ret) {
|
||||
/* already notified everyone. go. */
|
||||
dlm->reco.new_master = dlm->node_num;
|
||||
goto master_here;
|
||||
}
|
||||
mlog(0, "another node will master this recovery session.\n");
|
||||
@ -371,8 +397,13 @@ master_here:
|
||||
if (status < 0) {
|
||||
mlog(ML_ERROR, "error %d remastering locks for node %u, "
|
||||
"retrying.\n", status, dlm->reco.dead_node);
|
||||
/* yield a bit to allow any final network messages
|
||||
* to get handled on remaining nodes */
|
||||
msleep(100);
|
||||
} else {
|
||||
/* success! see if any other nodes need recovery */
|
||||
mlog(0, "DONE mastering recovery of %s:%u here(this=%u)!\n",
|
||||
dlm->name, dlm->reco.dead_node, dlm->node_num);
|
||||
dlm_reset_recovery(dlm);
|
||||
}
|
||||
dlm_end_recovery(dlm);
|
||||
@ -477,7 +508,7 @@ static int dlm_remaster_locks(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
BUG();
|
||||
break;
|
||||
case DLM_RECO_NODE_DATA_DEAD:
|
||||
mlog(0, "node %u died after "
|
||||
mlog(ML_NOTICE, "node %u died after "
|
||||
"requesting recovery info for "
|
||||
"node %u\n", ndata->node_num,
|
||||
dead_node);
|
||||
@ -485,6 +516,19 @@ static int dlm_remaster_locks(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
// start all over
|
||||
destroy = 1;
|
||||
status = -EAGAIN;
|
||||
/* instead of spinning like crazy here,
|
||||
* wait for the domain map to catch up
|
||||
* with the network state. otherwise this
|
||||
* can be hit hundreds of times before
|
||||
* the node is really seen as dead. */
|
||||
wait_event_timeout(dlm->dlm_reco_thread_wq,
|
||||
dlm_is_node_dead(dlm,
|
||||
ndata->node_num),
|
||||
msecs_to_jiffies(1000));
|
||||
mlog(0, "waited 1 sec for %u, "
|
||||
"dead? %s\n", ndata->node_num,
|
||||
dlm_is_node_dead(dlm, ndata->node_num) ?
|
||||
"yes" : "no");
|
||||
goto leave;
|
||||
case DLM_RECO_NODE_DATA_RECEIVING:
|
||||
case DLM_RECO_NODE_DATA_REQUESTED:
|
||||
@ -678,11 +722,27 @@ static void dlm_request_all_locks_worker(struct dlm_work_item *item, void *data)
|
||||
dlm = item->dlm;
|
||||
dead_node = item->u.ral.dead_node;
|
||||
reco_master = item->u.ral.reco_master;
|
||||
mres = (struct dlm_migratable_lockres *)data;
|
||||
|
||||
if (dead_node != dlm->reco.dead_node ||
|
||||
reco_master != dlm->reco.new_master) {
|
||||
/* show extra debug info if the recovery state is messed */
|
||||
mlog(ML_ERROR, "%s: bad reco state: reco(dead=%u, master=%u), "
|
||||
"request(dead=%u, master=%u)\n",
|
||||
dlm->name, dlm->reco.dead_node, dlm->reco.new_master,
|
||||
dead_node, reco_master);
|
||||
mlog(ML_ERROR, "%s: name=%.*s master=%u locks=%u/%u flags=%u "
|
||||
"entry[0]={c=%"MLFu64",l=%u,f=%u,t=%d,ct=%d,hb=%d,n=%u}\n",
|
||||
dlm->name, mres->lockname_len, mres->lockname, mres->master,
|
||||
mres->num_locks, mres->total_locks, mres->flags,
|
||||
mres->ml[0].cookie, mres->ml[0].list, mres->ml[0].flags,
|
||||
mres->ml[0].type, mres->ml[0].convert_type,
|
||||
mres->ml[0].highest_blocked, mres->ml[0].node);
|
||||
BUG();
|
||||
}
|
||||
BUG_ON(dead_node != dlm->reco.dead_node);
|
||||
BUG_ON(reco_master != dlm->reco.new_master);
|
||||
|
||||
mres = (struct dlm_migratable_lockres *)data;
|
||||
|
||||
/* lock resources should have already been moved to the
|
||||
* dlm->reco.resources list. now move items from that list
|
||||
* to a temp list if the dead owner matches. note that the
|
||||
@ -757,15 +817,18 @@ int dlm_reco_data_done_handler(struct o2net_msg *msg, u32 len, void *data)
|
||||
continue;
|
||||
|
||||
switch (ndata->state) {
|
||||
/* should have moved beyond INIT but not to FINALIZE yet */
|
||||
case DLM_RECO_NODE_DATA_INIT:
|
||||
case DLM_RECO_NODE_DATA_DEAD:
|
||||
case DLM_RECO_NODE_DATA_DONE:
|
||||
case DLM_RECO_NODE_DATA_FINALIZE_SENT:
|
||||
mlog(ML_ERROR, "bad ndata state for node %u:"
|
||||
" state=%d\n", ndata->node_num,
|
||||
ndata->state);
|
||||
BUG();
|
||||
break;
|
||||
/* these states are possible at this point, anywhere along
|
||||
* the line of recovery */
|
||||
case DLM_RECO_NODE_DATA_DONE:
|
||||
case DLM_RECO_NODE_DATA_RECEIVING:
|
||||
case DLM_RECO_NODE_DATA_REQUESTED:
|
||||
case DLM_RECO_NODE_DATA_REQUESTING:
|
||||
@ -799,13 +862,31 @@ static void dlm_move_reco_locks_to_list(struct dlm_ctxt *dlm,
|
||||
{
|
||||
struct dlm_lock_resource *res;
|
||||
struct list_head *iter, *iter2;
|
||||
struct dlm_lock *lock;
|
||||
|
||||
spin_lock(&dlm->spinlock);
|
||||
list_for_each_safe(iter, iter2, &dlm->reco.resources) {
|
||||
res = list_entry (iter, struct dlm_lock_resource, recovering);
|
||||
/* always prune any $RECOVERY entries for dead nodes,
|
||||
* otherwise hangs can occur during later recovery */
|
||||
if (dlm_is_recovery_lock(res->lockname.name,
|
||||
res->lockname.len))
|
||||
res->lockname.len)) {
|
||||
spin_lock(&res->spinlock);
|
||||
list_for_each_entry(lock, &res->granted, list) {
|
||||
if (lock->ml.node == dead_node) {
|
||||
mlog(0, "AHA! there was "
|
||||
"a $RECOVERY lock for dead "
|
||||
"node %u (%s)!\n",
|
||||
dead_node, dlm->name);
|
||||
list_del_init(&lock->list);
|
||||
dlm_lock_put(lock);
|
||||
break;
|
||||
}
|
||||
}
|
||||
spin_unlock(&res->spinlock);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (res->owner == dead_node) {
|
||||
mlog(0, "found lockres owned by dead node while "
|
||||
"doing recovery for node %u. sending it.\n",
|
||||
@ -1179,7 +1260,7 @@ static void dlm_mig_lockres_worker(struct dlm_work_item *item, void *data)
|
||||
again:
|
||||
ret = dlm_lockres_master_requery(dlm, res, &real_master);
|
||||
if (ret < 0) {
|
||||
mlog(0, "dlm_lockres_master_requery failure: %d\n",
|
||||
mlog(0, "dlm_lockres_master_requery ret=%d\n",
|
||||
ret);
|
||||
goto again;
|
||||
}
|
||||
@ -1757,6 +1838,7 @@ static void dlm_do_local_recovery_cleanup(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
struct dlm_lock_resource *res;
|
||||
int i;
|
||||
struct list_head *bucket;
|
||||
struct dlm_lock *lock;
|
||||
|
||||
|
||||
/* purge any stale mles */
|
||||
@ -1780,10 +1862,25 @@ static void dlm_do_local_recovery_cleanup(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
bucket = &(dlm->resources[i]);
|
||||
list_for_each(iter, bucket) {
|
||||
res = list_entry (iter, struct dlm_lock_resource, list);
|
||||
/* always prune any $RECOVERY entries for dead nodes,
|
||||
* otherwise hangs can occur during later recovery */
|
||||
if (dlm_is_recovery_lock(res->lockname.name,
|
||||
res->lockname.len))
|
||||
res->lockname.len)) {
|
||||
spin_lock(&res->spinlock);
|
||||
list_for_each_entry(lock, &res->granted, list) {
|
||||
if (lock->ml.node == dead_node) {
|
||||
mlog(0, "AHA! there was "
|
||||
"a $RECOVERY lock for dead "
|
||||
"node %u (%s)!\n",
|
||||
dead_node, dlm->name);
|
||||
list_del_init(&lock->list);
|
||||
dlm_lock_put(lock);
|
||||
break;
|
||||
}
|
||||
}
|
||||
spin_unlock(&res->spinlock);
|
||||
continue;
|
||||
|
||||
}
|
||||
spin_lock(&res->spinlock);
|
||||
/* zero the lvb if necessary */
|
||||
dlm_revalidate_lvb(dlm, res, dead_node);
|
||||
@ -1869,12 +1966,9 @@ void dlm_hb_node_up_cb(struct o2nm_node *node, int idx, void *data)
|
||||
return;
|
||||
|
||||
spin_lock(&dlm->spinlock);
|
||||
|
||||
set_bit(idx, dlm->live_nodes_map);
|
||||
|
||||
/* notify any mles attached to the heartbeat events */
|
||||
dlm_hb_event_notify_attached(dlm, idx, 1);
|
||||
|
||||
/* do NOT notify mle attached to the heartbeat events.
|
||||
* new nodes are not interesting in mastery until joined. */
|
||||
spin_unlock(&dlm->spinlock);
|
||||
|
||||
dlm_put(dlm);
|
||||
@ -1897,7 +1991,18 @@ static void dlm_reco_unlock_ast(void *astdata, enum dlm_status st)
|
||||
mlog(0, "unlockast for recovery lock fired!\n");
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* dlm_pick_recovery_master will continually attempt to use
|
||||
* dlmlock() on the special "$RECOVERY" lockres with the
|
||||
* LKM_NOQUEUE flag to get an EX. every thread that enters
|
||||
* this function on each node racing to become the recovery
|
||||
* master will not stop attempting this until either:
|
||||
* a) this node gets the EX (and becomes the recovery master),
|
||||
* or b) dlm->reco.new_master gets set to some nodenum
|
||||
* != O2NM_INVALID_NODE_NUM (another node will do the reco).
|
||||
* so each time a recovery master is needed, the entire cluster
|
||||
* will sync at this point. if the new master dies, that will
|
||||
* be detected in dlm_do_recovery */
|
||||
static int dlm_pick_recovery_master(struct dlm_ctxt *dlm)
|
||||
{
|
||||
enum dlm_status ret;
|
||||
@ -1906,23 +2011,45 @@ static int dlm_pick_recovery_master(struct dlm_ctxt *dlm)
|
||||
|
||||
mlog(0, "starting recovery of %s at %lu, dead=%u, this=%u\n",
|
||||
dlm->name, jiffies, dlm->reco.dead_node, dlm->node_num);
|
||||
retry:
|
||||
again:
|
||||
memset(&lksb, 0, sizeof(lksb));
|
||||
|
||||
ret = dlmlock(dlm, LKM_EXMODE, &lksb, LKM_NOQUEUE|LKM_RECOVERY,
|
||||
DLM_RECOVERY_LOCK_NAME, dlm_reco_ast, dlm, dlm_reco_bast);
|
||||
|
||||
mlog(0, "%s: dlmlock($RECOVERY) returned %d, lksb=%d\n",
|
||||
dlm->name, ret, lksb.status);
|
||||
|
||||
if (ret == DLM_NORMAL) {
|
||||
mlog(0, "dlm=%s dlmlock says I got it (this=%u)\n",
|
||||
dlm->name, dlm->node_num);
|
||||
/* I am master, send message to all nodes saying
|
||||
* that I am beginning a recovery session */
|
||||
status = dlm_send_begin_reco_message(dlm,
|
||||
dlm->reco.dead_node);
|
||||
|
||||
/* got the EX lock. check to see if another node
|
||||
* just became the reco master */
|
||||
if (dlm_reco_master_ready(dlm)) {
|
||||
mlog(0, "%s: got reco EX lock, but %u will "
|
||||
"do the recovery\n", dlm->name,
|
||||
dlm->reco.new_master);
|
||||
status = -EEXIST;
|
||||
} else {
|
||||
status = dlm_send_begin_reco_message(dlm,
|
||||
dlm->reco.dead_node);
|
||||
/* this always succeeds */
|
||||
BUG_ON(status);
|
||||
|
||||
/* set the new_master to this node */
|
||||
spin_lock(&dlm->spinlock);
|
||||
dlm->reco.new_master = dlm->node_num;
|
||||
spin_unlock(&dlm->spinlock);
|
||||
}
|
||||
|
||||
/* recovery lock is a special case. ast will not get fired,
|
||||
* so just go ahead and unlock it. */
|
||||
ret = dlmunlock(dlm, &lksb, 0, dlm_reco_unlock_ast, dlm);
|
||||
if (ret == DLM_DENIED) {
|
||||
mlog(0, "got DLM_DENIED, trying LKM_CANCEL\n");
|
||||
ret = dlmunlock(dlm, &lksb, LKM_CANCEL, dlm_reco_unlock_ast, dlm);
|
||||
}
|
||||
if (ret != DLM_NORMAL) {
|
||||
/* this would really suck. this could only happen
|
||||
* if there was a network error during the unlock
|
||||
@ -1930,20 +2057,42 @@ retry:
|
||||
* is actually "done" and the lock structure is
|
||||
* even freed. we can continue, but only
|
||||
* because this specific lock name is special. */
|
||||
mlog(0, "dlmunlock returned %d\n", ret);
|
||||
}
|
||||
|
||||
if (status < 0) {
|
||||
mlog(0, "failed to send recovery message. "
|
||||
"must retry with new node map.\n");
|
||||
goto retry;
|
||||
mlog(ML_ERROR, "dlmunlock returned %d\n", ret);
|
||||
}
|
||||
} else if (ret == DLM_NOTQUEUED) {
|
||||
mlog(0, "dlm=%s dlmlock says another node got it (this=%u)\n",
|
||||
dlm->name, dlm->node_num);
|
||||
/* another node is master. wait on
|
||||
* reco.new_master != O2NM_INVALID_NODE_NUM */
|
||||
* reco.new_master != O2NM_INVALID_NODE_NUM
|
||||
* for at most one second */
|
||||
wait_event_timeout(dlm->dlm_reco_thread_wq,
|
||||
dlm_reco_master_ready(dlm),
|
||||
msecs_to_jiffies(1000));
|
||||
if (!dlm_reco_master_ready(dlm)) {
|
||||
mlog(0, "%s: reco master taking awhile\n",
|
||||
dlm->name);
|
||||
goto again;
|
||||
}
|
||||
/* another node has informed this one that it is reco master */
|
||||
mlog(0, "%s: reco master %u is ready to recover %u\n",
|
||||
dlm->name, dlm->reco.new_master, dlm->reco.dead_node);
|
||||
status = -EEXIST;
|
||||
} else {
|
||||
struct dlm_lock_resource *res;
|
||||
|
||||
/* dlmlock returned something other than NOTQUEUED or NORMAL */
|
||||
mlog(ML_ERROR, "%s: got %s from dlmlock($RECOVERY), "
|
||||
"lksb.status=%s\n", dlm->name, dlm_errname(ret),
|
||||
dlm_errname(lksb.status));
|
||||
res = dlm_lookup_lockres(dlm, DLM_RECOVERY_LOCK_NAME,
|
||||
DLM_RECOVERY_LOCK_NAME_LEN);
|
||||
if (res) {
|
||||
dlm_print_one_lock_resource(res);
|
||||
dlm_lockres_put(res);
|
||||
} else {
|
||||
mlog(ML_ERROR, "recovery lock not found\n");
|
||||
}
|
||||
BUG();
|
||||
}
|
||||
|
||||
return status;
|
||||
@ -1982,7 +2131,7 @@ static int dlm_send_begin_reco_message(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
mlog(0, "not sending begin reco to self\n");
|
||||
continue;
|
||||
}
|
||||
|
||||
retry:
|
||||
ret = -EINVAL;
|
||||
mlog(0, "attempting to send begin reco msg to %d\n",
|
||||
nodenum);
|
||||
@ -1991,8 +2140,17 @@ static int dlm_send_begin_reco_message(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
/* negative status is handled ok by caller here */
|
||||
if (ret >= 0)
|
||||
ret = status;
|
||||
if (dlm_is_host_down(ret)) {
|
||||
/* node is down. not involved in recovery
|
||||
* so just keep going */
|
||||
mlog(0, "%s: node %u was down when sending "
|
||||
"begin reco msg (%d)\n", dlm->name, nodenum, ret);
|
||||
ret = 0;
|
||||
}
|
||||
if (ret < 0) {
|
||||
struct dlm_lock_resource *res;
|
||||
/* this is now a serious problem, possibly ENOMEM
|
||||
* in the network stack. must retry */
|
||||
mlog_errno(ret);
|
||||
mlog(ML_ERROR, "begin reco of dlm %s to node %u "
|
||||
" returned %d\n", dlm->name, nodenum, ret);
|
||||
@ -2004,7 +2162,10 @@ static int dlm_send_begin_reco_message(struct dlm_ctxt *dlm, u8 dead_node)
|
||||
} else {
|
||||
mlog(ML_ERROR, "recovery lock not found\n");
|
||||
}
|
||||
break;
|
||||
/* sleep for a bit in hopes that we can avoid
|
||||
* another ENOMEM */
|
||||
msleep(100);
|
||||
goto retry;
|
||||
}
|
||||
}
|
||||
|
||||
@ -2027,19 +2188,34 @@ int dlm_begin_reco_handler(struct o2net_msg *msg, u32 len, void *data)
|
||||
|
||||
spin_lock(&dlm->spinlock);
|
||||
if (dlm->reco.new_master != O2NM_INVALID_NODE_NUM) {
|
||||
mlog(0, "new_master already set to %u!\n",
|
||||
dlm->reco.new_master);
|
||||
if (test_bit(dlm->reco.new_master, dlm->recovery_map)) {
|
||||
mlog(0, "%s: new_master %u died, changing "
|
||||
"to %u\n", dlm->name, dlm->reco.new_master,
|
||||
br->node_idx);
|
||||
} else {
|
||||
mlog(0, "%s: new_master %u NOT DEAD, changing "
|
||||
"to %u\n", dlm->name, dlm->reco.new_master,
|
||||
br->node_idx);
|
||||
/* may not have seen the new master as dead yet */
|
||||
}
|
||||
}
|
||||
if (dlm->reco.dead_node != O2NM_INVALID_NODE_NUM) {
|
||||
mlog(0, "dead_node already set to %u!\n",
|
||||
dlm->reco.dead_node);
|
||||
mlog(ML_NOTICE, "%s: dead_node previously set to %u, "
|
||||
"node %u changing it to %u\n", dlm->name,
|
||||
dlm->reco.dead_node, br->node_idx, br->dead_node);
|
||||
}
|
||||
dlm->reco.new_master = br->node_idx;
|
||||
dlm->reco.dead_node = br->dead_node;
|
||||
if (!test_bit(br->dead_node, dlm->recovery_map)) {
|
||||
mlog(ML_ERROR, "recovery master %u sees %u as dead, but this "
|
||||
mlog(0, "recovery master %u sees %u as dead, but this "
|
||||
"node has not yet. marking %u as dead\n",
|
||||
br->node_idx, br->dead_node, br->dead_node);
|
||||
if (!test_bit(br->dead_node, dlm->domain_map) ||
|
||||
!test_bit(br->dead_node, dlm->live_nodes_map))
|
||||
mlog(0, "%u not in domain/live_nodes map "
|
||||
"so setting it in reco map manually\n",
|
||||
br->dead_node);
|
||||
set_bit(br->dead_node, dlm->recovery_map);
|
||||
__dlm_hb_node_down(dlm, br->dead_node);
|
||||
}
|
||||
spin_unlock(&dlm->spinlock);
|
||||
|
@ -188,6 +188,19 @@ static enum dlm_status dlmunlock_common(struct dlm_ctxt *dlm,
|
||||
actions &= ~(DLM_UNLOCK_REMOVE_LOCK|
|
||||
DLM_UNLOCK_REGRANT_LOCK|
|
||||
DLM_UNLOCK_CLEAR_CONVERT_TYPE);
|
||||
} else if (status == DLM_RECOVERING ||
|
||||
status == DLM_MIGRATING ||
|
||||
status == DLM_FORWARD) {
|
||||
/* must clear the actions because this unlock
|
||||
* is about to be retried. cannot free or do
|
||||
* any list manipulation. */
|
||||
mlog(0, "%s:%.*s: clearing actions, %s\n",
|
||||
dlm->name, res->lockname.len,
|
||||
res->lockname.name,
|
||||
status==DLM_RECOVERING?"recovering":
|
||||
(status==DLM_MIGRATING?"migrating":
|
||||
"forward"));
|
||||
actions = 0;
|
||||
}
|
||||
if (flags & LKM_CANCEL)
|
||||
lock->cancel_pending = 0;
|
||||
|
@ -27,7 +27,7 @@
|
||||
* Boston, MA 021110-1307, USA.
|
||||
*/
|
||||
|
||||
#include <asm/signal.h>
|
||||
#include <linux/signal.h>
|
||||
|
||||
#include <linux/module.h>
|
||||
#include <linux/fs.h>
|
||||
|
@ -262,8 +262,7 @@ static int ocfs2_extent_map_find_leaf(struct inode *inode,
|
||||
el = &eb->h_list;
|
||||
}
|
||||
|
||||
if (el->l_tree_depth)
|
||||
BUG();
|
||||
BUG_ON(el->l_tree_depth);
|
||||
|
||||
for (i = 0; i < le16_to_cpu(el->l_next_free_rec); i++) {
|
||||
rec = &el->l_recs[i];
|
||||
@ -364,8 +363,8 @@ static int ocfs2_extent_map_lookup_read(struct inode *inode,
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (ent->e_tree_depth)
|
||||
BUG(); /* FIXME: Make sure this isn't a corruption */
|
||||
/* FIXME: Make sure this isn't a corruption */
|
||||
BUG_ON(ent->e_tree_depth);
|
||||
|
||||
*ret_ent = ent;
|
||||
|
||||
@ -423,8 +422,7 @@ static int ocfs2_extent_map_try_insert(struct inode *inode,
|
||||
le32_to_cpu(rec->e_clusters), NULL,
|
||||
NULL);
|
||||
|
||||
if (!old_ent)
|
||||
BUG();
|
||||
BUG_ON(!old_ent);
|
||||
|
||||
ret = -EEXIST;
|
||||
if (old_ent->e_tree_depth < tree_depth)
|
||||
@ -988,7 +986,7 @@ int __init init_ocfs2_extent_maps(void)
|
||||
return 0;
|
||||
}
|
||||
|
||||
void __exit exit_ocfs2_extent_maps(void)
|
||||
void exit_ocfs2_extent_maps(void)
|
||||
{
|
||||
kmem_cache_destroy(ocfs2_em_ent_cachep);
|
||||
}
|
||||
|
@ -1022,8 +1022,9 @@ static ssize_t ocfs2_file_aio_write(struct kiocb *iocb,
|
||||
}
|
||||
newsize = count + saved_pos;
|
||||
|
||||
mlog(0, "pos=%lld newsize=%"MLFu64" cursize=%lld\n",
|
||||
saved_pos, newsize, i_size_read(inode));
|
||||
mlog(0, "pos=%lld newsize=%lld cursize=%lld\n",
|
||||
(long long) saved_pos, (long long) newsize,
|
||||
(long long) i_size_read(inode));
|
||||
|
||||
/* No need for a higher level metadata lock if we're
|
||||
* never going past i_size. */
|
||||
@ -1042,8 +1043,9 @@ static ssize_t ocfs2_file_aio_write(struct kiocb *iocb,
|
||||
spin_unlock(&OCFS2_I(inode)->ip_lock);
|
||||
|
||||
mlog(0, "Writing at EOF, may need more allocation: "
|
||||
"i_size = %lld, newsize = %"MLFu64", need %u clusters\n",
|
||||
i_size_read(inode), newsize, clusters);
|
||||
"i_size = %lld, newsize = %lld, need %u clusters\n",
|
||||
(long long) i_size_read(inode), (long long) newsize,
|
||||
clusters);
|
||||
|
||||
/* We only want to continue the rest of this loop if
|
||||
* our extend will actually require more
|
||||
|
@ -903,10 +903,10 @@ void ocfs2_clear_inode(struct inode *inode)
|
||||
"Clear inode of %"MLFu64", inode is locked\n",
|
||||
oi->ip_blkno);
|
||||
|
||||
mlog_bug_on_msg(down_trylock(&oi->ip_io_sem),
|
||||
"Clear inode of %"MLFu64", io_sem is locked\n",
|
||||
mlog_bug_on_msg(!mutex_trylock(&oi->ip_io_mutex),
|
||||
"Clear inode of %"MLFu64", io_mutex is locked\n",
|
||||
oi->ip_blkno);
|
||||
up(&oi->ip_io_sem);
|
||||
mutex_unlock(&oi->ip_io_mutex);
|
||||
|
||||
/*
|
||||
* down_trylock() returns 0, down_write_trylock() returns 1
|
||||
|
@ -46,10 +46,10 @@ struct ocfs2_inode_info
|
||||
struct list_head ip_io_markers;
|
||||
int ip_orphaned_slot;
|
||||
|
||||
struct semaphore ip_io_sem;
|
||||
struct mutex ip_io_mutex;
|
||||
|
||||
/* Used by the journalling code to attach an inode to a
|
||||
* handle. These are protected by ip_io_sem in order to lock
|
||||
* handle. These are protected by ip_io_mutex in order to lock
|
||||
* out other I/O to the inode until we either commit or
|
||||
* abort. */
|
||||
struct list_head ip_handle_list;
|
||||
|
@ -147,8 +147,7 @@ struct ocfs2_journal_handle *ocfs2_start_trans(struct ocfs2_super *osb,
|
||||
|
||||
mlog_entry("(max_buffs = %d)\n", max_buffs);
|
||||
|
||||
if (!osb || !osb->journal->j_journal)
|
||||
BUG();
|
||||
BUG_ON(!osb || !osb->journal->j_journal);
|
||||
|
||||
if (ocfs2_is_hard_readonly(osb)) {
|
||||
ret = -EROFS;
|
||||
@ -401,7 +400,7 @@ int ocfs2_journal_access(struct ocfs2_journal_handle *handle,
|
||||
* j_trans_barrier for us. */
|
||||
ocfs2_set_inode_lock_trans(OCFS2_SB(inode->i_sb)->journal, inode);
|
||||
|
||||
down(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_lock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
switch (type) {
|
||||
case OCFS2_JOURNAL_ACCESS_CREATE:
|
||||
case OCFS2_JOURNAL_ACCESS_WRITE:
|
||||
@ -416,7 +415,7 @@ int ocfs2_journal_access(struct ocfs2_journal_handle *handle,
|
||||
status = -EINVAL;
|
||||
mlog(ML_ERROR, "Uknown access type!\n");
|
||||
}
|
||||
up(&OCFS2_I(inode)->ip_io_sem);
|
||||
mutex_unlock(&OCFS2_I(inode)->ip_io_mutex);
|
||||
|
||||
if (status < 0)
|
||||
mlog(ML_ERROR, "Error %d getting %d access to buffer!\n",
|
||||
@ -561,7 +560,11 @@ int ocfs2_journal_init(struct ocfs2_journal *journal, int *dirty)
|
||||
SET_INODE_JOURNAL(inode);
|
||||
OCFS2_I(inode)->ip_open_count++;
|
||||
|
||||
status = ocfs2_meta_lock(inode, NULL, &bh, 1);
|
||||
/* Skip recovery waits here - journal inode metadata never
|
||||
* changes in a live cluster so it can be considered an
|
||||
* exception to the rule. */
|
||||
status = ocfs2_meta_lock_full(inode, NULL, &bh, 1,
|
||||
OCFS2_META_LOCK_RECOVERY);
|
||||
if (status < 0) {
|
||||
if (status != -ERESTARTSYS)
|
||||
mlog(ML_ERROR, "Could not get lock on journal!\n");
|
||||
@ -672,8 +675,7 @@ void ocfs2_journal_shutdown(struct ocfs2_super *osb)
|
||||
|
||||
mlog_entry_void();
|
||||
|
||||
if (!osb)
|
||||
BUG();
|
||||
BUG_ON(!osb);
|
||||
|
||||
journal = osb->journal;
|
||||
if (!journal)
|
||||
@ -805,8 +807,7 @@ int ocfs2_journal_wipe(struct ocfs2_journal *journal, int full)
|
||||
|
||||
mlog_entry_void();
|
||||
|
||||
if (!journal)
|
||||
BUG();
|
||||
BUG_ON(!journal);
|
||||
|
||||
status = journal_wipe(journal->j_journal, full);
|
||||
if (status < 0) {
|
||||
@ -1072,10 +1073,10 @@ restart:
|
||||
NULL);
|
||||
|
||||
bail:
|
||||
down(&osb->recovery_lock);
|
||||
mutex_lock(&osb->recovery_lock);
|
||||
if (!status &&
|
||||
!ocfs2_node_map_is_empty(osb, &osb->recovery_map)) {
|
||||
up(&osb->recovery_lock);
|
||||
mutex_unlock(&osb->recovery_lock);
|
||||
goto restart;
|
||||
}
|
||||
|
||||
@ -1083,7 +1084,7 @@ bail:
|
||||
mb(); /* sync with ocfs2_recovery_thread_running */
|
||||
wake_up(&osb->recovery_event);
|
||||
|
||||
up(&osb->recovery_lock);
|
||||
mutex_unlock(&osb->recovery_lock);
|
||||
|
||||
mlog_exit(status);
|
||||
/* no one is callint kthread_stop() for us so the kthread() api
|
||||
@ -1098,7 +1099,7 @@ void ocfs2_recovery_thread(struct ocfs2_super *osb, int node_num)
|
||||
mlog_entry("(node_num=%d, osb->node_num = %d)\n",
|
||||
node_num, osb->node_num);
|
||||
|
||||
down(&osb->recovery_lock);
|
||||
mutex_lock(&osb->recovery_lock);
|
||||
if (osb->disable_recovery)
|
||||
goto out;
|
||||
|
||||
@ -1120,7 +1121,7 @@ void ocfs2_recovery_thread(struct ocfs2_super *osb, int node_num)
|
||||
}
|
||||
|
||||
out:
|
||||
up(&osb->recovery_lock);
|
||||
mutex_unlock(&osb->recovery_lock);
|
||||
wake_up(&osb->recovery_event);
|
||||
|
||||
mlog_exit_void();
|
||||
@ -1271,8 +1272,7 @@ static int ocfs2_recover_node(struct ocfs2_super *osb,
|
||||
|
||||
/* Should not ever be called to recover ourselves -- in that
|
||||
* case we should've called ocfs2_journal_load instead. */
|
||||
if (osb->node_num == node_num)
|
||||
BUG();
|
||||
BUG_ON(osb->node_num == node_num);
|
||||
|
||||
slot_num = ocfs2_node_num_to_slot(si, node_num);
|
||||
if (slot_num == OCFS2_INVALID_SLOT) {
|
||||
|
@ -33,6 +33,7 @@
|
||||
#include <linux/rbtree.h>
|
||||
#include <linux/workqueue.h>
|
||||
#include <linux/kref.h>
|
||||
#include <linux/mutex.h>
|
||||
|
||||
#include "cluster/nodemanager.h"
|
||||
#include "cluster/heartbeat.h"
|
||||
@ -233,7 +234,7 @@ struct ocfs2_super
|
||||
struct proc_dir_entry *proc_sub_dir; /* points to /proc/fs/ocfs2/<maj_min> */
|
||||
|
||||
atomic_t vol_state;
|
||||
struct semaphore recovery_lock;
|
||||
struct mutex recovery_lock;
|
||||
struct task_struct *recovery_thread_task;
|
||||
int disable_recovery;
|
||||
wait_queue_head_t checkpoint_event;
|
||||
|
@ -932,7 +932,7 @@ static void ocfs2_inode_init_once(void *data,
|
||||
oi->ip_dir_start_lookup = 0;
|
||||
|
||||
init_rwsem(&oi->ip_alloc_sem);
|
||||
init_MUTEX(&(oi->ip_io_sem));
|
||||
mutex_init(&oi->ip_io_mutex);
|
||||
|
||||
oi->ip_blkno = 0ULL;
|
||||
oi->ip_clusters = 0;
|
||||
@ -1137,9 +1137,9 @@ static void ocfs2_dismount_volume(struct super_block *sb, int mnt_err)
|
||||
|
||||
/* disable any new recovery threads and wait for any currently
|
||||
* running ones to exit. Do this before setting the vol_state. */
|
||||
down(&osb->recovery_lock);
|
||||
mutex_lock(&osb->recovery_lock);
|
||||
osb->disable_recovery = 1;
|
||||
up(&osb->recovery_lock);
|
||||
mutex_unlock(&osb->recovery_lock);
|
||||
wait_event(osb->recovery_event, !ocfs2_recovery_thread_running(osb));
|
||||
|
||||
/* At this point, we know that no more recovery threads can be
|
||||
@ -1254,8 +1254,7 @@ static int ocfs2_initialize_super(struct super_block *sb,
|
||||
osb->sb = sb;
|
||||
/* Save off for ocfs2_rw_direct */
|
||||
osb->s_sectsize_bits = blksize_bits(sector_size);
|
||||
if (!osb->s_sectsize_bits)
|
||||
BUG();
|
||||
BUG_ON(!osb->s_sectsize_bits);
|
||||
|
||||
osb->net_response_ids = 0;
|
||||
spin_lock_init(&osb->net_response_lock);
|
||||
@ -1283,7 +1282,7 @@ static int ocfs2_initialize_super(struct super_block *sb,
|
||||
snprintf(osb->dev_str, sizeof(osb->dev_str), "%u,%u",
|
||||
MAJOR(osb->sb->s_dev), MINOR(osb->sb->s_dev));
|
||||
|
||||
init_MUTEX(&osb->recovery_lock);
|
||||
mutex_init(&osb->recovery_lock);
|
||||
|
||||
osb->disable_recovery = 0;
|
||||
osb->recovery_thread_task = NULL;
|
||||
|
@ -77,8 +77,7 @@ struct inode *ocfs2_get_system_file_inode(struct ocfs2_super *osb,
|
||||
if (arr && ((inode = *arr) != NULL)) {
|
||||
/* get a ref in addition to the array ref */
|
||||
inode = igrab(inode);
|
||||
if (!inode)
|
||||
BUG();
|
||||
BUG_ON(!inode);
|
||||
|
||||
return inode;
|
||||
}
|
||||
@ -89,8 +88,7 @@ struct inode *ocfs2_get_system_file_inode(struct ocfs2_super *osb,
|
||||
/* add one more if putting into array for first time */
|
||||
if (arr && inode) {
|
||||
*arr = igrab(inode);
|
||||
if (!*arr)
|
||||
BUG();
|
||||
BUG_ON(!*arr);
|
||||
}
|
||||
return inode;
|
||||
}
|
||||
|
@ -388,7 +388,7 @@ out_free:
|
||||
}
|
||||
}
|
||||
|
||||
/* Item insertion is guarded by ip_io_sem, so the insertion path takes
|
||||
/* Item insertion is guarded by ip_io_mutex, so the insertion path takes
|
||||
* advantage of this by not rechecking for a duplicate insert during
|
||||
* the slow case. Additionally, if the cache needs to be bumped up to
|
||||
* a tree, the code will not recheck after acquiring the lock --
|
||||
@ -418,7 +418,7 @@ void ocfs2_set_buffer_uptodate(struct inode *inode,
|
||||
(unsigned long long) bh->b_blocknr);
|
||||
|
||||
/* No need to recheck under spinlock - insertion is guarded by
|
||||
* ip_io_sem */
|
||||
* ip_io_mutex */
|
||||
spin_lock(&oi->ip_lock);
|
||||
if (ocfs2_insert_can_use_array(oi, ci)) {
|
||||
/* Fast case - it's an array and there's a free
|
||||
@ -440,7 +440,7 @@ void ocfs2_set_buffer_uptodate(struct inode *inode,
|
||||
|
||||
/* Called against a newly allocated buffer. Most likely nobody should
|
||||
* be able to read this sort of metadata while it's still being
|
||||
* allocated, but this is careful to take ip_io_sem anyway. */
|
||||
* allocated, but this is careful to take ip_io_mutex anyway. */
|
||||
void ocfs2_set_new_buffer_uptodate(struct inode *inode,
|
||||
struct buffer_head *bh)
|
||||
{
|
||||
@ -451,9 +451,9 @@ void ocfs2_set_new_buffer_uptodate(struct inode *inode,
|
||||
|
||||
set_buffer_uptodate(bh);
|
||||
|
||||
down(&oi->ip_io_sem);
|
||||
mutex_lock(&oi->ip_io_mutex);
|
||||
ocfs2_set_buffer_uptodate(inode, bh);
|
||||
up(&oi->ip_io_sem);
|
||||
mutex_unlock(&oi->ip_io_mutex);
|
||||
}
|
||||
|
||||
/* Requires ip_lock. */
|
||||
@ -537,7 +537,7 @@ int __init init_ocfs2_uptodate_cache(void)
|
||||
return 0;
|
||||
}
|
||||
|
||||
void __exit exit_ocfs2_uptodate_cache(void)
|
||||
void exit_ocfs2_uptodate_cache(void)
|
||||
{
|
||||
if (ocfs2_uptodate_cachep)
|
||||
kmem_cache_destroy(ocfs2_uptodate_cachep);
|
||||
|
@ -27,7 +27,7 @@
|
||||
#define OCFS2_UPTODATE_H
|
||||
|
||||
int __init init_ocfs2_uptodate_cache(void);
|
||||
void __exit exit_ocfs2_uptodate_cache(void);
|
||||
void exit_ocfs2_uptodate_cache(void);
|
||||
|
||||
void ocfs2_metadata_cache_init(struct inode *inode);
|
||||
void ocfs2_metadata_cache_purge(struct inode *inode);
|
||||
|
@ -126,7 +126,7 @@ extern struct config_item *config_group_find_obj(struct config_group *, const ch
|
||||
|
||||
|
||||
struct configfs_attribute {
|
||||
char *ca_name;
|
||||
const char *ca_name;
|
||||
struct module *ca_owner;
|
||||
mode_t ca_mode;
|
||||
};
|
||||
|
Loading…
Reference in New Issue
Block a user