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

Linux 驱动开发基础(3):pinctrl 子系统

Linux 驱动开发基础详解(3):pinctrl 子系统

前情回顾:在上一节的学习中,我们编写了基于设备树的 LED 驱动。但大家可能会发现,我们依然像裸机开发那样,通过直接获取并操作 GPIO 相关的底层寄存器来实现驱动。

然而,Linux 是一个庞大而高度抽象的系统,这种“原始”的寄存器硬编码方式不仅繁琐,而且违背了 Linux 驱动“分离与分层”的设计思想。为此,Linux 内核引入了pinctrlgpio子系统。本篇将聚焦于:pinctrl子系统


01 学习重点

在掌握了设备树基本语法后,本节我们需要搞清以下两个核心问题:

  1. 为什么需要 pinctrl 子系统?它到底负责干什么?
  2. 如何在设备树(Device Tree)中正确使用和配置 pinctrl?

02 告别裸机,引入 pinctrl?

传统的“裸机式”配置方式

回顾上一节 LED-GPIO 的初始化过程,我们通常需要做三件事:

  1. 查地址:在设备树节点中设置reg属性,填入相关的寄存器物理地址。
  2. 配引脚(PIN):初始化复用(MUX)寄存器和电气属性(PAD)寄存器,设置引脚的复用功能(如设为 GPIO)、上下拉速度等。
  3. 配 GPIO:初始化 GPIO 控制寄存器,设置输入/输出方向及电高等。

对于绝大多数 32 位 SOC 而言,引脚支持多路复用(Multiplexing)。如果每个驱动都自己去查手册、算地址、配寄存器,代码将极度冗余且难以维护。

pinctrl 子系统的使命

为了将开发者从繁琐的寄存器配置中解放出来,Linux 推出了pinctrl子系统。它的核心工作内容只有三个:

  1. 获取 PIN 信息:从设备树中读取引脚配置。
  2. 设置复用功能(MUX):根据信息,将引脚配置为特定的功能(如 I2C、SPI、GPIO)。
  3. 设置电气特性(PAD):配置引脚的上/下拉、驱动能力、速度等。

💡优势:引入pinctrl后,驱动开发者只需要在设备树中按规则写好 PIN 的属性,底层的寄存器读写全由内核pinctrl驱动自动配置!


03 实战:在设备树中配置 pinctrl

我们以 NXP 的I.MX6ULL芯片为例,实现在设备树(.dtsi文件)中配置 pinctrl。

1. 找到 IOMUXC 节点

在 I.MX 系列芯片中,引脚由IOMUXC外设管理。在设备树中,我们需要找到对应的iomuxc节点:

iomuxc: iomuxc@020e0000 { /* 内核会根据 compatible 匹配对应的 pinctrl 驱动文件 */ compatible = "fsl,imx6ul-iomuxc"; /* IOMUXC 外设的寄存器基地址为 0x020e0000,长度为 0x4000 */ reg = <0x020e0000 0x4000>; };

2. 追加 PIN 配置信息

我们需要配置UART1_RTS_B这个引脚,将其复用为GPIO1_IO19。我们需要在iomuxc节点下添加特定的子节点:

&iomuxc { pinctrl-names = "default"; pinctrl-0 = <&pinctrl_hog_1>; imx6ul-evk { /* 自定义的节点名称,常以驱动名或功能命名 */ pinctrl_hog_1: hoggrp-1 { /* fsl,pins 是核心属性,里面存放具体的引脚配置 */ fsl,pins = < /* 格式:<宏定义(复用配置) 自定义数值(电气特性配置)> */ MX6UL_PAD_UART1_RTS_B__GPIO1_IO19 0x17059 MX6UL_PAD_GPIO1_IO05__USDHC1_VSELECT 0x17059 MX6UL_PAD_GPIO1_IO09__GPIO1_IO09 0x17059 MX6UL_PAD_GPIO1_IO00__ANATOP_OTG1_ID 0x13058 >; }; }; };

⚠️注意解析
fsl,pins属性中,每一行代表一个引脚的配置,由两部分组成:

  • 前半部分(宏)MX6UL_PAD_UART1_RTS_B__GPIO1_IO19,决定引脚的复用功能
  • 后半部分(数值)0x17059,决定引脚的电气特性(由用户查手册自行计算填写,用于配置上/下拉、驱动能力等)。

04 深度解析:宏定义

MX6UL_PAD_UART1_RTS_B__GPIO1_IO19这么长的宏,到底代表了什么呢?
设备树是支持 C 语言#include语法的,这个宏定义在内核的.h头文件中(如imx6ul-pinfunc.h)。

宏定义源码

在头文件,关于UART1_RTS_B的复用宏多达 8 个(代表它可以复用为 8 种不同的功能):

/* 格式: <mux_reg conf_reg input_reg mux_mode input_val> */#defineMX6UL_PAD_UART1_RTS_B__UART1_DCE_RTS0x00900x031C0x06200x00x3#defineMX6UL_PAD_UART1_RTS_B__UART1_DTE_CTS0x00900x031C0x00000x00x0#defineMX6UL_PAD_UART1_RTS_B__ENET1_TX_ER0x00900x031C0x00000x10x0#defineMX6UL_PAD_UART1_RTS_B__USDHC1_CD_B0x00900x031C0x06680x20x1#defineMX6UL_PAD_UART1_RTS_B__CSI_DATA050x00900x031C0x04CC0x30x1#defineMX6UL_PAD_UART1_RTS_B__ENET2_1588_EVENT1_OUT0x00900x031C0x00000x40x0/* ⚠️这是我们用到的宏,将其复用为 GPIO1_IO19 ⚠️ */#defineMX6UL_PAD_UART1_RTS_B__GPIO1_IO190x00900x031C0x00000x50x0#defineMX6UL_PAD_UART1_RTS_B__USDHC2_CD_B0x00900x031C0x06740x80x2

