]> www.infradead.org Git - linux.git/commitdiff
gendwarfksyms: Add a kABI rule to override type strings
authorSami Tolvanen <samitolvanen@google.com>
Wed, 7 May 2025 23:14:07 +0000 (23:14 +0000)
committerMasahiro Yamada <masahiroy@kernel.org>
Sun, 25 May 2025 09:12:23 +0000 (18:12 +0900)
In rare situations where distributions must make significant
changes to otherwise opaque data structures that have
inadvertently been included in the published ABI, keeping
symbol versions stable using the existing kABI macros can
become tedious.

For example, Android decided to switch to a newer io_uring
implementation in the 5.10 GKI kernel "to resolve a huge number
of potential, and known, problems with the codebase," requiring
"horrible hacks" with genksyms:

  "A number of the io_uring structures get used in other core
  kernel structures, only as "opaque" pointers, so there is
  not any real ABI breakage.  But, due to the visibility of
  the structures going away, the CRC values of many scheduler
  variables and functions were changed."
    -- https://r.android.com/2425293

While these specific changes probably could have been hidden
from gendwarfksyms using the existing kABI macros, this may not
always be the case.

Add a last resort kABI rule that allows distribution
maintainers to fully override a type string for a symbol or a
type. Also add a more informative error message in case we find
a non-existent type references when calculating versions.

Suggested-by: Giuliano Procida <gprocida@google.com>
Signed-off-by: Sami Tolvanen <samitolvanen@google.com>
Reviewed-by: Petr Pavlu <petr.pavlu@suse.com>
Signed-off-by: Masahiro Yamada <masahiroy@kernel.org>
scripts/gendwarfksyms/examples/kabi.h
scripts/gendwarfksyms/examples/kabi_ex.c
scripts/gendwarfksyms/examples/kabi_ex.h
scripts/gendwarfksyms/gendwarfksyms.h
scripts/gendwarfksyms/kabi.c
scripts/gendwarfksyms/types.c

index 86f4428e047954071c9b302e2f74984d8ee6c56c..170733a3fba477577ebaf5cace1d4418169b0c6d 100644 (file)
 #define __stringify(x...) __stringify_1(x)
 #endif
 
-#define __KABI_RULE(hint, target, value)                             \
+#define ___KABI_RULE(hint, target, value)                            \
        static const char __PASTE(__gendwarfksyms_rule_,             \
                                  __COUNTER__)[] __used __aligned(1) \
                __section(".discard.gendwarfksyms.kabi_rules") =     \
