二进制明文字符串加密: 实现原理
2023-11-5 18:1:46 Author: 看雪学苑(查看原文) 阅读量:3 收藏

背景
这里就不多做解释了,明文字符串暴露就是最好的逆向分析指引。无论是恶意攻击样本还是有一定安全需求的组件,直接暴露程序中的明文字符串会大幅降低外部的分析成本。所以需要在编译出的二进制中隐藏字符串。不过需要明确下字符串加密的实现主要分两种,一种是借助llvm的pass去做的,另外一种是基于编译期模板的形式去做的,这里讨论的是后者。

常规的字符串表示

代码和16进制如下:
printf("conventional...\n");

解决明文暴露的问题

首先考虑转换下写法,代码和16进制如下:
char strArray[] = { 'c', 'o', 'n', 'v', 'e', 'n', 't',
   'i', 'o', 'n', 'a', 'l', '.', '.', '.', '\n', '\0' };
printf(strArray);
这么写似乎和直接写字符串编译后没有区别?但是将字符串转为字符数组表示便于下面的加密逻辑。
char strArray[] = { 'c' + 1, 'o' + 1, 'n' + 1, 'v' + 1, 'e' + 1, 'n' + 1, 't' + 1,
   'i' + 1, 'o' + 1, 'n' + 1, 'a' + 1, 'l' + 1, '.' + 1, '.' + 1, '.' + 1, '\n' + 1, '\0' + 1 };
for (unsigned int index = 0; index < sizeof(strArray); index++)
{
   strArray[index] -= 1;
}
printf(strArray);
可以看到实际每个字符以+1的模式存储,在使用前进行解密,从而实现明文字符串隐藏,同时这里的核心在于依靠编译器优化使得加密过程(这里指+1)在编译期可以直接算出加密后的结果,从而在编译完成的二进制中直接保存密文,而运行时进行解密从而使得实际结果保持不变。这里的两个需求非常重要,如果有一个不满足即会导致失败,这里将这两个总目标单独列出。
① 编译期可以直接算出加密后的结果。
② 运行时进行解密。

实现上述过程

当然我们不可能每次隐藏字符串的时候都要手动去编写类似上述的代码,而希望有一个自动生成上述代码的方式。所以xorstr的核心目的就是如果将上面的写法用C++模板自动生成,并且确保总结出的两个核心目标{编译期可以直接算出加密后的结果}和{运行时进行解密}。
两个开源实现如:andrivet/ADVobfuscator(https://github.com/andrivet/ADVobfuscator)以及JustasMasiulis/xorstr(https://github.com/JustasMasiulis/xorstr)
两个实现都稍显复杂,下面写个最小概念代码验证可行性。
**我们需要的使用形式:**我们希望使用起来和正常字符串使用差不多,代码如下:
#define Enc(str) EncryptString(str).decrypt()
printf(Enc("encrypt string?"));
因此我们需要EncryptString返回一个类的实例化对象,然后调用该对象的decrypt成员函数完成解密,代码如下:
template<size_t N>
class EncryptString
{
public:
   template<size_t... Index>
   constexpr EncryptString(const char* plainString, std::index_sequence<Index...>)
      : encBuffer{ (plainString[Index] + 1)... }
  {}

   __forceinline char* decrypt()
  {
       for (auto index = 0; index < N; index++)
      {
           encBuffer[index] -= 1;
      }
       return encBuffer;
  }
private:
   char encBuffer[N];
};

#define Enc(str) EncryptString<sizeof(str)>(str, std::make_index_sequence<sizeof(str)>{}).decrypt()

printf(Enc("conventional...\n"));

可以看到上述代码成功实现了我们的预定目标,实现方式概述:编译期求解字符串数组长度利用模板技术生成一个该字符串数组的加解密类,类的构造函数使用C++变参模板展开和std::index_sequence的方法实现对字符串中每个元素的提取和加密,同时保障上述过程编译时完成。同时解释下为什么这里既然所有字符串的decrypt的逻辑都是一样的,为什么还要把他实现成类的成员函数呢?因为如果设计成统一的外部解密函数会被反编译引用查找的方式找到统一的入口,大大降低分析的难度。

Next

下篇文章:二进制明文字符串加密:还原与反还原(https://bbs.kanxue.com/thread-279309.htm)

看雪ID:Istaroth

https://bbs.kanxue.com/user-home-777136.htm

*本文为看雪论坛优秀文章,由 Istaroth 原创,转载请注明来自看雪社区

# 往期推荐

1、IOFILE exploit入门

2、入门编译原理之前端体验

3、如何用纯猜的方式逆向喜马拉雅xm文件加密(wasm部分)

4、反恶意软件扫描接口(AMSI)如何帮助您防御恶意软件

5、sRDI — Shellcode反射式DLL注入技术

6、对APP的检测以及参数计算分析

球分享

球点赞

球在看


文章来源: http://mp.weixin.qq.com/s?__biz=MjM5NTc2MDYxMw==&mid=2458527425&idx=1&sn=23b47e525f65789aa361bf305b96217b&chksm=b18d164b86fa9f5dbe019e95dd0c026d8da0ad1de2327dbead8c19e66c6a51c546fe246321f9&scene=0&xtrack=1#rd
如有侵权请联系:admin#unsafe.sh