Zbigniew Jędrzejewski-Szmek 77e6ed
From 533cc35f09181971821d94b6e4ce242b4b966583 Mon Sep 17 00:00:00 2001
Zbigniew Jędrzejewski-Szmek 77e6ed
From: Michal Schmidt <mschmidt@redhat.com>
Zbigniew Jędrzejewski-Szmek 77e6ed
Date: Mon, 16 Mar 2015 21:58:35 +0100
Zbigniew Jędrzejewski-Szmek 77e6ed
Subject: [PATCH] shared: add path_compare(), an ordering path comparison
Zbigniew Jędrzejewski-Szmek 77e6ed
Zbigniew Jędrzejewski-Szmek 77e6ed
... and make path_equal() a simple wrapper around it.
Zbigniew Jędrzejewski-Szmek 77e6ed
Zbigniew Jędrzejewski-Szmek 77e6ed
(cherry picked from commit 2230852bd9755e1b7bfd1260082471f559b0a005)
Zbigniew Jędrzejewski-Szmek 77e6ed
---
Zbigniew Jędrzejewski-Szmek 77e6ed
 src/shared/path-util.c    | 37 +++++++++++++++++++++++++++----------
Zbigniew Jędrzejewski-Szmek 77e6ed
 src/shared/path-util.h    |  1 +
Zbigniew Jędrzejewski-Szmek 77e6ed
 src/test/test-path-util.c | 36 +++++++++++++++++++++++++-----------
Zbigniew Jędrzejewski-Szmek 77e6ed
 3 files changed, 53 insertions(+), 21 deletions(-)
Zbigniew Jędrzejewski-Szmek 77e6ed
Zbigniew Jędrzejewski-Szmek 77e6ed
diff --git a/src/shared/path-util.c b/src/shared/path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
index 70bc1caa2a..d5510bf56f 100644
Zbigniew Jędrzejewski-Szmek 77e6ed
--- a/src/shared/path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
+++ b/src/shared/path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
@@ -403,12 +403,18 @@ char* path_startswith(const char *path, const char *prefix) {
Zbigniew Jędrzejewski-Szmek 77e6ed
         }
