应用程序传递数据给驱动和驱动操作LED灯

news2025/1/13 10:13:37

目录

1. 应用程序将数据传递给驱动

1.1. 函数分析

1.2. 编写驱动.c文件

1.3. 编写编译驱动的makefile文件

1.4. 执行make命令,并安装驱动,生成设备文件

1.5. 写应用层.c文件

1.6. 执行可执行文件验证

2. 驱动操作LED灯

2.1. 函数分析

2.2. 手册和原理图

2.3. 编写驱动.c文件

2.4. 编写驱动的makefile文件

2.5. 执行make命令,生成.ko文件

2.6. 将驱动文件放到nfs同步到开发板中

2.7. 在开发板上安装此驱动


1. 应用程序将数据传递给驱动

1.1. 函数分析

//记忆技巧:从用户角度看
》1.函数原型:int copy_from_user(void *to, const void __user *from, int n)
功能:从用户空间拷贝数据到内核空间(from user)
参数:
	to:内核空间首地址
	from:用户空间首地址
	n:拷贝数据的长度
返回值:
	成功:0
	失败:未拷贝字符个数

》2.函数原型:int copy_to_user(void __user *to,const void *from,int n)
功能:从内核空间到用户空间(to user)
参数:
	to:用户空间
	from:内核空间
	n:拷贝数据长度
返回值:
	成功:0
	失败:未拷贝字符个数

1.2. 编写驱动.c文件

//应用程序和驱动程序传参
#include <linux/init.h>
#include <linux/module.h>
#include <linux/printk.h>
#include <linux/fs.h> 
#include <linux/uaccess.h> //添加头文件

unsigned int major =0; 
#define NAME "hello"

int error_num=0;//定义变量用来判错
char kbuf[128]={0};//定义数组用来传输数据


ssize_t mycdev_read (struct file *file, char __user *ubuf, size_t size, loff_t * offs)
{
if(size>sizeof(kbuf)) //添加判断让数据不过大
{
size=sizeof(kbuf);
}
//站在用户角度,数据到用户
error_num=copy_to_user(ubuf,kbuf,size);
if(error_num)
{
	printk("copy_to_user error\n");
	return -1;
}
return 0;
}

ssize_t mycdev_write (struct file *file, const char __user *ubuf, size_t size, loff_t *offs)
{
if(size>sizeof(kbuf)) //添加判断让数据不过大
{
size=sizeof(kbuf);
}
//站在用户角度,用户写是数据来自用户
error_num=copy_from_user(kbuf,ubuf,size);
if(error_num)
{
	printk("copy_from_user error\n");
	return -1;
}
return 0;
}

int mycdev_open(struct inode *inode, struct file *file)
{
printk("hello open\n");
return 0;
}

int mycdev_release (struct inode *inode, struct file *file)
{
printk("hello close\n");
return 0;
}


struct file_operations fops={
.open=mycdev_open,
.write=mycdev_write,
.release=mycdev_release,
.read=mycdev_read,
};

static int __init hello_init(void)
{
major= register_chrdev(major, NAME, &fops);
if(major<0)
{
printk("register_chrdev error\n");
return major;
}

return 0;
}

static void __exit hello_exit(void)
{

unregister_chrdev(major, NAME);

}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");

1.3. 编写编译驱动的makefile文件

KERNELDIR:=/lib/modules/$(shell uname -r)/build
#KERNELDIR:=/home/hq/temp/kernel-3.4.39/
PWD:=$(shell pwd)
all:
	make -C $(KERNELDIR) M=$(PWD) modules
clean:
	make -C $(KERNELDIR) M=$(PWD) clean
obj-m:=hello.o

1.4. 执行make命令,并安装驱动,生成设备文件

》1.执行make命令

》2.安装驱动

sudo insmod hello.ko 安装驱动命令

cat /proc/devices 查看主设备号

3.生成设备文件

sudo mknod hello c 244 1 生成设备文件

添加设备文件权限(不能忘记)

1.5. 写应用层.c文件

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>

char buf[128]={"my name is southernbird"};//写入数据
int main(int argc,const char *argv[])
{
  int fd;
  fd=open("./hello",O_RDWR);
  if(fd==-1)
  {
    perror("open error");
    return -1;
  }
  write(fd,buf,sizeof(buf));
  memset(buf,0,sizeof(buf)); //清空
  read(fd,buf,sizeof(buf));
  printf("%s\n",buf);//用来验证
  close(fd);
  return 0;
}

