【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 就活了。
相关推荐
朽棘不雕15 分钟前
Linux工具(上)
linux·运维·服务器
daad77731 分钟前
bitcoin HD钱包示例 真实使命7
运维·服务器
BestOrNothing_201538 分钟前
Ubuntu 22.04 下调整 VS Code 界面及字体教程
linux·vscode·ubuntu22.04·界面调整
Zero-Talent1 小时前
TCP/IP协议
运维·服务器·网络
桌面运维家1 小时前
Windows/Linux云桌面:高校VDisk方案部署指南
linux·运维·windows
Du_chong_huan1 小时前
1.7 计算机网络和因特网的历史 | 《计算机网络:自顶向下方法》精读版
运维·服务器·网络
ZZZKKKRTSAE1 小时前
rhel9快速上手Docker
运维·docker·容器
筱顾大牛1 小时前
Docker安装教程(加汉化!超详细!!!)
运维·docker·容器
没头脑的男大1 小时前
关于tailscale和ssh那些事儿
运维·服务器·ssh
mzhan0171 小时前
Linux:intel:Cache Allocation tech
linux·cpu