首页 > 编程语言 >记一次unicorn半自动化逆向——还原某东sign算法

记一次unicorn半自动化逆向——还原某东sign算法

时间:2024-11-27 11:59:51浏览次数:10  
标签:arr sub R2 text 某东 sign mu 半自动化 ADDR

分析准备

集成so文件

为了方便分析和调试,我选择了主动集成生成sign的so文件到自己写的apk中,然后主动调用。

可能是gradle版本的问题,搜索到的文章大都无效,踩坑十分多。

其实配置很简单,在src/main/目录下建立jniLibs/armeabi-v7a/目录,把so文件放在里面。

然后配置好build.gradleCMakeLists.txt就行了,当然,不同的gradle版本会有不同的问题,自己多搜索折腾下就好了。

image-20210309113547443

image-20210309113739384

以及so文件中会有两处简单的校验,直接nop两条跳转指令就解决了,十分简单,就不多说了。

总加密流程

总的加密函数执行流程:

sub_127E4——>sub_126AC——>sub_18B8——>sub_227C

核心的加密算法都在sub_126AC中,传入待加密字符串,待加密字符串长度和两个随机值,

sub_126AC会返回加密后的字节,然后sub_18B8进行base64加密,最后sub_227C会进行标准的md5加密。

sub_126AC——加密选择

sub_126AC只是加密的入口,会根据传入的两个随机数选择加密方式和相应的key。

image-20210309110812935

我们定义三种加密方式为version0,version1和version2,其中version0和version1加密流程全部一样,只不过里面函数传入的常量version0是32,version1是16,我们在分析过程中以Version0为例。

Version0 加密

执行流程

加密函数执行流程:

sub_10E4C——>sub_125F0——>sub_12580——>sub_10EA4——>sub_10D70

待加密字符串会被每8个字节分为一组传进sub_10EA4进行加密,如果最后还有字节剩余,会单独进入sub_10D70加密。

sub_10EA4比特位初始化

首先看下sub_10EA4函数的流程图,先进行初始化,接着有一个循环,循环次数是传入的key0的长度。

image-20210308171518031

最上面的两个长框框就是初始化过程,做了什么呢,IDA进行f5后比较简洁,我们截取一部分看下。

image-20210308173335943

我们注意到分别传入的8个字节分别和0x80(b'1000 0000'),0x40(b'0100 0000'),0x20(b'0010 0000'),0x10(b'0001 0000'),0x8(b’0000 1000‘),0x4(b'0000 0100'),0x2(b'0000 0010'),0x1(b'0000 0001')进行与操作,然后赋值给一些变量,我们观察这些与操作的对象,会发现其实很有规律,这些变量正是输入的8个字节的64个比特位,后面会进行打乱比特位然后还原。

sub_10EA4打乱比特位

然后我们来看流程图中很有规律的十六个小框框,

image-20210308174216954

截取一些IDA进行f5后的片段进行分析。

image-20210308174746290

image-20210308174524110

其中key0_i是key0的第i个字符,在循环中key0_i也会和0x80(b'1000 0000'),0x40(b'0100 0000'),0x20(b'0010 0000'),0x10(b'0001 0000'),0x8(b’0000 1000‘),0x4(b'0000 0100'),0x2(b'0000 0010'),0x1(b'0000 0001')进行与操作,结果作为条件判断,也就是判断key0_i二进制的对应比特为是0还是1。然后会进入对应的分支交换初始化过程中得到的变量,也就是打乱比特位。

sub_10EA4比特位复原

在函数最后进行的是比特位复原,四轮循环,每轮还原两个字节。

image-20210308175646222

image-20210308175711322

如有不清楚的地方,可以自行调试观察。

使用unicorn还原sub_10EA4算法

我们怎么还原sub_10EA4中的算法呢,过程不难就是量多,难道真的要一点点对比着IDA的f5反编译手动写出一样的逻辑吗?当然不是,这里有更简单的方法,我们可以借助unicorn来快速还原。

我们已经分析出来sub_10EA4算法做的不过是打乱传入八个字节的比特位,生成新的八个字节,也就说总的比特位只是顺便变了,并没有被改变值,那么我们可不可以找到每个比特位在打乱前和被打乱后的映射关系呢?当然可以。

我们只要借助unicorn,控制sub_10EA4函数输入的八个字节的64个二进制比特位,如果64个比特中只有一个是1,那么结果的比特位中会有几个1呢?也是只有一个,然后计算前和计算后比特位的1的索引位置就是要找的映射关系,我们进行64次计算,然后每次计算1的索引位置不同,最后就能得到全部比特位的计算前和计算后的映射关系了。

篇幅所限,unicorn的使用就不多说了。

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

from unicorn import *

from unicorn.arm_const import *

table = []

def bytes2bin(bytes):

    arr = []

    for in [m for in bytes]:

        arr.append(

            [(v & 128) >> 7, (v & 64) >> 6, (v & 32) >> 5, (v & 16) >> 4, (v & 8) >> 3, (v & 4) >> 2, (v & 2) >> 1,

             v & 1])

    return [i for in arr for in j]

