2-ARM Linux驱动开发-设备树平台驱动

2-ARM Linux驱动开发-设备树平台驱动

一、概述

  设备树(Device Tree)是一种描述硬件的数据结构,用于将硬件设备的信息传递给操作系统内核。它的主要作用是使内核能够以一种统一、灵活的方式了解硬件平台的细节,包括设备的拓扑结构、资源分配(如内存地址、中断号等)等信息,从而减少了将硬件信息硬编码到内核中的情况,增强了内核的可移植性和可维护性。

1、设备树的基本结构

1.节点(Node)

  设备树以节点为基本单元,节点用于描述一个设备或者一个设备集合。每个节点都有一个名字,名字通常反映了设备的类型或者功能。例如,一个典型的节点名字可能是 “uart@0x1000”,其中 “uart” 表示这是一个串口设备,“0x1000” 可能是该设备的基地址。节点可以包含属性(Properties)和子节点(Child Nodes)。

2.属性(Properties)

  属性是节点的重要组成部分,用于描述设备的各种特性。属性由键 - 值对(key - value pairs)组成。键是一个字符串,用于标识属性的名称,值可以是多种数据类型,如整数、字符串、字节数组等。例如,一个设备节点可能有一个属性 “compatible”,其值为 “nvidia,tegra - uart”,这个属性用于告诉内核该设备与哪种驱动程序兼容。

3.子节点(Child Nodes)

  一个节点可以包含多个子节点,用于描述设备的层次结构。比如,在一个 SoC(片上系统)设备树中,可能有一个 “cpu” 节点,它下面会有多个代表不同 CPU 核心的子节点,每个子节点可以包含自己的属性,用于描述该核心的特性,如频率、缓存大小等。

2、基本概念

1.DTS(Device Tree Source)

  DTS 是设备树源文件,它是一种文本格式的文件,用于描述硬件设备的详细信息。其内容包括设备的拓扑结构、设备之间的连接关系、资源分配(如内存地址、中断号等)以及设备的各种属性等。

