三、Linux内核模块编写3(多个c文件生成多个ko文件)


  
  模块是一个目标文件,可以完成某种独立的功能,但是自身不是一个独立的进程,不能单独运行,可以动态的载入模块,使其成为内核代码的一部分,与内核其他代码的地位完全相同,当不需要某个模块功能时,还可以卸载模块。

Ⅰ、 多个c文件生成多个ko文件。

( 注: ko文件为kernel object文件,也称内核模块 )
程序包括:hello.c 、world.c 、Makefile。

a. 编写Makefile文件程序。
注:该Makefile实用性更强,只需要修改第2行即可!!!凡要生成.ko文件的都可以用此程序!

#多个c文件生成一个ko文件。
obj-m +=hello.o world.o          #把.c文件的名字加.o写到这里

# KDIR 内核源码路径,根据自己需要设置
KDIR:=/home/qjl/work/lichee/linux-3.10

all:
#ARCH: 指当前编译的驱动模块的架构
#CROSS_COMPILE:指明交叉编译器的前缀
#-C: 指定去$(KDIR)目录下执行Makefile
#M:告知Makefile,需要的编译文件在哪
#modules: 这个规则是用于编译驱动模块的
	@make ARCH=arm64 CROSS_COMPILE=aarch64-linux- -C $(KDIR) M=$(PWD) modules 
	@rm -fr .tmp_versions *.o *.mod.o *.mod.c *.bak *.symvers *.markers *.unsigned *.order *~ .*.*.cmd .*.*.*.cmd
	
clean:
	@make ARCH=arm64 CROSS_COMPILE=aarch64-linux- -C $(KDIR) M=$(PWD) modules clean
	@rm -rf *.ko

b. 编写简单内核模块程序 ( hello.c )。

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

int add(int a,int b)
{
  return a+b;
}
EXPORT_SYMBOL(add);        //导出符号,其他模块就可以使用这个符号了。

//实现入口、出口函数
/*
 __int只有驱动被编译进内核时才有效。
	作用:执行完本函数后丢弃该函数,并释放其占用的空间。(节约空间)
	原因:它只有模块被安装后才能触发。
*/

static int __init a53_hello_init(void) //驱动模块被安装时触发的函数
{
  // 如果想把消息打印至前台,需要给printk输入等级如0 1 2 3。 如KERN_ERR为3.
	// 不赋予printk等级则 会将信息打印至后台。	
	//可使用 :dmesg -c  查看后台内.
  printk(KERN_ERR"hello world\r\n");  
  return 0;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
  
}
/*
  __exit只有驱动被编译进内核时才有效。
  作用:不将本函数进行链接,因为模块不可能被卸载,所以此函数无意义
*/
static void __exit a53_hello_exit(void) //驱动模块被卸载时触发的函数
{
  printk(KERN_ERR"BYE BYE\r\n"); // 不使用KERN_ERR 会将信息打印至后台
}

//声明驱动模块的入口、出口
module_init(a53_hello_init);
module_exit(a53_hello_exit);

MODULE_LICENSE("GPL"); //本驱动程序遵循GPL开源协议,必写
MODULE_AUTHOR("QJL <1033275663@qq.com>");//作者信息
MODULE_DESCRIPTION("This is hello world driver");//驱动功能的描述
MODULE_VERSION("v1.0");//驱动的版本

c.编写world.c模块文件。

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

extern int add(int , int); //声明外部函数

//实现入口、出口函数
/*
 __int只有驱动被编译进内核时才有效。
	作用:执行完本函数后丢弃该函数,并释放其占用的空间。(节约空间)
	原因:它只有模块被安装后才能触发。
*/

static int __init a53_world_init(void) //驱动模块被安装时触发的函数
{
  // 如果想把消息打印至前台,需要给printk输入等级如0 1 2 3。 如KERN_ERR为3.
	// 不赋予printk等级则 会将信息打印至后台。	
	//可使用 :dmesg -c  查看后台内.
  printk(KERN_ERR"a+b=%d\r\n",add(30,20));  
  return 0;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
  
}
/*
  __exit只有驱动被编译进内核时才有效。
  作用:不将本函数进行链接,因为模块不可能被卸载,所以此函数无意义
*/
static void __exit a53_world_exit(void) //驱动模块被卸载时触发的函数
{
  printk(KERN_ERR"BYE BYE\r\n"); // 不使用KERN_ERR 会将信息打印至后台
}

