imx6设备树pinctrl解析_imx6ull pinctrl内核怎么用-程序员宅基地

技术标签: linux系统  

最近在移植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个参数的确定方法。



版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/michaelcao1980/article/details/50730421

智能推荐

Android ViewPager Fragment 切换刷新数据,解决生命周期只走一次的问题_android fragment viewpager 切换刷新-程序员宅基地

文章浏览阅读1.6w次,点赞6次,收藏17次。前言Activity的生命周期都应该熟悉,Fragment其标准生命周期也仅比Activity多出一些流程,如onCreateView(),比较好理解。了解Activity、Fragment的基础知识可以先转头去看下我的博文:Android Fragment项目中使用及生命周期分析认识ViewPager页面加载机制ViewPager的页面缓存机制默认是两个,如果有三个Fragment,实际上三个都..._android fragment viewpager 切换刷新

SDL2系列教程4-事件处理_sdl2 c4-程序员宅基地

文章浏览阅读3k次。事件循环大多数多媒体程序依靠事件系统来处理输入。SDL为处理输入事件提供了灵活的API。本质上,SDL将来自设备(如键盘,鼠标或控制器)的输入记录为事件,将它们存储在“事件队列”中。您可以将此结构视为等待线 - 事件在线的后面排队并从线的前面取出。在您的程序中,您将始终拥有一个事件(或“游戏”或“主”)循环来处理这些事件并根据输入运行您的程序。每次运行事件循环时,必须从事件队列中拉出每..._sdl2 c4

Java面试题原理和底层-程序员宅基地

文章浏览阅读799次,点赞21次,收藏27次。可能有人会问我为什么愿意去花时间帮助大家实现求职梦想,因为我一直坚信时间是可以复制的。我牺牲了自己的大概十个小时写了这片文章,换来的是成千上万的求职者节约几天甚至几周时间浪费在无用的资源上。上面的这些(算法与数据结构)+(Java多线程学习手册)+(计算机网络顶级教程)等学习资源《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取![外链图片转存中…(img-UTZkDIID-1712634944586)]

SpringBoot整合ActiveMQ实现Queue和Topic两种模式-程序员宅基地

文章浏览阅读385次,点赞5次,收藏5次。关于分布式,限流+缓存+缓存,这三大技术(包含:ZooKeeper+Nginx+MongoDB+memcached+Redis+ActiveMQ+Kafka+RabbitMQ)等等。这些相关的面试也好,还有手写以及学习的笔记PDF,都是啃透分布式技术必不可少的宝藏。以上的每一个专题每一个小分类都有相关的介绍,并且小编也已经将其整理成PDF啦《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!65)]

【附带Java采用JDBC连接数据库】SQLServer2012的安装与注意事项详解-程序员宅基地

文章浏览阅读849次,点赞20次,收藏18次。各位读者,由于本篇幅度过长,为了避免影响阅读体验,下面我就大概概括了整理了《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!各位读者,由于本篇幅度过长,为了避免影响阅读体验,下面我就大概概括了整理了[外链图片转存中…(img-b9ox4iQ7-1713427035624)][外链图片转存中…(img-YTGkAqAk-1713427035624)][外链图片转存中…(img-3NR7iBZS-1713427035624)]

配置OpenCV3+VS2017发生错误:无法打开文件 opencv_world345d.dll 解决方案_无法打开文件“opencv_world345d.lib”-程序员宅基地

文章浏览阅读9.1k次,点赞10次,收藏8次。配置OpenCV3+VS2017的步骤:Visual Studio 2017 + OpenCV 3.4.1 全新安装与配置及可能遇到的问题解决但是我在按照该流程配置完成以后却报了错误:LNK1104 无法打开文件"opencv_world345d.dll"原因是我们在进行配置时错把".lib"写成了".dll"我的解决方案是:在属性管理器界面配置"Microsoft.Cpp.x64.u..._无法打开文件“opencv_world345d.lib”