使用gcc编译器进行编译,生成可执行文件

1.6. 执行可执行文件验证

》1.使用普通文件hello的情况

》2.使用设备文件

数据从应用层到驱动,又从驱动又回到了应用层

2. 驱动操作LED灯

2.1. 函数分析

驱动如何操作寄存器:

Led灯的寄存器是物理地址

Linux内核启动之后,操作的全是虚拟地址

这就牵扯到了虚拟地址和物理地址之间的转换

》1.函数原型:void * ioremap(phys_addr_t offset, unsigned long size)
功能:将物理地址映射成虚拟地址
参数:
	offset:要映射的地址
	size:间下方解释
返回值:
	成功:返回虚拟地址
	失败:NULL
》2.函数原型:void iounmap(void  *addr)
功能:将映射取消
参数:虚拟地址
返回值:
	无

2.2. 手册和原理图

1个为4字节,大小就应该写36

红灯引脚

绿灯引脚

蓝灯引脚

想要让灯灭,将三个引脚输出低电平就可

2.3. 编写驱动.c文件

#include <linux/init.h>
#include <linux/module.h>
#include <linux/printk.h>
#include <linux/fs.h> 
#include <linux/uaccess.h> 
#include <linux/io.h>//添加内存映射的函数

unsigned int major =0; 
#define NAME "hello"

int error_num=0;
char kbuf[128]={0};

//定义三个灯的基地址
#define RED_BASE 	0XC001A000
#define BLUE_BASE	0XC001B000
#define GREEN_BASE 	0XC001E000

//定义三个指针用来存放映射后的虚拟地址
unsigned int * red_base=NULL;
unsigned int * blue_base=NULL;
unsigned int * green_base=NULL;


ssize_t mycdev_read (struct file *file, char __user *ubuf, size_t size, loff_t * offs)
{
if(size>sizeof(kbuf))
{
size=sizeof(kbuf);
}
error_num=copy_to_user(ubuf,kbuf,size);
if(error_num)
{
	printk("copy_to_user error\n");
	return -1;
}
return 0;
}

ssize_t mycdev_write (struct file *file, const char __user *ubuf, size_t size, loff_t *offs)
{
if(size>sizeof(kbuf)) 
{
size=sizeof(kbuf);
}

error_num=copy_from_user(kbuf,ubuf,size);
if(error_num)
{
	printk("copy_from_user error\n");
	return -1;
}
return 0;
}

int mycdev_open(struct inode *inode, struct file *file)
{
printk("hello open\n");
return 0;
}

int mycdev_release (struct inode *inode, struct file *file)
{
printk("hello close\n");
return 0;
}


struct file_operations fops={
.open=mycdev_open,
.write=mycdev_write,
.release=mycdev_release,
.read=mycdev_read,
};

static int __init hello_init(void)
{
major= register_chrdev(major, NAME, &fops);
if(major<0)
{
printk("register_chrdev error\n");
return major;
}

//在入口处将物理地址映射成虚拟地址
red_base=ioremap(RED_BASE, 36);//大小最大就是36
if(red_base ==NULL)
{
printk("ioremap error RED_BASE\n");
return -ENOMEM;	//返回内存溢出
}
 //注意解引用
*red_base&=~(1<<28); //灭红灯
*(red_base+1)|=(1<<28);//设置为输出模式
*(red_base+9)&=~(3<<24);//选择function 0


green_base=ioremap(GREEN_BASE,36);
if(green_base==NULL)
{
printk("ioremap error GREEN_BASE\n");
return -ENOMEM;
}

*green_base&=~(1<<13);//灭绿灯
*(green_base+1)|=(1<<13);//设置为输出模式
*(green_base+8)&=~(3<<26);//选择function 0

blue_base=ioremap(BLUE_BASE,36);
if(blue_base==NULL)
{
printk("ioremap error BLUE_BASE\n");
return -ENOMEM;
}

*blue_base&=~(1<<12);//灭蓝灯
*(blue_base+1)|=(1<<12);//设置为输出模式
*(blue_base+8)|=(1<<25);//将25位置为1
*(blue_base+8)&=~(1<<24);//将24位置为0

return 0;
}

static void __exit hello_exit(void)
{
//注销内存
iounmap(blue_base);//先注册的后注销
iounmap(green_base);
iounmap(red_base);

unregister_chrdev(major, NAME);

}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");

2.4. 编写驱动的makefile文件

因为驱动是基于开发板写的,编译的时候使用开发板内核

