5.4.18 加载某三方模块使用内核 panic 问题分析

2024-06-06 09:44

本文主要是介绍5.4.18 加载某三方模块使用内核 panic 问题分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

环境信息

内核版本:5.4.18

cpu 架构:arm64

问题描述

加载了产品的某三方 ko 文件使用过程中,会触发如下 panic 信息:

[  218.133479][ 0]  Unable to handle kernel NULL pointer dereference at virtual address 00000000000001f8
........................................................................................................
[  218.164737][ 0]   dsa_slave_dev_check+0x24/0x40
[  218.165445][ 0]   __switchdev_handle_port_obj_add+0x4c/0xfc
[  218.166286][ 0]   switchdev_handle_port_obj_add+0x40/0x60
[  218.167104][ 0]   dsa_slave_switchdev_blocking_event+0xd0/0xdc
[  218.168101][ 0]   notifier_call_chain+0x8c/0xf0
[  218.168803][ 0]   blocking_notifier_call_chain+0x68/0x90
[  218.169606][ 0]   switchdev_port_obj_notify+0x64/0xcc
[  218.170367][ 0]   switchdev_port_obj_add_now+0x60/0x110
[  218.171151][ 0]   switchdev_port_obj_add+0x4c/0x1e0
[  218.171897][ 0]   br_switchdev_port_vlan_add+0x74/0xa0
[  218.172673][ 0]   __vlan_add+0x68/0x690
[  218.173286][ 0]   br_vlan_add+0xf4/0x270
[  218.173910][ 0]   br_vlan_bridge_event+0x12c/0x164
[  218.174649][ 0]   br_device_event+0x1f0/0x3ec
[  218.175340][ 0]   notifier_call_chain+0x8c/0xf0
[  218.176038][ 0]   raw_notifier_call_chain+0x40/0x50
[  218.176786][ 0]   call_netdevice_notifiers_info+0x40/0x84
[  218.177603][ 0]   register_netdevice+0x3d4/0x490
.........................................................................................................
[  218.183456][ 0]  Code: aa1e03e0 d503201f 90002421 913c6021 (f940fe60)
[  218.184412][ 0]  ---[ end trace b95290d7d16d2c50 ]---
[  218.191634][ 1]  Kernel panic - not syncing: Fatal exception

从 panic 信息看,问题出在 register_netdevice 执行过程中,调用内核 notifier chain 过程中触发了内存访问异常,出现问题的点在 dsa_slave_dev_check 函数的 0x24 偏移处。

dsa_slave_dev_check 函数反汇编代码:

