2016-01-13 07:16:41 -08:00
|
|
|
-- split lips.lua
|
2015-11-20 11:53:09 -08:00
|
|
|
|
2015-11-20 16:21:13 -08:00
|
|
|
local assembler = {
|
|
|
|
_DESCRIPTION = 'Assembles MIPS assembly files for the R4300i CPU.',
|
2015-11-23 14:16:31 -08:00
|
|
|
_URL = 'https://github.com/notwa/lips/',
|
2015-11-20 16:21:13 -08:00
|
|
|
_LICENSE = [[
|
2016-01-13 06:30:58 -08:00
|
|
|
Copyright (C) 2015,2016 Connor Olding
|
2015-11-20 16:21:13 -08:00
|
|
|
|
|
|
|
This program is licensed under the terms of the MIT License, and
|
|
|
|
is distributed without any warranty. You should have received a
|
|
|
|
copy of the license along with this program; see the file LICENSE.
|
|
|
|
]],
|
|
|
|
}
|
|
|
|
|
2016-01-13 07:16:41 -08:00
|
|
|
local Parser = require "Parser"
|
2015-12-02 09:09:44 -08:00
|
|
|
|
2015-12-18 23:42:54 -08:00
|
|
|
local function readfile(fn)
|
2016-01-13 06:30:58 -08:00
|
|
|
local f = open(fn, 'r')
|
2015-12-18 23:42:54 -08:00
|
|
|
if not f then
|
|
|
|
error('could not open assembly file for reading: '..tostring(fn), 2)
|
|
|
|
end
|
|
|
|
local asm = f:read('*a')
|
|
|
|
f:close()
|
|
|
|
return asm
|
|
|
|
end
|
|
|
|
|
2015-12-25 01:09:00 -08:00
|
|
|
function assembler.word_writer()
|
|
|
|
local buff = {}
|
2015-12-25 01:15:48 -08:00
|
|
|
local max = -1
|
2015-12-25 01:09:00 -08:00
|
|
|
return function(pos, b)
|
|
|
|
if pos then
|
|
|
|
buff[pos] = b
|
|
|
|
if pos > max then
|
|
|
|
max = pos
|
|
|
|
end
|
2016-01-13 06:30:58 -08:00
|
|
|
elseif max >= 0 then
|
2015-12-25 01:09:00 -08:00
|
|
|
for i=0, max, 4 do
|
|
|
|
local a = buff[i+0] or '00'
|
|
|
|
local b = buff[i+1] or '00'
|
|
|
|
local c = buff[i+2] or '00'
|
|
|
|
local d = buff[i+3] or '00'
|
|
|
|
print(a..b..c..d)
|
|
|
|
end
|
|
|
|
end
|
2015-11-20 11:53:09 -08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-11-26 19:45:13 -08:00
|
|
|
function assembler.assemble(fn_or_asm, writer, options)
|
2015-11-23 14:16:31 -08:00
|
|
|
-- assemble MIPS R4300i assembly code.
|
|
|
|
-- if fn_or_asm contains a newline; treat as assembly, otherwise load file.
|
|
|
|
-- returns error message on error, or nil on success.
|
2015-11-20 16:21:13 -08:00
|
|
|
fn_or_asm = tostring(fn_or_asm)
|
2015-12-25 01:09:00 -08:00
|
|
|
local default_writer = not writer
|
|
|
|
writer = writer or assembler.word_writer()
|
2015-11-26 19:45:13 -08:00
|
|
|
options = options or {}
|
2015-11-20 14:00:02 -08:00
|
|
|
|
2016-01-13 07:16:41 -08:00
|
|
|
local function main()
|
2015-11-21 17:35:43 -08:00
|
|
|
local fn = nil
|
2015-12-18 23:42:54 -08:00
|
|
|
local asm
|
2015-11-20 16:21:13 -08:00
|
|
|
if fn_or_asm:find('[\r\n]') then
|
|
|
|
asm = fn_or_asm
|
|
|
|
else
|
2015-11-21 17:35:43 -08:00
|
|
|
fn = fn_or_asm
|
2015-12-18 23:42:54 -08:00
|
|
|
asm = readfile(fn)
|
|
|
|
options.path = fn:match(".*/")
|
2015-11-20 16:21:13 -08:00
|
|
|
end
|
|
|
|
|
2015-12-17 07:19:24 -08:00
|
|
|
local parser = Parser(writer, fn, options)
|
2015-12-25 01:09:00 -08:00
|
|
|
parser:parse(asm)
|
|
|
|
|
|
|
|
if default_writer then
|
|
|
|
writer()
|
|
|
|
end
|
2015-11-20 15:27:22 -08:00
|
|
|
end
|
2015-11-20 11:53:09 -08:00
|
|
|
|
2015-11-26 19:45:13 -08:00
|
|
|
if options.unsafe then
|
2015-11-24 17:06:38 -08:00
|
|
|
return main()
|
|
|
|
else
|
|
|
|
local ok, err = pcall(main)
|
|
|
|
return err
|
|
|
|
end
|
2015-11-20 11:53:09 -08:00
|
|
|
end
|
2015-11-20 16:21:13 -08:00
|
|
|
|
|
|
|
return setmetatable(assembler, {
|
2015-11-26 19:45:13 -08:00
|
|
|
__call = function(self, ...)
|
|
|
|
return self.assemble(...)
|
2015-11-20 16:21:13 -08:00
|
|
|
end,
|
|
|
|
})
|