image

  这里的/{​表示根节点,node1​是根节点下的一个子节点,它有两个属性property1​和property2​,并且还有一个子节点child_node​,这个子节点有自己的属性child_property​。属性的值可以是整数(用<>​包围)、字符串(用双引号包围)等多种形式。

/{
    node1 {
        property1 = <value1>;
        property2 = "string_value";
        child_node {
            child_property = <value2>;
        }
    }
}

2.DTC(Device Tree Compiler)

  DTC是设备树编译器,它是一个工具,用于将DTS文件编译成二进制的DTB文件。DTC会对DTS文件进行语法检查、格式转换等操作,以确保设备树信息的正确性和有效性,并将其转换为内核能够解析的二进制格式。

  DTC读取DTS文件的文本内容,根据设备树的语法规则进行解析。它会识别节点、属性等元素,并将其转换为二进制格式存储在DTB文件中。在编译过程中,DTC还可以进行一些优化操作,例如去除注释、压缩数据等,以减小DTB文件的大小。

  通常在构建内核或者制作嵌入式系统镜像时使用。例如,在交叉编译环境中,开发人员在完成DTS文件的编写后,会使用DTC工具将DTS编译为DTB文件,然后将DTB文件与内核镜像一起部署到目标设备上。

3.DTB(Device Tree Blob)

  DTB是设备树二进制文件,它是DTS文件经过DTC编译后的产物。内核在启动过程中能够直接解析DTB文件来获取硬件设备的信息。文件包含了设备树的所有信息,不过是以二进制形式存储。它的结构包括头部信息(如文件版本、大小等)和设备树数据部分。数据部分按照一定的格式组织,内核通过特定的解析代码可以从中提取出设备节点、属性等信息。

  当系统启动时,内核会读取 DTB 文件。例如,在一个基于 ARM 架构的嵌入式系统中,引导加载程序(如 U - Boot)会将 DTB 文件加载到内存中,然后内核启动时会解析 DTB 文件,根据其中的信息来识别硬件设备,如发现系统中的存储设备、网络设备等,并为这些设备分配资源,同时将设备与相应的驱动程序进行匹配。

4.DTSI(Device Tree Source Include)

  DTSI 是设备树源文件包含(Include)文件。它也是一种文本格式的文件,主要用于将设备树描述信息进行模块化和复用。

  在复杂的硬件系统中,可能有多个设备具有相同的基本属性或者结构。DTSI 文件可以用来定义这些通用的部分,然后在多个 DTS 文件中进行引用。例如,对于一个具有多个相同类型的 UART 设备的系统,可以将 UART 设备的通用属性(如基本的寄存器定义、中断处理方式等)定义在一个 DTSI 文件中,然后在每个具体描述 UART 设备的 DTS 文件中包含这个 DTSI 文件,从而减少代码的重复编写。

  ​#include "xxx.dtsi"

二、实操

1、设备树文件修改

/include/ "system-conf.dtsi"
/ {
	chosen{
		bootargs = "earlycon console=ttyPS0,115200 clk_ignore_unused root=/dev/mmcblk0p2 rw rootwait";
	};

	gpio-led {
		compatible = "zynqMP-led";
		gpios = <	&gpio 0x2A 0x0 
				&gpio 0x28 0x0>;
	};
};

&gpio {
	emio-gpio-width = <32>;
	gpio-mask-high = <0x0>;
	gpio-mask-low = <0x5600>;
	status = "okay";
};

&sdhci1 {
	clock-frequency = <100000000>;
	status = "okay";
	xlnx,mio_bank = <0x1>;
	no-1-8-v;
	disable-wp;
};

&uart0 {
	cts-override ;
	device_type = "serial";
	port-number = <0>;
	status = "okay";
	u-boot,dm-pre-reloc ;
};

&usb0 {
	status = "okay";
	xlnx,tz-nonsecure = <0x1>;
	xlnx,usb-polarity = <0x0>;
	xlnx,usb-reset-mode = <0x0>;
};

2、驱动文件编写

//1、添加头文件
#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/of_gpio.h>
#include <linux/platform_device.h>
#include <linux/mod_devicetable.h>

static unsigned int led_major;
static struct class *led_class;

struct led_driver
{
	int gpio1;
	int gpio2;
	int irq;
	struct device dev;
};
struct led_driver *led_dri = NULL;

const struct file_operations led_fops = {

};

//在probe函数中打印获取数据包里面的名字及GPIO
int gpio_pdrv_probe(struct platform_device *pdev)
{
	struct device_node *node;
	unsigned int gpio1, gpio2;
	unsigned int ret = 0;

	printk("gpio pdrv probe!\n");
	printk("pdrv name = %s!\n", pdev->name);

	//申请主设备号
	led_major = register_chrdev(0, "led_drv", &led_fops);
	if (led_major < 0)
	{
		printk("register chrdev led major error!\n");
		return -ENOMEM;
	}

	//创建类
	led_class = class_create(THIS_MODULE, "led_class");

	//创建设备
	device_create(led_class, NULL, MKDEV(led_major, 0), NULL, "led_device%d", 0);

	//硬件初始化

	//申请空间
	led_dri = devm_kmalloc(&pdev->dev, sizeof(struct led_driver), GFP_KERNEL);
	if (led_dri == NULL)
	{
		printk("devm kmalloc led_driver error!\n");
		return -1;
	}

	//获取从设备节点传过来的pdev中的dev及node节点
	led_dri->dev = pdev->dev;
	node = pdev->dev.of_node;

	//3.从node节点处获得GPIO号
	gpio1 = of_get_gpio(node, 0);
	printk("of get gpio1 number = %d\n", gpio1); 	//GPIO基数 338 + 40,338 是 512 - 174 得到的。
	if (gpio1 < 0)
	{
		printk("of get gpio error!\n");
		return -1;
	}
	gpio2 = of_get_gpio(node, 1);
	printk("of get gpio2 number = %d\n", gpio2);    //GPIO基数 338 + 42
	if (gpio2 < 0)
	{
		printk("of get gpio error!\n");
		return -1;
	}

	//4.申请GPIO
	ret = gpio_request(gpio1, "plattree_led");
	if (ret < 0)
	{
		printk("plattree led gpio request error!\n");
		return ret;
	}
	ret = gpio_request(gpio2, "plattree_led");
	if (ret < 0)
	{
		printk("plattree led gpio request error!\n");
		return ret;
	}

	//5.设置GPIO为输出模式,并设备为0,灭灯
	gpio_direction_output(gpio1, 0);
	gpio_direction_output(gpio2, 0);

	led_dri->gpio1 = gpio1;
	led_dri->gpio2 = gpio2;

	return 0;
}

int gpio_pdrv_remove(struct platform_device *pdev)
{
	printk("led pdrv remove!\n");
	//6.将gpio设为输出。
	gpio_set_value(led_dri->gpio1, 1);
	gpio_set_value(led_dri->gpio2, 1);
	gpio_free(led_dri->gpio1);
	gpio_free(led_dri->gpio2);
	device_destroy(led_class, MKDEV(led_major, 0));
	class_destroy(led_class);
	unregister_chrdev(led_major, "led_drv");

	return 0;
}

//of_match_table实现
const struct of_device_id gpio_of_match_table[] = {
	{
		.compatible = "zynqMP-led",
	},
	{}
};

//2.当驱动在设备中找到name之后,进行配对获取resource资源,进入probe函数
struct platform_driver gpio_drv = {
	.driver = {
		.name = "zynqmp_led",
		.of_match_table = gpio_of_match_table,
	},
	.probe = gpio_pdrv_probe,
	.remove = gpio_pdrv_remove,
};

//实现装载入口函数和卸载入口函数
static __init int platform_gpio_drv_init(void)
{
	//1.创建pdrv,并且注册到总线中
	return platform_driver_register(&gpio_drv);
}
static __exit void platform_gpio_drv_exit(void)
{
	//7.注销设备
	platform_driver_unregister(&gpio_drv);
}

//声明装载入口函数和卸载入口函数
module_init(platform_gpio_drv_init);
module_exit(platform_gpio_drv_exit);

//添加GPL协议
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Popeye");

image

posted on 2024-11-04 20:25  naive156  阅读(8)  评论(0编辑  收藏  举报

导航