def bin2bytes(arr):

    length = len(arr) // 8

    arr1 = [0 for in range(length)]

    for in range(length):

        arr1[j] = arr[j * 8] << 7 | arr[j * 8 + 1] << 6 | arr[j * 8 + 2] << 5 | arr[j * 8 + 3] << 4 | arr[

            * 8 + 4] << 3 | arr[j * 8 + 5] << 2 | arr[j * 8 + 6] << 1 | arr[j * 8 + 7]

    return bytes(arr1)

def read(name):

    with open(name, 'rb') as f:

        return f.read()

def hook_code(mu, address, size, user_data):

    if address == BASE + 0x119cc:

        arr2 = []

        for byte in mu.mem_read(PLAINTEXT_ADDR, 8):

            arr2.append(byte)

        table.append([user_data.index(1), bytes2bin(arr2).index(1)])

if __name__ == "__main__":

    key0 = b'44e715a6e322ccb7d028f7a42fa55040'

    mu = Uc(UC_ARCH_ARM, UC_MODE_THUMB)

    BASE = 0x400000

    STACK_ADDR = 0x0

    STACK_SIZE = 1024

    PLAINTEXT_ADDR = 1024 * 2

    PLAINTEXT_SIZE = 1024

    KEY_ADDR = 1024 * 3

    KEY_SIZE = 1024

    mu.mem_map(BASE, 1024 * 1024)

    mu.mem_map(STACK_ADDR, STACK_SIZE)

    mu.mem_map(PLAINTEXT_ADDR, PLAINTEXT_SIZE)

    mu.mem_map(KEY_ADDR, KEY_SIZE)

    mu.reg_write(UC_ARM_REG_SP, STACK_ADDR + STACK_SIZE - 1)

    # mu.mem_write(BASE, read("F:\\Code\\Pycharm\\JDSign\\libjdbitmapkit.so"))

    mu.mem_write(BASE, read("./libjdbitmapkit.so"))

    mu.mem_write(KEY_ADDR, key0)

    for in range(64):

        arr1 = [0 for in range(64)]

        arr1[i] = 1

        = mu.hook_add(UC_HOOK_CODE, hook_code, arr1)

        mu.mem_write(PLAINTEXT_ADDR, bin2bytes(arr1))

        mu.reg_write(UC_ARM_REG_R1, KEY_ADDR)

        mu.reg_write(UC_ARM_REG_R2, 32)

        mu.reg_write(UC_ARM_REG_R3, PLAINTEXT_ADDR)

        mu.emu_start(BASE + 0x00010EA4 + 1, BASE + 0x000119D0)

        mu.hook_del(h)

    print(table)

可以得到映射表,例如[1,4]表示64个比特位在打乱后第1个位置的比特位会到第4个位置。

1

[[00], [14], [261], [315], [456], [540], [66], [759], [862], [958], [1017], [112], [1212], [138], [1432], [1560], [1613], [1745], [1834], [1914], [2036], [2121], [2222], [2339], [2423], [2525], [2626], [2720], [281], [2933], [3046], [3155], [3235], [3324], [3457], [3519], [3653], [3737], [3838], [395], [4030], [4141], [4242], [4318], [4447], [4527], [469], [4744], [4851], [497], [5049], [5163], [5228], [5343], [5454], [5552], [5631], [5710], [5829], [5911], [603], [6116], [6250], [6348]]

然后可轻松还原sub_10EA4算法。

1

2

3

4

5

6

7

8

9

10

11

12

13

def sub_10EA4(input):

    table = [[00], [14], [261], [315], [456], [540], [66], [759], [862], [958], [1017], [112],

             [1212], [138], [1432], [1560], [1613], [1745], [1834], [1914], [2036], [2121],

             [2222], [2339], [2423], [2525], [2626], [2720], [281], [2933], [3046], [3155],

             [3235], [3324], [3457], [3519], [3653], [3737], [3838], [395], [4030], [4141],

             [4242], [4318], [4447], [4527], [469], [4744], [4851], [497], [5049], [5163], [5228],

             [5343], [5454], [5552], [5631], [5710], [5829], [5911], [603], [6116], [6250],

             [6348]]

    arr = bytes2bin(input)

    arr1 = [0 for in range(len(arr))]

    for in range(len(table)):

        arr1[table[i][1]] = arr[table[i][0]]

    return bin2bytes(arr1)

sub_10D70函数分析

sub_10D70函数在IDA反编译也比较清晰,我们来看下。

image-20210308212310967

前面有说,待加密字符串会被每8个字节分为一组传进sub_10EA4进行加密,sub_10D70会加密最后余出来的几个字节,这6个case就是加密余出来的1到7个字节。case 0到case 6对应的六个函数都是一个模板,我们以case 0为例来分析,即sub_4B7C。

