2019 中关村网络与信息安全领域专项赛

2020-06-18  本文已影响0人  SamiraG

参考链接

https://apeng.fun/2019/08/16/2019zgc_quals/
PYC 文件的简单分析
PYC文件格式分析
Python字节码解混淆
Python字节码解混淆之反控制流扁平化

src_leak

题目给了源码如下:

#include<iostream>
using namespace std;

typedef unsigned int uint;


template <bool Flag, class MaybeA, class MaybeB> class IfElse;

template <class MaybeA, class MaybeB>
class IfElse<true, MaybeA, MaybeB> {
public:
    using ResultType = MaybeA;
};

template <class MaybeA, class MaybeB>
class IfElse<false, MaybeA, MaybeB> {
public:
    using ResultType = MaybeB;
};

template <uint N, uint L, uint R> struct func1 {
    enum { mid = (L + R + 1) / 2 };

    using ResultType = typename IfElse<(N < mid * mid),
        func1<N, L, mid - 1>, func1<N, mid, R> >::ResultType;

    enum { result = ResultType::result };
};

template <uint N, uint L> struct func1<N, L, L> { enum { result = L }; };

template <uint N> struct _func1 { enum { result = func1<N, 1, N>::result }; };


template<size_t Input>
constexpr size_t func2 = (Input % 2) + func2< (Input / 2) >;

template<>
constexpr size_t func2<0> = 0;

template<size_t num>
constexpr size_t func3 = num % 2;

template<uint n, uint m>struct NEXTN {
    const static uint value = ((n % m != 0) * n);
};
template<uint n, uint m>struct NEXTM {
    const static uint value = (m * m <= n ? (m + 1) : 0);
};
template<uint n, uint m>struct TEST {
    const static uint value = TEST<NEXTN<n, m>::value, NEXTM<n, m>::value>::value;
};
template<uint m>struct TEST<0, m> {
    const static uint value = 0;
};
template<uint n>struct TEST<n, 0> {
    const static uint value = 1;
};
template<uint n>struct func4 {
    const static uint value = TEST<n, 2>::value;
};
template<>struct func4<1> {
    const static uint value = 0;
};
template<>struct func4<2> {
    const static uint value = 1;
};


int main(int argc, char**argv) {
    //input 5 uint numbers ,x1,x2,x3,x4,x5
    //the sum of them should be MIN


    cout << func3< func2<x1> > << endl;
    cout << func3< func2<x2> > << endl;
    cout << func3< func2<x3> > << endl;
    cout << func3< func2<x4> > << endl;
    cout << func3< func2<x5> > << endl;

    // output: 1 1 1 1 1


    cout << _func1<x1>::result << endl;
    cout << _func1<x2>::result << endl;
    cout << _func1<x3>::result << endl;
    cout << _func1<x4>::result << endl;
    cout << _func1<x5>::result << endl;

    //output: 963 4396 6666 1999 3141

    //how many "1" will func4<1>,func4<2>,fun4<3>......fun4<10000> ::value  return?
    x6 = count;


    // your flag is flag{x1-x2-x3-x4-x5-x6}
    // if x1=1,x2=2,x3=3,x4=4,x5=5,x6=6
    // flag is     flag{1-2-3-4-5-6}

    return 0;
}

模板:
https://zh.cppreference.com/w/cpp/language/template_specialization
模板特化:
https://blog.csdn.net/gatieme/article/details/50953564

所以函数要求: x1-x5都是奇数,而且平方根分别为963 4396 6666 1999 3141, x6是10000以内素数的个数。

pyc

使用010Editor的相应模板打开,并修改pyc.bt中的以下部分:

