aboutsummaryrefslogtreecommitdiff
path: root/funccrypt.h
blob: 56d193010d386f63aaedb1dac8cbf2fdd71021d2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#ifndef FUNCCRYPT_H
#define FUNCCRYPT_H 1

#include <stdlib.h>
#include <stdint.h>

#if !defined(__GNUC__) || !defined(__GNUC_MINOR__)
#error "This is only verified to work with a GCC compiler!"
#endif

/* Force GCC struct for MingW compilers and pack them,
 * which means the struct is 1-byte aligned.
 */
#define GCC_PACKED __attribute__((packed, gcc_struct))

typedef struct crypt_header {
    uint64_t key;
    uint8_t crpyted;
    uint32_t marker;
    uint64_t func_body[0];
} GCC_PACKED crypt_header;

typedef enum crypt_return {
    CRET_ERROR,     /* Neither prologue marker nor epilogue marker found. */
    CRET_ERROR_MEM, /* set memory page protection failed */
    CRET_PROLOGUE,  /* prologue marker found */
    CRET_EPILOGUE,  /* epilogue marker found */
    CRET_CHECK,     /* all pre (en|de)cryption checks successful */
    CRET_OK_ENC,    /* encryption succeeded */
    CRET_OK_DEC,    /* decryption succeeded */
} crypt_return;

#define CRYPT_FUNC_MAXSIZ 0x100
#define CRYPT_FUNC(fn) \
    crypt_func((void *)fn)
#define CRYPT_RETVAL() __cr
#define CRYPT_PROLOGUE(fn) \
    crypt_return __cr; \
    { \
        __cr = CRYPT_FUNC(fn); \
        if (__cr != CRET_OK_DEC) \
            asm volatile goto("jmp %l0          \n" \
                : : : : cr_epilogue); \
        asm volatile goto("jmp %l0          \n" \
            : : : : cr_prologue); \
        asm volatile( \
            ".byte " \
            /* key: */     "0x00, 0x00, 0x00, 0x00," \
                           "0x00, 0x00, 0x00, 0x00," \
            /* crypted: */ "0x00," \
            /* marker: */  "0xDE, 0xC0, 0xDE, 0xC0;  \n" \
        ); \
    } \
    cr_prologue: {
#define CRYPT_EPILOGUE(fn) \
    } { \
        asm volatile goto("jmp %l0          \n" \
            : : : : cr_epilogue); \
        asm volatile( \
            ".byte " \
            /* Insert encryption pad, so we can find the end marker,
             * while the function body is encrypted.
             * (XOR 64 bit encryption == 8 byte)
             */ \
            "0x00, 0x00, 0x00, 0x00," \
            "0x00, 0x00, 0x00, 0x00," \
            /* marker: */   "0xFE, 0xCA, 0xFE, 0xCA;  \n" \
        ); \
    } \
    cr_epilogue: \
    if (CRYPT_RETVAL() == CRET_OK_DEC) \
        CRYPT_FUNC(fn);

#define CRYPT_FNDEF(name, ...) \
    void name( __VA_ARGS__ ) { \
        CRYPT_PROLOGUE( name );
#define CRYPT_FNEND(name) \
        CRYPT_EPILOGUE(name); \
    }

#ifdef _DEBUG
extern const char *crypt_strs[];
extern void printHexBuf(uint8_t *buf, size_t siz, size_t chars_per_line);
#endif
extern crypt_return crypt_func(void *fn_start);

#endif