imx6设备树pinctrl解析

article/2025/9/17 10:43:24

最近在移植linux,用到kernel版本为3.14.28,在高版本的内核源码中用到了设备树(device-tree),设备树中用到pinctrl的配置,记录一下。

1、普通设置

在配置串口时,pinctrl的配置信息如下所示:

[cpp]  view plain  copy
  1. &uart2 {  
  2.     pinctrl-names = "default";  
  3.     pinctrl-0 = <&pinctrl_uart2>;  
  4.     status = "okay";  
  5. };  
  6.   
  7. //。。。。。。。。  
  8. pinctrl_uart2: uart2grp {  
  9.             fsl,pins = <  
  10.                 MX6QDL_PAD_SD4_DAT7__UART2_TX_DATA  0x1b0b1  
  11.                 MX6QDL_PAD_SD4_DAT4__UART2_RX_DATA  0x1b0b1  
  12.             >;  
  13. };  

这里的MX6QDL_PAD_SD4_DAT7__UART2_TX_DATA在imx6dl-pinfunc.h文件中有如下定义:

[cpp]  view plain  copy
  1. MX6QDL_PAD_SD4_DAT7__UART2_TX_DATA          0x35c 0x744 0x000 0x2 0x0  

将管脚的配置展开即: 0x35c 0x744 0x000 0x2 0x00x1b0b1 

想知道这六个值都是什么意思,可以从两个路出发:①查找解读dts的文件,即看内核源码;②在网上查找相关知识。


1.1 查看源码对设备树文件的解读

首先在imx6dl-pinfunc.h文件中有对前5个变量的解释,如下图:


为了验证这5个变量,并查找第6个变量的含义,我们打开读取设备树文件的代码。

读取dts文件的文件为:drivers/pinctrl/freescale/pinctrl-imx.c,实现函数名为:static int imx_pinctrl_parse_groups(。。。),如下:

[cpp]  view plain  copy
  1. static int imx_pinctrl_parse_groups(struct device_node *np,  
  2.                     struct imx_pin_group *grp,  
  3.                     struct imx_pinctrl_soc_info *info,  
  4.                     u32 index)  
  5. {  
  6.     int size, pin_size;  
  7.     const __be32 *list;  
  8.     int i;  
  9.     u32 config;  
  10.   
  11.     dev_dbg(info->dev, "group(%d): %s\n", index, np->name);  
  12.   
  13.     if (info->flags & SHARE_MUX_CONF_REG)  
  14.         pin_size = SHARE_FSL_PIN_SIZE;  
  15.     else  
  16.         pin_size = FSL_PIN_SIZE;  
  17.     /* Initialise group */  
  18.     grp->name = np->name;  
  19.   
  20.     /* 
  21.      * the binding format is fsl,pins = <PIN_FUNC_ID CONFIG ...>, 
  22.      * do sanity check and calculate pins number 
  23.      */  
  24.     list = of_get_property(np, "fsl,pins", &size);  
  25.     if (!list) {  
  26.         dev_err(info->dev, "no fsl,pins property in node %s\n", np->full_name);  
  27.         return -EINVAL;  
  28.     }  
  29.   
  30.     /* we do not check return since it's safe node passed down */  
  31.     if (!size || size % pin_size) {  
  32.         dev_err(info->dev, "Invalid fsl,pins property in node %s\n", np->full_name);  
  33.         return -EINVAL;  
  34.     }  
  35.   
  36.     grp->npins = size / pin_size;  
  37.     grp->pins = devm_kzalloc(info->dev, grp->npins * sizeof(struct imx_pin),  
  38.                 GFP_KERNEL);  
  39.     grp->pin_ids = devm_kzalloc(info->dev, grp->npins * sizeof(unsigned int),  
  40.                 GFP_KERNEL);  
  41.     if (!grp->pins || ! grp->pin_ids)  
  42.         return -ENOMEM;  
  43.   
  44.     for (i = 0; i < grp->npins; i++) {  
  45.         u32 mux_reg = be32_to_cpu(*list++);  
  46.         u32 conf_reg;  
  47.         unsigned int pin_id;  
  48.         struct imx_pin_reg *pin_reg;  
  49.         struct imx_pin *pin = &grp->pins[i];  
  50.   
  51.         if (info->flags & SHARE_MUX_CONF_REG)  
  52.             conf_reg = mux_reg;  
  53.         else  
  54.             conf_reg = be32_to_cpu(*list++);  
  55.   
  56.         pin_id = mux_reg ? mux_reg / 4 : conf_reg / 4;  
  57.         pin_reg = &info->pin_regs[pin_id];  
  58.         pin->pin = pin_id;  
  59.         grp->pin_ids[i] = pin_id;  
  60.         pin_reg->mux_reg = mux_reg;  
  61.         pin_reg->conf_reg = conf_reg;  
  62.         pin->input_reg = be32_to_cpu(*list++);  
  63.         pin->mux_mode = be32_to_cpu(*list++);  
  64.         pin->input_val = be32_to_cpu(*list++);  
  65.   
  66.         /* SION bit is in mux register */  
  67.         config = be32_to_cpu(*list++);  
  68.         if (config & IMX_PAD_SION)  
  69.             pin->mux_mode |= IOMUXC_CONFIG_SION;  
  70.         pin->config = config & ~IMX_PAD_SION;  
  71.   
  72.         dev_dbg(info->dev, "%s: 0x%x 0x%08lx", info->pins[pin_id].name,  
  73.                 pin->mux_mode, pin->config);  
  74.     }  
  75.   
  76.     return 0;  
  77. }

