Theodosius v3.0
Jit linker, symbol mapper, and obfuscator
next_inst_pass.hpp
Go to the documentation of this file.
1// Copyright (c) 2022, _xeroxz
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are met:
6//
7// 1. Redistributions of source code must retain the above copyright notice,
8// this list of conditions and the following disclaimer.
9//
10// 2. Redistributions in binary form must reproduce the above copyright notice,
11// this list of conditions and the following disclaimer in the documentation
12// and/or other materials provided with the distribution.
13//
14// 3. Neither the name of the copyright holder nor the names of its
15// contributors may be used to endorse or promote products derived from
16// this software without specific prior written permission.
17//
18// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28// POSSIBILITY OF SUCH DAMAGE.
29//
30
31#pragma once
32#include <obf/pass.hpp>
33
34namespace theo::obf {
35/// <summary>
36/// This pass is used to generate transformations and jmp code to change RIP to
37/// the next instruction.
38///
39/// given the following code (get pml4 address from cr3):
40///
41/// get_pml4:
42/// 0: 48 c7 c0 ff 0f 00 00 mov rax,0xfff
43/// 7: 48 f7 d0 not rax
44/// a: 0f 20 da mov rdx,cr3
45/// d: 48 21 c2 and rdx,rax
46/// 10: b1 00 mov cl,0x0
47/// 12: 48 d3 e2 shl rdx,cl
48/// 15: 48 89 d0 mov rax,rdx
49/// 18: c3 ret
50///
51/// this pass will break up each instruction so that it can be anywhere in a
52/// linear virtual address space. this pass will not work on rip relative code,
53/// however clang will not generate such code when compiled with
54/// "-mcmodel=large"
55///
56/// get_pml4@0:
57/// mov rax, 0xFFF
58/// push [next_inst_addr_enc]
59/// xor [rsp], 0x3243342
60/// ; a random number of transformations here...
61/// ret
62/// next_inst_addr_enc:
63/// ; encrypted address of the next instruction goes here.
64///
65/// get_pml4@7:
66/// not rax
67/// push [next_inst_addr_enc]
68/// xor [rsp], 0x93983498
69/// ; a random number of transformations here...
70/// ret
71/// next_inst_addr_enc:
72/// ; encrypted address of the next instruction goes here.
73///
74/// this process is continued for each instruction in the function. the last
75/// instruction "ret" will have no code generated for it as there is no next
76/// instruction.
77///
78///
79/// this pass also only runs at the instruction level, theodosius internally
80/// breaks up functions inside of the ".split" section into individual
81/// instruction symbols. this process also creates a psuedo relocation which
82/// simply tells this pass that there needs to be a relocation to the next
83/// symbol. the offset for these psuedo relocations is zero.
84/// </summary>
85class next_inst_pass_t : public pass_t {
87 xed_state_t istate{XED_MACHINE_MODE_LONG_64, XED_ADDRESS_WIDTH_64b};
88 xed_decoded_inst_zero_set_mode(&m_tmp_inst, &istate);
89 xed_decode(&m_tmp_inst, m_type_inst_bytes, sizeof(m_type_inst_bytes));
90 }
91
92 public:
93 static next_inst_pass_t* get();
94 void run(decomp::symbol_t* sym);
95
96 private:
97 std::optional<recomp::reloc_t*> has_next_inst_reloc(decomp::symbol_t*);
98 xed_decoded_inst_t m_tmp_inst;
99 std::uint8_t m_type_inst_bytes[9] = {0x48, 0xC7, 0x44, 0x24, 0x08,
100 0x44, 0x33, 0x22, 0x11};
101};
102} // namespace theo::obf