forked from SBird1337/cexplore
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpycat.py
executable file
·92 lines (76 loc) · 3.62 KB
/
pycat.py
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
88
89
90
91
#!/usr/bin/env python3
import os
import argparse
import shutil
import re
parser = argparse.ArgumentParser(description='Simplified "compiler" frontend, cats any input')
parser.add_argument('--version', action='store_true', help = 'Get Version String of cc1', required = False, dest='version')
parser.add_argument('-o', action='store', help = 'Output Assembly file', required = False, dest='destination')
args, remainder = parser.parse_known_args()
if args.version:
print("pycat frontend for cexplore")
quit()
source = remainder[-1]
with open(source, 'r') as f_src, open(args.destination, 'w') as f_dst:
for line in f_src:
line2 = line.strip()
if line2 and ':' not in line2 and not line2.startswith('@') and not line2.startswith('//'):
instruction = line2.split(maxsplit=1)
if len(instruction) > 1:
opcode, operand = instruction
if opcode.endswith('s') and opcode != 'bls' and opcode != 'bhs' and opcode != 'bcs': # Remove s suffix from most opcodes
opcode = opcode[:-1]
# Fix bhs -> bcs and blo -> bcc
if opcode == 'bhs':
opcode = 'bcs'
if opcode == 'blo':
opcode = 'bcc'
rx_ry = re.search(r'r\d{1,2}-r\d{1,2}', operand, re.I)
if rx_ry:
rx_ry = rx_ry.group(0)
pos = rx_ry.find('-')
x = int(rx_ry[1:pos])
y = int(rx_ry[pos + 2:])
operands = operand.split(rx_ry, 1)
operand = operands[0]
for i in range(x, y + 1):
operand += 'r{}, '.format(i)
operand = operand[:-2] + operands[1]
dec_num = re.search(r'#\-?\d+', operand, re.I)
if dec_num:
dec_num = dec_num.group(0)
if dec_num[1] != '-':
value = int(dec_num[1:])
if value != 0:
s = operand.split(dec_num, 1)
operand = '{}#0x{:x}{}'.format(s[0], value, s[1])
else:
value = int(dec_num[2:])
if value != 0:
s = operand.split(dec_num, 1)
operand = '{}#-0x{:x}{}'.format(s[0], value, s[1])
if opcode in ('add', 'sub', 'lsl', 'lsr', 'asr', 'ror', 'and', 'orr', 'eor'):
operands = operand.split(',')
if len(operands) == 2:
operand = operands[0] + ', ' + operand
# Rename sb to r9
operands = operand.split(', ')
operands = map(lambda x: 'r9' if x=='sb' else x, operands)
operand = ', '.join(operands)
# fix movs rX, #0 -> mov rX, #0x0
if opcode == 'mov':
operands = operand.split(', ')
if operands[1] == '#0':
operand = operands[0] + ', #0x0'
# rsb rn, rn, #0 -> neg rn, rn
if opcode == 'rsb':
operands = operand.split(', ')
if operands[2] == '#0' and operands[0] == operands[1]:
opcode = 'neg'
operand = operands[0] + ', ' + operands[1]
line2 = '\t{}\t{}'.format(opcode, operand)
else:
# Replace .4byte with .word
if '.4byte' in line2:
line2 = line2.replace('.4byte', '.word ')
f_dst.write(line2 + '\n')