enum <uint16> MagicValue {
    PY_24a0 = 62041,
    PY_24a3 = 62051,
    PY_24b1 = 62061,
    PY_25a0_1 = 62071,
    PY_25a0_2 = 62081,
    PY_25a0_3 = 62091,
    PY_25a0_4 = 62092,
    PY_25b3_1 = 62101,
    PY_25b3_2 = 62111,
    PY_25c1 = 62121,
    PY_25c2 = 62131,
    PY_26a0 = 62151,
    PY_26a1 = 62161,
    PY_27a0_1 = 62171,
    PY_27a0_2 = 62181,
    PY_27a0_a = 62211,
};
//Include/code.h
typedef struct {
    PyObject_HEAD // 用它来表示一个 PyCodeObject的开始
    int co_argcount;        /* #arguments, except *args */
    int co_nlocals;     /* #local variables */
    int co_stacksize;       /* #entries needed for evaluation stack */
    int co_flags;       /* CO_..., see below */
    PyObject *co_code;      /* instruction opcodes */
    PyObject *co_consts;    /* list (constants used) */
    PyObject *co_names;     /* list of strings (names used) */
    PyObject *co_varnames;  /* tuple of strings (local variable names) */
    PyObject *co_freevars;  /* tuple of strings (free variable names) */
    PyObject *co_cellvars;      /* tuple of strings (cell variable names) */
    /* The rest doesn't count for hash/cmp */
    PyObject *co_filename;  /* string (where it was loaded from) */
    PyObject *co_name;      /* string (name, for reference) */
    int co_firstlineno;     /* first source line number */
    PyObject *co_lnotab;    /* string (encoding addr<->lineno mapping) See
                   Objects/lnotab_notes.txt for details. */
    void *co_zombieframe;     /* for optimization only (see frameobject.c) */
    PyObject *co_weakreflist;   /* to support weakrefs to code objects */
} PyCodeObject;

各个字段的含义如下:

argcount  参数个数
nlocals   局部变量个数
stacksize 栈空间大小
flags     N/A
TYPE_STRING 表示字节码开始
r_long n  字节码的数量
struct Instruction inst[] 字节码序列

PyObject 的序列化在 python/marshal.c 内实现,一般是先写入一个 byte 来标识此 PyObject 的类型,每种 PyObject 对应的类型也在 Python/marshal.c 内定义:

#define TYPE_NULL               '0'
#define TYPE_NONE               'N'
#define TYPE_FALSE              'F'
#define TYPE_TRUE               'T'
#define TYPE_STOPITER           'S'
#define TYPE_ELLIPSIS           '.'
#define TYPE_INT                'i'
#define TYPE_INT64              'I'
#define TYPE_FLOAT              'f'
#define TYPE_BINARY_FLOAT       'g'
#define TYPE_COMPLEX            'x'
#define TYPE_BINARY_COMPLEX     'y'
#define TYPE_LONG               'l'
#define TYPE_STRING             's'
#define TYPE_INTERNED           't'
#define TYPE_STRINGREF          'R'
#define TYPE_TUPLE              '('
#define TYPE_LIST               '['
#define TYPE_DICT               '{'
#define TYPE_CODE               'c'
#define TYPE_UNICODE            'u'
#define TYPE_UNKNOWN            '?'
#define TYPE_SET                '<'
#define TYPE_FROZENSET          '>'

如图type等于99就是TYPE_CODE类型,PyCodeObject 的第一个部分肯定是 TYPE_CODE,表示字节码区块
使用dis模块转化为反编译字节码区块部分:

import dis
class read():
    def __init__(self, filename, mode):
        # print filename, mode
        self.file = open(filename, mode)
    def read(self, num):
        return (self.file).read(num)

pyc = read("test.pyc", "rb")
pyc.read(30)
target = pyc.read(0x31)
# print bytes(target)
dis.dis(target)

结果:

 0 LOAD_CONST          0 (0)    # 读取常量列表中的 0 号常量
 3 MAKE_FUNCTION       0      # 制作一个函数
 6 STORE_NAME          0 (0)  # 函数命名为字符串列表中的 0 号
 9 LOAD_CONST          1 (1)
