В ядре Linux устранена следующая уязвимость:
gue: Исправлена утечка памяти skb с внутренним IP-протоколом 0.
syzbot сообщил об утечке памяти skb ниже. [0]
Воспроизведение сгенерировало пакет GUE с внутренним протоколом 0.
gue_udp_recv() возвращает -guehdr->proto_ctype для «повторной отправки»
в ip_protocol_deliver_rcu(), но это работает только с
ненулевой номер протокола. Давайте отбрасывать такие пакеты. Обратите внимание, что 0 — допустимое число (опция IPv6 Hop-by-Hop).
Я думаю, что HOPOPT нецелесообразно инкапировать в GUE, поэтому однажды
кто-то начинает жаловаться, мы могли бы отправить повторную заявку
указатель флага, чтобы отличить два нуля от верхнего слоя:
* нет ошибки
* повторно отправить HOPOPT
[0]
ОШИБКА: утечка памяти.
объект без ссылки 0xffff888109695a00 (размер 240):
comm "syz.0.17", pid 6088, jiffies 4294943096
шестнадцатеричный дамп (первые 32 байта):
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ..............
00 40 c2 10 81 88 ff ff 00 00 00 00 00 00 00 00 .@..............
обратная трассировка (crc a84b336f):
kmemleak_alloc_recursive include/linux/kmemleak.h:44 [встроенный]
slab_post_alloc_hook мм/slub.c:4958 [встроенный]
slab_alloc_node мм/slub.c:5263 [встроенный]
kmem_cache_alloc_noprof+0x3b4/0x590 мм/slub.c:5270
__build_skb+0x23/0x60 net/core/skbuff.c:474
build_skb+0x20/0x190 net/core/skbuff.c:490
__tun_build_skb driver/net/tun.c:1541 [встроенный]
tun_build_skb+0x4a1/0xa40 драйверы/net/tun.c:1636
tun_get_user+0xc12/0x2030 драйверы/net/tun.c:1770
tun_chr_write_iter+0x71/0x120 driver/net/tun.c:1999
new_sync_write fs/read_write.c:593 [встроенный]
vfs_write+0x45d/0x710 fs/read_write.c:686
ksys_write+0xa7/0x170 fs/read_write.c:738
do_syscall_x64 Arch/x86/entry/syscall_64.c:63 [встроенный]
do_syscall_64+0xa4/0xf80 Arch/x86/entry/syscall_64.c:94
запись_SYSCALL_64_after_hwframe+0x77/0x7f
Показать оригинальное описание (EN)
In the Linux kernel, the following vulnerability has been resolved: gue: Fix skb memleak with inner IP protocol 0. syzbot reported skb memleak below. [0] The repro generated a GUE packet with its inner protocol 0. gue_udp_recv() returns -guehdr->proto_ctype for "resubmit" in ip_protocol_deliver_rcu(), but this only works with non-zero protocol number. Let's drop such packets. Note that 0 is a valid number (IPv6 Hop-by-Hop Option). I think it is not practical to encap HOPOPT in GUE, so once someone starts to complain, we could pass down a resubmit flag pointer to distinguish two zeros from the upper layer: * no error * resubmit HOPOPT [0] BUG: memory leak unreferenced object 0xffff888109695a00 (size 240): comm "syz.0.17", pid 6088, jiffies 4294943096 hex dump (first 32 bytes): 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 00 40 c2 10 81 88 ff ff 00 00 00 00 00 00 00 00 .@.............. backtrace (crc a84b336f): kmemleak_alloc_recursive include/linux/kmemleak.h:44 [inline] slab_post_alloc_hook mm/slub.c:4958 [inline] slab_alloc_node mm/slub.c:5263 [inline] kmem_cache_alloc_noprof+0x3b4/0x590 mm/slub.c:5270 __build_skb+0x23/0x60 net/core/skbuff.c:474 build_skb+0x20/0x190 net/core/skbuff.c:490 __tun_build_skb drivers/net/tun.c:1541 [inline] tun_build_skb+0x4a1/0xa40 drivers/net/tun.c:1636 tun_get_user+0xc12/0x2030 drivers/net/tun.c:1770 tun_chr_write_iter+0x71/0x120 drivers/net/tun.c:1999 new_sync_write fs/read_write.c:593 [inline] vfs_write+0x45d/0x710 fs/read_write.c:686 ksys_write+0xa7/0x170 fs/read_write.c:738 do_syscall_x64 arch/x86/entry/syscall_64.c:63 [inline] do_syscall_64+0xa4/0xf80 arch/x86/entry/syscall_64.c:94 entry_SYSCALL_64_after_hwframe+0x77/0x7f