这段代码中list = of_get_property(np, "fsl,pins", &size);实现了读取dts文件中的fsl,pin属性值,并保存在了list指针变量中。紧接着,分别将list中的值mux_reg、conf_reg、input_reg、mux_mode、input_val、config六个变量中,由名字可以猜测个大概,前5个得以验证,第六个表示config,config的值说白了就是对寄存器配置(上拉电阻、频率等等)的值,就是pad_ctrl的值。

因此对应关系如下:

      0x35c     |     0x744      |     0x000        |      0x2        |      0x0     | 0x1b0b1
---------------------------------------------------------------------------------------------------------
mux_ctrl_ofs  |  pad_ctrl_ofs |  sel_input_ofs |  mux_mode   | sel_input   |  pad_ctrl


以上参数在参考手册怎么确定的呢?由于是对复用管脚的配置,于是在管脚复用的章节(IOMUXC)中查找。但是现确定pad name才方便,于是定义在External Signals and Pin Multiplexing章节,搜索MX6QDL_PAD_SD4_DAT7__UART2_TX_DATA的中间部分:SD4_DAT7,如下图


可知UART2_TX_DATA是属于SD4_DAT7的ALT2,于是mux_mode=0x2即可。上图表格中最后一列SW_PAD_CTL_PAD_SD4_DATA7是config配置需要查找的名称,跳到管脚复用的章节(IOMUXC)中,找到SW_PAD_CTL_PAD_SD4_DATA7,如下所示:


如果直接取默认值的话结果是config=0x1b0b0,这里可以根据自己的需要(硬件)更改为与自己的板子匹配的值,我把最后SRE的值设置为1,即Fast Slew Rate,如下图说明:


OK,接下来是mux_ctrl_ofs、pad_ctrl_ofs、sel_input_ofs三个偏移值,这些值都是在复用管脚的章节确定的。因为pad name为SD4_DATA7,所以在找的时候可以拿它当关键字。

首先是mux_ctrl_ofs,找到IOMUXC_SW_MUX_CTL_PAD*开头的部分,结尾选择SD4_DATA7即可,如下图,


由”Address: 20E_0000h base + 35Ch offset = 20E_035Ch“中可知offset=35C,即mux_ctrl_oft=0x35c

其他的查找方法类似。pad_ctrl_ofs,查找IOMUXC_SW_PAD_CTL_PAD_SD4_DATA7一节,可知偏移值pad_ctrl_ofs=0x744



