get rid of ->mnt_longterm
it's enough to set ->mnt_ns of internal vfsmounts to something distinct from all struct mnt_namespace out there; then we can just use the check for ->mnt_ns != NULL in the fast path of mntput_no_expire() Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
parent
d187663ef2
commit
f7a99c5b7c
5 changed files with 26 additions and 72 deletions
|
@ -2622,7 +2622,7 @@ static int prepend_path(const struct path *path,
|
||||||
if (!slash)
|
if (!slash)
|
||||||
error = prepend(buffer, buflen, "/", 1);
|
error = prepend(buffer, buflen, "/", 1);
|
||||||
if (!error)
|
if (!error)
|
||||||
error = real_mount(vfsmnt)->mnt_ns ? 1 : 2;
|
error = is_mounted(vfsmnt) ? 1 : 2;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,18 +6,6 @@
|
||||||
#include <linux/fs_struct.h>
|
#include <linux/fs_struct.h>
|
||||||
#include "internal.h"
|
#include "internal.h"
|
||||||
|
|
||||||
static inline void path_get_longterm(struct path *path)
|
|
||||||
{
|
|
||||||
path_get(path);
|
|
||||||
mnt_make_longterm(path->mnt);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void path_put_longterm(struct path *path)
|
|
||||||
{
|
|
||||||
mnt_make_shortterm(path->mnt);
|
|
||||||
path_put(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Replace the fs->{rootmnt,root} with {mnt,dentry}. Put the old values.
|
* Replace the fs->{rootmnt,root} with {mnt,dentry}. Put the old values.
|
||||||
* It can block.
|
* It can block.
|
||||||
|
@ -26,7 +14,7 @@ void set_fs_root(struct fs_struct *fs, struct path *path)
|
||||||
{
|
{
|
||||||
struct path old_root;
|
struct path old_root;
|
||||||
|
|
||||||
path_get_longterm(path);
|
path_get(path);
|
||||||
spin_lock(&fs->lock);
|
spin_lock(&fs->lock);
|
||||||
write_seqcount_begin(&fs->seq);
|
write_seqcount_begin(&fs->seq);
|
||||||
old_root = fs->root;
|
old_root = fs->root;
|
||||||
|
@ -34,7 +22,7 @@ void set_fs_root(struct fs_struct *fs, struct path *path)
|
||||||
write_seqcount_end(&fs->seq);
|
write_seqcount_end(&fs->seq);
|
||||||
spin_unlock(&fs->lock);
|
spin_unlock(&fs->lock);
|
||||||
if (old_root.dentry)
|
if (old_root.dentry)
|
||||||
path_put_longterm(&old_root);
|
path_put(&old_root);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -45,7 +33,7 @@ void set_fs_pwd(struct fs_struct *fs, struct path *path)
|
||||||
{
|
{
|
||||||
struct path old_pwd;
|
struct path old_pwd;
|
||||||
|
|
||||||
path_get_longterm(path);
|
path_get(path);
|
||||||
spin_lock(&fs->lock);
|
spin_lock(&fs->lock);
|
||||||
write_seqcount_begin(&fs->seq);
|
write_seqcount_begin(&fs->seq);
|
||||||
old_pwd = fs->pwd;
|
old_pwd = fs->pwd;
|
||||||
|
@ -54,7 +42,7 @@ void set_fs_pwd(struct fs_struct *fs, struct path *path)
|
||||||
spin_unlock(&fs->lock);
|
spin_unlock(&fs->lock);
|
||||||
|
|
||||||
if (old_pwd.dentry)
|
if (old_pwd.dentry)
|
||||||
path_put_longterm(&old_pwd);
|
path_put(&old_pwd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int replace_path(struct path *p, const struct path *old, const struct path *new)
|
static inline int replace_path(struct path *p, const struct path *old, const struct path *new)
|
||||||
|
@ -84,7 +72,7 @@ void chroot_fs_refs(struct path *old_root, struct path *new_root)
|
||||||
write_seqcount_end(&fs->seq);
|
write_seqcount_end(&fs->seq);
|
||||||
while (hits--) {
|
while (hits--) {
|
||||||
count++;
|
count++;
|
||||||
path_get_longterm(new_root);
|
path_get(new_root);
|
||||||
}
|
}
|
||||||
spin_unlock(&fs->lock);
|
spin_unlock(&fs->lock);
|
||||||
}
|
}
|
||||||
|
@ -92,13 +80,13 @@ void chroot_fs_refs(struct path *old_root, struct path *new_root)
|
||||||
} while_each_thread(g, p);
|
} while_each_thread(g, p);
|
||||||
read_unlock(&tasklist_lock);
|
read_unlock(&tasklist_lock);
|
||||||
while (count--)
|
while (count--)
|
||||||
path_put_longterm(old_root);
|
path_put(old_root);
|
||||||
}
|
}
|
||||||
|
|
||||||
void free_fs_struct(struct fs_struct *fs)
|
void free_fs_struct(struct fs_struct *fs)
|
||||||
{
|
{
|
||||||
path_put_longterm(&fs->root);
|
path_put(&fs->root);
|
||||||
path_put_longterm(&fs->pwd);
|
path_put(&fs->pwd);
|
||||||
kmem_cache_free(fs_cachep, fs);
|
kmem_cache_free(fs_cachep, fs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -132,9 +120,9 @@ struct fs_struct *copy_fs_struct(struct fs_struct *old)
|
||||||
|
|
||||||
spin_lock(&old->lock);
|
spin_lock(&old->lock);
|
||||||
fs->root = old->root;
|
fs->root = old->root;
|
||||||
path_get_longterm(&fs->root);
|
path_get(&fs->root);
|
||||||
fs->pwd = old->pwd;
|
fs->pwd = old->pwd;
|
||||||
path_get_longterm(&fs->pwd);
|
path_get(&fs->pwd);
|
||||||
spin_unlock(&old->lock);
|
spin_unlock(&old->lock);
|
||||||
}
|
}
|
||||||
return fs;
|
return fs;
|
||||||
|
|
|
@ -50,8 +50,6 @@ extern int copy_mount_string(const void __user *, char **);
|
||||||
extern struct vfsmount *lookup_mnt(struct path *);
|
extern struct vfsmount *lookup_mnt(struct path *);
|
||||||
extern int finish_automount(struct vfsmount *, struct path *);
|
extern int finish_automount(struct vfsmount *, struct path *);
|
||||||
|
|
||||||
extern void mnt_make_longterm(struct vfsmount *);
|
|
||||||
extern void mnt_make_shortterm(struct vfsmount *);
|
|
||||||
extern int sb_prepare_remount_readonly(struct super_block *);
|
extern int sb_prepare_remount_readonly(struct super_block *);
|
||||||
|
|
||||||
extern void __init mnt_init(void);
|
extern void __init mnt_init(void);
|
||||||
|
|
|
@ -22,7 +22,6 @@ struct mount {
|
||||||
struct vfsmount mnt;
|
struct vfsmount mnt;
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
struct mnt_pcp __percpu *mnt_pcp;
|
struct mnt_pcp __percpu *mnt_pcp;
|
||||||
atomic_t mnt_longterm; /* how many of the refs are longterm */
|
|
||||||
#else
|
#else
|
||||||
int mnt_count;
|
int mnt_count;
|
||||||
int mnt_writers;
|
int mnt_writers;
|
||||||
|
@ -49,6 +48,8 @@ struct mount {
|
||||||
int mnt_ghosts;
|
int mnt_ghosts;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#define MNT_NS_INTERNAL ERR_PTR(-EINVAL) /* distinct from any mnt_namespace */
|
||||||
|
|
||||||
static inline struct mount *real_mount(struct vfsmount *mnt)
|
static inline struct mount *real_mount(struct vfsmount *mnt)
|
||||||
{
|
{
|
||||||
return container_of(mnt, struct mount, mnt);
|
return container_of(mnt, struct mount, mnt);
|
||||||
|
@ -59,6 +60,12 @@ static inline int mnt_has_parent(struct mount *mnt)
|
||||||
return mnt != mnt->mnt_parent;
|
return mnt != mnt->mnt_parent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int is_mounted(struct vfsmount *mnt)
|
||||||
|
{
|
||||||
|
/* neither detached nor internal? */
|
||||||
|
return !IS_ERR_OR_NULL(real_mount(mnt));
|
||||||
|
}
|
||||||
|
|
||||||
extern struct mount *__lookup_mnt(struct vfsmount *, struct dentry *, int);
|
extern struct mount *__lookup_mnt(struct vfsmount *, struct dentry *, int);
|
||||||
|
|
||||||
static inline void get_mnt_ns(struct mnt_namespace *ns)
|
static inline void get_mnt_ns(struct mnt_namespace *ns)
|
||||||
|
|
|
@ -621,21 +621,6 @@ static void attach_mnt(struct mount *mnt, struct path *path)
|
||||||
list_add_tail(&mnt->mnt_child, &real_mount(path->mnt)->mnt_mounts);
|
list_add_tail(&mnt->mnt_child, &real_mount(path->mnt)->mnt_mounts);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void __mnt_make_longterm(struct mount *mnt)
|
|
||||||
{
|
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
atomic_inc(&mnt->mnt_longterm);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/* needs vfsmount lock for write */
|
|
||||||
static inline void __mnt_make_shortterm(struct mount *mnt)
|
|
||||||
{
|
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
atomic_dec(&mnt->mnt_longterm);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* vfsmount lock must be held for write
|
* vfsmount lock must be held for write
|
||||||
*/
|
*/
|
||||||
|
@ -649,10 +634,8 @@ static void commit_tree(struct mount *mnt)
|
||||||
BUG_ON(parent == mnt);
|
BUG_ON(parent == mnt);
|
||||||
|
|
||||||
list_add_tail(&head, &mnt->mnt_list);
|
list_add_tail(&head, &mnt->mnt_list);
|
||||||
list_for_each_entry(m, &head, mnt_list) {
|
list_for_each_entry(m, &head, mnt_list)
|
||||||
m->mnt_ns = n;
|
m->mnt_ns = n;
|
||||||
__mnt_make_longterm(m);
|
|
||||||
}
|
|
||||||
|
|
||||||
list_splice(&head, n->list.prev);
|
list_splice(&head, n->list.prev);
|
||||||
|
|
||||||
|
@ -804,7 +787,8 @@ static void mntput_no_expire(struct mount *mnt)
|
||||||
put_again:
|
put_again:
|
||||||
#ifdef CONFIG_SMP
|
#ifdef CONFIG_SMP
|
||||||
br_read_lock(&vfsmount_lock);
|
br_read_lock(&vfsmount_lock);
|
||||||
if (likely(atomic_read(&mnt->mnt_longterm))) {
|
if (likely(mnt->mnt_ns)) {
|
||||||
|
/* shouldn't be the last one */
|
||||||
mnt_add_count(mnt, -1);
|
mnt_add_count(mnt, -1);
|
||||||
br_read_unlock(&vfsmount_lock);
|
br_read_unlock(&vfsmount_lock);
|
||||||
return;
|
return;
|
||||||
|
@ -1074,8 +1058,6 @@ void umount_tree(struct mount *mnt, int propagate, struct list_head *kill)
|
||||||
list_del_init(&p->mnt_expire);
|
list_del_init(&p->mnt_expire);
|
||||||
list_del_init(&p->mnt_list);
|
list_del_init(&p->mnt_list);
|
||||||
__touch_mnt_namespace(p->mnt_ns);
|
__touch_mnt_namespace(p->mnt_ns);
|
||||||
if (p->mnt_ns)
|
|
||||||
__mnt_make_shortterm(p);
|
|
||||||
p->mnt_ns = NULL;
|
p->mnt_ns = NULL;
|
||||||
list_del_init(&p->mnt_child);
|
list_del_init(&p->mnt_child);
|
||||||
if (mnt_has_parent(p)) {
|
if (mnt_has_parent(p)) {
|
||||||
|
@ -2209,23 +2191,6 @@ static struct mnt_namespace *alloc_mnt_ns(void)
|
||||||
return new_ns;
|
return new_ns;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mnt_make_longterm(struct vfsmount *mnt)
|
|
||||||
{
|
|
||||||
__mnt_make_longterm(real_mount(mnt));
|
|
||||||
}
|
|
||||||
|
|
||||||
void mnt_make_shortterm(struct vfsmount *m)
|
|
||||||
{
|
|
||||||
#ifdef CONFIG_SMP
|
|
||||||
struct mount *mnt = real_mount(m);
|
|
||||||
if (atomic_add_unless(&mnt->mnt_longterm, -1, 1))
|
|
||||||
return;
|
|
||||||
br_write_lock(&vfsmount_lock);
|
|
||||||
atomic_dec(&mnt->mnt_longterm);
|
|
||||||
br_write_unlock(&vfsmount_lock);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Allocate a new namespace structure and populate it with contents
|
* Allocate a new namespace structure and populate it with contents
|
||||||
* copied from the namespace of the passed in task structure.
|
* copied from the namespace of the passed in task structure.
|
||||||
|
@ -2265,18 +2230,13 @@ static struct mnt_namespace *dup_mnt_ns(struct mnt_namespace *mnt_ns,
|
||||||
q = new;
|
q = new;
|
||||||
while (p) {
|
while (p) {
|
||||||
q->mnt_ns = new_ns;
|
q->mnt_ns = new_ns;
|
||||||
__mnt_make_longterm(q);
|
|
||||||
if (fs) {
|
if (fs) {
|
||||||
if (&p->mnt == fs->root.mnt) {
|
if (&p->mnt == fs->root.mnt) {
|
||||||
fs->root.mnt = mntget(&q->mnt);
|
fs->root.mnt = mntget(&q->mnt);
|
||||||
__mnt_make_longterm(q);
|
|
||||||
mnt_make_shortterm(&p->mnt);
|
|
||||||
rootmnt = &p->mnt;
|
rootmnt = &p->mnt;
|
||||||
}
|
}
|
||||||
if (&p->mnt == fs->pwd.mnt) {
|
if (&p->mnt == fs->pwd.mnt) {
|
||||||
fs->pwd.mnt = mntget(&q->mnt);
|
fs->pwd.mnt = mntget(&q->mnt);
|
||||||
__mnt_make_longterm(q);
|
|
||||||
mnt_make_shortterm(&p->mnt);
|
|
||||||
pwdmnt = &p->mnt;
|
pwdmnt = &p->mnt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2320,7 +2280,6 @@ static struct mnt_namespace *create_mnt_ns(struct vfsmount *m)
|
||||||
if (!IS_ERR(new_ns)) {
|
if (!IS_ERR(new_ns)) {
|
||||||
struct mount *mnt = real_mount(m);
|
struct mount *mnt = real_mount(m);
|
||||||
mnt->mnt_ns = new_ns;
|
mnt->mnt_ns = new_ns;
|
||||||
__mnt_make_longterm(mnt);
|
|
||||||
new_ns->root = mnt;
|
new_ns->root = mnt;
|
||||||
list_add(&new_ns->list, &mnt->mnt_list);
|
list_add(&new_ns->list, &mnt->mnt_list);
|
||||||
} else {
|
} else {
|
||||||
|
@ -2615,7 +2574,7 @@ struct vfsmount *kern_mount_data(struct file_system_type *type, void *data)
|
||||||
* it is a longterm mount, don't release mnt until
|
* it is a longterm mount, don't release mnt until
|
||||||
* we unmount before file sys is unregistered
|
* we unmount before file sys is unregistered
|
||||||
*/
|
*/
|
||||||
mnt_make_longterm(mnt);
|
real_mount(mnt)->mnt_ns = MNT_NS_INTERNAL;
|
||||||
}
|
}
|
||||||
return mnt;
|
return mnt;
|
||||||
}
|
}
|
||||||
|
@ -2625,7 +2584,9 @@ void kern_unmount(struct vfsmount *mnt)
|
||||||
{
|
{
|
||||||
/* release long term mount so mount point can be released */
|
/* release long term mount so mount point can be released */
|
||||||
if (!IS_ERR_OR_NULL(mnt)) {
|
if (!IS_ERR_OR_NULL(mnt)) {
|
||||||
mnt_make_shortterm(mnt);
|
br_write_lock(&vfsmount_lock);
|
||||||
|
real_mount(mnt)->mnt_ns = NULL;
|
||||||
|
br_write_unlock(&vfsmount_lock);
|
||||||
mntput(mnt);
|
mntput(mnt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue