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

字符设备驱动

能够以字节流的形式进行访问的设备是字符设备,编写的对字符设备进行驱动的驱动顺序叫做字符设备驱动

用户层:open(设备文件)


操作方法:

open 方法                                                                                                           当字符设备驱动注册到内核时,会从内核申请到一个设备号(32位)一个                                                                                                                        设备有两部分组成主设备号和次设备号主设备号占据高12位次设备号:标识                                                                                                                        一类设备中的一个具体的设备,占据设备号的低20位

read 方法                                                                         

write方法


编写基本的设备驱动需要解决的问题

如何注册字符设备驱动

驱动中的操作方法和管理如何实现

如何创建设备文件

应用程序中的如何回调驱动中的操作方法

如何将硬件的物理内存映射到虚拟内存

用户和内核之间如何进行数据传递 

int register_chdev(unsigned int maior,const char  *name,const struct file_operations *fops )
功能:进行字符设备驱动的注册,申请了256个设备号(0-255)
参数 :
major: >0 手动指定当前驱动的主设备号
==0 由系统内核动态分配的主设备号
name:设备名或驱动名
fops:操作方法对象指针
void unregister_cherdev(unsigned int major,const char *name)

功能:进行字符设备的驱动的注销
参数Lmajor 祖册的时候得到的设备号
name :注册填写时候的设备名

字符设备驱动注册的测试案例

填写字符设备的驱动代码

创建一个驱动程序的mycdev.c 在其中的编写如下内容

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
unsigned int major;
int mycdev_open(struct inode *inode ,struct file *file)
{
printk("%s,%s:%d\n",__FILE_,__FUNC_,_LINE_);
return 0;
}
ssize_t mycdev_read(struct file *file,char __user *ubuf,size_t ,loft_t *lft)
{

pprintk("%s,%s:%d\n",__FILE_,__FUNC_,_LINE_);
return 0;
}
ssize_t mycdev_write(struct file *file, const char __user *ubunf, size_t size, loff_t *lft)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
} 
int mycdev_close(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
} 
//定义一个操作方法结构体变量
struct file_operations fops={
    .open=mycdev_open,
    .read=mycdev_read,
    .write=mycdev_write,
    .release=mycdev_close,
};
 
static int __init mycdev_init(void)
{
    //进行字符设备驱动的注册
    major=register_chrdev(0,"mycdev",&fops);
    if(major<0)
    {
        printk("字符设备驱动注册失败\n");
    }
    printk("字符设备驱动注册成功 major=%d\n",major);
    return 0;
}
static void __exit mycdev_exit(void)
{
    //进行字符设备驱动的注销
    unregister_chrdev(major,"mycdev");
 
}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

安装驱动的镜像,查看/proc/devices文件中,可以发现驱动被注册

为驱动手动创建一个设备文件

创建的命令:

mknod 设备文件路径+名字 设备文件类型(c/b) 主设备号 ,次设备号

编写一个应用程序,在应用程序中操作设备文件,执行查看对应的操作是否被回调

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char const *argv[])
{
  int fd=open("/dev/mycdev",O_RDWR);
   if(fd<0)
{

   printf("设备文件打开失败");
    return fd;

}
 char buf[128]={};
  while(1)
{
   write(fd,buf,sizeof(buf));
   sleep(1);
}
close(fd);
return 0;










}

4 用户和内核之间的数据传递

内核和用户进行数据传递的api

#include <linux /uaccess.h>
unsigned long copy_to_user(void _user *to,const void *from,unsigned long n)
功能:从内核传递一个数据给用户进程
参数
:to 用户空间保存数据buf的首地址
from: 内核空间保存数据的buf首地址
n :实际传递的数据大小
返回值 :成功返回0,失败返回没有传递成功的数据大小


