【IMX6ULL驱动开发学习】12.Linux驱动之设备树

这篇具有很好参考价值的文章主要介绍了【IMX6ULL驱动开发学习】12.Linux驱动之设备树。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

承接上一篇博客 【IMX6ULL驱动开发学习】11.驱动设计之面向对象_分层思想(学习设备树过渡部分)

代码获取:https://gitee.com/chenshao777/imx6-ull_-drivers
我后面将三个层合并了(实际上只有前两层),合并成一个dev_drv.c了,暂时没有加GPIO操作,只是个框架
合并前的代码在 11.button_drv_chip_device-tree 文件夹中
合并后的代码在 12.led_button_drv_tree 文件夹中,文章最后把代码贴出来

打算在第13次代码中加入GPIO子系统的代码,并且根据Pinctrl子系统编写设备树,使得外设控制更简单,敬请期待哦!


之前我们将驱动程序分为了三层
1、驱动框架程序, 包含 file_operations 结构体(内核自带)
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动

2、硬件操作程序, 包含 chip_operations 结构体(自定义,初始化和操作GPIO),包含 platform_driver 结构体(内核自带,提取具体的引脚)
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动

3、硬件资源定义程序, 包含 platform_device 结构体(内核自带,resource 成员中了指定了具体的GPIO引脚)
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动


设备树的引入

像上面第三层那样定义硬件资源还是太麻烦,有没有一种更便捷的方式呢?
有,设备树!

1、修改设备树
介绍设备树语法的博客有很多,这就不做赘述了
直接介绍如何使用设备树来定义硬件资源吧!
以正点原子IMX6ULL阿尔法开发板来举例,首先找到你目前所使用的设备树文件
路径如下

linux内核文件名/arch/arm/boot/dts/imx6ull-alientek-emmc.dtd

后缀为 dtb 的是二进制的设备树文件,我们需要修改它,那么真正要操作的是其对应的 dts 文件,即 imx6ull-alientek-emmc.dts,使用任意工具打开,我这里使用 gedit 文本编辑器打开。

linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动
接着在根节点下添加自己的设备节点即可,例如
linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动

属性 含义
compatible 匹配设备节点和设备驱动,定义了该属性的节点,内核会自动生成 platform_device 结构体
pin 自定义属性,指定引脚,可通过 of_property_read_u32 函数取出该属性值
my_name 自定义属性
status 状态属性,“okay” 或者 “disabled”,表示是否使用该节点

2、编译设备树

回到Linux内核目录下,执行命令

make dtbs

只要更改了 dts 文件,就会重新编译生成新的 dtb 文件,然后使用 uboot 网络加载方式加载新的设备树 设置uboot使用网络加载zImage和dtb

即可替换成新的设备树了,如何查看是否替换成功呢
查看是否有我们自己添加的设备节点就行了,输入命令

ls /proc/device-tree

linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动
可以看到我后来添加的三个节点,第一步成功!

3、修改硬件操作程序的 platform_driver 结构体

(1)platform_driver 结构体添加 of_match_table 属性,添加 of_device_id 结构体,匹配设备树

linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动

(2)在 probe 函数中提取设备树中的引脚,提取设备树节点中的属性(of_property_read_u32等函数)

linux 设备树与驱动,IMX6ULL,驱动开发,linux,设备树,dtb,dts,IMX6ULL驱动
其他地方都不用修改,下次想添加外设时,直接修改设备树,然后修改 of_device_id 结构体即可


三层合并后的代码(实际上是两层合并后)OK,写完,晚安了各位!文章来源地址https://www.toymoban.com/news/detail-648825.html

#include <linux/module.h>
#include <linux/fs.h>
#include <linux/platform_device.h>
#include <linux/of.h>

int major;				//设备号
static struct class *my_dev_class;
int dev_cnt;
int dev_pins[10];

/*=============================file_operations ==============================*/
static ssize_t my_drv_read (struct file *filp, char __user *buf, size_t size, loff_t *offset)
{
	printk("drv_read function run....\n");
	return 1;
}

static ssize_t my_drv_write (struct file *filp, const char __user *buf, size_t size, loff_t *offset)
{
	printk("drv_write function run....\n");
	return 0;
}

static int my_drv_open (struct inode *node, struct file *filp)
{
	printk("drv_open function run....\n");
	return 0;
}

static int my_drv_release (struct inode *node, struct file *filp)
{

	printk("drv_release function run....\n");
	return 0;
}