Zbigniew Jędrzejewski-Szmek 77e6ed
 }
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-bool path_equal(const char *a, const char *b) {
Zbigniew Jędrzejewski-Szmek 77e6ed
+int path_compare(const char *a, const char *b) {
Zbigniew Jędrzejewski-Szmek 77e6ed
+        int d;
Zbigniew Jędrzejewski-Szmek 77e6ed
+
Zbigniew Jędrzejewski-Szmek 77e6ed
         assert(a);
Zbigniew Jędrzejewski-Szmek 77e6ed
         assert(b);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-        if ((a[0] == '/') != (b[0] == '/'))
Zbigniew Jędrzejewski-Szmek 77e6ed
-                return false;
Zbigniew Jędrzejewski-Szmek 77e6ed
+        /* A relative path and an abolute path must not compare as equal.
Zbigniew Jędrzejewski-Szmek 77e6ed
+         * Which one is sorted before the other does not really matter.
Zbigniew Jędrzejewski-Szmek 77e6ed
+         * Here a relative path is ordered before an absolute path. */
Zbigniew Jędrzejewski-Szmek 77e6ed
+        d = (a[0] == '/') - (b[0] == '/');
Zbigniew Jędrzejewski-Szmek 77e6ed
+        if (d)
Zbigniew Jędrzejewski-Szmek 77e6ed
+                return d;
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
         for (;;) {
Zbigniew Jędrzejewski-Szmek 77e6ed
                 size_t j, k;
Zbigniew Jędrzejewski-Szmek 77e6ed
@@ -417,25 +423,36 @@ bool path_equal(const char *a, const char *b) {
Zbigniew Jędrzejewski-Szmek 77e6ed
                 b += strspn(b, "/");
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
                 if (*a == 0 && *b == 0)
Zbigniew Jędrzejewski-Szmek 77e6ed
-                        return true;
Zbigniew Jędrzejewski-Szmek 77e6ed
+                        return 0;
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-                if (*a == 0 || *b == 0)
Zbigniew Jędrzejewski-Szmek 77e6ed
-                        return false;
Zbigniew Jędrzejewski-Szmek 77e6ed
+                /* Order prefixes first: "/foo" before "/foo/bar" */
Zbigniew Jędrzejewski-Szmek 77e6ed
+                if (*a == 0)
Zbigniew Jędrzejewski-Szmek 77e6ed
+                        return -1;
Zbigniew Jędrzejewski-Szmek 77e6ed
+                if (*b == 0)
Zbigniew Jędrzejewski-Szmek 77e6ed
+                        return 1;
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
                 j = strcspn(a, "/");
Zbigniew Jędrzejewski-Szmek 77e6ed
                 k = strcspn(b, "/");
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-                if (j != k)
Zbigniew Jędrzejewski-Szmek 77e6ed
-                        return false;
Zbigniew Jędrzejewski-Szmek 77e6ed
+                /* Alphabetical sort: "/foo/aaa" before "/foo/b" */
Zbigniew Jędrzejewski-Szmek 77e6ed
+                d = memcmp(a, b, MIN(j, k));
Zbigniew Jędrzejewski-Szmek 77e6ed
+                if (d)
Zbigniew Jędrzejewski-Szmek 77e6ed
+                        return (d > 0) - (d < 0); /* sign of d */
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-                if (memcmp(a, b, j) != 0)
Zbigniew Jędrzejewski-Szmek 77e6ed
-                        return false;
Zbigniew Jędrzejewski-Szmek 77e6ed
+                /* Sort "/foo/a" before "/foo/aaa" */
Zbigniew Jędrzejewski-Szmek 77e6ed
+                d = (j > k) - (j < k);  /* sign of (j - k) */
Zbigniew Jędrzejewski-Szmek 77e6ed
+                if (d)
Zbigniew Jędrzejewski-Szmek 77e6ed
+                        return d;
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
                 a += j;
Zbigniew Jędrzejewski-Szmek 77e6ed
                 b += k;
Zbigniew Jędrzejewski-Szmek 77e6ed
         }
Zbigniew Jędrzejewski-Szmek 77e6ed
 }
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
+bool path_equal(const char *a, const char *b) {
Zbigniew Jędrzejewski-Szmek 77e6ed
+        return path_compare(a, b) == 0;
Zbigniew Jędrzejewski-Szmek 77e6ed
+}
Zbigniew Jędrzejewski-Szmek 77e6ed
+
Zbigniew Jędrzejewski-Szmek 77e6ed
 bool path_equal_or_files_same(const char *a, const char *b) {
Zbigniew Jędrzejewski-Szmek 77e6ed
         return path_equal(a, b) || files_same(a, b) > 0;
Zbigniew Jędrzejewski-Szmek 77e6ed
 }
Zbigniew Jędrzejewski-Szmek 77e6ed
diff --git a/src/shared/path-util.h b/src/shared/path-util.h
Zbigniew Jędrzejewski-Szmek 77e6ed
index bcf116ed3d..ca81b49cbf 100644
Zbigniew Jędrzejewski-Szmek 77e6ed
--- a/src/shared/path-util.h
Zbigniew Jędrzejewski-Szmek 77e6ed
+++ b/src/shared/path-util.h
Zbigniew Jędrzejewski-Szmek 77e6ed
@@ -44,6 +44,7 @@ char* path_make_absolute_cwd(const char *p);
Zbigniew Jędrzejewski-Szmek 77e6ed
 int path_make_relative(const char *from_dir, const char *to_path, char **_r);
Zbigniew Jędrzejewski-Szmek 77e6ed
 char* path_kill_slashes(char *path);
Zbigniew Jędrzejewski-Szmek 77e6ed
 char* path_startswith(const char *path, const char *prefix) _pure_;
Zbigniew Jędrzejewski-Szmek 77e6ed
+int path_compare(const char *a, const char *b) _pure_;
Zbigniew Jędrzejewski-Szmek 77e6ed
 bool path_equal(const char *a, const char *b) _pure_;
Zbigniew Jędrzejewski-Szmek 77e6ed
 bool path_equal_or_files_same(const char *a, const char *b);
Zbigniew Jędrzejewski-Szmek 77e6ed
 char* path_join(const char *root, const char *path, const char *rest);
Zbigniew Jędrzejewski-Szmek 77e6ed
diff --git a/src/test/test-path-util.c b/src/test/test-path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
index 11aa52aaed..6396fcb398 100644
Zbigniew Jędrzejewski-Szmek 77e6ed
--- a/src/test/test-path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
+++ b/src/test/test-path-util.c
Zbigniew Jędrzejewski-Szmek 77e6ed
@@ -27,23 +27,37 @@
Zbigniew Jędrzejewski-Szmek 77e6ed
 #include "macro.h"
Zbigniew Jędrzejewski-Szmek 77e6ed
 #include "strv.h"
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
+#define test_path_compare(a, b, result) {                 \
Zbigniew Jędrzejewski-Szmek 77e6ed
+                assert_se(path_compare(a, b) == result);  \
Zbigniew Jędrzejewski-Szmek 77e6ed
+                assert_se(path_compare(b, a) == -result); \
Zbigniew Jędrzejewski-Szmek 77e6ed
+                assert_se(path_equal(a, b) == !result);   \
Zbigniew Jędrzejewski-Szmek 77e6ed
+                assert_se(path_equal(b, a) == !result);   \
Zbigniew Jędrzejewski-Szmek 77e6ed
+        }
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
 static void test_path(void) {
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("/goo", "/goo"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("//goo", "/goo"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("//goo/////", "/goo"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("goo/////", "goo"));
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/goo", "/goo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/goo", "/goo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("//goo", "/goo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("//goo/////", "/goo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("goo/////", "goo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/goo/boo", "/goo//boo", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("//goo/boo", "/goo/boo//", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("/goo/boo", "/goo//boo"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("//goo/boo", "/goo/boo//"));
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/", "///", 0);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(path_equal("/", "///"));
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/x", "x/", 1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("x/", "/", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(!path_equal("/x", "x/"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(!path_equal("x/", "/"));
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/x/./y", "x/y", 1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("x/.y", "x/y", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(!path_equal("/x/./y", "x/y"));
Zbigniew Jędrzejewski-Szmek 77e6ed
-        assert_se(!path_equal("x/.y", "x/y"));
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("foo", "/foo", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/foo", "/foo/bar", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/foo/aaa", "/foo/b", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/foo/aaa", "/foo/b/a", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/foo/a", "/foo/aaa", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
+        test_path_compare("/foo/a/b", "/foo/aaa", -1);
Zbigniew Jędrzejewski-Szmek 77e6ed
 
Zbigniew Jędrzejewski-Szmek 77e6ed
         assert_se(path_is_absolute("/"));
Zbigniew Jędrzejewski-Szmek 77e6ed
         assert_se(!path_is_absolute("./"));