当前位置: 首页 > news >正文

字符设备驱动(内核态用户态内存交互)

前言

内核驱动:运行在内核态的动态模块,遵循内核模块框架接口,更倾向于插件。
应用程序:运行在用户态的进程。
应用程序与内核驱动交互通过既定接口,内核态和用户态访问依然遵循内核既定接口。

环境搭建

系统:openEuler-20.03-LTS-SP3

yum install gcc kernel-devel

编写源码

  • char_module.c
#include <linux/module.h>
#include <linux/init.h>
#include <linux/fs.h>
#include <asm/uaccess.h>
#include <asm/device.h> //下面这三个头文件是由于动态创建需要加的
#include <linux/device.h>
#include <linux/cdev.h>MODULE_LICENSE("GPL");#define DEVICE_NAME "char_module"
#define BUF_SIZE 32static struct class *cdev_class;
dev_t dev_num = 0; // 这里是动态分配设备号和动态创建设备结点需要用到的
struct cdev dev_c;static char context_buf[BUF_SIZE]={"this a test context buffer\0"};static ssize_t read(struct file *, char *, size_t, loff_t *);
static ssize_t write(struct file *, const char *, size_t, loff_t *);
static int open(struct inode *, struct file *);
static int release(struct inode *, struct file *);// 初始化字符设备驱动的 file_operations 结构体
struct file_operations fops = {.read = read,.write = write,.open = open,.release = release
};static int __init demo_init(void)
{int ret, err;printk(KERN_INFO "%s: %s", DEVICE_NAME , __func__);// 注册设备驱动ret = alloc_chrdev_region(&dev_num, 0, 1, DEVICE_NAME); // 动态分配设备号if (ret){printk("demo_init register failure\n");unregister_chrdev_region(dev_num, 1);return ret;}printk("demo_init register success\n");// 初始化设备操作cdev_init(&dev_c, &fops);err = cdev_add(&dev_c, dev_num, 1);if (err){printk(KERN_NOTICE "error %d adding cdev\n", err);unregister_chrdev_region(dev_num, 1);return err;}// 动态创建设备结点cdev_class = class_create(THIS_MODULE, DEVICE_NAME); if (IS_ERR(cdev_class)){printk("ERR:cannot create a cdev_class\n");unregister_chrdev_region(dev_num, 1);return -1;}device_create(cdev_class, NULL, dev_num, 0, DEVICE_NAME);return ret;
}static void __exit demo_exit(void)
{printk(KERN_INFO "%s: %s", DEVICE_NAME , __func__);// 注销设备驱动device_destroy(cdev_class, dev_num);class_destroy(cdev_class);unregister_chrdev_region(dev_num, 1);
}static ssize_t read(struct file *filp, char *buf, size_t len, loff_t *off)
{// 内核空间到用户空间copyprintk(KERN_INFO "%s: %s", DEVICE_NAME , __func__);if (raw_copy_to_user(buf, &context_buf, sizeof(context_buf))){return -EFAULT;}printk(KERN_INFO "user space: %pF", buf);printk(KERN_INFO "read: %pF; size: %ld; data: %s", &context_buf, sizeof(context_buf), context_buf);return BUF_SIZE;
}static ssize_t write (struct file *filp, const char __user *buf, size_t len, loff_t *off)
{// 用户空间到内核空间copyprintk(KERN_INFO "%s: %s", DEVICE_NAME , __func__);if (raw_copy_from_user(&context_buf, buf, sizeof(context_buf))){return -EFAULT;}printk(KERN_INFO "user space: %pF", buf);printk(KERN_INFO "write: %pF; size: %ld; data: %s", &context_buf, sizeof(context_buf), context_buf);return BUF_SIZE;
}static int open(struct inode *inodp, struct file *filp)
{printk(KERN_INFO "%s: %s", DEVICE_NAME , __func__);return 0;
}static int release(struct inode *inodp, struct file *filp)
{printk(KERN_INFO "%s: %s", DEVICE_NAME, __func__);return 0;
}module_init(demo_init);
module_exit(demo_exit);
  • Makefile
ifneq ($(KERNELRELEASE),)
obj-m := char_module.oelse
PWD  := $(shell pwd)
KVER := $(shell uname -r)
KDIR := /lib/modules/$(KVER)/build
all:$(MAKE) -C $(KDIR) M=$(PWD) modules modules_install
clean:rm -rf .*.cmd *.o *.mod.c *.ko .tmp_versions modules.*  Module.*
endif

app.c

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>#define CHAR_DEV_NAME "/dev/char_module"int main()
{int ret;int fd;char buf[32];fd = open(CHAR_DEV_NAME, O_RDWR | O_NDELAY);if(fd < 0){printf("open failed!\n");return -1;}int size = read(fd, buf, 32);printf("read size: %d;\nbuffer:[%s]\n", size, buf);char *write_buf = "use a application wirte to driver buffer";int w_size = write(fd, write_buf, strlen(write_buf));printf("write size: %d;\nbuffer:[%s]\n", w_size, write_buf);close(fd);return 0;
}

构建并测试

  • 驱动构建
    make && insmod char_module.ko
    
  • 驱动信息确认
    在这里插入图片描述
  • 应用程序构建
    gcc app.c -o app
    ./app
    
  • 应用程序运行结果
    在这里插入图片描述
  • 查看驱动日志
    dmesg
    
    在这里插入图片描述
http://www.lryc.cn/news/144358.html

相关文章:

  • Qt基础 线程池
  • Django(8)-静态资源引用CSS和图片
  • C++ list模拟实现
  • 中国建筑出版传媒许少辉博士八一新书乡村振兴战略下传统村落文化旅游设计日京东当当畅销榜自由营九三学
  • C语言(第三十二天)
  • arcgis+postgresql+postgis使用介绍
  • 机器视觉之开运算和闭运算
  • 【python爬虫】—URL管理器的实现
  • Oracle 19C RAC安装PSU oui-patch.xml权限错误
  • 华为数通方向HCIP-DataCom H12-821题库(单选题:161-180)
  • ResNet详解:网络结构解读与PyTorch实现教程
  • ChatGPT 随机动态可视化图表分析
  • 国标视频融合云平台EasyCVR视频汇聚平台的应用场景及其功能说明
  • 后端面试话术集锦第三篇:spring cloud 面试话术
  • React 18 选择 State 结构
  • LNMT与动静分离
  • 【java】LinkedList 和 ArrayList的简介与对比
  • 机器学习基础14-算法调参(基于印第安糖尿病Pima数据集)
  • ASUS华硕天选4笔记本电脑FA507XV原厂Windows11系统22H2
  • IET独立出版 | EI检索 | 2023年第三届机械、航空航天与汽车工程国际会议
  • 【Pytorch】CUDA error: no kernel image is available for execution on the device
  • dolphinschedule配置企微告警服务(WeChat群组)
  • Git中smart Checkout与force checkout
  • Java“牵手”1688商品跨境属性数据,1688API接口申请指南
  • Win解答 | 解决键盘中 字母+空格 导致的输入法弹窗导致的一系列问题
  • WPF读取dicom序列:实现上一帧、下一帧、自动播放、暂停
  • homeassistant ubuntu自启动 网络设置
  • 生成式AI背景下编程工作者的技术和高级软考理论的演进融合之路
  • RabbitMQ的镜像队列
  • 【Spring Boot】数据库持久层框架MyBatis — Spring Boot构建MyBatis应用程序