dcavalca / rpms / util-linux

Forked from rpms/util-linux 2 years ago
Clone
05ad79
From 597807396a00e6f31aa03805ddca27875b81440b Mon Sep 17 00:00:00 2001
05ad79
From: Aurelien Aptel <aaptel@suse.com>
05ad79
Date: Thu, 1 Sep 2016 13:57:42 +0200
05ad79
Subject: [PATCH 182/185] libmount: fix mount -a for cifs
05ad79
05ad79
when mounting a cifs share, the src is actually an UNC path which can in
05ad79
in several forms:
05ad79
05ad79
simple:            //host/share, //host/share/
05ad79
including subpath: //host/share/sub/path
05ad79
05ad79
to check if the cifs fs is mounted we have to extract the subpath and
05ad79
compare *that* to the root.
05ad79
05ad79
Addresses: https://bugzilla.redhat.com/show_bug.cgi?id=1633657
05ad79
Upstream: http://github.com/karelzak/util-linux/commit/76d4fba2e47808264f5c4c883127b0d275e31949
05ad79
Signed-off-by: Aurelien Aptel <aaptel@suse.com>
05ad79
Signed-off-by: Karel Zak <kzak@redhat.com>
05ad79
---
05ad79
 include/strutils.h |  1 +
05ad79
 lib/strutils.c     | 50 ++++++++++++++++++++++++++++++++++++++++++++++
05ad79
 libmount/src/tab.c | 29 ++++++++++++++++++++++++---
05ad79
 3 files changed, 77 insertions(+), 3 deletions(-)
05ad79
05ad79
diff --git a/include/strutils.h b/include/strutils.h
05ad79
index 822fb7d49..83fdc7ead 100644
05ad79
--- a/include/strutils.h
05ad79
+++ b/include/strutils.h
05ad79
@@ -108,6 +108,7 @@ extern int string_to_bitmask(const char *list,
05ad79
 extern int parse_range(const char *str, int *lower, int *upper, int def);
05ad79
 
05ad79
 extern int streq_except_trailing_slash(const char *s1, const char *s2);
05ad79
+extern int streq_paths(const char *a, const char *b);
05ad79
 
05ad79
 extern char *strnappend(const char *s, const char *suffix, size_t b);
05ad79
 extern char *strappend(const char *s, const char *suffix);
05ad79
diff --git a/lib/strutils.c b/lib/strutils.c
05ad79
index 2458a2c2f..2f264daeb 100644
05ad79
--- a/lib/strutils.c
05ad79
+++ b/lib/strutils.c
05ad79
@@ -697,6 +697,56 @@ int streq_except_trailing_slash(const char *s1, const char *s2)
05ad79
 	return equal;
05ad79
 }
05ad79
 
05ad79
+static const char *next_path_segment(const char *str, size_t *sz)
05ad79
+{
05ad79
+	const char *start, *p;
05ad79
+
05ad79
+	start = str;
05ad79
+	*sz = 0;
05ad79
+	while (start && *start == '/' && *(start + 1) == '/')
05ad79
+		start++;
05ad79
+
05ad79
+	if (!start || !*start)
05ad79
+		return NULL;
05ad79
+
05ad79
+	for (*sz = 1, p = start + 1; *p && *p != '/'; p++) {
05ad79
+		(*sz)++;
05ad79
+	}
05ad79
+
05ad79
+	return start;
05ad79
+}
05ad79
+
05ad79
+int streq_paths(const char *a, const char *b)
05ad79
+{
05ad79
+	while (a && b) {
05ad79
+		size_t a_sz, b_sz;
05ad79
+		const char *a_seg = next_path_segment(a, &a_sz);
05ad79
+		const char *b_seg = next_path_segment(b, &b_sz);
05ad79
+
05ad79
+		/*
05ad79
+		fprintf(stderr, "A>>>(%zu) '%s'\n", a_sz, a_seg);
05ad79
+		fprintf(stderr, "B>>>(%zu) '%s'\n", b_sz, b_seg);
05ad79
+		*/
05ad79
+
05ad79
+		/* end of the path */
05ad79
+		if (a_sz + b_sz == 0)
05ad79
+			return 1;
05ad79
+
05ad79
+		/* ignore tailing slash */
05ad79
+		if (a_sz + b_sz == 1 &&
05ad79
+		    ((a_seg && *a_seg == '/') || (b_seg && *b_seg == '/')))
05ad79
+			return 1;
05ad79
+
05ad79
+		if (a_sz != b_sz || strncmp(a_seg, b_seg, a_sz) != 0)
05ad79
+			return 0;
05ad79
+
05ad79
+		a = a_seg + a_sz;
05ad79
+		b = b_seg + b_sz;
05ad79
+	};
05ad79
+
05ad79
+	return 0;
05ad79
+}
05ad79
+
05ad79
 char *strnappend(const char *s, const char *suffix, size_t b)
05ad79
 {
05ad79
         size_t a;
05ad79
diff --git a/libmount/src/tab.c b/libmount/src/tab.c
05ad79
index dfa1da822..3ee9c0042 100644
05ad79
--- a/libmount/src/tab.c
05ad79
+++ b/libmount/src/tab.c
05ad79
@@ -1053,6 +1053,22 @@ static int is_mountinfo(struct libmnt_table *tb)
05ad79
 	return 0;
05ad79
 }
05ad79
 
05ad79
+
05ad79
+
05ad79
+static const char *get_cifs_unc_subdir_path (const char *unc)
05ad79
+{
05ad79
+	/*
05ad79
+	 *  1 or more slash:     %*[/]
05ad79
+	 *  1 or more non-slash: %*[^/]
05ad79
+	 *  number of byte read: %n
05ad79
+	 */
05ad79
+	int share_end = 0;
05ad79
+	int r = sscanf(unc, "%*[/]%*[^/]%*[/]%*[^/]%n", &share_end);
05ad79
+	if (r == EOF || share_end == 0)
05ad79
+		return NULL;
05ad79
+	return unc + share_end;
05ad79
+}
05ad79
+
05ad79
 /**
05ad79
  * mnt_table_is_mounted:
05ad79
  * @tb: /proc/self/mountinfo file
05ad79
@@ -1150,9 +1166,16 @@ int mnt_table_is_fs_mounted(struct libmnt_table *tb, struct libmnt_fs *fstab_fs)
05ad79
 		}
05ad79
 
05ad79
 		if (root) {
05ad79
-			const char *r = mnt_fs_get_root(fs);
05ad79
-			if (!r || strcmp(r, root) != 0)
05ad79
-				continue;
05ad79
+			if (strcmp(mnt_fs_get_fstype(fs), "cifs") == 0) {
05ad79
+				const char *unc_subdir = get_cifs_unc_subdir_path(src);
05ad79
+				const char *path_on_fs = mnt_fs_get_root(fs);
05ad79
+				if (!unc_subdir || !path_on_fs || !streq_paths(unc_subdir, path_on_fs))
05ad79
+					continue;
05ad79
+			} else {
05ad79
+				const char *r = mnt_fs_get_root(fs);
05ad79
+				if (!r || strcmp(r, root) != 0)
05ad79
+					continue;
05ad79
+			}
05ad79
 		}
05ad79
 
05ad79
 		/*
05ad79
-- 
05ad79
2.20.1
05ad79