12 MAKE_FUNCTION       0
15 STORE_NAME          1 (1)
18 LOAD_NAME           0 (0)  # 取出函数 1
21 CALL_FUNCTION       0     # 调用
24 STORE_NAME          2 (2) # 存储返回值
27 LOAD_NAME           1 (1) 
30 CALL_FUNCTION       0
33 STORE_NAME          3 (3)
36 LOAD_NAME           2 (2)   # 取出两个返回值
39 LOAD_NAME           3 (3)
42 BINARY_ADD                 # 相加
43 PRINT_ITEM                 # 打印结果
44 PRINT_NEWLINE
45 LOAD_CONST          2 (2)
48 RETURN_VALUE

混淆的一般思路:



uncompyle 的工作原理和一般的反编译器类似,它会尽力去匹配每一条指令,尝试将所有指令都覆盖到,但是在解析上面的代码时,碰到 load 不存在的常量时就会出错,无法继续反编译。
所有的指令可以分为两类,不需要参数和需要参数的,Python字节码在设计的时候故意把没有参数的指令分配在了对应编号的低位,高位都是有参数的,以Include/opcode.h中的HAVE_ARGUMENT分界。他们的在二进制级别上的组织是这样的:

[指令] 不需要参数的指令只占用一个字节
[指令] [参数低字节] [参数高字节] 需要参数的指令占用三个字节,一个字节指令,两个字节参数
本题中经过混淆的字节码:

>>> list(map(ord,code.co_code[:9]))
[113, 158, 2, 136, 104, 110, 126, 58, 140]
>>> dis.opname[113]
'JUMP_ABSOLUTE'
>>> 2*256+158
670
>>> dis.opname[136]
'LOAD_DEREF'
>>> 110*256+104
28264

从上面可以看到,第一条指令是JUMP_ABSOLTE 670,这个offset的指令是真实存在的,所以指令合法。但是第二条指令应该是LOAD_DEREF 28264,这个index的对象并不存在,在dis尝试解析的时候就会崩溃。

实际上因为之前的跳转指令所以第二条的非法指令并不会被真实执行到,所以pyc文件作者是故意加入不影响执行的非法指令触发分析软件崩溃,阻碍对该pyc文件的分析。

去混淆

用Apeng师傅的方法可以去掉混淆

# python2 disasm_anti.py py.pyc
import dis, marshal, struct, sys, time, types
from opcode import *

def ana_branch(code, i, hits):
    if i > len(code):
        return
    if i in hits:
        return
    else:
        hits.append(i)
    c = code[i]
    op = ord(c)
    if op == 111 or op == 112 or op == 114 or op == 115 or op == 120 or op == 93:
        oparg = ord(code[i+1]) + ord(code[i+2])*256
        if op == 120 or op == 93:
            oparg += i
            oparg += 3
        ana_branch(code, oparg, hits)
        ana_branch(code, i+3, hits)
    elif op == 110:
        oparg = ord(code[i+1]) + ord(code[i+2])*256
        ana_branch(code, i + oparg + 3, hits)
    elif op == 113:
        oparg = ord(code[i+1]) + ord(code[i+2])*256
        ana_branch(code, oparg, hits)
    else:
        if op>=HAVE_ARGUMENT:
            ana_branch(code, i+3, hits)
        else:
            ana_branch(code, i+1, hits)

def findlinestarts(code):
    """Find the offsets in a byte code which are start of lines in the source.

    Generate pairs (offset, lineno) as described in Python/compile.c.

    """
    byte_increments = [ord(c) for c in code.co_lnotab[0::2]]
    line_increments = [ord(c) for c in code.co_lnotab[1::2]]

    lastlineno = None
    lineno = code.co_firstlineno
    addr = 0
    for byte_incr, line_incr in zip(byte_increments, line_increments):
        if byte_incr:
            if lineno != lastlineno:
                yield (addr, lineno)
                lastlineno = lineno
            addr += byte_incr
        lineno += line_incr
    if lineno != lastlineno:
        yield (addr, lineno)