//声明驱动模块的入口、出口
module_init(a53_world_init);
module_exit(a53_world_exit);
MODULE_LICENSE("GPL"); //本驱动程序遵循GPL开源协议,必写
MODULE_AUTHOR("QJL <1033275663@qq.com>");//作者信息
MODULE_DESCRIPTION("This is hello world driver");//驱动功能的描述
MODULE_VERSION("v1.0");//驱动的版本

d. 使用make生成hello.ko、world.ko模块驱动文件。

e.将驱动文件传给开发板,进行安装。(传输方式见:添加链接描述

Ⅱ、模块的安装与卸载

a.原程序目录文件:
在这里插入图片描述

b.使用make编译后目录文件:
在这里插入图片描述

c.将所有ko文件传给开发板。

d.开发板安装模块。
   注意:安装模块时只能先安装hello.ko模块,再安装insmod world.ko模块。world.ko模块依赖hello.ko模块文件,因为world.c中使用了hello.c中的函数!!否则会报错!!

insmod hello.ko
insmod world.ko

e.开发板卸载模块
   注意:卸载模块时只能先卸载world.ko模块,再卸载hello.ko模块。world.ko模块依赖hello.ko模块文件,因为world.c中使用了hello.c中的函数!!

rmmod world.ko
rmmod hello.ko

附加注意:
   当一个A模块中使用另一个B模块所定义的函数时,需要在B模块函数中使用EXPORT_SYMBOL( 函数名 ); 导出符号(函数)。且A模块程序中需要声明该函数为外部函数。这样A模块就可以使用这个符号(函数)了。
  

Ⅲ、附加: 参数传入(使用不多,一般用于调试)

使用参考:

int n;
module_param(n, int, 0700); //0777为权限

char *p;
module_param(p, charp, 0700); //0777为权限

static int __init a53_hello_init(void) //驱动模块被安装时触发的函数
{
  // 如果想把消息打印至前台,需要给printk输入等级如0 1 2 3。 如KERN_ERR为3.
	// 不赋予printk等级则 会将信息打印至后台。	
	//可使用 :dmesg -c  查看后台内.
  printk(KERN_ERR"hello world\r\n");  
  printk(KERN_ERR"n = %d\r\n", n);
  printk(KERN_ERR"p = %s\r\n", p);
  return 0;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
}

执行安装模块命令时 输入参数,且要输入规范。变量名要一致!!
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Linux内核中,通常有许多头文件和源文件需要编译成.ko(内核对象)文件,以实现对特定功能或驱动程序的支持。 首先,头文件包含了函数和数据结构的声明,被其他源文件引用。这些头文件通常以.h文件扩展名结尾。源文件则包含了函数和数据结构的实现,通常以.c或.cpp文件扩展名结尾。 为了将多个文件和源文件编译成.ko文件,使用Makefile是一个常见的方法。Makefile是一个文本文件,包含了一系列的规则,指定了文件依赖关系以及编译和链接的命令。 Makefile中的规则通常包括以下几个步骤: 1. 定义目标文件(.ko文件)的名称和相关的编译选项。 2. 定义依赖关系,即指定哪些源文件和头文件需要进行编译。 3. 定义编译命令,通常使用gcc或其他编译器执行编译操作。例如,可以使用gcc -c命令将源文件编译成目标文件(.o文件)。 4. 定义链接命令,将所有目标文件链接成一个.ko文件。例如,可以使用gcc -o命令将所有目标文件链接成一个.ko文件。 在执行Makefile时,会根据规则的定义逐步执行编译和链接的操作,生成最终的.ko文件。同时,Makefile还可用于指定其他操作,例如清除中间文件或执行其他自定义操作。 总而言之,将多个文件和源文件编译成.ko文件可以通过使用Makefile来定义编译和链接的规则,并将其作为输入参数传递给编译器。Makefile中的规则会根据文件依赖关系逐步执行编译和链接的操作,最终生成所需的.ko文件

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值