逆向分析rust - (1)
rust现在用的越来越多了,很多红队也在使用rust进行进攻性工具开发,作为一名逆向工程师肯定也需要了解一下rust,于是就有了本文,在个人学习rust的过程中,顺便对生成的二进制进行逆向分析,一举两得,何乐而不为呢?
为了方便学习,会加载符号来协助分析,且大部分使用Windows平台X64架构Debug版本编译的二进制文件
语法就不讲解了,有兴趣的可以自己去学习相关语法
根据惯例,第一个程序以"hello world"开始
fn main(){
println!("Hello World");
}
Rust默认生成的是动态链接的二进制文件(即运行时需要运行时库,没有运行时库会提示缺少xxx.dll),使用`rustc -C target-feature=+crt-static main.rs`编译生成静态链接文件
查看生成的文件发现文件大小为258kb
data:image/s3,"s3://crabby-images/77eea/77eeabb9422755b82d46ddc35cfd79d994b66e25" alt=""
作为对比,动态链接生成的文件大小为159kb
data:image/s3,"s3://crabby-images/514a3/514a3f43bccd9bd36fb984875127cf41b8ced762" alt=""
查看导入表,只有kernel32.dll
data:image/s3,"s3://crabby-images/3eb05/3eb055ad23d07f6c4db125c8eed9272818bd6a08" alt=""
作为对比,动态链接生成的文件导入表有运行时相关库
data:image/s3,"s3://crabby-images/45f03/45f0355c2b57f92e29e43d5272e79561ddc32cc6" alt=""
使用IDA Pro 7.5打开,可以看到入口函数为`mainCRTStartup`,这有点像MSVC编译器生成的C/C++程序,然后内部调用`__scrt_common_main_seh`
data:image/s3,"s3://crabby-images/7011f/7011f489da854621affdb2182ee10b9bdd44e6e1" alt=""
data:image/s3,"s3://crabby-images/5a3dd/5a3dd26ed45ee505f689c8671a4f66e9f3690ec3" alt=""
在`_cexit`上方两个函数左右进入`main`,真是非常像msvc编译出来的C/C++程序
data:image/s3,"s3://crabby-images/08fc2/08fc29337b9541feff86769d47f6e0ddf4c56d2c" alt=""
作为对比,看以下MSVC生成的C++程序,当然,也可能是因为使用了MSVC的工具链
data:image/s3,"s3://crabby-images/dbc8f/dbc8f506d32d649b071cf265377e580a8d7bf1da" alt=""
main
函数也很简单,开辟栈空间,操作了`r8`,`rdx`,`rcx`,`r9`四个寄存器,看出来了吗?是不是很像经典的x64调用(cd89),其中`rcx`指向sub_1400010B0
data:image/s3,"s3://crabby-images/22c31/22c318d773ec86705aa9ba8245b8219616b3f415" alt=""
我们看一下IDA的F5,可以判定就是cd89调用,`rcx`为sub_1400010B0,`rdx`为main函数的argc,也就是movsxd rdx,ecx的操作,`r8`为argv(mov r8,rdx),`r9`为0(xor r9d,r9d)
data:image/s3,"s3://crabby-images/35db5/35db5439a7beb145340a131408e572374f44593c" alt=""
函数std::rt::lang_start::xxx从名称std::rt上来看像是运行时(runtime)相关的函数,我们暂时先不管,先来看看sub_1400010B0
data:image/s3,"s3://crabby-images/f416e/f416e44360054443948f345aa4a5f8c6a9b04f9b" alt=""
因为我们的程序只有一个打印`hello world`,所以sub_1400010B0可能就是我们的主函数,首先调用了一个格式化字符串函数`core::fmt::xxx`,根据字符串`invalid args`猜测该函数还做了某些判断处理,通常这也符合Debug版本编译的程序的预期操作,然后通过print函数进行打印输出.程序返回后结束,看来没什么值得分析的地方,那么继续返回看看std::rt::lang_start::xxx函数,只有一个`std::rt::lang_start_internal::xxx`的调用
data:image/s3,"s3://crabby-images/88c42/88c42a88d251f3444372f870a20e6841b691fff6" alt=""
继续跟进去
data:image/s3,"s3://crabby-images/a0a81/a0a81d8c997923c4adae626b330630a3d2e24d84" alt=""
看着不是很舒服,还是看汇编吧
data:image/s3,"s3://crabby-images/d5eb9/d5eb9ecd38a67b812336bcad203ad7a189186d8d" alt=""
发现有几个线程相关的函数`Thread::new`,`thread_info::set::xxx`,然后一个间接调用`call qword ptr[rax+28h]`,我们看看间接调用上面的ecx从哪里来的
data:image/s3,"s3://crabby-images/077a1/077a1fb79234df87aaaf21075ec2d8d307844e94" alt=""
可以看到ecx从var_50里取到的,var_50在函数入口处保存的是rcx的值,rcx从上面的分析可以看到就是`sub_1400010B0`,使用x64dbg来动态分析
data:image/s3,"s3://crabby-images/cb60c/cb60cd79f666c608012e79f0663e7d9ea18fa17b" alt=""
call [rax+28h]函数内部调用了两个函数,一个为`rust_begin_short_backtrace`,一个为`std..process..Termination`,从名字上来看可能为栈追踪和进程结束.并且在函数入口获取了sub_1400010B0函数的地址
data:image/s3,"s3://crabby-images/c9d17/c9d178fa4aa46652c7076a526f0b07605f7d17fa" alt=""
跟进`rust_begin_short_backtrace`,又有一个名字为`call_once`的函数调用
data:image/s3,"s3://crabby-images/5e27b/5e27bf5620ba33eca24af1ddcfc7c0169fb273a2" alt=""
跟进去发现`call_once`函数内部有`call rcx`即上面分析的sub_1400010B0函数的流程,执行完返回后调用std..process..Termination,后续调用destroy等函数然后程序退出
data:image/s3,"s3://crabby-images/0dcf9/0dcf92fff42d33c645a452282671371286d2da55" alt=""
main ->
std::rt::lang_start::xxx ->
std::rt::lang_start_internal::xxx ->
call qword ptr[rax+28h] ->
rust_begin_short_backtrace ->
call_once ->
call rcx(helloworld) ->
std::process::Termination
至此我们分析完一个rust的`hello world`程序,从入口到结束的大部分流程