平台设备总线platfrom 框架开发

2024-08-26 10:12

本文主要是介绍平台设备总线platfrom 框架开发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1:认识平台设备总线

1.1 平台设备总线是什么

平台设备总线:在 Linux 内核中,设备驱动程序是操作系统与硬件设备之间的桥梁。为了管理不同的硬件设备,Linux 内核引入了不同的设备总线(Bus)模型,其中“平台设备总线”(Platform Bus)是一个非常常见的总线类型。

总线设备示意图:

platform 是更早与设备树之前的分层思想

platform是内核抽象/虚拟的一条管理设备驱动的总线

在plaform框架下驱动被分为两层:

        第一层:设备层只提供设备信息

        第二层:驱动层通用动

随着内核的发展:
        其中的设备层演变成->设备树

在platform这条总线的管理下:

        内核里面驱动会自动匹配设备树

        内核层的驱动会自动获取设备树节点

2:平台设备总线的匹配方法:

2.1 老式的匹配方法:

        platform_device * drvier.name 函数传参相对麻烦的方式进行的,需要靠代码驱动的方法加载内核传参。

2.2 新试匹配方法

        在设备树的出现后,匹配方法就改变成:

                compatible = “abb”,“sadc”;

        驱动层platform代码会有一个结构体里面 compatible,只需要保证传入的字符串和设备树的compatible其中的一个相同就可以匹配成功。

3:平台设备树总线的驱动层的相关接口函数

开发平台设备总线的框架:

//头文件
#include "linux/platform_device.h"
//入口函数
static int led_probe(struct platfrom_device *dev)
{return 0;
}
//出口函数
static int led_remove(struct platform_device *dev)
{return 0;
}
//内核注册信息
static struct platform_driver led_drv=
{.probe = led_probe,.remove = led_remove,匹配 compatible
}//加载函数
statict int __init led_init(void)
{platform_driver_register(&led_drv);
}
//卸载函数
static void __exit led_exit(void)
{//平台设备驱动注销platform_driver_unregister(&led_drv);
}


platform 只需要学习两个函数:

platform_driver_register();
platform_driver_unregister();

函数原型:

int platform_devicer_register(struct platform_driver * drv)

函数的参数:
drv:
        struct platform_driver {
        int (*probe)(struct platform_device *);//新入口
        int (*remove)(struct platform_device *);//新出口
        struct device_driver driver;//这里肯定有匹配的名字
};

* probe:
        你要提供的设备树匹配成功后执行的回调函数
        就是新入口函数
* remove:
        如果检测到驱动卸载/设备树异常就会执行 remove 回调函数
        新的出口函数
* driver:
        name: 是一个标签名->老式的匹配方法
        of_match_table:
        这就是有设备树的匹配的方法
        在这结构体里面虽说有四个成员
        只需要管 compatible 这个成员变量
函数返回值:
        注册成功返回 0
        注册失败返回 非 0
int platform_driver_unregister(struct platform_driver *drv)
取消注册驱动
怎么取消就行了

4:平台设备总线下LED灯驱动事例