随便推点

【附源码】JAVA计算机毕业设计远景民宿酒店预订小程序(springboot+mysql+开题+论文)-程序员宅基地

文章浏览阅读1k次,点赞28次,收藏20次。最后,小程序的推广和使用,还能带动相关产业的发展,如旅游、餐饮等,为地方经济注入新的活力。针对这些问题,我们将通过深入研究用户需求和市场现状,设计合理的系统架构和功能模块,采用先进的开发技术和安全策略,确保小程序能够满足用户需求并保障交易安全。首先,进行市场调研和用户需求分析,通过问卷调查、访谈等方式收集用户对于民宿预订小程序的期望和需求,为系统设计和功能开发提供依据。首先,开发出一款功能全面、操作便捷的远景民宿酒店预订小程序,满足用户对民宿预订的多样化需求,提升民宿行业的服务质量和用户体验。

pandas按条件筛选数据_ilocation based boolean indexing cannot use an ind-程序员宅基地

文章浏览阅读5.8w次,点赞20次,收藏126次。pandas中对DataFrame筛选数据的方法有很多的,以后会后续进行补充,这里只整理遇到错误的情况。1.使用布尔型DataFrame对数据进行筛选使用一个条件对数据进行筛选,代码类似如下:num_red=flags[flags['red']==1]使用多个条件对数据进行筛选,代码类似如下:stripes_or_bars=flags[(flags['stripes']&g..._ilocation based boolean indexing cannot use an indexable as a mask

讯飞离线命令词识别_科大讯飞离线的sdk可以用多长时间-程序员宅基地

文章浏览阅读2.3k次。可以借鉴这几个博客:https://blog.csdn.net/hfut_why/article/details/82117670https://blog.csdn.net/u013834525/article/details/80097253目前,讯飞的SDK是有90天试用时间,装机量10个,足够尝试了。大概分为几步:注册账号 创建相应设备(可以得到一个APPID,后面调用SDK需要使用) 根据相应平台下载官方Demo 找到Samples文件夹,可以查看Demo代码,编译并运行_科大讯飞离线的sdk可以用多长时间

复制构造函数调用关闭GCC编译器优化_c语言禁止编译器优化函数复制构造函数的调用情况-程序员宅基地

文章浏览阅读1.5k次,点赞3次,收藏3次。在linux下,编译器有时会对复制构造函数的调用做优化,避免不必要的复制构造函数调用。可以使用命令g++ xxx.cpp -fno-elide-constructors命令关闭编译器优化。例如下面代码的执行结果:#include <iostream>using namespace std;class Point{public: Point(int xx = 0,..._c语言禁止编译器优化函数复制构造函数的调用情况

在cmd的命令下 用Mysql中建立一个表_用cmd做表格-程序员宅基地

文章浏览阅读5.2k次。1, mysql -uroot -pxxxx //登陆数据库2, show databases; //查看数据库,例如要在test库中建表3, use test;//进入test4, create table users( //例如要建立users表,下面是建立字段5, id int(10) unsigned not null auto_inc_用cmd做表格

2024 Android开发项目开发必备底层技术知识点整理,面试了几家公司都被拒绝了-程序员宅基地

文章浏览阅读725次,点赞10次,收藏20次。理解RTTI在Java中的工作原理,首先需要知道类型信息在运行时是如何表示的,这是由Class对象来完成的,它包含了与类有关的信息。Class对象就是用来创建所有“常规”对象的,Java使用Class对象来执行RTTI,即使你正在执行的是类似类型转换这样的操作。只要是程序员,不管是Java还是Android,如果不去阅读源码,只看API文档,那就只是停留于皮毛,这对我们知识体系的建立和完备以及实战技术的提升都是不利的。真正最能锻炼能力的便是直接去阅读源码,不仅限于阅读各大系统源码,还包括各种优秀的开源库。