sel_input_ofs查找IOMUXC章节以SELECT_INPUT结尾的部分,中间选择UART2_TX,如果没有这里sel_input_ofs=0x000即可,对应的sel_input为0即可。如果有例如IOMUXC_UART2_UART_RX_DATA_SELECT_INPUT,即uart的rx管脚配置,如下图,所以RXsel_input_ofs=0x904,这里选择对应的值“110 SD4_DATA4_ALT2 — Selecting ALT2 mode of pad SD4_DAT4 for UART2_RX_DATA..“所以RXMX6QDL_PAD_SD4_DAT4__UART2_RX_DATA)的sel_input=0x6。



首先还是先看代码,看看到底特殊到哪里。

[cpp]  view plain  copy
  1.         pinctrl_gpio_leds: gpioledsgrp {  
  2.             fsl,pins = <  
  3.                 MX6QDL_PAD_DISP0_DAT21__GPIO5_IO15  0x80000000  
  4.             >;  
  5.         };  
  6.   
  7.         pinctrl_i2c2: i2c2grp {  
  8.             fsl,pins = <  
  9.                 MX6QDL_PAD_EIM_EB2__I2C2_SCL    0x4001b8b1  
  10.                 MX6QDL_PAD_KEY_ROW3__I2C2_SDA   0x4001b8b1  
  11.             >;  
  12.         };

可以看出来特殊的配置就是后面的值也就是上一篇讲的config(pad_ctrl)的值改变了,变为0x80000000和0x4001b8b1了,当我们查找相应的pad值时是这样的:



这明显不和常理,在上图中显示高15位全部置0,取值也没啥用,那么为什么设置为0x80000000和0x4001b8b1呢?在网上搜罗一番没有任何有帮助的文档,只能靠自己了。还是老思路,查找设备树文件的读取源码,drivers/pinctrl/freescale/pinctrl-imx.c中,找到了惊喜!!!代码如下

[cpp]  view plain  copy
  1. /* The bits in CONFIG cell defined in binding doc*/  
  2. #define IMX_NO_PAD_CTL  0x80000000  /* no pin config need */  
  3. #define IMX_PAD_SION 0x40000000     /* set SION */</span>  
再将IMX_NO_PAD_CTL使用部分的代码贴上(随便找一处)

[cpp]  view plain  copy
  1.     for (i = j = 0; i < grp->npins; i++) {  
  2.         if (!(grp->pins[i].config & IMX_NO_PAD_CTL)) {  
  3.             new_map[j].type = PIN_MAP_TYPE_CONFIGS_PIN;  
  4.             new_map[j].data.configs.group_or_pin =  
  5.                     pin_get_name(pctldev, grp->pins[i].pin);  
  6.             new_map[j].data.configs.configs = &grp->pins[i].config;  
  7.             new_map[j].data.configs.num_configs = 1;  
  8.             j++;  
  9.         }  
  10.     }
可以看出来确实如注释(/* no pin config need */)所述,表示该管脚的配置config(pad_ctrl)无效,或者说不需要。

同理0x40000000表示设置了SION。但是0x4001b8b1表示什么意思呢,从上一个注释(/* The bits in CONFIG cell defined in binding doc*/)可以找到方向,即取binding doc中找,所以打开Documentation/devicetree/bindings/pinctrl目录下的fsl,imx6dl-pinctrl.txt文件,里面有关于SION的介绍,如下:


再从芯片的参考手册中查阅可知,SION就相当于一个标志为(第30位),去掉这一位后config=0x1b8b1,这个值就是从pad_ctrl一节找到的,具体可以参见第6个参数的确定方法。




http://chatgpt.dhexx.cn/article/fwpB2V5h.shtml

相关文章

IMX6UL核心板

CPU:IMX6UL DDR:256M EMMC:4G NANDFLASH:1G 分立DCDC 需定制请联系QQ402979209 广州瀚忆科技有限公司 http://www.gzhysmart.com/

imx6 LCD 参数配置(lvds为例)

