Blame SOURCES/autofs-5.1.6-use-struct-mnt_list-to-track-mounted-mounts.patch

9a499a
autofs-5.1.6 - use struct mnt_list to track mounted mounts
9a499a
9a499a
From: Ian Kent <raven@themaw.net>
9a499a
9a499a
Use struct mnt_list to track mounted mounts so that it's no longer
9a499a
necessary to use the system mount table to get the list of mounted
9a499a
mounts.
9a499a
9a499a
Signed-off-by: Ian Kent <raven@themaw.net>
9a499a
---
9a499a
 CHANGELOG          |    1 
9a499a
 daemon/automount.c |    5 ++
9a499a
 daemon/direct.c    |   19 ++++++++
9a499a
 daemon/indirect.c  |    3 +
9a499a
 include/mounts.h   |   10 ++++
9a499a
 lib/mounts.c       |  125 +++++++++++++++++++++++++++++++++++++++++++++++++----
9a499a
 6 files changed, 154 insertions(+), 9 deletions(-)
9a499a
9a499a
--- autofs-5.1.4.orig/CHANGELOG
9a499a
+++ autofs-5.1.4/CHANGELOG
9a499a
@@ -128,6 +128,7 @@ xx/xx/2018 autofs-5.1.5
9a499a
 - remove force parameter from umount_all().
9a499a
 - fix remount expire.
9a499a
 - fix stale offset directories disable mount.
9a499a
+- use struct mnt_list to track mounted mounts.
9a499a
 
9a499a
 19/12/2017 autofs-5.1.4
9a499a
 - fix spec file url.
9a499a
--- autofs-5.1.4.orig/daemon/automount.c
9a499a
+++ autofs-5.1.4/daemon/automount.c
9a499a
@@ -677,6 +677,9 @@ int umount_multi(struct autofs_point *ap
9a499a
 			mnts_put_mount(mnt);
9a499a
 		}
9a499a
 
9a499a
+		/* Check for mounted mount and remove it if found */
9a499a
+		mnts_remove_mount(path, MNTS_MOUNTED);
9a499a
+
9a499a
 		return 0;
9a499a
 	}
9a499a
 
9a499a
@@ -1719,6 +1722,8 @@ static void handle_mounts_cleanup(void *
9a499a
 
9a499a
 		/* Submount at ap->path belongs to parent submount list. */
9a499a
 		mnts_remove_submount(ap->path);
9a499a
+		/* Also remove from parent mounted list */
9a499a
+		mnts_remove_mount(ap->path, MNTS_MOUNTED);
9a499a
 		mnt = mnts_find_amdmount(ap->path);
9a499a
 		if (mnt) {
9a499a
 			mnts_remove_amdmount(ap->path);
9a499a
--- autofs-5.1.4.orig/daemon/direct.c
9a499a
+++ autofs-5.1.4/daemon/direct.c
9a499a
@@ -606,6 +606,9 @@ force_umount:
9a499a
 	} else
9a499a
 		info(ap->logopt, "umounted offset mount %s", me->key);
9a499a
 
9a499a
+	if (!rv)
9a499a
+		mnts_remove_mount(me->key, MNTS_OFFSET);
9a499a
+
9a499a
 	return rv;
9a499a
 }
9a499a
 
