Hello World C ++内核模块中的未定义符号

时间:2018-07-01 09:16:14

标签: linux gcc linux-kernel g++ kernel-module

我已将C ++支持添加到Linux内核版本4.14.41,对其进行了编译并成功使用内核启动。我可以通过插入LKM来检查C ++模块的正确性。这是我要加载的模块:

#include<c++/begin_include.h>
#include<linux/module.h>
#include<linux/kernel.h>
#include<c++/end_include.h>


MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("LKM in c++");
MODULE_AUTHOR("MOOL");

class hello
{

public:
        hello();
       void hi();
};
void hello::hi()
{
  printk("Hello world!! \n");
}

hello::hello()
{
        printk("Constructor is being called \n");
        }
 extern "C"
{
   static int __init test_classes_init()
   {
        class hello obj;
        obj.hi();
        printk("Module inserted:\n");
        return 0;
   }
   static void __exit test_classes_fini()
   {
        printk("Module removed:\n");
   }

   module_init(test_classes_init);
   module_exit(test_classes_fini);

}

Makefile:

obj-m = helloworld.o
KVERSION=$(shell uname -r)
all:
        make -C /lib/modules/$(KVERSION)/build M=$(PWD) modules
clean:
        make -C /lib/modules/$(KVERSION)/build M=$(PWD) clean

当我输入make命令时,会生成helloworld.ko并显示警告

WARNING: "begin_fini" [/home/jai/Downloads/helloworld/helloworld.ko] undefined !
WARNING: "end_init" [/home/jai/Downloads/helloworld/helloworld.ko] undefined !
WARNING: "begin_init" [/home/jai/Downloads/helloworld/helloworld.ko] undefined !

但是当我尝试使用insmod helloworld.ko插入它时,会发生undefined symbol错误。

dmesg

loading out-of-tree module taints kernel
Unknown symbol begin_init (err 0)
Unknown symbol end_init (err 0)
Unknown symbol begin_fini (err 0)

这些begin_initend_initbegin_finilib/gcc/crtstuff.c中定义(已移植到内核中)。这些功能在externcrtstuff.c中都被声明为linux/module.h。该module.h已包含在上面的helloworld模块中,但这些符号仍然不确定。那么,如何定义这些功能?

1 个答案:

答案 0 :(得分:0)

您的内核C ++实现不完整。您将必须实现全局构造函数和析构函数支持(正确处理.init_array.fini_array部分),或停止在源代码中使用这些C ++功能。这需要内核模块加载器的配合。由于启动代码未链接到内核模块中,因此无法更改启动代码。