4

你好stackoverflowers :)

在过去的几个小时里,我一直在尝试编译+加载一个多文件模块。编译发出一个奇怪的警告,模块无法加载。这里是模块、Makefile、编译输出和 dmesg。

标题:

// header.h

#ifndef _HEADER_H
#define _HEADER_H
void do_module_func(void);
void do_other_func(void);
#endif

“主”模块文件:

//mymodule.c

#include <linux/module.h>
#include <linux/kernel.h>
#include "header.h"

void do_module_func(void)
{
    printk(KERN_INFO "module_func\n");
}

static int mymodule_init(void)
{
    printk(KERN_INFO "Hello world\n");
    do_other_func();
    return 0;
}
module_init(mymodule_init);


static void mymodule_exit(void)
{
    printk(KERN_INFO "Goodbye, cruel world\n");
}
module_exit(mymodule_exit);

MODULE_LICENSE("GPL")

其他 c 文件,它调用do_module_func()位于“主”模块中

//other_file.c

#include "header.h"
#include <linux/kernel.h>

void do_other_func(void)
{
    printk(KERN_INFO "other_func\n");
    do_module_func();
}

生成文件

//Makefile

obj-m := mymodule.o
mymodule-objs := other_file.o

CROSS:=arm-unknown-linux-gnueabi-
KERNEL:= ~/work/linux-davinci-2.6.38/
ARCH:=arm

PWD:=$(shell pwd)

all:
    $(MAKE) CROSS_COMPILE=$(CROSS) ARCH=$(ARCH) -C $(KERNEL) M=$(PWD) modules

clean:
    $(MAKE) CROSS_COMPILE=$(CROSS) ARCH=$(ARCH) -C $(KERNEL) M=$(PWD) clean

我正在交叉编译,但我相信这应该不是问题。制作输出:

make CROSS_COMPILE....
make[1]: Entering directory .../linux-davinci-2.6.38
  CC [M] .../other_file.o
  LD [M] .../mymodule.o
  Building modules, stage 2.
  MODPOST 1 modules
WARNING: "do_module_func" [.../mymodule.o] undefined! <--- warning here
  CC .../mymodule.mod.o
  LD [M] .../mymodule.ko
make[1]: Leaving directory .../linux-davinci-2.6.38

insmod 输出:

can't insert 'mymodule.ko': unknown symbol in module, or unknown parameter

dmesg:

mymodule: Unknown symbol do_mdule_func (err 0)

因此,模块编译时带有(链接?)警告并且模块不加载。

现在,我看到在 make 输出中似乎在编译 other_file.c 之后进行了链接尝试,但在链接之前不应该也编译 mymodule.c 吗?

谢谢!:)

4

3 回答 3

10

原来问题出在Makefile中。“诀窍”是您在obj-m将要编译的模块中定义(成 .ko)并在-objs您编写所有源文件中。
因此,此 Makefile 中的定义变为:

obj-m := moduleko.o
moduleko-objs := other_file.o mymodule.o

这被编译成moduleko.ko。

于 2013-05-23T12:17:45.047 回答
0

这是因为 file_2 需要 file_1 符号引用才能将 file_2 构建为 LKM。为了克服这个问题,构建 file_1(LKM) 并将 file_1 的 Module.symvers 放在 file_2 位置。并再次构建 file_2。

于 2014-04-10T09:30:51.537 回答
-1
all:
    $(MAKE) CROSS_COMPILE=$(CROSS) ARCH=$(ARCH) -C $(KERNEL) M=$(PWD) modules
instead try like this
     $(MAKE) -C  $(KERNEL) ARCH=$(ARCH)  CROSS_COMPILE=$(CROSS) M=$(PWD) /
          this will run each file and links with object.hope this would solve your problem
于 2014-05-12T14:05:05.437 回答