MTK8766 LK GPIO初始化状态设置分析
问题来源是M.2 Dongle的LED灯在kernel起来之前就亮了,kernel起来之后又初始化成熄灭状态。通过排查硬件规格书、GPIO表格,大概判定是前期软件初始化不正确造成的。通过观察串口打印的log,发现灯开始亮起来的瞬间是运行在lk中。
找到MTK 的LK代码路径
MT8766_A11_AP/vendor/mediatek/proprietary/bootable/bootloader/lk/
找到MTK的LK代码编译out目录
MT8766_A11_AP/out/target/product/tb8766p1_bsp_1g/obj/LK_OBJ/build-tb8766p1_bsp_1g/
再从out目录下的plarform目录确定当前编译的平台是mt6761,从而在对应的gpio配置输出函数中添加log
MT8766_A11_AP/vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6761/mt_gpio.c
S32 mt_set_gpio_mode_chip(u32 pin, u32 mode)
{
......
添加以下代码对改GPIO进行debug
if (pin == 174)
{
GPIOVER("============ fuck gpio ===============\n");
mdelay(1000);
}
return RSUCCESS;
}
通过log大概确定相关动作是在lk_dtb_init()中或者后面一点点
[254] [SBC] image dtbo auth init pass
[254] [SBC] image dtbo cert vfy pass(22 ms)
[255] [PROFILE] mmc read 1 blks in 0 ms: 8KB/s
[256] dtbo_total_len: 0x8a17
[257] [PROFILE] mmc read 69 blks in 1 ms: 34500KB/s
[258] [PROFILE] mmc read 1 blks in 0 ms: 8KB/s
[259] [SBC][oem] img auth pass
[259] [SBC] dtbo vfy pass(1 ms)
[260] dtbo_offset: 0x40
[260] [PROFILE] ::: lvl(3) load_dtbo_buffer takes 30 ms
[261] dtbo_size : 35287
[272] [PROFILE] ::: lvl(3) dtb_overlay takes 11 ms
[273] [PROFILE] ::: lvl(2) lk_dtb_init takes 100 ms
[275] [GPIO] mt_set_gpio_mode_chip: ============ fuck gpio ===============
[1278] [PROFILE] mmc read 32 blks in 2 ms: 8000KB/s
[1279] [LK_ENV]ENV of area 0 initialize sucess
[1280] [PROFILE] mmc read 32 blks in 1 ms: 16000KB/s
[1280] [LK_ENV]ENV SIG of area 1 is wrong
[1281] [LK_ENV]env area: 0
[1281] [LK_ENV]env:
[1281] [LK_ENV]md1_ccb_cap_gear=1
[1282] [LK_ENV]md1_ccb_gear_list=1(2,20);2(2,10);3(0,0);11(2,2);12(2,62)
[1283] [LK_ENV]env area: 1
[1283] [LK_ENV]no valid env
[1283] [PROFILE] ::: lvl(2) ENV init takes 7 ms
[1284] mblock_reserve-R start: 0x7f040000, sz: 0xdb0000 map:0 name:framebuffer
最终定位到以下调用关系
platform_early_init();
|
mt_gpio_set_default();
|
mt_gpio_set_default_chip();
观察mt_gpio_set_default,发现相关配置参数是从设备树中获取的,展开mt_gpio_set_default()函数中的mt_gpio_get_default_chip_from_dtb()调用,发现相关设备树节点名字是gpio@10005000,相关属性的名字是gpio_init_default。用adb pull /sys/firmware/fdt,反编译设备树确认存在这个属性和节点。
在kernel中过滤文本搜索gpio_init_default属性,发现并不存在与当前硬件相关的设备树文件!那么,猜测该属性应该是从别的设备树文件中合并进来的。再在代码中使用grep -nr gpio_init_default
全局过滤搜索,发现有以下文件与之相关。
./kernel-4.19/arch/arm64/boot/dts/mediatek/mt6779-evb.dts:604: gpio_init_default = <0 0 0 0 1 1 1>,
./vendor/mediatek/proprietary/tools/dct/python/obj/GpioObj.py:746: gen_str += '''\tgpio_init_default = '''
./vendor/mediatek/proprietary/tools/dct/python/obj/GpioObj.py:772: gen_str += '''\tgpio_init_default = '''
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/GpioObj.py:710: gen_str += '''\tgpio_init_default = '''
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/GpioObj.py:736: gen_str += '''\tgpio_init_default = '''
./vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6768/gpio_init.c:79:#define GPIO_DT_NODE_PROP_NAME "gpio_init_default"
./vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6765/mt_gpio_init.c:134:#define GPIO_DT_NODE_PROP_NAME "gpio_init_default"
./vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6761/mt_gpio_init.c:134:#define GPIO_DT_NODE_PROP_NAME "gpio_init_default"
./vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6785/gpio_init.c:79:#define GPIO_DT_NODE_PROP_NAME "gpio_init_default"
./vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt6771/gpio_init.c:76:#define GPIO_DT_NODE_PROP_NAME "gpio_init_default"
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py:710: gen_str += '''\tgpio_init_default = '''
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py:736: gen_str += '''\tgpio_init_default = '''
打开与之相关的文件
vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py
发现相关的python调用方法是fill_init_default_dtsiFile
,那么再全局过滤搜索这个方法名,发现有以下文件与之相关。
./vendor/mediatek/proprietary/tools/dct/python/obj/ChipObj.py:260: gen_str += gpioObj.fill_init_default_dtsiFile()
./vendor/mediatek/proprietary/tools/dct/python/obj/GpioObj.py:649: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/tools/dct/python/obj/GpioObj.py:744: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/tools/dct/python/obj/GpioObj.py:770: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/ChipObj.py:224: gen_str += gpioObj.fill_init_default_dtsiFile()
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/GpioObj.py:613: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/GpioObj.py:708: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/preloader/tools/dct/obj/GpioObj.py:734: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/ChipObj.py:207: gen_str += gpioObj.fill_init_default_dtsiFile()
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py:613: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py:708: def fill_init_default_dtsiFile(self):
./vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/GpioObj.py:734: def fill_init_default_dtsiFile(self):
打开与之相关的文件
vendor/mediatek/proprietary/bootable/bootloader/lk/scripts/dct/obj/ChipObj.py
内容如下:
def gen_custDtsi(self):
log(LogLevel.info, 'Start to generate cust_dtsi file...')
fp = open(os.path.join(ModuleObj.get_genPath(), 'cust.dtsi'), 'w')
gen_str = ModuleObj.writeComment()
# if early porting, gen empty dtsi file for kernel
if self.__epFlag:
fp.write(gen_str)
fp.close()
return
#sorted_list = sorted(self.__objs.keys())
#for tag in sorted_list:
for tag in self.__objs.keys():
if cmp(tag, 'gpio') == 0:
gpioObj = self.create_obj(tag)
gen_str += ModuleObj.writeHeader(gpioObj.get_dtsiFileName())
gen_str += gpioObj.fill_mapping_dtsiFile()
gen_str += gpioObj.fill_init_default_dtsiFile()
else:
obj = self.create_obj(tag)
gen_str += ModuleObj.writeHeader(obj.get_dtsiFileName())
gen_str += obj.fill_dtsiFile()
gen_str += '''\n\n'''
fp.write(gen_str)
fp.close()
log(LogLevel.info, 'Generate cust_dtsi file successfully!')
可以从中判断出最终生成了cust.dtsi,全局搜索这个文件find -name cust.dtsi
,发现生成路径是
./out/target/product/tb8766p1_bsp_1g/obj/KERNEL_OBJ/arch/arm/boot/dts/tb8766p1_bsp_1g/cust.dtsi
打开后确认生成的文件存在gpio@10005000
节点和gpio_init_default
属性。但是目前又不能判断到底是从哪个文件中解析了配置,从而生成了最终的设备树文件。好在该方法中打印了log,再从Android的编译log中搜索Start to generate cust_dtsi file...
,发现以下内容:
[ 60% 688/1130] build out/target/product/tb8766p1_bsp_1g/obj/FAKE/treble_sepolicy_tests_27.0_intermediates/27.0_mapping.combined.cil
[ 60% 689/1130] build out/target/product/tb8766p1_bsp_1g/obj/KERNEL_OBJ/arch/arm/boot/dts/tb8766p1_bsp_1g/cust.dtsi
[DCT_INFO]: DWS file path is /home/123/workspace/MT8766_A11_AP/kernel-4.19/drivers/misc/mediatek/dws/mt6761/tb8766p1_bsp_1g.dws
[DCT_INFO]: Gen files path is /home/123/workspace/MT8766_A11_AP/out/target/product/tb8766p1_bsp_1g/obj/KERNEL_OBJ/arch/arm/boot/dts/tb8766p1_bsp_1g
[DCT_INFO]: Log files path is /home/123/workspace/MT8766_A11_AP/out/target/product/tb8766p1_bsp_1g/obj/KERNEL_OBJ/arch/arm/boot/dts/tb8766p1_bsp_1g
[DCT_INFO]: Parameter is cust_dtsi
[DCT_INFO]: chip id: MT6761
[DCT_INFO]: Chip ID : MT6761
[DCT_INFO]: Project Info: Merlot_Phone
[DCT_INFO]: Start to generate cust_dtsi file...
[DCT_INFO]: Generate cust_dtsi file successfully!
[ 61% 690/1130] //system/core/init:libinit static link libinit.a
[ 61% 691/1130] //system/core/init:libinit static link libinit.a
好家伙,绕了一圈原来是tb8766p1_bsp_1g.dws!
那么修改dws 的GPIO配置,重新编译。。。发现cust.dtsi中对应的GPIO配置变了,烧录验证。。。成功!
MTK8766 LK GPIO初始化状态设置分析的更多相关文章
- HGE初始化状态设置
HGE_FRAMEFUNC: 最重要的设置,每个HGE应用必须设置.游戏的主循环就是他了.类型为bool*(),返回真那么主循环退出,游戏也就结束了.否则进行必要的处理后返回假.必须在调用进入 ...
- Linux USB Host-Controller的初始化代码框架分析【转】
转自:http://blog.csdn.net/zkami/article/details/2496770 usb_hcd_omap_probe (const struct hc_driver *dr ...
- k8s replicaset controller分析(1)-初始化与启动分析
replicaset controller分析 replicaset controller简介 replicaset controller是kube-controller-manager组件中众多控制 ...
- kube-scheduler源码分析(1)-初始化与启动分析
kube-scheduler源码分析(1)-初始化与启动分析 kube-scheduler简介 kube-scheduler组件是kubernetes中的核心组件之一,主要负责pod资源对象的调度工作 ...
- 因为相同类型的其他实体已具有相同的主键值。在使用 "Attach" 方法或者将实体的状态设置为 "Unchanged" 或 "Modified" 。。。
因为相同类型的其他实体已具有相同的主键值.在使用 "Attach" 方法或者将实体的状态设置为 "Unchanged" 或 "Modified&quo ...
- tableview 编辑状态设置
#pragma mark - tableview 编辑状态设置 -(BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSI ...
- Java线程Thread的状态解析以及状态转换分析 多线程中篇(七)
线程与操作系统中线程(进程)的概念同根同源,尽管千差万别. 操作系统中有状态以及状态的切换,Java线程中照样也有. State 在Thread类中有内部类 枚举State,用于抽象描述Java线程的 ...
- Wireshark抓包分析TCP建立/释放链接的过程以及状态变迁分析
Wireshark抓包分析TCP建立/释放链接的过程以及状态变迁分析 一.介绍计算机网络体系结构 1.计算机的网络体系结构 在抓包分析TCP建立链接之前首先了解下计算机的网络通信的模型,我相信学习过计 ...
- ContextMune上下文菜单中,二级菜单获取及状态设置
ContextMune上下文菜单中,二级菜单获取及状态设置 在使用ContextMune上下文菜单中,能够通过二级菜单来获取及状态设置 //二级菜单获取和状态设置((ToolStripDropDown ...
- 关于Entity Framework更新的几种方式以及可能遇到的问题(附加类型“Model”的实体失败,因为相同类型的其他实体已具有相同的主键值)在使用 "Attach" 方法或者将实体的状态设置为 "Unchanged" 或 "Modified" 时如果图形中的任何实体具有冲突键值,则可能会发生上述行为
在日常使用Entity Framework中,数据更新通常会用到.下面就简单封装了一个DBContext类 public partial class EFContext<T> : DbCo ...
随机推荐
- CSS动画(登录页面)
1.整体效果 https://mmbiz.qpic.cn/sz_mmbiz_gif/EGZdlrTDJa6Pfop3IiakrZOtiaiaKniaxeH2Gs407DmkXPsMo8AFKdsYib ...
- python语言中三个奇妙的返回值
从公众号看到的,记录下 d = {} d[5] = "test1" d[5.0] = "test2" d["5"] = "test ...
- Power BI新卡片更改显示单位
Power BI 不知道什么时候发布了新卡片,照现在官方来说,该视觉对象目前还属于预览版,但已经可以正常使用了,对比旧的卡片,显示效果个人觉得会友好一些,详见官方说明:创建"新"卡 ...
- awk优化钉钉通知测试报告
一.背景 在之前的博客 Go服务自动触发单元测试覆盖率 中钉钉通知的效果实现如下图: 最近RD提出对本次和上次覆盖率的比对需求,并把比对结果也显示在钉钉通知上. 二.实现思路 要实现数据比对,就需要对 ...
- 每秒550万亿次算力!打破世界纪录!中国造全球首例纯电驱全尺寸人形机器人!直击全球最快人形机器人“天工”The world's first purely electric humanoid robot
地址: https://www.youtube.com/watch?v=uRc-885NpD4
- Java真的没出路了吗?
Java从1991年由James Gosling和他的同事们开发, 至今已经三十多年, 我们知道,任何产品都有生命周期, 都要经历从诞生.发展.成熟.消亡四个阶段, 目前的Java已经处在成熟阶段, ...
- Nuxt.js 应用中的 webpack:change 事件钩子
title: Nuxt.js 应用中的 webpack:change 事件钩子 date: 2024/11/24 updated: 2024/11/24 author: cmdragon excerp ...
- 使用 ASM 实现 Java 语言的“多重继承”
问题的提出 在大部分情况下,需要多重继承往往意味着糟糕的设计.但在处理一些遗留项目的时候,多重继承可能是我们能做出的选择中代价最小的.由于 Java 语言本身不支持多重继承,这常常会给我们带来麻烦,最 ...
- MySQL底层概述—4.InnoDB数据文件
大纲 1.表空间文件结构 (1)表空间Tablesapce (2)段Segment (3)区Extend (4)页Page (5)行Row 2.Page结构 (1)页结构各部分说明 (2)页结构整体划 ...
- 使用 JsonSchema 校验 JSON数据
有时候JSON 数据格式需要校验是否合法,我们可以使用 JsonSchema 来校验数据是否合法. 引入 pom.xml https://json-schema.org/ <dependency ...