mirror of
https://github.com/odin-lang/Odin.git
synced 2026-06-20 00:52:33 +00:00
Predicated FP round (FRINTN/P/M/Z/A/X/I, FRECPX), reversed predicated shifts (ASRR/LSLR/LSRR) and FP (FSUBR/FDIVR), FP compare (FCMEQ/GE/GT/ NE/UO + vs-zero FCMLE/FCMLT), integer compare aliases (CMPLE/LO/LS/LT), predicate logical (NANDS/NORS/ORNS), predicate break (BRKPA/BRKPB, BRKA/BRKB + flag-setting BRKAS/BRKBS), SVE2 EOR3/BCAX, INSR, COMPACT. New specgen SVE section: a generic emitter assembles each form all-zero then one variant per field at its max (Z 31, 3-bit Pg 7, 4-bit Pd/Pg/Pn/ Pm 15, GPR wzr/xzr) and derives mask = ~union. Operand placements verified vs llvm-mc: the reversed/destructive ops put Zm at VN (5-9); the CMPLE/LO/LS/LT aliases swap operands (VM/VN); EOR3/BCAX place the 3rd src at VM and 4th at VN. All 22 representative forms byte-exact and decode-clean; 461 tests green. (BRKN + CPY/EXT/MOV/NOT_P/FFR/XAR stragglers next.)
697 lines
34 KiB
Lua
697 lines
34 KiB
Lua
#!/usr/bin/env luajit
|
|
-- rexcode · Brendan Punsky (dotbmp@github), original author
|
|
--
|
|
-- Encode-form spec generator (arm64). Expands compact per-instruction specs
|
|
-- into ENCODING_TABLE entries, deriving `bits` from llvm-mc (the oracle) and
|
|
-- `mask` empirically: assemble each form with operand registers at 0 and at 31;
|
|
-- the differing bits are operand-driven, so mask = ~(bits0 ^ bits31). Per-form
|
|
-- assembly makes it robust: an arrangement llvm-mc rejects is reported, skipped.
|
|
--
|
|
-- Output replaces the SPECGEN:BEGIN..SPECGEN:END region of encoding_table.odin
|
|
-- in place; the hand-written core is untouched. Every bit pattern is therefore
|
|
-- reproducible and llvm-mc-backed.
|
|
--
|
|
-- Two spec kinds:
|
|
-- * uniform — every operand shares one arrangement T (iterate ALL_ARR).
|
|
-- * diff — operands have different arrangements (long/wide/narrow/XTN);
|
|
-- each item lists the arrangement TUPLE per size-variant.
|
|
--
|
|
-- Run: luajit tablegen/specgen.lua (from arm64/, or with a full path)
|
|
|
|
local bit = require("bit")
|
|
local LLVM = "llvm-mc --assemble --arch=aarch64 --mattr=+fullfp16 --show-encoding"
|
|
local DIR = (arg[0]:match("^(.*)/[^/]*$")) or "."
|
|
local TABLE = DIR .. "/encoding_table.odin"
|
|
|
|
local ARR = {
|
|
["8B"]={vt="V_8B",asm="8b"}, ["16B"]={vt="V_16B",asm="16b"},
|
|
["4H"]={vt="V_4H",asm="4h"}, ["8H"] ={vt="V_8H", asm="8h"},
|
|
["2S"]={vt="V_2S",asm="2s"}, ["4S"] ={vt="V_4S", asm="4s"},
|
|
["1D"]={vt="V_1D",asm="1d"}, ["2D"] ={vt="V_2D", asm="2d"},
|
|
-- half-precision FP arrangements (distinct operand type + FP16 feature)
|
|
["4HF"]={vt="V_4H_FP16",asm="4h",feat="FP16"}, ["8HF"]={vt="V_8H_FP16",asm="8h",feat="FP16"},
|
|
}
|
|
local ALL_ARR = {"8B","16B","4H","8H","2S","4S","2D"}
|
|
|
|
-- scalar SIMD register destinations (across-lanes reductions): token -> reg
|
|
local SCAL = { B={vt="B_REG",asm="b"}, H={vt="H_REG",asm="h"}, S={vt="S_REG",asm="s"}, D={vt="D_REG",asm="d"} }
|
|
-- a tuple token is either a vector arrangement (in ARR) or a scalar size (in SCAL)
|
|
local function tok_asm(t, r) if ARR[t] then return "v"..r.."."..ARR[t].asm else return SCAL[t].asm..r end end
|
|
local function tok_vt(t) if ARR[t] then return ARR[t].vt else return SCAL[t].vt end end
|
|
|
|
local function word(line)
|
|
local p = io.popen(string.format("printf '%%s\\n' '%s' | %s 2>/dev/null", line, LLVM))
|
|
local out = p:read("*a"); p:close()
|
|
local b1,b2,b3,b4 = out:match("0x(%x%x),0x(%x%x),0x(%x%x),0x(%x%x)")
|
|
if not b1 then return nil end
|
|
return tonumber(b4..b3..b2..b1, 16)
|
|
end
|
|
|
|
local function padded(tokens, n)
|
|
local t = {}
|
|
for i = 1, 4 do t[i] = tokens[i] or ".NONE" end
|
|
return "{" .. table.concat(t, ", ") .. "}"
|
|
end
|
|
|
|
local sections, skips, n_forms, n_mnem = {}, {}, 0, 0
|
|
|
|
-- Emit one mnemonic's block from a list of arrangement tuples (operand order).
|
|
-- enc_str is the prebuilt "{.VD, .VN, .VM, .NONE}" enc array text.
|
|
local function emit(mnem, llvm, enc_str, feature, variants)
|
|
local rows = {}
|
|
for _, tup in ipairs(variants) do
|
|
local function mk(r)
|
|
local parts = {}
|
|
for i, t in ipairs(tup) do parts[i] = tok_asm(t, r) end
|
|
return llvm.." "..table.concat(parts, ", ")
|
|
end
|
|
local w0, w31 = word(mk(0)), word(mk(31))
|
|
if w0 and w31 then
|
|
local mask = bit.band(bit.bnot(bit.bxor(w0, w31)), 0xFFFFFFFF)
|
|
local ops = {}
|
|
for i, t in ipairs(tup) do ops[i] = "."..tok_vt(t) end
|
|
local f = feature
|
|
for _, tk in ipairs(tup) do if ARR[tk] and ARR[tk].feat then f = ARR[tk].feat end end
|
|
rows[#rows+1] = string.format("\t\t{.%s, %s, %s, 0x%s, 0x%s, .%s, {}},",
|
|
mnem, padded(ops, #tup), enc_str, bit.tohex(w0):upper(), bit.tohex(mask):upper(), f)
|
|
n_forms = n_forms + 1
|
|
else
|
|
skips[#skips+1] = mnem.." "..table.concat(tup, "/")
|
|
end
|
|
end
|
|
if #rows == 0 then return nil end
|
|
n_mnem = n_mnem + 1
|
|
return string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n"))
|
|
end
|
|
|
|
-- Compare-against-zero: a 2-register shape whose asm carries a literal #0
|
|
-- (integer) or #0.0 (FP) third operand that emits no bits. Modeled with two
|
|
-- register operands (VD, VN); mask derived by varying the registers only.
|
|
local function emit_cmp0(mnem, llvm, fp, arr)
|
|
local rows = {}
|
|
local suf = fp and ", #0.0" or ", #0"
|
|
for _, a in ipairs(arr) do
|
|
local function mk(r) return string.format("%s v%d.%s, v%d.%s%s", llvm, r, ARR[a].asm, r, ARR[a].asm, suf) end
|
|
local w0, w31 = word(mk(0)), word(mk(31))
|
|
if w0 and w31 then
|
|
local mask = bit.band(bit.bnot(bit.bxor(w0, w31)), 0xFFFFFFFF)
|
|
local f = ARR[a].feat or "NEON"
|
|
rows[#rows+1] = string.format(
|
|
"\t\t{.%s, {.%s, .%s, .NONE, .NONE}, {.VD, .VN, .NONE, .NONE}, 0x%s, 0x%s, .%s, {}},",
|
|
mnem, ARR[a].vt, ARR[a].vt, bit.tohex(w0):upper(), bit.tohex(mask):upper(), f)
|
|
n_forms = n_forms + 1
|
|
else
|
|
skips[#skips+1] = mnem.." "..a
|
|
end
|
|
end
|
|
if #rows == 0 then return nil end
|
|
n_mnem = n_mnem + 1
|
|
return string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n"))
|
|
end
|
|
|
|
-- ---- Uniform shapes (all operands share one arrangement) -------------------
|
|
local VD_VN_VM = padded({".VD",".VN",".VM"}, 3)
|
|
local VD_VN = padded({".VD",".VN"}, 2)
|
|
|
|
local UNIFORM = {
|
|
{ title="three-same (integer)", enc=VD_VN_VM, nreg=3, items={
|
|
{"SHADD","shadd"},{"UHADD","uhadd"},{"SHSUB","shsub"},{"UHSUB","uhsub"},
|
|
{"SRHADD","srhadd"},{"URHADD","urhadd"},
|
|
{"SQADD","sqadd"},{"UQADD","uqadd"},{"SQSUB","sqsub"},{"UQSUB","uqsub"},
|
|
{"SMAX","smax"},{"UMAX","umax"},{"SMIN","smin"},{"UMIN","umin"},
|
|
{"SABD","sabd"},{"UABD","uabd"},{"SABA","saba"},{"UABA","uaba"},
|
|
{"MLA_V","mla"},{"MLS_V","mls"},
|
|
{"CMGE","cmge"},{"CMHS","cmhs"},{"CMTST","cmtst"},
|
|
{"SQDMULH","sqdmulh"},{"SQRDMULH","sqrdmulh"},
|
|
{"ADDP_V","addp"},{"SMAXP","smaxp"},{"SMINP","sminp"},{"UMAXP","umaxp"},{"UMINP","uminp"},
|
|
{"SSHL","sshl"},{"USHL","ushl"},{"SRSHL","srshl"},{"URSHL","urshl"},
|
|
}},
|
|
{ title="permute (ZIP/UZP/TRN)", enc=VD_VN_VM, nreg=3, items={
|
|
{"ZIP1","zip1"},{"ZIP2","zip2"},{"UZP1","uzp1"},{"UZP2","uzp2"},{"TRN1","trn1"},{"TRN2","trn2"},
|
|
}},
|
|
{ title="two-register misc", enc=VD_VN, nreg=2, items={
|
|
{"ABS_V","abs"},{"NEG_V","neg"},
|
|
{"NOT_V","not"},{"RBIT_V","rbit"},
|
|
{"REV16_V","rev16"},{"REV32_V","rev32"},{"REV64","rev64"},
|
|
{"CLS_V","cls"},{"CLZ_V","clz"},{"CNT","cnt"},
|
|
{"URECPE_V","urecpe"},{"URSQRTE_V","ursqrte"},
|
|
}},
|
|
{ title="floating-point three-same", enc=VD_VN_VM, nreg=3, arr={"2S","4S","2D","4HF","8HF"}, items={
|
|
{"FMAX_V","fmax"},{"FMIN_V","fmin"},{"FMAXNM_V","fmaxnm"},{"FMINNM_V","fminnm"},
|
|
{"FMULX","fmulx"},{"FRECPS","frecps"},{"FRSQRTS","frsqrts"},
|
|
{"FACGE","facge"},{"FACGT","facgt"},
|
|
{"FCMEQ","fcmeq"},{"FCMGE","fcmge"},{"FCMGT","fcmgt"},
|
|
{"FADDP_V","faddp"},{"FMAXP_V","fmaxp"},{"FMINP_V","fminp"},
|
|
{"FMAXNMP","fmaxnmp"},{"FMINNMP","fminnmp"},
|
|
}},
|
|
{ title="floating-point two-register", enc=VD_VN, nreg=2, arr={"2S","4S","2D","4HF","8HF"}, items={
|
|
{"FABS_V","fabs"},{"FNEG_V","fneg"},{"FSQRT_V","fsqrt"},
|
|
{"FRINTA_V","frinta"},{"FRINTI_V","frinti"},{"FRINTM_V","frintm"},{"FRINTN_V","frintn"},
|
|
{"FRINTP_V","frintp"},{"FRINTX_V","frintx"},{"FRINTZ_V","frintz"},
|
|
{"FRECPE","frecpe"},{"FRSQRTE","frsqrte"},
|
|
}},
|
|
{ title="floating-point convert (vector, register form)", enc=VD_VN, nreg=2, arr={"2S","4S","2D","4HF","8HF"}, items={
|
|
{"FCVTAS_V","fcvtas"},{"FCVTAU_V","fcvtau"},{"FCVTMS_V","fcvtms"},{"FCVTMU_V","fcvtmu"},
|
|
{"FCVTNS_V","fcvtns"},{"FCVTNU_V","fcvtnu"},{"FCVTPS_V","fcvtps"},{"FCVTPU_V","fcvtpu"},
|
|
{"FCVTZS_V","fcvtzs"},{"FCVTZU_V","fcvtzu"},{"SCVTF_V","scvtf"},{"UCVTF_V","ucvtf"},
|
|
}},
|
|
}
|
|
for _, fam in ipairs(UNIFORM) do
|
|
local blk = {}
|
|
for _, it in ipairs(fam.items) do
|
|
local variants = {}
|
|
for _, a in ipairs(fam.arr or ALL_ARR) do
|
|
local tup = {}; for i = 1, fam.nreg do tup[i] = a end
|
|
variants[#variants+1] = tup
|
|
end
|
|
local b = emit(it[1], it[2], fam.enc, "NEON", variants)
|
|
if b then blk[#blk+1] = b end
|
|
end
|
|
sections[#sections+1] = "\t// Advanced SIMD "..fam.title..".\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- Mixed-arrangement shapes (long / wide / narrow / XTN) ------------------
|
|
-- arrangement tuples per size-variant (operand order: dst, n, [m]). Base
|
|
-- mnemonics take the low-half source, the "2" variants the high half.
|
|
local LONG_LO = {{"8H","8B","8B"},{"4S","4H","4H"},{"2D","2S","2S"}}
|
|
local LONG_HI = {{"8H","16B","16B"},{"4S","8H","8H"},{"2D","4S","4S"}}
|
|
local WIDE_LO = {{"8H","8H","8B"},{"4S","4S","4H"},{"2D","2D","2S"}}
|
|
local WIDE_HI = {{"8H","8H","16B"},{"4S","4S","8H"},{"2D","2D","4S"}}
|
|
local NARR_LO = {{"8B","8H","8H"},{"4H","4S","4S"},{"2S","2D","2D"}}
|
|
local NARR_HI = {{"16B","8H","8H"},{"8H","4S","4S"},{"4S","2D","2D"}}
|
|
local XTN_LO = {{"8B","8H"},{"4H","4S"},{"2S","2D"}}
|
|
local XTN_HI = {{"16B","8H"},{"8H","4S"},{"4S","2D"}}
|
|
-- pairwise-long: Vd.<wide>, Vn.<narrow> (half the lanes, double the element size)
|
|
local PLONG = {{"4H","8B"},{"8H","16B"},{"2S","4H"},{"4S","8H"},{"1D","2S"},{"2D","4S"}}
|
|
-- across-lanes: scalar dst of the element size, Vn.<T>
|
|
local ACROSS = {{"B","8B"},{"B","16B"},{"H","4H"},{"H","8H"},{"S","4S"}}
|
|
-- across-lanes long: scalar dst of 2x the element size, Vn.<T>
|
|
local ACROSSL = {{"H","8B"},{"H","16B"},{"S","4H"},{"S","8H"},{"D","4S"}}
|
|
-- FP across-lanes: scalar dst (S for .4S, H for the FP16 forms), Vn.<T>
|
|
local ACROSSF = {{"S","4S"},{"H","4HF"},{"H","8HF"}}
|
|
-- FP widen (FCVTL): Vd.<wide>, Vn.<narrow>; narrow (FCVTN/FCVTXN): Vd.<narrow>, Vn.<wide>
|
|
local FCVTL_LO = {{"4S","4HF"},{"2D","2S"}}
|
|
local FCVTL_HI = {{"4S","8HF"},{"2D","4S"}}
|
|
local FCVTN_LO = {{"4HF","4S"},{"2S","2D"}}
|
|
local FCVTN_HI = {{"8HF","4S"},{"4S","2D"}}
|
|
local FCVTXN_LO = {{"2S","2D"}}
|
|
local FCVTXN_HI = {{"4S","2D"}}
|
|
-- SXTL/UXTL widen (= SSHLL/USHLL #0): Vd.<wide>, Vn.<narrow>; shift is implicit
|
|
-- in the static bits, so this is a plain 2-register form (no shift operand).
|
|
local SXTL_LO = {{"8H","8B"},{"4S","4H"},{"2D","2S"}}
|
|
local SXTL_HI = {{"8H","16B"},{"4S","8H"},{"2D","4S"}}
|
|
|
|
local DIFF = {
|
|
{ title="three-different (long)", enc=VD_VN_VM, items={
|
|
{"SADDL","saddl",LONG_LO},{"SADDL2","saddl2",LONG_HI},
|
|
{"UADDL","uaddl",LONG_LO},{"UADDL2","uaddl2",LONG_HI},
|
|
{"SSUBL","ssubl",LONG_LO},{"SSUBL2","ssubl2",LONG_HI},
|
|
{"USUBL","usubl",LONG_LO},{"USUBL2","usubl2",LONG_HI},
|
|
{"SMULL_V","smull",LONG_LO},{"SMULL2_V","smull2",LONG_HI},
|
|
{"UMULL_V","umull",LONG_LO},{"UMULL2_V","umull2",LONG_HI},
|
|
{"SMLAL","smlal",LONG_LO},{"SMLAL2","smlal2",LONG_HI},
|
|
{"UMLAL","umlal",LONG_LO},{"UMLAL2","umlal2",LONG_HI},
|
|
{"SMLSL","smlsl",LONG_LO},{"SMLSL2","smlsl2",LONG_HI},
|
|
{"UMLSL","umlsl",LONG_LO},{"UMLSL2","umlsl2",LONG_HI},
|
|
{"SQDMULL","sqdmull",LONG_LO},{"SQDMULL2","sqdmull2",LONG_HI},
|
|
{"SQDMLAL","sqdmlal",LONG_LO},{"SQDMLAL2","sqdmlal2",LONG_HI},
|
|
{"SQDMLSL","sqdmlsl",LONG_LO},{"SQDMLSL2","sqdmlsl2",LONG_HI},
|
|
}},
|
|
{ title="three-different (wide)", enc=VD_VN_VM, items={
|
|
{"SADDW","saddw",WIDE_LO},{"SADDW2","saddw2",WIDE_HI},
|
|
{"UADDW","uaddw",WIDE_LO},{"UADDW2","uaddw2",WIDE_HI},
|
|
{"SSUBW","ssubw",WIDE_LO},{"SSUBW2","ssubw2",WIDE_HI},
|
|
{"USUBW","usubw",WIDE_LO},{"USUBW2","usubw2",WIDE_HI},
|
|
}},
|
|
{ title="three-different (narrow, halving)", enc=VD_VN_VM, items={
|
|
{"ADDHN","addhn",NARR_LO},{"ADDHN2","addhn2",NARR_HI},
|
|
{"SUBHN","subhn",NARR_LO},{"SUBHN2","subhn2",NARR_HI},
|
|
{"RADDHN","raddhn",NARR_LO},{"RADDHN2","raddhn2",NARR_HI},
|
|
{"RSUBHN","rsubhn",NARR_LO},{"RSUBHN2","rsubhn2",NARR_HI},
|
|
}},
|
|
{ title="two-register narrowing (XTN)", enc=VD_VN, items={
|
|
{"XTN","xtn",XTN_LO},{"XTN2","xtn2",XTN_HI},
|
|
{"SQXTN","sqxtn",XTN_LO},{"SQXTN2","sqxtn2",XTN_HI},
|
|
{"UQXTN","uqxtn",XTN_LO},{"UQXTN2","uqxtn2",XTN_HI},
|
|
{"SQXTUN","sqxtun",XTN_LO},{"SQXTUN2","sqxtun2",XTN_HI},
|
|
}},
|
|
{ title="two-register widen (SXTL/UXTL = SSHLL/USHLL #0)", enc=VD_VN, items={
|
|
{"SXTL","sxtl",SXTL_LO},{"SXTL2","sxtl2",SXTL_HI},
|
|
{"UXTL","uxtl",SXTL_LO},{"UXTL2","uxtl2",SXTL_HI},
|
|
}},
|
|
{ title="two-register pairwise long", enc=VD_VN, items={
|
|
{"SADDLP","saddlp",PLONG},{"UADDLP","uaddlp",PLONG},
|
|
{"SADALP","sadalp",PLONG},{"UADALP","uadalp",PLONG},
|
|
}},
|
|
{ title="across lanes", enc=VD_VN, items={
|
|
{"ADDV","addv",ACROSS},{"SMAXV","smaxv",ACROSS},{"SMINV","sminv",ACROSS},
|
|
{"UMAXV","umaxv",ACROSS},{"UMINV","uminv",ACROSS},
|
|
{"SADDLV","saddlv",ACROSSL},{"UADDLV","uaddlv",ACROSSL},
|
|
}},
|
|
{ title="floating-point across lanes", enc=VD_VN, items={
|
|
{"FMAXV_V","fmaxv",ACROSSF},{"FMINV_V","fminv",ACROSSF},
|
|
{"FMAXNMV","fmaxnmv",ACROSSF},{"FMINNMV","fminnmv",ACROSSF},
|
|
}},
|
|
{ title="floating-point widen / narrow", enc=VD_VN, items={
|
|
{"FCVTL","fcvtl",FCVTL_LO},{"FCVTL2","fcvtl2",FCVTL_HI},
|
|
{"FCVTN","fcvtn",FCVTN_LO},{"FCVTN2","fcvtn2",FCVTN_HI},
|
|
{"FCVTXN","fcvtxn",FCVTXN_LO},{"FCVTXN2","fcvtxn2",FCVTXN_HI},
|
|
}},
|
|
}
|
|
for _, fam in ipairs(DIFF) do
|
|
local blk = {}
|
|
for _, it in ipairs(fam.items) do
|
|
local b = emit(it[1], it[2], fam.enc, "NEON", it[3])
|
|
if b then blk[#blk+1] = b end
|
|
end
|
|
sections[#sections+1] = "\t// Advanced SIMD "..fam.title..".\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- Compare against zero (integer + floating-point) -----------------------
|
|
do
|
|
local FP_ARR = {"2S","4S","2D","4HF","8HF"}
|
|
local blk = {}
|
|
for _, b in ipairs({
|
|
emit_cmp0("CMLE","cmle",false,ALL_ARR), emit_cmp0("CMLT","cmlt",false,ALL_ARR),
|
|
emit_cmp0("FCMLE","fcmle",true,FP_ARR), emit_cmp0("FCMLT","fcmlt",true,FP_ARR),
|
|
}) do blk[#blk+1] = b end
|
|
sections[#sections+1] = "\t// Advanced SIMD compare against zero.\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- NEON shift-by-immediate ----------------------------------------------
|
|
-- <mnem> Vd.T, Vn.T, #shift. immh:immb carries element size + amount; mask is
|
|
-- derived empirically by also varying the shift (canon = operand bits 0, other =
|
|
-- all shift bits set). The encoder/decoder compute the amount (NEON_SHL/SHR_IMM).
|
|
local ESIZE = {["8B"]=8,["16B"]=8,["4H"]=16,["8H"]=16,["2S"]=32,["4S"]=32,["1D"]=64,["2D"]=64}
|
|
-- variants are arrangement pairs {dst, src}; the shift element size is the SOURCE
|
|
-- (same for same-arrangement shifts, the narrow input for widening shifts).
|
|
local function emit_shift(mnem, llvm, dir, variants)
|
|
local enc_tok = (dir == "L") and ".NEON_SHL_IMM" or ".NEON_SHR_IMM"
|
|
local rows = {}
|
|
for _, v in ipairs(variants) do
|
|
local dst, src = v[1], v[2]
|
|
-- Left shifts (incl. widening SSHLL) key immh on the source element;
|
|
-- right shifts (incl. narrowing SHRN) key it on the destination
|
|
-- (narrow result) element. For same-arrangement shifts these coincide.
|
|
local es = (dir == "L") and ESIZE[src] or ESIZE[dst]
|
|
local canon = (dir == "L") and 0 or es
|
|
local other = (dir == "L") and (es - 1) or 1
|
|
local da, sa = ARR[dst].asm, ARR[src].asm
|
|
local function mk(r, sh) return string.format("%s v%d.%s, v%d.%s, #%d", llvm, r, da, r, sa, sh) end
|
|
local bits, regV, shV = word(mk(0, canon)), word(mk(31, canon)), word(mk(0, other))
|
|
if bits and regV and shV then
|
|
local mask = bit.band(bit.bnot(bit.bor(bit.bxor(bits, regV), bit.bxor(bits, shV))), 0xFFFFFFFF)
|
|
rows[#rows+1] = string.format(
|
|
"\t\t{.%s, {.%s, .%s, .VEC_SHIFT, .NONE}, {.VD, .VN, %s, .NONE}, 0x%s, 0x%s, .NEON, {}},",
|
|
mnem, ARR[dst].vt, ARR[src].vt, enc_tok, bit.tohex(bits):upper(), bit.tohex(mask):upper())
|
|
n_forms = n_forms + 1
|
|
else
|
|
skips[#skips+1] = mnem.." "..dst.."/"..src
|
|
end
|
|
end
|
|
if #rows == 0 then return nil end
|
|
n_mnem = n_mnem + 1
|
|
return string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n"))
|
|
end
|
|
local SAME_SH = {}
|
|
for _, a in ipairs({"8B","16B","4H","8H","2S","4S","2D"}) do SAME_SH[#SAME_SH+1] = {a, a} end
|
|
local WSHL_LO = {{"8H","8B"},{"4S","4H"},{"2D","2S"}}
|
|
local WSHL_HI = {{"8H","16B"},{"4S","8H"},{"2D","4S"}}
|
|
-- narrowing right shifts: {narrow dst, wide src}; the `2` variants write the
|
|
-- high half (full 128-bit dst arrangement). esize keys on the narrow dst.
|
|
local NSHR_LO = {{"8B","8H"},{"4H","4S"},{"2S","2D"}}
|
|
local NSHR_HI = {{"16B","8H"},{"8H","4S"},{"4S","2D"}}
|
|
local SHIFTS = {
|
|
{"SHL_V","shl","L",SAME_SH},{"SLI","sli","L",SAME_SH},{"SQSHLU","sqshlu","L",SAME_SH},{"SQSHL_V","sqshl","L",SAME_SH},
|
|
{"SSHR","sshr","R",SAME_SH},{"USHR","ushr","R",SAME_SH},{"SRSHR","srshr","R",SAME_SH},{"URSHR","urshr","R",SAME_SH},
|
|
{"SSRA","ssra","R",SAME_SH},{"USRA","usra","R",SAME_SH},{"SRSRA","srsra","R",SAME_SH},{"URSRA","ursra","R",SAME_SH},
|
|
{"SRI","sri","R",SAME_SH},
|
|
{"SSHLL","sshll","L",WSHL_LO},{"SSHLL2","sshll2","L",WSHL_HI},
|
|
{"USHLL","ushll","L",WSHL_LO},{"USHLL2","ushll2","L",WSHL_HI},
|
|
-- narrowing right shifts (immh keys on the narrow destination element).
|
|
{"SHRN","shrn","R",NSHR_LO},{"SHRN2","shrn2","R",NSHR_HI},
|
|
{"RSHRN","rshrn","R",NSHR_LO},{"RSHRN2","rshrn2","R",NSHR_HI},
|
|
{"SQSHRN","sqshrn","R",NSHR_LO},{"SQSHRN2","sqshrn2","R",NSHR_HI},
|
|
{"UQSHRN","uqshrn","R",NSHR_LO},{"UQSHRN2","uqshrn2","R",NSHR_HI},
|
|
{"SQRSHRN","sqrshrn","R",NSHR_LO},{"SQRSHRN2","sqrshrn2","R",NSHR_HI},
|
|
{"UQRSHRN","uqrshrn","R",NSHR_LO},{"UQRSHRN2","uqrshrn2","R",NSHR_HI},
|
|
{"SQSHRUN","sqshrun","R",NSHR_LO},{"SQSHRUN2","sqshrun2","R",NSHR_HI},
|
|
{"SQRSHRUN","sqrshrun","R",NSHR_LO},{"SQRSHRUN2","sqrshrun2","R",NSHR_HI},
|
|
}
|
|
do
|
|
local blk = {}
|
|
for _, it in ipairs(SHIFTS) do
|
|
local b = emit_shift(it[1], it[2], it[3], it[4])
|
|
if b then blk[#blk+1] = b end
|
|
end
|
|
sections[#sections+1] = "\t// Advanced SIMD shift by immediate.\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- NEON copy / permute (MOV/MVN/DUP/INS/EXT) -----------------------------
|
|
-- These carry a lane index in imm5/imm4 whose element-size marker is fixed in
|
|
-- `bits`; the operand drives the bits above it. mask is derived by varying the
|
|
-- registers AND each index field to its maximum (mask = ~union of the deltas).
|
|
local function mask_of(base, variants)
|
|
local x = 0
|
|
for _, w in ipairs(variants) do x = bit.bor(x, bit.bxor(base, w)) end
|
|
return bit.band(bit.bnot(x), 0xFFFFFFFF)
|
|
end
|
|
-- element metadata: asm token, valid dst arrangements, max lane index, V_ELEM type.
|
|
local ELEM = {
|
|
B = {tok="b", dst={"8B","16B"}, idxmax=15, vt="V_ELEM_B"},
|
|
H = {tok="h", dst={"4H","8H"}, idxmax=7, vt="V_ELEM_H"},
|
|
S = {tok="s", dst={"2S","4S"}, idxmax=3, vt="V_ELEM_S"},
|
|
D = {tok="d", dst={"2D"}, idxmax=1, vt="V_ELEM_D"},
|
|
}
|
|
do
|
|
local blk = {}
|
|
local function emit_rows(mnem, rows)
|
|
if #rows > 0 then blk[#blk+1] = string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n")); n_mnem = n_mnem + 1 end
|
|
end
|
|
local function mkrow(mnem, ops, enc, b0, variants)
|
|
n_forms = n_forms + 1
|
|
return string.format("\t\t{.%s, %s, %s, 0x%s, 0x%s, .NEON, {}},",
|
|
mnem, ops, enc, bit.tohex(b0):upper(), bit.tohex(mask_of(b0, variants)):upper())
|
|
end
|
|
|
|
-- MOV Vd.T, Vn.T (= ORR Vd,Vn,Vn): the single source feeds both Vn and Vm.
|
|
local mov = {}
|
|
for _, a in ipairs({"8B","16B"}) do
|
|
local function mk(x) return string.format("mov v%d.%s, v%d.%s", x, ARR[a].asm, x, ARR[a].asm) end
|
|
local b0, b31 = word(mk(0)), word(mk(31))
|
|
if b0 and b31 then mov[#mov+1] = mkrow("MOV_V",
|
|
string.format("{.%s, .%s, .NONE, .NONE}", ARR[a].vt, ARR[a].vt),
|
|
"{.VD, .VN_VM_DUP, .NONE, .NONE}", b0, {b31}) end
|
|
end
|
|
emit_rows("MOV_V", mov)
|
|
|
|
-- MVN Vd.T, Vn.T (= NOT alias): plain two-register.
|
|
local mvn = {}
|
|
for _, a in ipairs({"8B","16B"}) do
|
|
local function mk(x) return string.format("mvn v%d.%s, v%d.%s", x, ARR[a].asm, x, ARR[a].asm) end
|
|
local b0, b31 = word(mk(0)), word(mk(31))
|
|
if b0 and b31 then mvn[#mvn+1] = mkrow("MVN_V",
|
|
string.format("{.%s, .%s, .NONE, .NONE}", ARR[a].vt, ARR[a].vt),
|
|
"{.VD, .VN, .NONE, .NONE}", b0, {b31}) end
|
|
end
|
|
emit_rows("MVN_V", mvn)
|
|
|
|
-- DUP: element form (Vd.T, Vn.Ts[i]) + general form (Vd.T, Wn/Xn).
|
|
local dup = {}
|
|
for _, ek in ipairs({"B","H","S","D"}) do
|
|
local e = ELEM[ek]
|
|
for _, a in ipairs(e.dst) do
|
|
local function mk(r,i) return string.format("dup v%d.%s, v%d.%s[%d]", r, ARR[a].asm, r, e.tok, i) end
|
|
local b0, br, bi = word(mk(0,0)), word(mk(31,0)), word(mk(0,e.idxmax))
|
|
if b0 and br and bi then dup[#dup+1] = mkrow("DUP_V",
|
|
string.format("{.%s, .%s, .VEC_INDEX, .NONE}", ARR[a].vt, e.vt),
|
|
"{.VD, .VN, .NEON_IDX5, .NONE}", b0, {br, bi}) end
|
|
end
|
|
end
|
|
for _, g in ipairs({{"8B","w","W_REG"},{"16B","w","W_REG"},{"4H","w","W_REG"},{"8H","w","W_REG"},{"2S","w","W_REG"},{"4S","w","W_REG"},{"2D","x","X_REG"}}) do
|
|
-- Vary Vd and Rn independently to 31 (Rn 31 is the zero register, asm
|
|
-- "wzr"/"xzr") so the low bit of each register field toggles.
|
|
local zr = (g[2] == "x") and "xzr" or "wzr"
|
|
local function mk(vr, gs) return string.format("dup v%d.%s, %s", vr, ARR[g[1]].asm, gs) end
|
|
local b0, bV, bR = word(mk(0, g[2].."0")), word(mk(31, g[2].."0")), word(mk(0, zr))
|
|
if b0 and bV and bR then dup[#dup+1] = mkrow("DUP_V",
|
|
string.format("{.%s, .%s, .NONE, .NONE}", ARR[g[1]].vt, g[3]),
|
|
"{.VD, .RN, .NONE, .NONE}", b0, {bV, bR}) end
|
|
end
|
|
emit_rows("DUP_V", dup)
|
|
|
|
-- INS: element form (Vd.Ts[i], Vn.Ts[j]) + general form (Vd.Ts[i], Wn/Xn).
|
|
local ins = {}
|
|
for _, ek in ipairs({"B","H","S","D"}) do
|
|
local e = ELEM[ek]
|
|
local function mk(r,i,j) return string.format("ins v%d.%s[%d], v%d.%s[%d]", r, e.tok, i, r, e.tok, j) end
|
|
local b0, br, bi, bj = word(mk(0,0,0)), word(mk(31,0,0)), word(mk(0,e.idxmax,0)), word(mk(0,0,e.idxmax))
|
|
if b0 and br and bi and bj then ins[#ins+1] = mkrow("INS",
|
|
string.format("{.%s, .VEC_INDEX, .%s, .VEC_INDEX}", e.vt, e.vt),
|
|
"{.VD, .NEON_IDX5, .VN, .NEON_IDX4}", b0, {br, bi, bj}) end
|
|
end
|
|
for _, ek in ipairs({"B","H","S","D"}) do
|
|
local e = ELEM[ek]
|
|
local gpr = (ek == "D") and "x" or "w"
|
|
local gvt = (ek == "D") and "X_REG" or "W_REG"
|
|
local zr = (ek == "D") and "xzr" or "wzr"
|
|
local function mk(vr, gs, i) return string.format("ins v%d.%s[%d], %s", vr, e.tok, i, gs) end
|
|
local b0 = word(mk(0, gpr.."0", 0))
|
|
local bV = word(mk(31, gpr.."0", 0)) -- Vd field
|
|
local bR = word(mk(0, zr, 0)) -- Rn field (zero register)
|
|
local bi = word(mk(0, gpr.."0", e.idxmax))
|
|
if b0 and bV and bR and bi then ins[#ins+1] = mkrow("INS",
|
|
string.format("{.%s, .VEC_INDEX, .%s, .NONE}", e.vt, gvt),
|
|
"{.VD, .NEON_IDX5, .RN, .NONE}", b0, {bV, bR, bi}) end
|
|
end
|
|
emit_rows("INS", ins)
|
|
|
|
-- EXT Vd.T, Vn.T, Vm.T, #index (imm4 byte index; .8b idx 0..7, .16b 0..15).
|
|
local ext = {}
|
|
for _, ai in ipairs({{"8B",7},{"16B",15}}) do
|
|
local a, imax = ai[1], ai[2]
|
|
local function mk(r,i) return string.format("ext v%d.%s, v%d.%s, v%d.%s, #%d", r, ARR[a].asm, r, ARR[a].asm, r, ARR[a].asm, i) end
|
|
local b0, br, bi = word(mk(0,0)), word(mk(31,0)), word(mk(0,imax))
|
|
if b0 and br and bi then ext[#ext+1] = mkrow("EXT_V",
|
|
string.format("{.%s, .%s, .%s, .VEC_INDEX}", ARR[a].vt, ARR[a].vt, ARR[a].vt),
|
|
"{.VD, .VN, .VM, .NEON_EXT_IDX}", b0, {br, bi}) end
|
|
end
|
|
emit_rows("EXT_V", ext)
|
|
|
|
sections[#sections+1] = "\t// Advanced SIMD copy / permute (MOV/MVN/DUP/INS/EXT).\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- Scalar FP round/reciprocal + FP-to-GPR convert ------------------------
|
|
-- All use Rd at 0:4, Rn at 5:9; element type (H/S/D) selected by static ftype
|
|
-- bits, so each is a register-only form. H variants require FP16.
|
|
do
|
|
local blk = {}
|
|
local function block(mnem, rows)
|
|
if #rows > 0 then blk[#blk+1] = string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n")); n_mnem = n_mnem + 1 end
|
|
end
|
|
-- scalar FP two-register (FRINTx, FRECPX): <op> Sd, Sn.
|
|
for _, it in ipairs({
|
|
{"FRINTN","frintn"},{"FRINTP","frintp"},{"FRINTM","frintm"},{"FRINTZ","frintz"},
|
|
{"FRINTA","frinta"},{"FRINTX","frintx"},{"FRINTI","frinti"},{"FRECPX","frecpx"},
|
|
}) do
|
|
local rows = {}
|
|
for _, sc in ipairs({{"s","S_REG","FP"},{"d","D_REG","FP"},{"h","H_REG","FP16"}}) do
|
|
local function mk(r) return string.format("%s %s%d, %s%d", it[2], sc[1], r, sc[1], r) end
|
|
local b0, b31 = word(mk(0)), word(mk(31))
|
|
if b0 and b31 then
|
|
rows[#rows+1] = string.format("\t\t{.%s, {.%s, .%s, .NONE, .NONE}, {.RD, .RN, .NONE, .NONE}, 0x%s, 0x%s, .%s, {}},",
|
|
it[1], sc[2], sc[2], bit.tohex(b0):upper(), bit.tohex(mask_of(b0,{b31})):upper(), sc[3])
|
|
n_forms = n_forms + 1
|
|
end
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- FP -> GPR convert (FCVTxS/xU): <op> Wd/Xd, Hn/Sn/Dn.
|
|
for _, it in ipairs({
|
|
{"FCVTAS","fcvtas"},{"FCVTAU","fcvtau"},{"FCVTMS","fcvtms"},{"FCVTMU","fcvtmu"},
|
|
{"FCVTNS","fcvtns"},{"FCVTNU","fcvtnu"},{"FCVTPS","fcvtps"},{"FCVTPU","fcvtpu"},
|
|
}) do
|
|
local rows = {}
|
|
for _, gp in ipairs({{"w","W_REG","wzr"},{"x","X_REG","xzr"}}) do
|
|
for _, fp in ipairs({{"s","S_REG","FP"},{"d","D_REG","FP"},{"h","H_REG","FP16"}}) do
|
|
local function mk(g, f) return string.format("%s %s, %s", it[2], g, f) end
|
|
local b0 = word(mk(gp[1].."0", fp[1].."0"))
|
|
local bD = word(mk(gp[3], fp[1].."0")) -- Rd field (zero reg)
|
|
local bN = word(mk(gp[1].."0", fp[1].."31")) -- Rn field
|
|
if b0 and bD and bN then
|
|
rows[#rows+1] = string.format("\t\t{.%s, {.%s, .%s, .NONE, .NONE}, {.RD, .RN, .NONE, .NONE}, 0x%s, 0x%s, .%s, {}},",
|
|
it[1], gp[2], fp[2], bit.tohex(b0):upper(), bit.tohex(mask_of(b0,{bD,bN})):upper(), fp[3])
|
|
n_forms = n_forms + 1
|
|
end
|
|
end
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
sections[#sections+1] = "\t// Scalar FP round/reciprocal + FP-to-GPR convert.\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- NEON modified immediate (MOVI / MVNI) --------------------------------
|
|
-- imm8 split abc(18:16):defgh(9:5); cmode/op/Q are static per arrangement.
|
|
-- (LSL/MSL-shifted variants share the operand signature and are omitted here.)
|
|
do
|
|
local blk = {}
|
|
local function imm8_block(mnem, llvm, arrs)
|
|
local rows = {}
|
|
for _, a in ipairs(arrs) do
|
|
-- For .2D the asm immediate is the full 64-bit value (each imm8 bit
|
|
-- replicates to a byte), so probe with all-ones to toggle every imm8
|
|
-- bit; other arrangements take the imm8 directly. The builder/encoder
|
|
-- treat the operand as the raw 8-bit pattern regardless.
|
|
local vmax = (a == "2D") and "0xffffffffffffffff" or "255"
|
|
local function mk(r,v) return string.format("%s v%d.%s, #%s", llvm, r, ARR[a].asm, tostring(v)) end
|
|
local b0, bR, bI = word(mk(0,"0")), word(mk(31,"0")), word(mk(0,vmax))
|
|
if b0 and bR and bI then
|
|
rows[#rows+1] = string.format("\t\t{.%s, {.%s, .IMM_8, .NONE, .NONE}, {.VD, .NEON_IMM8_FMOV, .NONE, .NONE}, 0x%s, 0x%s, .NEON, {}},",
|
|
mnem, ARR[a].vt, bit.tohex(b0):upper(), bit.tohex(mask_of(b0,{bR,bI})):upper())
|
|
n_forms = n_forms + 1
|
|
else skips[#skips+1] = mnem.." "..a end
|
|
end
|
|
if #rows > 0 then blk[#blk+1] = string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(rows, "\n")); n_mnem = n_mnem + 1 end
|
|
end
|
|
imm8_block("MOVI", "movi", {"8B","16B","4H","8H","2S","4S","2D"})
|
|
imm8_block("MVNI", "mvni", {"4H","8H","2S","4S"})
|
|
sections[#sections+1] = "\t// NEON modified immediate (MOVI/MVNI).\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- SVE (predicated FP/round/compare, predicate logical/break, SVE2) -------
|
|
-- These iterate element sizes; the predicate-destination forms select size via
|
|
-- the Z-register operands. `bits`/`mask` come from llvm-mc: assemble all-zero,
|
|
-- then one variant per field set to its max (Z reg 31, 3-bit Pg 7, 4-bit Pd/
|
|
-- Pg/Pn/Pm 15, GPR 31 = wzr/xzr) and mask = ~union of the deltas.
|
|
SVE_LLVM = "llvm-mc --assemble --arch=aarch64 --mattr=+sve,+sve2,+sve2-aes,+fullfp16 --show-encoding"
|
|
local function sve_word(line)
|
|
local p = io.popen(string.format("printf '%%s\\n' '%s' | %s 2>/dev/null", line, SVE_LLVM))
|
|
local out = p:read("*a"); p:close()
|
|
local b1,b2,b3,b4 = out:match("0x(%x%x),0x(%x%x),0x(%x%x),0x(%x%x)")
|
|
if not b1 then return nil end
|
|
return tonumber(b4..b3..b2..b1, 16)
|
|
end
|
|
do
|
|
local blk = {}
|
|
local E_HSD = {{"h","Z_REG_H",""},{"s","Z_REG_S",""},{"d","Z_REG_D","is_64=true"}}
|
|
local E_BHSD = {{"b","Z_REG_B",""},{"h","Z_REG_H",""},{"s","Z_REG_S",""},{"d","Z_REG_D","is_64=true"}}
|
|
local E_SD = {{"s","Z_REG_S",""},{"d","Z_REG_D","is_64=true"}}
|
|
local function emit1(mnem, ops, enc, flags, fields, build)
|
|
local zero = {}; for i=1,#fields do zero[i]=0 end
|
|
local b0 = sve_word(build(zero)); if not b0 then skips[#skips+1]=mnem..":b0"; return nil end
|
|
local vs = {}
|
|
for i=1,#fields do
|
|
local v={}; for j=1,#fields do v[j]=0 end; v[i]=fields[i]
|
|
local w=sve_word(build(v)); if not w then skips[#skips+1]=mnem..":v"..i; return nil end; vs[#vs+1]=w
|
|
end
|
|
n_forms=n_forms+1
|
|
return string.format("\t\t{.%s, %s, %s, 0x%s, 0x%s, .SVE, {%s}},",
|
|
mnem, ops, enc, bit.tohex(b0):upper(), bit.tohex(mask_of(b0,vs)):upper(), flags)
|
|
end
|
|
local function block(mnem, rows)
|
|
local r={}; for _,x in ipairs(rows) do if x then r[#r+1]=x end end
|
|
if #r>0 then blk[#blk+1]=string.format("\t.%s = {\n%s\n\t},", mnem, table.concat(r,"\n")); n_mnem=n_mnem+1 end
|
|
end
|
|
local function g(v, x) if v==0 then return x and "x0" or "w0" else return x and "xzr" or "wzr" end end
|
|
|
|
-- predicated FP round: Zd.T, Pg/m, Zn.T (H/S/D)
|
|
for _, it in ipairs({{"SVE_FRINTN","frintn"},{"SVE_FRINTP","frintp"},{"SVE_FRINTM","frintm"},
|
|
{"SVE_FRINTZ","frintz"},{"SVE_FRINTA","frinta"},{"SVE_FRINTX","frintx"},{"SVE_FRINTI","frinti"},{"SVE_FRECPX_Z","frecpx"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_HSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.%s, .P_REG_MERGE, .%s, .NONE}",e[2],e[2]),
|
|
"{.VD, .PG, .VN, .NONE}", e[3], {31,7,31},
|
|
function(v) return string.format("%s z%d.%s, p%d/m, z%d.%s", it[2], v[1],t, v[2], v[3],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- reversed predicated shifts: Zd.T, Pg/m, Zd.T, Zm.T (B/H/S/D)
|
|
for _, it in ipairs({{"SVE_ASRR_PRED","asrr"},{"SVE_LSLR_PRED","lslr"},{"SVE_LSRR_PRED","lsrr"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_BHSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.%s, .P_REG_MERGE, .%s, .%s}",e[2],e[2],e[2]),
|
|
"{.VD, .PG, .VD, .VN}", e[3], {31,7,31},
|
|
function(v) return string.format("%s z%d.%s, p%d/m, z%d.%s, z%d.%s", it[2], v[1],t, v[2], v[1],t, v[3],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- reversed predicated FP: Zd.T, Pg/m, Zd.T, Zm.T (H/S/D)
|
|
for _, it in ipairs({{"SVE_FSUBR_PRED","fsubr"},{"SVE_FDIVR_PRED","fdivr"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_HSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.%s, .P_REG_MERGE, .%s, .%s}",e[2],e[2],e[2]),
|
|
"{.VD, .PG, .VD, .VN}", e[3], {31,7,31},
|
|
function(v) return string.format("%s z%d.%s, p%d/m, z%d.%s, z%d.%s", it[2], v[1],t, v[2], v[1],t, v[3],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- FP compare (reg) -> Pd.T: Pd.T, Pg/z, Zn.T, Zm.T (H/S/D)
|
|
for _, it in ipairs({{"SVE_FCMEQ","fcmeq"},{"SVE_FCMGE","fcmge"},{"SVE_FCMGT","fcmgt"},{"SVE_FCMNE","fcmne"},{"SVE_FCMUO","fcmuo"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_HSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.P_REG, .P_REG_ZERO, .%s, .%s}",e[2],e[2]),
|
|
"{.PD, .PG, .VN, .VM}", e[3], {15,7,31,31},
|
|
function(v) return string.format("%s p%d.%s, p%d/z, z%d.%s, z%d.%s", it[2], v[1],t, v[2], v[3],t, v[4],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- FP compare vs zero -> Pd.T: Pd.T, Pg/z, Zn.T, #0.0 (H/S/D)
|
|
for _, it in ipairs({{"SVE_FCMLE","fcmle"},{"SVE_FCMLT","fcmlt"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_HSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.P_REG, .P_REG_ZERO, .%s, .NONE}",e[2]),
|
|
"{.PD, .PG, .VN, .NONE}", e[3], {15,7,31},
|
|
function(v) return string.format("%s p%d.%s, p%d/z, z%d.%s, #0.0", it[2], v[1],t, v[2], v[3],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- integer compare (reg) -> Pd.T: Pd.T, Pg/z, Zn.T, Zm.T (B/H/S/D)
|
|
for _, it in ipairs({{"SVE_CMPLE","cmple"},{"SVE_CMPLO","cmplo"},{"SVE_CMPLS","cmpls"},{"SVE_CMPLT","cmplt"}}) do
|
|
local rows={}
|
|
for _,e in ipairs(E_BHSD) do local t=e[1]
|
|
rows[#rows+1]=emit1(it[1], string.format("{.P_REG, .P_REG_ZERO, .%s, .%s}",e[2],e[2]),
|
|
"{.PD, .PG, .VM, .VN}", "sets_flags=true"..(e[3]~="" and ", "..e[3] or ""), {15,7,31,31},
|
|
function(v) return string.format("%s p%d.%s, p%d/z, z%d.%s, z%d.%s", it[2], v[1],t, v[2], v[3],t, v[4],t) end)
|
|
end
|
|
block(it[1], rows)
|
|
end
|
|
-- predicate logical (3-source) -> Pd.B: Pd.B, Pg/z, Pn.B, Pm.B
|
|
for _, it in ipairs({{"SVE_NANDS_P","nands"},{"SVE_NORS_P","nors"},{"SVE_ORNS_P","orns"}}) do
|
|
block(it[1], { emit1(it[1], "{.P_REG, .P_REG_ZERO, .P_REG, .P_REG}", "{.PD, .PG4, .PN, .PM}", "sets_flags=true", {15,15,15,15},
|
|
function(v) return string.format("%s p%d.b, p%d/z, p%d.b, p%d.b", it[2], v[1], v[2], v[3], v[4]) end) })
|
|
end
|
|
-- predicate break (pair) -> Pd.B: Pd.B, Pg/z, Pn.B, Pm.B
|
|
for _, it in ipairs({{"SVE_BRKPA","brkpa"},{"SVE_BRKPB","brkpb"},{"SVE_BRKN","brkn"}}) do
|
|
block(it[1], { emit1(it[1], "{.P_REG, .P_REG_ZERO, .P_REG, .P_REG}", "{.PD, .PG4, .PN, .PM}", "", {15,15,15,15},
|
|
function(v) return string.format("%s p%d.b, p%d/z, p%d.b, p%d.b", it[2], v[1], v[2], v[3], v[4]) end) })
|
|
end
|
|
-- predicate break (single) -> Pd.B: Pd.B, Pg/m, Pn.B (BRKA/BRKB merge; BRKAS/BRKBS /z+flags)
|
|
for _, it in ipairs({{"SVE_BRKA","brka","/m",""},{"SVE_BRKB","brkb","/m",""},
|
|
{"SVE_BRKAS","brkas","/z","sets_flags=true"},{"SVE_BRKBS","brkbs","/z","sets_flags=true"}}) do
|
|
local pred = it[3]=="/m" and ".P_REG_MERGE" or ".P_REG_ZERO"
|
|
block(it[1], { emit1(it[1], "{.P_REG, "..pred..", .P_REG, .NONE}", "{.PD, .PG4, .PN, .NONE}", it[4], {15,15,15},
|
|
function(v) return string.format("%s p%d.b, p%d%s, p%d.b", it[2], v[1], v[2], it[3], v[3]) end) })
|
|
end
|
|
-- SVE2 crypto three-source (.D): Zd.D, Zd.D, Zn.D, Zm.D
|
|
for _, it in ipairs({{"SVE_EOR3_Z","eor3"},{"SVE_BCAX_Z","bcax"}}) do
|
|
block(it[1], { emit1(it[1], "{.Z_REG_D, .Z_REG_D, .Z_REG_D, .Z_REG_D}", "{.VD, .VD, .VM, .VN}", "is_64=true", {31,31,31},
|
|
function(v) return string.format("%s z%d.d, z%d.d, z%d.d, z%d.d", it[2], v[1], v[1], v[2], v[3]) end) })
|
|
end
|
|
-- INSR: Zd.T, Wn/Xn (X for .D)
|
|
do
|
|
local rows={}
|
|
for _,e in ipairs(E_BHSD) do local t=e[1]; local x=(t=="d")
|
|
rows[#rows+1]=emit1("SVE_INSR", string.format("{.%s, .%s, .NONE, .NONE}",e[2], x and "X_REG" or "W_REG"),
|
|
"{.VD, .VN, .NONE, .NONE}", e[3], {31,31},
|
|
function(v) return string.format("insr z%d.%s, %s", v[1],t, g(v[2],x)) end)
|
|
end
|
|
block("SVE_INSR", rows)
|
|
end
|
|
-- COMPACT: Zd.T, Pg, Zn.T (S/D)
|
|
do
|
|
local rows={}
|
|
for _,e in ipairs(E_SD) do local t=e[1]
|
|
rows[#rows+1]=emit1("SVE_COMPACT", string.format("{.%s, .P_REG_GOV, .%s, .NONE}",e[2],e[2]),
|
|
"{.VD, .PG, .VN, .NONE}", e[3], {31,7,31},
|
|
function(v) return string.format("compact z%d.%s, p%d, z%d.%s", v[1],t, v[2], v[3],t) end)
|
|
end
|
|
block("SVE_COMPACT", rows)
|
|
end
|
|
sections[#sections+1] = "\t// SVE predicated / compare / predicate-logical / SVE2.\n" .. table.concat(blk, "\n")
|
|
end
|
|
|
|
-- ---- splice into the SoT ---------------------------------------------------
|
|
local region = "\t// SPECGEN:BEGIN\n" .. table.concat(sections, "\n\n") .. "\n\t// SPECGEN:END"
|
|
local fh = assert(io.open(TABLE, "r")); local src = fh:read("*a"); fh:close()
|
|
local new, n = src:gsub("\t// SPECGEN:BEGIN.-\t// SPECGEN:END", (region:gsub("%%", "%%%%")))
|
|
if n ~= 1 then
|
|
io.stderr:write("FATAL: expected exactly one SPECGEN:BEGIN..END region, found "..n.."\n")
|
|
os.exit(1)
|
|
end
|
|
local wh = assert(io.open(TABLE, "w")); wh:write(new); wh:close()
|
|
|
|
io.write(string.format("specgen: wrote %d mnemonics / %d forms into %s\n", n_mnem, n_forms, TABLE))
|
|
if #skips > 0 then
|
|
io.write(" skipped "..#skips.." invalid arrangement(s)\n")
|
|
end
|