【Linux驱动-快速回顾】简单了解一下PinCtrl子系统:设备树如何被接解析与匹配

下面用 "一段真实板级 .dts 代码"

"i2c1 节点 + pinctrl-0 从设备树 → 内核 → 寄存器" 的 全流程 逐行拆开讲清。

dts 复制代码
&i2c1 {
    clock-frequency = <100000>;
    pinctrl-names = "default";
    pinctrl-0 = <&pinctrl_i2c1>;
    status = "okay";
};

&iomuxc {
    pinctrl_i2c1: i2c1grp {
        fsl,pins = <
            MX6UL_PAD_UART4_TX_DATA__I2C1_SCL 0x4001b8b0
            MX6UL_PAD_UART4_RX_DATA__I2C1_SDA 0x4001b8b0
        >;
    };
};

Step-0 先记住 3 个文件角色

文件 作用
imx6ull.dtsi SoC 通用:声明"芯片里有 I²C1 控制器",默认 status = "disabled"
imx6ull-evk.dts 板级:把 I²C1 的时钟、引脚、状态补全
imx6ull-pinfunc.h 芯片头文件:把宏 MX6UL_PAD_UART4_TX_DATA__I2C1_SCL 翻译成寄存器偏移

Step-1 编译阶段(dtc)

复制代码
dtc -I dts -O dtb imx6ull-evk.dts -o imx6ull-evk.dtb
  • &i2c1 展开后变成 节点路径 /soc/i2c@021a0000

  • pinctrl-0 = <&pinctrl_i2c1> 变成 phandle 指向 i2c1grp

  • 宏展开(以 SCL 为例)

    MX6UL_PAD_UART4_TX_DATA__I2C1_SCL
    └─展开为 _MX6_PIN(0x01B0, 0x0460, 0x0000, ALT2, 0x4001b8b0)

得到 4 组数字:

  • MUX 偏移 0x01B0
  • PAD 偏移 0x0460
  • ALT 功能号 ALT2
  • 电气值 0x4001b8b0

Step-2 U-Boot 把 .dtb 塞进内存,启动内核

复制代码
bootz zImage - dtb_addr

内核把整块 DTB 解析成 device_node 树

复制代码
struct device_node *dn = of_find_node_by_path("/soc/i2c@021a0000");

此时 dn->properties 里已经有:

复制代码
clock-frequency = 100000
pinctrl-names   = "default"
pinctrl-0       = <phandle_to_i2c1grp>
status          = "okay"

Step-3 内核 probe 链

① pinctrl 子系统先行

复制代码
imx_pinctrl_probe()
 ├─ 解析 phandle → 找到 i2c1grp
 ├─ 对每条 fsl,pins 调用
     imx_pinctrl_set_mux()   /* 写 MUX 寄存器 0x020E0000+0x01B0 = ALT2 */
     imx_pinctrl_set_pad()   /* 写 PAD 寄存器 0x020E0000+0x0460 = 0x4001b8b0 */

结果:

  • UART4_TX_DATA 物理球I²C1_SCL 功能
  • 上拉 50 Ω、100 MHz、开漏 等电气特性生效

② I²C 驱动匹配

复制代码
static const struct of_device_id imx_i2c_dt_ids[] = {
    { .compatible = "fsl,imx6ul-i2c", },
    { .compatible = "fsl,imx21-i2c", },
    { /* sentinel */ }
};
MODULE_DEVICE_TABLE(of, imx_i2c_dt_ids);
  • i2c_imx_probe() 通过 of_match_table 匹配到 i2c@021a0000
  • clock-frequency = 100000 读出来 → 设置 I²C 波特率
  • platform_get_resource_byname() 拿到中断号、寄存器基址
  • 注册 i2c-1 总线到内核

Step-4 用户空间验证

复制代码
# 板子上电后
ls /sys/bus/i2c/devices/0-0050    # 设备节点出现
i2cdetect -y 1                    # 扫描到 EEPROM 0x50

至此,I²C1 控制器 + 两根引脚 + 外设 打通。


一句话回顾

  1. 板厂.dts 里用宏"连线"。
  2. dtc 把宏翻译成寄存器偏移 + 电气值。
  3. 内核 probe 时把寄存器写到位,I²C1 就活了。
相关推荐
珹洺26 分钟前
计算机网络:(十一)多协议标记交换 MPLS
运维·计算机网络
绵绵细雨中的乡音29 分钟前
消息队列与信号量:System V 进程间通信的基础
linux
简CC43 分钟前
Linux——文件压缩和解压
linux·运维·服务器
Stark-C1 小时前
nastools继任者?极空间部署影视自动化订阅系统『MediaMaster』
运维·自动化
cq_run1 小时前
centos9部署jdk
linux·运维·服务器
程序员小胡06192 小时前
操作系统系统面试常问(进程、线程、协程相关知识)
linux·面试·职场和发展
小张是铁粉2 小时前
linux配置nvc远程连接
linux·服务器·centos
知北游天3 小时前
Linux:多线程---深入生产消费模型&&环形队列生产消费模型
linux·缓存
阿巴~阿巴~3 小时前
Linux“一切皆文件“设计哲学 与 Linux文件抽象层:struct file与file_operations的架构解析
linux·运维·服务器
TE-茶叶蛋3 小时前
React 服务器组件 (RSC)
服务器·前端·react.js