data:image/s3,"s3://crabby-images/64aa2/64aa20ea697a48467c7ec27303b3b2564e6c7f7b" alt="img"
在一年前笔者研究过一款嵌入式设备,当时就对其声波配网的逻辑感到十分新奇,但后面专注于漏洞挖掘就没有深入研究
最近在研究一些诸如sub-ghz、蓝牙、射频等无线通信的东西,于是通过声波传递信息的方式又激发了我研究的兴趣,所以就有了这篇文章,简要记录一下我的一些发现和心得
将主控芯片上的散热片去除后,我们可以通过放大镜读取主控芯片上的丝印:
data:image/s3,"s3://crabby-images/5fcf0/5fcf012b17698cdf37918dc890338daaf04fd47a" alt="img"
使用的是lngenic t20
芯片的官方网站:Ingenic-北京君正集成电路股份有限公司
音频功放为XPT4890:
data:image/s3,"s3://crabby-images/4f1b7/4f1b7f17093680d192834461547f097c055733e1" alt="img"
具体的音频接收是通过设备上的咪头,咪头接收到声波后产生电信号,然后主控芯片对电信号进行处理
首先采集一段配网时候的音频进行分析
通过aux公对公线,能直接从手机的耳机孔输出音频到电脑上,通过windows自带的录音软件或者OBS就可以采集
data:image/s3,"s3://crabby-images/00af3/00af3e54792c287f43af1ba5486e0aa986a111c8" alt="img"
查阅资料发现声波配网和声音的频率有很大关系,所以这一步主要是通过脚本提取声波的频率特征
这里使用librosa这个库,但是它不支持m4a,只支持对wav进行处理,所以首先需要将m4a转换为wav
转换使用的是pydub这个库,它对音频的转换要安装FFmpeg:
Windows 10系统下安装FFmpeg教程详解_ffmpeg window10-CSDN博客
下载链接:https://www.gyan.dev/ffmpeg/builds/ffmpeg-release-essentials.zip
把bin目录添加到环境变量就行了:
data:image/s3,"s3://crabby-images/4d834/4d83434858295395b3e3f25dfdace666c587bb2b" alt="img"
转换代码:
代码:
效果:
data:image/s3,"s3://crabby-images/ce0e9/ce0e96ba8c3cf09bea2c2fbfde25e450637ea236" alt="img"
代码:
效果:
data:image/s3,"s3://crabby-images/f4dab/f4dabd3134e16c6fca7e4009ddff4953834e341d" alt="img"
当wifi的名称是8个1,密码是32个1的情况
data:image/s3,"s3://crabby-images/7a2b6/7a2b68e366690eb76a6c0c7479411e7434854dae" alt="img"
发现有一部分和上面的图片是重叠的,而且这张图片的频率分布确实更平整了,说明确实和频率是有关系的
但仍然看不出什么东西,需要具体逆向分析一下通信过程中传输协议是什么样的
整个通信大体可以概括为:
设备配套的app将wifi的ssid和passwd通过声波这种媒介传递到设备中,设备接收到信息后连接上对应的wifi并访问运营商的云端服务器,并给予用户连接成功/失败的反馈
在固件中能看到一个名为"connect_wifi.sh"的sh脚本,脚本主要的作用是通过wpa_supplicant.conf中的内容去连接wifi
data:image/s3,"s3://crabby-images/9c3fd/9c3fd5defe4f3c0ebe86ca41619c3f06727e85c2" alt="img"
wpa_supplicant.conf的格式如下:
data:image/s3,"s3://crabby-images/44e71/44e71e7f357f5d25557f3588779d4cf78116c327" alt="img"
搜索存在"wpa_supplicant.conf"的elf可以发现有如下调用:
data:image/s3,"s3://crabby-images/dc523/dc52368de29a8b01c211306c699b1f1f7f76ad7a" alt="img"
在"ipcam"这个elf中可以看到如下内容:
data:image/s3,"s3://crabby-images/0b262/0b262894f7ec7fce205fa1815a09ed07310a3877" alt="img"
那么可以确定是ipcam这个elf对声波进行识别,并把内容保存到"wpa_supplicant.conf"中,然后在"daemon"中执行"connect_wifi.sh"连接wifi
该函数会被当作参数传递到"vr_setRecognizerListener"中进行处理:
data:image/s3,"s3://crabby-images/c1681/c1681a4f2ece6516aac74bafd7953d430c4e8f24" alt="img"
data:image/s3,"s3://crabby-images/3f5e2/3f5e2f040415c04d815cd15059de97f688668dfa" alt="img"
然后就到了c++的魅力时刻:
data:image/s3,"s3://crabby-images/ed54a/ed54ac45d7e6876948f6d00807a8be3803d86f6b" alt="img"
这里应该是某个结构体的虚表跳转,那么想继续从固件里看出东西的话就很困难了
但其实可以看出有很多地方都有调试的打印,而且这个设备是有UART调试串口的,所以应该可以通过调试打印的信息定位后续处理的位置
遗憾的是,调试口被笔者焊坏掉了(QAQ):
data:image/s3,"s3://crabby-images/67de1/67de10ac032f941328e143510be5b4c413c67e2a" alt="img"
而且从板子上飞线出来也不是很好飞
所以暂时放弃对接收方的研究,去逆向一下发送方(app)的逻辑
对app进行解压,确定不是flutter架构:
data:image/s3,"s3://crabby-images/5d7f2/5d7f27125271d3c02cde0765eff33327ede59855" alt="img"
Pkid查壳发现是《梆梆》加固:
data:image/s3,"s3://crabby-images/34a0f/34a0f7cc6353fec5a9a0d5cc73364efb3fef2b39" alt="img"
用jadx直接打开也能看到比较明显的加固特征
data:image/s3,"s3://crabby-images/1c0a3/1c0a363415d5286af2bea69cee9a79d3c9d9263d" alt="img"
通过frida dexdump脱壳试试:
data:image/s3,"s3://crabby-images/037ca/037cad77dc543278d9d609caba0f3b278d13b59f" alt="img"
能成功dump下来59个dex,并且每一个dex都可以通过JEB进行查看了:
data:image/s3,"s3://crabby-images/d07d8/d07d8bf604e9b395f5ce9f0929ba0aae7523cefd" alt="img"
首先需要从59个dex中定位到哪些是我们需要的
我们可以让app运行到声波配网的页面:
data:image/s3,"s3://crabby-images/f7925/f792581e3538d3f457d3e89e249370ef00bcb1eb" alt="img"
然后通过adb shell执行"dumpsys activity top | grep ACTIVITY"这行命令
就可以列出顶层的activity,也就是需要尤为关注的内容:
data:image/s3,"s3://crabby-images/fa219/fa21947d0adadc7d618f95f2b7fb233f24c799f7" alt="img"
通过"Everthing"或者其它字符串搜索工具搜索"SoundWaveGuideActivity",能够定位到如下dex
data:image/s3,"s3://crabby-images/2c459/2c459a6b4d03ad5c7fb9013cbc44832a8cc83640" alt="img"
用JEB对该dex进行简单的逆向可以知道整个流程应该是在"soundWaveGuide.control"中进行的
data:image/s3,"s3://crabby-images/83c6f/83c6f3f6f9380b63a9853e3765813803d8568112" alt="img"
首先输入的SSID和passwd会被分别存入this.F和this.G:
data:image/s3,"s3://crabby-images/3ffeb/3ffeb4c0ed556b98776c3dfc3abe2a020b926467" alt="img"
然后到下面这个地方会对传入的SSID和passwd进行处理然后通过"play"方法进行播放
data:image/s3,"s3://crabby-images/e4dc9/e4dc963900833c0491f33409cb49f9897d05ab6d" alt="img"
"play"和"encodeString"方法都不是在这个dex实现的,同样可以通过搜关键字符串的形式进行查找,最后在16.dex里找到了"play"方法的具体实现:
data:image/s3,"s3://crabby-images/ea57e/ea57e2bca73de1dbfc383d199e21a0cae80d657e" alt="img"
并且可以发现这里用的是一个原生库的函数:
data:image/s3,"s3://crabby-images/58bd5/58bd56c5a18a55e0389b8175d75a5ec051edb73e" alt="img"
直接检索"play"发现有很多的原生库都能匹配:
data:image/s3,"s3://crabby-images/9505e/9505e0b563166b9348e4dbd873bb6dc8edabe7bd" alt="img"
好在这个app在运行的时候会有一些关键的log,能够通过"adb logcat"进行打印获取:
data:image/s3,"s3://crabby-images/48810/48810b740a52d8f8b5e444b22d6420bc2bb69694" alt="img"
通过搜索"play text"最终定位了具体的原生库文件:
data:image/s3,"s3://crabby-images/dcd95/dcd95aaa483fa1f79d5bb8d90a30761327728762" alt="img"
原生库文件一样是用c++写的,直接逆向分析也会遇到虚表跳转
data:image/s3,"s3://crabby-images/41004/410041e611c6569369144c5770406aaf8e210de4" alt="img"
好在对于安卓有frida这种神器,可以打印具体的内存信息获取虚表跳转的地址
但是要hook的地方实在是太多了,基本上是一步一hook非常麻烦
好在app的开发者偷懒了,通过在github上搜原生库中的一个函数名"slg_gen"就能直接搜到一个名为"SmartBoy"的项目:
data:image/s3,"s3://crabby-images/f5d89/f5d892e08442209b79767d9e72b7f8db612d52c4" alt="img"
和原生库中的一模一样,而且其相关调用链也高度一致,也能在项目中搜到adb logcat的打印字符串:
data:image/s3,"s3://crabby-images/37c84/37c841a1b822c1a5958cd1cb1a5fb170d76494eb" alt="img"
项目地址:https://github.com/npnet/SmartBoy/tree/af0403fc7151f22381b0d0da7df3812f64ea5b3a
更为夸张的是,笔者在项目的"main.c"里发现了如下内容:
data:image/s3,"s3://crabby-images/9b2a8/9b2a8cabd794d091f825c0dde8878716e96e47d7" alt="img"
好家伙,直接不演了,黑盒测试变成白盒了
data:image/s3,"s3://crabby-images/24a94/24a94582a32c1db9303c3134b07e5be8f93cfdd1" alt="img"
CSDN链接:http://blog.csdn.net/softlgh
通过CSDN的链接我们能下载下来linux、windows、macos、android这几个主流平台的声波配网的全部demo,这里面显然就包括了android的源码
在源码里我们能看到熟悉的原生库,以及熟悉的代码
data:image/s3,"s3://crabby-images/7f577/7f5771fa35049f7e0011aad23dc8e6a2d5b2df14" alt="img"
data:image/s3,"s3://crabby-images/9a874/9a874244221fcfcfea51ae1756549f1598b0facd" alt="img"
只不过demo的baseFreq是16000,而app里的是4000:
data:image/s3,"s3://crabby-images/1a6b9/1a6b90ade9cf137bc0f6c3beb968f1a1e37ceb1c" alt="img"
至此,我们的逆向工作其实已经可以结束了,因为我们完成了逆向的最高境界——逆出源码(笑)
后续的工作更多的就是对传输的协议(对ssid+passwd字符串的处理)的分析了
分析协议主要是分析"play"的字符串长啥样
官方app提供了设备的日志,从日志里能看见对传入声波的解析后的内容:
data:image/s3,"s3://crabby-images/7b9d6/7b9d677647b75777ab36fb74d5b0244addd4f2b6" alt="img"
主要是ssid psd phone这三个东西,猜测"play"的字符串就和这三个东西有关
在JEB反编译的java代码中可以看见这三个内容经过了两次处理:
data:image/s3,"s3://crabby-images/b64d4/b64d4cfb12870cd5c6cef5cd63dd5c465e4c6097" alt="img"
第一次处理是对三个字符串的拼接,这里就不赘述
第二次处理是对第一次拼接的字符串进行编码:
data:image/s3,"s3://crabby-images/c9f00/c9f00dd588d1d61f33710a72da5705497de27b91" alt="img"
hexChar是一个数组:
data:image/s3,"s3://crabby-images/c35f1/c35f1414f9f4c2d45719aa1cc10495bd4716ca2d" alt="img"
在encodeData里会把传入的字符串分为8类:
data:image/s3,"s3://crabby-images/32d43/32d436ef5fe284e4a58a5af3e842188e33772d84" alt="img"
[培训]内核驱动高级班,冲击BAT一流互联网大厂工作,每周日13:00-18:00直播授课
最后于 2023-12-5 00:05
被Nameless_a编辑
,原因: