inat.h 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. #ifndef _ASM_X86_INAT_H
  2. #define _ASM_X86_INAT_H
  3. /*
  4. * x86 instruction attributes
  5. *
  6. * Written by Masami Hiramatsu <[email protected]>
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  21. *
  22. */
  23. #include <asm/inat_types.h>
  24. /*
  25. * Internal bits. Don't use bitmasks directly, because these bits are
  26. * unstable. You should use checking functions.
  27. */
  28. #define INAT_OPCODE_TABLE_SIZE 256
  29. #define INAT_GROUP_TABLE_SIZE 8
  30. /* Legacy last prefixes */
  31. #define INAT_PFX_OPNDSZ 1 /* 0x66 */ /* LPFX1 */
  32. #define INAT_PFX_REPE 2 /* 0xF3 */ /* LPFX2 */
  33. #define INAT_PFX_REPNE 3 /* 0xF2 */ /* LPFX3 */
  34. /* Other Legacy prefixes */
  35. #define INAT_PFX_LOCK 4 /* 0xF0 */
  36. #define INAT_PFX_CS 5 /* 0x2E */
  37. #define INAT_PFX_DS 6 /* 0x3E */
  38. #define INAT_PFX_ES 7 /* 0x26 */
  39. #define INAT_PFX_FS 8 /* 0x64 */
  40. #define INAT_PFX_GS 9 /* 0x65 */
  41. #define INAT_PFX_SS 10 /* 0x36 */
  42. #define INAT_PFX_ADDRSZ 11 /* 0x67 */
  43. /* x86-64 REX prefix */
  44. #define INAT_PFX_REX 12 /* 0x4X */
  45. /* AVX VEX prefixes */
  46. #define INAT_PFX_VEX2 13 /* 2-bytes VEX prefix */
  47. #define INAT_PFX_VEX3 14 /* 3-bytes VEX prefix */
  48. #define INAT_PFX_EVEX 15 /* EVEX prefix */
  49. #define INAT_LSTPFX_MAX 3
  50. #define INAT_LGCPFX_MAX 11
  51. /* Immediate size */
  52. #define INAT_IMM_BYTE 1
  53. #define INAT_IMM_WORD 2
  54. #define INAT_IMM_DWORD 3
  55. #define INAT_IMM_QWORD 4
  56. #define INAT_IMM_PTR 5
  57. #define INAT_IMM_VWORD32 6
  58. #define INAT_IMM_VWORD 7
  59. /* Legacy prefix */
  60. #define INAT_PFX_OFFS 0
  61. #define INAT_PFX_BITS 4
  62. #define INAT_PFX_MAX ((1 << INAT_PFX_BITS) - 1)
  63. #define INAT_PFX_MASK (INAT_PFX_MAX << INAT_PFX_OFFS)
  64. /* Escape opcodes */
  65. #define INAT_ESC_OFFS (INAT_PFX_OFFS + INAT_PFX_BITS)
  66. #define INAT_ESC_BITS 2
  67. #define INAT_ESC_MAX ((1 << INAT_ESC_BITS) - 1)
  68. #define INAT_ESC_MASK (INAT_ESC_MAX << INAT_ESC_OFFS)
  69. /* Group opcodes (1-16) */
  70. #define INAT_GRP_OFFS (INAT_ESC_OFFS + INAT_ESC_BITS)
  71. #define INAT_GRP_BITS 5
  72. #define INAT_GRP_MAX ((1 << INAT_GRP_BITS) - 1)
  73. #define INAT_GRP_MASK (INAT_GRP_MAX << INAT_GRP_OFFS)
  74. /* Immediates */
  75. #define INAT_IMM_OFFS (INAT_GRP_OFFS + INAT_GRP_BITS)
  76. #define INAT_IMM_BITS 3
  77. #define INAT_IMM_MASK (((1 << INAT_IMM_BITS) - 1) << INAT_IMM_OFFS)
  78. /* Flags */
  79. #define INAT_FLAG_OFFS (INAT_IMM_OFFS + INAT_IMM_BITS)
  80. #define INAT_MODRM (1 << (INAT_FLAG_OFFS))
  81. #define INAT_FORCE64 (1 << (INAT_FLAG_OFFS + 1))
  82. #define INAT_SCNDIMM (1 << (INAT_FLAG_OFFS + 2))
  83. #define INAT_MOFFSET (1 << (INAT_FLAG_OFFS + 3))
  84. #define INAT_VARIANT (1 << (INAT_FLAG_OFFS + 4))
  85. #define INAT_VEXOK (1 << (INAT_FLAG_OFFS + 5))
  86. #define INAT_VEXONLY (1 << (INAT_FLAG_OFFS + 6))
  87. #define INAT_EVEXONLY (1 << (INAT_FLAG_OFFS + 7))
  88. /* Attribute making macros for attribute tables */
  89. #define INAT_MAKE_PREFIX(pfx) (pfx << INAT_PFX_OFFS)
  90. #define INAT_MAKE_ESCAPE(esc) (esc << INAT_ESC_OFFS)
  91. #define INAT_MAKE_GROUP(grp) ((grp << INAT_GRP_OFFS) | INAT_MODRM)
  92. #define INAT_MAKE_IMM(imm) (imm << INAT_IMM_OFFS)
  93. /* Attribute search APIs */
  94. extern insn_attr_t inat_get_opcode_attribute(insn_byte_t opcode);
  95. extern int inat_get_last_prefix_id(insn_byte_t last_pfx);
  96. extern insn_attr_t inat_get_escape_attribute(insn_byte_t opcode,
  97. int lpfx_id,
  98. insn_attr_t esc_attr);
  99. extern insn_attr_t inat_get_group_attribute(insn_byte_t modrm,
  100. int lpfx_id,
  101. insn_attr_t esc_attr);
  102. extern insn_attr_t inat_get_avx_attribute(insn_byte_t opcode,
  103. insn_byte_t vex_m,
  104. insn_byte_t vex_pp);
  105. /* Attribute checking functions */
  106. static inline int inat_is_legacy_prefix(insn_attr_t attr)
  107. {
  108. attr &= INAT_PFX_MASK;
  109. return attr && attr <= INAT_LGCPFX_MAX;
  110. }
  111. static inline int inat_is_address_size_prefix(insn_attr_t attr)
  112. {
  113. return (attr & INAT_PFX_MASK) == INAT_PFX_ADDRSZ;
  114. }
  115. static inline int inat_is_operand_size_prefix(insn_attr_t attr)
  116. {
  117. return (attr & INAT_PFX_MASK) == INAT_PFX_OPNDSZ;
  118. }
  119. static inline int inat_is_rex_prefix(insn_attr_t attr)
  120. {
  121. return (attr & INAT_PFX_MASK) == INAT_PFX_REX;
  122. }
  123. static inline int inat_last_prefix_id(insn_attr_t attr)
  124. {
  125. if ((attr & INAT_PFX_MASK) > INAT_LSTPFX_MAX)
  126. return 0;
  127. else
  128. return attr & INAT_PFX_MASK;
  129. }
  130. static inline int inat_is_vex_prefix(insn_attr_t attr)
  131. {
  132. attr &= INAT_PFX_MASK;
  133. return attr == INAT_PFX_VEX2 || attr == INAT_PFX_VEX3 ||
  134. attr == INAT_PFX_EVEX;
  135. }
  136. static inline int inat_is_evex_prefix(insn_attr_t attr)
  137. {
  138. return (attr & INAT_PFX_MASK) == INAT_PFX_EVEX;
  139. }
  140. static inline int inat_is_vex3_prefix(insn_attr_t attr)
  141. {
  142. return (attr & INAT_PFX_MASK) == INAT_PFX_VEX3;
  143. }
  144. static inline int inat_is_escape(insn_attr_t attr)
  145. {
  146. return attr & INAT_ESC_MASK;
  147. }
  148. static inline int inat_escape_id(insn_attr_t attr)
  149. {
  150. return (attr & INAT_ESC_MASK) >> INAT_ESC_OFFS;
  151. }
  152. static inline int inat_is_group(insn_attr_t attr)
  153. {
  154. return attr & INAT_GRP_MASK;
  155. }
  156. static inline int inat_group_id(insn_attr_t attr)
  157. {
  158. return (attr & INAT_GRP_MASK) >> INAT_GRP_OFFS;
  159. }
  160. static inline int inat_group_common_attribute(insn_attr_t attr)
  161. {
  162. return attr & ~INAT_GRP_MASK;
  163. }
  164. static inline int inat_has_immediate(insn_attr_t attr)
  165. {
  166. return attr & INAT_IMM_MASK;
  167. }
  168. static inline int inat_immediate_size(insn_attr_t attr)
  169. {
  170. return (attr & INAT_IMM_MASK) >> INAT_IMM_OFFS;
  171. }
  172. static inline int inat_has_modrm(insn_attr_t attr)
  173. {
  174. return attr & INAT_MODRM;
  175. }
  176. static inline int inat_is_force64(insn_attr_t attr)
  177. {
  178. return attr & INAT_FORCE64;
  179. }
  180. static inline int inat_has_second_immediate(insn_attr_t attr)
  181. {
  182. return attr & INAT_SCNDIMM;
  183. }
  184. static inline int inat_has_moffset(insn_attr_t attr)
  185. {
  186. return attr & INAT_MOFFSET;
  187. }
  188. static inline int inat_has_variant(insn_attr_t attr)
  189. {
  190. return attr & INAT_VARIANT;
  191. }
  192. static inline int inat_accept_vex(insn_attr_t attr)
  193. {
  194. return attr & INAT_VEXOK;
  195. }
  196. static inline int inat_must_vex(insn_attr_t attr)
  197. {
  198. return attr & (INAT_VEXONLY | INAT_EVEXONLY);
  199. }
  200. static inline int inat_must_evex(insn_attr_t attr)
  201. {
  202. return attr & INAT_EVEXONLY;
  203. }
  204. #endif