def findhits(code):
    hits = []
    n = len(code)
    i = 0
    ana_branch(code, i, hits)
    hits.sort()
    return hits

def anti_findlabels(code):
    """Detect all offsets in a byte code which are jump targets.
    Return the list of offsets.
    """
    hits = findhits(code)
    labels = []
    n = len(code)
    i = 0
    while i < n:
        if i not in hits:
            i+=1
            continue
        c = code[i]
        op = ord(c)
        i = i+1
        if op >= HAVE_ARGUMENT:
            oparg = ord(code[i]) + ord(code[i+1])*256
            i = i+2
            label = -1
            if op in hasjrel:
                label = i+oparg
            elif op in hasjabs:
                label = oparg
            if label >= 0:
                if label not in labels:
                    labels.append(label)
    return labels

def dis_anti_obf(co, lasti = -1):
    """Disassemble a code object, anti obf"""
    anti_code = ""
    code = co.co_code
    hits = findhits(code)
    labels = anti_findlabels(code)
    linestarts = dict(findlinestarts(co))
    n = len(code)
    i = 0
    i = 0
    extended_arg = 0
    free = None
    while i < n:
        if i not in hits:
            i+=1
            anti_code+="\x09"
            continue
        c = code[i]
        op = ord(c)
        if i in linestarts:
            if i > 0:
                print
            print "%3d" % linestarts[i],
        else:
            print '   ',
        
        if i == lasti: print '-->',
        else: print '   ',
        if i in labels: print '>>',
        else: print '  ',
        print repr(i).rjust(4),
        print opname[op].ljust(20),
        anti_code += code[i]
        i = i+1
        if op >= HAVE_ARGUMENT:
            oparg = ord(code[i]) + ord(code[i+1])*256 + extended_arg
            extended_arg = 0
            anti_code+=code[i]
            anti_code+=code[i+1]
            i = i+2
            if op == EXTENDED_ARG:
                extended_arg = oparg*65536L
            print repr(oparg).rjust(5),
            if op in hasconst:
                print '(' + repr(co.co_consts[oparg]) + ')',
            elif op in hasname:
                print '(' + co.co_names[oparg] + ')',
            elif op in hasjrel:
                print '(to ' + repr(i + oparg) + ')',
            elif op in haslocal:
                print '(' + co.co_varnames[oparg] + ')',
            elif op in hascompare:
                print '(' + cmp_op[oparg] + ')',
            elif op in hasfree:
                if free is None:
                    free = co.co_cellvars + co.co_freevars
                print '(' + free[oparg] + ')',
        print
    print "patch code:"
    print(anti_code.encode("hex"))




def show_file(fname):
    f = open(fname, "rb")
    magic = f.read(4)
    moddate = f.read(4)
    modtime = time.asctime(time.localtime(struct.unpack('L', moddate)[0]))
    print "magic %s" % (magic.encode('hex'))
    print "moddate %s (%s)" % (moddate.encode('hex'), modtime)
    code = marshal.load(f)
    show_code(code)
     
def show_code(code, indent=''):
    print "%scode" % indent
    indent += '   '
    print "%sargcount %d" % (indent, code.co_argcount)
    print "%snlocals %d" % (indent, code.co_nlocals)
    print "%sstacksize %d" % (indent, code.co_stacksize)
    print "%sflags %04x" % (indent, code.co_flags)
    show_hex("code", code.co_code, indent=indent)
    dis_anti_obf(code)
    print "%sconsts" % indent
    for const in code.co_consts:
        if type(const) == types.CodeType:
            show_code(const, indent+'   ')
        else:
            print "   %s%r" % (indent, const)
    print "%snames %r" % (indent, code.co_names)
    print "%svarnames %r" % (indent, code.co_varnames)
    print "%sfreevars %r" % (indent, code.co_freevars)
    print "%scellvars %r" % (indent, code.co_cellvars)
    print "%sfilename %r" % (indent, code.co_filename)
    print "%sname %r" % (indent, code.co_name)
    print "%sfirstlineno %d" % (indent, code.co_firstlineno)
    show_hex("lnotab", code.co_lnotab, indent=indent)
     
