inat.h 5.9 KB

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