rk3568 | rk平台GPIO冲突检测小技巧

news2024/11/24 6:03:30

上一篇我们讲解了如何编写gpio驱动,但是实际操作中,经常发现gpio引脚被占用的情况发生,那么本篇文章就详细讲解rxw平台下如何快速定位gpio复用问题以及如何解决。

一、GPIO寄存器查找

要想查看某个GPIO引脚可以配置的功能以及地址信息,需要查看TRM手册:

《Rockchip RK3568 TRM Part1》
  • 第一步:
    对于GPIO2 A2,我们转换成下面字符串然后搜索
gpio2a2_sel


这样我们就可以直接找到该引脚iomux配置寄存器,bit[10:8]。
该寄存器地址:

基地址+0x0020

那么如何找到基地址呢?

  • 第二步:

搜索该引脚寄存器的名字:

GRF_GPIO2A_IOMUX_L

注意向上搜索


可以得到该寄存器基地址名称:

SYS_GRF
  • 第三步
    直接进入chapter 3


可得SYS_GRF 基地址

0xFDC60000

那么我们就得到了GPIO2 A2的IOMUX配置寄存器地址为

0xFDC60000 + 0x0020

该寄存器的bit[10:8]用于设置该寄存器功能。

当然也可以用下图来查找,更加方便:

那么找到了这个寄存器地址,我要如何来直接读取这个寄存器的值呢?

这个可以借助于一个命令io。

二、io命令

io命令可以直接操作某个寄存器,用于查看设置某个PIO 引脚配置了什么iomux,非常方便。

1. 移植io命令需要的驱动

RK 的 Android 平台,默认有包含 io 工具(源码位置: external\io), linux 系统平台如果没有此源码, 可以将 Android 平台此源码打包过去编译即可( linux 平台代码同步最新都已带有 IO 工具,可直接使用命令)。

  • 第一步:修改Makefile、Kconfig
vim  drivers/char/Makefile

增加
+obj-$(CONFIG_DEVMEM)           += mem.o
vim drivers/char/Kconfig
改文件已经包含下面信息
 10 config DEVMEM
 11     bool "/dev/mem virtual device support"
 12     default y
 13     help
 14       Say Y here if you want to support the /dev/mem device.
 15       The /dev/mem device is used to access areas of physical
 16       memory.
 17       When in doubt, say "Y".

  • 第二步:修改驱动文件

io命令需要内核驱动支持,驱动文件如下:

drivers/char/mem.c

找到下面代码:

#ifdef CONFIG_DEVMEM
         [1] = { "mem", 0, &mem_fops, FMODE_UNSIGNED_OFFSET },
#endif

修改为


+//#ifdef CONFIG_DEVMEM
         [1] = { "mem", 0, &mem_fops, FMODE_UNSIGNED_OFFSET },
+//#endif
  • 第三步:修改配置文件rockchip_defconfig
vim  arch/arm64/configs/rockchip_defconfig
增加
+CONFIG_DEVMEM=y

重新编译内核,烧录重启。

rockchip_defconfig需要根据平台选择

  • 第四步:查看mem字符设备:
rk3568_r:/ # ls /dev/mem -l
ls /dev/mem -l
crw------- 1 media media 1,   1 2017-08-04 09:00 /dev/mem

io命令正是通过这个字符设备来实现寄存器的读写的。

2. 命令说明

rk3568_r:/dev # io
io
Raw memory i/o utility - $Revision: 1.5 $

io -v -1|2|4 -r|w [-l <len>] [-f <file>] <addr> [<value>]

    -v         Verbose, asks for confirmation
    -1|2|4     Sets memory access size in bytes (default byte)
    -l <len>   Length in bytes of area to access (defaults to
               one access, or whole file length)
    -r|w       Read from or Write to memory (default read)
    -f <file>  File to write on memory read, or
               to read on memory write
    <addr>     The memory address to access
    <val>      The value to write (implies -w)

Examples:
    io 0x1000                  Reads one byte from 0x1000
    io 0x1000 0x12             Writes 0x12 to location 0x1000
    io -2 -l 8 0x1000          Reads 8 words from 0x1000
    io -r -f dmp -l 100 200    Reads 100 bytes from addr 200 to file
    io -w -f img 0x10000       Writes the whole of file to memory

Note access size (-1|2|4) does not apply to file based accesses.

3. 举例1,通过IO读寄存器:

读取gpio2 A2引脚配置寄存器

