x86_table.js 48 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972
  1. "use strict";
  2. const { hex } = require("./util");
  3. // http://ref.x86asm.net/coder32.html
  4. const zf = 1 << 6;
  5. const of = 1 << 11;
  6. const cf = 1 << 0;
  7. const af = 1 << 4;
  8. const pf = 1 << 2;
  9. const sf = 1 << 7;
  10. // === Types of instructions
  11. //
  12. // create entry | check for compiled code | instruction
  13. // -------------+-------------------------+-----------------------------------------------------------
  14. // 1 | optional | pop ds (may change cpu state)
  15. // | | trigger_ud, div (exception that doesn't generate conditional return from BB)
  16. // | | port io, popf, sti (may call interrupt or continue at next instruction)
  17. // | | hlt
  18. // -------------+-------------------------+-----------------------------------------------------------
  19. // 1 | 1 | call [eax], jmp [eax], int, iret, ret, jmpf, callf, sysenter, sysexit
  20. // | | Special case: normal instruction with fallthough to next page
  21. // | | Special case: after execution of compiled code
  22. // | | -> may create redundant entry points depending on last instruction?
  23. // -------------+-------------------------+-----------------------------------------------------------
  24. // 1 | 0 | rep movs, rep lods, rep stos, rep cmps, rep scas
  25. // | | -> Executed as follows:
  26. // | | - Upto including the first call in compiled mode
  27. // | | - Back to main loop and repeated in interpreted mode (as entry point is after instruction, not on)
  28. // | | - When finished entry pointer *after* instruction is hit and execution continues in compiled mode
  29. // -------------+-------------------------+-----------------------------------------------------------
  30. // 0 | optional | jmp foo, jnz foo
  31. // | | (foo is in the same page as the instruction)
  32. // -------------+-------------------------+-----------------------------------------------------------
  33. // 1 | 1 | call foo
  34. // | | (foo is in the same page as the instruction)
  35. // | | -> The entry point is not created for jumps within
  36. // | | this page, but speculatively for calls from
  37. // | | other pages to the function in this page
  38. // -------------+-------------------------+-----------------------------------------------------------
  39. // 1 | 1 | call foo, jmp foo, jnz foo
  40. // | | (foo is in a different page than the instruction)
  41. // os: the instruction behaves differently depending on the operand size
  42. const encodings = [
  43. { opcode: 0x00, custom: 1, e: 1, },
  44. { opcode: 0x01, custom: 1, os: 1, e: 1, },
  45. { opcode: 0x02, custom: 1, e: 1, },
  46. { opcode: 0x03, custom: 1, os: 1, e: 1, },
  47. { opcode: 0x08, custom: 1, e: 1, },
  48. { opcode: 0x09, custom: 1, os: 1, e: 1, },
  49. { opcode: 0x0A, custom: 1, e: 1, },
  50. { opcode: 0x0B, custom: 1, os: 1, e: 1, },
  51. { opcode: 0x10, custom: 1, e: 1, },
  52. { opcode: 0x11, custom: 1, os: 1, e: 1, },
  53. { opcode: 0x12, custom: 1, e: 1, },
  54. { opcode: 0x13, custom: 1, os: 1, e: 1, },
  55. { opcode: 0x18, custom: 1, e: 1, },
  56. { opcode: 0x19, custom: 1, os: 1, e: 1, },
  57. { opcode: 0x1A, custom: 1, e: 1, },
  58. { opcode: 0x1B, custom: 1, os: 1, e: 1, },
  59. { opcode: 0x20, custom: 1, e: 1, },
  60. { opcode: 0x21, custom: 1, os: 1, e: 1, },
  61. { opcode: 0x22, custom: 1, e: 1, },
  62. { opcode: 0x23, custom: 1, os: 1, e: 1, },
  63. { opcode: 0x28, custom: 1, e: 1, },
  64. { opcode: 0x29, custom: 1, os: 1, e: 1, },
  65. { opcode: 0x2A, custom: 1, e: 1, },
  66. { opcode: 0x2B, custom: 1, os: 1, e: 1, },
  67. { opcode: 0x30, custom: 1, e: 1, },
  68. { opcode: 0x31, custom: 1, os: 1, e: 1, },
  69. { opcode: 0x32, custom: 1, e: 1, },
  70. { opcode: 0x33, custom: 1, os: 1, e: 1, },
  71. { opcode: 0x38, custom: 1, e: 1, },
  72. { opcode: 0x39, custom: 1, os: 1, e: 1, },
  73. { opcode: 0x3A, custom: 1, e: 1, },
  74. { opcode: 0x3B, custom: 1, os: 1, e: 1, },
  75. { opcode: 0x06, os: 1, custom: 1 },
  76. { opcode: 0x07, os: 1, skip: 1, block_boundary: 1, }, // pop es: block_boundary since it uses non-raising cpu exceptions
  77. { opcode: 0x0E, os: 1, custom: 1 },
  78. { opcode: 0x0F, os: 1, prefix: 1, },
  79. { opcode: 0x16, os: 1, custom: 1 },
  80. { opcode: 0x17, block_boundary: 1, os: 1, skip: 1, }, // pop ss
  81. { opcode: 0x1E, os: 1, custom: 1 },
  82. { opcode: 0x1F, block_boundary: 1, os: 1, skip: 1, }, // pop ds
  83. { opcode: 0x26, prefix: 1, },
  84. { opcode: 0x27, mask_flags: of, },
  85. { opcode: 0x2E, prefix: 1, },
  86. { opcode: 0x2F, mask_flags: of, },
  87. { opcode: 0x36, prefix: 1, },
  88. { opcode: 0x37, mask_flags: of | sf | pf | zf, },
  89. { opcode: 0x3E, prefix: 1, },
  90. { opcode: 0x3F, mask_flags: of | sf | pf | zf, },
  91. { opcode: 0x40, os: 1, custom: 1 },
  92. { opcode: 0x41, os: 1, custom: 1 },
  93. { opcode: 0x42, os: 1, custom: 1 },
  94. { opcode: 0x43, os: 1, custom: 1 },
  95. { opcode: 0x44, os: 1, custom: 1 },
  96. { opcode: 0x45, os: 1, custom: 1 },
  97. { opcode: 0x46, os: 1, custom: 1 },
  98. { opcode: 0x47, os: 1, custom: 1 },
  99. { opcode: 0x48, os: 1, custom: 1 },
  100. { opcode: 0x49, os: 1, custom: 1 },
  101. { opcode: 0x4A, os: 1, custom: 1 },
  102. { opcode: 0x4B, os: 1, custom: 1 },
  103. { opcode: 0x4C, os: 1, custom: 1 },
  104. { opcode: 0x4D, os: 1, custom: 1 },
  105. { opcode: 0x4E, os: 1, custom: 1 },
  106. { opcode: 0x4F, os: 1, custom: 1 },
  107. { opcode: 0x50, custom: 1, os: 1 },
  108. { opcode: 0x51, custom: 1, os: 1 },
  109. { opcode: 0x52, custom: 1, os: 1 },
  110. { opcode: 0x53, custom: 1, os: 1 },
  111. { opcode: 0x54, custom: 1, os: 1 },
  112. { opcode: 0x55, custom: 1, os: 1 },
  113. { opcode: 0x56, custom: 1, os: 1 },
  114. { opcode: 0x57, custom: 1, os: 1 },
  115. { opcode: 0x58, custom: 1, os: 1, },
  116. { opcode: 0x59, custom: 1, os: 1, },
  117. { opcode: 0x5A, custom: 1, os: 1, },
  118. { opcode: 0x5B, custom: 1, os: 1, },
  119. { opcode: 0x5C, custom: 1, os: 1, },
  120. { opcode: 0x5D, custom: 1, os: 1, },
  121. { opcode: 0x5E, custom: 1, os: 1, },
  122. { opcode: 0x5F, custom: 1, os: 1, },
  123. { opcode: 0x60, os: 1, },
  124. { opcode: 0x61, os: 1, },
  125. { opcode: 0x62, e: 1, skip: 1, },
  126. { opcode: 0x63, e: 1, block_boundary: 1, }, // arpl
  127. { opcode: 0x64, prefix: 1, },
  128. { opcode: 0x65, prefix: 1, },
  129. { opcode: 0x66, prefix: 1, },
  130. { opcode: 0x67, prefix: 1, },
  131. { opcode: 0x68, custom: 1, os: 1, imm1632: 1 },
  132. { opcode: 0x69, os: 1, e: 1, custom: 1, imm1632: 1, mask_flags: af, }, // zf?
  133. { opcode: 0x6A, custom: 1, os: 1, imm8s: 1 },
  134. { opcode: 0x6B, os: 1, e: 1, custom: 1, imm8s: 1, mask_flags: af, }, // zf?
  135. { opcode: 0x6C, block_boundary: 1, is_string: 1, skip: 1, }, // ins
  136. { opcode: 0xF26C, block_boundary: 1, is_string: 1, skip: 1, },
  137. { opcode: 0xF36C, block_boundary: 1, is_string: 1, skip: 1, },
  138. { opcode: 0x6D, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  139. { opcode: 0xF26D, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  140. { opcode: 0xF36D, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  141. { opcode: 0x6E, block_boundary: 1, is_string: 1, skip: 1, }, // outs
  142. { opcode: 0xF26E, block_boundary: 1, is_string: 1, skip: 1, },
  143. { opcode: 0xF36E, block_boundary: 1, is_string: 1, skip: 1, },
  144. { opcode: 0x6F, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  145. { opcode: 0xF26F, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  146. { opcode: 0xF36F, block_boundary: 1, is_string: 1, os: 1, skip: 1, },
  147. { opcode: 0x84, custom: 1, e: 1, },
  148. { opcode: 0x85, custom: 1, e: 1, os: 1, },
  149. { opcode: 0x86, e: 1, },
  150. { opcode: 0x87, os: 1, e: 1, },
  151. { opcode: 0x88, custom: 1, e: 1, },
  152. { opcode: 0x89, custom: 1, os: 1, e: 1, },
  153. { opcode: 0x8A, custom: 1, e: 1, },
  154. { opcode: 0x8B, custom: 1, os: 1, e: 1, },
  155. { opcode: 0x8C, os: 1, e: 1, custom: 1 }, // mov reg, sreg
  156. { opcode: 0x8D, reg_ud: 1, os: 1, e: 1, custom_modrm_resolve: 1, custom: 1, }, // lea
  157. { opcode: 0x8E, block_boundary: 1, e: 1, skip: 1, }, // mov sreg
  158. { opcode: 0x8F, os: 1, e: 1, fixed_g: 0, custom_modrm_resolve: 1, custom: 1, block_boundary: 1, }, // pop r/m
  159. { opcode: 0x90, custom: 1, },
  160. { opcode: 0x91, custom: 1, os: 1, },
  161. { opcode: 0x92, custom: 1, os: 1, },
  162. { opcode: 0x93, custom: 1, os: 1, },
  163. { opcode: 0x94, custom: 1, os: 1, },
  164. { opcode: 0x95, custom: 1, os: 1, },
  165. { opcode: 0x96, custom: 1, os: 1, },
  166. { opcode: 0x97, custom: 1, os: 1, },
  167. { opcode: 0x98, os: 1, custom: 1 },
  168. { opcode: 0x99, os: 1, custom: 1 },
  169. { opcode: 0x9A, os: 1, imm1632: 1, extra_imm16: 1, skip: 1, block_boundary: 1, }, // callf
  170. { opcode: 0x9B, skip: 1, }, // fwait: block_boundary since it uses non-raising cpu exceptions
  171. { opcode: 0x9C, os: 1, custom: 1 },
  172. // popf: not a jump, but can cause an eip change due to updating the interrupt flag
  173. { opcode: 0x9D, os: 1, block_boundary: 1, skip: 1, },
  174. { opcode: 0x9E, custom: 1 },
  175. { opcode: 0x9F, },
  176. { opcode: 0xA0, custom: 1, immaddr: 1 },
  177. { opcode: 0xA1, custom: 1, os: 1, immaddr: 1 },
  178. { opcode: 0xA2, custom: 1, immaddr: 1 },
  179. { opcode: 0xA3, custom: 1, os: 1, immaddr: 1 },
  180. // string instructions aren't jumps, but they modify eip due to how they're implemented
  181. // TODO: The block_boundary on the non-rep instructions can be removed once they're custom
  182. { opcode: 0xA4, block_boundary: 1, is_string: 1, },
  183. { opcode: 0xF2A4, block_boundary: 1, is_string: 1, },
  184. { opcode: 0xF3A4, block_boundary: 1, is_string: 1, },
  185. { opcode: 0xA5, block_boundary: 1, is_string: 1, os: 1, },
  186. { opcode: 0xF2A5, block_boundary: 1, is_string: 1, os: 1, },
  187. { opcode: 0xF3A5, block_boundary: 1, is_string: 1, os: 1, },
  188. { opcode: 0xA6, block_boundary: 1, is_string: 1, },
  189. { opcode: 0xF2A6, block_boundary: 1, is_string: 1, },
  190. { opcode: 0xF3A6, block_boundary: 1, is_string: 1, },
  191. { opcode: 0xA7, block_boundary: 1, is_string: 1, os: 1, },
  192. { opcode: 0xF2A7, block_boundary: 1, is_string: 1, os: 1, },
  193. { opcode: 0xF3A7, block_boundary: 1, is_string: 1, os: 1, },
  194. { opcode: 0xA8, custom: 1, imm8: 1, },
  195. { opcode: 0xA9, custom: 1, os: 1, imm1632: 1, },
  196. { opcode: 0xAA, block_boundary: 1, is_string: 1, },
  197. { opcode: 0xF2AA, block_boundary: 1, is_string: 1, },
  198. { opcode: 0xF3AA, block_boundary: 1, is_string: 1, },
  199. { opcode: 0xAB, block_boundary: 1, is_string: 1, os: 1, },
  200. { opcode: 0xF2AB, block_boundary: 1, is_string: 1, os: 1, },
  201. { opcode: 0xF3AB, block_boundary: 1, is_string: 1, os: 1, },
  202. { opcode: 0xAC, block_boundary: 1, is_string: 1, },
  203. { opcode: 0xF2AC, block_boundary: 1, is_string: 1, },
  204. { opcode: 0xF3AC, block_boundary: 1, is_string: 1, },
  205. { opcode: 0xAD, block_boundary: 1, is_string: 1, os: 1, },
  206. { opcode: 0xF2AD, block_boundary: 1, is_string: 1, os: 1, },
  207. { opcode: 0xF3AD, block_boundary: 1, is_string: 1, os: 1, },
  208. { opcode: 0xAE, block_boundary: 1, is_string: 1, },
  209. { opcode: 0xF2AE, block_boundary: 1, is_string: 1, },
  210. { opcode: 0xF3AE, block_boundary: 1, is_string: 1, },
  211. { opcode: 0xAF, block_boundary: 1, is_string: 1, os: 1, },
  212. { opcode: 0xF2AF, block_boundary: 1, is_string: 1, os: 1, },
  213. { opcode: 0xF3AF, block_boundary: 1, is_string: 1, os: 1, },
  214. { opcode: 0xC2, custom: 1, block_boundary: 1, no_next_instruction: 1, os: 1, imm16: 1, skip: 1, }, // ret
  215. { opcode: 0xC3, custom: 1, block_boundary: 1, no_next_instruction: 1, os: 1, skip: 1, },
  216. { opcode: 0xC4, block_boundary: 1, os: 1, e: 1, skip: 1, }, // les
  217. { opcode: 0xC5, block_boundary: 1, os: 1, e: 1, skip: 1, }, // lds
  218. { opcode: 0xC6, custom: 1, e: 1, fixed_g: 0, imm8: 1 },
  219. { opcode: 0xC7, custom: 1, os: 1, e: 1, fixed_g: 0, imm1632: 1 },
  220. // XXX: Temporary block boundary
  221. { opcode: 0xC8, os: 1, imm16: 1, extra_imm8: 1, block_boundary: 1, }, // enter
  222. { opcode: 0xC9, custom: 1, os: 1, skip: 1 }, // leave
  223. { opcode: 0xCA, block_boundary: 1, no_next_instruction: 1, os: 1, imm16: 1, skip: 1, }, // retf
  224. { opcode: 0xCB, block_boundary: 1, no_next_instruction: 1, os: 1, skip: 1, },
  225. { opcode: 0xCC, block_boundary: 1, skip: 1, }, // int
  226. { opcode: 0xCD, block_boundary: 1, skip: 1, imm8: 1, },
  227. { opcode: 0xCE, block_boundary: 1, skip: 1, },
  228. { opcode: 0xCF, block_boundary: 1, no_next_instruction: 1, os: 1, skip: 1, }, // iret
  229. { opcode: 0xD4, imm8: 1, block_boundary: 1, }, // aam, may trigger #de
  230. { opcode: 0xD5, imm8: 1, mask_flags: of | cf | af, },
  231. { opcode: 0xD6, },
  232. // XXX: Temporary block boundary
  233. { opcode: 0xD7, skip: 1, block_boundary: 1, },
  234. { opcode: 0xD8, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  235. { opcode: 0xD8, e: 1, fixed_g: 1, custom: 1, is_fpu: 1, task_switch_test: 1, },
  236. { opcode: 0xD8, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1, },
  237. { opcode: 0xD8, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1, },
  238. { opcode: 0xD8, e: 1, fixed_g: 4, custom: 1, is_fpu: 1, task_switch_test: 1, },
  239. { opcode: 0xD8, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  240. { opcode: 0xD8, e: 1, fixed_g: 6, custom: 1, is_fpu: 1, task_switch_test: 1, },
  241. { opcode: 0xD8, e: 1, fixed_g: 7, custom: 1, is_fpu: 1, task_switch_test: 1, },
  242. { opcode: 0xD9, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  243. { opcode: 0xD9, e: 1, fixed_g: 1, custom: 1, is_fpu: 1, task_switch_test: 1, },
  244. { opcode: 0xD9, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1, },
  245. { opcode: 0xD9, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1, },
  246. { opcode: 0xD9, e: 1, fixed_g: 4, custom: 1, is_fpu: 1, task_switch_test: 1, },
  247. { opcode: 0xD9, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  248. { opcode: 0xD9, e: 1, fixed_g: 6, custom: 1, is_fpu: 1, task_switch_test: 1, skip: 1, }, // fstenv (mem), fprem (reg)
  249. { opcode: 0xD9, e: 1, fixed_g: 7, custom: 1, is_fpu: 1, task_switch_test: 1, only_mem: 1, }, // fprem, fyl2xp1
  250. { opcode: 0xDA, e: 1, fixed_g: 0, custom: 0, is_fpu: 1, task_switch_test: 1, },
  251. { opcode: 0xDA, e: 1, fixed_g: 1, custom: 0, is_fpu: 1, task_switch_test: 1, },
  252. { opcode: 0xDA, e: 1, fixed_g: 2, custom: 0, is_fpu: 1, task_switch_test: 1, },
  253. { opcode: 0xDA, e: 1, fixed_g: 3, custom: 0, is_fpu: 1, task_switch_test: 1, },
  254. { opcode: 0xDA, e: 1, fixed_g: 4, custom: 0, is_fpu: 1, task_switch_test: 1, },
  255. { opcode: 0xDA, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  256. { opcode: 0xDA, e: 1, fixed_g: 6, custom: 0, is_fpu: 1, task_switch_test: 1, },
  257. { opcode: 0xDA, e: 1, fixed_g: 7, custom: 0, is_fpu: 1, task_switch_test: 1, },
  258. { opcode: 0xDB, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  259. { opcode: 0xDB, e: 1, fixed_g: 1, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1, }, // unimplemented: fisttp (sse3)
  260. { opcode: 0xDB, e: 1, fixed_g: 2, custom: 2, is_fpu: 1, task_switch_test: 1, },
  261. { opcode: 0xDB, e: 1, fixed_g: 3, custom: 2, is_fpu: 1, task_switch_test: 1, },
  262. { opcode: 0xDB, e: 1, fixed_g: 4, custom: 0, is_fpu: 1, task_switch_test: 1, },
  263. { opcode: 0xDB, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  264. { opcode: 0xDB, e: 1, fixed_g: 6, custom: 0, is_fpu: 1, task_switch_test: 1, },
  265. { opcode: 0xDB, e: 1, fixed_g: 7, custom: 0, is_fpu: 1, task_switch_test: 1, },
  266. { opcode: 0xDC, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  267. { opcode: 0xDC, e: 1, fixed_g: 1, custom: 1, is_fpu: 1, task_switch_test: 1, },
  268. { opcode: 0xDC, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1, },
  269. { opcode: 0xDC, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1, },
  270. { opcode: 0xDC, e: 1, fixed_g: 4, custom: 1, is_fpu: 1, task_switch_test: 1, },
  271. { opcode: 0xDC, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  272. { opcode: 0xDC, e: 1, fixed_g: 6, custom: 1, is_fpu: 1, task_switch_test: 1, },
  273. { opcode: 0xDC, e: 1, fixed_g: 7, custom: 1, is_fpu: 1, task_switch_test: 1, },
  274. { opcode: 0xDD, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  275. { opcode: 0xDD, e: 1, fixed_g: 1, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1, }, // unimplemented: fisttp (sse3)
  276. { opcode: 0xDD, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1, },
  277. { opcode: 0xDD, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1, },
  278. { opcode: 0xDD, e: 1, fixed_g: 4, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1, }, // frstor
  279. { opcode: 0xDD, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  280. { opcode: 0xDD, e: 1, fixed_g: 6, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1, }, // fsave
  281. { opcode: 0xDD, e: 1, fixed_g: 7, custom: 0, is_fpu: 1, task_switch_test: 1, },
  282. { opcode: 0xDE, e: 1, fixed_g: 0, custom: 1, is_fpu: 1, task_switch_test: 1, },
  283. { opcode: 0xDE, e: 1, fixed_g: 1, custom: 1, is_fpu: 1, task_switch_test: 1, },
  284. { opcode: 0xDE, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1, },
  285. { opcode: 0xDE, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1, },
  286. { opcode: 0xDE, e: 1, fixed_g: 4, custom: 1, is_fpu: 1, task_switch_test: 1, },
  287. { opcode: 0xDE, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  288. { opcode: 0xDE, e: 1, fixed_g: 6, custom: 1, is_fpu: 1, task_switch_test: 1, },
  289. { opcode: 0xDE, e: 1, fixed_g: 7, custom: 1, is_fpu: 1, task_switch_test: 1, },
  290. { opcode: 0xDF, e: 1, fixed_g: 0, custom: 0, is_fpu: 1, task_switch_test: 1 },
  291. { opcode: 0xDF, e: 1, fixed_g: 1, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1 }, // unimplemented: fisttp (sse3)
  292. { opcode: 0xDF, e: 1, fixed_g: 2, custom: 1, is_fpu: 1, task_switch_test: 1 },
  293. { opcode: 0xDF, e: 1, fixed_g: 3, custom: 1, is_fpu: 1, task_switch_test: 1 },
  294. { opcode: 0xDF, e: 1, fixed_g: 4, custom: 1, is_fpu: 1, task_switch_test: 1, only_reg: 1 }, // unimplemented: Binary Coded Decimals
  295. { opcode: 0xDF, e: 1, fixed_g: 5, custom: 1, is_fpu: 1, task_switch_test: 1, },
  296. { opcode: 0xDF, e: 1, fixed_g: 6, custom: 0, is_fpu: 1, task_switch_test: 1, only_reg: 1 }, // unimplemented: Binary Coded Decimals
  297. { opcode: 0xDF, e: 1, fixed_g: 7, custom: 1, is_fpu: 1, task_switch_test: 1, },
  298. // loop, jcxz, etc.
  299. // Conditional jumps, but condition code not supported by code generator
  300. // (these are never generated by modern compilers)
  301. { opcode: 0xE0, os: 1, imm8s: 1, skip: 1, block_boundary: 1, /* jump_offset_imm: 1, conditional_jump: 1, */ },
  302. { opcode: 0xE1, os: 1, imm8s: 1, skip: 1, block_boundary: 1, /* jump_offset_imm: 1, conditional_jump: 1, */ },
  303. { opcode: 0xE2, os: 1, imm8s: 1, skip: 1, block_boundary: 1, /* jump_offset_imm: 1, conditional_jump: 1, */ },
  304. { opcode: 0xE3, os: 1, imm8s: 1, skip: 1, block_boundary: 1, /* jump_offset_imm: 1, conditional_jump: 1, */ },
  305. // port functions aren't jumps, but they may modify eip due to how they are implemented
  306. { opcode: 0xE4, block_boundary: 1, imm8: 1, skip: 1, }, // in
  307. { opcode: 0xE5, block_boundary: 1, os: 1, imm8: 1, skip: 1, },
  308. { opcode: 0xE6, block_boundary: 1, imm8: 1, skip: 1, }, // out
  309. { opcode: 0xE7, block_boundary: 1, os: 1, imm8: 1, skip: 1, },
  310. { opcode: 0xE8, block_boundary: 1, jump_offset_imm: 1, os: 1, imm1632: 1, custom: 1, skip: 1, }, // call
  311. { opcode: 0xE9, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, no_next_instruction: 1, os: 1, imm1632: 1, custom: 1, skip: 1, },
  312. { opcode: 0xEA, block_boundary: 1, no_next_instruction: 1, os: 1, imm1632: 1, extra_imm16: 1, skip: 1, }, // jmpf
  313. { opcode: 0xEB, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, no_next_instruction: 1, os: 1, imm8s: 1, custom: 1, skip: 1, },
  314. { opcode: 0xEC, block_boundary: 1, skip: 1, }, // in
  315. { opcode: 0xED, block_boundary: 1, os: 1, skip: 1, },
  316. { opcode: 0xEE, block_boundary: 1, skip: 1, }, // out
  317. { opcode: 0xEF, block_boundary: 1, os: 1, skip: 1, },
  318. { opcode: 0xF0, prefix: 1, },
  319. { opcode: 0xF1, skip: 1, },
  320. { opcode: 0xF2, prefix: 1, },
  321. { opcode: 0xF3, prefix: 1, },
  322. { opcode: 0xF4, block_boundary: 1, no_next_instruction: 1, skip: 1, }, // hlt
  323. { opcode: 0xF5, },
  324. { opcode: 0xF6, e: 1, fixed_g: 0, imm8: 1, custom: 1 },
  325. { opcode: 0xF6, e: 1, fixed_g: 1, imm8: 1, custom: 1 },
  326. { opcode: 0xF6, e: 1, fixed_g: 2, },
  327. { opcode: 0xF6, e: 1, fixed_g: 3, },
  328. { opcode: 0xF6, e: 1, fixed_g: 4, mask_flags: af | zf, },
  329. { opcode: 0xF6, e: 1, fixed_g: 5, mask_flags: af | zf, },
  330. { opcode: 0xF6, e: 1, fixed_g: 6, block_boundary: 1, }, // div/idiv: Not a block boundary, but doesn't use control flow exceptions
  331. { opcode: 0xF6, e: 1, fixed_g: 7, block_boundary: 1, },
  332. { opcode: 0xF7, os: 1, e: 1, fixed_g: 0, imm1632: 1, custom: 1 },
  333. { opcode: 0xF7, os: 1, e: 1, fixed_g: 1, imm1632: 1, custom: 1 },
  334. { opcode: 0xF7, os: 1, e: 1, fixed_g: 2, custom: 1 },
  335. { opcode: 0xF7, os: 1, e: 1, fixed_g: 3, custom: 1 },
  336. { opcode: 0xF7, os: 1, e: 1, fixed_g: 4, mask_flags: zf | af, custom: 1 },
  337. { opcode: 0xF7, os: 1, e: 1, fixed_g: 5, mask_flags: zf | af, custom: 1 },
  338. { opcode: 0xF7, os: 1, e: 1, fixed_g: 6, block_boundary: 1, }, // div/idiv: Not a block boundary, but doesn't use control flow exceptions
  339. { opcode: 0xF7, os: 1, e: 1, fixed_g: 7, block_boundary: 1, },
  340. { opcode: 0xF8, },
  341. { opcode: 0xF9, },
  342. { opcode: 0xFA, custom: 1, skip: 1 },
  343. // sti: not a jump, but can cause a change in eip
  344. { opcode: 0xFB, block_boundary: 1, skip: 1, },
  345. { opcode: 0xFC, custom: 1, },
  346. { opcode: 0xFD, custom: 1, },
  347. { opcode: 0xFE, e: 1, fixed_g: 0, },
  348. { opcode: 0xFE, e: 1, fixed_g: 1, },
  349. { opcode: 0xFF, os: 1, e: 1, fixed_g: 0, custom: 1, },
  350. { opcode: 0xFF, os: 1, e: 1, fixed_g: 1, custom: 1, },
  351. { opcode: 0xFF, os: 1, e: 1, fixed_g: 2, custom: 1, block_boundary: 1, skip: 1, },
  352. { opcode: 0xFF, os: 1, e: 1, fixed_g: 3, block_boundary: 1, skip: 1, },
  353. { opcode: 0xFF, os: 1, e: 1, fixed_g: 4, custom: 1, block_boundary: 1, no_next_instruction: 1, skip: 1, },
  354. { opcode: 0xFF, os: 1, e: 1, fixed_g: 5, block_boundary: 1, no_next_instruction: 1, skip: 1, },
  355. { opcode: 0xFF, custom: 1, os: 1, e: 1, fixed_g: 6, },
  356. { opcode: 0x0F00, fixed_g: 0, e: 1, skip: 1, block_boundary: 1, }, // sldt, ...
  357. { opcode: 0x0F00, fixed_g: 1, e: 1, skip: 1, block_boundary: 1, },
  358. { opcode: 0x0F00, fixed_g: 2, e: 1, skip: 1, block_boundary: 1, },
  359. { opcode: 0x0F00, fixed_g: 3, e: 1, skip: 1, block_boundary: 1, },
  360. { opcode: 0x0F00, fixed_g: 4, e: 1, skip: 1, block_boundary: 1, },
  361. { opcode: 0x0F00, fixed_g: 5, e: 1, skip: 1, block_boundary: 1, },
  362. { opcode: 0x0F01, fixed_g: 0, e: 1, skip: 1, block_boundary: 1, }, // sgdt, ...
  363. { opcode: 0x0F01, fixed_g: 1, e: 1, skip: 1, block_boundary: 1, },
  364. { opcode: 0x0F01, fixed_g: 2, e: 1, skip: 1, block_boundary: 1, },
  365. { opcode: 0x0F01, fixed_g: 3, e: 1, skip: 1, block_boundary: 1, },
  366. { opcode: 0x0F01, fixed_g: 4, e: 1, skip: 1, block_boundary: 1, },
  367. { opcode: 0x0F01, fixed_g: 6, e: 1, skip: 1, block_boundary: 1, },
  368. { opcode: 0x0F01, fixed_g: 7, e: 1, skip: 1, block_boundary: 1, },
  369. { opcode: 0x0F02, os: 1, e: 1, skip: 1, block_boundary: 1, }, // lar
  370. { opcode: 0x0F03, os: 1, e: 1, skip: 1, block_boundary: 1, }, // lsl
  371. { opcode: 0x0F04, skip: 1, block_boundary: 1, },
  372. { opcode: 0x0F05, skip: 1, block_boundary: 1, },
  373. { opcode: 0x0F06, skip: 1, block_boundary: 1, }, // clts
  374. { opcode: 0x0F07, skip: 1, block_boundary: 1, },
  375. { opcode: 0x0F08, skip: 1, block_boundary: 1, },
  376. { opcode: 0x0F09, skip: 1, block_boundary: 1, }, // wbinvd
  377. { opcode: 0x0F0A, skip: 1, block_boundary: 1, },
  378. // ud2
  379. // Technically has a next instruction, but Linux uses this for assertions
  380. // and embeds the assertion message after this instruction, which is likely
  381. // the most common use case of ud2
  382. { opcode: 0x0F0B, skip: 1, block_boundary: 1, no_next_instruction: 1, },
  383. { opcode: 0x0F0C, skip: 1, block_boundary: 1, },
  384. { opcode: 0x0F0D, skip: 1, block_boundary: 1, },
  385. { opcode: 0x0F0E, skip: 1, block_boundary: 1, },
  386. { opcode: 0x0F0F, skip: 1, block_boundary: 1, },
  387. { opcode: 0x0F18, e: 1, custom: 1 },
  388. { opcode: 0x0F19, non_faulting: 1, custom: 1, e: 1, },
  389. { opcode: 0x0F1A, skip: 1, block_boundary: 1, },
  390. { opcode: 0x0F1B, skip: 1, block_boundary: 1, },
  391. { opcode: 0x0F1C, non_faulting: 1, custom: 1, e: 1, },
  392. { opcode: 0x0F1D, non_faulting: 1, custom: 1, e: 1, },
  393. { opcode: 0x0F1E, non_faulting: 1, custom: 1, e: 1, },
  394. { opcode: 0x0F1F, non_faulting: 1, custom: 1, e: 1, },
  395. { opcode: 0x0F20, ignore_mod: 1, e: 1, skip: 1, block_boundary: 1, }, // mov reg, creg
  396. { opcode: 0x0F21, ignore_mod: 1, e: 1, skip: 1, block_boundary: 1, }, // mov reg, dreg
  397. { opcode: 0x0F22, ignore_mod: 1, e: 1, skip: 1, block_boundary: 1, }, // mov creg, reg
  398. { opcode: 0x0F23, ignore_mod: 1, e: 1, skip: 1, block_boundary: 1, }, // mov dreg, reg
  399. { opcode: 0x0F24, skip: 1, block_boundary: 1, },
  400. { opcode: 0x0F25, skip: 1, block_boundary: 1, },
  401. { opcode: 0x0F26, skip: 1, block_boundary: 1, },
  402. { opcode: 0x0F27, skip: 1, block_boundary: 1, },
  403. { opcode: 0x0F30, skip: 1, block_boundary: 1, }, // wrmsr
  404. { opcode: 0x0F31, skip: 1, block_boundary: 1, }, // rdtsc
  405. { opcode: 0x0F32, skip: 1, block_boundary: 1, }, // rdmsr
  406. { opcode: 0x0F33, skip: 1, block_boundary: 1, }, // rdpmc
  407. { opcode: 0x0F34, skip: 1, block_boundary: 1, no_next_instruction: 1, }, // sysenter
  408. { opcode: 0x0F35, skip: 1, block_boundary: 1, no_next_instruction: 1, }, // sysexit
  409. { opcode: 0x0F36, skip: 1, block_boundary: 1, }, // ud
  410. { opcode: 0x0F37, skip: 1, block_boundary: 1, }, // getsec
  411. // sse3+
  412. { opcode: 0x0F38, skip: 1, block_boundary: 1, },
  413. { opcode: 0x0F39, skip: 1, block_boundary: 1, },
  414. { opcode: 0x0F3A, skip: 1, block_boundary: 1, },
  415. { opcode: 0x0F3B, skip: 1, block_boundary: 1, },
  416. { opcode: 0x0F3C, skip: 1, block_boundary: 1, },
  417. { opcode: 0x0F3D, skip: 1, block_boundary: 1, },
  418. { opcode: 0x0F3E, skip: 1, block_boundary: 1, },
  419. { opcode: 0x0F3F, skip: 1, block_boundary: 1, },
  420. { opcode: 0x0F40, e: 1, os: 1, custom: 1, },
  421. { opcode: 0x0F41, e: 1, os: 1, custom: 1, },
  422. { opcode: 0x0F42, e: 1, os: 1, custom: 1, },
  423. { opcode: 0x0F43, e: 1, os: 1, custom: 1, },
  424. { opcode: 0x0F44, e: 1, os: 1, custom: 1, },
  425. { opcode: 0x0F45, e: 1, os: 1, custom: 1, },
  426. { opcode: 0x0F46, e: 1, os: 1, custom: 1, },
  427. { opcode: 0x0F47, e: 1, os: 1, custom: 1, },
  428. { opcode: 0x0F48, e: 1, os: 1, custom: 1, },
  429. { opcode: 0x0F49, e: 1, os: 1, custom: 1, },
  430. { opcode: 0x0F4A, e: 1, os: 1, custom: 1, },
  431. { opcode: 0x0F4B, e: 1, os: 1, custom: 1, },
  432. { opcode: 0x0F4C, e: 1, os: 1, custom: 1, },
  433. { opcode: 0x0F4D, e: 1, os: 1, custom: 1, },
  434. { opcode: 0x0F4E, e: 1, os: 1, custom: 1, },
  435. { opcode: 0x0F4F, e: 1, os: 1, custom: 1, },
  436. { opcode: 0x0F80, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  437. { opcode: 0x0F81, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  438. { opcode: 0x0F82, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  439. { opcode: 0x0F83, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  440. { opcode: 0x0F84, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  441. { opcode: 0x0F85, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  442. { opcode: 0x0F86, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  443. { opcode: 0x0F87, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  444. { opcode: 0x0F88, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  445. { opcode: 0x0F89, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  446. { opcode: 0x0F8A, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  447. { opcode: 0x0F8B, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  448. { opcode: 0x0F8C, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  449. { opcode: 0x0F8D, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  450. { opcode: 0x0F8E, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  451. { opcode: 0x0F8F, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, imm1632: 1, os: 1, custom: 1, skip: 1, },
  452. { opcode: 0x0F90, e: 1, custom: 1, },
  453. { opcode: 0x0F91, e: 1, custom: 1, },
  454. { opcode: 0x0F92, e: 1, custom: 1, },
  455. { opcode: 0x0F93, e: 1, custom: 1, },
  456. { opcode: 0x0F94, e: 1, custom: 1, },
  457. { opcode: 0x0F95, e: 1, custom: 1, },
  458. { opcode: 0x0F96, e: 1, custom: 1, },
  459. { opcode: 0x0F97, e: 1, custom: 1, },
  460. { opcode: 0x0F98, e: 1, custom: 1, },
  461. { opcode: 0x0F99, e: 1, custom: 1, },
  462. { opcode: 0x0F9A, e: 1, custom: 1, },
  463. { opcode: 0x0F9B, e: 1, custom: 1, },
  464. { opcode: 0x0F9C, e: 1, custom: 1, },
  465. { opcode: 0x0F9D, e: 1, custom: 1, },
  466. { opcode: 0x0F9E, e: 1, custom: 1, },
  467. { opcode: 0x0F9F, e: 1, custom: 1, },
  468. { opcode: 0x0FA0, os: 1, custom: 1, },
  469. { opcode: 0x0FA1, os: 1, block_boundary: 1, skip: 1, }, // pop fs: block_boundary since it uses non-raising cpu exceptions
  470. { opcode: 0x0FA2, skip: 1, },
  471. { opcode: 0x0FA8, os: 1, custom: 1, },
  472. { opcode: 0x0FA9, os: 1, block_boundary: 1, skip: 1, }, // pop gs
  473. { opcode: 0x0FA3, os: 1, e: 1, only_reg: 1, }, // bt (can also index memory, but not supported by test right now)
  474. { opcode: 0x0FAB, os: 1, e: 1, only_reg: 1, },
  475. { opcode: 0x0FB3, os: 1, e: 1, only_reg: 1, },
  476. { opcode: 0x0FBB, os: 1, e: 1, only_reg: 1, },
  477. { opcode: 0x0FBA, os: 1, e: 1, fixed_g: 4, imm8: 1, only_reg: 1, }, // bt
  478. { opcode: 0x0FBA, os: 1, e: 1, fixed_g: 5, imm8: 1, only_reg: 1, },
  479. { opcode: 0x0FBA, os: 1, e: 1, fixed_g: 6, imm8: 1, only_reg: 1, },
  480. { opcode: 0x0FBA, os: 1, e: 1, fixed_g: 7, imm8: 1, only_reg: 1, },
  481. { opcode: 0x0FBC, os: 1, e: 1, mask_flags: af, }, // bsf
  482. { opcode: 0x0FBD, os: 1, e: 1, mask_flags: af, },
  483. // note: overflow flag only undefined if shift is > 1
  484. { opcode: 0x0FA4, os: 1, e: 1, custom: 1, imm8: 1, mask_flags: af | of, }, // shld
  485. { opcode: 0x0FA5, os: 1, e: 1, custom: 1, mask_flags: af | of, },
  486. { opcode: 0x0FAC, os: 1, e: 1, custom: 1, imm8: 1, mask_flags: af | of, },
  487. { opcode: 0x0FAD, os: 1, e: 1, custom: 1, mask_flags: af | of, },
  488. { opcode: 0x0FA6, skip: 1, block_boundary: 1, }, // ud
  489. { opcode: 0x0FA7, skip: 1, block_boundary: 1, }, // ud
  490. { opcode: 0x0FAA, skip: 1 },
  491. { opcode: 0x0FAE, e: 1, fixed_g: 0, reg_ud: 1, task_switch_test: 1, skip: 1, block_boundary: 1, }, // fxsave
  492. { opcode: 0x0FAE, e: 1, fixed_g: 1, reg_ud: 1, task_switch_test: 1, skip: 1, block_boundary: 1, }, // fxrstor
  493. { opcode: 0x0FAE, e: 1, fixed_g: 2, reg_ud: 1, sse: 1, skip: 1, block_boundary: 1, }, // ldmxcsr
  494. { opcode: 0x0FAE, e: 1, fixed_g: 3, reg_ud: 1, sse: 1, skip: 1, block_boundary: 1, }, // stmxcsr
  495. { opcode: 0x0FAE, e: 1, fixed_g: 4, reg_ud: 1, skip: 1, block_boundary: 1, }, // xsave (mem, not implemented)
  496. { opcode: 0x0FAE, e: 1, fixed_g: 5, skip: 1, block_boundary: 1, custom: 1 }, // lfence (reg, only 0), xrstor (mem, not implemented)
  497. { opcode: 0x0FAE, e: 1, fixed_g: 6, skip: 1, block_boundary: 1, }, // mfence (reg, only 0), xsaveopt (mem, not implemented)
  498. { opcode: 0x0FAE, e: 1, fixed_g: 7, skip: 1, block_boundary: 1, }, // sfence (reg, only 0), clflush (mem)
  499. { opcode: 0x0FAF, os: 1, e: 1, mask_flags: af | zf, custom: 1, }, // imul
  500. { opcode: 0x0FB0, e: 1 }, // cmxchg
  501. { opcode: 0x0FB1, os: 1, e: 1, custom: 1 },
  502. { opcode: 0x0FC7, e: 1, fixed_g: 1, reg_ud: 1, }, // cmpxchg8b (memory)
  503. { opcode: 0x0FC7, e: 1, fixed_g: 6, mem_ud: 1, skip: 1, }, // rdrand
  504. { opcode: 0x0FB2, block_boundary: 1, os: 1, e: 1, skip: 1, }, // lss
  505. { opcode: 0x0FB4, block_boundary: 1, os: 1, e: 1, skip: 1, }, // lfs
  506. { opcode: 0x0FB5, block_boundary: 1, os: 1, e: 1, skip: 1, }, // lgs
  507. { opcode: 0x0FB6, os: 1, e: 1, custom: 1 }, // movzx
  508. { opcode: 0x0FB7, os: 1, e: 1, custom: 1 },
  509. { opcode: 0xF30FB8, os: 1, e: 1 }, // popcnt
  510. { opcode: 0x0FB8, os: 1, e: 1, block_boundary: 1, }, // ud
  511. { opcode: 0x0FB9, block_boundary: 1, }, // ud2
  512. { opcode: 0x0FBE, os: 1, e: 1, custom: 1 }, // movsx
  513. { opcode: 0x0FBF, os: 1, e: 1, custom: 1 },
  514. { opcode: 0x0FC0, e: 1, }, // xadd
  515. { opcode: 0x0FC1, os: 1, e: 1, custom: 1 },
  516. { opcode: 0x0FC8, }, // bswap
  517. { opcode: 0x0FC9, },
  518. { opcode: 0x0FCA, },
  519. { opcode: 0x0FCB, },
  520. { opcode: 0x0FCC, },
  521. { opcode: 0x0FCD, },
  522. { opcode: 0x0FCE, },
  523. { opcode: 0x0FCF, },
  524. // mmx, sse
  525. // - skipped or missing are sse3+
  526. { sse: 1, opcode: 0x0F10, e: 1 },
  527. { sse: 1, opcode: 0xF30F10, e: 1 },
  528. { sse: 1, opcode: 0x660F10, e: 1 },
  529. { sse: 1, opcode: 0xF20F10, e: 1 },
  530. { sse: 1, opcode: 0x0F11, e: 1 },
  531. { sse: 1, opcode: 0xF30F11, e: 1 },
  532. { sse: 1, opcode: 0x660F11, e: 1 },
  533. { sse: 1, opcode: 0xF20F11, e: 1 },
  534. { sse: 1, opcode: 0x0F12, e: 1 },
  535. { sse: 1, opcode: 0x660F12, reg_ud: 1, e: 1 },
  536. { sse: 1, opcode: 0xF20F12, e: 1, skip: 1, block_boundary: 1, }, // sse3
  537. { sse: 1, opcode: 0xF30F12, e: 1, skip: 1, block_boundary: 1, }, // sse3
  538. { sse: 1, opcode: 0x0F13, reg_ud: 1, e: 1 },
  539. { sse: 1, opcode: 0x660F13, reg_ud: 1, e: 1 },
  540. { sse: 1, opcode: 0x0F14, e: 1 },
  541. { sse: 1, opcode: 0x660F14, e: 1 },
  542. { sse: 1, opcode: 0x0F15, e: 1 },
  543. { sse: 1, opcode: 0x660F15, e: 1 },
  544. { sse: 1, opcode: 0x0F16, e: 1 },
  545. { sse: 1, opcode: 0x660F16, reg_ud: 1, e: 1 },
  546. { sse: 1, opcode: 0xF30F16, skip: 1, e: 1, block_boundary: 1, }, // sse3
  547. { sse: 1, opcode: 0x0F17, reg_ud: 1, e: 1 },
  548. { sse: 1, opcode: 0x660F17, reg_ud: 1, e: 1 },
  549. { sse: 1, opcode: 0x0F28, e: 1 },
  550. { sse: 1, opcode: 0x660F28, e: 1 },
  551. { sse: 1, opcode: 0x0F29, e: 1, custom: 1 },
  552. { sse: 1, opcode: 0x660F29, e: 1, custom: 1 },
  553. { sse: 1, opcode: 0x0F2A, e: 1, },
  554. { sse: 1, opcode: 0x660F2A, e: 1, },
  555. { sse: 1, opcode: 0xF20F2A, e: 1, },
  556. { sse: 1, opcode: 0xF30F2A, e: 1, },
  557. { sse: 1, opcode: 0x0F2B, reg_ud: 1, e: 1 },
  558. { sse: 1, opcode: 0x660F2B, reg_ud: 1, e: 1 },
  559. { sse: 1, opcode: 0x0F2C, e: 1, },
  560. { sse: 1, opcode: 0x660F2C, e: 1, },
  561. { sse: 1, opcode: 0xF20F2C, e: 1, },
  562. { sse: 1, opcode: 0xF30F2C, e: 1, },
  563. { sse: 1, opcode: 0x0F2D, e: 1, },
  564. { sse: 1, opcode: 0x660F2D, e: 1, },
  565. { sse: 1, opcode: 0xF20F2D, e: 1, },
  566. { sse: 1, opcode: 0xF30F2D, e: 1, },
  567. { sse: 1, opcode: 0x0F2E, e: 1 },
  568. { sse: 1, opcode: 0x660F2E, e: 1 },
  569. { sse: 1, opcode: 0x0F2F, e: 1 },
  570. { sse: 1, opcode: 0x660F2F, e: 1 },
  571. { sse: 1, opcode: 0x0F50, mem_ud: 1, e: 1 },
  572. { sse: 1, opcode: 0x660F50, mem_ud: 1, e: 1 },
  573. { sse: 1, opcode: 0x0F51, e: 1 },
  574. { sse: 1, opcode: 0x660F51, e: 1 },
  575. { sse: 1, opcode: 0xF20F51, e: 1 },
  576. { sse: 1, opcode: 0xF30F51, e: 1 },
  577. // approximation of 1/sqrt(x). Skipped because our approximation doesn't match intel's
  578. { sse: 1, opcode: 0x0F52, e: 1, skip: 1, },
  579. { sse: 1, opcode: 0xF30F52, e: 1, skip: 1, },
  580. // reciprocal: approximation of 1/x. Skipped because our approximation doesn't match intel's
  581. { sse: 1, opcode: 0x0F53, e: 1, skip: 1, },
  582. { sse: 1, opcode: 0xF30F53, e: 1, skip: 1, },
  583. { sse: 1, opcode: 0x0F54, e: 1 },
  584. { sse: 1, opcode: 0x660F54, e: 1 },
  585. { sse: 1, opcode: 0x0F55, e: 1 },
  586. { sse: 1, opcode: 0x660F55, e: 1 },
  587. { sse: 1, opcode: 0x0F56, e: 1 },
  588. { sse: 1, opcode: 0x660F56, e: 1 },
  589. { sse: 1, opcode: 0x0F57, e: 1 },
  590. { sse: 1, opcode: 0x660F57, e: 1 },
  591. { sse: 1, opcode: 0x0F58, e: 1, },
  592. { sse: 1, opcode: 0x660F58, e: 1, },
  593. { sse: 1, opcode: 0xF20F58, e: 1, },
  594. { sse: 1, opcode: 0xF30F58, e: 1, },
  595. { sse: 1, opcode: 0x0F59, e: 1, },
  596. { sse: 1, opcode: 0x660F59, e: 1, },
  597. { sse: 1, opcode: 0xF20F59, e: 1, },
  598. { sse: 1, opcode: 0xF30F59, e: 1, },
  599. { sse: 1, opcode: 0x0F5A, e: 1, },
  600. { sse: 1, opcode: 0x660F5A, e: 1, },
  601. { sse: 1, opcode: 0xF20F5A, e: 1, },
  602. { sse: 1, opcode: 0xF30F5A, e: 1, },
  603. { sse: 1, opcode: 0x0F5B, e: 1, },
  604. { sse: 1, opcode: 0x660F5B, e: 1, },
  605. // no F2 variant
  606. { sse: 1, opcode: 0xF30F5B, e: 1, },
  607. { sse: 1, opcode: 0x0F5C, e: 1, },
  608. { sse: 1, opcode: 0x660F5C, e: 1, },
  609. { sse: 1, opcode: 0xF20F5C, e: 1, },
  610. { sse: 1, opcode: 0xF30F5C, e: 1, },
  611. { sse: 1, opcode: 0x0F5D, e: 1, },
  612. { sse: 1, opcode: 0x660F5D, e: 1, },
  613. { sse: 1, opcode: 0xF20F5D, e: 1, },
  614. { sse: 1, opcode: 0xF30F5D, e: 1, },
  615. { sse: 1, opcode: 0x0F5E, e: 1, },
  616. { sse: 1, opcode: 0x660F5E, e: 1, },
  617. { sse: 1, opcode: 0xF20F5E, e: 1, },
  618. { sse: 1, opcode: 0xF30F5E, e: 1, },
  619. { sse: 1, opcode: 0x0F5F, e: 1, },
  620. { sse: 1, opcode: 0x660F5F, e: 1, },
  621. { sse: 1, opcode: 0xF20F5F, e: 1, },
  622. { sse: 1, opcode: 0xF30F5F, e: 1, },
  623. { sse: 1, opcode: 0x660F60, e: 1, custom: 1 },
  624. { sse: 1, opcode: 0x0F60, e: 1 },
  625. { sse: 1, opcode: 0x660F61, e: 1, custom: 1 },
  626. { sse: 1, opcode: 0x0F61, e: 1 },
  627. { sse: 1, opcode: 0x660F62, e: 1 },
  628. { sse: 1, opcode: 0x0F62, e: 1 },
  629. { sse: 1, opcode: 0x660F63, e: 1 },
  630. { sse: 1, opcode: 0x0F63, e: 1 },
  631. { sse: 1, opcode: 0x660F64, e: 1 },
  632. { sse: 1, opcode: 0x0F64, e: 1 },
  633. { sse: 1, opcode: 0x660F65, e: 1 },
  634. { sse: 1, opcode: 0x0F65, e: 1 },
  635. { sse: 1, opcode: 0x660F66, e: 1 },
  636. { sse: 1, opcode: 0x0F66, e: 1 },
  637. { sse: 1, opcode: 0x660F67, e: 1, custom: 1 },
  638. { sse: 1, opcode: 0x0F67, e: 1 },
  639. { sse: 1, opcode: 0x660F68, e: 1, custom: 1 },
  640. { sse: 1, opcode: 0x0F68, e: 1 },
  641. { sse: 1, opcode: 0x660F69, e: 1 },
  642. { sse: 1, opcode: 0x0F69, e: 1 },
  643. { sse: 1, opcode: 0x660F6A, e: 1 },
  644. { sse: 1, opcode: 0x0F6A, e: 1 },
  645. { sse: 1, opcode: 0x660F6B, e: 1 },
  646. { sse: 1, opcode: 0x0F6B, e: 1 },
  647. { sse: 1, opcode: 0x660F6C, e: 1 },
  648. { sse: 1, opcode: 0x0F6C, e: 1, block_boundary: 1, }, // ud
  649. { sse: 1, opcode: 0x660F6D, e: 1 },
  650. { sse: 1, opcode: 0x0F6D, e: 1, block_boundary: 1, }, // ud
  651. { sse: 1, opcode: 0x660F6E, e: 1, custom: 1 },
  652. { sse: 1, opcode: 0x0F6E, e: 1 },
  653. { sse: 1, opcode: 0xF30F6F, e: 1, custom: 1 },
  654. { sse: 1, opcode: 0x660F6F, e: 1, custom: 1 },
  655. { sse: 1, opcode: 0x0F6F, e: 1 },
  656. { sse: 1, opcode: 0x0F70, e: 1, imm8: 1, },
  657. { sse: 1, opcode: 0x660F70, e: 1, imm8: 1, custom: 1 },
  658. { sse: 1, opcode: 0xF20F70, e: 1, imm8: 1, custom: 1 },
  659. { sse: 1, opcode: 0xF30F70, e: 1, imm8: 1, custom: 1 },
  660. { sse: 1, opcode: 0x0F71, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1, },
  661. { sse: 1, opcode: 0x660F71, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1 },
  662. { sse: 1, opcode: 0x0F71, e: 1, fixed_g: 4, imm8: 1, mem_ud: 1, },
  663. { sse: 1, opcode: 0x660F71, e: 1, fixed_g: 4, imm8: 1, mem_ud: 1 },
  664. { sse: 1, opcode: 0x0F71, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1, },
  665. { sse: 1, opcode: 0x660F71, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1 },
  666. { sse: 1, opcode: 0x0F72, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1, },
  667. { sse: 1, opcode: 0x660F72, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1 },
  668. { sse: 1, opcode: 0x0F72, e: 1, fixed_g: 4, imm8: 1, mem_ud: 1, },
  669. { sse: 1, opcode: 0x660F72, e: 1, fixed_g: 4, imm8: 1, mem_ud: 1 },
  670. { sse: 1, opcode: 0x0F72, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1, },
  671. { sse: 1, opcode: 0x660F72, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1 },
  672. { sse: 1, opcode: 0x0F73, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1, },
  673. { sse: 1, opcode: 0x660F73, e: 1, fixed_g: 2, imm8: 1, mem_ud: 1, },
  674. { sse: 1, opcode: 0x660F73, e: 1, fixed_g: 3, imm8: 1, mem_ud: 1 },
  675. { sse: 1, opcode: 0x0F73, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1, },
  676. { sse: 1, opcode: 0x660F73, e: 1, fixed_g: 6, imm8: 1, mem_ud: 1 },
  677. { sse: 1, opcode: 0x660F73, e: 1, fixed_g: 7, imm8: 1, mem_ud: 1 },
  678. { sse: 1, opcode: 0x0F74, e: 1, },
  679. { sse: 1, opcode: 0x660F74, e: 1, custom: 1 },
  680. { sse: 1, opcode: 0x0F75, e: 1, },
  681. { sse: 1, opcode: 0x660F75, e: 1, },
  682. { sse: 1, opcode: 0x0F76, e: 1, },
  683. { sse: 1, opcode: 0x660F76, e: 1, },
  684. { sse: 1, opcode: 0x0F77, skip: 1 }, // emms (skip as it breaks gdb printing of float registers)
  685. // vmx instructions
  686. { opcode: 0x0F78, skip: 1, block_boundary: 1, },
  687. { opcode: 0x0F79, skip: 1, block_boundary: 1, },
  688. { opcode: 0x0F7A, skip: 1, block_boundary: 1, }, // ud
  689. { opcode: 0x0F7B, skip: 1, block_boundary: 1, }, // ud
  690. { sse: 1, opcode: 0x0F7C, skip: 1, block_boundary: 1, }, // sse3
  691. { sse: 1, opcode: 0x0F7D, skip: 1, block_boundary: 1, }, // sse3
  692. { sse: 1, opcode: 0x0F7E, e: 1 },
  693. { sse: 1, opcode: 0x660F7E, e: 1, custom: 1 },
  694. { sse: 1, opcode: 0xF30F7E, e: 1, custom: 1 },
  695. { sse: 1, opcode: 0x0F7F, e: 1 },
  696. { sse: 1, opcode: 0x660F7F, e: 1, custom: 1 },
  697. { sse: 1, opcode: 0xF30F7F, e: 1, custom: 1 },
  698. { sse: 1, opcode: 0x0FC2, e: 1, imm8: 1 },
  699. { sse: 1, opcode: 0x660FC2, e: 1, imm8: 1 },
  700. { sse: 1, opcode: 0xF20FC2, e: 1, imm8: 1 },
  701. { sse: 1, opcode: 0xF30FC2, e: 1, imm8: 1 },
  702. { opcode: 0x0FC3, e: 1, reg_ud: 1, }, // movnti: Uses normal registers, hence not marked as sse
  703. { sse: 1, opcode: 0x0FC4, e: 1, imm8: 1 },
  704. { sse: 1, opcode: 0x660FC4, e: 1, imm8: 1 },
  705. { sse: 1, opcode: 0x0FC5, e: 1, mem_ud: 1, imm8: 1 },
  706. { sse: 1, opcode: 0x660FC5, e: 1, mem_ud: 1, imm8: 1, },
  707. { sse: 1, opcode: 0x0FC6, e: 1, imm8: 1 },
  708. { sse: 1, opcode: 0x660FC6, e: 1, imm8: 1 },
  709. { sse: 1, opcode: 0x0FD0, skip: 1, block_boundary: 1, }, // sse3
  710. { sse: 1, opcode: 0x0FD1, e: 1 },
  711. { sse: 1, opcode: 0x660FD1, e: 1 },
  712. { sse: 1, opcode: 0x0FD2, e: 1 },
  713. { sse: 1, opcode: 0x660FD2, e: 1 },
  714. { sse: 1, opcode: 0x0FD3, e: 1 },
  715. { sse: 1, opcode: 0x660FD3, e: 1 },
  716. { sse: 1, opcode: 0x0FD4, e: 1 },
  717. { sse: 1, opcode: 0x660FD4, e: 1 },
  718. { sse: 1, opcode: 0x0FD5, e: 1 },
  719. { sse: 1, opcode: 0x660FD5, e: 1, custom: 1 },
  720. { sse: 1, opcode: 0x660FD6, e: 1, custom: 1 },
  721. { sse: 1, opcode: 0xF20FD6, mem_ud: 1, e: 1 },
  722. { sse: 1, opcode: 0xF30FD6, mem_ud: 1, e: 1 },
  723. { sse: 1, opcode: 0x0FD6, e: 1, block_boundary: 1, }, // ud
  724. { sse: 1, opcode: 0x0FD7, e: 1, mem_ud: 1 },
  725. { sse: 1, opcode: 0x660FD7, e: 1, mem_ud: 1, },
  726. { sse: 1, opcode: 0x0FD8, e: 1 },
  727. { sse: 1, opcode: 0x660FD8, e: 1 },
  728. { sse: 1, opcode: 0x0FD9, e: 1 },
  729. { sse: 1, opcode: 0x660FD9, e: 1 },
  730. { sse: 1, opcode: 0x0FDA, e: 1 },
  731. { sse: 1, opcode: 0x660FDA, e: 1 },
  732. { sse: 1, opcode: 0x0FDB, e: 1 },
  733. { sse: 1, opcode: 0x660FDB, e: 1 },
  734. { sse: 1, opcode: 0x0FDC, e: 1 },
  735. { sse: 1, opcode: 0x660FDC, e: 1, custom: 1 },
  736. { sse: 1, opcode: 0x0FDD, e: 1 },
  737. { sse: 1, opcode: 0x660FDD, e: 1, custom: 1 },
  738. { sse: 1, opcode: 0x0FDE, e: 1 },
  739. { sse: 1, opcode: 0x660FDE, e: 1 },
  740. { sse: 1, opcode: 0x0FDF, e: 1 },
  741. { sse: 1, opcode: 0x660FDF, e: 1 },
  742. { sse: 1, opcode: 0x0FE0, e: 1 },
  743. { sse: 1, opcode: 0x660FE0, e: 1 },
  744. { sse: 1, opcode: 0x0FE1, e: 1 },
  745. { sse: 1, opcode: 0x660FE1, e: 1 },
  746. { sse: 1, opcode: 0x0FE2, e: 1 },
  747. { sse: 1, opcode: 0x660FE2, e: 1 },
  748. { sse: 1, opcode: 0x0FE3, e: 1 },
  749. { sse: 1, opcode: 0x660FE3, e: 1 },
  750. { sse: 1, opcode: 0x0FE4, e: 1 },
  751. { sse: 1, opcode: 0x660FE4, e: 1, custom: 1 },
  752. { sse: 1, opcode: 0x0FE5, e: 1 },
  753. { sse: 1, opcode: 0x660FE5, e: 1 },
  754. { sse: 1, opcode: 0x660FE6, e: 1 },
  755. { sse: 1, opcode: 0xF20FE6, e: 1 },
  756. { sse: 1, opcode: 0xF30FE6, e: 1 },
  757. { sse: 1, opcode: 0x0FE6, e: 1, block_boundary: 1, }, // ud
  758. { sse: 1, opcode: 0x0FE7, e: 1, reg_ud: 1 },
  759. { sse: 1, opcode: 0x660FE7, e: 1, reg_ud: 1, },
  760. { sse: 1, opcode: 0x0FE8, e: 1 },
  761. { sse: 1, opcode: 0x660FE8, e: 1 },
  762. { sse: 1, opcode: 0x0FE9, e: 1 },
  763. { sse: 1, opcode: 0x660FE9, e: 1 },
  764. { sse: 1, opcode: 0x0FEA, e: 1 },
  765. { sse: 1, opcode: 0x660FEA, e: 1 },
  766. { sse: 1, opcode: 0x0FEB, e: 1 },
  767. { sse: 1, opcode: 0x660FEB, e: 1, custom: 1 },
  768. { sse: 1, opcode: 0x0FEC, e: 1 },
  769. { sse: 1, opcode: 0x660FEC, e: 1 },
  770. { sse: 1, opcode: 0x0FED, e: 1 },
  771. { sse: 1, opcode: 0x660FED, e: 1 },
  772. { sse: 1, opcode: 0x0FEE, e: 1 },
  773. { sse: 1, opcode: 0x660FEE, e: 1 },
  774. { sse: 1, opcode: 0x0FEF, e: 1 },
  775. { sse: 1, opcode: 0x660FEF, e: 1, custom: 1 },
  776. { sse: 1, opcode: 0x0FF0, skip: 1, block_boundary: 1, }, // sse3
  777. { sse: 1, opcode: 0x0FF1, e: 1 },
  778. { sse: 1, opcode: 0x660FF1, e: 1 },
  779. { sse: 1, opcode: 0x0FF2, e: 1 },
  780. { sse: 1, opcode: 0x660FF2, e: 1 },
  781. { sse: 1, opcode: 0x0FF3, e: 1 },
  782. { sse: 1, opcode: 0x660FF3, e: 1, },
  783. { sse: 1, opcode: 0x0FF4, e: 1 },
  784. { sse: 1, opcode: 0x660FF4, e: 1 },
  785. { sse: 1, opcode: 0x0FF5, e: 1 },
  786. { sse: 1, opcode: 0x660FF5, e: 1 },
  787. { sse: 1, opcode: 0x0FF6, e: 1 },
  788. { sse: 1, opcode: 0x660FF6, e: 1 },
  789. // maskmovq (0FF7), maskmovdqu (660FF7) tested manually
  790. // Generated tests don't setup EDI as required (yet)
  791. { sse: 1, opcode: 0x0FF7, mem_ud: 1, e: 1, skip: 1, },
  792. { sse: 1, opcode: 0x660FF7, mem_ud: 1, e: 1, skip: 1, },
  793. { sse: 1, opcode: 0x0FF8, e: 1 },
  794. { sse: 1, opcode: 0x660FF8, e: 1 },
  795. { sse: 1, opcode: 0x0FF9, e: 1 },
  796. { sse: 1, opcode: 0x660FF9, e: 1 },
  797. { sse: 1, opcode: 0x0FFA, e: 1 },
  798. { sse: 1, opcode: 0x660FFA, e: 1 },
  799. { sse: 1, opcode: 0x0FFB, e: 1 },
  800. { sse: 1, opcode: 0x660FFB, e: 1 },
  801. { sse: 1, opcode: 0x0FFC, e: 1 },
  802. { sse: 1, opcode: 0x660FFC, e: 1 },
  803. { sse: 1, opcode: 0x0FFD, e: 1 },
  804. { sse: 1, opcode: 0x660FFD, e: 1 },
  805. { sse: 1, opcode: 0x0FFE, e: 1 },
  806. { sse: 1, opcode: 0x660FFE, e: 1 },
  807. { opcode: 0x0FFF, block_boundary: 1, }, // ud
  808. ];
  809. for(let i = 0; i < 8; i++)
  810. {
  811. encodings.push.apply(encodings, [
  812. { opcode: 0x04 | i << 3, custom: 1, imm8: 1, },
  813. { opcode: 0x05 | i << 3, custom: 1, os: 1, imm1632: 1, },
  814. { opcode: 0x70 | i, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, os: 1, imm8s: 1, custom: 1, skip: 1, },
  815. { opcode: 0x78 | i, block_boundary: 1, no_block_boundary_in_interpreted: 1, jump_offset_imm: 1, conditional_jump: 1, os: 1, imm8s: 1, custom: 1, skip: 1, },
  816. { opcode: 0x80, e: 1, fixed_g: i, imm8: 1, custom: 1, },
  817. { opcode: 0x81, os: 1, e: 1, fixed_g: i, imm1632: 1, custom: 1, },
  818. { opcode: 0x82, e: 1, fixed_g: i, imm8: 1, },
  819. { opcode: 0x83, os: 1, e: 1, fixed_g: i, imm8s: 1, custom: 1, },
  820. { opcode: 0xB0 | i, custom: 1, imm8: 1, },
  821. { opcode: 0xB8 | i, custom: 1, os: 1, imm1632: 1, },
  822. // note: overflow flag only undefined if shift is > 1
  823. // note: the adjust flag is undefined for shifts > 0 and unaffected by rotates
  824. { opcode: 0xC0, e: 1, fixed_g: i, imm8: 1, mask_flags: of | af, custom: 1, },
  825. { opcode: 0xC1, os: 1, e: 1, fixed_g: i, imm8: 1, mask_flags: of | af, custom: 1, },
  826. { opcode: 0xD0, e: 1, fixed_g: i, mask_flags: af, custom: 1 },
  827. { opcode: 0xD1, os: 1, e: 1, fixed_g: i, mask_flags: af, custom: 1, },
  828. { opcode: 0xD2, e: 1, fixed_g: i, mask_flags: of | af, custom: 1 },
  829. { opcode: 0xD3, os: 1, e: 1, fixed_g: i, mask_flags: of | af, custom: 1, },
  830. ]);
  831. }
  832. encodings.sort((e1, e2) => {
  833. let o1 = (e1.opcode & 0xFF00) === 0x0F00 ? e1.opcode & 0xFFFF : e1.opcode & 0xFF;
  834. let o2 = (e2.opcode & 0xFF00) === 0x0F00 ? e2.opcode & 0xFFFF : e2.opcode & 0xFF;
  835. return o1 - o2 || e1.fixed_g - e2.fixed_g;
  836. });
  837. module.exports = Object.freeze(encodings.map(entry => Object.freeze(entry)));