目前imx6的BSP开发中&#xff0c;lvds的参数配置一般在两个地方&#xff1a; 1. uboot的CMDLINE的参数设置&#xff0c;形如&#xff1a; videomxcfb0:devldb,bpp32 2. uboot板级代码中对struct display_info_t的配置&#xff0c;形如&#xff1a; static struct display_i…

IMX6U主频详解和配置

IMX6U默认396Mhz主频 可以达到528Mhz 一、时钟来源及组成 1、时钟的来源都是晶振 32.768khz的晶振 给RTC使用 24MHz的晶振给系统来使用 接在T16和T17这两个IO上 2、芯片时钟构成 七路PLL 全都由24Mhz晶振产生 使用这些的原因是为了方便外设使用 倍频产生 ①ARM PLL供给AR…

iMX6平台MAX9288调试

因为有之前Ti ds90ub964的调试经验&#xff0c;这次调9288只是当做一个常规的驱动调试&#xff0c;坑大也得到了宝贵的经验 9288&#xff08;GMSL--->MIPI CSI2&#xff09;解码器解码9291和9277编码器过来的gmsl信号 iMX6DQ的mipi csi接口寄存器控制的特性&#xff1a; …

【imx6ul】从头搭建imx6ul开发环境(uboot、内核编译及烧入、mfgtools详细使用方法)

实现一个自制的Bootloader见: 01-基于imx6ul从0自制Bootloader专栏实现总结_【星星之火】的博客-CSDN博客 硬件平台&#xff1a;自制imx6ul&#xff08;详见【ALLEGRO】从零制作imx6-PCB核心板全流程_【星星之火】的博客-CSDN博客&#xff09;系统&#xff1a;linux-4.1.15文件…

飞思卡尔IMX6处理器的GPIO配置方式

在linux或android系统中&#xff0c;假如我们要配置飞思卡尔IMX6处理器的GPIO管脚&#xff0c;比如是GPIO_19这个管脚&#xff0c;那么要像这样&#xff1a; #define MX6Q_PAD_GPIO_19__GPIO_4_5 \(_MX6Q_PAD_GPIO_19__GPIO_4_5 | MUX_PAD_CTRL(NO_PAD_CTRL))其中_MX6Q_PAD_…

迅为IMX6Q四核核心板商业级|工业级|IMX6Plus版本|IMX6D双核核心板

迅为IMX6Q核心板工业级/Plus版本 IMX6Q处理器&#xff1a;兼容单核&#xff0c;双核&#xff0c;工业级&#xff0c;汽车级&#xff0c;IMX6Q最新Plus版本&#xff0c;共用同一底板&#xff0c;高端产品无忧。 i.MX6系列针对消费电子、工业控制和汽车应用领域&#xff0c;它将…

imx6ull驱动开发经验

1、背景 imx6ull驱动开发基于正点原子的开发板&#xff0c;上面运行linux 4.1.15内核&#xff0c;根文件系统为ubuntu 16.05.5 LTS. 2、加载驱动文件chrdevbase.ko文件时&#xff0c;先使用depmod生成依赖文件时&#xff0c;提示无modules.order, modules.builtin无这两个文件…

【盘点】imx6的应用

导读: i.mx6系列应用处理器因其高性能&#xff0c;低功耗以及令人瞩目的多媒体功能等特性&#xff0c;被广泛运用于消费电子、工业以及汽车车载娱乐系统等新应用中。 今天为大家盘点一下启扬i.MX6嵌入式主板的部分实际应用。 以智能、多媒体为中心、基于触摸的设备市场正在迅速…

imx6 usb增强信号强度

USB信号 参考&#xff1a;官方文档 USB信号完整性取决于许多因素&#xff0c;如电路设计、PCB布局、堆叠和阻抗。每个产品可能彼此不同&#xff0c;因此客户需要微调参数&#xff0c;以获得最佳的信号质量。 测试板已经路由出两个USB端口:一个OTG1&#xff0c;一个主机。每个端…

iMX6ULL上手体验

CSDN仅用于增加百度收录权重&#xff0c;排版未优化&#xff0c;日常不维护。请访问&#xff1a;www.hceng.cn 查看、评论。 本博文对应地址: https://hceng.cn/2018/03/28/iMX6ULL上手体验/#more 第一次接触NXP/Freescale的SOC&#xff0c;记录拿到板子后快速上手的技巧和思维…

