嵌入式Linux设备树定制:GPIO与中断资源映射深度解析
扫描二维码
随时随地手机看文章
在嵌入式Linux开发中,设备树(Device Tree)已成为硬件抽象的核心机制,其通过动态描述硬件资源,使内核能够灵活适配不同硬件平台。本文聚焦GPIO与中断资源的设备树映射技术,结合实际案例解析其配置方法与优化策略。
一、GPIO资源映射:从硬件到内核的桥梁
1.1 GPIO控制器节点配置
GPIO控制器的设备树节点需包含关键属性:
dts
gpio0: gpio@10000000 {
compatible = "vendor,gpio-v1"; // 驱动匹配标识
reg = <0x10000000 0x1000>; // 寄存器基地址与长度
gpio-controller; // 声明为GPIO控制器
#gpio-cells = <2>; // 每个GPIO需2个参数:编号+配置
interrupts = <5 IRQ_TYPE_LEVEL_HIGH>; // 中断配置
};
其中#gpio-cells定义了GPIO描述符的参数数量,典型值2表示第一个参数为GPIO编号,第二个参数为电气特性(如GPIO_ACTIVE_LOW表示低电平有效)。
1.2 设备级GPIO绑定
设备节点通过gpios属性引用GPIO控制器资源:
dts
led: led-controller {
compatible = "custom,led-driver";
gpios = <&gpio0 12 GPIO_ACTIVE_LOW>; // 使用gpio0的第12脚,低电平有效
};
驱动程序通过gpiod_get()获取GPIO描述符:
c
struct gpio_desc *led_pin;
led_pin = gpiod_get(&pdev->dev, "led", GPIOD_OUT_LOW); // 配置为输出,初始低电平
if (IS_ERR(led_pin)) {
dev_err(&pdev->dev, "Failed to get GPIO\n");
return PTR_ERR(led_pin);
}
二、中断资源映射:多级中断控制器协同
2.1 中断控制器层级建模
现代SoC通常采用多级中断控制器(如GICv3+GPIO控制器):
dts
gic: interrupt-controller@2f000000 {
compatible = "arm,gic-v3";
#interrupt-cells = <3>; // 每个中断需3个参数:中断号/触发方式/优先级
interrupt-controller;
};
gpio0: gpio@10000000 {
compatible = "vendor,gpio-v1";
interrupt-parent = <&gic>; // 指定父中断控制器
interrupts = <GIC_SPI 33 IRQ_TYPE_LEVEL_HIGH>; // SPI中断33,高电平触发
};
2.2 设备中断绑定与处理
设备节点通过interrupts属性声明中断需求:
dts
button: button-controller {
compatible = "custom,button-driver";
interrupt-parent = <&gpio0>; // 使用gpio0作为中断控制器
interrupts = <3 IRQ_TYPE_EDGE_FALLING>; // GPIO3,下降沿触发
};
驱动程序通过gpiod_to_irq()获取Linux IRQ号:
c
int irq_num = gpiod_to_irq(button_pin); // 获取中断号
if (devm_request_irq(&pdev->dev, irq_num, button_isr,
IRQF_TRIGGER_FALLING, "button-irq", NULL) < 0) {
dev_err(&pdev->dev, "Failed to request IRQ\n");
return -EINVAL;
}
中断处理函数需高效执行,复杂任务可委托给工作队列:
c
static irqreturn_t button_isr(int irq, void *dev_id) {
schedule_work(&button_work); // 调度工作队列
return IRQ_HANDLED;
}
三、实战案例:RK3568平台触摸屏驱动
在RK3568平台中,触摸屏通过GPIO中断实现事件上报:
dts
ts: touchscreen@38 {
compatible = "edt,edt-ft5306";
reg = <0x38>;
touch-gpio = <&gpio0 RK_PB5 IRQ_TYPE_EDGE_RISING>; // PB5上升沿触发
interrupt-parent = <&gpio0>;
interrupts = <RK_PB5 IRQ_TYPE_LEVEL_LOW>; // 低电平中断(备用)
};
驱动程序关键代码:
c
// 获取GPIO与中断号
struct gpio_desc *ts_gpio = gpiod_get(&pdev->dev, "touch", GPIOD_IN);
int irq_num = gpiod_to_irq(ts_gpio);
// 配置中断触发方式
if (devm_request_irq(&pdev->dev, irq_num, ts_isr,
IRQF_TRIGGER_RISING, "ts-irq", NULL) < 0) {
dev_err(&pdev->dev, "Failed to request IRQ\n");
return -EINVAL;
}
四、优化与调试技巧
资源释放:使用devm_系列函数(如devm_gpiod_get())自动管理资源,避免内存泄漏。
并发控制:对共享GPIO资源使用spin_lock_irqsave()保护。
调试工具:
查看GPIO状态:cat /sys/kernel/debug/gpio
动态修改设备树:通过dtoverlay机制在运行时加载覆盖片段。
通过合理配置设备树,开发者可实现GPIO与中断资源的高效映射,显著提升嵌入式Linux系统的硬件适配能力与稳定性。掌握这些技术,是迈向高级嵌入式开发的关键一步。