unsigned long copy_from_user(void _user *to,const void *from,unsigned long n)
功能:用户传递数据给内核空间
参数:
to :内核空间保存数据的buf首地址
from:用户空间保存buf的首地址
返回值 :成功返回0,失败返回没有传递成功的数据的大小
ssize_t mycdev_read(struct *file ,char _user *ubuf,size_t size,loft_t *lft)
{
unsigned long ret=copy_to_user(ubuf,kbuf,size);  
if(ret)
{
printk("copy_to_user filad\n" );
return ret;


}
return 0;


}
ssize_t mycdev_write(struct *file,const char _user*ubuf,size_t size,loff_t*lft)
{
unsigned long ret=copy_from_user(kbuf,ubuf,size);
if(ret)
{
printf("copy_from_user filad\n");
return ret;


}
return 0;


}
int main(int argc ,char const *argv[])
{
int fd=open("/dev/mycdev",O_RDWR);
if(fd<0)
{

printf("设备文件打开失败\n");
return fd;

}
char buf[128]="hello world";
write(fd,buf,sizeof(buf));
memset(buf,0,sizeof(buf)));
read(fd,buf,sizeof(buf));
printf("buf:%s\n",buf);
close(fd);


}

物理内存的内存映射API

想要完成硬件的控制直接操作硬件的寄存器,硬件的寄存器属于物理内存,我们的驱动程序会加载到虚拟内存中,我们需要寄存器的物理内存映射成虚拟内存才能完成设备的控制

void *ioremap(phys——addr_t paddr,unsigned long size)
功能将指定大小的物理内存映射为虚拟内存
参数:
paddr :要映射的物理内存首地址数值
size :要映射的物理内存的大小
返回值
成功返回映射的虚拟内存的首地址
失败返回NULL;
void ioumap(const void __iomem *addr)
功能:取消物理内存的映射
参数:
addr :物理内存对应的首地址
返回值 :无

LED1 ->PE10

LED2->PF10

LED3->PE8

要对LED1进行控制

将PE10设置为输出模式 GPIOE_MODER[21:20]->01 0X50006000

PE10输出高低电平 GPIOE_ODR[10]  1输出高电平 

rcc_mp_ahb4enster [4]->1 使能GPIOE外设时钟 050000A28

ssize_t myled_write(struct file *file, const char __user *ubuf, size_t size, loff_t *lft)
{
    //printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    unsigned long ret= copy_from_user(kbuf,ubuf,size);//读取用户写入的数据
   if(ret)
   {
  printk("copy from user filad\n");
return ret;


}
if(kbuf[0]=='1')
{

(*vir_odr)|=(0x1<<10);

}
else if(kbuf[0]=='0')
{
(*vir_odr) &=(~(0X1)<<10);

}

return 0;
static int __init mycdev_init(void)
{

major =register_chrdev(0,"myled",&fops);
if(major<0)
{


printk("字符设备驱动祖册失败\n");



}

printk("字符设备驱动注册成功 major=%d\n",major);
vir_moder=ioremap(PHY_GPIOE_MODER,4);
if(vir_moder ==NULL)
{
printk("物理内存映射失败% ")
}


}

相关文章:

  • 2024华为OD机试真题-第k个排列(C++/Java/Python)-E卷-100分
  • Mac OS JAVA_HOME设置
  • 【数据分析】2.数据分析业务全流程
  • LLM 推理中推理-时间计算技巧
  • debian 12 安装 NVIDIA 390驱动记录
  • 6.3 DBMS的功能和特征
  • 网络安全治理模型
  • 自由学习记录(36)
  • C#的序列化[Serializable()]
  • [AI]docker封装包含cuda cudnn的paddlepaddle PaddleOCR
  • OkHttp使用和源码分析学习(一)
  • 【设计模式】【创建型模式】原型模式(Prototype)
  • 免费搭建个人网站
  • Java 比较器:Comparable vs. Comparator
  • XTOP3D的DIC技术在极端条件下的应用解决方案
  • Pyecharts系列课程07——饼图(Pie)
  • 【AI实践】阿里百炼文本对话Agent安卓版搭建
  • SpringBoot速成概括
  • Dfs分布式文件存储
  • 如何在Windows下使用Ollama本地部署DeepSeek R1
  • 上交所五方面落实募资新规:强化关键少数责任和股东权利保障
  • 商务部回应稀土出口管制问题
  • 鸿海下调全年营收展望:AI服务器业务强劲,预计今年营收增超50%
  • 制造四十余年血腥冲突后,库尔德工人党为何自行解散?
  • 绿景中国地产:洛杉矶酒店出售事项未能及时披露纯属疏忽,已采取补救措施
  • 图讯丨习近平出席中国-拉美和加勒比国家共同体论坛第四届部长级会议开幕式