-                       "1\0" #hint "\0" #target "\0" #value
+                       "1\0" #hint "\0" target "\0" value
+
+#define __KABI_RULE(hint, target, value) \
+       ___KABI_RULE(hint, #target, #value)
 
 #define __KABI_NORMAL_SIZE_ALIGN(_orig, _new)                                             \
        union {                                                                           \
  */
 #define KABI_BYTE_SIZE(fqn, value) __KABI_RULE(byte_size, fqn, value)
 
+/*
+ * KABI_TYPE_STRING(type, str)
+ *   For the given type, override the type string used in symtypes
+ *   output and version calculation with str.
+ */
+#define KABI_TYPE_STRING(type, str) ___KABI_RULE(type_string, type, str)
+
 /*
  * KABI_RESERVE
  *   Reserve some "padding" in a structure for use by LTS backports.
index b73ee5399a59ce2cfd2576d8f5ead7f0539f7663..1f799eb7c7568014c90bb304b4f670090966cf5f 100644 (file)
@@ -30,3 +30,8 @@ struct ex3b ex3b;
 struct ex3c ex3c;
 
 struct ex4a ex4a;
+
+struct ex5a ex5a;
+struct ex5b ex5b;
+
+int ex6a;
index 092c8cb7bcd7cc585c1285cdf835b6f60cf70590..785b211d9c58a98540723c1eff691ddfcca0ae6c 100644 (file)
  *     ./gendwarfksyms --stable --dump-dies \
  *             examples/kabi_ex.o 2>&1 >/dev/null | \
  *     FileCheck examples/kabi_ex.h --check-prefix=STABLE
+
+ * $ nm examples/kabi_ex.o | awk '{ print $NF }' | \
+ *     ./gendwarfksyms --stable --dump-versions \
+ *             examples/kabi_ex.o 2>&1 >/dev/null | \
+ *     sort | \
+ *     FileCheck examples/kabi_ex.h --check-prefix=VERSIONS
  */
 
 #ifndef __KABI_EX_H__
@@ -170,7 +176,7 @@ struct ex2a {
 /*
  * STABLE:      variable structure_type ex2a {
  * STABLE-NEXT:   member base_type int byte_size(4) encoding(5) a data_member_location(0) ,
- * STABLE-NEXT:   member base_type [[ULONG:long unsigned int|unsigned long]] byte_size(8) encoding(7) b data_member_location(8)
+ * STABLE-NEXT:   member base_type [[ULONG]] byte_size(8) encoding(7) b data_member_location(8)
  * STABLE-NEXT:   member base_type int byte_size(4) encoding(5) c data_member_location(16) ,
  * STABLE-NEXT:   member base_type [[ULONG]] byte_size(8) encoding(7) d data_member_location(24)
  * STABLE-NEXT: } byte_size(32)
@@ -227,7 +233,7 @@ struct ex3a {
 
 /*
  * STABLE:      variable structure_type ex3a {
- * STABLE-NEXT:   member base_type [[ULONG:long unsigned int|unsigned long]] byte_size(8) encoding(7) a data_member_location(0)
+ * STABLE-NEXT:   member base_type [[ULONG]] byte_size(8) encoding(7) a data_member_location(0)
  * STABLE-NEXT:   member base_type [[ULONG]] byte_size(8) encoding(7) unused data_member_location(8)
  * STABLE-NEXT: } byte_size(16)
  */
@@ -282,4 +288,73 @@ KABI_BYTE_SIZE(ex4a, 8);
  * STABLE-NEXT: } byte_size(8)
  */
 
+/*
+ * Example: A type string override.
+ */
+
+struct ex5a {
+       unsigned long a;
+};
+
+/*
+ * This may be safe if the structure is fully opaque to modules, even though
+ * its definition has inadvertently become part of the ABI.
+ */
+KABI_TYPE_STRING(
+       "s#ex5a",
+       "structure_type ex5a { member pointer_type { s#ex4a } byte_size(8) p data_member_location(0) } byte_size(8)");
+
+/*
+ * Make sure the fully expanded type string includes ex4a.
+ *
+ * VERSIONS:      ex5a variable structure_type ex5a {
+ * VERSIONS-SAME:   member pointer_type {
+ * VERSIONS-SAME:     structure_type ex4a {
+ * VERSIONS-SAME:       member base_type [[ULONG:long unsigned int|unsigned long]] byte_size(8) encoding(7) a data_member_location(0)
+ * VERSIONS-SAME:     } byte_size(8)
+ * VERSIONS-SAME:   } byte_size(8) p data_member_location(0)
+ * VERSIONS-SAME: } byte_size(8)
+ */
+
+/*
+ * Example: A type string definition for a non-existent type.
+ */
+
+struct ex5b {
+       unsigned long a;
+};
+
+/* Replace the type string for struct ex5b */
+KABI_TYPE_STRING(
+       "s#ex5b",
+       "structure_type ex5b { member pointer_type { s#ex5c } byte_size(8) p data_member_location(0) } byte_size(8)");
+
+/* Define a type string for a non-existent struct ex5c */
+KABI_TYPE_STRING(
+       "s#ex5c",
+       "structure_type ex5c { member base_type int byte_size(4) encoding(5) n data_member_location(0) } byte_size(8)");
+
+/*
+ * Make sure the fully expanded type string includes the definition for ex5c.
+ *
+ * VERSIONS:      ex5b variable structure_type ex5b {
+ * VERSIONS-SAME:   member pointer_type {
+ * VERSIONS-SAME:     structure_type ex5c {
+ * VERSIONS-SAME:       member base_type int byte_size(4) encoding(5) n data_member_location(0)
+ * VERSIONS-SAME:     } byte_size(8)
+ * VERSIONS-SAME:   } byte_size(8) p data_member_location(0)
+ * VERSIONS-SAME: } byte_size(8)
+ */
+
+/*
+ * Example: A type string override for a symbol.
+ */
+
+KABI_TYPE_STRING("ex6a", "variable s#ex5c");
+
+/*
+ * VERSIONS:      ex6a variable structure_type ex5c {
+ * VERSIONS-SAME:   member base_type int byte_size(4) encoding(5) n data_member_location(0)
+ * VERSIONS-SAME: } byte_size(8)
+ */
 #endif /* __KABI_EX_H__ */
index 2db49c2ad50e9f5aa69d968ba3e98042f84f6d6e..7dd03ffe0c5c829d32cadfe9ac9109ed7455dd92 100644 (file)
@@ -292,6 +292,7 @@ bool kabi_is_enumerator_ignored(const char *fqn, const char *field);
 bool kabi_get_enumerator_value(const char *fqn, const char *field,
                               unsigned long *value);
 bool kabi_is_declonly(const char *fqn);
+bool kabi_get_type_string(const char *type, const char **str);
 
 void kabi_read_rules(int fd);
 void kabi_free(void);
index 61620ff647bd18c683550d89540d941bd6b17eba..b3ade713778f1be725a70fd1209c7ccb4a387d0e 100644 (file)
  */
 #define KABI_RULE_TAG_BYTE_SIZE "byte_size"
 
+/*
+ * Rule: type_string
+ * - For the type reference in the fqn field, use the type string
+ *   in the value field.
+ */
+#define KABI_RULE_TAG_TYPE_STRING "type_string"
+
 enum kabi_rule_type {
        KABI_RULE_TYPE_UNKNOWN,
        KABI_RULE_TYPE_DECLONLY,
        KABI_RULE_TYPE_ENUMERATOR_IGNORE,
        KABI_RULE_TYPE_ENUMERATOR_VALUE,
        KABI_RULE_TYPE_BYTE_SIZE,
+       KABI_RULE_TYPE_TYPE_STRING,
 };
 
 #define RULE_HASH_BITS 7
@@ -139,6 +147,10 @@ void kabi_read_rules(int fd)
                        .type = KABI_RULE_TYPE_BYTE_SIZE,
                        .tag = KABI_RULE_TAG_BYTE_SIZE,
                },
+               {
+                       .type = KABI_RULE_TYPE_TYPE_STRING,
+                       .tag = KABI_RULE_TAG_TYPE_STRING,
+               },
        };
 
        if (!stable)
