inat.h 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. #ifndef _ASM_X86_INAT_H
  2. #define _ASM_X86_INAT_H
  3. /*
  4. * x86 instruction attributes
  5. *
  6. * Written by Masami Hiramatsu <mhiramat@redhat.com>
  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 "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_LSTPFX_MAX 3
  49. #define INAT_LGCPFX_MAX 11
  50. /* Immediate size */
  51. #define INAT_IMM_BYTE 1
  52. #define INAT_IMM_WORD 2
  53. #define INAT_IMM_DWORD 3
  54. #define INAT_IMM_QWORD 4
  55. #define INAT_IMM_PTR 5
  56. #define INAT_IMM_VWORD32 6
  57. #define INAT_IMM_VWORD 7
  58. /* Legacy prefix */
  59. #define INAT_PFX_OFFS 0
  60. #define INAT_PFX_BITS 4
  61. #define INAT_PFX_MAX ((1 << INAT_PFX_BITS) - 1)
  62. #define INAT_PFX_MASK (INAT_PFX_MAX << INAT_PFX_OFFS)
  63. /* Escape opcodes */
  64. #define INAT_ESC_OFFS (INAT_PFX_OFFS + INAT_PFX_BITS)
  65. #define INAT_ESC_BITS 2
  66. #define INAT_ESC_MAX ((1 << INAT_ESC_BITS) - 1)
  67. #define INAT_ESC_MASK (INAT_ESC_MAX << INAT_ESC_OFFS)
  68. /* Group opcodes (1-16) */
  69. #define INAT_GRP_OFFS (INAT_ESC_OFFS + INAT_ESC_BITS)
  70. #define INAT_GRP_BITS 5
  71. #define INAT_GRP_MAX ((1 << INAT_GRP_BITS) - 1)
  72. #define INAT_GRP_MASK (INAT_GRP_MAX << INAT_GRP_OFFS)
  73. /* Immediates */
  74. #define INAT_IMM_OFFS (INAT_GRP_OFFS + INAT_GRP_BITS)
  75. #define INAT_IMM_BITS 3
  76. #define INAT_IMM_MASK (((1 << INAT_IMM_BITS) - 1) << INAT_IMM_OFFS)
  77. /* Flags */
  78. #define INAT_FLAG_OFFS (INAT_IMM_OFFS + INAT_IMM_BITS)
  79. #define INAT_MODRM (1 << (INAT_FLAG_OFFS))
  80. #define INAT_FORCE64 (1 << (INAT_FLAG_OFFS + 1))
  81. #define INAT_SCNDIMM (1 << (INAT_FLAG_OFFS + 2))
  82. #define INAT_MOFFSET (1 << (INAT_FLAG_OFFS + 3))
  83. #define INAT_VARIANT (1 << (INAT_FLAG_OFFS + 4))
  84. #define INAT_VEXOK (1 << (INAT_FLAG_OFFS + 5))
  85. #define INAT_VEXONLY (1 << (INAT_FLAG_OFFS + 6))
  86. /* Attribute making macros for attribute tables */
  87. #define INAT_MAKE_PREFIX(pfx) (pfx << INAT_PFX_OFFS)
  88. #define INAT_MAKE_ESCAPE(esc) (esc << INAT_ESC_OFFS)
  89. #define INAT_MAKE_GROUP(grp) ((grp << INAT_GRP_OFFS) | INAT_MODRM)
  90. #define INAT_MAKE_IMM(imm) (imm << INAT_IMM_OFFS)
  91. /* Attribute search APIs */
  92. extern insn_attr_t inat_get_opcode_attribute(insn_byte_t opcode);
  93. extern int inat_get_last_prefix_id(insn_byte_t last_pfx);
  94. extern insn_attr_t inat_get_escape_attribute(insn_byte_t opcode,
  95. int lpfx_id,
  96. insn_attr_t esc_attr);
  97. extern insn_attr_t inat_get_group_attribute(insn_byte_t modrm,
  98. int lpfx_id,
  99. insn_attr_t esc_attr);
  100. extern insn_attr_t inat_get_avx_attribute(insn_byte_t opcode,
  101. insn_byte_t vex_m,
  102. insn_byte_t vex_pp);
  103. /* Attribute checking functions */
  104. static inline int inat_is_legacy_prefix(insn_attr_t attr)
  105. {
  106. attr &= INAT_PFX_MASK;
  107. return attr && attr <= INAT_LGCPFX_MAX;
  108. }
  109. static inline int inat_is_address_size_prefix(insn_attr_t attr)
  110. {
  111. return (attr & INAT_PFX_MASK) == INAT_PFX_ADDRSZ;
  112. }
  113. static inline int inat_is_operand_size_prefix(insn_attr_t attr)
  114. {
  115. return (attr & INAT_PFX_MASK) == INAT_PFX_OPNDSZ;
  116. }
  117. static inline int inat_is_rex_prefix(insn_attr_t attr)
  118. {
  119. return (attr & INAT_PFX_MASK) == INAT_PFX_REX;
  120. }
  121. static inline int inat_last_prefix_id(insn_attr_t attr)
  122. {
  123. if ((attr & INAT_PFX_MASK) > INAT_LSTPFX_MAX)
  124. return 0;
  125. else
  126. return attr & INAT_PFX_MASK;
  127. }
  128. static inline int inat_is_vex_prefix(insn_attr_t attr)
  129. {
  130. attr &= INAT_PFX_MASK;
  131. return attr == INAT_PFX_VEX2 || attr == INAT_PFX_VEX3;
  132. }
  133. static inline int inat_is_vex3_prefix(insn_attr_t attr)
  134. {
  135. return (attr & INAT_PFX_MASK) == INAT_PFX_VEX3;
  136. }
  137. static inline int inat_is_escape(insn_attr_t attr)
  138. {
  139. return attr & INAT_ESC_MASK;
  140. }
  141. static inline int inat_escape_id(insn_attr_t attr)
  142. {
  143. return (attr & INAT_ESC_MASK) >> INAT_ESC_OFFS;
  144. }
  145. static inline int inat_is_group(insn_attr_t attr)
  146. {
  147. return attr & INAT_GRP_MASK;
  148. }
  149. static inline int inat_group_id(insn_attr_t attr)
  150. {
  151. return (attr & INAT_GRP_MASK) >> INAT_GRP_OFFS;
  152. }
  153. static inline int inat_group_common_attribute(insn_attr_t attr)
  154. {
  155. return attr & ~INAT_GRP_MASK;
  156. }
  157. static inline int inat_has_immediate(insn_attr_t attr)
  158. {
  159. return attr & INAT_IMM_MASK;
  160. }
  161. static inline int inat_immediate_size(insn_attr_t attr)
  162. {
  163. return (attr & INAT_IMM_MASK) >> INAT_IMM_OFFS;
  164. }
  165. static inline int inat_has_modrm(insn_attr_t attr)
  166. {
  167. return attr & INAT_MODRM;
  168. }
  169. static inline int inat_is_force64(insn_attr_t attr)
  170. {
  171. return attr & INAT_FORCE64;
  172. }
  173. static inline int inat_has_second_immediate(insn_attr_t attr)
  174. {
  175. return attr & INAT_SCNDIMM;
  176. }
  177. static inline int inat_has_moffset(insn_attr_t attr)
  178. {
  179. return attr & INAT_MOFFSET;
  180. }
  181. static inline int inat_has_variant(insn_attr_t attr)
  182. {
  183. return attr & INAT_VARIANT;
  184. }
  185. static inline int inat_accept_vex(insn_attr_t attr)
  186. {
  187. return attr & INAT_VEXOK;
  188. }
  189. static inline int inat_must_vex(insn_attr_t attr)
  190. {
  191. return attr & INAT_VEXONLY;
  192. }
  193. #endif