本文分享自华为云社区《go语言逆向技术之---常量字符串解密》,作者:安全技术猿。

Go语言源代码编译成二进制文件后,源代码中的字符串存放在哪里?是如何组织的?

以下面go语言源代码为例:
package main
  • import “fmt”
  • func main() {
  • fmt.Println(“Hello, World!”)
  • }
  • 复制代码
    虽然只是打印一个字符串"Hello, World!",生成的二进制文件中字符串却是非常的多:

    v2-1d8228c1fc8818697adf48ab0f28ce69_720w.jpg

    Go语言二进制文件中字符串是存放在哪里的呢?其实存放的位置遵循的elf格式原则,在.rodata节中(如上图所示)。

    有同学会问C语言编译出来的elf格式中字符串也是这样存放的,Go语言的字符串组织方法和C语言的字符串组织方法有什么不同的呢?最大的不同点就是C语言字符串是以’\x00’结尾的,这样不同字符串之间可以方便的以’\x00’来切割,而Go语言的字符串你可以发现是没有’\x00’结尾的,比如上图中"Hello, World!"后面紧接着就是"SIGKILL:"了,没有’\x00’分隔。

      Go语言字符串按以下方法来组织:


    • 字符串按长度从小到大排列
    • 相同长度的字符串按字符比较的方法从小到大排列
    • 非可见字符串先转义再存放

    另外Go语言字符串一般从go_string位置开始存放,如下图所示

    v2-7e744851d0df22301b3dc67f5eac1a78_720w.jpg

      逆向时正确切割Go语言字符串方法有两种:


    • 进行反汇编,解析汇编指令,确定字符串起始位置和处理长度
    v2-082f6311106e79eb8ff2a13c3643ca4c_720w.jpg


    • 直接根据Go语言字符串组织原则进行切割
      
    基于Go语言字符串组织原则的快速切割算法:


    • 1、 搜索确定go_string起始位置
    • 2、 设置字符串搜索起始长度为1
    • 3、 根据当前字符串长度切割字符串
    • 4、 预切割下一个字符串,和当前字符串进行比较,检查是否违反原则,若符合保存当前字符串,继续第3步切割下一个字符串,否则表示字符串有变化进行第5步操作
    • 5、 把当前字符串长度加1,继续第3步的搜索,直至全部正确搜索完毕

    如下图所示(长度为13个字节的字符串):

    v2-29a32f7040b960a45fb7cc304a1bd284_720w.jpg

    总结:

    go语言字符串的切割在二进制安全检测中可以真实的还原源代码中引用字符串的信息,提升检测准确率。