我已将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_init
,end_init
和begin_fini
在lib/gcc/crtstuff.c
中定义(已移植到内核中)。这些功能在extern
和crtstuff.c
中都被声明为linux/module.h
。该module.h
已包含在上面的helloworld
模块中,但这些符号仍然不确定。那么,如何定义这些功能?
答案 0 :(得分:0)
您的内核C ++实现不完整。您将必须实现全局构造函数和析构函数支持(正确处理.init_array
和.fini_array
部分),或停止在源代码中使用这些C ++功能。这需要内核模块加载器的配合。由于启动代码未链接到内核模块中,因此无法更改启动代码。