加载内核模块时出现未知符号

Unknown symbol when loading a kernel module

我试图重做在这个主题中找到的内核模块的代码How can I obtain battery level inside a Linux kernel module?。但是当我尝试使用 power_supply.h header 文件中包含的函数时,模块加载失败,因为它无法识别 power_supply_get_by_name 函数。

这是我在 Ubuntu 18.04 上使用的代码,内核版本为 4.15.0-101-generic:

#include <linux/module.h>
#include <linux/power_supply.h>

static int __init test_init (void)
{
        struct power_supply *psy;
        char name[] = "BAT1";

        psy = power_supply_get_by_name(name);

        printk(KERN_DEBUG "Test module inserted");

        return 0;
}

static void __exit test_exit (void)
{
        printk(KERN_DEBUG "Test module removed");
}

module_init (test_init);
module_exit (test_exit);

除了有关模块许可证的警告外,我在编译时没有遇到任何错误,我认为这与我的问题无关,但我收到以下错误:

  1. 当 运行 在终端中 insmod 时:"insmod: ERROR: could not insert module test.ko: Unknown symbol in module"
  2. 在 /var/log/kern.log 文件中:"test: Unknown symbol power_supply_get_by_name (err 0)"

我检查了 kallsyms proc 文件,如果我很好地理解了这个主题,该函数被指示在其他内核模块中可用 What is the difference between T and t in /proc/kallsyms。这是读取 kallsyms 文件的输出:

ffffffff8e9bd270 T power_supply_get_by_name

有谁知道为什么这不起作用,而我可以毫无问题地使用其他 linux headers 函数,如果是这样,我该如何解决我的问题?

提前致谢

这实际上可能与模块许可证有关!如果你看内核源码,函数power_supply_get_by_name被导出here. You can see it's using EXPORT_SYMBOL_GPL. As this answer解释:

EXPORT_SYMBOL_GPL will show the symbol only in GPL-licensed modules

这个宏的使用是有争议的,但这就是项目的运作方式...要访问您需要的符号,您需要将您的模块许可为 GPL:

MODULE_LICENSE("GPL");