linux驱动----模块符号导出使用

本文介绍一种,一个模块调动另一个模块中定义的函数的方法

在内核中,函数又叫做符号,为了便于理解,此处说成我们在C语言中熟悉的函数调用,回想一下C语言中的函数调用,如果我们需要在一个.C文件中调用另一个.C文件中的函数,很简单,我们只需要在头文件中声明一下即可。但是,这件事,在内核中模块间却要麻烦一些,下面听我细细道来。

在这之前,我先假定一个背景,以此来更形象得阐述这件事,现下有两个不同的模块,以及一个函数,模块2(hehe)要调用模块1(hello)中定义的mystrlen函数。

(1)在编写模块1的.c文件时,将所定义函数,用EXPORT_SYMBOL导出

EXPORT_SYMBOL(mystrlen);

(2)在编写模块2的.c文件时,在顶部声明一下是调用的外部函数

extern int mystrlen(char *);

(3)将模块1make一下, 模块1的makefile文件放在文章末尾,此处我们继续。make完成之后,加载一下模块1,然后使用下面这条命令查看一下

cat /proc/kallsyms | grep mystrlen

会发现

db569f82c1104b91becec549d18ffc53.png

这说明在我们内核里的符号表里已经有这个函数了(/proc/kallsyms这个文件夹包含的是所有内核知道的符号)

(4)拷贝make模块1时所生成的Module.symvers文件到模块2所在的位置。

插一句,Module.symvers这个是什么东西?我们可以vim打开看一下,这个文件包含了内存位置、符号名称、模块所在位置。有了这三条信息,模块2就知道了去哪调用了,不然模块2怎么知道要调用谁,要去哪调用。

7bc21fd1b9f74bdeb5253ff4b9811e79.png

(5)把模块2也make一下

(6)在两个模块所在的目录下,分别执行sudo make modules_install,其中modules_install的具体实现在Makefile中有写,见文末。此举主要是将模块安装到根目录下的对应文件夹下,安装成功后我们可以查看一下有没有

07e89e388fdf4547a1c735a33dac29ae.png

 (7)sudo depmod -a 建立起模块之间的依赖关系,因为,模块2需要调动模块1中的函数,所以需要这样来建立起依赖关系。

(8)sudo modprobe hehe 直接加载这个模块2,因为上一条命令已经将模块间的依赖建立,此处直接用modprobe就可以直接加载,不用再去用insmod一个个地去加载,那样不是很方便。

文末给出两个.c文件和Makefile文件

模块1

#include <linux/module.h>
#include <linux/init.h>
MODULE_LICENSE("GPL");
int mystrlen(char *p)
{
        int i=0;
        while(*p)
        {
                i++;
                p++;
        }
        return i;
}
EXPORT_SYMBOL(mystrlen);


//int init_module()
static int __init hello_init(void)
{
        printk("hello, init .....\n");

        printk("lcd = %d\n",lcd);
        return 0;
}
static void __exit hello_exit(void)
{
        printk("hello,exit...\n");
}
module_init(hello_init); //指出当前模块的入口
module_exit(hello_exit);

 模块2

#include <linux/module.h>
#include <linux/init.h>
extern int mystrlen(char *);
MODULE_LICENSE("GPL");
char *p="abcdefg";
//int init_module()
static int __init hello_init(void)
{
        printk("hello, init .....\n");

        printk("length of p is %d\n",mystrlen(p));
        return 0;
}

static void __exit hello_exit(void)
{
        printk("hello,exit...\n");
}
module_init(hello_init); //指出当前模块的入口
module_exit(hello_exit);

Makefile 

ifeq ($(KERNELRELEASE),)
KERNELDIR ?= /lib/modules/$(shell uname -r)/build
#KERNELDIR ?=/home/farsight/linux-add-usb/linux-3.14   
PWD := $(shell pwd)
modules:
        $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
modules_install:
        $(MAKE) -C $(KERNELDIR) M=$(PWD) INSTALL_MOD_PATH= modules_install
else
  obj-m := hello.o
endif

clean:
        rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions Module* modules*


 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

翔在天上飞

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值