def show_hex(label, h, indent):
    h = h.encode('hex')
    if len(h) < 60:
        print "%s%s %s" % (indent, label, h)
    else:
        print "%s%s" % (indent, label)
        for i in range(0, len(h), 60):
            print "%s   %s" % (indent, h[i:i+60])

show_file(sys.argv[1])

或者使用FXTi师傅的方法将不合法的指令转换为nop:

import marshal, sys, opcode, types, dis

NOP = 9

HAVE_ARGUMENT = 90

JUMP_FORWARD = 110
JUMP_IF_FALSE_OR_POP = 111
JUMP_IF_TRUE_OR_POP = 112
JUMP_ABSOLUTE = 113
POP_JUMP_IF_FALSE = 114
POP_JUMP_IF_TRUE = 115

CONTINUE_LOOP = 119
FOR_ITER = 93

RETURN_VALUE = 83

used_set = set()

def deconf_inner(code, now):
    global used_set

    while code[now] != RETURN_VALUE:
        if now in used_set:
            break
        used_set.add(now)
        if code[now] >= HAVE_ARGUMENT:
            used_set.add(now+1)
            used_set.add(now+2)
        op = code[now]

        #print(str(now) + " " + opcode.opname[op])

        if op == JUMP_FORWARD:
            arg = code[now+2] << 8 | code[now+1]
            now += arg + 3
            continue

        elif op == JUMP_ABSOLUTE:
            arg = code[now+2] << 8 | code[now+1]
            now = arg
            continue

        elif op == JUMP_IF_TRUE_OR_POP:
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, arg)

        elif op == JUMP_IF_FALSE_OR_POP:
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, arg)

        elif op == POP_JUMP_IF_TRUE:
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, arg)

        elif op == POP_JUMP_IF_FALSE: 
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, arg)

        elif op == CONTINUE_LOOP:
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, arg)

        elif op == FOR_ITER: 
            arg = code[now+2] << 8 | code[now+1] 
            deconf_inner(code, now + arg + 3)

        if op < HAVE_ARGUMENT:
            now += 1
        else:
            now += 3

    used_set.add(now)
    if code[now] >= HAVE_ARGUMENT:
        used_set.add(now+1)
        used_set.add(now+2)

def deconf(code):
    global used_set

    used_set = set() #Remember to clean up used_set for every target function

    cod = list(map(ord, code))
    deconf_inner(cod, 0)

    for i in range(len(cod)):
        if i not in used_set:
            cod[i] = NOP

    return "".join(list(map(chr, cod)))

with open(sys.argv[1], 'rb') as f:
    header = f.read(8)
    code = marshal.load(f)

'''
print(code.co_consts[3].co_name)
print(dis.dis(deconf(code.co_consts[3].co_code)))
'''

consts = list()

for i in range(len(code.co_consts)):
    if hasattr(code.co_consts[i], 'co_code'):
        consts.append(types.CodeType(code.co_consts[i].co_argcount,
            # c.co_kwonlyargcount,  Add this in Python3
            code.co_consts[i].co_nlocals,
            code.co_consts[i].co_stacksize,
            code.co_consts[i].co_flags,
            deconf(code.co_consts[i].co_code),
            code.co_consts[i].co_consts,
            code.co_consts[i].co_names,
            code.co_consts[i].co_varnames,
            code.co_consts[i].co_filename,
            code.co_consts[i].co_name,
            code.co_consts[i].co_firstlineno,
            code.co_consts[i].co_lnotab,   # In general, You should adjust this
            code.co_consts[i].co_freevars,
            code.co_consts[i].co_cellvars))
    else:
        consts.append(code.co_consts[i])