sub_4B7C初始化

sub_4B87只会加密处理一个字节,首先我们来看下sub_4B7C的流程图。

image-20210308212954647

可以看到密密麻麻,其他五个case也都长这样,看到后首先第一个反应就是,这是ollvm吗?在经过仔细的分析以及动态调试之后,我判断这个并不是ollvm,没有看到控制流平坦化会带有的标志性大量的常量,也没有找到不可到达的分支,虚假控制流以及指令替换的痕迹,当然也可能是我水平太低了,没有认出来这种ollvm,总之,我还是铁着头把整个流程从头到尾看了一遍。

来看下初始化,

image-20210308214035243

看样子有点像sub_10EA4,但仔细一看又很多不一样,上半部分取了传入的一个字节的比特位,还做了些其他的计算操作,赋值给变量。

而下半部分呢,则是取一些变量的地址,然后放到另一些变量地址加偏移处,这些说起来可能很模糊,但是如果看一下sub_4B7C的栈空间就会清晰很多。

为了方便理解,我们把这样连着五个变量在一起的当作一个数组,这样的数组往下拉可以看到是有八个,每个数组的前四个位置都存放着其他数组首的地址,而第五个位置则存放着前面说的输入字节的比特位经过计算后的值,其实分析后面的case就会发现,有多少输入的比特位,就会有多少个这样的数组。

image-20210308214910555

初始化就这么多,而后面就开始根据这些东西进行大量的循环计算。

sub_4B7C计算分析

image-20210308215648511

整个循环是根据key0的每个字节的每个比特位作为判断条件来选择分支,然后进行循环的,所有的计算类型只有两种,就是下面的两个大方框,是两个小循环。

它们的汇编如下。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

.text:00005BE2 loc_5BE2                                ; CODE XREF: sub_4B7C+104↑j

.text:00005BE2                 CMP             R1, R2

.text:00005BE4                 MOV             R12, R1

.text:00005BE6                 BEQ             loc_5C00

.text:00005BE8                 MOV             R6, R1

.text:00005BEA                 MOV             R8, R2

.text:00005BEC                 B               loc_5BF0

.text:00005BEE ---------------------------------------------------------------------------

.text:00005BEE

.text:00005BEE loc_5BEE                                ; CODE XREF: sub_4B7C+1082↓j

.text:00005BEE                 MOV             R12, R6

.text:00005BF0

.text:00005BF0 loc_5BF0                                ; CODE XREF: sub_4B7C+1070↑j

