2 # Copyright (C) 2020 Michael Nolan <mtnolan2640@gmail.com>
3 # Copyright (C) 2020 Luke Kenneth Casson Leighton <lkcl@lkcl.net>
7 takes powerpc assembly instructions and turns them into LE/BE binary
8 data. calls powerpc64-linux-gnu-as, ld and objcopy to do so.
16 from io
import BytesIO
18 from openpower
.simulator
.envcmds
import cmds
20 filedir
= os
.path
.dirname(os
.path
.realpath(__file__
))
21 memmap
= os
.path
.join(filedir
, "memmap")
25 def __init__(self
, instructions
, bigendian
):
26 self
.bigendian
= bigendian
28 self
.endian_fmt
= "elf64-big"
33 self
.endian_fmt
= "elf64-little"
36 if isinstance(instructions
, bytes
): # actual bytes
37 self
.binfile
= BytesIO(instructions
)
38 self
.binfile
.name
= "assembly"
39 self
.assembly
= '' # noo disassemble number fiiive
40 print("binary", self
.binfile
)
41 elif isinstance(instructions
, str): # filename
42 # read instructions into a BytesIO to avoid "too many open files"
43 with
open(instructions
, "rb") as f
:
45 self
.binfile
= BytesIO(b
)
46 self
.assembly
= '' # noo disassemble number fiiive
47 print("program", self
.binfile
)
49 if isinstance(instructions
, list):
50 instructions
= '\n'.join(instructions
)
51 self
.assembly
= instructions
+ '\n' # plus final newline
53 self
._instructions
= list(self
._get
_instructions
())
58 def __exit__(self
, type, value
, traceback
):
61 def _get_binary(self
, elffile
):
62 self
.binfile
= tempfile
.NamedTemporaryFile(suffix
=".bin")
63 args
= [cmds
['objcopy'],
65 "-I", self
.endian_fmt
,
68 subprocess
.check_output(args
)
70 def _link(self
, ofile
):
71 with tempfile
.NamedTemporaryFile(suffix
=".elf") as elffile
:
77 subprocess
.check_output(args
)
78 self
._get
_binary
(elffile
)
81 with tempfile
.NamedTemporaryFile(suffix
=".o") as outfile
:
88 p
= subprocess
.Popen(args
, stdin
=subprocess
.PIPE
)
89 p
.communicate(self
.assembly
.encode('utf-8'))
91 print("Error in program:")
96 def _get_instructions(self
):
98 data
= self
.binfile
.read(4)
101 yield struct
.unpack('<I', data
)[0] # unsigned int
103 def generate_instructions(self
):
104 yield from self
._instructions
110 curpos
= self
.binfile
.tell()
111 self
.binfile
.seek(0, 2) # Seek to end of file
112 size
= self
.binfile
.tell()
113 self
.binfile
.seek(curpos
, 0)
116 def write_bin(self
, fname
):
118 data
= self
.binfile
.read()
119 with
open(fname
, "wb") as f
:
125 if __name__
== '__main__':
126 lst
= ['addi 5, 0, 4660/2',
132 lst
= ["addi 9, 0, 0x10", # i = 16
133 "addi 9,9,-1", # i = i - 1
134 "cmpi 2,1,9,12", # compare 9 to value 12, store in CR2
135 "bc 4,10,-8", # branch if CR2 "test was != 12"
139 with
Program(lst
, False) as p
:
140 for instruction
in p
.generate_instructions():
141 print (hex(instruction
))
142 p
.write_bin("/tmp/test.bin")