io -4 -l 0x30 0xFDC60000
-4         : 按字(4个字节) 访问内存 
0x30       : 读取0x30(48)个字节
0xFDC60000 : 基地址

由上图可得0xFDC60020的bit[10:8]值为1

所以该io引脚被设置的iomux为

3'h1: SDMMC0_CLK

如果我们是希望通过设备树设置该引脚为普通gpio,那么该值应该为0,

那么这就说明了,我们设置失败了,

那就可以到设备树中查找设置SDMMC0的地方,将其注释掉。

要读取单个寄存器,可以用下面命令:

 io -4 -r 0xFDC60024

4. 举例2,通过IO写寄存器

比如已经通过命令: io -4 -r 0xFDC60024读出了寄存器的值,那么此时想把gpio2 A2修改为普通GPIO,
那么只需要对0xFDC60024这个寄存器的第 3 个 1 写 0,那么可以如下操作:

io -4 –w 0xFDC60024 0x01002011

注意:

  • 第三个1对应是bit[10:8]
  • 通过 io 写的寄存器值 reboot 后不会保留
  • 为什么寄存器地址后面的十六进制值的 bit [24]写 1?
    因为该寄存器的 16bit 至31 bit 是写有效位,默认为 0,即不可写。因为要往[8] bit 写 1,所以[8] bit 对应的
    写有效位 16 bit 也要对应置 1 才可写入,这个是根据寄存器描述而定的。

修改过后,再查看,可以发现对应的位的值变为了1:

三、 通过函数gpiod_direction_output()

思 路 :
驱 动 里 如 果 想 要 去 操 作 GPIO , 肯 定 会 调 用 到 gpio_direction_output 、gpio_direction_input、 gpiod_direction_output、 gpiod_direction_input 这几个接口, 他们的定义位置是:

 vim kernel/drivers/gpio/gpiolib.c

在这些接口里添加判断对应查询的 IO 口条件,通过以下函数就可以打印出哪些模块复用了该引脚:

dump_stack(); 

注意:
在 linux3.10 内核的 sdk 中用的是 gpio_direction_output 和gpio_direction_input 接 口 ,
在 linux4.4 内 核 中 则 是 gpiod_direction_output 和gpiod_direction_input。

下面以 linux4.4 版本为例来讲解如何查看引脚:gpio4 b3。

首先,需要计算出代表 gpio4b3 的值,算法如下:

gpio4_B3 = 4 *32 + (B-A) * 8 + 3 = 3 *32 + 1 * 8 + 3 = 139

计算方法参考:
《rk3568 | 瑞芯微平台GPIO引脚驱动编写》

  • 最前面和 32 相乘的数字因为是 gpio4,所以是 432。如果是 gpio3,那就是 332;
  • 括号里面的 A、 B、 C、 D 分别代表数值 0、 1、 2、 3,在计算时候分别对应去减即可,这里因为是 B3,所以用 B-A,如果是 C3,就是 C-A;
  • 最后的+3 是因为是 B3,如果是 GPIO4B2,那么最后就+2。

注:在 linux4.4 内核, io 引脚的值有些变化,也就是按照上文算法计算的结
果要+1000,所以 GPIO4B3 如果是 linux4.4 内核里要填 1139。

int gpiod_direction_output(struct gpio_desc *desc, int value)
{
	if (!desc || !desc->chip) {
		pr_warn("%s: invalid GPIO\n", __func__);
		return -EINVAL;
	}
+ 	if ( desc_to_gpio(desc) == 1139)
+ 	{
+ 		printk("dump_stack_start\n");
+		 dump_stack();
+ 		printk("dump_stack_end\n");
+ 	}
	if (test_bit(FLAG_ACTIVE_LOW, &desc->flags))
		value = !value;
	return _gpiod_direction_output_raw(desc, value);
}

添加后编译烧录,只要对应判断的引脚有被调用,启动 log 中就会打印出堆栈,可以根据找出结果查看,找到驱动调用函数。

四、通过函数rockchip_set_mux()

配置IOMUX会调用该接口,

仍然以引脚gpio4 b3为例。

[drivers/pinctrl/pinctrl-rockchip.c]

	 @@ -1224,6 +1224,17 @@ static int rockchip_set_mux(struct rockchip_pin_bank *bank, int pin, int mux)
			dev_dbg(info->dev, "setting mux of GPIO%d-%d to %d\n",
													bank->bank_num, pin, mux);