/* operations结构体:为应用层提供驱动接口 */
static struct file_operations my_dev_ops = {
	.owner		= 	THIS_MODULE,
	.read 		=	my_drv_read,
	.write		=	my_drv_write,
	.open		=	my_drv_open,
	.release	=	my_drv_release,
};

/*=============================platform_driver==============================*/
/*  如果匹配到了内核根据设备树生成的platform_device,
	该函数会被调用,如果有多个匹配的设备节点,该函数
	会被多次调用
*/
static int my_probe(struct platform_device *pdev)
{
	/*  从内核根据设备树生成的 platform_device 
		结构体中获取到设备节点
	*/
	struct device *dev = &pdev->dev;
	struct device_node *np = dev->of_node;
	
	int pin;
	char a[20];
	const char *str = a;
	
	of_property_read_u32(np, "pin", &pin);
	of_property_read_string(np, "my_name", &str);

	//保存设备的引脚
	dev_pins[dev_cnt] = pin;  
	//创建设备节点 /dev/xxx
	device_create(my_dev_class, NULL, MKDEV(major, dev_cnt), NULL, str);
	dev_cnt++;

	printk("my_probe run, my_name = %s\n", str);
	
	return 0;
}

static int my_remove(struct platform_device *pdev)
{
	/*  从内核根据设备树生成的 platform_device 
		结构体中获取到设备节点
	*/
	struct device *dev = &pdev->dev;
	struct device_node *np = dev->of_node;
	
	int pin, i;
	char a[20];
	const char *str = a;
	
	of_property_read_u32(np, "pin", &pin);
	of_property_read_string(np, "my_name", &str);

	for(i = 0; i < dev_cnt; i++){
		if(dev_pins[i] == pin){
			dev_pins[i] = -1;
			device_destroy(my_dev_class, MKDEV(major, i));
			break;
		}
	}
	
	printk("my_remove run, device_destroy %s\n", str);
	return 0;
}

static struct of_device_id my_dev_match[] = {
	{.compatible = "hc-led-beep"}, 
	{.compatible = "hc-led-beep"}, 
	{.compatible = "hc-key"}, 
	{},
};

static struct platform_driver dev_driver = {
	.probe		=	my_probe,	
	.remove		= 	my_remove,
	.driver		= {
		.name	= "my_platform_driver",
		.of_match_table = my_dev_match,
	},
};

/*=============================驱动出入口函数==============================*/
/* 驱动入口函数:insmod xx.ko 时会被调用 */
static int dev_init(void)
{	
	major = register_chrdev(0, "hc_dev_drv", &my_dev_ops);
	if(major < 0){
		printk("register_chrdev famy\n");
		return major;
	}

	my_dev_class = class_create
(THIS_MODULE, "my_dev_class");
	if(IS_ERR(my_dev_class)){
		printk("class_create failed\n");
		return 1;
	}

	platform_driver_register(&dev_driver);

	return 0;
}

/* 驱动出口函数: rmmod xx.ko 时会被调用 */
static void dev_exit(void)
{
	platform_driver_unregister(&dev_driver);
	class_destroy(my_dev_class);
	unregister_chrdev(major, "hc_dev_drv");
	printk("my_dev driver exit\n");
}

module_init(dev_init);
module_exit(dev_exit);
MODULE_LICENSE("GPL");