#KERNELDIR:=/lib/modules/$(shell uname -r)/build  
KERNELDIR:=/home/hq/temp/kernel-3.4.39/ 
PWD:=$(shell pwd)
all:
	make -C $(KERNELDIR) M=$(PWD) modules
clean:
	make -C $(KERNELDIR) M=$(PWD) clean
obj-m:=hello.o

2.5. 执行make命令,生成.ko文件

2.6. 将驱动文件放到nfs同步到开发板中

结合我驱动系统移植篇的博客可以知道我的nfs搭建在

/opt/6818/rootfs/rootfs下

将hello.ko放到此文件下

2.7. 在开发板上安装此驱动

下面是观测的现象

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/667269.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C语言】第一个C语言项目——“猜数字”游戏(内附源码)

君兮_的个人主页 勤时当勉励 岁月不待人 C/C 游戏开发 Hello米娜桑&#xff0c;这里是君兮_&#xff0c;今天又抽空为大家更新我们的主线0基础C语言啦&#xff01;鉴于最近讲解了非常多的选择语句与循环语句&#xff0c;咱们今天就来讲讲两者结合的一个简单的实战应用。 同时…

Python Locust全过程使用代码详解

下方查看历史精选文章 重磅发布 - 自动化框架基础指南pdfv1.1大数据测试过程、策略及挑战 测试框架原理&#xff0c;构建成功的基石 在自动化测试工作之前&#xff0c;你应该知道的10条建议 在自动化测试中&#xff0c;重要的不是工具 Python locust 是一个基于 Python 的开源负…

MKS SERVO4257D 闭环步进电机_系列9 上位机通讯示例

第1部分 产品介绍 MKS SERVO 28D/35D/42D/57D 系列闭环步进电机是创客基地为满足市场需求而自主研发的一款产品。具备脉冲接口和RS485/CAN串行接口&#xff0c;支持MODBUS-RTU通讯协议&#xff0c;内置高效FOC矢量算法&#xff0c;采用高精度编码器&#xff0c;通过位置反馈&a…

视觉SLAM十四讲——ch13实践(设计SLAM系统)

视觉SLAM十四讲——ch13的实践操作及避坑 1. 实践操作前的准备工作2. 实践过程2.1 运行测试程序2.2 运行00数据集2.3 更改代码画出运动轨迹 3. 遇到的问题及解决办法3.1 cmake ..时出现的问题3.2 make时出现的问题3.3 头文件下红色报错 1. 实践操作前的准备工作 下载Kitti数据…

使用dat.gui更改three.js中的物体变量

一、dat.gui介绍 gui是一种JavaScript库&#xff0c;用于创建可视化控件和调试工具。它是dat.gui的简称。dat.gui是一个用于在Web应用程序中创建可定制GUI的JavaScript库。它可以轻松创建滑块、复选框、颜色选择器等控件&#xff0c;用户可以直接在GUI上进行交互和调整。dat.g…

一起来看看 K-verse LAND 销售活动中的合作伙伴给大家的祝福吧~

K-verse 是 The Sandbox 中的韩国内容主题空间&#xff0c;自去年 12 月首次推出以来&#xff0c;已吸引多家合作伙伴加入。此外&#xff0c;现有的合作伙伴公司和品牌正在积极准备以新的形式展示元宇宙内容。 这里有着许多可能性&#xff0c;K-verse LAND 销售活动是不是让你们…

Tomcat及项目部署

一、Tomcat是什么&#xff1f; Tomcat 是基于 Java 实现的⼀个开源免费, 也是被⼴泛使⽤的 HTTP 服务器。 二、下载安装 官⽅⽹站&#xff1a;https://tomcat.apache.org/ 选择其中的 zip 压缩包, 下载后解压缩即可. 解压缩的⽬录最好不要带 "中⽂" 或者 特殊符号…

vue-cli 如何修改默认环境变量名称

比如想要修改开发环境 NODE_ENV 的默认值 &#xff1f; 1. 新建文件 .env.development 2. 在 packjson.json 的 script 中添加一行代码 --mode [文件 env 后面的环境名称] "dev": "vue-cli-service serve --mode development", 3. 然后 npm run dev 环境变…

JavaScript ES12新特性有哪些?

文章目录 导文Promise.any()WeakRef 和 FinalizationRegistry数字分隔符String.prototype.replaceAll()Logical Assignment Operators数字类型的新增方法私有字段和方法 导文 JavaScript ES12&#xff08;也称为ECMAScript 2022&#xff09;是JavaScript的最新版本&#xff0c;…