5个十六进制数

宏定义后面跟着的0x0090 0x031C 0x0000 0x5 0x0这 5 个数字,正是内核pinctrl驱动用来操作底层寄存器的方式。

它们对应<mux_reg conf_reg input_reg mux_mode input_val>,具体含义如下表:

参数本例数值含义说明实际物理地址计算 (基地址0x020e0000)
mux_reg0x0090复用寄存器偏移地址0x020e0000 + 0x0090 = 0x020e0090
(指向 MUX_CTL 寄存器)
conf_reg0x031C电气特性寄存器偏移地址0x020e0000 + 0x031C = 0x020e031C
(指向 PAD_CTL 寄存器)
input_reg0x0000输入寄存器偏移地址有些外设没有此寄存器(如做 GPIO 时),故填 0 无效。
mux_mode0x5复用寄存器(mux_reg)要写入的值将底层寄存器0x020e0090的值配置为0x5,即可复用为 GPIO。
input_val0x0输入寄存器要写入的值在本例中无效。

配置完成

现在,结合我们在设备树中写的那行代码:
MX6UL_PAD_UART1_RTS_B__GPIO1_IO19 0x17059

相当于我们向pinctrl子系统下达了如下指令:

“把偏移地址为0x0090的寄存器值写为0x5(搞定复用);然后把偏移地址为0x031C的寄存器值写为0x17059(搞定电气特性)。”

底层的fsl,imx6ul-pinctrl驱动拿到这些参数后,会自动完成所有寄存器的映射和写入,完全不需要我们在驱动 C 代码中写一行ioremap映射!


05 小结

通过本节学习,我们掌握了 pinctrl 子系统的工作原理。后续驱动开发中,配置引脚的标准方式:

步骤操作内容方式
1打开内核提供的引脚复用头文件(.h),找到目标引脚的复用宏定义查阅源码
2查阅 SOC 参考手册,计算出需要的电气特性值(如0x17059)。查阅手册
3在设备树(.dts)的iomuxc节点下新建子节点,在fsl,pins中填入<宏定义 电气特性值>设备树修改
4驱动程序加载时,内核 pinctrl 自动接管并完成寄存器初始化。内核自动处理

搞定了引脚的复用和上下拉(pinctrl),这只是第一步。下一节,我们将学习如何使用GPIO 子系统来真正控制这个被配置好的引脚(输出高低电平或读取输入状态)。

http://www.jsqmd.com/news/526711/

相关文章:

  • Linux System V标准简介
  • MedGemma-X企业应用:构建放射科AI能力中台,统一调度多模型服务
  • 企业数据架构、应用架构、技术架构设计方案(PPT文件)
  • Raptor子程序应用:区间数字和统计的5种高效实现方案对比
  • 工业4.0会取代精益生产吗?看懂两者关系,企业才不会走错路
  • 【STM32实战】机械臂快递分拣系统(三)——云端交互与远程控制实现
  • 工业物联网实时分析卡脖子?DolphinDB用两大核心能力破局,筑牢国产时序数据底座
  • Phi-4-mini-reasoning推理链可视化|ollama+LangChain实现思维过程可追溯
  • 重磅升级| G5501 SDK内核从5.10全面升级至6.1 LTS
  • Qt导航栏组件C05:抽屉式侧边栏
  • 多旋翼无人机系统组成(三)(动力系统详解)
  • 希尔排序:从原理到代码
  • 零门槛掌握网络安全数据处理:CyberChef全指南
  • StructBERT零样本分类-中文-base开源镜像部署:低成本GPU显存优化方案(<3GB)
  • Qwen3-TTS创意玩法:克隆声音制作多语种短视频、个性化语音助手
  • kotlin:函数式参数
  • OpCore-Simplify:当黑苹果遇上智能决策,传统配置的终结者
  • SpringBoot 内置服务器(Tomcat/Jetty/Undertow)切换
  • 单相桥式全控整流电路在电力电子技术中的应用与Simulink仿真分析
  • CoPaw模型赋能数字人:驱动虚拟角色生成动态对话与表情
  • 用Python自动生成Verilog Testbench?这5个脚本让仿真效率提升300%
  • 企业级网络安全深度解析:从协议层到云原生的攻防实战与架构设计
  • SuperGrok 额度管理全攻略:从查看剩余到永久省额度,一文搞定(附带高ROI Prompt 模板)
  • 读2025世界前沿技术发展报告21载人航天和深太空探索
  • MusePublic艺术创作引擎算法优化:提升艺术生成效率
  • SpringAiAlibaba使用模型出现404报错问题
  • IDEA 新建web工程实战
  • 第一次降AI率不知道用什么?比话可能是最适合新手的选择
  • Qwen2.5-VL-7B-Instruct图文对话:支持多轮图片上下文+历史记忆回溯
  • AI原生城市服务平台:不是加个AI插件,而是城市服务的“原生进化”