imx6ul 驱动详解

链表的知识&#xff1a; struct list_head { struct list_head *next, *prev; };API函数 函数 功能LIST_HEAD声明并初始化双向链表。INIT_LIST_HEAD初始化双向链表。list_add在链表头head节点后面插入一个新的节点new。list_add_tail在链表末尾tail节点后面插入一个新的节点n…

IMX6 / IMX8 系列芯片简单对比

imx6有&#xff1a; imx8 分为&#xff1a; IMX8/IMX8M/IMX8X三个系列 IMX8有&#xff1a;iMX 8qm 和 IMX8-QP IMX8M有&#xff1a;IMX8mn 、 IMX8mm 、 IMX8mq 、IMX8mp 3.IMX8X 有&#xff1a;iMX8dxp / iMX8qxp 和 iMX8dx

IMX6启动流程

目录 1. 流程图 2. u-boot.imx的构造 1. 流程图 正常情况下&#xff0c;一台IMX6设备的启动路径就是图中自上而下最左边那条路&#xff1a; 启动/复位后先检查CPU的ID(每颗芯片独一无二) ------> 检查复位状态(有没有按下复位键) ------> 检查boot模式(mmc? USB? SPI…

ARM恩智浦IMX6系列-产品开发无忧-四种主控核心

ITOP-IMX6 开发平台是基于 NXP 的 IMX6 系类的芯片开发的一款产品&#xff0c;CPU 采用 Cortex-A9 内核&#xff0c;主频达1Ghz。 ITOP-IMX6 开发板平台均由核心板和底板构成&#xff0c;核心板的主要芯片有 CPU,内存&#xff0c;EMMC&#xff0c;电源管理&#xff0c;EEPROM 等…

关于NXP公司的IMX6系列芯片参考资料体系介绍

IMX6系列芯片由FreeScale公司基于ARM核设计开发&#xff0c;该公司于2015年被NXP公司收购。之前由FreeScale提供支持的资源被整合进NXP支持体系&#xff0c;资料内部FreeScale和NXP的标记都有&#xff0c;文档内部原有资源链接失效&#xff0c;给人混乱感。加之IMX6系列芯片本身…

伺服电机矢量控制原理与仿真(1)控制系统的建立

自动控制原理综合实践 直流伺服电机及其控制为什么是六步换向法类比有刷直流电机类比三相异步电机这些差异意味着什么 从六步换向到FOCFOC的大体流程FOC的目的电流环的执行器&#xff1a;SVPWM 从物理走向数学直流伺服电机的数学模型&#xff08;正弦式&#xff09;从数学模型到…

台达PLC模拟量或台达伺服电机控制模块程序案例

台达PLC模拟量或台达伺服电机控制模块程序案例&#xff0c;二选一 ID:6958618562901252Anh13309639501 以下内容不属于本博内容&#xff0c;仅仅是相关参考&#xff1a; 台达PLC采用可以编制程序的存储器&#xff0c;用来在其部存储执行逻辑运算、顺序运算、计时、计数和算术运…

使用Arduino开发板控制伺服电机

大多数情况下&#xff0c;直流电机控制的控制方法都是非常简单的&#xff08;只需要连接电源和地&#xff09;。伺服电机有一些不同&#xff0c;它使用3根线连接&#xff08;电源、地和信号&#xff09;将电机移动到某个旋转位置。该位置由信号线上发送的信号决定。一旦电机到达…

步进伺服电机控制程序: 1.实现步进电机的正转、反转控制 2.相对和绝对位置运动

步进伺服电机控制程序: 1.实现步进电机的正转、反转控制 2.相对和绝对位置运动 3.电机复位找原点功能 4.显示电机的当前位置 硬件:1西门子S7-200PLC 2.昆仑通泰MCGS触摸屏 3.步进电机驱动器(不限制厂家) 西门子200步进电机控制 4810625077520119gao699592