dynasm.lua 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095
  1. ------------------------------------------------------------------------------
  2. -- DynASM. A dynamic assembler for code generation engines.
  3. -- Originally designed and implemented for LuaJIT.
  4. --
  5. -- Copyright (C) 2005-2021 Mike Pall. All rights reserved.
  6. -- See below for full copyright notice.
  7. ------------------------------------------------------------------------------
  8. -- Application information.
  9. local _info = {
  10. name = "DynASM",
  11. description = "A dynamic assembler for code generation engines",
  12. version = "1.5.0",
  13. vernum = 10500,
  14. release = "2021-05-02",
  15. author = "Mike Pall",
  16. url = "https://luajit.org/dynasm.html",
  17. license = "MIT",
  18. copyright = [[
  19. Copyright (C) 2005-2021 Mike Pall. All rights reserved.
  20. Permission is hereby granted, free of charge, to any person obtaining
  21. a copy of this software and associated documentation files (the
  22. "Software"), to deal in the Software without restriction, including
  23. without limitation the rights to use, copy, modify, merge, publish,
  24. distribute, sublicense, and/or sell copies of the Software, and to
  25. permit persons to whom the Software is furnished to do so, subject to
  26. the following conditions:
  27. The above copyright notice and this permission notice shall be
  28. included in all copies or substantial portions of the Software.
  29. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  30. EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  31. MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
  32. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
  33. CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
  34. TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
  35. SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  36. [ MIT license: https://www.opensource.org/licenses/mit-license.php ]
  37. ]],
  38. }
  39. -- Cache library functions.
  40. local type, pairs, ipairs = type, pairs, ipairs
  41. local pcall, error, assert = pcall, error, assert
  42. local _s = string
  43. local sub, match, gmatch, gsub = _s.sub, _s.match, _s.gmatch, _s.gsub
  44. local format, rep, upper = _s.format, _s.rep, _s.upper
  45. local _t = table
  46. local insert, remove, concat, sort = _t.insert, _t.remove, _t.concat, _t.sort
  47. local exit = os.exit
  48. local io = io
  49. local stdin, stdout, stderr = io.stdin, io.stdout, io.stderr
  50. ------------------------------------------------------------------------------
  51. -- Program options.
  52. local g_opt = {}
  53. -- Global state for current file.
  54. local g_fname, g_curline, g_indent, g_lineno, g_synclineno, g_arch
  55. local g_errcount = 0
  56. -- Write buffer for output file.
  57. local g_wbuffer, g_capbuffer
  58. ------------------------------------------------------------------------------
  59. -- Write an output line (or callback function) to the buffer.
  60. local function wline(line, needindent)
  61. local buf = g_capbuffer or g_wbuffer
  62. buf[#buf+1] = needindent and g_indent..line or line
  63. g_synclineno = g_synclineno + 1
  64. end
  65. -- Write assembler line as a comment, if requested.
  66. local function wcomment(aline)
  67. if g_opt.comment then
  68. wline(g_opt.comment..aline..g_opt.endcomment, true)
  69. end
  70. end
  71. -- Resync CPP line numbers.
  72. local function wsync()
  73. if g_synclineno ~= g_lineno and g_opt.cpp then
  74. wline("#line "..g_lineno..' "'..g_fname..'"')
  75. g_synclineno = g_lineno
  76. end
  77. end
  78. -- Dummy action flush function. Replaced with arch-specific function later.
  79. local function wflush(term)
  80. end
  81. -- Dump all buffered output lines.
  82. local function wdumplines(out, buf)
  83. for _,line in ipairs(buf) do
  84. if type(line) == "string" then
  85. assert(out:write(line, "\n"))
  86. else
  87. -- Special callback to dynamically insert lines after end of processing.
  88. line(out)
  89. end
  90. end
  91. end
  92. ------------------------------------------------------------------------------
  93. -- Emit an error. Processing continues with next statement.
  94. local function werror(msg)
  95. error(format("%s:%s: error: %s:\n%s", g_fname, g_lineno, msg, g_curline), 0)
  96. end
  97. -- Emit a fatal error. Processing stops.
  98. local function wfatal(msg)
  99. g_errcount = "fatal"
  100. werror(msg)
  101. end
  102. -- Print a warning. Processing continues.
  103. local function wwarn(msg)
  104. stderr:write(format("%s:%s: warning: %s:\n%s\n",
  105. g_fname, g_lineno, msg, g_curline))
  106. end
  107. -- Print caught error message. But suppress excessive errors.
  108. local function wprinterr(...)
  109. if type(g_errcount) == "number" then
  110. -- Regular error.
  111. g_errcount = g_errcount + 1
  112. if g_errcount < 21 then -- Seems to be a reasonable limit.
  113. stderr:write(...)
  114. elseif g_errcount == 21 then
  115. stderr:write(g_fname,
  116. ":*: warning: too many errors (suppressed further messages).\n")
  117. end
  118. else
  119. -- Fatal error.
  120. stderr:write(...)
  121. return true -- Stop processing.
  122. end
  123. end
  124. ------------------------------------------------------------------------------
  125. -- Map holding all option handlers.
  126. local opt_map = {}
  127. local opt_current
  128. -- Print error and exit with error status.
  129. local function opterror(...)
  130. stderr:write("dynasm.lua: ERROR: ", ...)
  131. stderr:write("\n")
  132. exit(1)
  133. end
  134. -- Get option parameter.
  135. local function optparam(args)
  136. local argn = args.argn
  137. local p = args[argn]
  138. if not p then
  139. opterror("missing parameter for option `", opt_current, "'.")
  140. end
  141. args.argn = argn + 1
  142. return p
  143. end
  144. ------------------------------------------------------------------------------
  145. -- Core pseudo-opcodes.
  146. local map_coreop = {}
  147. -- Dummy opcode map. Replaced by arch-specific map.
  148. local map_op = {}
  149. -- Forward declarations.
  150. local dostmt
  151. local readfile
  152. ------------------------------------------------------------------------------
  153. -- Map for defines (initially empty, chains to arch-specific map).
  154. local map_def = {}
  155. -- Pseudo-opcode to define a substitution.
  156. map_coreop[".define_2"] = function(params, nparams)
  157. if not params then return nparams == 1 and "name" or "name, subst" end
  158. local name, def = params[1], params[2] or "1"
  159. if not match(name, "^[%a_][%w_]*$") then werror("bad or duplicate define") end
  160. map_def[name] = def
  161. end
  162. map_coreop[".define_1"] = map_coreop[".define_2"]
  163. -- Define a substitution on the command line.
  164. function opt_map.D(args)
  165. local namesubst = optparam(args)
  166. local name, subst = match(namesubst, "^([%a_][%w_]*)=(.*)$")
  167. if name then
  168. map_def[name] = subst
  169. elseif match(namesubst, "^[%a_][%w_]*$") then
  170. map_def[namesubst] = "1"
  171. else
  172. opterror("bad define")
  173. end
  174. end
  175. -- Undefine a substitution on the command line.
  176. function opt_map.U(args)
  177. local name = optparam(args)
  178. if match(name, "^[%a_][%w_]*$") then
  179. map_def[name] = nil
  180. else
  181. opterror("bad define")
  182. end
  183. end
  184. -- Helper for definesubst.
  185. local gotsubst
  186. local function definesubst_one(word)
  187. local subst = map_def[word]
  188. if subst then gotsubst = word; return subst else return word end
  189. end
  190. -- Iteratively substitute defines.
  191. local function definesubst(stmt)
  192. -- Limit number of iterations.
  193. for i=1,100 do
  194. gotsubst = false
  195. stmt = gsub(stmt, "#?[%w_]+", definesubst_one)
  196. if not gotsubst then break end
  197. end
  198. if gotsubst then wfatal("recursive define involving `"..gotsubst.."'") end
  199. return stmt
  200. end
  201. -- Dump all defines.
  202. local function dumpdefines(out, lvl)
  203. local t = {}
  204. for name in pairs(map_def) do
  205. t[#t+1] = name
  206. end
  207. sort(t)
  208. out:write("Defines:\n")
  209. for _,name in ipairs(t) do
  210. local subst = map_def[name]
  211. if g_arch then subst = g_arch.revdef(subst) end
  212. out:write(format(" %-20s %s\n", name, subst))
  213. end
  214. out:write("\n")
  215. end
  216. ------------------------------------------------------------------------------
  217. -- Support variables for conditional assembly.
  218. local condlevel = 0
  219. local condstack = {}
  220. -- Evaluate condition with a Lua expression. Substitutions already performed.
  221. local function cond_eval(cond)
  222. local func, err
  223. if setfenv then
  224. func, err = loadstring("return "..cond, "=expr")
  225. else
  226. -- No globals. All unknown identifiers evaluate to nil.
  227. func, err = load("return "..cond, "=expr", "t", {})
  228. end
  229. if func then
  230. if setfenv then
  231. setfenv(func, {}) -- No globals. All unknown identifiers evaluate to nil.
  232. end
  233. local ok, res = pcall(func)
  234. if ok then
  235. if res == 0 then return false end -- Oh well.
  236. return not not res
  237. end
  238. err = res
  239. end
  240. wfatal("bad condition: "..err)
  241. end
  242. -- Skip statements until next conditional pseudo-opcode at the same level.
  243. local function stmtskip()
  244. local dostmt_save = dostmt
  245. local lvl = 0
  246. dostmt = function(stmt)
  247. local op = match(stmt, "^%s*(%S+)")
  248. if op == ".if" then
  249. lvl = lvl + 1
  250. elseif lvl ~= 0 then
  251. if op == ".endif" then lvl = lvl - 1 end
  252. elseif op == ".elif" or op == ".else" or op == ".endif" then
  253. dostmt = dostmt_save
  254. dostmt(stmt)
  255. end
  256. end
  257. end
  258. -- Pseudo-opcodes for conditional assembly.
  259. map_coreop[".if_1"] = function(params)
  260. if not params then return "condition" end
  261. local lvl = condlevel + 1
  262. local res = cond_eval(params[1])
  263. condlevel = lvl
  264. condstack[lvl] = res
  265. if not res then stmtskip() end
  266. end
  267. map_coreop[".elif_1"] = function(params)
  268. if not params then return "condition" end
  269. if condlevel == 0 then wfatal(".elif without .if") end
  270. local lvl = condlevel
  271. local res = condstack[lvl]
  272. if res then
  273. if res == "else" then wfatal(".elif after .else") end
  274. else
  275. res = cond_eval(params[1])
  276. if res then
  277. condstack[lvl] = res
  278. return
  279. end
  280. end
  281. stmtskip()
  282. end
  283. map_coreop[".else_0"] = function(params)
  284. if condlevel == 0 then wfatal(".else without .if") end
  285. local lvl = condlevel
  286. local res = condstack[lvl]
  287. condstack[lvl] = "else"
  288. if res then
  289. if res == "else" then wfatal(".else after .else") end
  290. stmtskip()
  291. end
  292. end
  293. map_coreop[".endif_0"] = function(params)
  294. local lvl = condlevel
  295. if lvl == 0 then wfatal(".endif without .if") end
  296. condlevel = lvl - 1
  297. end
  298. -- Check for unfinished conditionals.
  299. local function checkconds()
  300. if g_errcount ~= "fatal" and condlevel ~= 0 then
  301. wprinterr(g_fname, ":*: error: unbalanced conditional\n")
  302. end
  303. end
  304. ------------------------------------------------------------------------------
  305. -- Search for a file in the given path and open it for reading.
  306. local function pathopen(path, name)
  307. local dirsep = package and match(package.path, "\\") and "\\" or "/"
  308. for _,p in ipairs(path) do
  309. local fullname = p == "" and name or p..dirsep..name
  310. local fin = io.open(fullname, "r")
  311. if fin then
  312. g_fname = fullname
  313. return fin
  314. end
  315. end
  316. end
  317. -- Include a file.
  318. map_coreop[".include_1"] = function(params)
  319. if not params then return "filename" end
  320. local name = params[1]
  321. -- Save state. Ugly, I know. but upvalues are fast.
  322. local gf, gl, gcl, gi = g_fname, g_lineno, g_curline, g_indent
  323. -- Read the included file.
  324. local fatal = readfile(pathopen(g_opt.include, name) or
  325. wfatal("include file `"..name.."' not found"))
  326. -- Restore state.
  327. g_synclineno = -1
  328. g_fname, g_lineno, g_curline, g_indent = gf, gl, gcl, gi
  329. if fatal then wfatal("in include file") end
  330. end
  331. -- Make .include and conditionals initially available, too.
  332. map_op[".include_1"] = map_coreop[".include_1"]
  333. map_op[".if_1"] = map_coreop[".if_1"]
  334. map_op[".elif_1"] = map_coreop[".elif_1"]
  335. map_op[".else_0"] = map_coreop[".else_0"]
  336. map_op[".endif_0"] = map_coreop[".endif_0"]
  337. ------------------------------------------------------------------------------
  338. -- Support variables for macros.
  339. local mac_capture, mac_lineno, mac_name
  340. local mac_active = {}
  341. local mac_list = {}
  342. -- Pseudo-opcode to define a macro.
  343. map_coreop[".macro_*"] = function(mparams)
  344. if not mparams then return "name [, params...]" end
  345. -- Split off and validate macro name.
  346. local name = remove(mparams, 1)
  347. if not name then werror("missing macro name") end
  348. if not (match(name, "^[%a_][%w_%.]*$") or match(name, "^%.[%w_%.]*$")) then
  349. wfatal("bad macro name `"..name.."'")
  350. end
  351. -- Validate macro parameter names.
  352. local mdup = {}
  353. for _,mp in ipairs(mparams) do
  354. if not match(mp, "^[%a_][%w_]*$") then
  355. wfatal("bad macro parameter name `"..mp.."'")
  356. end
  357. if mdup[mp] then wfatal("duplicate macro parameter name `"..mp.."'") end
  358. mdup[mp] = true
  359. end
  360. -- Check for duplicate or recursive macro definitions.
  361. local opname = name.."_"..#mparams
  362. if map_op[opname] or map_op[name.."_*"] then
  363. wfatal("duplicate macro `"..name.."' ("..#mparams.." parameters)")
  364. end
  365. if mac_capture then wfatal("recursive macro definition") end
  366. -- Enable statement capture.
  367. local lines = {}
  368. mac_lineno = g_lineno
  369. mac_name = name
  370. mac_capture = function(stmt) -- Statement capture function.
  371. -- Stop macro definition with .endmacro pseudo-opcode.
  372. if not match(stmt, "^%s*.endmacro%s*$") then
  373. lines[#lines+1] = stmt
  374. return
  375. end
  376. mac_capture = nil
  377. mac_lineno = nil
  378. mac_name = nil
  379. mac_list[#mac_list+1] = opname
  380. -- Add macro-op definition.
  381. map_op[opname] = function(params)
  382. if not params then return mparams, lines end
  383. -- Protect against recursive macro invocation.
  384. if mac_active[opname] then wfatal("recursive macro invocation") end
  385. mac_active[opname] = true
  386. -- Setup substitution map.
  387. local subst = {}
  388. for i,mp in ipairs(mparams) do subst[mp] = params[i] end
  389. local mcom
  390. if g_opt.maccomment and g_opt.comment then
  391. mcom = " MACRO "..name.." ("..#mparams..")"
  392. wcomment("{"..mcom)
  393. end
  394. -- Loop through all captured statements
  395. for _,stmt in ipairs(lines) do
  396. -- Substitute macro parameters.
  397. local st = gsub(stmt, "[%w_]+", subst)
  398. st = definesubst(st)
  399. st = gsub(st, "%s*%.%.%s*", "") -- Token paste a..b.
  400. if mcom and sub(st, 1, 1) ~= "|" then wcomment(st) end
  401. -- Emit statement. Use a protected call for better diagnostics.
  402. local ok, err = pcall(dostmt, st)
  403. if not ok then
  404. -- Add the captured statement to the error.
  405. wprinterr(err, "\n", g_indent, "| ", stmt,
  406. "\t[MACRO ", name, " (", #mparams, ")]\n")
  407. end
  408. end
  409. if mcom then wcomment("}"..mcom) end
  410. mac_active[opname] = nil
  411. end
  412. end
  413. end
  414. -- An .endmacro pseudo-opcode outside of a macro definition is an error.
  415. map_coreop[".endmacro_0"] = function(params)
  416. wfatal(".endmacro without .macro")
  417. end
  418. -- Dump all macros and their contents (with -PP only).
  419. local function dumpmacros(out, lvl)
  420. sort(mac_list)
  421. out:write("Macros:\n")
  422. for _,opname in ipairs(mac_list) do
  423. local name = sub(opname, 1, -3)
  424. local params, lines = map_op[opname]()
  425. out:write(format(" %-20s %s\n", name, concat(params, ", ")))
  426. if lvl > 1 then
  427. for _,line in ipairs(lines) do
  428. out:write(" |", line, "\n")
  429. end
  430. out:write("\n")
  431. end
  432. end
  433. out:write("\n")
  434. end
  435. -- Check for unfinished macro definitions.
  436. local function checkmacros()
  437. if mac_capture then
  438. wprinterr(g_fname, ":", mac_lineno,
  439. ": error: unfinished .macro `", mac_name ,"'\n")
  440. end
  441. end
  442. ------------------------------------------------------------------------------
  443. -- Support variables for captures.
  444. local cap_lineno, cap_name
  445. local cap_buffers = {}
  446. local cap_used = {}
  447. -- Start a capture.
  448. map_coreop[".capture_1"] = function(params)
  449. if not params then return "name" end
  450. wflush()
  451. local name = params[1]
  452. if not match(name, "^[%a_][%w_]*$") then
  453. wfatal("bad capture name `"..name.."'")
  454. end
  455. if cap_name then
  456. wfatal("already capturing to `"..cap_name.."' since line "..cap_lineno)
  457. end
  458. cap_name = name
  459. cap_lineno = g_lineno
  460. -- Create or continue a capture buffer and start the output line capture.
  461. local buf = cap_buffers[name]
  462. if not buf then buf = {}; cap_buffers[name] = buf end
  463. g_capbuffer = buf
  464. g_synclineno = 0
  465. end
  466. -- Stop a capture.
  467. map_coreop[".endcapture_0"] = function(params)
  468. wflush()
  469. if not cap_name then wfatal(".endcapture without a valid .capture") end
  470. cap_name = nil
  471. cap_lineno = nil
  472. g_capbuffer = nil
  473. g_synclineno = 0
  474. end
  475. -- Dump a capture buffer.
  476. map_coreop[".dumpcapture_1"] = function(params)
  477. if not params then return "name" end
  478. wflush()
  479. local name = params[1]
  480. if not match(name, "^[%a_][%w_]*$") then
  481. wfatal("bad capture name `"..name.."'")
  482. end
  483. cap_used[name] = true
  484. wline(function(out)
  485. local buf = cap_buffers[name]
  486. if buf then wdumplines(out, buf) end
  487. end)
  488. g_synclineno = 0
  489. end
  490. -- Dump all captures and their buffers (with -PP only).
  491. local function dumpcaptures(out, lvl)
  492. out:write("Captures:\n")
  493. for name,buf in pairs(cap_buffers) do
  494. out:write(format(" %-20s %4s)\n", name, "("..#buf))
  495. if lvl > 1 then
  496. local bar = rep("=", 76)
  497. out:write(" ", bar, "\n")
  498. for _,line in ipairs(buf) do
  499. out:write(" ", line, "\n")
  500. end
  501. out:write(" ", bar, "\n\n")
  502. end
  503. end
  504. out:write("\n")
  505. end
  506. -- Check for unfinished or unused captures.
  507. local function checkcaptures()
  508. if cap_name then
  509. wprinterr(g_fname, ":", cap_lineno,
  510. ": error: unfinished .capture `", cap_name,"'\n")
  511. return
  512. end
  513. for name in pairs(cap_buffers) do
  514. if not cap_used[name] then
  515. wprinterr(g_fname, ":*: error: missing .dumpcapture ", name ,"\n")
  516. end
  517. end
  518. end
  519. ------------------------------------------------------------------------------
  520. -- Sections names.
  521. local map_sections = {}
  522. -- Pseudo-opcode to define code sections.
  523. -- TODO: Data sections, BSS sections. Needs extra C code and API.
  524. map_coreop[".section_*"] = function(params)
  525. if not params then return "name..." end
  526. if #map_sections > 0 then werror("duplicate section definition") end
  527. wflush()
  528. for sn,name in ipairs(params) do
  529. local opname = "."..name.."_0"
  530. if not match(name, "^[%a][%w_]*$") or
  531. map_op[opname] or map_op["."..name.."_*"] then
  532. werror("bad section name `"..name.."'")
  533. end
  534. map_sections[#map_sections+1] = name
  535. wline(format("#define DASM_SECTION_%s\t%d", upper(name), sn-1))
  536. map_op[opname] = function(params) g_arch.section(sn-1) end
  537. end
  538. wline(format("#define DASM_MAXSECTION\t\t%d", #map_sections))
  539. end
  540. -- Dump all sections.
  541. local function dumpsections(out, lvl)
  542. out:write("Sections:\n")
  543. for _,name in ipairs(map_sections) do
  544. out:write(format(" %s\n", name))
  545. end
  546. out:write("\n")
  547. end
  548. ------------------------------------------------------------------------------
  549. -- Replacement for customized Lua, which lacks the package library.
  550. local prefix = ""
  551. if not require then
  552. function require(name)
  553. local fp = assert(io.open(prefix..name..".lua"))
  554. local s = fp:read("*a")
  555. assert(fp:close())
  556. return assert(loadstring(s, "@"..name..".lua"))()
  557. end
  558. end
  559. -- Load architecture-specific module.
  560. local function loadarch(arch)
  561. if not match(arch, "^[%w_]+$") then return "bad arch name" end
  562. _G._map_def = map_def
  563. local ok, m_arch = pcall(require, "dasm_"..arch)
  564. if not ok then return "cannot load module: "..m_arch end
  565. g_arch = m_arch
  566. wflush = m_arch.passcb(wline, werror, wfatal, wwarn)
  567. m_arch.setup(arch, g_opt)
  568. map_op, map_def = m_arch.mergemaps(map_coreop, map_def)
  569. end
  570. -- Dump architecture description.
  571. function opt_map.dumparch(args)
  572. local name = optparam(args)
  573. if not g_arch then
  574. local err = loadarch(name)
  575. if err then opterror(err) end
  576. end
  577. local t = {}
  578. for name in pairs(map_coreop) do t[#t+1] = name end
  579. for name in pairs(map_op) do t[#t+1] = name end
  580. sort(t)
  581. local out = stdout
  582. local _arch = g_arch._info
  583. out:write(format("%s version %s, released %s, %s\n",
  584. _info.name, _info.version, _info.release, _info.url))
  585. g_arch.dumparch(out)
  586. local pseudo = true
  587. out:write("Pseudo-Opcodes:\n")
  588. for _,sname in ipairs(t) do
  589. local name, nparam = match(sname, "^(.+)_([0-9%*])$")
  590. if name then
  591. if pseudo and sub(name, 1, 1) ~= "." then
  592. out:write("\nOpcodes:\n")
  593. pseudo = false
  594. end
  595. local f = map_op[sname]
  596. local s
  597. if nparam ~= "*" then nparam = nparam + 0 end
  598. if nparam == 0 then
  599. s = ""
  600. elseif type(f) == "string" then
  601. s = map_op[".template__"](nil, f, nparam)
  602. else
  603. s = f(nil, nparam)
  604. end
  605. if type(s) == "table" then
  606. for _,s2 in ipairs(s) do
  607. out:write(format(" %-12s %s\n", name, s2))
  608. end
  609. else
  610. out:write(format(" %-12s %s\n", name, s))
  611. end
  612. end
  613. end
  614. out:write("\n")
  615. exit(0)
  616. end
  617. -- Pseudo-opcode to set the architecture.
  618. -- Only initially available (map_op is replaced when called).
  619. map_op[".arch_1"] = function(params)
  620. if not params then return "name" end
  621. local err = loadarch(params[1])
  622. if err then wfatal(err) end
  623. wline(format("#if DASM_VERSION != %d", _info.vernum))
  624. wline('#error "Version mismatch between DynASM and included encoding engine"')
  625. wline("#endif")
  626. end
  627. -- Dummy .arch pseudo-opcode to improve the error report.
  628. map_coreop[".arch_1"] = function(params)
  629. if not params then return "name" end
  630. wfatal("duplicate .arch statement")
  631. end
  632. ------------------------------------------------------------------------------
  633. -- Dummy pseudo-opcode. Don't confuse '.nop' with 'nop'.
  634. map_coreop[".nop_*"] = function(params)
  635. if not params then return "[ignored...]" end
  636. end
  637. -- Pseudo-opcodes to raise errors.
  638. map_coreop[".error_1"] = function(params)
  639. if not params then return "message" end
  640. werror(params[1])
  641. end
  642. map_coreop[".fatal_1"] = function(params)
  643. if not params then return "message" end
  644. wfatal(params[1])
  645. end
  646. -- Dump all user defined elements.
  647. local function dumpdef(out)
  648. local lvl = g_opt.dumpdef
  649. if lvl == 0 then return end
  650. dumpsections(out, lvl)
  651. dumpdefines(out, lvl)
  652. if g_arch then g_arch.dumpdef(out, lvl) end
  653. dumpmacros(out, lvl)
  654. dumpcaptures(out, lvl)
  655. end
  656. ------------------------------------------------------------------------------
  657. -- Helper for splitstmt.
  658. local splitlvl
  659. local function splitstmt_one(c)
  660. if c == "(" then
  661. splitlvl = ")"..splitlvl
  662. elseif c == "[" then
  663. splitlvl = "]"..splitlvl
  664. elseif c == "{" then
  665. splitlvl = "}"..splitlvl
  666. elseif c == ")" or c == "]" or c == "}" then
  667. if sub(splitlvl, 1, 1) ~= c then werror("unbalanced (), [] or {}") end
  668. splitlvl = sub(splitlvl, 2)
  669. elseif splitlvl == "" then
  670. return " \0 "
  671. end
  672. return c
  673. end
  674. -- Split statement into (pseudo-)opcode and params.
  675. local function splitstmt(stmt)
  676. -- Convert label with trailing-colon into .label statement.
  677. local label = match(stmt, "^%s*(.+):%s*$")
  678. if label then return ".label", {label} end
  679. -- Split at commas and equal signs, but obey parentheses and brackets.
  680. splitlvl = ""
  681. stmt = gsub(stmt, "[,%(%)%[%]{}]", splitstmt_one)
  682. if splitlvl ~= "" then werror("unbalanced () or []") end
  683. -- Split off opcode.
  684. local op, other = match(stmt, "^%s*([^%s%z]+)%s*(.*)$")
  685. if not op then werror("bad statement syntax") end
  686. -- Split parameters.
  687. local params = {}
  688. for p in gmatch(other, "%s*(%Z+)%z?") do
  689. params[#params+1] = gsub(p, "%s+$", "")
  690. end
  691. if #params > 16 then werror("too many parameters") end
  692. params.op = op
  693. return op, params
  694. end
  695. -- Process a single statement.
  696. dostmt = function(stmt)
  697. -- Ignore empty statements.
  698. if match(stmt, "^%s*$") then return end
  699. -- Capture macro defs before substitution.
  700. if mac_capture then return mac_capture(stmt) end
  701. stmt = definesubst(stmt)
  702. -- Emit C code without parsing the line.
  703. if sub(stmt, 1, 1) == "|" then
  704. local tail = sub(stmt, 2)
  705. wflush()
  706. if sub(tail, 1, 2) == "//" then wcomment(tail) else wline(tail, true) end
  707. return
  708. end
  709. -- Split into (pseudo-)opcode and params.
  710. local op, params = splitstmt(stmt)
  711. -- Get opcode handler (matching # of parameters or generic handler).
  712. local f = map_op[op.."_"..#params] or map_op[op.."_*"]
  713. if not f then
  714. if not g_arch then wfatal("first statement must be .arch") end
  715. -- Improve error report.
  716. for i=0,9 do
  717. if map_op[op.."_"..i] then
  718. werror("wrong number of parameters for `"..op.."'")
  719. end
  720. end
  721. werror("unknown statement `"..op.."'")
  722. end
  723. -- Call opcode handler or special handler for template strings.
  724. if type(f) == "string" then
  725. map_op[".template__"](params, f)
  726. else
  727. f(params)
  728. end
  729. end
  730. -- Process a single line.
  731. local function doline(line)
  732. if g_opt.flushline then wflush() end
  733. -- Assembler line?
  734. local indent, aline = match(line, "^(%s*)%|(.*)$")
  735. if not aline then
  736. -- No, plain C code line, need to flush first.
  737. wflush()
  738. wsync()
  739. wline(line, false)
  740. return
  741. end
  742. g_indent = indent -- Remember current line indentation.
  743. -- Emit C code (even from macros). Avoids echo and line parsing.
  744. if sub(aline, 1, 1) == "|" then
  745. if not mac_capture then
  746. wsync()
  747. elseif g_opt.comment then
  748. wsync()
  749. wcomment(aline)
  750. end
  751. dostmt(aline)
  752. return
  753. end
  754. -- Echo assembler line as a comment.
  755. if g_opt.comment then
  756. wsync()
  757. wcomment(aline)
  758. end
  759. -- Strip assembler comments.
  760. aline = gsub(aline, "//.*$", "")
  761. -- Split line into statements at semicolons.
  762. if match(aline, ";") then
  763. for stmt in gmatch(aline, "[^;]+") do dostmt(stmt) end
  764. else
  765. dostmt(aline)
  766. end
  767. end
  768. ------------------------------------------------------------------------------
  769. -- Write DynASM header.
  770. local function dasmhead(out)
  771. out:write(format([[
  772. /*
  773. ** This file has been pre-processed with DynASM.
  774. ** %s
  775. ** DynASM version %s, DynASM %s version %s
  776. ** DO NOT EDIT! The original file is in "%s".
  777. */
  778. ]], _info.url,
  779. _info.version, g_arch._info.arch, g_arch._info.version,
  780. g_fname))
  781. end
  782. -- Read input file.
  783. readfile = function(fin)
  784. g_indent = ""
  785. g_lineno = 0
  786. g_synclineno = -1
  787. -- Process all lines.
  788. for line in fin:lines() do
  789. g_lineno = g_lineno + 1
  790. g_curline = line
  791. local ok, err = pcall(doline, line)
  792. if not ok and wprinterr(err, "\n") then return true end
  793. end
  794. wflush()
  795. -- Close input file.
  796. assert(fin == stdin or fin:close())
  797. end
  798. -- Write output file.
  799. local function writefile(outfile)
  800. local fout
  801. -- Open output file.
  802. if outfile == nil or outfile == "-" then
  803. fout = stdout
  804. else
  805. fout = assert(io.open(outfile, "w"))
  806. end
  807. -- Write all buffered lines
  808. wdumplines(fout, g_wbuffer)
  809. -- Close output file.
  810. assert(fout == stdout or fout:close())
  811. -- Optionally dump definitions.
  812. dumpdef(fout == stdout and stderr or stdout)
  813. end
  814. -- Translate an input file to an output file.
  815. local function translate(infile, outfile)
  816. g_wbuffer = {}
  817. g_indent = ""
  818. g_lineno = 0
  819. g_synclineno = -1
  820. -- Put header.
  821. wline(dasmhead)
  822. -- Read input file.
  823. local fin
  824. if infile == "-" then
  825. g_fname = "(stdin)"
  826. fin = stdin
  827. else
  828. g_fname = infile
  829. fin = assert(io.open(infile, "r"))
  830. end
  831. readfile(fin)
  832. -- Check for errors.
  833. if not g_arch then
  834. wprinterr(g_fname, ":*: error: missing .arch directive\n")
  835. end
  836. checkconds()
  837. checkmacros()
  838. checkcaptures()
  839. if g_errcount ~= 0 then
  840. stderr:write(g_fname, ":*: info: ", g_errcount, " error",
  841. (type(g_errcount) == "number" and g_errcount > 1) and "s" or "",
  842. " in input file -- no output file generated.\n")
  843. dumpdef(stderr)
  844. exit(1)
  845. end
  846. -- Write output file.
  847. writefile(outfile)
  848. end
  849. ------------------------------------------------------------------------------
  850. -- Print help text.
  851. function opt_map.help()
  852. stdout:write("DynASM -- ", _info.description, ".\n")
  853. stdout:write("DynASM ", _info.version, " ", _info.release, " ", _info.url, "\n")
  854. stdout:write[[
  855. Usage: dynasm [OPTION]... INFILE.dasc|-
  856. -h, --help Display this help text.
  857. -V, --version Display version and copyright information.
  858. -o, --outfile FILE Output file name (default is stdout).
  859. -I, --include DIR Add directory to the include search path.
  860. -c, --ccomment Use /* */ comments for assembler lines.
  861. -C, --cppcomment Use // comments for assembler lines (default).
  862. -N, --nocomment Suppress assembler lines in output.
  863. -M, --maccomment Show macro expansions as comments (default off).
  864. -L, --nolineno Suppress CPP line number information in output.
  865. -F, --flushline Flush action list for every line.
  866. -D NAME[=SUBST] Define a substitution.
  867. -U NAME Undefine a substitution.
  868. -P, --dumpdef Dump defines, macros, etc. Repeat for more output.
  869. -A, --dumparch ARCH Load architecture ARCH and dump description.
  870. ]]
  871. exit(0)
  872. end
  873. -- Print version information.
  874. function opt_map.version()
  875. stdout:write(format("%s version %s, released %s\n%s\n\n%s",
  876. _info.name, _info.version, _info.release, _info.url, _info.copyright))
  877. exit(0)
  878. end
  879. -- Misc. options.
  880. function opt_map.outfile(args) g_opt.outfile = optparam(args) end
  881. function opt_map.include(args) insert(g_opt.include, 1, optparam(args)) end
  882. function opt_map.ccomment() g_opt.comment = "/*|"; g_opt.endcomment = " */" end
  883. function opt_map.cppcomment() g_opt.comment = "//|"; g_opt.endcomment = "" end
  884. function opt_map.nocomment() g_opt.comment = false end
  885. function opt_map.maccomment() g_opt.maccomment = true end
  886. function opt_map.nolineno() g_opt.cpp = false end
  887. function opt_map.flushline() g_opt.flushline = true end
  888. function opt_map.dumpdef() g_opt.dumpdef = g_opt.dumpdef + 1 end
  889. ------------------------------------------------------------------------------
  890. -- Short aliases for long options.
  891. local opt_alias = {
  892. h = "help", ["?"] = "help", V = "version",
  893. o = "outfile", I = "include",
  894. c = "ccomment", C = "cppcomment", N = "nocomment", M = "maccomment",
  895. L = "nolineno", F = "flushline",
  896. P = "dumpdef", A = "dumparch",
  897. }
  898. -- Parse single option.
  899. local function parseopt(opt, args)
  900. opt_current = #opt == 1 and "-"..opt or "--"..opt
  901. local f = opt_map[opt] or opt_map[opt_alias[opt]]
  902. if not f then
  903. opterror("unrecognized option `", opt_current, "'. Try `--help'.\n")
  904. end
  905. f(args)
  906. end
  907. -- Parse arguments.
  908. local function parseargs(args)
  909. -- Default options.
  910. g_opt.comment = "//|"
  911. g_opt.endcomment = ""
  912. g_opt.cpp = true
  913. g_opt.dumpdef = 0
  914. g_opt.include = { "" }
  915. -- Process all option arguments.
  916. args.argn = 1
  917. repeat
  918. local a = args[args.argn]
  919. if not a then break end
  920. local lopt, opt = match(a, "^%-(%-?)(.+)")
  921. if not opt then break end
  922. args.argn = args.argn + 1
  923. if lopt == "" then
  924. -- Loop through short options.
  925. for o in gmatch(opt, ".") do parseopt(o, args) end
  926. else
  927. -- Long option.
  928. parseopt(opt, args)
  929. end
  930. until false
  931. -- Check for proper number of arguments.
  932. local nargs = #args - args.argn + 1
  933. if nargs ~= 1 then
  934. if nargs == 0 then
  935. if g_opt.dumpdef > 0 then return dumpdef(stdout) end
  936. end
  937. opt_map.help()
  938. end
  939. -- Translate a single input file to a single output file
  940. -- TODO: Handle multiple files?
  941. translate(args[args.argn], g_opt.outfile)
  942. end
  943. ------------------------------------------------------------------------------
  944. -- Add the directory dynasm.lua resides in to the Lua module search path.
  945. local arg = arg
  946. if arg and arg[0] then
  947. prefix = match(arg[0], "^(.*[/\\])")
  948. if package and prefix then package.path = prefix.."?.lua;"..package.path end
  949. end
  950. -- Start DynASM.
  951. parseargs{...}
  952. ------------------------------------------------------------------------------