9a499a
@@ -622,6 +625,7 @@ int mount_autofs_offset(struct autofs_po
9a499a
 	const char *map_name = hosts_map_name;
9a499a
 	const char *type;
9a499a
 	char mountpoint[PATH_MAX];
9a499a
+	struct mnt_list *mnt;
9a499a
 
9a499a
 	if (ops->version && ap->flags & MOUNT_FLAG_REMOUNT) {
9a499a
 		ret = try_remount(ap, me, t_offset);
9a499a
@@ -635,6 +639,11 @@ int mount_autofs_offset(struct autofs_po
9a499a
 			if (ap->state != ST_READMAP)
9a499a
 				warn(ap->logopt,
9a499a
 				     "trigger %s already mounted", me->key);
9a499a
+			mnt = mnts_add_mount(ap, me->key, MNTS_OFFSET);
9a499a
+			if (!mnt)
9a499a
+				error(ap->logopt,
9a499a
+				      "failed to add offset mount %s to mounted list",
9a499a
+				      me->key);
9a499a
 			return MOUNT_OFFSET_OK;
9a499a
 		}
9a499a
 
9a499a
@@ -757,6 +766,12 @@ int mount_autofs_offset(struct autofs_po
9a499a
 		notify_mount_result(ap, me->key, timeout, str_offset);
9a499a
 	ops->close(ap->logopt, ioctlfd);
9a499a
 
9a499a
+	mnt = mnts_add_mount(ap, mountpoint, MNTS_OFFSET);
9a499a
+	if (!mnt)
9a499a
+		error(ap->logopt,
9a499a
+		      "failed to add offset mount %s to mounted list",
9a499a
+		      mountpoint);
9a499a
+
9a499a
 	debug(ap->logopt, "mounted trigger %s at %s", me->key, mountpoint);
9a499a
 
9a499a
 	return MOUNT_OFFSET_OK;
9a499a
@@ -877,6 +892,7 @@ void *expire_proc_direct(void *arg)
9a499a
 				ops->close(ap->logopt, me->ioctlfd);
9a499a
 				me->ioctlfd = -1;
9a499a
 				cache_unlock(me->mc);
9a499a
+				mnts_remove_mount(next->mp, MNTS_MOUNTED);
9a499a
 				pthread_setcancelstate(cur_state, NULL);
9a499a
 				continue;
9a499a
 			}
9a499a
@@ -1238,7 +1254,10 @@ static void *do_mount_direct(void *arg)
9a499a
 		cache_unlock(mt.mc);
9a499a
 		if (close_fd)
9a499a
 			ops->close(ap->logopt, mt.ioctlfd);
9a499a
+
9a499a
 		info(ap->logopt, "mounted %s", mt.name);
9a499a
+
9a499a
+		mnts_set_mounted_mount(ap, mt.name);
9a499a
 	} else {
9a499a
 		/* TODO: get mount return status from lookup_nss_mount */
9a499a
 		ops->send_fail(ap->logopt,
9a499a
--- autofs-5.1.4.orig/daemon/indirect.c
9a499a
+++ autofs-5.1.4/daemon/indirect.c
9a499a
@@ -751,7 +751,10 @@ static void *do_mount_indirect(void *arg
9a499a
 	if (status) {
9a499a
 		ops->send_ready(ap->logopt,
9a499a
 				ap->ioctlfd, mt.wait_queue_token);
9a499a
+
9a499a
 		info(ap->logopt, "mounted %s", buf);
9a499a
+
9a499a
+		mnts_set_mounted_mount(ap, mt.name);
9a499a
 	} else {
9a499a
 		/* TODO: get mount return status from lookup_nss_mount */
9a499a
 		ops->send_fail(ap->logopt,
9a499a
--- autofs-5.1.4.orig/include/mounts.h
9a499a
+++ autofs-5.1.4/include/mounts.h
9a499a
@@ -39,6 +39,7 @@
9a499a
 #define MNTS_DIRECT	0x0010
9a499a
 #define MNTS_OFFSET	0x0020
9a499a
 #define MNTS_AMD_MOUNT	0x0040
9a499a
+#define MNTS_MOUNTED	0x0080
9a499a
 
9a499a
 #define REMOUNT_SUCCESS		0x0000
9a499a
 #define REMOUNT_FAIL		0x0001
9a499a
@@ -60,6 +61,9 @@ struct mnt_list {
9a499a
 	struct hlist_node hash;
9a499a
 	unsigned int ref;
9a499a
 
9a499a
+	/* List of mounts of an autofs_point */
9a499a
+	struct list_head mount;
9a499a
+
9a499a
 	/* List of sub-mounts of an autofs_point */
9a499a
 	struct autofs_point *ap;
9a499a
 	struct list_head submount;
9a499a
@@ -130,7 +134,13 @@ void mnts_put_submount_list(struct list_
9a499a
 struct mnt_list *mnts_find_amdmount(const char *path);
9a499a
 struct mnt_list *mnts_add_amdmount(struct autofs_point *ap, struct amd_entry *entry);
9a499a
 void mnts_remove_amdmount(const char *mp);
9a499a
+struct mnt_list *mnts_add_mount(struct autofs_point *ap, const char *name, unsigned int flags);
9a499a
+void mnts_remove_mount(const char *mp, unsigned int flags);
9a499a
 struct mnt_list *get_mnt_list(const char *path, int include);
9a499a
+unsigned int mnts_has_mounted_mounts(struct autofs_point *ap);
9a499a
+void mnts_get_expire_list(struct list_head *mnts, struct autofs_point *ap);
9a499a
+void mnts_put_expire_list(struct list_head *mnts);
9a499a
+void mnts_set_mounted_mount(struct autofs_point *ap, const char *name);
9a499a
 int unlink_mount_tree(struct autofs_point *ap, const char *mp);
9a499a
 void free_mnt_list(struct mnt_list *list);
9a499a
 int is_mounted(const char *mp, unsigned int type);
9a499a
--- autofs-5.1.4.orig/lib/mounts.c
9a499a
+++ autofs-5.1.4/lib/mounts.c
9a499a
@@ -884,6 +884,7 @@ static struct mnt_list *mnts_alloc_mount
9a499a
 
9a499a
 	this->ref = 1;
9a499a
 	INIT_HLIST_NODE(&this->hash);
9a499a
+	INIT_LIST_HEAD(&this->mount);
9a499a
 	INIT_LIST_HEAD(&this->submount);
9a499a
 	INIT_LIST_HEAD(&this->submount_work);
9a499a
 	INIT_LIST_HEAD(&this->amdmount);
9a499a
@@ -1149,6 +1150,90 @@ done:
9a499a
 	mnts_hash_mutex_unlock();
9a499a
 }
9a499a
 
9a499a
+struct mnt_list *mnts_add_mount(struct autofs_point *ap,
9a499a
+				const char *name, unsigned int flags)
9a499a
+{
9a499a
+	struct mnt_list *this;
9a499a
+	char *mp;
9a499a
+
9a499a
+	if (*name == '/') {
9a499a
+		mp = strdup(name);
9a499a
+		if (!mp)
9a499a
+			goto fail;
9a499a
+	} else {
9a499a
+		int len = strlen(ap->path) + strlen(name) + 2;
9a499a
+
9a499a
+		mp = malloc(len);
9a499a
+		if (!mp)
9a499a
+			goto fail;
9a499a
+		strcpy(mp, ap->path);
9a499a
+		strcat(mp, "/");
9a499a
+		strcat(mp, name);
9a499a
+	}
9a499a
+
9a499a
+	mnts_hash_mutex_lock();
9a499a
+	this = mnts_get_mount(mp);
9a499a
+	if (this) {
9a499a
+		this->flags |= flags;
9a499a
+		if (list_empty(&this->mount))
9a499a
+			list_add(&this->mount, &ap->mounts);
9a499a
+	}
9a499a
+	mnts_hash_mutex_unlock();
9a499a
+	free(mp);
9a499a
+
9a499a
+	return this;
9a499a
+fail:
9a499a
+	if (mp)
9a499a
+		free(mp);
9a499a
+	return NULL;
9a499a
+}
9a499a
+
9a499a
+void mnts_remove_mount(const char *mp, unsigned int flags)
9a499a
+{
9a499a
+	struct mnt_list *this;
9a499a
+
9a499a
+	mnts_hash_mutex_lock();
9a499a
+	this = mnts_lookup(mp);
9a499a
+	if (this && this->flags & flags) {
9a499a
+		this->flags &= ~flags;
9a499a
+		if (!(this->flags & (MNTS_OFFSET|MNTS_MOUNTED)))
9a499a
+			list_del_init(&this->mount);
9a499a
+		__mnts_put_mount(this);
9a499a
+	}
9a499a
+	mnts_hash_mutex_unlock();
9a499a
+}
9a499a
+
9a499a
+void mnts_set_mounted_mount(struct autofs_point *ap, const char *name)
9a499a
+{
9a499a
+	struct mnt_list *mnt;
9a499a
+
9a499a
+	mnt = mnts_add_mount(ap, name, MNTS_MOUNTED);
9a499a
+	if (!mnt) {
9a499a
+		error(ap->logopt,
9a499a
+		      "failed to add mount %s to mounted list", name);
9a499a
+		return;
9a499a
+	}
9a499a
+
9a499a
+	/* Offset mount failed but non-strict returns success */
9a499a
+	if (mnt->flags & MNTS_OFFSET &&
9a499a
+	    !is_mounted(mnt->mp, MNTS_REAL)) {
9a499a
+		mnt->flags &= ~MNTS_MOUNTED;
9a499a
+		mnts_put_mount(mnt);
9a499a
+	}
9a499a
+
9a499a
+	/* Housekeeping.
9a499a
+	 * Set the base type of the mounted mount.
9a499a
+	 * MNTS_AUTOFS and MNTS_OFFSET are set at mount time and
9a499a
+	 * are used during expire.
9a499a
+	 */
9a499a
+	if (!(mnt->flags & (MNTS_AUTOFS|MNTS_OFFSET))) {
9a499a
+		if (ap->type == LKP_INDIRECT)
9a499a
+			mnt->flags |= MNTS_INDIRECT;
9a499a
+		else
9a499a
+			mnt->flags |= MNTS_DIRECT;
9a499a
+	}
9a499a
+}
9a499a
+
9a499a
 /* From glibc decode_name() */
9a499a
 /* Since the values in a line are separated by spaces, a name cannot
9a499a
  * contain a space.  Therefore some programs encode spaces in names
9a499a
@@ -1962,7 +2047,8 @@ void notify_mount_result(struct autofs_p
9a499a
 	return;
9a499a
 }
9a499a
 
9a499a
-static int do_remount_direct(struct autofs_point *ap, int fd, const char *path)
9a499a
+static int do_remount_direct(struct autofs_point *ap,
9a499a
+			     const unsigned int type, int fd, const char *path)
9a499a
 {
9a499a
 	struct ioctl_ops *ops = get_ioctl_ops();
9a499a
 	int status = REMOUNT_SUCCESS;
9a499a
@@ -1975,9 +2061,21 @@ static int do_remount_direct(struct auto
9a499a
 		set_tsd_user_vars(ap->logopt, uid, gid);
9a499a
 
9a499a
 	ret = lookup_nss_mount(ap, NULL, path, strlen(path));
9a499a
-	if (ret)
9a499a
+	if (ret) {
9a499a
+		struct mnt_list *mnt;
9a499a
+
9a499a
+		/* If it's an offset mount add a mount reference */
9a499a
+		if (type == t_offset) {
9a499a
+			mnt = mnts_add_mount(ap, path, MNTS_OFFSET);
9a499a
+			if (!mnt)
9a499a
+				error(ap->logopt,
9a499a
+				      "failed to add mount %s to mounted list", path);
9a499a
+		}
9a499a
+
9a499a
+		mnts_set_mounted_mount(ap, path);
9a499a
+
9a499a
 		info(ap->logopt, "re-connected to %s", path);
9a499a
-	else {
9a499a
+	} else {
9a499a
 		status = REMOUNT_FAIL;
9a499a
 		info(ap->logopt, "failed to re-connect %s", path);
9a499a
 	}
9a499a
@@ -1985,7 +2083,7 @@ static int do_remount_direct(struct auto
9a499a
 	return status;
9a499a
 }
9a499a
 
9a499a
-static int do_remount_indirect(struct autofs_point *ap, int fd, const char *path)
9a499a
+static int do_remount_indirect(struct autofs_point *ap, const unsigned int type, int fd, const char *path)
9a499a
 {
9a499a
 	struct ioctl_ops *ops = get_ioctl_ops();
9a499a
 	int status = REMOUNT_SUCCESS;
9a499a
@@ -2046,9 +2144,11 @@ static int do_remount_indirect(struct au
9a499a
 		len = strlen(de[n]->d_name);
9a499a
 
9a499a
 		ret = lookup_nss_mount(ap, NULL, de[n]->d_name, len);
9a499a
-		if (ret)
9a499a
+		if (ret) {
9a499a
+			mnts_set_mounted_mount(ap, buf);
9a499a
+
9a499a
 			info(ap->logopt, "re-connected to %s", buf);
9a499a
-		else {
9a499a
+		} else {
9a499a
 			status = REMOUNT_FAIL;
9a499a
 			info(ap->logopt, "failed to re-connect %s", buf);
9a499a
 		}
9a499a
@@ -2149,9 +2249,9 @@ static int remount_active_mount(struct a
9a499a
 		 * following will be broken?
9a499a
 		 */
9a499a
 		if (type == t_indirect)
9a499a
-			do_remount_indirect(ap, fd, path);
9a499a
+			do_remount_indirect(ap, type, fd, path);
9a499a
 		else
9a499a
-			do_remount_direct(ap, fd, path);
9a499a
+			do_remount_direct(ap, type, fd, path);
9a499a
 	}
9a499a
 
9a499a
 	debug(ap->logopt, "re-connected to mount %s", path);
9a499a
@@ -2389,7 +2489,7 @@ int umount_ent(struct autofs_point *ap,
9a499a
 		 * so that we do not try to call rmdir_path on the
9a499a
 		 * directory.
9a499a
 		 */
9a499a
-		if (!rv && is_mounted(path, MNTS_REAL)) {
9a499a
+		if (is_mounted(path, MNTS_REAL)) {
9a499a
 			crit(ap->logopt,
9a499a
 			     "the umount binary reported that %s was "
9a499a
 			     "unmounted, but there is still something "
9a499a
@@ -2398,6 +2498,10 @@ int umount_ent(struct autofs_point *ap,
9a499a
 		}
9a499a
 	}
9a499a
 
9a499a
+	/* On success, check for mounted mount and remove it if found */
9a499a
+	if (!rv)
9a499a
+		mnts_remove_mount(path, MNTS_MOUNTED);
9a499a
+
9a499a
 	return rv;
9a499a
 }
9a499a
 
9a499a
@@ -2690,6 +2794,9 @@ int umount_multi_triggers(struct autofs_
9a499a
 		status = cache_delete_offset_list(mc, me->key);
9a499a
 		if (status != CHE_OK)
9a499a
 			warn(ap->logopt, "couldn't delete offset list");
9a499a
+
9a499a
+	       /* check for mounted mount entry and remove it if found */
9a499a
+               mnts_remove_mount(root, MNTS_MOUNTED);
9a499a
 	}
9a499a
 
9a499a
 	return left;