首页
社区
课程
招聘
[翻译]通过Javascript中的CFI实现混淆阻止解密分析
2018-4-6 17:19 7278

[翻译]通过Javascript中的CFI实现混淆阻止解密分析

2018-4-6 17:19
7278

通过Javascript中的CFI实现混淆

能够读懂恶意软件的代码对于恶意软件分析来说是一个绝好的方式,因为它使我们能够了解恶意软件的真实行为。不幸的是,并不是在任何时候我们都能够获得恶意软件的实际代码,比如在分析的时候可能需要使用反汇编工具和调试器来猜测恶意软件的行为。然而,当样本是由解释性语言,比如Java、JS、VBS、.NET等写成时是存在几种方法来阻止获知源代码的。

 

由于攻击者知道安全人员常使用的一些分析技术,因而会使用一些混淆技术来使理解和分析变得困难。存在一种技术可以检测代码是否运行在虚拟机中或者可以保证代码只有在指定的环境下才能运行,并且可以避免调试器链接和反汇编引擎分析。

 

今天的文章就是关于这一点,我想向读者介绍一种有趣的创新方式,用于逃避基于Javascript的逆向工程技术。

 

Javascript作为攻击向量显得日益重要,它的实现受到许多编码风格的影响。但是,几乎所有的Javascript恶意软件都是经过混淆的了。 下图显示了一个经过混淆的JavaScript有效载荷的示例(从我的一个分析中取出)
此处输入图片的描述

 

作为分析的第一步,需要尝试将这些代码消除混淆。从简单的剪切和粘贴开始到使用更强大的替换脚本,分析师会尝试重新命名函数和变量的名称,以分离复杂性并明确每个代码段的功能。在Javascript中,有一个好办法来于获取被调用函数的名称,这个名称可用于了解函数名称是否在随时间进行改变。该函数是arguments.callee.caller。通过使用该函数,攻击者可以创建一个栈回溯其中储存着已执行函数的名称链。攻击者会抓取函数名称并将其用作动态解密特定Javascript代码的密钥。使用这种技术,攻击者会拥有一个隐式的控制流完整性,因为如果函数被重命名或者函数顺序与设计的顺序有所不同,那么产生的hash也将会不同。如果hash值不同,则生成的密钥也会不同,所以就无法解密并加载被加密过的代码。

 

下面的片段解释了这种技术原生的(而不是混淆后的)示例。 我在这里展示未混淆的代码,只是为了简单起见。

var _ = require("underscore");

function keyCharAt(key, i) {
    return key.charCodeAt( Math.floor(i % key.length) );
  }

  function xor_encrypt(key, data) {
    return _.map(data, function(c, i) {
      return c.charCodeAt(0) ^ keyCharAt(key, i);
    });
  }

  function xor_decrypt(key, data) {
    return _.map(data, function(c, i) {
      return String.fromCharCode( c ^ keyCharAt(key, i) );
    }).join("");
  }


function cow001(){
eval(xor_decrypt(arguments.callee.name,[0,0,25,67,95,93,6,65,27,95,87,25,68,34,22,92,89,82,10,0,2,67,16,114,12,1,3,85,94,69,67,59,5,89,87,86,6,29,4,16,120,84,17,10,87,17,23,24]));
}

function pyth001(){
eval(xor_decrypt(arguments.callee.name,[19,22,3,88,0,1,25,89,66]));
}

function pippo(){
pyth001();
}

pippo();

每个内部阶段执行(eval())一个content。在第21行、第25行的cow001和pyth001函数进行xor运行解密content。 xor_decrypt函数接收两个参数:decoding_key和欲被解密的payload。每个内部阶段都使用arguments.callee.name函数将被调用者的名称用作解密密钥。 如果函数名称是攻击者设计的那个(用来加密payload的那个),加密内容将被正常执行没有例外。但是如果函数名称被重命名(比如被分析人员更改),执行函数就会失败,攻击者很有可能会去触发其他的代码路径。

 

在进行实际的野外攻击之前,攻击者需要编写恶意JS并对其进行混淆来准备“攻击路径”。进行混淆之后,攻击者需要使用额外的脚本(如下面的脚本)根据混淆的函数名称对有效载荷进行加密,并将加密好的有效载荷替换为最终的Javascipt文件。

"use strict";  

var _ = require("underscore");

function keyCharAt(key, i) {
    return key.charCodeAt( Math.floor(i % key.length) );
  }

  function xor_encrypt(key, data) {
    return _.map(data, function(c, i) {
      return c.charCodeAt(0) ^ keyCharAt(key, i);
    });
  }

  function xor_decrypt(key, data) {
    return _.map(data, function(c, i) {
      return String.fromCharCode( c ^ keyCharAt(key, i) );
    }).join("");
  }

var final_payload = "console.log('Malicious Content Triggers Here !')";
var k_final = "cow001";
var encrypted_final = xor_encrypt(k_final,final_payload);
var decrypted_final = xor_decrypt(k_final, encrypted_final);
console.log(encrypted_final.toString());
console.log(decrypted_final);

var _1_payload = "cow001();";
var k_1 = "pyth001";
var encrypted_1 = xor_encrypt(k_1,_1_payload);
var decrypted_1 = xor_decrypt(k_1, encrypted_1);
console.log(encrypted_1.toString());
console.log(decrypted_1);

攻击者现在可以编写拥有自己的控制流的Javascript代码。 如果攻击者一遍又一遍地重复这样的概念,他会完成一个完整的逆向工程规避技术来阻止或控制代码执行。


[培训]二进制漏洞攻防(第3期);满10人开班;模糊测试与工具使用二次开发;网络协议漏洞挖掘;Linux内核漏洞挖掘与利用;AOSP漏洞挖掘与利用;代码审计。

收藏
点赞1
打赏
分享
最新回复 (5)
雪    币: 6096
活跃值: (1197)
能力值: (RANK:30 )
在线值:
发帖
回帖
粉丝
CCkicker 2018-4-10 10:37
2
0
原文链接在?
雪    币: 5697
活跃值: (984)
能力值: ( LV9,RANK:400 )
在线值:
发帖
回帖
粉丝
玉涵 5 2018-4-19 10:20
3
0
求师傅给个原文链接
雪    币: 799
活跃值: (452)
能力值: ( LV12,RANK:280 )
在线值:
发帖
回帖
粉丝
Ox9A82 3 2018-4-26 14:12
4
0
哆啦咪 原文链接在?
https://marcoramilli.blogspot.com/2018/02/control-flow-integrity-javascript.html
雪    币: 799
活跃值: (452)
能力值: ( LV12,RANK:280 )
在线值:
发帖
回帖
粉丝
Ox9A82 3 2018-4-26 14:13
5
0
玉涵 求师傅给个原文链接
见楼上
雪    币: 1700
活跃值: (676)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
弱冠甕卿还仓 2019-9-1 10:30
6
0
想念我当初看到的一篇控制流完整性的文章,找不到了,后悔没有右键保存,现在只能看这篇阉割版,真的是气死我了,我记得好像也是这论坛看的,咋找不到了
游客
登录 | 注册 方可回帖
返回