/* -*- c -*- --------------------------------------------------------------- * * * linux/fs/autofs/expire.c * * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved * Copyright 1999-2000 Jeremy Fitzhardinge * * This file is part of the Linux kernel and is made available under * the terms of the GNU General Public License, version 2, or at your * option, any later version, incorporated herein by reference. * * ------------------------------------------------------------------------- */ #include "autofs_i.h" /* * Determine if a subtree of the namespace is busy. * * mnt is the mount tree under the autofs mountpoint */ static inline int is_vfsmnt_tree_busy(struct vfsmount *mnt) { struct vfsmount *this_parent = mnt; struct list_head *next; int count; count = atomic_read(&mnt->mnt_count) - 1; repeat: next = this_parent->mnt_mounts.next; DPRINTK(("is_vfsmnt_tree_busy: mnt=%p, this_parent=%p, next=%p\n", mnt, this_parent, next)); resume: for( ; next != &this_parent->mnt_mounts; next = next->next) { struct vfsmount *p = list_entry(next, struct vfsmount, mnt_child); /* -1 for struct vfs_mount's normal count, -1 to compensate for child's reference to parent */ count += atomic_read(&p->mnt_count) - 1 - 1; DPRINTK(("is_vfsmnt_tree_busy: p=%p, count now %d\n", p, count)); if (!list_empty(&p->mnt_mounts)) { this_parent = p; goto repeat; } /* root is busy if any leaf is busy */ if (atomic_read(&p->mnt_count) > 1) return 1; } /* All done at this level ... ascend and resume the search. */ if (this_parent != mnt) { next = this_parent->mnt_child.next; this_parent = this_parent->mnt_parent; goto resume; } DPRINTK(("is_vfsmnt_tree_busy: count=%d\n", count)); return count != 0; /* remaining users? */ } /* Traverse a dentry's list of vfsmounts and return the number of non-busy mounts */ static int check_vfsmnt(struct vfsmount *mnt, struct dentry *dentry) { int ret = dentry->d_mounted; struct vfsmount *vfs = lookup_mnt(mnt, dentry); if (vfs && is_vfsmnt_tree_busy(vfs)) ret--; DPRINTK(("check_vfsmnt: ret=%d\n", ret)); return ret; } /* Check dentry tree for busyness. If a dentry appears to be busy because it is a mountpoint, check to see if the mounted filesystem is busy. */ static int is_tree_busy(struct vfsmount *topmnt, struct dentry *top) { struct dentry *this_parent; struct list_head *next; int count; count = atomic_read(&top->d_count); DPRINTK(("is_tree_busy: top=%p initial count=%d\n", top, count)); this_parent = top; if (is_autofs4_dentry(top)) { count--; DPRINTK(("is_tree_busy: autofs; count=%d\n", count)); } if (d_mountpoint(top)) count -= check_vfsmnt(topmnt, top); repeat: next = this_parent->d_subdirs.next; resume: while (next != &this_parent->d_subdirs) { int adj = 0; struct dentry *dentry = list_entry(next, struct dentry, d_child); next = next->next; count += atomic_read(&dentry->d_count) - 1; if (d_mountpoint(dentry)) adj += check_vfsmnt(topmnt, dentry); if (is_autofs4_dentry(dentry)) { adj++; DPRINTK(("is_tree_busy: autofs; adj=%d\n", adj)); } count -= adj; if (!list_empty(&dentry->d_subdirs)) { this_parent = dentry; goto repeat; } if (atomic_read(&dentry->d_count) != adj) { DPRINTK(("is_tree_busy: busy leaf (d_count=%d adj=%d)\n", atomic_read(&dentry->d_count), adj)); return 1; } } /* All done at this level ... ascend and resume the search. */ if (this_parent != top) { next = this_parent->d_child.next; this_parent = this_parent->d_parent; goto resume; } DPRINTK(("is_tree_busy: count=%d\n", count)); return count != 0; /* remaining users? */ } /* * Find an eligible tree to time-out * A tree is eligible if :- * - it is unused by any user process * - it has been unused for exp_timeout time */ static struct dentry *autofs4_expire(struct super_block *sb, struct vfsmount *mnt, struct autofs_sb_info *sbi, int do_now) { unsigned long now = jiffies; unsigned long timeout; struct dentry *root = sb->s_root; struct list_head *tmp; if (!sbi->exp_timeout || !root) return NULL; timeout = sbi->exp_timeout; spin_lock(&dcache_lock); for(tmp = root->d_subdirs.next; tmp != &root->d_subdirs; tmp = tmp->next) { struct autofs_info *ino; struct dentry *dentry = list_entry(tmp, struct dentry, d_child); if (dentry->d_inode == NULL) continue; ino = autofs4_dentry_ino(dentry); if (ino == NULL) { /* dentry in the process of being deleted */ continue; } /* No point expiring a pending mount */ if (dentry->d_flags & DCACHE_AUTOFS_PENDING) continue; if (!do_now) { /* Too young to die */ if (time_after(ino->last_used + timeout, now)) continue; /* update last_used here :- - obviously makes sense if it is in use now - less obviously, prevents rapid-fire expire attempts if expire fails the first time */ ino->last_used = now; } if (!is_tree_busy(mnt, dentry)) { DPRINTK(("autofs_expire: returning %p %.*s\n", dentry, (int)dentry->d_name.len, dentry->d_name.name)); /* Start from here next time */ list_del(&root->d_subdirs); list_add(&root->d_subdirs, &dentry->d_child); dget(dentry); spin_unlock(&dcache_lock); return dentry; } } spin_unlock(&dcache_lock); return NULL; } /* Perform an expiry operation */ int autofs4_expire_run(struct super_block *sb, struct vfsmount *mnt, struct autofs_sb_info *sbi, struct autofs_packet_expire *pkt_p) { struct autofs_packet_expire pkt; struct dentry *dentry; memset(&pkt,0,sizeof pkt); pkt.hdr.proto_version = sbi->version; pkt.hdr.type = autofs_ptype_expire; if ((dentry = autofs4_expire(sb, mnt, sbi, 0)) == NULL) return -EAGAIN; pkt.len = dentry->d_name.len; memcpy(pkt.name, dentry->d_name.name, pkt.len); pkt.name[pkt.len] = '\0'; dput(dentry); if ( copy_to_user(pkt_p, &pkt, sizeof(struct autofs_packet_expire)) ) return -EFAULT; return 0; } /* Call repeatedly until it returns -EAGAIN, meaning there's nothing more to be done */ int autofs4_expire_multi(struct super_block *sb, struct vfsmount *mnt, struct autofs_sb_info *sbi, int *arg) { struct dentry *dentry; int ret = -EAGAIN; int do_now = 0; if (arg && get_user(do_now, arg)) return -EFAULT; if ((dentry = autofs4_expire(sb, mnt, sbi, do_now)) != NULL) { struct autofs_info *de_info = autofs4_dentry_ino(dentry); /* This is synchronous because it makes the daemon a little easier */ de_info->flags |= AUTOFS_INF_EXPIRING; ret = autofs4_wait(sbi, &dentry->d_name, NFY_EXPIRE); de_info->flags &= ~AUTOFS_INF_EXPIRING; dput(dentry); } return ret; }