123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836 |
- local type = type
- local sub, byte, format = string.sub, string.byte, string.format
- local match, gmatch, gsub = string.match, string.gmatch, string.gsub
- local lower, rep = string.lower, string.rep
- local map_opc1_32 = {
- [0]="addBmr","addVmr","addBrm","addVrm","addBai","addVai","push es","pop es",
- "orBmr","orVmr","orBrm","orVrm","orBai","orVai","push cs","opc2*",
- "adcBmr","adcVmr","adcBrm","adcVrm","adcBai","adcVai","push ss","pop ss",
- "sbbBmr","sbbVmr","sbbBrm","sbbVrm","sbbBai","sbbVai","push ds","pop ds",
- "andBmr","andVmr","andBrm","andVrm","andBai","andVai","es:seg","daa",
- "subBmr","subVmr","subBrm","subVrm","subBai","subVai","cs:seg","das",
- "xorBmr","xorVmr","xorBrm","xorVrm","xorBai","xorVai","ss:seg","aaa",
- "cmpBmr","cmpVmr","cmpBrm","cmpVrm","cmpBai","cmpVai","ds:seg","aas",
- "incVR","incVR","incVR","incVR","incVR","incVR","incVR","incVR",
- "decVR","decVR","decVR","decVR","decVR","decVR","decVR","decVR",
- "pushUR","pushUR","pushUR","pushUR","pushUR","pushUR","pushUR","pushUR",
- "popUR","popUR","popUR","popUR","popUR","popUR","popUR","popUR",
- "sz*pushaw,pusha","sz*popaw,popa","boundVrm","arplWmr",
- "fs:seg","gs:seg","o16:","a16",
- "pushUi","imulVrmi","pushBs","imulVrms",
- "insb","insVS","outsb","outsVS",
- "joBj","jnoBj","jbBj","jnbBj","jzBj","jnzBj","jbeBj","jaBj",
- "jsBj","jnsBj","jpeBj","jpoBj","jlBj","jgeBj","jleBj","jgBj",
- "arith!Bmi","arith!Vmi","arith!Bmi","arith!Vms",
- "testBmr","testVmr","xchgBrm","xchgVrm",
- "movBmr","movVmr","movBrm","movVrm",
- "movVmg","leaVrm","movWgm","popUm",
- "nop*xchgVaR|pause|xchgWaR|repne nop","xchgVaR","xchgVaR","xchgVaR",
- "xchgVaR","xchgVaR","xchgVaR","xchgVaR",
- "sz*cbw,cwde,cdqe","sz*cwd,cdq,cqo","call farViw","wait",
- "sz*pushfw,pushf","sz*popfw,popf","sahf","lahf",
- "movBao","movVao","movBoa","movVoa",
- "movsb","movsVS","cmpsb","cmpsVS",
- "testBai","testVai","stosb","stosVS",
- "lodsb","lodsVS","scasb","scasVS",
- "movBRi","movBRi","movBRi","movBRi","movBRi","movBRi","movBRi","movBRi",
- "movVRI","movVRI","movVRI","movVRI","movVRI","movVRI","movVRI","movVRI",
- "shift!Bmu","shift!Vmu","retBw","ret","$lesVrm","$ldsVrm","movBmi","movVmi",
- "enterBwu","leave","retfBw","retf","int3","intBu","into","iretVS",
- "shift!Bm1","shift!Vm1","shift!Bmc","shift!Vmc","aamBu","aadBu","salc","xlatb",
- "fp*0","fp*1","fp*2","fp*3","fp*4","fp*5","fp*6","fp*7",
- "loopneBj","loopeBj","loopBj","sz*jcxzBj,jecxzBj,jrcxzBj",
- "inBau","inVau","outBua","outVua",
- "callVj","jmpVj","jmp farViw","jmpBj","inBad","inVad","outBda","outVda",
- "lock:","int1","repne:rep","rep:","hlt","cmc","testb!Bm","testv!Vm",
- "clc","stc","cli","sti","cld","std","incb!Bm","incd!Vm",
- }
- assert(#map_opc1_32 == 255)
- local map_opc1_64 = setmetatable({
- [0x06]=false, [0x07]=false, [0x0e]=false,
- [0x16]=false, [0x17]=false, [0x1e]=false, [0x1f]=false,
- [0x27]=false, [0x2f]=false, [0x37]=false, [0x3f]=false,
- [0x60]=false, [0x61]=false, [0x62]=false, [0x63]="movsxdVrDmt", [0x67]="a32:",
- [0x40]="rex*", [0x41]="rex*b", [0x42]="rex*x", [0x43]="rex*xb",
- [0x44]="rex*r", [0x45]="rex*rb", [0x46]="rex*rx", [0x47]="rex*rxb",
- [0x48]="rex*w", [0x49]="rex*wb", [0x4a]="rex*wx", [0x4b]="rex*wxb",
- [0x4c]="rex*wr", [0x4d]="rex*wrb", [0x4e]="rex*wrx", [0x4f]="rex*wrxb",
- [0x82]=false, [0x9a]=false, [0xc4]=false, [0xc5]=false, [0xce]=false,
- [0xd4]=false, [0xd5]=false, [0xd6]=false, [0xea]=false,
- }, { __index = map_opc1_32 })
- local map_opc2 = {
- [0]="sldt!Dmp","sgdt!Ump","larVrm","lslVrm",nil,"syscall","clts","sysret",
- "invd","wbinvd",nil,"ud1",nil,"$prefetch!Bm","femms","3dnowMrmu",
- "movupsXrm|movssXrm|movupdXrm|movsdXrm",
- "movupsXmr|movssXmr|movupdXmr|movsdXmr",
- "movhlpsXrm$movlpsXrm|movsldupXrm|movlpdXrm|movddupXrm",
- "movlpsXmr||movlpdXmr",
- "unpcklpsXrm||unpcklpdXrm",
- "unpckhpsXrm||unpckhpdXrm",
- "movlhpsXrm$movhpsXrm|movshdupXrm|movhpdXrm",
- "movhpsXmr||movhpdXmr",
- "$prefetcht!Bm","hintnopVm","hintnopVm","hintnopVm",
- "hintnopVm","hintnopVm","hintnopVm","hintnopVm",
- "movUmx$","movUmy$","movUxm$","movUym$","movUmz$",nil,"movUzm$",nil,
- "movapsXrm||movapdXrm",
- "movapsXmr||movapdXmr",
- "cvtpi2psXrMm|cvtsi2ssXrVmt|cvtpi2pdXrMm|cvtsi2sdXrVmt",
- "movntpsXmr|movntssXmr|movntpdXmr|movntsdXmr",
- "cvttps2piMrXm|cvttss2siVrXm|cvttpd2piMrXm|cvttsd2siVrXm",
- "cvtps2piMrXm|cvtss2siVrXm|cvtpd2piMrXm|cvtsd2siVrXm",
- "ucomissXrm||ucomisdXrm",
- "comissXrm||comisdXrm",
- "wrmsr","rdtsc","rdmsr","rdpmc","sysenter","sysexit",nil,"getsec",
- "opc3*38",nil,"opc3*3a",nil,nil,nil,nil,nil,
- "cmovoVrm","cmovnoVrm","cmovbVrm","cmovnbVrm",
- "cmovzVrm","cmovnzVrm","cmovbeVrm","cmovaVrm",
- "cmovsVrm","cmovnsVrm","cmovpeVrm","cmovpoVrm",
- "cmovlVrm","cmovgeVrm","cmovleVrm","cmovgVrm",
- "movmskpsVrXm$||movmskpdVrXm$","sqrtpsXrm|sqrtssXrm|sqrtpdXrm|sqrtsdXrm",
- "rsqrtpsXrm|rsqrtssXrm","rcppsXrm|rcpssXrm",
- "andpsXrm||andpdXrm","andnpsXrm||andnpdXrm",
- "orpsXrm||orpdXrm","xorpsXrm||xorpdXrm",
- "addpsXrm|addssXrm|addpdXrm|addsdXrm","mulpsXrm|mulssXrm|mulpdXrm|mulsdXrm",
- "cvtps2pdXrm|cvtss2sdXrm|cvtpd2psXrm|cvtsd2ssXrm",
- "cvtdq2psXrm|cvttps2dqXrm|cvtps2dqXrm",
- "subpsXrm|subssXrm|subpdXrm|subsdXrm","minpsXrm|minssXrm|minpdXrm|minsdXrm",
- "divpsXrm|divssXrm|divpdXrm|divsdXrm","maxpsXrm|maxssXrm|maxpdXrm|maxsdXrm",
- "punpcklbwPrm","punpcklwdPrm","punpckldqPrm","packsswbPrm",
- "pcmpgtbPrm","pcmpgtwPrm","pcmpgtdPrm","packuswbPrm",
- "punpckhbwPrm","punpckhwdPrm","punpckhdqPrm","packssdwPrm",
- "||punpcklqdqXrm","||punpckhqdqXrm",
- "movPrVSm","movqMrm|movdquXrm|movdqaXrm",
- "pshufwMrmu|pshufhwXrmu|pshufdXrmu|pshuflwXrmu","pshiftw!Pmu",
- "pshiftd!Pmu","pshiftq!Mmu||pshiftdq!Xmu",
- "pcmpeqbPrm","pcmpeqwPrm","pcmpeqdPrm","emms|",
- "vmreadUmr||extrqXmuu$|insertqXrmuu$","vmwriteUrm||extrqXrm$|insertqXrm$",
- nil,nil,
- "||haddpdXrm|haddpsXrm","||hsubpdXrm|hsubpsXrm",
- "movVSmMr|movqXrm|movVSmXr","movqMmr|movdquXmr|movdqaXmr",
- "joVj","jnoVj","jbVj","jnbVj","jzVj","jnzVj","jbeVj","jaVj",
- "jsVj","jnsVj","jpeVj","jpoVj","jlVj","jgeVj","jleVj","jgVj",
- "setoBm","setnoBm","setbBm","setnbBm","setzBm","setnzBm","setbeBm","setaBm",
- "setsBm","setnsBm","setpeBm","setpoBm","setlBm","setgeBm","setleBm","setgBm",
- "push fs","pop fs","cpuid","btVmr","shldVmru","shldVmrc",nil,nil,
- "push gs","pop gs","rsm","btsVmr","shrdVmru","shrdVmrc","fxsave!Dmp","imulVrm",
- "cmpxchgBmr","cmpxchgVmr","$lssVrm","btrVmr",
- "$lfsVrm","$lgsVrm","movzxVrBmt","movzxVrWmt",
- "|popcntVrm","ud2Dp","bt!Vmu","btcVmr",
- "bsfVrm","bsrVrm|lzcntVrm|bsrWrm","movsxVrBmt","movsxVrWmt",
- "xaddBmr","xaddVmr",
- "cmppsXrmu|cmpssXrmu|cmppdXrmu|cmpsdXrmu","$movntiVmr|",
- "pinsrwPrWmu","pextrwDrPmu",
- "shufpsXrmu||shufpdXrmu","$cmpxchg!Qmp",
- "bswapVR","bswapVR","bswapVR","bswapVR","bswapVR","bswapVR","bswapVR","bswapVR",
- "||addsubpdXrm|addsubpsXrm","psrlwPrm","psrldPrm","psrlqPrm",
- "paddqPrm","pmullwPrm",
- "|movq2dqXrMm|movqXmr|movdq2qMrXm$","pmovmskbVrMm||pmovmskbVrXm",
- "psubusbPrm","psubuswPrm","pminubPrm","pandPrm",
- "paddusbPrm","padduswPrm","pmaxubPrm","pandnPrm",
- "pavgbPrm","psrawPrm","psradPrm","pavgwPrm",
- "pmulhuwPrm","pmulhwPrm",
- "|cvtdq2pdXrm|cvttpd2dqXrm|cvtpd2dqXrm","$movntqMmr||$movntdqXmr",
- "psubsbPrm","psubswPrm","pminswPrm","porPrm",
- "paddsbPrm","paddswPrm","pmaxswPrm","pxorPrm",
- "|||lddquXrm","psllwPrm","pslldPrm","psllqPrm",
- "pmuludqPrm","pmaddwdPrm","psadbwPrm","maskmovqMrm||maskmovdquXrm$",
- "psubbPrm","psubwPrm","psubdPrm","psubqPrm",
- "paddbPrm","paddwPrm","padddPrm","ud",
- }
- assert(map_opc2[255] == "ud")
- local map_opc3 = {
- ["38"] = {
- [0]="pshufbPrm","phaddwPrm","phadddPrm","phaddswPrm",
- "pmaddubswPrm","phsubwPrm","phsubdPrm","phsubswPrm",
- "psignbPrm","psignwPrm","psigndPrm","pmulhrswPrm",
- nil,nil,nil,nil,
- "||pblendvbXrma",nil,nil,nil,
- "||blendvpsXrma","||blendvpdXrma",nil,"||ptestXrm",
- nil,nil,nil,nil,
- "pabsbPrm","pabswPrm","pabsdPrm",nil,
- "||pmovsxbwXrm","||pmovsxbdXrm","||pmovsxbqXrm","||pmovsxwdXrm",
- "||pmovsxwqXrm","||pmovsxdqXrm",nil,nil,
- "||pmuldqXrm","||pcmpeqqXrm","||$movntdqaXrm","||packusdwXrm",
- nil,nil,nil,nil,
- "||pmovzxbwXrm","||pmovzxbdXrm","||pmovzxbqXrm","||pmovzxwdXrm",
- "||pmovzxwqXrm","||pmovzxdqXrm",nil,"||pcmpgtqXrm",
- "||pminsbXrm","||pminsdXrm","||pminuwXrm","||pminudXrm",
- "||pmaxsbXrm","||pmaxsdXrm","||pmaxuwXrm","||pmaxudXrm",
- "||pmulddXrm","||phminposuwXrm",
- [0xf0] = "|||crc32TrBmt",[0xf1] = "|||crc32TrVmt",
- },
- ["3a"] = {
- [0x00]=nil,nil,nil,nil,nil,nil,nil,nil,
- "||roundpsXrmu","||roundpdXrmu","||roundssXrmu","||roundsdXrmu",
- "||blendpsXrmu","||blendpdXrmu","||pblendwXrmu","palignrPrmu",
- nil,nil,nil,nil,
- "||pextrbVmXru","||pextrwVmXru","||pextrVmSXru","||extractpsVmXru",
- nil,nil,nil,nil,nil,nil,nil,nil,
- "||pinsrbXrVmu","||insertpsXrmu","||pinsrXrVmuS",nil,
- [0x40] = "||dppsXrmu",
- [0x41] = "||dppdXrmu",
- [0x42] = "||mpsadbwXrmu",
- [0x60] = "||pcmpestrmXrmu",[0x61] = "||pcmpestriXrmu",
- [0x62] = "||pcmpistrmXrmu",[0x63] = "||pcmpistriXrmu",
- },
- }
- local map_opcvm = {
- [0xc1]="vmcall",[0xc2]="vmlaunch",[0xc3]="vmresume",[0xc4]="vmxoff",
- [0xc8]="monitor",[0xc9]="mwait",
- [0xd8]="vmrun",[0xd9]="vmmcall",[0xda]="vmload",[0xdb]="vmsave",
- [0xdc]="stgi",[0xdd]="clgi",[0xde]="skinit",[0xdf]="invlpga",
- [0xf8]="swapgs",[0xf9]="rdtscp",
- }
- local map_opcfp = {
- [0]="faddFm","fmulFm","fcomFm","fcompFm","fsubFm","fsubrFm","fdivFm","fdivrFm",
- "fldFm",nil,"fstFm","fstpFm","fldenvVm","fldcwWm","fnstenvVm","fnstcwWm",
- "fiaddDm","fimulDm","ficomDm","ficompDm",
- "fisubDm","fisubrDm","fidivDm","fidivrDm",
- "fildDm","fisttpDm","fistDm","fistpDm",nil,"fld twordFmp",nil,"fstp twordFmp",
- "faddGm","fmulGm","fcomGm","fcompGm","fsubGm","fsubrGm","fdivGm","fdivrGm",
- "fldGm","fisttpQm","fstGm","fstpGm","frstorDmp",nil,"fnsaveDmp","fnstswWm",
- "fiaddWm","fimulWm","ficomWm","ficompWm",
- "fisubWm","fisubrWm","fidivWm","fidivrWm",
- "fildWm","fisttpWm","fistWm","fistpWm",
- "fbld twordFmp","fildQm","fbstp twordFmp","fistpQm",
- "faddFf","fmulFf","fcomFf","fcompFf","fsubFf","fsubrFf","fdivFf","fdivrFf",
- "fldFf","fxchFf",{"fnop"},nil,
- {"fchs","fabs",nil,nil,"ftst","fxam"},
- {"fld1","fldl2t","fldl2e","fldpi","fldlg2","fldln2","fldz"},
- {"f2xm1","fyl2x","fptan","fpatan","fxtract","fprem1","fdecstp","fincstp"},
- {"fprem","fyl2xp1","fsqrt","fsincos","frndint","fscale","fsin","fcos"},
- "fcmovbFf","fcmoveFf","fcmovbeFf","fcmovuFf",nil,{nil,"fucompp"},nil,nil,
- "fcmovnbFf","fcmovneFf","fcmovnbeFf","fcmovnuFf",
- {nil,nil,"fnclex","fninit"},"fucomiFf","fcomiFf",nil,
- "fadd toFf","fmul toFf",nil,nil,
- "fsub toFf","fsubr toFf","fdivr toFf","fdiv toFf",
- "ffreeFf",nil,"fstFf","fstpFf","fucomFf","fucompFf",nil,nil,
- "faddpFf","fmulpFf",nil,{nil,"fcompp"},
- "fsubrpFf","fsubpFf","fdivrpFf","fdivpFf",
- nil,nil,nil,nil,{"fnstsw ax"},"fucomipFf","fcomipFf",nil,
- }
- assert(map_opcfp[126] == "fcomipFf")
- local map_opcgroup = {
- arith = { "add", "or", "adc", "sbb", "and", "sub", "xor", "cmp" },
- shift = { "rol", "ror", "rcl", "rcr", "shl", "shr", "sal", "sar" },
- testb = { "testBmi", "testBmi", "not", "neg", "mul", "imul", "div", "idiv" },
- testv = { "testVmi", "testVmi", "not", "neg", "mul", "imul", "div", "idiv" },
- incb = { "inc", "dec" },
- incd = { "inc", "dec", "callUmp", "$call farDmp",
- "jmpUmp", "$jmp farDmp", "pushUm" },
- sldt = { "sldt", "str", "lldt", "ltr", "verr", "verw" },
- sgdt = { "vm*$sgdt", "vm*$sidt", "$lgdt", "vm*$lidt",
- "smsw", nil, "lmsw", "vm*$invlpg" },
- bt = { nil, nil, nil, nil, "bt", "bts", "btr", "btc" },
- cmpxchg = { nil, "sz*,cmpxchg8bQmp,cmpxchg16bXmp", nil, nil,
- nil, nil, "vmptrld|vmxon|vmclear", "vmptrst" },
- pshiftw = { nil, nil, "psrlw", nil, "psraw", nil, "psllw" },
- pshiftd = { nil, nil, "psrld", nil, "psrad", nil, "pslld" },
- pshiftq = { nil, nil, "psrlq", nil, nil, nil, "psllq" },
- pshiftdq = { nil, nil, "psrlq", "psrldq", nil, nil, "psllq", "pslldq" },
- fxsave = { "$fxsave", "$fxrstor", "$ldmxcsr", "$stmxcsr",
- nil, "lfenceDp$", "mfenceDp$", "sfenceDp$clflush" },
- prefetch = { "prefetch", "prefetchw" },
- prefetcht = { "prefetchnta", "prefetcht0", "prefetcht1", "prefetcht2" },
- }
- local map_regs = {
- B = { "al", "cl", "dl", "bl", "ah", "ch", "dh", "bh",
- "r8b", "r9b", "r10b", "r11b", "r12b", "r13b", "r14b", "r15b" },
- B64 = { "al", "cl", "dl", "bl", "spl", "bpl", "sil", "dil",
- "r8b", "r9b", "r10b", "r11b", "r12b", "r13b", "r14b", "r15b" },
- W = { "ax", "cx", "dx", "bx", "sp", "bp", "si", "di",
- "r8w", "r9w", "r10w", "r11w", "r12w", "r13w", "r14w", "r15w" },
- D = { "eax", "ecx", "edx", "ebx", "esp", "ebp", "esi", "edi",
- "r8d", "r9d", "r10d", "r11d", "r12d", "r13d", "r14d", "r15d" },
- Q = { "rax", "rcx", "rdx", "rbx", "rsp", "rbp", "rsi", "rdi",
- "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15" },
- M = { "mm0", "mm1", "mm2", "mm3", "mm4", "mm5", "mm6", "mm7",
- "mm0", "mm1", "mm2", "mm3", "mm4", "mm5", "mm6", "mm7" },
- X = { "xmm0", "xmm1", "xmm2", "xmm3", "xmm4", "xmm5", "xmm6", "xmm7",
- "xmm8", "xmm9", "xmm10", "xmm11", "xmm12", "xmm13", "xmm14", "xmm15" },
- }
- local map_segregs = { "es", "cs", "ss", "ds", "fs", "gs", "segr6", "segr7" }
- local map_sz2n = {
- B = 1, W = 2, D = 4, Q = 8, M = 8, X = 16,
- }
- local map_sz2prefix = {
- B = "byte", W = "word", D = "dword",
- Q = "qword",
- M = "qword", X = "xword",
- F = "dword", G = "qword",
- }
- local function putop(ctx, text, operands)
- local code, pos, hex = ctx.code, ctx.pos, ""
- local hmax = ctx.hexdump
- if hmax > 0 then
- for i=ctx.start,pos-1 do
- hex = hex..format("%02X", byte(code, i, i))
- end
- if #hex > hmax then hex = sub(hex, 1, hmax)..". "
- else hex = hex..rep(" ", hmax-#hex+2) end
- end
- if operands then text = text.." "..operands end
- if ctx.o16 then text = "o16 "..text; ctx.o16 = false end
- if ctx.a32 then text = "a32 "..text; ctx.a32 = false end
- if ctx.rep then text = ctx.rep.." "..text; ctx.rep = false end
- if ctx.rex then
- local t = (ctx.rexw and "w" or "")..(ctx.rexr and "r" or "")..
- (ctx.rexx and "x" or "")..(ctx.rexb and "b" or "")
- if t ~= "" then text = "rex."..t.." "..text end
- ctx.rexw = false; ctx.rexr = false; ctx.rexx = false; ctx.rexb = false
- ctx.rex = false
- end
- if ctx.seg then
- local text2, n = gsub(text, "%[", "["..ctx.seg..":")
- if n == 0 then text = ctx.seg.." "..text else text = text2 end
- ctx.seg = false
- end
- if ctx.lock then text = "lock "..text; ctx.lock = false end
- local imm = ctx.imm
- if imm then
- local sym = ctx.symtab[imm]
- if sym then text = text.."\t->"..sym end
- end
- ctx.out(format("%08x %s%s\n", ctx.addr+ctx.start, hex, text))
- ctx.mrm = false
- ctx.start = pos
- ctx.imm = nil
- end
- local function clearprefixes(ctx)
- ctx.o16 = false; ctx.seg = false; ctx.lock = false; ctx.rep = false
- ctx.rexw = false; ctx.rexr = false; ctx.rexx = false; ctx.rexb = false
- ctx.rex = false; ctx.a32 = false
- end
- local function incomplete(ctx)
- ctx.pos = ctx.stop+1
- clearprefixes(ctx)
- return putop(ctx, "(incomplete)")
- end
- local function unknown(ctx)
- clearprefixes(ctx)
- return putop(ctx, "(unknown)")
- end
- local function getimm(ctx, pos, n)
- if pos+n-1 > ctx.stop then return incomplete(ctx) end
- local code = ctx.code
- if n == 1 then
- local b1 = byte(code, pos, pos)
- return b1
- elseif n == 2 then
- local b1, b2 = byte(code, pos, pos+1)
- return b1+b2*256
- else
- local b1, b2, b3, b4 = byte(code, pos, pos+3)
- local imm = b1+b2*256+b3*65536+b4*16777216
- ctx.imm = imm
- return imm
- end
- end
- local function putpat(ctx, name, pat)
- local operands, regs, sz, mode, sp, rm, sc, rx, sdisp
- local code, pos, stop = ctx.code, ctx.pos, ctx.stop
-
- for p in gmatch(pat, ".") do
- local x = nil
- if p == "V" or p == "U" then
- if ctx.rexw then sz = "Q"; ctx.rexw = false
- elseif ctx.o16 then sz = "W"; ctx.o16 = false
- elseif p == "U" and ctx.x64 then sz = "Q"
- else sz = "D" end
- regs = map_regs[sz]
- elseif p == "T" then
- if ctx.rexw then sz = "Q"; ctx.rexw = false else sz = "D" end
- regs = map_regs[sz]
- elseif p == "B" then
- sz = "B"
- regs = ctx.rex and map_regs.B64 or map_regs.B
- elseif match(p, "[WDQMXFG]") then
- sz = p
- regs = map_regs[sz]
- elseif p == "P" then
- sz = ctx.o16 and "X" or "M"; ctx.o16 = false
- regs = map_regs[sz]
- elseif p == "S" then
- name = name..lower(sz)
- elseif p == "s" then
- local imm = getimm(ctx, pos, 1); if not imm then return end
- x = imm <= 127 and format("+0x%02x", imm)
- or format("-0x%02x", 256-imm)
- pos = pos+1
- elseif p == "u" then
- local imm = getimm(ctx, pos, 1); if not imm then return end
- x = format("0x%02x", imm)
- pos = pos+1
- elseif p == "w" then
- local imm = getimm(ctx, pos, 2); if not imm then return end
- x = format("0x%x", imm)
- pos = pos+2
- elseif p == "o" then
- if ctx.x64 then
- local imm1 = getimm(ctx, pos, 4); if not imm1 then return end
- local imm2 = getimm(ctx, pos+4, 4); if not imm2 then return end
- x = format("[0x%08x%08x]", imm2, imm1)
- pos = pos+8
- else
- local imm = getimm(ctx, pos, 4); if not imm then return end
- x = format("[0x%08x]", imm)
- pos = pos+4
- end
- elseif p == "i" or p == "I" then
- local n = map_sz2n[sz]
- if n == 8 and ctx.x64 and p == "I" then
- local imm1 = getimm(ctx, pos, 4); if not imm1 then return end
- local imm2 = getimm(ctx, pos+4, 4); if not imm2 then return end
- x = format("0x%08x%08x", imm2, imm1)
- else
- if n == 8 then n = 4 end
- local imm = getimm(ctx, pos, n); if not imm then return end
- if sz == "Q" and (imm < 0 or imm > 0x7fffffff) then
- imm = (0xffffffff+1)-imm
- x = format(imm > 65535 and "-0x%08x" or "-0x%x", imm)
- else
- x = format(imm > 65535 and "0x%08x" or "0x%x", imm)
- end
- end
- pos = pos+n
- elseif p == "j" then
- local n = map_sz2n[sz]
- if n == 8 then n = 4 end
- local imm = getimm(ctx, pos, n); if not imm then return end
- if sz == "B" and imm > 127 then imm = imm-256
- elseif imm > 2147483647 then imm = imm-4294967296 end
- pos = pos+n
- imm = imm + pos + ctx.addr
- if imm > 4294967295 and not ctx.x64 then imm = imm-4294967296 end
- ctx.imm = imm
- if sz == "W" then
- x = format("word 0x%04x", imm%65536)
- elseif ctx.x64 then
- local lo = imm % 0x1000000
- x = format("0x%02x%06x", (imm-lo) / 0x1000000, lo)
- else
- x = format("0x%08x", imm)
- end
- elseif p == "R" then
- local r = byte(code, pos-1, pos-1)%8
- if ctx.rexb then r = r + 8; ctx.rexb = false end
- x = regs[r+1]
- elseif p == "a" then x = regs[1]
- elseif p == "c" then x = "cl"
- elseif p == "d" then x = "dx"
- elseif p == "1" then x = "1"
- else
- if not mode then
- mode = ctx.mrm
- if not mode then
- if pos > stop then return incomplete(ctx) end
- mode = byte(code, pos, pos)
- pos = pos+1
- end
- rm = mode%8; mode = (mode-rm)/8
- sp = mode%8; mode = (mode-sp)/8
- sdisp = ""
- if mode < 3 then
- if rm == 4 then
- if pos > stop then return incomplete(ctx) end
- sc = byte(code, pos, pos)
- pos = pos+1
- rm = sc%8; sc = (sc-rm)/8
- rx = sc%8; sc = (sc-rx)/8
- if ctx.rexx then rx = rx + 8; ctx.rexx = false end
- if rx == 4 then rx = nil end
- end
- if mode > 0 or rm == 5 then
- local dsz = mode
- if dsz ~= 1 then dsz = 4 end
- local disp = getimm(ctx, pos, dsz); if not disp then return end
- if mode == 0 then rm = nil end
- if rm or rx or (not sc and ctx.x64 and not ctx.a32) then
- if dsz == 1 and disp > 127 then
- sdisp = format("-0x%x", 256-disp)
- elseif disp >= 0 and disp <= 0x7fffffff then
- sdisp = format("+0x%x", disp)
- else
- sdisp = format("-0x%x", (0xffffffff+1)-disp)
- end
- else
- sdisp = format(ctx.x64 and not ctx.a32 and
- not (disp >= 0 and disp <= 0x7fffffff)
- and "0xffffffff%08x" or "0x%08x", disp)
- end
- pos = pos+dsz
- end
- end
- if rm and ctx.rexb then rm = rm + 8; ctx.rexb = false end
- if ctx.rexr then sp = sp + 8; ctx.rexr = false end
- end
- if p == "m" then
- if mode == 3 then x = regs[rm+1]
- else
- local aregs = ctx.a32 and map_regs.D or ctx.aregs
- local srm, srx = "", ""
- if rm then srm = aregs[rm+1]
- elseif not sc and ctx.x64 and not ctx.a32 then srm = "rip" end
- ctx.a32 = false
- if rx then
- if rm then srm = srm.."+" end
- srx = aregs[rx+1]
- if sc > 0 then srx = srx.."*"..(2^sc) end
- end
- x = format("[%s%s%s]", srm, srx, sdisp)
- end
- if mode < 3 and
- (not match(pat, "[aRrgp]") or match(pat, "t")) then
- x = map_sz2prefix[sz].." "..x
- end
- elseif p == "r" then x = regs[sp+1]
- elseif p == "g" then x = map_segregs[sp+1]
- elseif p == "p" then
- elseif p == "f" then x = "st"..rm
- elseif p == "x" then
- if sp == 0 and ctx.lock and not ctx.x64 then
- x = "CR8"; ctx.lock = false
- else
- x = "CR"..sp
- end
- elseif p == "y" then x = "DR"..sp
- elseif p == "z" then x = "TR"..sp
- elseif p == "t" then
- else
- error("bad pattern `"..pat.."'")
- end
- end
- if x then operands = operands and operands..", "..x or x end
- end
- ctx.pos = pos
- return putop(ctx, name, operands)
- end
- local map_act
- local function getmrm(ctx)
- local mrm = ctx.mrm
- if not mrm then
- local pos = ctx.pos
- if pos > ctx.stop then return nil end
- mrm = byte(ctx.code, pos, pos)
- ctx.pos = pos+1
- ctx.mrm = mrm
- end
- return mrm
- end
- local function dispatch(ctx, opat, patgrp)
- if not opat then return unknown(ctx) end
- if match(opat, "%|") then
- local p
- if ctx.rep then
- p = ctx.rep=="rep" and "%|([^%|]*)" or "%|[^%|]*%|[^%|]*%|([^%|]*)"
- ctx.rep = false
- elseif ctx.o16 then p = "%|[^%|]*%|([^%|]*)"; ctx.o16 = false
- else p = "^[^%|]*" end
- opat = match(opat, p)
- if not opat then return unknown(ctx) end
-
-
- end
- if match(opat, "%$") then
- local mrm = getmrm(ctx); if not mrm then return incomplete(ctx) end
- opat = match(opat, mrm >= 192 and "^[^%$]*" or "%$(.*)")
- if opat == "" then return unknown(ctx) end
- end
- if opat == "" then return unknown(ctx) end
- local name, pat = match(opat, "^([a-z0-9 ]*)(.*)")
- if pat == "" and patgrp then pat = patgrp end
- return map_act[sub(pat, 1, 1)](ctx, name, pat)
- end
- local function dispatchmap(ctx, opcmap)
- local pos = ctx.pos
- local opat = opcmap[byte(ctx.code, pos, pos)]
- pos = pos + 1
- ctx.pos = pos
- return dispatch(ctx, opat)
- end
- map_act = {
-
- [""] = function(ctx, name, pat)
- return putop(ctx, name)
- end,
-
- B = putpat, W = putpat, D = putpat, Q = putpat,
- V = putpat, U = putpat, T = putpat,
- M = putpat, X = putpat, P = putpat,
- F = putpat, G = putpat,
-
- [":"] = function(ctx, name, pat)
- ctx[pat == ":" and name or sub(pat, 2)] = name
- if ctx.pos - ctx.start > 5 then return unknown(ctx) end
- end,
-
- ["*"] = function(ctx, name, pat)
- return map_act[name](ctx, name, sub(pat, 2))
- end,
-
- ["!"] = function(ctx, name, pat)
- local mrm = getmrm(ctx); if not mrm then return incomplete(ctx) end
- return dispatch(ctx, map_opcgroup[name][((mrm-(mrm%8))/8)%8+1], sub(pat, 2))
- end,
-
- sz = function(ctx, name, pat)
- if ctx.o16 then ctx.o16 = false
- else
- pat = match(pat, ",(.*)")
- if ctx.rexw then
- local p = match(pat, ",(.*)")
- if p then pat = p; ctx.rexw = false end
- end
- end
- pat = match(pat, "^[^,]*")
- return dispatch(ctx, pat)
- end,
-
- opc2 = function(ctx, name, pat)
- return dispatchmap(ctx, map_opc2)
- end,
-
- opc3 = function(ctx, name, pat)
- return dispatchmap(ctx, map_opc3[pat])
- end,
-
- vm = function(ctx, name, pat)
- return dispatch(ctx, map_opcvm[ctx.mrm])
- end,
-
- fp = function(ctx, name, pat)
- local mrm = getmrm(ctx); if not mrm then return incomplete(ctx) end
- local rm = mrm%8
- local idx = pat*8 + ((mrm-rm)/8)%8
- if mrm >= 192 then idx = idx + 64 end
- local opat = map_opcfp[idx]
- if type(opat) == "table" then opat = opat[rm+1] end
- return dispatch(ctx, opat)
- end,
-
- rex = function(ctx, name, pat)
- if ctx.rex then return unknown(ctx) end
- for p in gmatch(pat, ".") do ctx["rex"..p] = true end
- ctx.rex = true
- end,
-
- nop = function(ctx, name, pat)
- return dispatch(ctx, ctx.rex and pat or "nop")
- end,
- }
- local function disass_block(ctx, ofs, len)
- if not ofs then ofs = 0 end
- local stop = len and ofs+len or #ctx.code
- ofs = ofs + 1
- ctx.start = ofs
- ctx.pos = ofs
- ctx.stop = stop
- ctx.imm = nil
- ctx.mrm = false
- clearprefixes(ctx)
- while ctx.pos <= stop do dispatchmap(ctx, ctx.map1) end
- if ctx.pos ~= ctx.start then incomplete(ctx) end
- end
- local function create_(code, addr, out)
- local ctx = {}
- ctx.code = code
- ctx.addr = (addr or 0) - 1
- ctx.out = out or io.write
- ctx.symtab = {}
- ctx.disass = disass_block
- ctx.hexdump = 16
- ctx.x64 = false
- ctx.map1 = map_opc1_32
- ctx.aregs = map_regs.D
- return ctx
- end
- local function create64_(code, addr, out)
- local ctx = create_(code, addr, out)
- ctx.x64 = true
- ctx.map1 = map_opc1_64
- ctx.aregs = map_regs.Q
- return ctx
- end
- local function disass_(code, addr, out)
- create_(code, addr, out):disass()
- end
- local function disass64_(code, addr, out)
- create64_(code, addr, out):disass()
- end
- local function regname_(r)
- if r < 8 then return map_regs.D[r+1] end
- return map_regs.X[r-7]
- end
- local function regname64_(r)
- if r < 16 then return map_regs.Q[r+1] end
- return map_regs.X[r-15]
- end
- module(...)
- create = create_
- create64 = create64_
- disass = disass_
- disass64 = disass64_
- regname = regname_
- regname64 = regname64_
|