mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-12-27 11:06:41 -05:00
gendwarfksyms: Add a kABI rule to override byte_size attributes
A data structure can be partially opaque to modules if its
allocation is handled by the core kernel, and modules only need
to access some of its members. In this situation, it's possible
to append new members to the structure without breaking the ABI,
as long as the layout for the original members remains unchanged.
For example, consider the following struct:
struct s {
unsigned long a;
void *p;
};
gendwarfksyms --stable --dump-dies produces the following type
expansion:
variable structure_type s {
member base_type long unsigned int byte_size(8) encoding(7) a
data_member_location(0) ,
member pointer_type {
base_type void
} byte_size(8) p data_member_location(8)
} byte_size(16)
To append new members, we can use the KABI_IGNORE() macro to
hide them from gendwarfksyms --stable:
struct s {
/* old members with unchanged layout */
unsigned long a;
void *p;
/* new members not accessed by modules */
KABI_IGNORE(0, unsigned long n);
};
However, we can't hide the fact that adding new members changes
the struct size, as seen in the updated type string:
variable structure_type s {
member base_type long unsigned int byte_size(8) encoding(7) a
data_member_location(0) ,
member pointer_type {
base_type void
} byte_size(8) p data_member_location(8)
} byte_size(24)
In order to support this use case, add a kABI rule that makes it
possible to override the byte_size attribute for types:
/*
* struct s allocation is handled by the kernel, so
* appending new members without changing the original
* layout won't break the ABI.
*/
KABI_BYTE_SIZE(s, 16);
This results in a type string that's unchanged from the original
and therefore, won't change versions for symbols that reference
the changed structure.
Signed-off-by: Sami Tolvanen <samitolvanen@google.com>
Reviewed-by: Petr Pavlu <petr.pavlu@suse.com>
Signed-off-by: Masahiro Yamada <masahiroy@kernel.org>
This commit is contained in:
committed by
Masahiro Yamada
parent
ff2c5f5a9e
commit
db59d74e5d
@@ -228,12 +228,24 @@ static void process_fqn(struct die *cache, Dwarf_Die *die)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(accessibility)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(alignment)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(bit_size)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(byte_size)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(encoding)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(data_bit_offset)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(data_member_location)
|
||||
DEFINE_PROCESS_UDATA_ATTRIBUTE(discr_value)
|
||||
|
||||
static void process_byte_size_attr(struct die *cache, Dwarf_Die *die)
|
||||
{
|
||||
Dwarf_Word value;
|
||||
unsigned long override;
|
||||
|
||||
if (get_udata_attr(die, DW_AT_byte_size, &value)) {
|
||||
if (stable && kabi_get_byte_size(cache->fqn, &override))
|
||||
value = override;
|
||||
|
||||
process_fmt(cache, " byte_size(%" PRIu64 ")", value);
|
||||
}
|
||||
}
|
||||
|
||||
/* Match functions -- die_match_callback_t */
|
||||
#define DEFINE_MATCH(type) \
|
||||
static bool match_##type##_type(Dwarf_Die *die) \
|
||||
|
||||
@@ -89,6 +89,13 @@
|
||||
#define KABI_ENUMERATOR_VALUE(fqn, field, value) \
|
||||
__KABI_RULE(enumerator_value, fqn field, value)
|
||||
|
||||
/*
|
||||
* KABI_BYTE_SIZE(fqn, value)
|
||||
* Set the byte_size attribute for the struct/union/enum fqn to
|
||||
* value bytes.
|
||||
*/
|
||||
#define KABI_BYTE_SIZE(fqn, value) __KABI_RULE(byte_size, fqn, value)
|
||||
|
||||
/*
|
||||
* KABI_RESERVE
|
||||
* Reserve some "padding" in a structure for use by LTS backports.
|
||||
|
||||
@@ -28,3 +28,5 @@ struct ex2c ex2c;
|
||||
struct ex3a ex3a;
|
||||
struct ex3b ex3b;
|
||||
struct ex3c ex3c;
|
||||
|
||||
struct ex4a ex4a;
|
||||
|
||||
@@ -260,4 +260,26 @@ _Static_assert(sizeof(struct ex3a) == sizeof(struct ex3c), "ex3a size doesn't ma
|
||||
* STABLE-NEXT: } byte_size(16)
|
||||
*/
|
||||
|
||||
/*
|
||||
* Example: An ignored field added to an end of a partially opaque struct,
|
||||
* while keeping the byte_size attribute unchanged.
|
||||
*/
|
||||
|
||||
struct ex4a {
|
||||
unsigned long a;
|
||||
KABI_IGNORE(0, unsigned long b);
|
||||
};
|
||||
|
||||
/*
|
||||
* This may be safe if the structure allocation is managed by the core kernel
|
||||
* and the layout remains unchanged except for appended new members.
|
||||
*/
|
||||
KABI_BYTE_SIZE(ex4a, 8);
|
||||
|
||||
/*
|
||||
* STABLE: variable structure_type ex4a {
|
||||
* STABLE-NEXT: member base_type [[ULONG]] byte_size(8) encoding(7) a data_member_location(0)
|
||||
* STABLE-NEXT: } byte_size(8)
|
||||
*/
|
||||
|
||||
#endif /* __KABI_EX_H__ */
|
||||
|
||||
@@ -287,6 +287,7 @@ void generate_symtypes_and_versions(FILE *file);
|
||||
* kabi.c
|
||||
*/
|
||||
|
||||
bool kabi_get_byte_size(const char *fqn, unsigned long *value);
|
||||
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);
|
||||
|
||||
@@ -54,11 +54,19 @@
|
||||
*/
|
||||
#define KABI_RULE_TAG_ENUMERATOR_VALUE "enumerator_value"
|
||||
|
||||
/*
|
||||
* Rule: byte_size
|
||||
* - For the fqn_field in the target field, set the byte_size
|
||||
* attribute to the value in the value field.
|
||||
*/
|
||||
#define KABI_RULE_TAG_BYTE_SIZE "byte_size"
|
||||
|
||||
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,
|
||||
};
|
||||
|
||||
#define RULE_HASH_BITS 7
|
||||
@@ -127,6 +135,10 @@ void kabi_read_rules(int fd)
|
||||
.type = KABI_RULE_TYPE_ENUMERATOR_VALUE,
|
||||
.tag = KABI_RULE_TAG_ENUMERATOR_VALUE,
|
||||
},
|
||||
{
|
||||
.type = KABI_RULE_TYPE_BYTE_SIZE,
|
||||
.tag = KABI_RULE_TAG_BYTE_SIZE,
|
||||
},
|
||||
};
|
||||
|
||||
if (!stable)
|
||||
@@ -308,6 +320,19 @@ bool kabi_get_enumerator_value(const char *fqn, const char *field,
|
||||
return false;
|
||||
}
|
||||
|
||||
bool kabi_get_byte_size(const char *fqn, unsigned long *value)
|
||||
{
|
||||
struct rule *rule;
|
||||
|
||||
rule = find_rule(KABI_RULE_TYPE_BYTE_SIZE, fqn);
|
||||
if (rule) {
|
||||
*value = get_ulong_value(rule->value);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void kabi_free(void)
|
||||
{
|
||||
struct hlist_node *tmp;
|
||||
|
||||
Reference in New Issue
Block a user