内核驱动开发hello world

内核模块初识

 1. 模块的加载和卸载

 hello.c

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

//实现入口、出口函数
static int __init _hello_init(void) //驱动模块被安装时触发的函数
{
  printk(KERN_INFO"hello world\r\n");  // 不使用KERN_ERR 会将信息打印至后台。
  //dmesg -c  查看后台内容(-c 清除历史buff)
  return 0;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
}

static void __exit _hello_exit(void) //驱动模块被卸载时触发的函数
{
  printk(KERN_INFO"BYE BYE\r\n"); // 不使用KERN_ERR 会将信息打印至后台
}

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

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

Makeflie:

# hello 是模块名,也是对应的 c 文件名
obj-m +=hello.o

 # KDIR 内核源码路径,根据自己需要设置
KDIR:=$(shell uname -r)


build: kernel_modules

kernel_modules:
        make -C /lib/modules/$(KDIR)/build M=$(CURDIR) modules

clean:
        make -C /lib/modules/$(KDIR)/build M=$(CURDIR) clean

#上面是x86的简单方式,如果是arm的,需要采用下方方式:
#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

 3. 模块符号导出(可以没有初始化函数,仅作为api提供给内核使用)

#include <linux/module.h>

int my_add(int a, int b) //驱动模块被安装时触发的函数
{
  return a + b;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
}

int my_sub(int a, int b) //驱动模块被安装时触发的函数
{
  return a - b;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
}


EXPORT_SYMBOL(my_add);
EXPORT_SYMBOL(my_sub);

MODULE_LICENSE("GPL"); //本驱动程序遵循GPL开源协议,必写
MODULE_AUTHOR("SX <sx417823439@163.com>");//作者信息
MODULE_DESCRIPTION("This is export symbol");//驱动功能的描述
MODULE_VERSION("v1.0");//驱动的版本


 4. 模块参数传递

#include <linux/module.h>

static int my_param = 0;  // 默认参数值为 0
static char *my_name = "sunxin";//默认参数为sunxin

module_param(my_param, int, S_IRUGO);  // 定义模块参数
module_param(my_name, charp, S_IRUGO);  // 定义模块参数

//实现入口、出口函数
static int __init _hello_init(void) //驱动模块被安装时触发的函数
{
  printk(KERN_INFO"my_param = %d\n", my_param);  // 不使用KERN_ERR 会将信息打印至后台。
  printk(KERN_INFO"my_name = %s\n", my_name);  // 不使用KERN_ERR 会将信息打印至后台。
  //dmesg -c  查看后台内容(-c 清除历史buff)
  return 0;  //此函数返回0时驱动才能正常安装,返回其他值则安装失败
}

static void __exit _hello_exit(void) //驱动模块被卸载时触发的函数
{
  printk(KERN_INFO"BYE BYE\r\n"); // 不使用KERN_ERR 会将信息打印至后台
}

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

MODULE_LICENSE("GPL"); //本驱动程序遵循GPL开源协议,必写
MODULE_AUTHOR("SX <sx417823439@163.com>");//作者信息
MODULE_DESCRIPTION("This is export symbol");//驱动功能的描述
MODULE_VERSION("v1.0");//驱动的版本

一个ko依赖多个源文件
使用多个源文件时,需要在Makefile中修改

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值