summaryrefslogtreecommitdiff
path: root/gnu/usr.bin/binutils/opcodes
diff options
context:
space:
mode:
authorPhilip Guenther <guenther@cvs.openbsd.org>2014-02-09 22:42:28 +0000
committerPhilip Guenther <guenther@cvs.openbsd.org>2014-02-09 22:42:28 +0000
commit3d28e9ace295091b0d7b011e31f3f5344ef7a4e7 (patch)
tree682c2df4e4d999ba9289d9ea3df3ddd56969fbd1 /gnu/usr.bin/binutils/opcodes
parent1479cfc9acfbc2f084be12b3af9c73435479da5d (diff)
Add support for i386 XSAVE family of instructions: xgetbv, xsetbv, xsave,
xrstor, and xsaveopt. based on kettenis's original that did xgetbv and xsetbv ok kettenis@
Diffstat (limited to 'gnu/usr.bin/binutils/opcodes')
-rw-r--r--gnu/usr.bin/binutils/opcodes/i386-dis.c43
1 files changed, 36 insertions, 7 deletions
diff --git a/gnu/usr.bin/binutils/opcodes/i386-dis.c b/gnu/usr.bin/binutils/opcodes/i386-dis.c
index f33929fd8f8..cbdd567b226 100644
--- a/gnu/usr.bin/binutils/opcodes/i386-dis.c
+++ b/gnu/usr.bin/binutils/opcodes/i386-dis.c
@@ -100,6 +100,7 @@ static void OP_xcrypt (int, int);
static void OP_SIMD_Suffix (int, int);
static void SIMD_Fixup (int, int);
static void PNI_Fixup (int, int);
+static void XCR_Fixup (int, int);
static void INVLPG_Fixup (int, int);
static void OP_0f38 (int, int);
static void OP_0f3a (int, int);
@@ -1366,7 +1367,7 @@ static const struct dis386 grps[][8] = {
{
{ "sgdtQ", M, XX, XX },
{ "sidtQ", PNI_Fixup, 0, XX, XX },
- { "lgdtQ", M, XX, XX },
+ { "lgdtQ", XCR_Fixup, 0, XX, XX },
{ "lidtQ", M, XX, XX },
{ "smswQ", Ev, XX, XX },
{ "(bad)", XX, XX, XX },
@@ -1434,10 +1435,10 @@ static const struct dis386 grps[][8] = {
{ "fxrstor", Ev, XX, XX },
{ "ldmxcsr", Ev, XX, XX },
{ "stmxcsr", Ev, XX, XX },
- { "(bad)", XX, XX, XX },
- { "lfence", OP_0fae, 0, XX, XX },
- { "mfence", OP_0fae, 0, XX, XX },
- { "clflush", OP_0fae, 0, XX, XX },
+ { "xsave", Ev, XX, XX },
+ { "xrstor", OP_0fae, v_mode, XX, XX },
+ { "xsaveopt", OP_0fae, v_mode, XX, XX },
+ { "clflush", OP_0fae, v_mode, XX, XX },
},
/* GRP14 */
{
@@ -3962,6 +3963,11 @@ OP_0fae (int bytemode, int sizeflag)
{
if (reg == 7)
strcpy (obuf + strlen (obuf) - sizeof ("clflush") + 1, "sfence");
+ else if (reg == 6)
+ strcpy (obuf + strlen (obuf) - sizeof ("xsaveopt") + 1, "mfence");
+ else if (reg == 5)
+ strcpy (obuf + strlen (obuf) - sizeof ("xrstor") + 1, "lfence");
+ bytemode = 0;
if (reg < 5 || rm != 0)
{
@@ -3969,9 +3975,9 @@ OP_0fae (int bytemode, int sizeflag)
return;
}
}
- else if (reg != 7)
+ else if (reg < 5)
{
- BadOp (); /* bad clflush */
+ BadOp (); /* bad sfence, mfence, or lfence */
return;
}
@@ -4181,6 +4187,29 @@ PNI_Fixup (int extrachar ATTRIBUTE_UNUSED, int sizeflag)
}
static void
+XCR_Fixup (int extrachar ATTRIBUTE_UNUSED, int sizeflag)
+{
+ if (mod == 3 && reg == 2 && rm <= 1)
+ {
+ char *p = obuf + strlen (obuf);
+
+ /* Override "lgdt". */
+ if (rm)
+ {
+ strcpy (p - 4, "xsetbv");
+ }
+ else
+ {
+ strcpy (p - 4, "xgetbv");
+ }
+
+ codep++;
+ }
+ else
+ OP_M (0, sizeflag);
+}
+
+static void
INVLPG_Fixup (int bytemode, int sizeflag)
{
if (*codep == 0xf8)