问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
CobaltStrike逆向学习系列(6):Beacon sleep_mask 分析
安全工具
只有对工具足够的了解,才能够更好的进行修改
这是\[**信安成长计划**\]的第 6 篇文章 CobaltStrike 提供了一个内存混淆功能,它会在 Sleep 的时候将自身混淆从而避免一定的检测 0x01 C2Profile 分析 ================= 因为 sleep\_mask 是从 C2Profile 中设置的,所以就需要先搞清楚 TeamServer 是如何解析的 data:image/s3,"s3://crabby-images/d2916/d2916490f69edac4f3a2773f95ab822e22e3d338" alt="图片" 很明显它还跟其他的设置项有关,这里我们重点关注一下 rwx 的设置 data:image/s3,"s3://crabby-images/e985b/e985b91747e2fa463b226b344aaee3cc5db48f52" alt="图片" 首先会将 text 段的结尾地址存储到 index=41 的位置 data:image/s3,"s3://crabby-images/9e04a/9e04a9b9b8cbc55c4c87b95fdd6b0618df14c35b" alt="图片" 接着判断了 text 段与 rdata 段中间的空白位置够不够 256 个字节,推测会将加解密函数放在这里 data:image/s3,"s3://crabby-images/d1e16/d1e168f754600717df4754b33a369d7eb9d3b8bd" alt="图片" obfuscate 就让它默认为 false,它不是我们这次关注的重点,接下来就会将 0 和 4096 添加到其中 data:image/s3,"s3://crabby-images/b7919/b7919dc964051ff2bc46c2207e116387d18e51eb" alt="图片" 再看一眼 text 段的 Virtual Address,这块很明显就是 PE 头的那段内容 data:image/s3,"s3://crabby-images/3d8dd/3d8dd8d98dda57ebd2a510459eff4aedf89160aa" alt="图片" 接下来就是循环添加内容了,可以很明显的看到 do while 的循环条件是 text 段且不允许使用 rwx,这也就意味着,当我们不允许使用 rwx 的时候,text 段并不会被添加到 index=42 的项中,应该也就不会被混淆了,在最后还添加了零,用于标示结尾 data:image/s3,"s3://crabby-images/9a9c7/9a9c77fcb49dbd78e6bfcec1bbace0abb7ebffb1" alt="图片" 0x02 set userwx "true" ====================== 为了快速定位到加解密函数所在,可以设置断点来完成,因为按照之前的分析,使用 rwx 的时候,text 段也会被混淆的,所以跟入函数以后,直接滑倒开头下写入断点即可 data:image/s3,"s3://crabby-images/d7425/d742536c829b38792ec331cc3c31de50a2fc1a0f" alt="图片" 接着直接放过也就到达了位置 data:image/s3,"s3://crabby-images/a094e/a094efb180d10a66c621d48dd09ee79c5bd06393" alt="图片" 接着根据这个位置到 IDA 中进行查找,然后通过回溯也很容易能够找到整个的调用链 data:image/s3,"s3://crabby-images/343a4/343a4dd5b511a70e60e017b5f9fc406f44a2bb9a" alt="图片" 在主循环的最后有一个用于处理 Sleep 的函数 data:image/s3,"s3://crabby-images/7691c/7691c08491f5fc329a7b0cf0dbdbe737d7523b71" alt="图片" 它通过判断 0x29 来决定是否直接调用 Sleep,如果使用了 sleep\_mask,0x29 中存储的就是 text 段的结尾地址 data:image/s3,"s3://crabby-images/f3c38/f3c38f0b70f1bcaf9e50b148eb20232416b4d231" alt="图片" 首先它会将自己当前函数与加解密函数一起传入 data:image/s3,"s3://crabby-images/f0660/f0660e1b2e00c41e82a47e1c6ce1d6e860653d7a" alt="图片" 在函数中先计算了两个函数地址的差值,通过对比很明显能够看出这个就是加解密函数的长度 data:image/s3,"s3://crabby-images/6fdbc/6fdbc58787f8eb10fd2f63788a64fa4c9c153f66" alt="图片" 接着从 0x29 中取出 text 段的结尾地址,并从全局变量中取出 PE 头的位置,相加也就得到了在内存中的 text 的结尾地址,也就是后面要存放加解密函数的地址,接着传统的拷贝也就不提了,重点是后面的赋值操作 data:image/s3,"s3://crabby-images/89ed7/89ed7c7092a98d291e37361c27e0b83eef0bcc98" alt="图片" 申请了 0x20 的空间,也就是 32 个字节,8 个字节存储 PE 头地址,8 个字节存储 0x2A 的地址 data:image/s3,"s3://crabby-images/4012d/4012def9879173c6399bdba9993ed21d9d4faf21" alt="图片" 这个 0x2A 也就是之前构造的那一串结构 data:image/s3,"s3://crabby-images/5a04c/5a04c235dcad6be4e3f230cf963ea9523a74a54f" alt="图片" 然后将加 16 的位置传入 data:image/s3,"s3://crabby-images/62984/62984d97fbb30fb5f8187adcadcf6987a59bebc8" alt="图片" 第一个使用 CryptoAPI 生成密钥 data:image/s3,"s3://crabby-images/73840/73840626ed856c320abd74129ca073f2a49f930b" alt="图片" 如果失败的话,用第二个自己实现的算法生成密钥 data:image/s3,"s3://crabby-images/5abad/5abad334fff839fb15e1af25a7e4a46f3eda8789" alt="图片" 最后就来调用加解密函数了,第一个是构造的结构,第二个是 Sleep 函数,第三个是时间 data:image/s3,"s3://crabby-images/3a78c/3a78cd6f2ae08999bcffdb8a99a15726836b01e3" alt="图片" 接着就来分析加解密函数,根据中间 Sleep 调用可以推断出,上面是加密函数,下面是解密函数 data:image/s3,"s3://crabby-images/524e7/524e7d73dc81838a80222bb3a4de105e54ba9d00" alt="图片" 参数是指针类型,取数组,下标 1,也就是 0x2A 所取到的值了 然后 v6 就是第一个值 sectionAddress,v7 就是第二个值 sectionEnd,然后将 v3 再后移 用来判断跳出规则的就是最后添加的两个零 do while 的混淆条件就是 sectionAddress<sectionEnd,也就是将所有的代码都进行混淆 data:image/s3,"s3://crabby-images/2121e/2121e97321407e54f783a1d671db9e5f7edf5c76" alt="图片" 之后的解密也是一样的逻辑就不提了 接下来就是验证的时候了 Sleep 时候的状态 data:image/s3,"s3://crabby-images/7bac6/7bac68221d9a4bf18d7eb1bcaa3ff4f839abc036" alt="图片" 此时的代码段 data:image/s3,"s3://crabby-images/6ae21/6ae21d3d9025e8836c276b3199c530bd66a5b260" alt="图片" 接收命令时候的状态 data:image/s3,"s3://crabby-images/663fb/663fbe2fa4107d63f7e521fc65f4864c225d7bb1" alt="图片" 此时的代码段 data:image/s3,"s3://crabby-images/b9919/b99192c94942cfeffaeb23609e9f0a3a8482eb51" alt="图片" 在允许 RWX 权限的时候,代码段是混淆的 0x03 set userwx "false" ======================= 理论在之前也都讲完了,这里就是验证一下最终的效果 Sleep 时候的状态 data:image/s3,"s3://crabby-images/cd0b9/cd0b9a26549836a70fb0cd318c2323b6d8725d78" alt="图片" 此时的代码段 data:image/s3,"s3://crabby-images/c230f/c230fbca6ce16d1a63e9a10647ef73c6fa5a5a78" alt="图片" 接收命令时的状态 data:image/s3,"s3://crabby-images/72f2e/72f2e02937bd8672db83d1b17c21eabe0f8ae3e7" alt="图片" 此时的代码段 data:image/s3,"s3://crabby-images/b400f/b400f2664f6d1ce41dfc5b15161aaf381c6d23eb" alt="图片" 在禁止 RWX 权限的时候,代码段是不混淆的 [文章首发公众号平台](https://mp.weixin.qq.com/s/56qpeeEOoayGakx0pWbtRg)
发表于 2022-02-14 10:32:59
阅读 ( 6330 )
分类:
安全工具
0 推荐
收藏
0 条评论
请先
登录
后评论
信安成长计划
9 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!