到了这里,关于【IMX6ULL驱动开发学习】12.Linux驱动之设备树的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【IMX6ULL驱动开发学习】22.IMX6ULL开发板读取ADC(以MQ-135为例)

    IMX6ULL一共有两个ADC,每个ADC都有八个通道,但他们共用一个ADC控制器 在imx6ull.dtsi文件中已经帮我们定义好了adc1的节点部分信息 注意 num-channels = 2; ,这个表示指定使用ADC1的两个通道,即通道1和通道2 如果你要使用多个ADC通道,修改这个值即可 配置ADC引脚的 pinctrl ,在自己的

    2024年02月12日
    浏览(46)
  • 【IMX6ULL驱动开发学习】15.IMX6ULL驱动开发问题记录(sleep被kill_fasync打断)

    发现问题的契机: 学习异步通知的时候,自己实现一个功能:按键控制蜂鸣器,同时LED灯在闪烁 结果:LED好像也同时被按键控制了 最后调试结果发现: 应用层的 sleep 被驱动层的 kill_fasync 打断,所以sleep没有执行完就重新进入下一次循环了 修改代码后解决该问题 解决逻辑就

    2024年02月13日
    浏览(42)
  • 【IMX6ULL驱动开发学习】08.IMX6ULL通过GPIO子系统函数点亮LED

    通过GPIO子系统函数点亮LED 1、GPIO子系统函数 1.1 确定 led 的GPIO标号,查看内核中的gpiochip 查看 gpiochip ,以正点原子的IMX6ULL阿尔法开发板为例 查看原理图,发现led接的引脚是 GPIO1_IO3,对应 /sys/kernel/debug/gpio 中的 gpiochip0 组,gpiochip0 组从0开始算起, 所以 GPIO1_IO3 对应的标号就

    2024年02月10日
    浏览(67)
  • 【IMX6ULL驱动开发学习】14.Linux驱动开发 - GPIO中断(设备树 + GPIO子系统)

    代码自取 【14.key_tree_pinctrl_gpios_interrupt】: https://gitee.com/chenshao777/imx6-ull_-drivers 主要接口函数: 1. of_gpio_count (获得GPIO的数量) 2. kzalloc (向内核申请空间) 3. of_get_gpio (获取GPIO子系统标号) 4. gpio_to_irq (根据GPIO子系统标号得到软件中断号) 5. request_irq (根据软件中断号

    2024年02月12日
    浏览(40)
  • 【IMX6ULL驱动开发学习】03.设置IMX6ULL开发板与虚拟机在同一网段(设置开发板静态IP)

    为什么要设置IMX6ULL与虚拟机通信? 因为要把在虚拟机下编译的文件传到IMX6ULL开发板上运行 设置好同一网段,可以互ping后,可以参考这篇博客,实现开发板与虚拟机的文件互传 IMX6ULL开发板与虚拟机互传文件 一、设置windows有线网卡 二、配置虚拟机双网卡(原本有一个NAT网卡

    2024年02月07日
    浏览(43)
  • 【IMX6ULL驱动开发学习】09.Linux驱动之GPIO中断(附SR501人体红外感应驱动代码)

    Linux驱动的GPIO中断编程主要有以下几个步骤: 1、 通过GPIO号获取 软件中断号 (中断编程不需要设置GPIO输入输出,当然申请GPIO,设置输入也没问题) 参数 含义 gpio GPIO引脚编号 2、 注册 中断处理函数 ,设置中断 触发方式 (上升沿、下降沿等) 参数 含义 irq 软件中断号(通过

    2024年02月11日
    浏览(45)
  • 【IMX6ULL驱动开发学习】21.Linux驱动之PWM子系统(以SG90舵机为例)

    首先在 imx6ull.dtsi 文件中已经帮我们定义好了一些pwm的设备树节点,这里以pwm2为例 我们要在设备树(.dts)文件中引用和使能该节点,同时指定好pwm映射到的GPIO引脚(即pinctrl子系统,我这里映射到了GPIO1_9上) 使用pwm 只需要在设备树节点中添加两条属性信息,如下所示 pwms :属

    2024年02月12日
    浏览(109)
  • 【IMX6ULL驱动开发学习】19.mmap内存映射

    mmap将一个文件或者其它对象映射进内存 ,使得应用层可以直接读取到驱动层的数据,无需通过copy_to_user函数 可以用于像LCD这样的外设, 需要读写大量数据的 一、应用层 mmap用法: 用open系统调用打开文件, 并返回描述符fd. 用mmap建立内存映射, 并返回映射首地址指针start. 对映

    2024年02月16日
    浏览(40)
  • 【IMX6ULL驱动开发学习】10.Linux I2C驱动实战:AT24C02驱动设计流程

    前情回顾:【IMX6ULL驱动开发学习】09.Linux之I2C框架简介和驱动程序模板_阿龙还在写代码的博客-CSDN博客 目录 一、修改设备树(设备树用来指定引脚资源) 二、编写驱动 2.1 i2c_drv_read 2.2 i2c_drv_write 2.3 完整驱动程序 三、上机测试 放在哪个I2C控制器下面 AT24C02的I2C设备地址(查

    2024年02月11日
    浏览(41)
  • iMX6ULL驱动开发 | 让imx6ull开发板支持usb接口FC游戏手柄

    手边有一闲置的linux开发板iMX6ULL一直在吃灰,不用来搞点事情,总觉得对不住它。业余打发时间就玩起来吧,总比刷某音强。从某多多上买来一个usb接口的游戏手柄,让开发板支持以下它,后续就可以接着在上面玩童年经典游戏啦。  我使用的是正点原子的I.MX6U-ALPHA 开发板,

    2024年02月14日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包