dasm_arm.lua 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125
  1. ------------------------------------------------------------------------------
  2. -- DynASM ARM module.
  3. --
  4. -- Copyright (C) 2005-2021 Mike Pall. All rights reserved.
  5. -- See dynasm.lua for full copyright notice.
  6. ------------------------------------------------------------------------------
  7. -- Module information:
  8. local _info = {
  9. arch = "arm",
  10. description = "DynASM ARM module",
  11. version = "1.5.0",
  12. vernum = 10500,
  13. release = "2021-05-02",
  14. author = "Mike Pall",
  15. license = "MIT",
  16. }
  17. -- Exported glue functions for the arch-specific module.
  18. local _M = { _info = _info }
  19. -- Cache library functions.
  20. local type, tonumber, pairs, ipairs = type, tonumber, pairs, ipairs
  21. local assert, setmetatable, rawget = assert, setmetatable, rawget
  22. local _s = string
  23. local sub, format, byte, char = _s.sub, _s.format, _s.byte, _s.char
  24. local match, gmatch, gsub = _s.match, _s.gmatch, _s.gsub
  25. local concat, sort, insert = table.concat, table.sort, table.insert
  26. local bit = bit or require("bit")
  27. local band, shl, shr, sar = bit.band, bit.lshift, bit.rshift, bit.arshift
  28. local ror, tohex = bit.ror, bit.tohex
  29. -- Inherited tables and callbacks.
  30. local g_opt, g_arch
  31. local wline, werror, wfatal, wwarn
  32. -- Action name list.
  33. -- CHECK: Keep this in sync with the C code!
  34. local action_names = {
  35. "STOP", "SECTION", "ESC", "REL_EXT",
  36. "ALIGN", "REL_LG", "LABEL_LG",
  37. "REL_PC", "LABEL_PC", "IMM", "IMM12", "IMM16", "IMML8", "IMML12", "IMMV8",
  38. }
  39. -- Maximum number of section buffer positions for dasm_put().
  40. -- CHECK: Keep this in sync with the C code!
  41. local maxsecpos = 25 -- Keep this low, to avoid excessively long C lines.
  42. -- Action name -> action number.
  43. local map_action = {}
  44. for n,name in ipairs(action_names) do
  45. map_action[name] = n-1
  46. end
  47. -- Action list buffer.
  48. local actlist = {}
  49. -- Argument list for next dasm_put(). Start with offset 0 into action list.
  50. local actargs = { 0 }
  51. -- Current number of section buffer positions for dasm_put().
  52. local secpos = 1
  53. ------------------------------------------------------------------------------
  54. -- Dump action names and numbers.
  55. local function dumpactions(out)
  56. out:write("DynASM encoding engine action codes:\n")
  57. for n,name in ipairs(action_names) do
  58. local num = map_action[name]
  59. out:write(format(" %-10s %02X %d\n", name, num, num))
  60. end
  61. out:write("\n")
  62. end
  63. -- Write action list buffer as a huge static C array.
  64. local function writeactions(out, name)
  65. local nn = #actlist
  66. if nn == 0 then nn = 1; actlist[0] = map_action.STOP end
  67. out:write("static const unsigned int ", name, "[", nn, "] = {\n")
  68. for i = 1,nn-1 do
  69. assert(out:write("0x", tohex(actlist[i]), ",\n"))
  70. end
  71. assert(out:write("0x", tohex(actlist[nn]), "\n};\n\n"))
  72. end
  73. ------------------------------------------------------------------------------
  74. -- Add word to action list.
  75. local function wputxw(n)
  76. assert(n >= 0 and n <= 0xffffffff and n % 1 == 0, "word out of range")
  77. actlist[#actlist+1] = n
  78. end
  79. -- Add action to list with optional arg. Advance buffer pos, too.
  80. local function waction(action, val, a, num)
  81. local w = assert(map_action[action], "bad action name `"..action.."'")
  82. wputxw(w * 0x10000 + (val or 0))
  83. if a then actargs[#actargs+1] = a end
  84. if a or num then secpos = secpos + (num or 1) end
  85. end
  86. -- Flush action list (intervening C code or buffer pos overflow).
  87. local function wflush(term)
  88. if #actlist == actargs[1] then return end -- Nothing to flush.
  89. if not term then waction("STOP") end -- Terminate action list.
  90. wline(format("dasm_put(Dst, %s);", concat(actargs, ", ")), true)
  91. actargs = { #actlist } -- Actionlist offset is 1st arg to next dasm_put().
  92. secpos = 1 -- The actionlist offset occupies a buffer position, too.
  93. end
  94. -- Put escaped word.
  95. local function wputw(n)
  96. if n <= 0x000fffff then waction("ESC") end
  97. wputxw(n)
  98. end
  99. -- Reserve position for word.
  100. local function wpos()
  101. local pos = #actlist+1
  102. actlist[pos] = ""
  103. return pos
  104. end
  105. -- Store word to reserved position.
  106. local function wputpos(pos, n)
  107. assert(n >= 0 and n <= 0xffffffff and n % 1 == 0, "word out of range")
  108. if n <= 0x000fffff then
  109. insert(actlist, pos+1, n)
  110. n = map_action.ESC * 0x10000
  111. end
  112. actlist[pos] = n
  113. end
  114. ------------------------------------------------------------------------------
  115. -- Global label name -> global label number. With auto assignment on 1st use.
  116. local next_global = 20
  117. local map_global = setmetatable({}, { __index = function(t, name)
  118. if not match(name, "^[%a_][%w_]*$") then werror("bad global label") end
  119. local n = next_global
  120. if n > 2047 then werror("too many global labels") end
  121. next_global = n + 1
  122. t[name] = n
  123. return n
  124. end})
  125. -- Dump global labels.
  126. local function dumpglobals(out, lvl)
  127. local t = {}
  128. for name, n in pairs(map_global) do t[n] = name end
  129. out:write("Global labels:\n")
  130. for i=20,next_global-1 do
  131. out:write(format(" %s\n", t[i]))
  132. end
  133. out:write("\n")
  134. end
  135. -- Write global label enum.
  136. local function writeglobals(out, prefix)
  137. local t = {}
  138. for name, n in pairs(map_global) do t[n] = name end
  139. out:write("enum {\n")
  140. for i=20,next_global-1 do
  141. out:write(" ", prefix, t[i], ",\n")
  142. end
  143. out:write(" ", prefix, "_MAX\n};\n")
  144. end
  145. -- Write global label names.
  146. local function writeglobalnames(out, name)
  147. local t = {}
  148. for name, n in pairs(map_global) do t[n] = name end
  149. out:write("static const char *const ", name, "[] = {\n")
  150. for i=20,next_global-1 do
  151. out:write(" \"", t[i], "\",\n")
  152. end
  153. out:write(" (const char *)0\n};\n")
  154. end
  155. ------------------------------------------------------------------------------
  156. -- Extern label name -> extern label number. With auto assignment on 1st use.
  157. local next_extern = 0
  158. local map_extern_ = {}
  159. local map_extern = setmetatable({}, { __index = function(t, name)
  160. -- No restrictions on the name for now.
  161. local n = next_extern
  162. if n > 2047 then werror("too many extern labels") end
  163. next_extern = n + 1
  164. t[name] = n
  165. map_extern_[n] = name
  166. return n
  167. end})
  168. -- Dump extern labels.
  169. local function dumpexterns(out, lvl)
  170. out:write("Extern labels:\n")
  171. for i=0,next_extern-1 do
  172. out:write(format(" %s\n", map_extern_[i]))
  173. end
  174. out:write("\n")
  175. end
  176. -- Write extern label names.
  177. local function writeexternnames(out, name)
  178. out:write("static const char *const ", name, "[] = {\n")
  179. for i=0,next_extern-1 do
  180. out:write(" \"", map_extern_[i], "\",\n")
  181. end
  182. out:write(" (const char *)0\n};\n")
  183. end
  184. ------------------------------------------------------------------------------
  185. -- Arch-specific maps.
  186. -- Ext. register name -> int. name.
  187. local map_archdef = { sp = "r13", lr = "r14", pc = "r15", }
  188. -- Int. register name -> ext. name.
  189. local map_reg_rev = { r13 = "sp", r14 = "lr", r15 = "pc", }
  190. local map_type = {} -- Type name -> { ctype, reg }
  191. local ctypenum = 0 -- Type number (for Dt... macros).
  192. -- Reverse defines for registers.
  193. function _M.revdef(s)
  194. return map_reg_rev[s] or s
  195. end
  196. local map_shift = { lsl = 0, lsr = 1, asr = 2, ror = 3, }
  197. local map_cond = {
  198. eq = 0, ne = 1, cs = 2, cc = 3, mi = 4, pl = 5, vs = 6, vc = 7,
  199. hi = 8, ls = 9, ge = 10, lt = 11, gt = 12, le = 13, al = 14,
  200. hs = 2, lo = 3,
  201. }
  202. ------------------------------------------------------------------------------
  203. -- Template strings for ARM instructions.
  204. local map_op = {
  205. -- Basic data processing instructions.
  206. and_3 = "e0000000DNPs",
  207. eor_3 = "e0200000DNPs",
  208. sub_3 = "e0400000DNPs",
  209. rsb_3 = "e0600000DNPs",
  210. add_3 = "e0800000DNPs",
  211. adc_3 = "e0a00000DNPs",
  212. sbc_3 = "e0c00000DNPs",
  213. rsc_3 = "e0e00000DNPs",
  214. tst_2 = "e1100000NP",
  215. teq_2 = "e1300000NP",
  216. cmp_2 = "e1500000NP",
  217. cmn_2 = "e1700000NP",
  218. orr_3 = "e1800000DNPs",
  219. mov_2 = "e1a00000DPs",
  220. bic_3 = "e1c00000DNPs",
  221. mvn_2 = "e1e00000DPs",
  222. and_4 = "e0000000DNMps",
  223. eor_4 = "e0200000DNMps",
  224. sub_4 = "e0400000DNMps",
  225. rsb_4 = "e0600000DNMps",
  226. add_4 = "e0800000DNMps",
  227. adc_4 = "e0a00000DNMps",
  228. sbc_4 = "e0c00000DNMps",
  229. rsc_4 = "e0e00000DNMps",
  230. tst_3 = "e1100000NMp",
  231. teq_3 = "e1300000NMp",
  232. cmp_3 = "e1500000NMp",
  233. cmn_3 = "e1700000NMp",
  234. orr_4 = "e1800000DNMps",
  235. mov_3 = "e1a00000DMps",
  236. bic_4 = "e1c00000DNMps",
  237. mvn_3 = "e1e00000DMps",
  238. lsl_3 = "e1a00000DMws",
  239. lsr_3 = "e1a00020DMws",
  240. asr_3 = "e1a00040DMws",
  241. ror_3 = "e1a00060DMws",
  242. rrx_2 = "e1a00060DMs",
  243. -- Multiply and multiply-accumulate.
  244. mul_3 = "e0000090NMSs",
  245. mla_4 = "e0200090NMSDs",
  246. umaal_4 = "e0400090DNMSs", -- v6
  247. mls_4 = "e0600090DNMSs", -- v6T2
  248. umull_4 = "e0800090DNMSs",
  249. umlal_4 = "e0a00090DNMSs",
  250. smull_4 = "e0c00090DNMSs",
  251. smlal_4 = "e0e00090DNMSs",
  252. -- Halfword multiply and multiply-accumulate.
  253. smlabb_4 = "e1000080NMSD", -- v5TE
  254. smlatb_4 = "e10000a0NMSD", -- v5TE
  255. smlabt_4 = "e10000c0NMSD", -- v5TE
  256. smlatt_4 = "e10000e0NMSD", -- v5TE
  257. smlawb_4 = "e1200080NMSD", -- v5TE
  258. smulwb_3 = "e12000a0NMS", -- v5TE
  259. smlawt_4 = "e12000c0NMSD", -- v5TE
  260. smulwt_3 = "e12000e0NMS", -- v5TE
  261. smlalbb_4 = "e1400080NMSD", -- v5TE
  262. smlaltb_4 = "e14000a0NMSD", -- v5TE
  263. smlalbt_4 = "e14000c0NMSD", -- v5TE
  264. smlaltt_4 = "e14000e0NMSD", -- v5TE
  265. smulbb_3 = "e1600080NMS", -- v5TE
  266. smultb_3 = "e16000a0NMS", -- v5TE
  267. smulbt_3 = "e16000c0NMS", -- v5TE
  268. smultt_3 = "e16000e0NMS", -- v5TE
  269. -- Miscellaneous data processing instructions.
  270. clz_2 = "e16f0f10DM", -- v5T
  271. rev_2 = "e6bf0f30DM", -- v6
  272. rev16_2 = "e6bf0fb0DM", -- v6
  273. revsh_2 = "e6ff0fb0DM", -- v6
  274. sel_3 = "e6800fb0DNM", -- v6
  275. usad8_3 = "e780f010NMS", -- v6
  276. usada8_4 = "e7800010NMSD", -- v6
  277. rbit_2 = "e6ff0f30DM", -- v6T2
  278. movw_2 = "e3000000DW", -- v6T2
  279. movt_2 = "e3400000DW", -- v6T2
  280. -- Note: the X encodes width-1, not width.
  281. sbfx_4 = "e7a00050DMvX", -- v6T2
  282. ubfx_4 = "e7e00050DMvX", -- v6T2
  283. -- Note: the X encodes the msb field, not the width.
  284. bfc_3 = "e7c0001fDvX", -- v6T2
  285. bfi_4 = "e7c00010DMvX", -- v6T2
  286. -- Packing and unpacking instructions.
  287. pkhbt_3 = "e6800010DNM", pkhbt_4 = "e6800010DNMv", -- v6
  288. pkhtb_3 = "e6800050DNM", pkhtb_4 = "e6800050DNMv", -- v6
  289. sxtab_3 = "e6a00070DNM", sxtab_4 = "e6a00070DNMv", -- v6
  290. sxtab16_3 = "e6800070DNM", sxtab16_4 = "e6800070DNMv", -- v6
  291. sxtah_3 = "e6b00070DNM", sxtah_4 = "e6b00070DNMv", -- v6
  292. sxtb_2 = "e6af0070DM", sxtb_3 = "e6af0070DMv", -- v6
  293. sxtb16_2 = "e68f0070DM", sxtb16_3 = "e68f0070DMv", -- v6
  294. sxth_2 = "e6bf0070DM", sxth_3 = "e6bf0070DMv", -- v6
  295. uxtab_3 = "e6e00070DNM", uxtab_4 = "e6e00070DNMv", -- v6
  296. uxtab16_3 = "e6c00070DNM", uxtab16_4 = "e6c00070DNMv", -- v6
  297. uxtah_3 = "e6f00070DNM", uxtah_4 = "e6f00070DNMv", -- v6
  298. uxtb_2 = "e6ef0070DM", uxtb_3 = "e6ef0070DMv", -- v6
  299. uxtb16_2 = "e6cf0070DM", uxtb16_3 = "e6cf0070DMv", -- v6
  300. uxth_2 = "e6ff0070DM", uxth_3 = "e6ff0070DMv", -- v6
  301. -- Saturating instructions.
  302. qadd_3 = "e1000050DMN", -- v5TE
  303. qsub_3 = "e1200050DMN", -- v5TE
  304. qdadd_3 = "e1400050DMN", -- v5TE
  305. qdsub_3 = "e1600050DMN", -- v5TE
  306. -- Note: the X for ssat* encodes sat_imm-1, not sat_imm.
  307. ssat_3 = "e6a00010DXM", ssat_4 = "e6a00010DXMp", -- v6
  308. usat_3 = "e6e00010DXM", usat_4 = "e6e00010DXMp", -- v6
  309. ssat16_3 = "e6a00f30DXM", -- v6
  310. usat16_3 = "e6e00f30DXM", -- v6
  311. -- Parallel addition and subtraction.
  312. sadd16_3 = "e6100f10DNM", -- v6
  313. sasx_3 = "e6100f30DNM", -- v6
  314. ssax_3 = "e6100f50DNM", -- v6
  315. ssub16_3 = "e6100f70DNM", -- v6
  316. sadd8_3 = "e6100f90DNM", -- v6
  317. ssub8_3 = "e6100ff0DNM", -- v6
  318. qadd16_3 = "e6200f10DNM", -- v6
  319. qasx_3 = "e6200f30DNM", -- v6
  320. qsax_3 = "e6200f50DNM", -- v6
  321. qsub16_3 = "e6200f70DNM", -- v6
  322. qadd8_3 = "e6200f90DNM", -- v6
  323. qsub8_3 = "e6200ff0DNM", -- v6
  324. shadd16_3 = "e6300f10DNM", -- v6
  325. shasx_3 = "e6300f30DNM", -- v6
  326. shsax_3 = "e6300f50DNM", -- v6
  327. shsub16_3 = "e6300f70DNM", -- v6
  328. shadd8_3 = "e6300f90DNM", -- v6
  329. shsub8_3 = "e6300ff0DNM", -- v6
  330. uadd16_3 = "e6500f10DNM", -- v6
  331. uasx_3 = "e6500f30DNM", -- v6
  332. usax_3 = "e6500f50DNM", -- v6
  333. usub16_3 = "e6500f70DNM", -- v6
  334. uadd8_3 = "e6500f90DNM", -- v6
  335. usub8_3 = "e6500ff0DNM", -- v6
  336. uqadd16_3 = "e6600f10DNM", -- v6
  337. uqasx_3 = "e6600f30DNM", -- v6
  338. uqsax_3 = "e6600f50DNM", -- v6
  339. uqsub16_3 = "e6600f70DNM", -- v6
  340. uqadd8_3 = "e6600f90DNM", -- v6
  341. uqsub8_3 = "e6600ff0DNM", -- v6
  342. uhadd16_3 = "e6700f10DNM", -- v6
  343. uhasx_3 = "e6700f30DNM", -- v6
  344. uhsax_3 = "e6700f50DNM", -- v6
  345. uhsub16_3 = "e6700f70DNM", -- v6
  346. uhadd8_3 = "e6700f90DNM", -- v6
  347. uhsub8_3 = "e6700ff0DNM", -- v6
  348. -- Load/store instructions.
  349. str_2 = "e4000000DL", str_3 = "e4000000DL", str_4 = "e4000000DL",
  350. strb_2 = "e4400000DL", strb_3 = "e4400000DL", strb_4 = "e4400000DL",
  351. ldr_2 = "e4100000DL", ldr_3 = "e4100000DL", ldr_4 = "e4100000DL",
  352. ldrb_2 = "e4500000DL", ldrb_3 = "e4500000DL", ldrb_4 = "e4500000DL",
  353. strh_2 = "e00000b0DL", strh_3 = "e00000b0DL",
  354. ldrh_2 = "e01000b0DL", ldrh_3 = "e01000b0DL",
  355. ldrd_2 = "e00000d0DL", ldrd_3 = "e00000d0DL", -- v5TE
  356. ldrsb_2 = "e01000d0DL", ldrsb_3 = "e01000d0DL",
  357. strd_2 = "e00000f0DL", strd_3 = "e00000f0DL", -- v5TE
  358. ldrsh_2 = "e01000f0DL", ldrsh_3 = "e01000f0DL",
  359. ldm_2 = "e8900000oR", ldmia_2 = "e8900000oR", ldmfd_2 = "e8900000oR",
  360. ldmda_2 = "e8100000oR", ldmfa_2 = "e8100000oR",
  361. ldmdb_2 = "e9100000oR", ldmea_2 = "e9100000oR",
  362. ldmib_2 = "e9900000oR", ldmed_2 = "e9900000oR",
  363. stm_2 = "e8800000oR", stmia_2 = "e8800000oR", stmfd_2 = "e8800000oR",
  364. stmda_2 = "e8000000oR", stmfa_2 = "e8000000oR",
  365. stmdb_2 = "e9000000oR", stmea_2 = "e9000000oR",
  366. stmib_2 = "e9800000oR", stmed_2 = "e9800000oR",
  367. pop_1 = "e8bd0000R", push_1 = "e92d0000R",
  368. -- Branch instructions.
  369. b_1 = "ea000000B",
  370. bl_1 = "eb000000B",
  371. blx_1 = "e12fff30C",
  372. bx_1 = "e12fff10M",
  373. -- Miscellaneous instructions.
  374. nop_0 = "e1a00000",
  375. mrs_1 = "e10f0000D",
  376. bkpt_1 = "e1200070K", -- v5T
  377. svc_1 = "ef000000T", swi_1 = "ef000000T",
  378. ud_0 = "e7f001f0",
  379. -- VFP instructions.
  380. ["vadd.f32_3"] = "ee300a00dnm",
  381. ["vadd.f64_3"] = "ee300b00Gdnm",
  382. ["vsub.f32_3"] = "ee300a40dnm",
  383. ["vsub.f64_3"] = "ee300b40Gdnm",
  384. ["vmul.f32_3"] = "ee200a00dnm",
  385. ["vmul.f64_3"] = "ee200b00Gdnm",
  386. ["vnmul.f32_3"] = "ee200a40dnm",
  387. ["vnmul.f64_3"] = "ee200b40Gdnm",
  388. ["vmla.f32_3"] = "ee000a00dnm",
  389. ["vmla.f64_3"] = "ee000b00Gdnm",
  390. ["vmls.f32_3"] = "ee000a40dnm",
  391. ["vmls.f64_3"] = "ee000b40Gdnm",
  392. ["vnmla.f32_3"] = "ee100a40dnm",
  393. ["vnmla.f64_3"] = "ee100b40Gdnm",
  394. ["vnmls.f32_3"] = "ee100a00dnm",
  395. ["vnmls.f64_3"] = "ee100b00Gdnm",
  396. ["vdiv.f32_3"] = "ee800a00dnm",
  397. ["vdiv.f64_3"] = "ee800b00Gdnm",
  398. ["vabs.f32_2"] = "eeb00ac0dm",
  399. ["vabs.f64_2"] = "eeb00bc0Gdm",
  400. ["vneg.f32_2"] = "eeb10a40dm",
  401. ["vneg.f64_2"] = "eeb10b40Gdm",
  402. ["vsqrt.f32_2"] = "eeb10ac0dm",
  403. ["vsqrt.f64_2"] = "eeb10bc0Gdm",
  404. ["vcmp.f32_2"] = "eeb40a40dm",
  405. ["vcmp.f64_2"] = "eeb40b40Gdm",
  406. ["vcmpe.f32_2"] = "eeb40ac0dm",
  407. ["vcmpe.f64_2"] = "eeb40bc0Gdm",
  408. ["vcmpz.f32_1"] = "eeb50a40d",
  409. ["vcmpz.f64_1"] = "eeb50b40Gd",
  410. ["vcmpze.f32_1"] = "eeb50ac0d",
  411. ["vcmpze.f64_1"] = "eeb50bc0Gd",
  412. vldr_2 = "ed100a00dl|ed100b00Gdl",
  413. vstr_2 = "ed000a00dl|ed000b00Gdl",
  414. vldm_2 = "ec900a00or",
  415. vldmia_2 = "ec900a00or",
  416. vldmdb_2 = "ed100a00or",
  417. vpop_1 = "ecbd0a00r",
  418. vstm_2 = "ec800a00or",
  419. vstmia_2 = "ec800a00or",
  420. vstmdb_2 = "ed000a00or",
  421. vpush_1 = "ed2d0a00r",
  422. ["vmov.f32_2"] = "eeb00a40dm|eeb00a00dY", -- #imm is VFPv3 only
  423. ["vmov.f64_2"] = "eeb00b40Gdm|eeb00b00GdY", -- #imm is VFPv3 only
  424. vmov_2 = "ee100a10Dn|ee000a10nD",
  425. vmov_3 = "ec500a10DNm|ec400a10mDN|ec500b10GDNm|ec400b10GmDN",
  426. vmrs_0 = "eef1fa10",
  427. vmrs_1 = "eef10a10D",
  428. vmsr_1 = "eee10a10D",
  429. ["vcvt.s32.f32_2"] = "eebd0ac0dm",
  430. ["vcvt.s32.f64_2"] = "eebd0bc0dGm",
  431. ["vcvt.u32.f32_2"] = "eebc0ac0dm",
  432. ["vcvt.u32.f64_2"] = "eebc0bc0dGm",
  433. ["vcvtr.s32.f32_2"] = "eebd0a40dm",
  434. ["vcvtr.s32.f64_2"] = "eebd0b40dGm",
  435. ["vcvtr.u32.f32_2"] = "eebc0a40dm",
  436. ["vcvtr.u32.f64_2"] = "eebc0b40dGm",
  437. ["vcvt.f32.s32_2"] = "eeb80ac0dm",
  438. ["vcvt.f64.s32_2"] = "eeb80bc0GdFm",
  439. ["vcvt.f32.u32_2"] = "eeb80a40dm",
  440. ["vcvt.f64.u32_2"] = "eeb80b40GdFm",
  441. ["vcvt.f32.f64_2"] = "eeb70bc0dGm",
  442. ["vcvt.f64.f32_2"] = "eeb70ac0GdFm",
  443. -- VFPv4 only:
  444. ["vfma.f32_3"] = "eea00a00dnm",
  445. ["vfma.f64_3"] = "eea00b00Gdnm",
  446. ["vfms.f32_3"] = "eea00a40dnm",
  447. ["vfms.f64_3"] = "eea00b40Gdnm",
  448. ["vfnma.f32_3"] = "ee900a40dnm",
  449. ["vfnma.f64_3"] = "ee900b40Gdnm",
  450. ["vfnms.f32_3"] = "ee900a00dnm",
  451. ["vfnms.f64_3"] = "ee900b00Gdnm",
  452. -- NYI: Advanced SIMD instructions.
  453. -- NYI: I have no need for these instructions right now:
  454. -- swp, swpb, strex, ldrex, strexd, ldrexd, strexb, ldrexb, strexh, ldrexh
  455. -- msr, nopv6, yield, wfe, wfi, sev, dbg, bxj, smc, srs, rfe
  456. -- cps, setend, pli, pld, pldw, clrex, dsb, dmb, isb
  457. -- stc, ldc, mcr, mcr2, mrc, mrc2, mcrr, mcrr2, mrrc, mrrc2, cdp, cdp2
  458. }
  459. -- Add mnemonics for "s" variants.
  460. do
  461. local t = {}
  462. for k,v in pairs(map_op) do
  463. if sub(v, -1) == "s" then
  464. local v2 = sub(v, 1, 2)..char(byte(v, 3)+1)..sub(v, 4, -2)
  465. t[sub(k, 1, -3).."s"..sub(k, -2)] = v2
  466. end
  467. end
  468. for k,v in pairs(t) do
  469. map_op[k] = v
  470. end
  471. end
  472. ------------------------------------------------------------------------------
  473. local function parse_gpr(expr)
  474. local tname, ovreg = match(expr, "^([%w_]+):(r1?[0-9])$")
  475. local tp = map_type[tname or expr]
  476. if tp then
  477. local reg = ovreg or tp.reg
  478. if not reg then
  479. werror("type `"..(tname or expr).."' needs a register override")
  480. end
  481. expr = reg
  482. end
  483. local r = match(expr, "^r(1?[0-9])$")
  484. if r then
  485. r = tonumber(r)
  486. if r <= 15 then return r, tp end
  487. end
  488. werror("bad register name `"..expr.."'")
  489. end
  490. local function parse_gpr_pm(expr)
  491. local pm, expr2 = match(expr, "^([+-]?)(.*)$")
  492. return parse_gpr(expr2), (pm == "-")
  493. end
  494. local function parse_vr(expr, tp)
  495. local t, r = match(expr, "^([sd])([0-9]+)$")
  496. if t == tp then
  497. r = tonumber(r)
  498. if r <= 31 then
  499. if t == "s" then return shr(r, 1), band(r, 1) end
  500. return band(r, 15), shr(r, 4)
  501. end
  502. end
  503. werror("bad register name `"..expr.."'")
  504. end
  505. local function parse_reglist(reglist)
  506. reglist = match(reglist, "^{%s*([^}]*)}$")
  507. if not reglist then werror("register list expected") end
  508. local rr = 0
  509. for p in gmatch(reglist..",", "%s*([^,]*),") do
  510. local rbit = shl(1, parse_gpr(gsub(p, "%s+$", "")))
  511. if band(rr, rbit) ~= 0 then
  512. werror("duplicate register `"..p.."'")
  513. end
  514. rr = rr + rbit
  515. end
  516. return rr
  517. end
  518. local function parse_vrlist(reglist)
  519. local ta, ra, tb, rb = match(reglist,
  520. "^{%s*([sd])([0-9]+)%s*%-%s*([sd])([0-9]+)%s*}$")
  521. ra, rb = tonumber(ra), tonumber(rb)
  522. if ta and ta == tb and ra and rb and ra <= 31 and rb <= 31 and ra <= rb then
  523. local nr = rb+1 - ra
  524. if ta == "s" then
  525. return shl(shr(ra,1),12)+shl(band(ra,1),22) + nr
  526. else
  527. return shl(band(ra,15),12)+shl(shr(ra,4),22) + nr*2 + 0x100
  528. end
  529. end
  530. werror("register list expected")
  531. end
  532. local function parse_imm(imm, bits, shift, scale, signed)
  533. imm = match(imm, "^#(.*)$")
  534. if not imm then werror("expected immediate operand") end
  535. local n = tonumber(imm)
  536. if n then
  537. local m = sar(n, scale)
  538. if shl(m, scale) == n then
  539. if signed then
  540. local s = sar(m, bits-1)
  541. if s == 0 then return shl(m, shift)
  542. elseif s == -1 then return shl(m + shl(1, bits), shift) end
  543. else
  544. if sar(m, bits) == 0 then return shl(m, shift) end
  545. end
  546. end
  547. werror("out of range immediate `"..imm.."'")
  548. else
  549. waction("IMM", (signed and 32768 or 0)+scale*1024+bits*32+shift, imm)
  550. return 0
  551. end
  552. end
  553. local function parse_imm12(imm)
  554. local n = tonumber(imm)
  555. if n then
  556. local m = band(n)
  557. for i=0,-15,-1 do
  558. if shr(m, 8) == 0 then return m + shl(band(i, 15), 8) end
  559. m = ror(m, 2)
  560. end
  561. werror("out of range immediate `"..imm.."'")
  562. else
  563. waction("IMM12", 0, imm)
  564. return 0
  565. end
  566. end
  567. local function parse_imm16(imm)
  568. imm = match(imm, "^#(.*)$")
  569. if not imm then werror("expected immediate operand") end
  570. local n = tonumber(imm)
  571. if n then
  572. if shr(n, 16) == 0 then return band(n, 0x0fff) + shl(band(n, 0xf000), 4) end
  573. werror("out of range immediate `"..imm.."'")
  574. else
  575. waction("IMM16", 32*16, imm)
  576. return 0
  577. end
  578. end
  579. local function parse_imm_load(imm, ext)
  580. local n = tonumber(imm)
  581. if n then
  582. if ext then
  583. if n >= -255 and n <= 255 then
  584. local up = 0x00800000
  585. if n < 0 then n = -n; up = 0 end
  586. return shl(band(n, 0xf0), 4) + band(n, 0x0f) + up
  587. end
  588. else
  589. if n >= -4095 and n <= 4095 then
  590. if n >= 0 then return n+0x00800000 end
  591. return -n
  592. end
  593. end
  594. werror("out of range immediate `"..imm.."'")
  595. else
  596. waction(ext and "IMML8" or "IMML12", 32768 + shl(ext and 8 or 12, 5), imm)
  597. return 0
  598. end
  599. end
  600. local function parse_shift(shift, gprok)
  601. if shift == "rrx" then
  602. return 3 * 32
  603. else
  604. local s, s2 = match(shift, "^(%S+)%s*(.*)$")
  605. s = map_shift[s]
  606. if not s then werror("expected shift operand") end
  607. if sub(s2, 1, 1) == "#" then
  608. return parse_imm(s2, 5, 7, 0, false) + shl(s, 5)
  609. else
  610. if not gprok then werror("expected immediate shift operand") end
  611. return shl(parse_gpr(s2), 8) + shl(s, 5) + 16
  612. end
  613. end
  614. end
  615. local function parse_label(label, def)
  616. local prefix = sub(label, 1, 2)
  617. -- =>label (pc label reference)
  618. if prefix == "=>" then
  619. return "PC", 0, sub(label, 3)
  620. end
  621. -- ->name (global label reference)
  622. if prefix == "->" then
  623. return "LG", map_global[sub(label, 3)]
  624. end
  625. if def then
  626. -- [1-9] (local label definition)
  627. if match(label, "^[1-9]$") then
  628. return "LG", 10+tonumber(label)
  629. end
  630. else
  631. -- [<>][1-9] (local label reference)
  632. local dir, lnum = match(label, "^([<>])([1-9])$")
  633. if dir then -- Fwd: 1-9, Bkwd: 11-19.
  634. return "LG", lnum + (dir == ">" and 0 or 10)
  635. end
  636. -- extern label (extern label reference)
  637. local extname = match(label, "^extern%s+(%S+)$")
  638. if extname then
  639. return "EXT", map_extern[extname]
  640. end
  641. end
  642. werror("bad label `"..label.."'")
  643. end
  644. local function parse_load(params, nparams, n, op)
  645. local oplo = band(op, 255)
  646. local ext, ldrd = (oplo ~= 0), (oplo == 208)
  647. local d
  648. if (ldrd or oplo == 240) then
  649. d = band(shr(op, 12), 15)
  650. if band(d, 1) ~= 0 then werror("odd destination register") end
  651. end
  652. local pn = params[n]
  653. local p1, wb = match(pn, "^%[%s*(.-)%s*%](!?)$")
  654. local p2 = params[n+1]
  655. if not p1 then
  656. if not p2 then
  657. if match(pn, "^[<>=%-]") or match(pn, "^extern%s+") then
  658. local mode, n, s = parse_label(pn, false)
  659. waction("REL_"..mode, n + (ext and 0x1800 or 0x0800), s, 1)
  660. return op + 15 * 65536 + 0x01000000 + (ext and 0x00400000 or 0)
  661. end
  662. local reg, tailr = match(pn, "^([%w_:]+)%s*(.*)$")
  663. if reg and tailr ~= "" then
  664. local d, tp = parse_gpr(reg)
  665. if tp then
  666. waction(ext and "IMML8" or "IMML12", 32768 + 32*(ext and 8 or 12),
  667. format(tp.ctypefmt, tailr))
  668. return op + shl(d, 16) + 0x01000000 + (ext and 0x00400000 or 0)
  669. end
  670. end
  671. end
  672. werror("expected address operand")
  673. end
  674. if wb == "!" then op = op + 0x00200000 end
  675. if p2 then
  676. if wb == "!" then werror("bad use of '!'") end
  677. local p3 = params[n+2]
  678. op = op + shl(parse_gpr(p1), 16)
  679. local imm = match(p2, "^#(.*)$")
  680. if imm then
  681. local m = parse_imm_load(imm, ext)
  682. if p3 then werror("too many parameters") end
  683. op = op + m + (ext and 0x00400000 or 0)
  684. else
  685. local m, neg = parse_gpr_pm(p2)
  686. if ldrd and (m == d or m-1 == d) then werror("register conflict") end
  687. op = op + m + (neg and 0 or 0x00800000) + (ext and 0 or 0x02000000)
  688. if p3 then op = op + parse_shift(p3) end
  689. end
  690. else
  691. local p1a, p2 = match(p1, "^([^,%s]*)%s*(.*)$")
  692. op = op + shl(parse_gpr(p1a), 16) + 0x01000000
  693. if p2 ~= "" then
  694. local imm = match(p2, "^,%s*#(.*)$")
  695. if imm then
  696. local m = parse_imm_load(imm, ext)
  697. op = op + m + (ext and 0x00400000 or 0)
  698. else
  699. local p2a, p3 = match(p2, "^,%s*([^,%s]*)%s*,?%s*(.*)$")
  700. local m, neg = parse_gpr_pm(p2a)
  701. if ldrd and (m == d or m-1 == d) then werror("register conflict") end
  702. op = op + m + (neg and 0 or 0x00800000) + (ext and 0 or 0x02000000)
  703. if p3 ~= "" then
  704. if ext then werror("too many parameters") end
  705. op = op + parse_shift(p3)
  706. end
  707. end
  708. else
  709. if wb == "!" then werror("bad use of '!'") end
  710. op = op + (ext and 0x00c00000 or 0x00800000)
  711. end
  712. end
  713. return op
  714. end
  715. local function parse_vload(q)
  716. local reg, imm = match(q, "^%[%s*([^,%s]*)%s*(.*)%]$")
  717. if reg then
  718. local d = shl(parse_gpr(reg), 16)
  719. if imm == "" then return d end
  720. imm = match(imm, "^,%s*#(.*)$")
  721. if imm then
  722. local n = tonumber(imm)
  723. if n then
  724. if n >= -1020 and n <= 1020 and n%4 == 0 then
  725. return d + (n >= 0 and n/4+0x00800000 or -n/4)
  726. end
  727. werror("out of range immediate `"..imm.."'")
  728. else
  729. waction("IMMV8", 32768 + 32*8, imm)
  730. return d
  731. end
  732. end
  733. else
  734. if match(q, "^[<>=%-]") or match(q, "^extern%s+") then
  735. local mode, n, s = parse_label(q, false)
  736. waction("REL_"..mode, n + 0x2800, s, 1)
  737. return 15 * 65536
  738. end
  739. local reg, tailr = match(q, "^([%w_:]+)%s*(.*)$")
  740. if reg and tailr ~= "" then
  741. local d, tp = parse_gpr(reg)
  742. if tp then
  743. waction("IMMV8", 32768 + 32*8, format(tp.ctypefmt, tailr))
  744. return shl(d, 16)
  745. end
  746. end
  747. end
  748. werror("expected address operand")
  749. end
  750. ------------------------------------------------------------------------------
  751. -- Handle opcodes defined with template strings.
  752. local function parse_template(params, template, nparams, pos)
  753. local op = tonumber(sub(template, 1, 8), 16)
  754. local n = 1
  755. local vr = "s"
  756. -- Process each character.
  757. for p in gmatch(sub(template, 9), ".") do
  758. local q = params[n]
  759. if p == "D" then
  760. op = op + shl(parse_gpr(q), 12); n = n + 1
  761. elseif p == "N" then
  762. op = op + shl(parse_gpr(q), 16); n = n + 1
  763. elseif p == "S" then
  764. op = op + shl(parse_gpr(q), 8); n = n + 1
  765. elseif p == "M" then
  766. op = op + parse_gpr(q); n = n + 1
  767. elseif p == "d" then
  768. local r,h = parse_vr(q, vr); op = op+shl(r,12)+shl(h,22); n = n + 1
  769. elseif p == "n" then
  770. local r,h = parse_vr(q, vr); op = op+shl(r,16)+shl(h,7); n = n + 1
  771. elseif p == "m" then
  772. local r,h = parse_vr(q, vr); op = op+r+shl(h,5); n = n + 1
  773. elseif p == "P" then
  774. local imm = match(q, "^#(.*)$")
  775. if imm then
  776. op = op + parse_imm12(imm) + 0x02000000
  777. else
  778. op = op + parse_gpr(q)
  779. end
  780. n = n + 1
  781. elseif p == "p" then
  782. op = op + parse_shift(q, true); n = n + 1
  783. elseif p == "L" then
  784. op = parse_load(params, nparams, n, op)
  785. elseif p == "l" then
  786. op = op + parse_vload(q)
  787. elseif p == "B" then
  788. local mode, n, s = parse_label(q, false)
  789. waction("REL_"..mode, n, s, 1)
  790. elseif p == "C" then -- blx gpr vs. blx label.
  791. if match(q, "^([%w_]+):(r1?[0-9])$") or match(q, "^r(1?[0-9])$") then
  792. op = op + parse_gpr(q)
  793. else
  794. if op < 0xe0000000 then werror("unconditional instruction") end
  795. local mode, n, s = parse_label(q, false)
  796. waction("REL_"..mode, n, s, 1)
  797. op = 0xfa000000
  798. end
  799. elseif p == "F" then
  800. vr = "s"
  801. elseif p == "G" then
  802. vr = "d"
  803. elseif p == "o" then
  804. local r, wb = match(q, "^([^!]*)(!?)$")
  805. op = op + shl(parse_gpr(r), 16) + (wb == "!" and 0x00200000 or 0)
  806. n = n + 1
  807. elseif p == "R" then
  808. op = op + parse_reglist(q); n = n + 1
  809. elseif p == "r" then
  810. op = op + parse_vrlist(q); n = n + 1
  811. elseif p == "W" then
  812. op = op + parse_imm16(q); n = n + 1
  813. elseif p == "v" then
  814. op = op + parse_imm(q, 5, 7, 0, false); n = n + 1
  815. elseif p == "w" then
  816. local imm = match(q, "^#(.*)$")
  817. if imm then
  818. op = op + parse_imm(q, 5, 7, 0, false); n = n + 1
  819. else
  820. op = op + shl(parse_gpr(q), 8) + 16
  821. end
  822. elseif p == "X" then
  823. op = op + parse_imm(q, 5, 16, 0, false); n = n + 1
  824. elseif p == "Y" then
  825. local imm = tonumber(match(q, "^#(.*)$")); n = n + 1
  826. if not imm or shr(imm, 8) ~= 0 then
  827. werror("bad immediate operand")
  828. end
  829. op = op + shl(band(imm, 0xf0), 12) + band(imm, 0x0f)
  830. elseif p == "K" then
  831. local imm = tonumber(match(q, "^#(.*)$")); n = n + 1
  832. if not imm or shr(imm, 16) ~= 0 then
  833. werror("bad immediate operand")
  834. end
  835. op = op + shl(band(imm, 0xfff0), 4) + band(imm, 0x000f)
  836. elseif p == "T" then
  837. op = op + parse_imm(q, 24, 0, 0, false); n = n + 1
  838. elseif p == "s" then
  839. -- Ignored.
  840. else
  841. assert(false)
  842. end
  843. end
  844. wputpos(pos, op)
  845. end
  846. map_op[".template__"] = function(params, template, nparams)
  847. if not params then return template:gsub("%x%x%x%x%x%x%x%x", "") end
  848. -- Limit number of section buffer positions used by a single dasm_put().
  849. -- A single opcode needs a maximum of 3 positions.
  850. if secpos+3 > maxsecpos then wflush() end
  851. local pos = wpos()
  852. local lpos, apos, spos = #actlist, #actargs, secpos
  853. local ok, err
  854. for t in gmatch(template, "[^|]+") do
  855. ok, err = pcall(parse_template, params, t, nparams, pos)
  856. if ok then return end
  857. secpos = spos
  858. actlist[lpos+1] = nil
  859. actlist[lpos+2] = nil
  860. actlist[lpos+3] = nil
  861. actargs[apos+1] = nil
  862. actargs[apos+2] = nil
  863. actargs[apos+3] = nil
  864. end
  865. error(err, 0)
  866. end
  867. ------------------------------------------------------------------------------
  868. -- Pseudo-opcode to mark the position where the action list is to be emitted.
  869. map_op[".actionlist_1"] = function(params)
  870. if not params then return "cvar" end
  871. local name = params[1] -- No syntax check. You get to keep the pieces.
  872. wline(function(out) writeactions(out, name) end)
  873. end
  874. -- Pseudo-opcode to mark the position where the global enum is to be emitted.
  875. map_op[".globals_1"] = function(params)
  876. if not params then return "prefix" end
  877. local prefix = params[1] -- No syntax check. You get to keep the pieces.
  878. wline(function(out) writeglobals(out, prefix) end)
  879. end
  880. -- Pseudo-opcode to mark the position where the global names are to be emitted.
  881. map_op[".globalnames_1"] = function(params)
  882. if not params then return "cvar" end
  883. local name = params[1] -- No syntax check. You get to keep the pieces.
  884. wline(function(out) writeglobalnames(out, name) end)
  885. end
  886. -- Pseudo-opcode to mark the position where the extern names are to be emitted.
  887. map_op[".externnames_1"] = function(params)
  888. if not params then return "cvar" end
  889. local name = params[1] -- No syntax check. You get to keep the pieces.
  890. wline(function(out) writeexternnames(out, name) end)
  891. end
  892. ------------------------------------------------------------------------------
  893. -- Label pseudo-opcode (converted from trailing colon form).
  894. map_op[".label_1"] = function(params)
  895. if not params then return "[1-9] | ->global | =>pcexpr" end
  896. if secpos+1 > maxsecpos then wflush() end
  897. local mode, n, s = parse_label(params[1], true)
  898. if mode == "EXT" then werror("bad label definition") end
  899. waction("LABEL_"..mode, n, s, 1)
  900. end
  901. ------------------------------------------------------------------------------
  902. -- Pseudo-opcodes for data storage.
  903. map_op[".long_*"] = function(params)
  904. if not params then return "imm..." end
  905. for _,p in ipairs(params) do
  906. local n = tonumber(p)
  907. if not n then werror("bad immediate `"..p.."'") end
  908. if n < 0 then n = n + 2^32 end
  909. wputw(n)
  910. if secpos+2 > maxsecpos then wflush() end
  911. end
  912. end
  913. -- Alignment pseudo-opcode.
  914. map_op[".align_1"] = function(params)
  915. if not params then return "numpow2" end
  916. if secpos+1 > maxsecpos then wflush() end
  917. local align = tonumber(params[1])
  918. if align then
  919. local x = align
  920. -- Must be a power of 2 in the range (2 ... 256).
  921. for i=1,8 do
  922. x = x / 2
  923. if x == 1 then
  924. waction("ALIGN", align-1, nil, 1) -- Action byte is 2**n-1.
  925. return
  926. end
  927. end
  928. end
  929. werror("bad alignment")
  930. end
  931. ------------------------------------------------------------------------------
  932. -- Pseudo-opcode for (primitive) type definitions (map to C types).
  933. map_op[".type_3"] = function(params, nparams)
  934. if not params then
  935. return nparams == 2 and "name, ctype" or "name, ctype, reg"
  936. end
  937. local name, ctype, reg = params[1], params[2], params[3]
  938. if not match(name, "^[%a_][%w_]*$") then
  939. werror("bad type name `"..name.."'")
  940. end
  941. local tp = map_type[name]
  942. if tp then
  943. werror("duplicate type `"..name.."'")
  944. end
  945. -- Add #type to defines. A bit unclean to put it in map_archdef.
  946. map_archdef["#"..name] = "sizeof("..ctype..")"
  947. -- Add new type and emit shortcut define.
  948. local num = ctypenum + 1
  949. map_type[name] = {
  950. ctype = ctype,
  951. ctypefmt = format("Dt%X(%%s)", num),
  952. reg = reg,
  953. }
  954. wline(format("#define Dt%X(_V) (int)(ptrdiff_t)&(((%s *)0)_V)", num, ctype))
  955. ctypenum = num
  956. end
  957. map_op[".type_2"] = map_op[".type_3"]
  958. -- Dump type definitions.
  959. local function dumptypes(out, lvl)
  960. local t = {}
  961. for name in pairs(map_type) do t[#t+1] = name end
  962. sort(t)
  963. out:write("Type definitions:\n")
  964. for _,name in ipairs(t) do
  965. local tp = map_type[name]
  966. local reg = tp.reg or ""
  967. out:write(format(" %-20s %-20s %s\n", name, tp.ctype, reg))
  968. end
  969. out:write("\n")
  970. end
  971. ------------------------------------------------------------------------------
  972. -- Set the current section.
  973. function _M.section(num)
  974. waction("SECTION", num)
  975. wflush(true) -- SECTION is a terminal action.
  976. end
  977. ------------------------------------------------------------------------------
  978. -- Dump architecture description.
  979. function _M.dumparch(out)
  980. out:write(format("DynASM %s version %s, released %s\n\n",
  981. _info.arch, _info.version, _info.release))
  982. dumpactions(out)
  983. end
  984. -- Dump all user defined elements.
  985. function _M.dumpdef(out, lvl)
  986. dumptypes(out, lvl)
  987. dumpglobals(out, lvl)
  988. dumpexterns(out, lvl)
  989. end
  990. ------------------------------------------------------------------------------
  991. -- Pass callbacks from/to the DynASM core.
  992. function _M.passcb(wl, we, wf, ww)
  993. wline, werror, wfatal, wwarn = wl, we, wf, ww
  994. return wflush
  995. end
  996. -- Setup the arch-specific module.
  997. function _M.setup(arch, opt)
  998. g_arch, g_opt = arch, opt
  999. end
  1000. -- Merge the core maps and the arch-specific maps.
  1001. function _M.mergemaps(map_coreop, map_def)
  1002. setmetatable(map_op, { __index = function(t, k)
  1003. local v = map_coreop[k]
  1004. if v then return v end
  1005. local k1, cc, k2 = match(k, "^(.-)(..)([._].*)$")
  1006. local cv = map_cond[cc]
  1007. if cv then
  1008. local v = rawget(t, k1..k2)
  1009. if type(v) == "string" then
  1010. local scv = format("%x", cv)
  1011. return gsub(scv..sub(v, 2), "|e", "|"..scv)
  1012. end
  1013. end
  1014. end })
  1015. setmetatable(map_def, { __index = map_archdef })
  1016. return map_op, map_def
  1017. end
  1018. return _M
  1019. ------------------------------------------------------------------------------