.text:00005BF0                 LDRB            R6, [R6,#0x10]

.text:00005BF2                 STRB.W          R6, [R8,#0x10]

.text:00005BF6                 MOV             R8, R12

.text:00005BF8                 LDR.W           R6, [R12,#0xC]

.text:00005BFC                 CMP             R6, R2

.text:00005BFE                 BNE             loc_5BEE

.text:00005C00

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

text:00005C00                 CMP             R11, R2

.text:00005C02                 STRB.W          R4, [R12,#0x10]

.text:00005C06                 MOV             R4, R11

.text:00005C08                 LDRB.W          R8, [SP,#0xE0+var_98]

.text:00005C0C                 IT NE

.text:00005C0E                 MOVNE           R12, R2

.text:00005C10                 BNE             loc_5C16

.text:00005C12                 B               loc_5C2C

.text:00005C14 ---------------------------------------------------------------------------

.text:00005C14

.text:00005C14 loc_5C14                                ; CODE XREF: sub_4B7C+10AE↓j

.text:00005C14                 MOV             R4, R6

.text:00005C16

.text:00005C16 loc_5C16                                ; CODE XREF: sub_4B7C+1094↑j

.text:00005C16                 LDRB            R6, [R4,#0x10]

.text:00005C18                 RSBS.W          R6, R6, #1

.text:00005C1C                 IT CC

.text:00005C1E                 MOVCC           R6, #0

.text:00005C20                 STRB.W          R6, [R12,#0x10]

.text:00005C24                 LDR             R6, [R4,#4]

.text:00005C26                 MOV             R12, R4

.text:00005C28                 CMP             R6, R2

.text:00005C2A                 BNE             loc_5C14

这两个小循环做了什么呢?其实十分简单,就是不断地查刚才我们定义的数组的前四个位置存放的变量地址值,判断是否等于另一个变量的地址值,只不过判断的过程中会不断地移动这些数组的第五个位置的值。

使用unicorn还原case0——sub_4B7C算法

这个算法似乎很难搞,确实很难搞,不同于sub_10EA4的对输入字节的比特位进行简单的交换,在获取了比特位后又进行了很多计算。

该怎么办呢,我们来看下sub_4B7C算法的最后部分。

image-20210308220953273

可以看到和sub_10EA4函数最后的比特位还原部分几乎是一样的,这时候我们进行下大胆的猜测,sub_4B7C函数其实也是打乱输入字节的比特位进行了还原,只不过比特位在打乱后还进行了计算,而且每个比特位进行计算的规则都是一样的。即输入的字节第x个比特位是0的话,打乱计算还原后,第y个比特位会是m;如果输入的字节第x个比特位是1的话,第y个比特位则会是n,(x, 0)——>(y,m)(x,1)——>(y,n)。后面经过实践,证明了这样的猜想是正确的。

然后我们还是可以借助unicorn来找到所有的映射关系,还原sub_4B7C算法。思路我们进行下改变,我们可以unicorn控制输入字节的比特位,用八个比特位只有一个1的计算结果和八个比特位全是0的计算结果进行对比从而得到所有的映射关系。

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

from unicorn import *

from unicorn.arm_const import *

table = []

table1 = []

def bytes2bin(bytes):

    arr = []

    for in [m for in bytes]:

        arr.append(

            [(v & 128) >> 7, (v & 64) >> 6, (v & 32) >> 5, (v & 16) >> 4, (v & 8) >> 3, (v & 4) >> 2, (v & 2) >> 1,

             v & 1])

    return [i for in arr for in j]

def bin2bytes(arr):

    length = len(arr) // 8

    arr1 = [0 for in range(length)]

    for in range(length):

        arr1[j] = arr[j * 8] << 7 | arr[j * 8 + 1] << 6 | arr[j * 8 + 2] << 5 | arr[j * 8 + 3] << 4 | arr[

            * 8 + 4] << 3 | arr[j * 8 + 5] << 2 | arr[j * 8 + 6] << 1 | arr[j * 8 + 7]

    return bytes(arr1)

def read(name):

    with open(name, 'rb') as f:

        return f.read()

def hook_code(mu, address, size, user_data):

    if address == BASE + 0x5284:

        table.append(bytes2bin(mu.mem_read(PLAINTEXT_ADDR, 1)))

if __name__ == "__main__":

    key0 = b'44e715a6e322ccb7d028f7a42fa55040'

    mu = Uc(UC_ARCH_ARM, UC_MODE_THUMB)

    BASE = 0x400000

    STACK_ADDR = 0x0

    STACK_SIZE = 1024

    PLAINTEXT_ADDR = 1024 * 2

    PLAINTEXT_SIZE = 1024

    KEY_ADDR = 1024 * 3

    KEY_SIZE = 1024

    mu.mem_map(BASE, 1024 * 1024)

    mu.mem_map(STACK_ADDR, STACK_SIZE)

    mu.mem_map(PLAINTEXT_ADDR, PLAINTEXT_SIZE)

    mu.mem_map(KEY_ADDR, KEY_SIZE)

    mu.reg_write(UC_ARM_REG_SP, STACK_ADDR + STACK_SIZE - 1)

    # mu.mem_write(BASE, read("F:\\Code\\Pycharm\\JDSign\\libjdbitmapkit.so"))

    mu.mem_write(BASE, read("./libjdbitmapkit.so"))

    mu.mem_write(KEY_ADDR, key0)

    for in range(9):

        arr1 = [0 for in range(8)]

        if i != 0:

            arr1[i-1= 1

        = mu.hook_add(UC_HOOK_CODE, hook_code, arr1)

        mu.mem_write(PLAINTEXT_ADDR, bin2bytes(arr1))

        mu.reg_write(UC_ARM_REG_R0, KEY_ADDR)

        mu.reg_write(UC_ARM_REG_R1, 32)

        mu.reg_write(UC_ARM_REG_R2, 1)

        mu.reg_write(UC_ARM_REG_R3, PLAINTEXT_ADDR)

        mu.emu_start(BASE + 0x0004B7C + 1, BASE + 0x0005288)

        mu.hook_del(h)

    for in range(8):

        for in range(8):

            arr3 = []

            if table[0][j] != table[i+1][j]:

                table1.append([i, j, table[0][j], table[i+1][j]])

    print(table1)

就结果为

1

[[0601], [1410], [2501], [3001], [4201], [5301], [6110], [7701]]

我们可以得到为[[0, 6, 0, 1], [1, 4, 1, 0], [2, 5, 0, 1], [3, 0, 0, 1], [4, 2, 0, 1], [5, 3, 0, 1], [6, 1, 1, 0], [7, 7, 0, 1]],里面的列表每个即为[x,y,m,n],比如[6, 1, 0, 1]意味,第6个比特位在打乱计算后会放到第1个比特位上,如果第6个比特位是0,则在打乱计算后会放到第6个比特位上是0,反之是1。

然后可轻松还原sub_4B7C算法。

1

2

3

4

5

6

7

8

9

10

11

def sub_4B7C(input):

    table = [[0601], [1410], [2501], [3001], [4201], [5301], [6110],

             [7701]]

    arr = bytes2bin(input)

    arr1 = [0 for in range(8)]

    for in range(8):

        if arr[i] == 0:

            arr1[table[i][1]] = table[i][2]

        else:

            arr1[table[i][1]] = table[i][3]

    return bin2bytes(arr1)

使用unicorn一步求出所有case的映射关系

好了,现在我们已经还原出第一个case,我们再会过头来看下sub_10D70,可以看到一共有七个case,分别对应的七个函数不同的只有函数起始结束地址,以及要处理的字节数,开拓下思维,这时候我们完全可以all in one,一次得到所有case的映射关系。

image-20210308212310967

代码如下:

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

from unicorn import *

from unicorn.arm_const import *

table = []

table1 = []

def bytes2bin(bytes):

    arr = []

    for in [m for in bytes]:

        arr.append(

            [(v & 128) >> 7, (v & 64) >> 6, (v & 32) >> 5, (v & 16) >> 4, (v & 8) >> 3, (v & 4) >> 2, (v & 2) >> 1,

             v & 1])

    return [i for in arr for in j]

def bin2bytes(arr):

    length = len(arr) // 8

    arr1 = [0 for in range(length)]

    for in range(length):

        arr1[j] = arr[j * 8] << 7 | arr[j * 8 + 1] << 6 | arr[j * 8 + 2] << 5 | arr[j * 8 + 3] << 4 | arr[

            * 8 + 4] << 3 | arr[j * 8 + 5] << 2 | arr[j * 8 + 6] << 1 | arr[j * 8 + 7]

    return bytes(arr1)

def read(name):

    with open(name, 'rb') as f:

        return f.read()

def hook_code(mu, address, size, user_data):

    if address == BASE + user_data[2-4:

        table.append(bytes2bin(mu.mem_read(PLAINTEXT_ADDR, m[0])))

if __name__ == "__main__":

    key0 = b'44e715a6e322ccb7d028f7a42fa55040'

    mu = Uc(UC_ARCH_ARM, UC_MODE_THUMB)

    BASE = 0x400000

    STACK_ADDR = 0x0

    STACK_SIZE = 1024 * 10

    PLAINTEXT_ADDR = 1024 * 10

    PLAINTEXT_SIZE = 1024

    KEY_ADDR = 1024 * 11

    KEY_SIZE = 1024

    mu.mem_map(BASE, 1024 * 1024)

    mu.mem_map(STACK_ADDR, STACK_SIZE)

    mu.mem_map(PLAINTEXT_ADDR, PLAINTEXT_SIZE)

    mu.mem_map(KEY_ADDR, KEY_SIZE)

    mu.reg_write(UC_ARM_REG_SP, STACK_ADDR + STACK_SIZE - 1)

    # mu.mem_write(BASE, read("F:\\Code\\Pycharm\\JDSign\\libjdbitmapkit.so"))

    mu.mem_write(BASE, read("./libjdbitmapkit.so"))

    mu.mem_write(KEY_ADDR, key0)

    arr = [[10x0004B7C0x0005288], [20x00061A00x0006AE8], [30x00079940x000084A6], [40x000091AC0x00009DF4],

           [50x0000ABF80x0000BA8C], [60x0000C8C00x0000D9A0], [70x0000E7FC0x0000FC1C]]

    for in arr:

        for in range(m[0]*8+1):

            arr1 = [0 for in range(m[0]*8)]

            if i != 0:

                arr1[i - 1= 1

            = mu.hook_add(UC_HOOK_CODE, hook_code, m)

            mu.mem_write(PLAINTEXT_ADDR, bin2bytes(arr1))

            mu.reg_write(UC_ARM_REG_R0, KEY_ADDR)

            mu.reg_write(UC_ARM_REG_R1, 32)

            mu.reg_write(UC_ARM_REG_R2, 1)

            mu.reg_write(UC_ARM_REG_R3, PLAINTEXT_ADDR)

            mu.emu_start(BASE + m[1+ 1, BASE + m[2])

            mu.hook_del(h)

        for in range(m[0]*8):

            for in range(m[0]*8):

                arr3 = []

                if table[0][j] != table[i + 1][j]:

                    table1.append([i, j, table[0][j], table[i + 1][j]])

        print("case %s 映射关系:"%(m[0]-1))

        print(table1)

        table.clear()

        table1.clear()

运行结果:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

case 0 映射关系:

[[0601], [1410], [2501], [3001], [4201], [5301], [6110], [7701]]

case 1 映射关系:

[[0501], [1901], [2010], [3710], [41001], [5601], [61310], [7101], [8401], [91101], [101410], [11310], [121201], [131510], [14801], [15201]]

case 2 映射关系:

[[01701], [1701], [2501], [31910], [41801], [51510], [62201], [72101], [81601], [9401], [101201], [11210], [121010], [131310], [142010], [15810], [16901], [172301], [181110], [19601], [20101], [21310], [22010], [231401]]

case 3 映射关系:

[[02510], [1401], [22901], [3101], [42710], [51810], [62310], [71410], [82810], [91101], [10910], [111301], [122410], [13010], [14501], [15210], [162601], [171201], [183110], [191610], [203001], [211501], [221001], [232210], [24710], [252101], [26610], [27310], [28810], [292001], [301910], [311701]]

case 4 映射关系:

[[01101], [11201], [22810], [33001], [41310], [52401], [62210], [72510], [82310], [9301], [101601], [11810], [123401], [13201], [14501], [15710], [16401], [171401], [183910], [193301], [201501], [21001], [223101], [23910], [242901], [252610], [261901], [27610], [282710], [291010], [303701], [313810], [322001], [332110], [34101], [353601], [363201], [371701], [381801], [393510]]

case 5 映射关系:

[[01101], [14501], [21510], [32201], [41001], [5701], [6301], [74201], [81710], [92101], [10401], [11810], [121910], [133201], [142810], [153110], [162901], [171410], [183910], [192710], [20210], [212401], [222610], [23910], [244101], [25110], [264701], [274401], [282310], [29010], [301210], [311801], [323301], [333601], [344010], [353401], [362501], [371610], [38510], [393501], [403801], [413710], [421301], [432010], [44601], [454301], [463001], [474610]]

case 6 映射关系:

[[0710], [1901], [25310], [31910], [41510], [5801], [6301], [72410], [81801], [95101], [104210], [113901], [122001], [131201], [142810], [152710], [162301], [174901], [181010], [195510], [205210], [211701], [224801], [231410], [243301], [252510], [26410], [271101], [284710], [29001], [302110], [314401], [321601], [334101], [342901], [35101], [364601], [37501], [383001], [394501], [403110], [414310], [423610], [432601], [443401], [45201], [46601], [475010], [481310], [493710], [503201], [514001], [523501], [533801], [545401], [552201]]

Version2 加密

执行流程

加密执行函数流程:sub_10DE4——>sub_12FF0——>sub_12ECC——>sub_130D0,其中sub_12ECC是重点,我们直接看sub_12ECC。

sub_12ECC初始化

进入函数sub_12ECC,先看下开始处的汇编片段。r1寄存器存放着key2字符串地址,r2寄存器存放着常量1,r3寄存器存放着待加密字符串地址,在开辟栈空间后SP,#0x48+arg_0地址存放着是待加密字节的长度。

1

2

3

4

5

6

7

8

9

10

11

12

13

.text:00012ECC                 LDR.W           R12, =(__stack_chk_guard_ptr - 0x12ED8)

.text:00012ED0                 PUSH.W          {R4-R11,LR}

.text:00012ED4                 ADD             R12, PC ; __stack_chk_guard_ptr

.text:00012ED6                 LDR.W           R12, [R12] ; __stack_chk_guard

.text:00012EDA                 SUB             SP, SP, #0x24

.text:00012EDC                 MOV             R10, R3

.text:00012EDE                 LDR.W           R3, [R12]

.text:00012EE2                 MOV             R7, R1

.text:00012EE4                 LDR.W           R8, [SP,#0x48+arg_0]

.text:00012EE8                 MOV             R9, R2

.text:00012EEA                 STR             R3, [SP,#0x48+var_2C]

.text:00012EEC                 CMP.W           R8, #0

.text:00012EF0                 BNE             loc_12F02

我们注意到下面这四句汇编指令,是把待加密字符串存放的地址放到r10寄存器中,key2字符串地址放到r7寄存器中,常量1放到r9寄存器,从SP,#0x48+arg_0地址取出待加密字符串长度后放到r8寄存器中。

1

2

3

4

5

6

7

.text:00012EDC                 MOV             R10, R3

.text:00012EE2                 MOV             R7, R1

.text:00012EE8                 MOV             R9, R2

.text:00012EE4                 LDR.W           R8, [SP,#0x48+arg_0]

sub_12ECC加密流程

然后我们在函数sub_12ECC中往下走,找到0x12F68地址处开始的关键汇编片段,即sign算法加密部分,这里是一个循环。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

.text:00012F66                 MOVS            R3, #0

.text:00012F68

.text:00012F68 loc_12F68                              

.text:00012F68                 AND.W           R2, R3, #0xF

.text:00012F6C                 ADD             R0, SP, #0x48+var_28

.text:00012F6E                 ADD             R2, R0

.text:00012F70                 AND.W           R1, R3, #7

.text:00012F74                 LDRB.W          R0, [R10]

.text:00012F78                 ADDS            R3, #1

.text:00012F7A                 LDRB.W          R2, [R2,#-0x14]

.text:00012F7E                 CMP             R3, R8

.text:00012F80                 LDRB            R4, [R7,R1]

.text:00012F82                 EOR.W           R0, R2, R0

.text:00012F86                 EOR.W           R0, R0, R4

.text:00012F8A                 ADD             R0, R2

.text:00012F8C                 EOR.W           R2, R2, R0

.text:00012F90                 UXTB            R2, R2

.text:00012F92                 STRB.W          R2, [R10],#1

.text:00012F96                 LDRB            R1, [R7,R1]

.text:00012F98                 EOR.W           R2, R2, R1

.text:00012F9C                 STRB.W          R2, [R10,#-1]

.text:00012FA0                 BNE             loc_12F68

r10寄存器

我们抓重点,看待加密字符串是怎么被加密的,待加密字符串地址放在了r10寄存器中,涉及r10寄存器的汇编指令有三条。

00012F74处的LDRB.W R0, [R10]是取出待加密字符串的一个字节放到r0寄存器中随后会进行计算操作;

00012F92处的STRB.W R2, [R10],#1,这条指令首先会把放在r2寄存器中的计算结果值放到r10寄存器当前存储的地址上,这个并不重要,因为还没有计算完成,后面的指令才是把最后的计算值存放值到这个地址上,重要的是这条指令随后r10寄存器中的地址值会加一,因为这是一个基于索引后置修改取址模式;

00012F9200012F9C之间,我们可以看到取了r7寄存器存放的地址值的一个偏移地址的值(偏移值为r1寄存器中的值)放到了r1寄存器中,随后把r1r2寄存器中的值进行亦或,结果存放在r2寄存器中。

最后是涉及到r10寄存器的第三条指令00012F9C处,STRB.W R2, [R10,#-1],把计算结果值放到R10,#-1地址处,注意到r10寄存器中的地址值刚才已经加一,所以现在减一后还是刚才存储的地址,所以这里才是计算结果最后存放的指令。

r3寄存器

好了,我们已经知道计算结果是放到了r2寄存器中,我们从00012F66开始看,一步步看是怎么计算出来的。

可以看到,r3作为计数器,每轮循环会在.text:00012F78 ADDS R3, #1处加一,并在.text:00012F7E CMP R3, R8处和r8寄存器值即加密字符串长度进行比较。

r2寄存器

在开始处,r3首先和0xf进行与操作,结果放在r2寄存器中,随后将与操作的结果值和SP, #0x48+var_28地址值进行相加,相加结果仍放在r2寄存器中,随后会进行.text:00012F7A LDRB.W R2, [R2,#-0x14],也就是说现在r2寄存器中存放的是SP, #0x48+var_28地址加上一个偏移值(即(i &0xf) - 0x14)这个地址上存放的值。

r4寄存器

然后在.text:00012F70 AND.W R1, R3, #7处把r3寄存器中的值和7进行与操作,结果放在r1寄存器中,然后在.text:00012F80 LDRB R4, [R7,R1]处把r1寄存器中的地址值作为偏移加上r7寄存器中的值,取出这个地址中的值放在r4寄存器中,r7寄存器前面我们说了放的是key2,所以放在r4寄存器中这个的值即是key2[i&7]

计算部分

随后便是计算部分:

1

2

3

4

5

6

7

8

.text:00012F82                 EOR.W           R0, R2, R0

.text:00012F86                 EOR.W           R0, R0, R4

.text:00012F8A                 ADD             R0, R2

.text:00012F8C                 EOR.W           R2, R2, R0

.text:00012F90                 UXTB            R2, R2

.text:00012F96                 LDRB            R1, [R7,R1]

.text:00012F98                 EOR.W           R2, R2, R1

此时,涉及计算的r0 r2 r4我们都已经知道是什么了,r0寄存器中的值我们在提r10寄存器时候有知道它是待加密字符串取出的一个字节。

IDA F5

这部分使用IDA进行f5的效果是这样的。

image-20210308163440724

sub_12ECC算法还原

现在我们想还原这个加密过程该怎么办,要看我们缺什么。待加密字符串,key2和加密计算过程我们都有了,只有一个SP, #0x48+var_28地址加上一个偏移 -0x14,这个地址存放有一个数组,每轮加密循环会取出一个值(数组偏移(i & 0xf)处)放在r2寄存器中然后进行计算,我们通过静态分析无法知道这个数组值是什么,所以进行ida动态调试,可以得到这个地方存放的数组值为[0x37, 0x92, 0x44, 0x68, 0xA5, 0x3D, 0xCC, 0x7F, 0xBB, 0xF, 0xD9, 0x88, 0xEE, 0x9A, 0xE9, 0x5A]。

image-20210309140639735

随后可python写出sub_12ECC函数中的加密计算过程。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

def sub_12ECC(input):

    arr = [0x370x920x440x680xA50x3D0xCC0x7F0xBB0xF0xD90x880xEE0x9A0xE90x5A]

    key2 = b"80306f4370b39fd5630ad0529f77adb6"

    arr1 = [0 for in range(len(input))]

    for in range(len(input)):

        r0 = int(input[i])

        r2 = arr[i & 0xf]

        r4 = int(key2[i & 7])

        r0 = r2 ^ r0

        r0 = r0 ^ r4

        r0 = r0 + r2

        r2 = r2 ^ r0

        r1 = int(key2[i & 7])

        r2 = r2 ^ r1

        arr1[i] = r2 & 0xff

    return bytes(arr1)

sign算法全流程还原

好了,到现在所有的关键函数已经被我们分析出来了,想还原出来所有的sign加密流程已经成了一个时间问题的体力劳动了,本来想放个半成品给大家参考下,经评论区老哥友情提醒,分析流程已经足够了,这部分就先和谐掉吧...

以及文章目的是学习交流的,请勿不正确使用,用于违法行为。

后记

最近经常会在想自己大部分时候只会用frida去hook十分像一个脚本小子(逃,迫切需要提升汇编分析水平,分析到这里对我自己来说进步不小,但是要学习的东西还很多,如果有什么希望的话,就是希望能早一点摆脱对IDA f5的依赖,可以手撕汇编:)

最后,如果你有学到有用的东西,请不要忘记点赞,不枉我写了这么多。

标签:arr,sub,R2,text,某东,sign,mu,半自动化,ADDR
From: https://blog.csdn.net/2403_87755661/article/details/144079560

相关文章

  • AntDesign - Vue Table组件 实现动态表格、列宽计算(方式二)
    朋友们,按照上文(方式一)思路,实现了动态表格和表头分组,这篇按照方式一的需求,扩展出另一种代码写法;一、表格头表格columns还是定义在data(){}中,初始化静态列数组,配置项列由后端接口返回(第二点写动态配置项代码);在方式一基础上加了筛选菜单功能,因此变化代码部分如下......
  • 【Azure 环境】使用Azure bicep对多个ServicePrinciple 进行role assignment分配
    问题描述使用Azurebicep对多个ServicePrinciple 进行roleassignment分配 步骤如下第一步:定义传参,里面包括objectID和role的一个map如:paramservicePrincipalsarray=[ {  objectId:'service-principal-object-id-1'  roles:[   'Contributor'......
  • [Whole Web] Vue design and React design
    VirtualDOMAdvantage:OneoftheadvantagesofthevirtualDOMiscross-platformrenderingabstraction.ThevirtualDOMcanconnecttodifferenthostenvironments,enablingoneframeworktorenderacrossmultipleplatforms.1.WhydidReactswitchfromt......
  • Adobe InDesign(ID)排版设计软件下载安装与win/mac安装包
    获取安装包链接:https://pan.baidu.com/s/1wlnx3rqgqD5HDWxiqyGFpQ?pwd=xy9e提取码:xy9e 一、AdobeInDesign软件简介AdobeInDesign是一款功能强大的排版设计软件,专为专业出版设计师打造。它不仅可以进行高效的页面布局,还能完美集成文本、图形和多媒体内容,广泛应用于各......
  • AntDesign - Vue Table组件 实现动态表格、表头分组的功能(方式一)
     一、功能分析产品经理要求企微主体名称是配置项且后期可修改或增加,各企微主体账号的数据一对应。前端开发设计方案为:静态列(左部分)在前端写,配置项由后端接口返回,再动态追加到columns中,根据表头dataIndex对应的数据项,填充到数据数组dataSource。至此,开发思路出来......
  • 说说响应式设计(responsive design)和自适应设计(adaptive design)的区别?
    响应式设计(ResponsiveDesign)和自适应设计(AdaptiveDesign)都是为了解决同一个问题:如何让网站在不同设备(桌面电脑、平板电脑、手机等)上都能提供良好的用户体验。它们的核心区别在于处理方式的不同:响应式设计(ResponsiveDesign):核心思想:像液体一样,根据容器(浏览器窗......
  • app&小程序&web安全—sign签名绕过
    一、环境准备本文测试环境为JeecgBoot,代码链接:jeecgboot/Github环境配置链接:IDEA启动项目-JeecgBoot文档中心配置好依赖和数据库后启动后端:org.jeecg.JeecgSystemApplication启动Vue3前端,点击dev,Jeecg启动!二、阅读代码(后端)关于签名函数的位置我不太会定位,是通过......
  • Altium Designer 入门基础教程(六)
    本文章继续接着《Altium Designer入门基础教程(五)》的内容往下介绍:七、AD画板的整个流程步骤 L.板层数和设计规则的设置a.板层数的设置开始画板前,根据板框大小和元器件的多少可以大概评估一下要用到几层板,先设置好层数,再开始布局画板工作;或者在布局画板的过程中,发现实在......
  • CBDD-Chemical Biology & Drug Design
    文章目录一、征稿简介二、重要信息三、服务简述四、投稿须知五、联系咨询一、征稿简介二、重要信息期刊官网:https://ais.cn/u/3eEJNv三、服务简述本次征文主题包括但不限于:虚拟筛选全新药物设计药物再利用毒性预测临床试验优化性质优化关键词:人工智能;自然......
  • AntDesign树形组件tree和输入input组件融合使用
    <a-tree :tree-data="selectItem.options.options" :replace-fields="{ children:'children', title:'label', code:'code' }" >......