0000000000000540 <dsa_slave_dev_check>:540:       a9be7bfd        stp     x29, x30, [sp,#-32]!544:       910003fd        mov     x29, sp548:       f9000bf3        str     x19, [sp,#16]54c:       aa0003f3        mov     x19, x0550:       d50320ff        hint    #0x7554:       aa1e03e0        mov     x0, x30558:       94000000        bl      0 <_mcount>55c:       90000001        adrp    x1, 0 <dsa_slave_phy_read>560:       91000021        add     x1, x1, #0x0564:       f940fe60        ldr     x0, [x19,#504]568:       f9400bf3        ldr     x19, [sp,#16]56c:       eb01001f        cmp     x0, x1570:       1a9f17e0        cset    w0, eq574:       a8c27bfd        ldp     x29, x30, [sp],#32578:       d65f03c0        ret57c:       d503201f        nop

出现问题的地方:

     564:       f940fe60        ldr     x0, [x19,#504]

504 的十六进制数就是 1f8,dsa_slave_dev_check 函数代码如下:

static bool dsa_slave_dev_check(const struct net_device *dev)
{return dev->netdev_ops == &dsa_slave_netdev_ops;
}

显然,问题出在获取 netdev_ops 过程中,表明传入的 net_device 结构为 NULL,需要进一步分析问题触发的原因。

源码分析

追溯源码确定问题的关键在 br_vlan_bridge_event 函数中,相关代码如下:

/* Must be protected by RTNL. */
int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr)
{struct netdev_notifier_changeupper_info *info;struct net_bridge *br = netdev_priv(dev);bool changed;int ret = 0;switch (event) {case NETDEV_REGISTER:ret = br_vlan_add(br, br->default_pvid,BRIDGE_VLAN_INFO_PVID |BRIDGE_VLAN_INFO_UNTAGGED |BRIDGE_VLAN_INFO_BRENTRY, &changed, NULL);

这里直接将 net device 的 priv 数据转化为 net_bridge 结构使用,继续向上追溯,确认关键逻辑如下:

        if (dev->priv_flags & IFF_EBRIDGE) {err = br_vlan_bridge_event(dev, event, ptr);

当 netdev 的 priv_flags 设置了 IFF_EBRIDGE 标志后,会将 net_device 的 priv 结构作为 net_bridge 使用,而此结构并未初始化,进而触发了问题。

产品反馈

产品同步 5.10 内核此内核模块都能正常使用,没有 panic 问题。怀疑是 dsa-core 模块的问题。分析两个版本内核中这两个项目的编译信息,得到如下内容:

5.10:编译为模块

5.4.18:编译进内核

于是怀疑是 5.10 并未加载 dsa-core 模块触发的问题,为了验证修改 5.4.18 将 dsa-core 编译为模块,测试验证堆栈变更为如下内容:


[  451.784754][ 1] Unable to handle kernel NULL pointer dereference at virtual address 0000000000000338
....................................................................................................
[  451.817166][ 1]   __vlan_add+0x2fc/0x690
[  451.817784][ 1]   br_vlan_add+0xf4/0x270
[  451.818404][ 1]   br_vlan_bridge_event+0x12c/0x164
[  451.819135][ 1]   br_device_event+0x1f0/0x3ec
[  451.819810][ 1]   notifier_call_chain+0x8c/0xf0
[  451.820510][ 1]   raw_notifier_call_chain+0x40/0x50
[  451.821261][ 1]   call_netdevice_notifiers_info+0x40/0x84
[  451.822068][ 1]   register_netdevice+0x3d4/0x490
..............................................................................................
[  451.827680][ 1] [  T963] Code: a94573fb 3707ece0 79402263 aa1703e1 (f9419f42) 
[  451.828637][ 1] [  T963] ---[ end trace 9c20b105b7211e5f ]---
[  451.836189][ 1] [  T963] Kernel panic - not syncing: Fatal exception

反汇编相关内容如下:

/* check if we should use the vlan entry, returns false if it's only context */
static inline bool br_vlan_should_use(const struct net_bridge_vlan *v)
{if (br_vlan_is_master(v)) {1a80:       3707ece0        tbnz    w0, #0, 181c <__vlan_add+0x8c>goto out;}/* Add the dev mac and count the vlan only if it's usable */if (br_vlan_should_use(v)) {err = br_fdb_insert(br, p, dev->dev_addr, v->vid);1a84:       79402263        ldrh    w3, [x19,#16]1a88:       aa1703e1        mov     x1, x231a8c:       f9419f42        ldr     x2, [x26,#824]

异常指令位于 1a8c 处,此处是在访问 dev->dev_addr,表明问题还是 dev 结构为 NULL。

5.10 与 5.4.18 相关代码对比

未见明显差异,代码几乎完全相同。

5.10 上加载 dsa-core 模块测试

5.10 上加载了 dsa-core 模块后使用产品 ko 也不会触发问题,明明是几乎完全一致的代码,不应该有这样的区别,分析一度陷入僵局。

进一步分析发现 5.10 上未加载 bridge 内核模块,尝试加载后复现问题,仍旧能够正常工作,实在有点不能解释,难道是函数未调用到?

于是进一步使用 ftrace 跟踪相关函数,很久不用 ftrace 使用起来有些不太顺畅,最后也没有跟踪到相关函数调用,一度怀疑是用法不对。

为什么 5.10 不出问题?

一通抠脑壳后发现 5.10 的 /proc/kallsyms 文件中并没有 vlan_add 相关符号,于是检索内核源码,发现这个函数在开启了 BRIDGE_VLAN_FILTERING 之后才会编译进 bridge 内核模块中。bridge 模块 Makefile 文件内容如下:

bridge-y        := br.o br_device.o br_fdb.o br_forward.o br_if.o br_input.o \br_ioctl.o br_stp.o br_stp_bpdu.o \br_stp_if.o br_stp_timer.o br_netlink.o \br_netlink_tunnel.o br_arp_nd_proxy.obridge-$(CONFIG_SYSFS) += br_sysfs_if.o br_sysfs_br.obridge-$(subst m,y,$(CONFIG_BRIDGE_NETFILTER)) += br_nf_core.obr_netfilter-y := br_netfilter_hooks.o
br_netfilter-$(subst m,y,$(CONFIG_IPV6)) += br_netfilter_ipv6.o
obj-$(CONFIG_BRIDGE_NETFILTER) += br_netfilter.obridge-$(CONFIG_BRIDGE_IGMP_SNOOPING) += br_multicast.o br_mdb.obridge-$(CONFIG_BRIDGE_VLAN_FILTERING) += br_vlan.o br_vlan_tunnel.o br_vlan_options.o

5.10 内核中的相关配置如下:

# CONFIG_BRIDGE_VLAN_FILTERING is not set
CONFIG_NET_SWITCHDEV=y
CONFIG_BRIDGE=m
CONFIG_NET_DSA=m

可以看到并没有开启此功能,于是单独编译 bridge,开启 VLAN_FILTERING,加载 bridge 模块后加载产品 ko 测试也会触发 panic,部分信息摘录如下:

[  234.935733] BUG: kernel NULL pointer dereference, address: 0000000000000320
...............................................................................
[  234.945997] RIP: 0010:__vlan_add+0x309/0xa50 [bridge]
...............................................................................
[  234.967415]  br_vlan_add+0x1dc/0x3a0 [bridge]
[  234.968293]  ? __ipv6_dev_mc_inc+0x1d9/0x350
[  234.969144]  br_vlan_bridge_event+0x175/0x1a0 [bridge]
[  234.970139]  ? failover_get_bymac+0x8e/0xa0
[  234.970979]  br_device_event+0x1bf/0x300 [bridge]
[  234.971902]  raw_notifier_call_chain+0x46/0x60
[  234.972762]  call_netdevice_notifiers_info+0x50/0x90
[  234.973724]  register_netdevice+0x479/0x5a0
...............................................................................

堆栈与 5.4.18 一致,表明问题出在不同的内核 bridge 相关内核配置不同上,实际是产品使用方法不正确。

解决方法

  1. 将 bridge 编译为模块、关闭 BRIDGE_VLAN_FILTER 功能
  2. 产品内核模块修改,去掉设置的 IFF_EBRIDGE 标志(建议)

总结

最开始一头扎进代码中,通过分析改代码得出的结论是产品异常使用问题。同步这个结论,产品反馈好几个内核版本都没有问题,于是并未坚持这一结论,继续进行分析。

分析对比了正常运行的 5.10.64 内核的代码,发现与 5.4.18 几乎一模一样,理论上应当出现相同的问题,实际测试确定正常工作,一度有些不知道怎么开展。

不过既然有正常的基线,就先基于基线进行分析,尝试使用 ftrace 跟踪内核符号调用,发现并没有跟踪到相应的调用信息,却并没有怀疑两个内核在相关模块编译上的差异。经过一通折腾,最后看到 bridge 模块编译配置,确认了问题为 5.10.64 上并未加载 bridge 功能。尝试加载后重试也能正常工作,在 /proc/kallsyms 中搜索 5.4.18 崩溃的关键函数名,没有找到,这才发现有一个关键的配置项目——BRIDGE_VLAN_FILTER 并未开启,才找到了问题点。

为了进一步验证,手动编译 5.10.64 内核的 bridge 模块,设置开启 BRIDGE_VLAN_FILTER 功能,再次配置产品业务,此时触发了与 5.4.18 相同的问题。

最后产品去掉 net_device 初始化过程中对 priv_flags IFF_EBRIDGE 标志的设定,验证功能正常,此问题得到解决,这时我已经抠了两天脑壳。。。

这篇关于5.4.18 加载某三方模块使用内核 panic 问题分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1035752

相关文章

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

Spring的RedisTemplate的json反序列泛型丢失问题解决

《Spring的RedisTemplate的json反序列泛型丢失问题解决》本文主要介绍了SpringRedisTemplate中使用JSON序列化时泛型信息丢失的问题及其提出三种解决方案,可以根据性... 目录背景解决方案方案一方案二方案三总结背景在使用RedisTemplate操作redis时我们针对

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

MySQL中比较运算符的具体使用

《MySQL中比较运算符的具体使用》本文介绍了SQL中常用的符号类型和非符号类型运算符,符号类型运算符包括等于(=)、安全等于(=)、不等于(/!=)、大小比较(,=,,=)等,感兴趣的可以了解一下... 目录符号类型运算符1. 等于运算符=2. 安全等于运算符<=>3. 不等于运算符<>或!=4. 小于运

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处

Kotlin Map映射转换问题小结

《KotlinMap映射转换问题小结》文章介绍了Kotlin集合转换的多种方法,包括map(一对一转换)、mapIndexed(带索引)、mapNotNull(过滤null)、mapKeys/map... 目录Kotlin 集合转换:map、mapIndexed、mapNotNull、mapKeys、map

nginx中端口无权限的问题解决

《nginx中端口无权限的问题解决》当Nginx日志报错bind()to80failed(13:Permissiondenied)时,这通常是由于权限不足导致Nginx无法绑定到80端口,下面就来... 目录一、问题原因分析二、解决方案1. 以 root 权限运行 Nginx(不推荐)2. 为 Nginx

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种