+       if((bank->bank_num == 4)&&(pin == 11)){
+               printk("6902 setting mux of GPIO%d-%d to %d\n",
+                               bank->bank_num, pin, mux);
+               dump_stack();
+       }
    if (bank->iomux[iomux_num].type & IOMUX_SOURCE_PMU)
            regmap = info->regmap_pmu;
  • bank_num 表示gpio4
  • 这里“ pin ==”后面跟的值计算方式为:
    将 A0 至 D7 32 个引脚顺序对应数值 0 至 31,b3为11。

五、如何去掉设备树中的复用引脚信息?

刚才分析,发现GPIO2 A2被SDMMC0占用,那么如何来解决这个冲突呢?

只要从设备树下手即可。

瑞芯微平台的设备树,根据平台区分,往往前缀是:

rk + 平台  + 板子型号 + ddr型号 + 版本

比如rk3568系列设计的设备树文件如下:

arch/arm64/boot/dts/rockchip/rk3568-amp.dtsi
arch/arm64/boot/dts/rockchip/rk3568-android9.dtsi
arch/arm64/boot/dts/rockchip/rk3568-android.dtsi
arch/arm64/boot/dts/rockchip/rk3568-dram-default-timing.dtsi
arch/arm64/boot/dts/rockchip/rk3568.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10-android9.dts
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10.dtb
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10-linux.dts
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10-linux-spi-nor.dts
arch/arm64/boot/dts/rockchip/rk3568-evb2-lp4x-v10-bt1120-to-hdmi.dts
arch/arm64/boot/dts/rockchip/rk3568-evb2-lp4x-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb2-lp4x-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb4-lp3-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb5-ddr4-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb5-ddr4-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10-linux.dts
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10-rk628-bt1120-to-hdmi.dts
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10-rk628-rgb2hdmi.dts
arch/arm64/boot/dts/rockchip/rk3568-evb6-ddr3-v10-rk630-bt656-to-cvbs.dts
arch/arm64/boot/dts/rockchip/rk3568-evb7-ddr4-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb.dtsi
arch/arm64/boot/dts/rockchip/rk3568-iotest-ddr3-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-iotest-ddr3-v10-linux.dts
arch/arm64/boot/dts/rockchip/rk3568-linux.dtsi
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v10-linux.dts
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v10-linux-spi-nand.dts
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v12.dtsi
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v12-linux.dts
arch/arm64/boot/dts/rockchip/rk3568-nvr-demo-v12-linux-spi-nand.dts
arch/arm64/boot/dts/rockchip/rk3568-nvr.dtsi
arch/arm64/boot/dts/rockchip/rk3568-nvr-linux.dtsi
arch/arm64/boot/dts/rockchip/rk3568-pinctrl.dtsi

一口君的板子是evb1,ddr4,v10版本,所以去掉其他的文件,

我们只需要关注以下文件即可。

arch/arm64/boot/dts/rockchip/rk3568-android.dtsi
与安卓相关的信息

arch/arm64/boot/dts/rockchip/rk3568.dtsi 
描述cpu、memory、timer、clk、sata、usb host、gic、视频控制器、sram、cru、i2c控制器、uart、pwm、pmu等各种Soc内部硬件信息

arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10.dts
arch/arm64/boot/dts/rockchip/rk3568-evb1-ddr4-v10.dtsi
arch/arm64/boot/dts/rockchip/rk3568-evb.dtsi
与evb1底板相关的外设硬件信息

arch/arm64/boot/dts/rockchip/rk3568-pinctrl.dtsi
pinctl相关硬件信息

还有1个描述pinctl引脚驱动能力的文件:

rockchip-pinconf.dtsi

使用grep命令来查询:

grep sdmmc0 arch/arm64/boot/dts/rockchip/rk3568* -nr

好在信息不多,逐个查看,下面这个文件,我们找到了复用的地方。

rk3568-pinctrl.dtsi