mode = types.CodeType(code.co_argcount,
    # c.co_kwonlyargcount,  Add this in Python3
    code.co_nlocals,
    code.co_stacksize,
    code.co_flags,
    deconf(code.co_code),
    tuple(consts),
    code.co_names,
    code.co_varnames,
    code.co_filename,
    code.co_name,
    code.co_firstlineno,
    code.co_lnotab,   # In general, You should adjust this
    code.co_freevars,
    code.co_cellvars)

f = open(sys.argv[1]+".mod", 'wb') 
f.write(header)
marshal.dump(mode, f)

NOP指令去除与块的合并

fxti师傅给了一个工具:
https://github.com/extremecoders-re/bytecode_simplifier
利用这个工具可以去除上面去混淆以后生成的全是NOP指令的块,以及合并一些块
完成之后使用uncompyle6可以反汇编一下(反编译还是会失败)

 L.   1         0  LOAD_CONST               2694209818L
                3  STORE_FAST            0  'DIVIDER'
                6  JUMP_FORWARD          0  'to 9'
              9_0  COME_FROM             6  '6'
                9  LOAD_CONST               4130330538L

 L.   3        12  LOAD_FAST             0  'DIVIDER'
               15  COMPARE_OP            2  ==
               18  POP_JUMP_IF_TRUE    366  'to 366'

 L.   9        21  LOAD_CONST               3168701571L
               24  LOAD_FAST             0  'DIVIDER'
               27  COMPARE_OP            2  ==

 L.  14        30  POP_JUMP_IF_TRUE    345  'to 345'
               33  LOAD_CONST               3715947653L
               36  LOAD_FAST             0  'DIVIDER'

 L.  17        39  COMPARE_OP            2  ==
               42  POP_JUMP_IF_TRUE    324  'to 324'
               45  LOAD_CONST               2694209818L

 L.  20        48  LOAD_FAST             0  'DIVIDER'
               51  COMPARE_OP            2  ==
               54  POP_JUMP_IF_TRUE    306  'to 306'

 L.  38        57  LOAD_CONST               651787064
               60  LOAD_FAST             0  'DIVIDER'
               63  COMPARE_OP            2  ==
               66  POP_JUMP_IF_TRUE    285  'to 285'
               69  LOAD_CONST               3521152606L
               72  LOAD_FAST             0  'DIVIDER'
               75  COMPARE_OP            2  ==
               78  POP_JUMP_IF_TRUE    264  'to 264'
               81  LOAD_CONST               2730391645L
               84  LOAD_FAST             0  'DIVIDER'
               87  COMPARE_OP            2  ==
               90  POP_JUMP_IF_TRUE    246  'to 246'
               93  LOAD_CONST               4084147187L
               96  LOAD_FAST             0  'DIVIDER'
               99  COMPARE_OP            2  ==
              102  POP_JUMP_IF_TRUE    222  'to 222'
              105  LOAD_CONST               1860581437
              108  LOAD_FAST             0  'DIVIDER'
              111  COMPARE_OP            2  ==
              114  POP_JUMP_IF_TRUE    204  'to 204'
              117  LOAD_CONST               3816944324L
              120  LOAD_FAST             0  'DIVIDER'
              123  COMPARE_OP            2  ==
              126  POP_JUMP_IF_TRUE    203  'to 203'
              129  LOAD_CONST               394367122
              132  LOAD_FAST             0  'DIVIDER'
              135  COMPARE_OP            2  ==
              138  POP_JUMP_IF_TRUE    181  'to 181'
              141  LOAD_CONST               1627830889
              144  LOAD_FAST             0  'DIVIDER'
              147  COMPARE_OP            2  ==
              150  POP_JUMP_IF_TRUE    157  'to 157'
              153  LOAD_CONST               None
              156  RETURN_END_IF    
            157_0  COME_FROM           150  '150'
              157  STORE_NAME            0  'sys'
              160  LOAD_CODE                <code_object str2hex>
              163  MAKE_FUNCTION_0       0  None
              166  STORE_NAME            1  'str2hex'
              169  LOAD_CODE                <code_object hex2str>
              172  LOAD_CONST               3715947653L
              175  STORE_FAST            0  'DIVIDER'
              178  JUMP_BACK             9  'to 9'
              181  LOAD_NAME            10  'flag'
              184  CALL_FUNCTION_1       1  None
              187  CALL_FUNCTION_1       1  None
              190  POP_TOP          
              191  LOAD_CONST               None
              194  LOAD_CONST               3816944324L
              197  STORE_FAST            0  'DIVIDER'
              200  JUMP_BACK             9  'to 9'
              203  RETURN_END_IF    
            204_0  COME_FROM           114  '114'
              204  LOAD_CONST               97
              207  LOAD_CONST               103
              210  LOAD_CONST               58
              213  LOAD_CONST               2730391645L
              216  STORE_FAST            0  'DIVIDER'
              219  JUMP_BACK             9  'to 9'
              222  LOAD_ATTR             6  'stdout'
              225  LOAD_ATTR             7  'write'
              228  LOAD_NAME             2  'hex2str'
              231  LOAD_CONST               102
              234  LOAD_CONST               108
              237  LOAD_CONST               1860581437
              240  STORE_FAST            0  'DIVIDER'
              243  JUMP_BACK             9  'to 9'
              246  BUILD_LIST_5          5 
              249  CALL_FUNCTION_1       1  None
              252  CALL_FUNCTION_1       1  None
              255  LOAD_CONST               4130330538L
              258  STORE_FAST            0  'DIVIDER'
              261  JUMP_BACK             9  'to 9'
              264  CALL_FUNCTION_1       1  None
              267  STORE_NAME           10  'flag'
              270  LOAD_NAME             5  'count'
              273  LOAD_NAME             1  'str2hex'
              276  LOAD_CONST               394367122
              279  STORE_FAST            0  'DIVIDER'
              282  JUMP_BACK             9  'to 9'
              285  STORE_NAME            3  'p_s'
              288  LOAD_CODE                <code_object p_f>
              291  MAKE_FUNCTION_0       0  None
              294  STORE_NAME            4  'p_f'
              297  LOAD_CONST               3168701571L
              300  STORE_FAST            0  'DIVIDER'
              303  JUMP_BACK             9  'to 9'
              306  LOAD_CONST               -1
              309  LOAD_CONST               None
              312  IMPORT_NAME           0  'sys'
              315  LOAD_CONST               1627830889
              318  STORE_FAST            0  'DIVIDER'
              321  JUMP_BACK             9  'to 9'
              324  MAKE_FUNCTION_0       0  None
              327  STORE_NAME            2  'hex2str'
              330  LOAD_CODE                <code_object p_s>
              333  MAKE_FUNCTION_0       0  None
              336  LOAD_CONST               651787064
              339  STORE_FAST            0  'DIVIDER'
              342  JUMP_BACK             9  'to 9'
              345  LOAD_CODE                <code_object count>
              348  MAKE_FUNCTION_0       0  None
              351  STORE_NAME            5  'count'
              354  LOAD_NAME             0  'sys'
              357  LOAD_CONST               4084147187L
              360  STORE_FAST            0  'DIVIDER'
              363  JUMP_BACK             9  'to 9'
              366  POP_TOP          
              367  LOAD_NAME             0  'sys'
              370  LOAD_ATTR             8  'stdin'
              373  LOAD_ATTR             9  'read'
              376  LOAD_CONST               38
              379  LOAD_CONST               3521152606L
              382  STORE_FAST            0  'DIVIDER'
              385  JUMP_BACK             9  'to 9'

然后我暂时只会看这个字节码,fxti师傅那个自己去混淆的方案没开源,我打算自己仔细研究一下这个发编译的代码,看能不能按照那个思路复现一下。

上一篇 下一篇

猜你喜欢

热点阅读