当前位置: 首页 > news >正文

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

下面用 “一段真实板级 .dts 代码”
“i2c1 节点 + pinctrl-0 从设备树 → 内核 → 寄存器” 的 全流程 逐行拆开讲清。

&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 0x4001b8b0MX6UL_PAD_UART4_RX_DATA__I2C1_SDA 0x4001b8b0>;};
};

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

文件作用
imx6ull.dtsiSoC 通用:声明“芯片里有 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 就活了。
http://www.dtcms.com/a/287643.html

相关文章:

  • 【取消分仓-分布式锁】
  • PCIe RAS学习专题(3):AER内核处理流程梳理
  • windows docker-03-如何一步步学习 docker
  • VSCode用Python操作MySQL:环境配置与代码验证
  • CentOS 清理技巧
  • 音视频学习(四十):H264码流结构
  • Tailwind CSS中设定宽度和高度的方法
  • ubuntu下好用的录屏工具
  • TCP 和 UDP 在创建套接字(Socket)时的区别
  • Claude Code 最新详细安装教程
  • 在 Solidity 中,abi是啥
  • day11 ADC
  • [spring6: AspectMetadata AspectInstanceFactory]-源码解析
  • 【Unity】YooAsset问题记录
  • 深度学习-线性神经网络
  • 剧本杀小程序开发:科技赋能,重塑推理娱乐新形态
  • 大模型军备竞赛升级!Grok 4 携 “多智能体内生化” 破局,重构 AI 算力与 Agent 2.0 时代
  • 1 渗透基础
  • FOC算法六步算法 以及 Vds保护是什么
  • 石子问题(区间dp)
  • 【c++】提升用户体验:问答系统的交互优化实践——关于我用AI编写了一个聊天机器人……(12)
  • QCC系列显示交互层的自研技术突破与实践
  • 论文reading学习记录4 - weekly - 视觉端到端开创-LOAM
  • 6 STM32单片机的智能家居安防系统设计(STM32代码+手机APP设计+PCB设计+Proteus仿真)
  • VRRP-虚拟路由冗余协议
  • leetcode3_435 and 605
  • 在服务器(ECS)部署 MySQL 操作流程
  • C++控制台贪吃蛇开发:从0到1绘制游戏世界
  • Linux --进程信号
  • 2025年燃气从业人员证考试题库及答案