@@ -333,6 +345,19 @@ bool kabi_get_byte_size(const char *fqn, unsigned long *value)
        return false;
 }
 
+bool kabi_get_type_string(const char *type, const char **str)
+{
+       struct rule *rule;
+
+       rule = find_rule(KABI_RULE_TYPE_TYPE_STRING, type);
+       if (rule) {
+               *str = rule->value;
+               return true;
+       }
+
+       return false;
+}
+
 void kabi_free(void)
 {
        struct hlist_node *tmp;
index 6f37289104ffdc52f393e0566e90924809bf7bb6..39ce1770e46364bafa275debf9327ec130b6bc4e 100644 (file)
@@ -100,7 +100,7 @@ static void type_expansion_append(struct type_expansion *type, const char *s,
 #define TYPE_HASH_BITS 12
 static HASHTABLE_DEFINE(type_map, 1 << TYPE_HASH_BITS);
 
-static int type_map_get(const char *name, struct type_expansion **res)
+static int __type_map_get(const char *name, struct type_expansion **res)
 {
        struct type_expansion *e;
 
@@ -114,11 +114,12 @@ static int type_map_get(const char *name, struct type_expansion **res)
        return -1;
 }
 
-static void type_map_add(const char *name, struct type_expansion *type)
+static struct type_expansion *type_map_add(const char *name,
+                                          struct type_expansion *type)
 {
        struct type_expansion *e;
 
-       if (type_map_get(name, &e)) {
+       if (__type_map_get(name, &e)) {
                e = xmalloc(sizeof(struct type_expansion));
                type_expansion_init(e);
                e->name = xstrdup(name);
@@ -130,7 +131,7 @@ static void type_map_add(const char *name, struct type_expansion *type)
        } else {
                /* Use the longest available expansion */
                if (type->len <= e->len)
-                       return;
+                       return e;
 
                type_list_free(&e->expanded);
 
@@ -148,6 +149,34 @@ static void type_map_add(const char *name, struct type_expansion *type)
                type_list_write(&e->expanded, stderr);
                checkp(fputs("\n", stderr));
        }
+
+       return e;
+}
+
+static void type_parse(const char *name, const char *str,
+                      struct type_expansion *type);
+
+static int type_map_get(const char *name, struct type_expansion **res)
+{
+       struct type_expansion type;
+       const char *override;
+
+       if (!__type_map_get(name, res))
+               return 0;
+
+       /*
+        * If die_map didn't contain a type, we might still have
+        * a type_string kABI rule that defines it.
+        */
+       if (stable && kabi_get_type_string(name, &override)) {
+               type_expansion_init(&type);
+               type_parse(name, override, &type);
+               *res = type_map_add(name, &type);
+               type_expansion_free(&type);
+               return 0;
+       }
+
+       return -1;
 }
 
 static void type_map_write(FILE *file)
@@ -267,15 +296,18 @@ static char *get_type_name(struct die *cache)
        return name;
 }
 
-static void __calculate_version(struct version *version, struct list_head *list)
+static void __calculate_version(struct version *version,
+                               struct type_expansion *type)
 {
        struct type_list_entry *entry;
        struct type_expansion *e;
 
        /* Calculate a CRC over an expanded type string */
-       list_for_each_entry(entry, list, list) {
+       list_for_each_entry(entry, &type->expanded, list) {
                if (is_type_prefix(entry->str)) {
-                       check(type_map_get(entry->str, &e));
+                       if (type_map_get(entry->str, &e))
+                               error("unknown type reference to '%s' when expanding '%s'",
+                                     entry->str, type->name);
 
                        /*
                         * It's sufficient to expand each type reference just
@@ -285,7 +317,7 @@ static void __calculate_version(struct version *version, struct list_head *list)
                                version_add(version, entry->str);
                        } else {
                                cache_mark_expanded(&expansion_cache, e);
-                               __calculate_version(version, &e->expanded);
+                               __calculate_version(version, e);
                        }
                } else {
                        version_add(version, entry->str);
@@ -293,10 +325,11 @@ static void __calculate_version(struct version *version, struct list_head *list)
        }
 }
 
-static void calculate_version(struct version *version, struct list_head *list)
+static void calculate_version(struct version *version,
+                             struct type_expansion *type)
 {
        version_init(version);
-       __calculate_version(version, list);
+       __calculate_version(version, type);
        cache_free(&expansion_cache);
 }
 
@@ -372,9 +405,80 @@ static void type_expand(struct die *cache, struct type_expansion *type,
        cache_free(&expansion_cache);
 }
 
+static void type_parse(const char *name, const char *str,
+                      struct type_expansion *type)
+{
+       char *fragment;
+       size_t start = 0;
+       size_t end;
+       size_t pos;
+
+       if (!*str)
+               error("empty type string override for '%s'", name);
+
+       type_expansion_init(type);
+
+       for (pos = 0; str[pos]; ++pos) {
+               bool empty;
+               char marker = ' ';
+
+               if (!is_type_prefix(&str[pos]))
+                       continue;
+
+               end = pos + 2;
+
+               /*
+                * Find the end of the type reference. If the type name contains
+                * spaces, it must be in single quotes.
+                */
+               if (str[end] == '\'') {
+                       marker = '\'';
+                       ++end;
+               }
+               while (str[end] && str[end] != marker)
+                       ++end;
+
+               /* Check that we have a non-empty type name */
+               if (marker == '\'') {
+                       if (str[end] != marker)
+                               error("incomplete %c# type reference for '%s' (string : '%s')",
+                                     str[pos], name, str);
+                       empty = end == pos + 3;
+                       ++end;
+               } else {
+                       empty = end == pos + 2;
+               }
+               if (empty)
+                       error("empty %c# type name for '%s' (string: '%s')",
+                             str[pos], name, str);
+
+               /* Append the part of the string before the type reference */
+               if (pos > start) {
+                       fragment = xstrndup(&str[start], pos - start);
+                       type_expansion_append(type, fragment, fragment);
+               }
+
+               /*
+                * Append the type reference -- note that if the reference
+                * is invalid, i.e. points to a non-existent type, we will
+                * print out an error when calculating versions.
+                */
+               fragment = xstrndup(&str[pos], end - pos);
+               type_expansion_append(type, fragment, fragment);
+
+               start = end;
+               pos = end - 1;
+       }
+
+       /* Append the rest of the type string, if there's any left */
+       if (str[start])
+               type_expansion_append(type, &str[start], NULL);
+}
+
 static void expand_type(struct die *cache, void *arg)
 {
        struct type_expansion type;
+       const char *override;
        char *name;
 
        if (cache->mapped)
@@ -399,9 +503,13 @@ static void expand_type(struct die *cache, void *arg)
                return;
 
        debug("%s", name);
-       type_expand(cache, &type, true);
-       type_map_add(name, &type);
 
+       if (stable && kabi_get_type_string(name, &override))
+               type_parse(name, override, &type);
+       else
+               type_expand(cache, &type, true);
+
+       type_map_add(name, &type);
        type_expansion_free(&type);
        free(name);
 }
@@ -410,6 +518,7 @@ static void expand_symbol(struct symbol *sym, void *arg)
 {
        struct type_expansion type;
        struct version version;
+       const char *override;
        struct die *cache;
 
        /*
@@ -423,11 +532,14 @@ static void expand_symbol(struct symbol *sym, void *arg)
        if (__die_map_get(sym->die_addr, DIE_SYMBOL, &cache))
                return; /* We'll warn about missing CRCs later. */
 
-       type_expand(cache, &type, false);
+       if (stable && kabi_get_type_string(sym->name, &override))
+               type_parse(sym->name, override, &type);
+       else
+               type_expand(cache, &type, false);
 
        /* If the symbol already has a version, don't calculate it again. */
        if (sym->state != SYMBOL_PROCESSED) {
-               calculate_version(&version, &type.expanded);
+               calculate_version(&version, &type);
                symbol_set_crc(sym, version.crc);
                debug("%s = %lx", sym->name, version.crc);