该引脚被定义为sdmmc0_clk,作为时钟被使用了。

	sdmmc0 {
		………………
		/omit-if-no-ref/
		sdmmc0_clk: sdmmc0-clk {
			rockchip,pins =
				/* sdmmc0_clk */
				<2 RK_PA2 1 &pcfg_pull_up_drv_level_2>;
		};

修改的方法有很多种:

  1. 投机取巧法
    将sdmmc0_clk改成其他没有用的gpio

  2. 简单粗暴法
    如果确定没有使用sdmmc0,可以将所有sdmmc0地方全部注释掉

设备树支持下面这种方法:

#if 0
#endif
  1. 硬件飞线法

找硬件工程师飞线,改用其他的GPIO

  1. 最优法
    如果sdmmc0也用到了,

那就只能修改冲突的GPIO,

但是这种情况,往往会牵一发而动全身,

要改好几处,那就需要各位老铁细心慢慢修改了。

好了,本文到底结束。

一口君目标是写100篇瑞芯微平台的文章,

有喜欢瑞芯微的老铁,

欢迎大家关注学习。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/115606.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Http协议基础

Http 是什么&#xff1f; Http 作为 WWW 的重要组成部分被提出 (World Wide Web) 而当时的 WWW 是为了帮助研究者更好的进行知识共享 基本理念&#xff1a;文档之间的相互关联形成超文本&#xff0c;最终连成相互参阅的 WWW Http (Hyper Text Transfer Protocol) 用于定义文…

QUIC的优势与缺陷

被寄予厚望的下一代互联网传输协议&#xff0c;QUIC究竟有哪些优点呢&#xff1f; 总结如下&#xff1a; 多路复用&#xff1a;QUIC升华了HTTP/2中的多路复用技术&#xff0c;实现了基于互相独立的多流&#xff08;多通道&#xff09;数据传输&#xff0c;从根本上解决了TCP存在…

基于C语言学生信息教务管理系统编程设计

一.实现功能 1.从键盘添加学生信息 2.从文件添加学生信息 3.显示学生信息到屏幕 4.显示学生信息到文件 5.删除学生信息 6.插入学生信息 7.查找学生信息 8.成绩排名 二、相关代码 #include<stdio.h> #include<stdlib.h> //使用malloc函数以及exit函数 #include<…

力扣(LeetCode)1759. 统计同构子字符串的数目(C++)

题目描述 双指针数学 根据同构字符串的定义&#xff0c;还有示例&#xff0c;发现同构子字符串的数量&#xff0c;只和字母相同的区间有关。如abbcccaa&#xff0c;有 444 个影响答案的区间&#xff0c;直观切分为a bb ccc aa&#xff0c;用空格划分区间。遍历的任务就是维护这…

灵动岛前端Ui

一、前言 灵动岛&#xff08;Dynamic Island &#xff09;是什么&#xff1f; 灵动岛&#xff0c;是苹果公司iPhone 14 Pro系列 [2] 交互UI&#xff0c;让虚拟软件和硬件的交互变得更为流畅。当有来电、短信等通知时&#xff0c;灵动岛会变化它的形态&#xff0c;以便让用户能…

【大数据】M1 mac win docker安装kafka+mysql+canal

文章目录kafkadocker-compose创建kafka容器启动以后&#xff0c;访问容器&#xff0c;并且发送消息测试问题Exception in thread "main" kafka.zookeeper.ZooKeeperClientTimeoutException: Timed out waiting for connection while in state: CONNECTINGmysqldocker…

LAPS本地管理员密码之使用PowerShell查看和重置密码

目录 一、PowerShell策略设置 二、引入AdmPwd.PS模块 三、查看密码 四、强制重置密码 文章主要介绍在部署了LAPS后&#xff0c;怎么使用PowerShell查看和管理域内本地管理员密码。需要注意的是被操作的电脑需要加域&#xff0c;所有操作都在域内环境下进行。 LAPS介绍 LAP…

Spring Boot 知识总结

Spring Boot 知识总结 一、Spring Boot基础 1.1 什么是Spring Spring是一个开源框架&#xff0c;2003年兴起的一个Java轻量级开发框架&#xff0c;作者&#xff1a;Rod Johnson。 Spring是为了解决企业级应用开发的复杂性而创建的&#xff0c;简化开发。 Spring是如何简化…

LeetCode 每日一题——1759. 统计同构子字符串的数目

1.题目描述 1759. 统计同构子字符串的数目 难度中等43 给你一个字符串 s &#xff0c;返回 s 中 同构子字符串 的数目。由于答案可能很大&#xff0c;只需返回对 109 7 取余 后的结果。 同构字符串 的定义为&#xff1a;如果一个字符串中的所有字符都相同&#xff0c;那么…

Rancher RFO 正式 GA

Rancher RFO GA RFO 是 Rancher For openEuler 的缩写&#xff0c;旨在面向 openEuler 打造 Rancher 基础平台。其中最核心的工作是打造一款面向 openEuler 生态的 Kubernetes 发行版。它基于上游 RKE2 的技术栈&#xff0c;构建物采用 openEuler base image&#xff0c;致力于…

C语言及算法设计课程实验一:C程序的运行环境和运行C程序的方法

C语言及算法设计课程实验一&#xff1a;C程序的运行环境和运行C程序的方法一、实验目的二、实验内容2.1、输人并运行一个简单的正确的程序2.2、输人并编辑一个有错误的C程序2.3、输入并运行一个需要在运行时输入数据的程序2.4、运行一个自己编写的程序三、实验步骤3.1、输人并运…

Android OpenGL ES 学习(十一) –渲染YUV视频以及视频抖音特效

OpenGL 学习教程 Android OpenGL ES 学习(一) – 基本概念 Android OpenGL ES 学习(二) – 图形渲染管线和GLSL Android OpenGL ES 学习(三) – 绘制平面图形 Android OpenGL ES 学习(四) – 正交投影 Android OpenGL ES 学习(五) – 渐变色 Android OpenGL ES 学习(六) – 使用…

基于MWORKS.Sysplorer的电子控制器应用案例——永磁同步电机FOC算法建模

1 前言 MWORKS是面向数字工程的新一代科学计算与系统建模仿真平台&#xff0c;可提供机械、电子、液压、控制、热、信息等多领域统一建模仿真环境。经过同元持续攻关&#xff0c;全新推出的MWORKS.Sysplorer嵌入式代码生成器&#xff0c;现已支持面向电子控制器的产品级的嵌入…

循环神经网络的简洁实现

参考8.6. 循环神经网络的简洁实现 — 动手学深度学习 2.0.0 documentation 本节将展示如何使用深度学习框架的高级API提供的函数更有效地实现相同的语言模型。 我们仍然从读取时光机器数据集开始。 pip install mxnet1.7.0.post1 pip install d2l0.15.0 from mxnet import n…

ubuntu18.04下用Fiddler抓取curl库网络数据包总结

本人在ubuntu18.04下进行开发&#xff0c;需要使用http和服务端进行通信&#xff0c;为了确认自己发送给服务端和服务端返回数据字段&#xff0c;所以需要进行抓包分析参数。本文就说明一下如何在ubuntu18.04使用fidder对自己编写的应用程序进行http协议数据包抓取。 目录 1.…

无线网络渗透测试清单

©网络研究院 无线渗透测试积极检查 WiFi 网络中的信息安全措施的过程&#xff0c;并分析弱点、技术流程和关键无线漏洞。 我们应该关注的最重要的对策是威胁评估、数据盗窃检测、安全控制审计、风险预防和检测、信息系统管理和升级基础设施&#xff0c;并且应该准备一份…

13-14-15-RabbitMq工作模式深度剖析与Spring整合MQ以及RabbitMq高级特性

RabbitMQ消息传递流程 连接( Connection) 在RabbitMQ中&#xff0c;生产者和消费者与RabbitMQ的通信就是基于TCP连接的。不过呢我们知道TCP连接的创建和销毁在高并发场景下对于操作系统来说都是特别昂贵的开销&#xff0c;所以RabbitMQ又引入了信道的概念 信道&#xff08;Chan…

云原生之使用Docker部署轻量级web服务器lighthttpd

云原生之使用Docker部署轻量级web服务器lighthttpd一、Lighthttpd介绍二、检查系统版本三、检查docker状态四、下载lighthttpd镜像五、部署lighthttpd1.创建数据目录2.创建lighthttpd容器3.查看容器状态六、访问lighthttpd服务七、编辑index.html1.编辑index.html文件2.重新访问…

Hadoop大数据存算分离方案:计算层无缝对接存储系统

Hadoop的诞生改变了企业对数据的存储、处理和分析的过程&#xff0c;加速了大数据的发展。随着大数据系统建设的深入&#xff0c;企业的数据基础设施易出现计算资源浪费、存储性能低、管理成本过高等挑战。相比存算一体架构&#xff0c;存算分离架构具有性能与成本最优、兼具灵…

3D地图app

3D三维地图APP 发布时间&#xff1a;2018-07-19 版权&#xff1a; 3D地图依据高程数据等对地表进行渲染&#xff0c;实现地表的起伏&#xff0c;模拟出真实的三维场景&#xff0c;让你有如身临其境般的感觉。 &#xff08;注&#xff1a;Bigemap 3D地图是一个三维地图浏览功能…