]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
thunderbolt: property: Reject u32 wrap in tb_property_entry_valid()
authorMichael Bommarito <michael.bommarito@gmail.com>
Sun, 10 May 2026 23:16:56 +0000 (19:16 -0400)
committerMika Westerberg <mika.westerberg@linux.intel.com>
Mon, 11 May 2026 09:31:48 +0000 (11:31 +0200)
entry->value is u32 and entry->length is u16; the sum is performed in
u32 and wraps.  A malicious XDomain peer can pick
value = 0xffffff00, length = 0x100 so the sum 0x100000000 wraps to 0
and passes the > block_len check.  tb_property_parse() then passes
entry->value to parse_dwdata() as a dword offset into the property
block, reading attacker-directed memory far past the allocation.

For TEXT-typed entries with the "deviceid" or "vendorid" keys this
lands in xd->device_name / xd->vendor_name and is readable back via
the per-XDomain device_name / vendor_name sysfs attributes; the leak
is NUL-bounded (kstrdup() stops at the first zero byte) and
untargeted (the attacker picks a delta, not an absolute address).
DATA-typed entries are parsed into property->value.data but not
generically surfaced to userspace.

Use check_add_overflow() so a wrapped sum is rejected.

Fixes: cdae7c07e3e3 ("thunderbolt: Add support for XDomain properties")
Cc: stable@vger.kernel.org
Assisted-by: Claude:claude-opus-4-6
Assisted-by: Codex:gpt-5-4
Signed-off-by: Michael Bommarito <michael.bommarito@gmail.com>
Signed-off-by: Mika Westerberg <mika.westerberg@linux.intel.com>
drivers/thunderbolt/property.c

index 50cbfc92fe65ed8fdcc0351afbd590efa2e7e152..29cd60c11ac4cdfda9b24f8936654774a0a7f5f5 100644 (file)
@@ -8,6 +8,7 @@
  */
 
 #include <linux/err.h>
+#include <linux/overflow.h>
 #include <linux/slab.h>
 #include <linux/string.h>
 #include <linux/uuid.h>
@@ -52,13 +53,16 @@ static inline void format_dwdata(void *dst, const void *src, size_t dwords)
 static bool tb_property_entry_valid(const struct tb_property_entry *entry,
                                  size_t block_len)
 {
+       u32 end;
+
        switch (entry->type) {
        case TB_PROPERTY_TYPE_DIRECTORY:
        case TB_PROPERTY_TYPE_DATA:
        case TB_PROPERTY_TYPE_TEXT:
                if (entry->length > block_len)
                        return false;
-               if (entry->value + entry->length > block_len)
+               if (check_add_overflow(entry->value, entry->length, &end) ||
+                   end > block_len)
                        return false;
                break;