#include "linux/kernel.h"
#include "linux/module.h"
#include "linux/of.h"
#include "linux/cdev.h"
#include "linux/fs.h"
#include "linux/gpio.h"
#include "linux/of_gpio.h"
#include "linux/device/class.h"
#include "linux/device.h"
#include "linux/platform_device.h"
struct class * cls;
dev_t devnum;
struct cdev  xydledcdev;
struct file_operations ops;
struct device_node * led_node;
struct xyd_led_gpio{int gpio_num;int gpio_flag;char name[32];
};
struct xyd_led_gpio myxyd_led_info;struct of_device_id device_id[]={{.compatible = "xyd_led"},};struct platform_driver drv;//开灯回调函数
int led_open (struct inode *i, struct file *f)
{gpio_set_value(myxyd_led_info.gpio_num,myxyd_led_info.gpio_flag);return 0;
}
//关灯回调函数
int led_close (struct inode *i, struct file *f)
{gpio_set_value(myxyd_led_info.gpio_num,!myxyd_led_info.gpio_flag);return 0;
}//新入口
int led_probe(struct platform_device * dev)
{//在这里进行资源初始化,设备注册 硬件初始化led_node = dev->dev.of_node;//设备树节点信息//然后一一获取设备树属性 GPIO口 statues 有效点平等//1.获取GPIO的属性myxyd_led_info.gpio_num = of_get_named_gpio(led_node, "xyd-gpios", 0);if(myxyd_led_info.gpio_num < 0){printk("该设备树未提供 GPIO 信息! \r\n");return -EINVAL;}//2.获取GPIO的有效点平enum of_gpio_flags flags =0;of_get_named_gpio_flags(led_node,"xyd-gpios" , 0,&flags);if(flags == OF_GPIO_ACTIVE_LOW){myxyd_led_info.gpio_flag=0;	}else{myxyd_led_info.gpio_flag=1;}//3:封装设别名sprintf(myxyd_led_info.name,"xyd_led_%d",myxyd_led_info.gpio_num);//4:初始化GPIOgpio_request(myxyd_led_info.gpio_num,myxyd_led_info.name);gpio_direction_output(myxyd_led_info.gpio_num,!myxyd_led_info.gpio_flag);//5:申请一个设备号alloc_chrdev_region(&devnum, 0,1,myxyd_led_info.name);//6:初始化cdevops.open = led_open;ops.release = led_close;ops.owner= THIS_MODULE;cdev_init(&xydledcdev,&ops);//7:添加到内核cdev_add(&xydledcdev,devnum,1);//8:生成类结构体cls = class_create(THIS_MODULE,myxyd_led_info.name);//9:生成设备文件device_create(cls, NULL,devnum,NULL,myxyd_led_info.name);return 0;
}
//新出口               
int led_remove(struct platform_device *dev)
{//这里进行资源卸载 设备的取消注册 硬件释放//根据倒序的思想//1:销毁设备文件device_destroy(cls, myxyd_led_info.gpio_num);//2:销毁类文件class_destroy(cls);//3:从内核中删除cdev_del(&xydledcdev);//4:释放设备号unregister_chrdev_region(devnum, 1);//5:注销GPIO设备gpio_free(myxyd_led_info.gpio_num);return 0;
}//内核注册信息
//配置对应结构体
static struct platform_driver led_drv=
{.probe = led_probe,.remove = led_remove,.driver = {.name = "myxyd_led",.of_match_table =  device_id,},
};//工程原版框架入口函数
static int __init  leddrv_init(void)
{platform_driver_register(&led_drv);	return 0;
}//出口函数
static void __exit leddrv_exit(void)
{platform_driver_unregister(&led_drv);
}module_init(leddrv_init);
module_exit(leddrv_exit);
MODULE_LICENSE("GPL");



 

这篇关于平台设备总线platfrom 框架开发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1108271

相关文章

一文详解Python如何开发游戏

《一文详解Python如何开发游戏》Python是一种非常流行的编程语言,也可以用来开发游戏模组,:本文主要介绍Python如何开发游戏的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、python简介二、Python 开发 2D 游戏的优劣势优势缺点三、Python 开发 3D

基于Python开发Windows自动更新控制工具

《基于Python开发Windows自动更新控制工具》在当今数字化时代,操作系统更新已成为计算机维护的重要组成部分,本文介绍一款基于Python和PyQt5的Windows自动更新控制工具,有需要的可... 目录设计原理与技术实现系统架构概述数学建模工具界面完整代码实现技术深度分析多层级控制理论服务层控制注

Java中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例解析

《Java中的分布式系统开发基于Zookeeper与Dubbo的应用案例解析》本文将通过实际案例,带你走进基于Zookeeper与Dubbo的分布式系统开发,本文通过实例代码给大家介绍的非常详... 目录Java 中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例一、分布式系统中的挑战二

Java 缓存框架 Caffeine 应用场景解析

《Java缓存框架Caffeine应用场景解析》文章介绍Caffeine作为高性能Java本地缓存框架,基于W-TinyLFU算法,支持异步加载、灵活过期策略、内存安全机制及统计监控,重点解析其... 目录一、Caffeine 简介1. 框架概述1.1 Caffeine的核心优势二、Caffeine 基础2

基于Go语言开发一个 IP 归属地查询接口工具

《基于Go语言开发一个IP归属地查询接口工具》在日常开发中,IP地址归属地查询是一个常见需求,本文将带大家使用Go语言快速开发一个IP归属地查询接口服务,有需要的小伙伴可以了解下... 目录功能目标技术栈项目结构核心代码(main.go)使用方法扩展功能总结在日常开发中,IP 地址归属地查询是一个常见需求:

录音功能在哪里? 电脑手机等设备打开录音功能的技巧

《录音功能在哪里?电脑手机等设备打开录音功能的技巧》很多时候我们需要使用录音功能,电脑和手机这些常用设备怎么使用录音功能呢?下面我们就来看看详细的教程... 我们在会议讨论、采访记录、课堂学习、灵感创作、法律取证、重要对话时,都可能有录音需求,便于留存关键信息。下面分享一下如何在电脑端和手机端上找到录音功能

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

GSON框架下将百度天气JSON数据转JavaBean

《GSON框架下将百度天气JSON数据转JavaBean》这篇文章主要为大家详细介绍了如何在GSON框架下实现将百度天气JSON数据转JavaBean,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录前言一、百度天气jsON1、请求参数2、返回参数3、属性映射二、GSON属性映射实战1、类对象映