Blame SOURCES/0117-libmultipath-cleanup-remove_feature.patch

108c2a
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
108c2a
From: Benjamin Marzinski <bmarzins@redhat.com>
108c2a
Date: Fri, 7 Oct 2022 12:35:37 -0500
108c2a
Subject: [PATCH] libmultipath: cleanup remove_feature
108c2a
108c2a
remove_feature() didn't correctly handle feature strings that used
108c2a
whitespace other than spaces, which the kernel allows. It also didn't
108c2a
check if the feature string to be removed was part of a larger feature
108c2a
token. Finally, it did a lot of unnecessary work. By failing if the
108c2a
feature string to be removed contains leading or trailing whitespace,
108c2a
the function can be significanly simplified.
108c2a
108c2a
Signed-off-by: Benjamin Marzinski <bmarzins@redhat.com>
108c2a
Reviewed-by: Martin Wilck <mwilck@suse.com>
108c2a
---
108c2a
 libmultipath/structs.c | 82 +++++++++++++++---------------------------
108c2a
 1 file changed, 29 insertions(+), 53 deletions(-)
108c2a
108c2a
diff --git a/libmultipath/structs.c b/libmultipath/structs.c
108c2a
index 9f86eb69..471087e2 100644
108c2a
--- a/libmultipath/structs.c
108c2a
+++ b/libmultipath/structs.c
108c2a
@@ -6,6 +6,7 @@
108c2a
 #include <unistd.h>
108c2a
 #include <libdevmapper.h>
108c2a
 #include <libudev.h>
108c2a
+#include <ctype.h>
108c2a
 
108c2a
 #include "checkers.h"
108c2a
 #include "memory.h"
108c2a
@@ -633,7 +634,7 @@ int add_feature(char **f, const char *n)
108c2a
 
108c2a
 int remove_feature(char **f, const char *o)
108c2a
 {
108c2a
-	int c = 0, d, l;
108c2a
+	int c = 0, d;
108c2a
 	char *e, *p, *n;
108c2a
 	const char *q;
108c2a
 
108c2a
@@ -644,33 +645,35 @@ int remove_feature(char **f, const char *o)
108c2a
 	if (!o || *o == '\0')
108c2a
 		return 0;
108c2a
 
108c2a
-	/* Check if not present */
108c2a
-	if (!strstr(*f, o))
108c2a
+	d = strlen(o);
108c2a
+	if (isspace(*o) || isspace(*(o + d - 1))) {
108c2a
+		condlog(0, "internal error: feature \"%s\" has leading or trailing spaces", o);
108c2a
+		return 1;
108c2a
+	}
108c2a
+
108c2a
+	/* Check if present and not part of a larger feature token*/
108c2a
+	p = *f + 1; /* the size must be at the start of the features string */
108c2a
+	while ((p = strstr(p, o)) != NULL) {
108c2a
+		if (isspace(*(p - 1)) &&
108c2a
+		    (isspace(*(p + d)) || *(p + d) == '\0'))
108c2a
+			break;
108c2a
+		p += d;
108c2a
+	}
108c2a
+	if (!p)
108c2a
 		return 0;
108c2a
 
108c2a
 	/* Get feature count */
108c2a
 	c = strtoul(*f, &e, 10);
108c2a
-	if (*f == e)
108c2a
-		/* parse error */
108c2a
+	if (*f == e || !isspace(*e)) {
108c2a
+		condlog(0, "parse error in feature string \"%s\"", *f);
108c2a
 		return 1;
108c2a
-
108c2a
-	/* Normalize features */
108c2a
-	while (*o == ' ') {
108c2a
-		o++;
108c2a
 	}
108c2a
-	/* Just spaces, return */
108c2a
-	if (*o == '\0')
108c2a
-		return 0;
108c2a
-	q = o + strlen(o);
108c2a
-	while (*q == ' ')
108c2a
-		q--;
108c2a
-	d = (int)(q - o);
108c2a
 
108c2a
 	/* Update feature count */
108c2a
 	c--;
108c2a
 	q = o;
108c2a
-	while (q[0] != '\0') {
108c2a
-		if (q[0] == ' ' && q[1] != ' ' && q[1] != '\0')
108c2a
+	while (*q != '\0') {
108c2a
+		if (isspace(*q) && !isspace(*(q + 1)) && *(q + 1) != '\0')
108c2a
 			c--;
108c2a
 		q++;
108c2a
 	}
108c2a
@@ -684,15 +687,8 @@ int remove_feature(char **f, const char *o)
108c2a
 		goto out;
108c2a
 	}
108c2a
 
108c2a
-	/* Search feature to be removed */
108c2a
-	e = strstr(*f, o);
108c2a
-	if (!e)
108c2a
-		/* Not found, return */
108c2a
-		return 0;
108c2a
-
108c2a
 	/* Update feature count space */
108c2a
-	l = strlen(*f) - d;
108c2a
-	n =  MALLOC(l + 1);
108c2a
+	n =  MALLOC(strlen(*f) - d + 1);
108c2a
 	if (!n)
108c2a
 		return 1;
108c2a
 
108c2a
@@ -702,36 +698,16 @@ int remove_feature(char **f, const char *o)
108c2a
 	 * Copy existing features up to the feature
108c2a
 	 * about to be removed
108c2a
 	 */
108c2a
-	p = strchr(*f, ' ');
108c2a
-	if (!p) {
108c2a
-		/* Internal error, feature string inconsistent */
108c2a
-		FREE(n);
108c2a
-		return 1;
108c2a
-	}
108c2a
-	while (*p == ' ')
108c2a
-		p++;
108c2a
-	p--;
108c2a
-	if (e != p) {
108c2a
-		do {
108c2a
-			e--;
108c2a
-			d++;
108c2a
-		} while (*e == ' ');
108c2a
-		e++; d--;
108c2a
-		strncat(n, p, (size_t)(e - p));
108c2a
-		p += (size_t)(e - p);
108c2a
-	}
108c2a
+	strncat(n, e, (size_t)(p - e));
108c2a
 	/* Skip feature to be removed */
108c2a
 	p += d;
108c2a
-
108c2a
 	/* Copy remaining features */
108c2a
-	if (strlen(p)) {
108c2a
-		while (*p == ' ')
108c2a
-			p++;
108c2a
-		if (strlen(p)) {
108c2a
-			p--;
108c2a
-			strcat(n, p);
108c2a
-		}
108c2a
-	}
108c2a
+	while (isspace(*p))
108c2a
+		p++;
108c2a
+	if (*p != '\0')
108c2a
+		strcat(n, p);
108c2a
+	else
108c2a
+		strchop(n);
108c2a
 
108c2a
 out:
108c2a
 	FREE(*f);