]> git.ipfire.org Git - thirdparty/systemd.git/commitdiff
cap-list: add CAPABILITY_TO_STRING() macro using compound initialization to allocate...
authorLennart Poettering <lennart@poettering.net>
Mon, 20 Feb 2023 10:43:13 +0000 (11:43 +0100)
committerLennart Poettering <lennart@poettering.net>
Mon, 20 Feb 2023 15:27:26 +0000 (16:27 +0100)
Let's add a helper that can return a numeric string in case we don't
recognize a name for a capability.

src/basic/cap-list.c
src/basic/cap-list.h
src/test/test-cap-list.c

index bcf7e597c2def35b50dc32395a81274d6e7942d2..7843efc99496459e17736558abc435550cfb12b2 100644 (file)
@@ -19,13 +19,28 @@ static const struct capability_name* lookup_capability(register const char *str,
 const char *capability_to_name(int id) {
         if (id < 0)
                 return NULL;
-
         if ((size_t) id >= ELEMENTSOF(capability_names))
                 return NULL;
 
         return capability_names[id];
 }
 
+const char *capability_to_string(int id, char buf[static CAPABILITY_TO_STRING_MAX]) {
+        const char *p;
+
+        if (id < 0)
+                return NULL;
+        if (id >= 63) /* refuse caps >= 63 since we can't store them in a uint64_t mask anymore, and still retain UINT64_MAX as marker for "unset" */
+                return NULL;
+
+        p = capability_to_name(id);
+        if (p)
+                return p;
+
+        sprintf(buf, "0x%x", (unsigned) id); /* numerical fallback */
+        return buf;
+}
+
 int capability_from_name(const char *name) {
         const struct capability_name *sc;
         int r, i;
index 888e9223e7b41b66885091e02e19b0cb6d8c8945..bddadcbae5f171ba3917ef1b78dca832680fba07 100644 (file)
@@ -3,7 +3,15 @@
 
 #include <inttypes.h>
 
+/* Space for capability_to_string() in case we write out a numeric capability because we don't know the name
+ * for it. "0x3e" is the largest string we might output, in both sensese of the word "largest": two chars for
+ * "0x", two bytes for the hex value, and one trailing NUL byte. */
+#define CAPABILITY_TO_STRING_MAX (2 + 2 + 1)
+
 const char *capability_to_name(int id);
+const char *capability_to_string(int id, char buf[static CAPABILITY_TO_STRING_MAX]);
+#define CAPABILITY_TO_STRING(id) capability_to_string(id, (char[CAPABILITY_TO_STRING_MAX]) {})
+
 int capability_from_name(const char *name);
 int capability_list_length(void);
 
index 517641a019a2f0e19564153b1a090d1df5e06ff9..8df425bdd29f9d21ed27fae82e2bc51ed6430228 100644 (file)
 TEST(cap_list) {
         assert_se(!capability_to_name(-1));
         assert_se(!capability_to_name(capability_list_length()));
+        assert_se(!capability_to_name(63));
+        assert_se(!capability_to_name(64));
+
+        assert_se(!CAPABILITY_TO_STRING(-1));
+        if (capability_list_length() <= 62)
+                assert_se(streq(CAPABILITY_TO_STRING(62), "0x3e"));
+        assert_se(!CAPABILITY_TO_STRING(64));
 
         for (int i = 0; i < capability_list_length(); i++) {
                 const char *n;
@@ -21,6 +28,8 @@ TEST(cap_list) {
                 assert_se(n = capability_to_name(i));
                 assert_se(capability_from_name(n) == i);
                 printf("%s = %i\n", n, i);
+
+                assert_se(streq(CAPABILITY_TO_STRING(i), n));
         }
 
         assert_se(capability_from_name("asdfbsd") == -EINVAL);