写了个platform架构的UART测试驱动,在卸载的时候出了问题,驱动卸载不掉,感觉很奇怪,
用户空间使用的rmmod最终会调用kernel/module.c 中的 SYSCALL_DEFINE2(delete_module, const char __user*, name_user, unsigned int, flags)
1、按找lsmod显示的模块名查找模块
mod = find_module(name);
if (!mod) { ret = -ENOENT; printk("lccdandeion we cant find mod\n"); goto out;}2、检查是否有模块依赖于本模块
if (!list_empty(&mod->modules_which_use_me)) {
/* Other modules depend on us: get rid of them first. */ printk("lccdandeion there are modules using me\n"); ret = -EWOULDBLOCK; goto out; }3、检查模块是否存活/* If it has an init func, it must have an exit func to unload */
if (mod->init && !mod->exit) { forced = try_force_unload(flags); if (!forced) { /* This module can't be removed */ ret = -EBUSY; goto out; } }4、若用户空间传递的参数要求强制卸载则卸载(内核编译时要选中可强制卸载模块),检查模块的引用计数是否为零(不为零,则一直等待到引用计数为零)/* Never wait if forced. */
if (!forced && module_refcount(mod) != 0) { printk("lccdandlion wait for zero refcount\n"); wait_for_zero_refcount(mod); printk("lccdandelion wait end\n"); }5、/* Final destruction now noone is using it. */
if (mod->exit != NULL) { printk("lccdandeion did we blocked\n"); mod->exit(); }写的UART驱动测试在调用mod->exit()崩溃。面对panic,我们该如何定位问题
1、Google
2、利用panic信息、反汇编内核
对于2:
源码树顶层下的Makefile 中KBUILD_CFLAGS := -Wall -Wundef -Wstrict-prototype ....
添加 KBUILD_CFLAGS := -g -Wall -Wunde -Wstrict-prototype
这样生成的内核将含有调试信息,611内核源码不支持gdb,但是我们可以反汇编,
unicore32-linux-objdump -Dz -S vmlinux >> linux.dump
根据panic信息可以定位到最终报错时的几个函数调用路径,在函数中打桩,一步步的定位问题,虽然611有些封闭,但是内核核心出现的问题google一般都可以查到。
虽然问题解决了,但是panic的信息还是怪怪的,panic时,pc值竟然不再内核空间中????!!!!