c55a67d1d68d06deb320394da6c76ca8c5f31492
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 soc
.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 elif isinstance(instructions
, str): # filename
41 # read instructions into a BytesIO to avoid "too many open files"
42 with
open(instructions
, "rb") as f
:
44 self
.binfile
= BytesIO(b
)
45 self
.assembly
= '' # noo disassemble number fiiive
46 print("program", self
.binfile
)
48 if isinstance(instructions
, list):
49 instructions
= '\n'.join(instructions
)
50 self
.assembly
= instructions
+ '\n' # plus final newline
52 self
._instructions
= list(self
._get
_instructions
())
57 def __exit__(self
, type, value
, traceback
):
60 def _get_binary(self
, elffile
):
61 self
.binfile
= tempfile
.NamedTemporaryFile(suffix
=".bin")
62 args
= [cmds
['objcopy'],
64 "-I", self
.endian_fmt
,
67 subprocess
.check_output(args
)
69 def _link(self
, ofile
):
70 with tempfile
.NamedTemporaryFile(suffix
=".elf") as elffile
:
76 subprocess
.check_output(args
)
77 self
._get
_binary
(elffile
)
80 with tempfile
.NamedTemporaryFile(suffix
=".o") as outfile
:
87 p
= subprocess
.Popen(args
, stdin
=subprocess
.PIPE
)
88 p
.communicate(self
.assembly
.encode('utf-8'))
90 print("Error in program:")
95 def _get_instructions(self
):
97 data
= self
.binfile
.read(4)
100 yield struct
.unpack('<I', data
)[0] # unsigned int
102 def generate_instructions(self
):
103 yield from self
._instructions
109 curpos
= self
.binfile
.tell()
110 self
.binfile
.seek(0, 2) # Seek to end of file
111 size
= self
.binfile
.tell()
112 self
.binfile
.seek(curpos
, 0)
115 def write_bin(self
, fname
):
117 data
= self
.binfile
.read()
118 with
open(fname
, "wb") as f
:
124 if __name__
== '__main__':
125 lst
= ['addi 5, 0, 4660/2',
131 lst
= ["addi 9, 0, 0x10", # i = 16
132 "addi 9,9,-1", # i = i - 1
133 "cmpi 2,1,9,12", # compare 9 to value 12, store in CR2
134 "bc 4,10,-8", # branch if CR2 "test was != 12"
138 with
Program(lst
, False) as p
:
139 for instruction
in p
.generate_instructions():
140 print (hex(instruction
))
141 p
.write_bin("/tmp/test.bin")