如何解决报错:nginx error!

目录 Nginx报错问题 nginx error! The page you are looking for is not found. Website Administrator 解决方法 Nginx报错问题 当访问搭建好的Nginx服务网站时 有以下报错 nginx error! The page you are looking for is not found. Website Administrator Someth…

猪齿鱼开源发布2.0版本:DevOps能力全面升级,研发效能显著提升,欢迎即刻体验!

近日&#xff0c;甄知科技猪齿鱼Choerodon数智化开发管理平台正式发布了开源2.0版本&#xff01; 开源发布会上&#xff0c;甄知产研团队、业内伙伴和社区开发者们齐聚一堂&#xff0c;共同见证猪齿鱼开源2.0的重磅发布&#xff01;发布会由上海甄知科技创始合伙人兼CTO张礼军先…

前端添加代理通过nginx进行转发解决跨域

记录在项目中遇到跨域并进行解决的方案 解决方案 记录在项目中遇到跨域并进行解决的方案前端代理部分nginx转发配置origin限制,修复CORS跨域漏洞 前端代理部分 代理后页面请求地址截图&#xff1a; 这里地址栏的地址是&#xff1a;http://127.0.0.1:13908 调用登录接口请求地…

OrCAD Capture 元件位号Part Reference有下划线

原因&#xff1a; 提示用户曾经修改过原理图封装。 现象&#xff1a; USB20_12 解决办法&#xff1a; 对着元器件右键>User Assigned Reference > Uset&#xff0c;即可消除下划线。 修改后&#xff1a;

通过域名的方式访问服务器里的资源

大家好&#xff0c;我是雄雄。欢迎关注微信公众号&#xff1a;雄雄的小课堂 前言 在平时的项目过程中&#xff0c;我们可能经常会遇到这样的场景。 上传资源&#xff0c;比如图片或者视频到服务器中&#xff0c;上传上去后&#xff0c;我们给数据库中存的是文件所在路径&…

SSMP整合案例(3) 创建数据层并在测试类中运行数据库增删查改操作

上文 SSMP整合案例(2) Spring Boot整合Lombok简化实体类开发我们已经开发完了实体类 我们就可以做数据层了 目前来讲 数据层技术 使用了最大的自然是 MyBatis 但其实MyBatis-Plus在国内很多中小企业还是使用的挺多的 这次 我们主要是通过MyBatis-Plus和Druid来做这件事情 这两…

5款界面简洁无广告的轻量级小软件

今天的主题是简洁&#xff0c;轻便&#xff0c;都是轻量级的小软件&#xff0c;界面都是非常简洁&#xff0c;而且无广告的。 文件同步——Syncthing Syncthing是一款用于同步和分享文件的工具。它可以让你在不同的设备上同步你的文件夹&#xff0c;并提供多种功能和选项来设…

鱼眼相机成像模型以及基于OpenCV标定鱼眼镜头(C++)

opencv系列 文章目录 opencv系列一、鱼眼镜头模型二、投影函数等距投影模型等立体角投影模型正交投影模型体视投影模型 三、OpenCV中的鱼眼相机模型四、标定&#xff08;C&#xff09;实现使用的函数采集标定图像标定代码标定结果 一、鱼眼镜头模型 鱼眼镜头一般是由十几个不同…

新能源充电桩4G无线物联网解决方案|4G路由器ZR2000

日常生活中新能源汽车已随处可见&#xff0c;新能源也逐渐普遍&#xff0c;绿色出行、低碳生活的环保概念也随着科普深入人心&#xff0c;新能源汽车必备的充电桩行业随之崛起&#xff0c;为保证用户体验及运营管理&#xff0c;充电桩需要通过网络实现数据传输、远程监控、位置…

19-递归的理解、场景

一、递归 &#x1f32d;&#x1f32d;&#x1f32d;在函数内部&#xff0c;可以调用其他函数。如果一个函数在内部调用自身本身&#xff0c;这个函数就是递归函数 核心思想是把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解 一般来说&#xff0c;递归…

Azure-FunctionApp入门实战

介绍 FuntionApp 是微软云推出的一款serverless产品服务&#xff0c;作为coder可以无需关心程序部署所需的infra信息&#xff0c;只需要focus自己的业务代码实现即可 使用场景 是不是所有的业务都可以使用serverless产品呢&#xff1f;理论上是可以的&#xff0c;但是从实际…