2020-08-25 11:23:25 +00:00
|
|
|
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
|
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
|
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
|
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
|
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
|
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
2020-08-25 11:23:25 +00:00
|
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
|
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
|
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
|
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
|
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
|
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
|
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
|
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
2020-08-25 11:23:25 +00:00
|
|
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
2020-12-27 15:02:35 +00:00
|
|
|
|
#include "libc/bits/safemacros.h"
|
2020-12-09 23:04:54 +00:00
|
|
|
|
#include "libc/fmt/conv.h"
|
2020-08-25 11:23:25 +00:00
|
|
|
|
#include "libc/macros.h"
|
|
|
|
|
#include "libc/runtime/runtime.h"
|
|
|
|
|
#include "libc/stdio/stdio.h"
|
|
|
|
|
#include "libc/str/str.h"
|
|
|
|
|
#include "libc/sysv/consts/ex.h"
|
|
|
|
|
#include "libc/sysv/consts/exit.h"
|
|
|
|
|
#include "third_party/getopt/getopt.h"
|
|
|
|
|
|
|
|
|
|
#define USAGE \
|
|
|
|
|
" [FLAGS] OPERAND..\n\
|
|
|
|
|
Decodes geek operand notation used by ref.x86asm.net\n\
|
|
|
|
|
\n\
|
|
|
|
|
Flags:\n\
|
|
|
|
|
-s succinct mode\n\
|
|
|
|
|
-b BITS sets 16/32/64 bit mode [default 64]\n\
|
|
|
|
|
-? shows this information\n\
|
|
|
|
|
\n\
|
|
|
|
|
Examples:\n\
|
|
|
|
|
o/tool/cpu/help-operand -b64 Evqp\n\
|
|
|
|
|
\n"
|
|
|
|
|
|
|
|
|
|
int bits_;
|
|
|
|
|
bool succinct_;
|
|
|
|
|
|
|
|
|
|
void PrintUsage(int rc, FILE *f) {
|
|
|
|
|
fputs("Usage: ", f);
|
|
|
|
|
fputs(program_invocation_name, f);
|
|
|
|
|
fputs(USAGE, f);
|
|
|
|
|
exit(rc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void GetOpts(int argc, char *argv[]) {
|
|
|
|
|
int opt;
|
|
|
|
|
bits_ = 64;
|
|
|
|
|
while ((opt = getopt(argc, argv, "?hbs")) != -1) {
|
|
|
|
|
switch (opt) {
|
|
|
|
|
case 's':
|
|
|
|
|
succinct_ = true;
|
|
|
|
|
break;
|
|
|
|
|
case 'b':
|
|
|
|
|
bits_ = atoi(optarg);
|
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
case 'h':
|
|
|
|
|
PrintUsage(EXIT_SUCCESS, stdout);
|
|
|
|
|
default:
|
|
|
|
|
PrintUsage(EX_USAGE, stderr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const struct Descriptors {
|
|
|
|
|
const char prefix[8];
|
|
|
|
|
const char *succinct;
|
|
|
|
|
const char *description;
|
|
|
|
|
} kDescriptors[] = {
|
|
|
|
|
|
|
|
|
|
{"AL", "AL", "AL register"},
|
|
|
|
|
{"CS", "CS", "CS register (code segment)"},
|
|
|
|
|
|
|
|
|
|
{"va", "?",
|
|
|
|
|
"Word or doubleword, according to asz address-size attribute (only REP "
|
|
|
|
|
"and LOOP families)."},
|
|
|
|
|
{"dqa", "?",
|
|
|
|
|
"Doubleword or quadword, according to asz address-size attribute (only "
|
|
|
|
|
"REP and LOOP families)."},
|
|
|
|
|
{"wa", "?",
|
|
|
|
|
"Word, according to asz address-size attribute (only JCXZ instruction)."},
|
|
|
|
|
{"wo", "?",
|
|
|
|
|
"Word, according to current operand size (e. g., MOVSW instruction)."},
|
|
|
|
|
{"ws", "?",
|
|
|
|
|
"Word, according to current stack size (only PUSHF and POPF instructions "
|
|
|
|
|
"in 64-bit mode)."},
|
|
|
|
|
{"da", "?",
|
|
|
|
|
"Doubleword, according to asz address-size attribute (only JECXZ "
|
|
|
|
|
"instruction)."},
|
|
|
|
|
{"do", "?",
|
|
|
|
|
"Doubleword, according to current osz operand size (e. g., MOVSD "
|
|
|
|
|
"instruction)."},
|
|
|
|
|
{"qa", "?",
|
|
|
|
|
"Quadword, according to asz address-size attribute (only JRCXZ "
|
|
|
|
|
"instruction)."},
|
|
|
|
|
{"qs", "64/16",
|
|
|
|
|
"Quadword, according to current stack size via osz operand-size (only "
|
|
|
|
|
"PUSHFQ and POPFQ instructions)."},
|
|
|
|
|
|
|
|
|
|
{"va", "16/32",
|
|
|
|
|
"Word or doubleword sign extended to the size of the stack pointer (for "
|
|
|
|
|
"example, PUSH (68))."},
|
|
|
|
|
|
|
|
|
|
{"vqp", "16/32/64",
|
|
|
|
|
"Word or doubleword, depending on operand-size attribute, or "
|
|
|
|
|
"quadword, promoted by REX.W in 64-bit mode."},
|
|
|
|
|
{"vs", "16/32",
|
|
|
|
|
"Word or doubleword sign extended to the size of the stack pointer (for "
|
|
|
|
|
"example, PUSH (68))."},
|
|
|
|
|
{"vds", "16/32",
|
|
|
|
|
"Word or doubleword, depending on operand-size attribute, or doubleword, "
|
|
|
|
|
"sign-extended to 64 bits for 64-bit operand size."},
|
|
|
|
|
{"vq", "64/16",
|
|
|
|
|
"Quadword (default) or word if operand-size prefix is used (for "
|
|
|
|
|
"example, PUSH (50))."},
|
|
|
|
|
|
|
|
|
|
{"EST", "STi",
|
|
|
|
|
"A ModR/M byte follows the opcode and specifies the x87 FPU stack "
|
|
|
|
|
"register."},
|
|
|
|
|
{"ES", "STi/m",
|
|
|
|
|
"A ModR/M byte follows the opcode and specifies the operand. The "
|
|
|
|
|
"operand is either a x87 FPU stack register or a memory address. If "
|
|
|
|
|
"it is a memory address, the address is computed from a segment "
|
|
|
|
|
"register and any of the following values: a base register, an "
|
|
|
|
|
"index register, a scaling factor, or a displacement."},
|
|
|
|
|
|
|
|
|
|
{"SC", "",
|
|
|
|
|
"Stack operand, used by instructions which either push an operand to the "
|
|
|
|
|
"stack or pop an operand from the stack. Pop-like instructions are, for "
|
|
|
|
|
"example, POP, RET, IRET, LEAVE. Push-like are, for example, PUSH, CALL, "
|
|
|
|
|
"INT. No Operand type is provided along with this method because it "
|
|
|
|
|
"depends on source/destination operand(s)."},
|
|
|
|
|
|
|
|
|
|
{"BA", "m",
|
|
|
|
|
"Memory addressed by DS:EAX, or by rAX in 64-bit mode (only 0F01C8 "
|
|
|
|
|
"MONITOR)."},
|
|
|
|
|
{"BB", "m",
|
|
|
|
|
"Memory addressed by DS:eBX+AL, or by rBX+AL in 64-bit mode (only XLAT)."},
|
|
|
|
|
{"BD", "m",
|
|
|
|
|
"Memory addressed by DS:eDI or by RDI (only 0FF7 MASKMOVQ and 660FF7 "
|
|
|
|
|
"MASKMOVDQU)"},
|
|
|
|
|
|
|
|
|
|
{"A", "ptr",
|
|
|
|
|
"Direct address. The instruction has no ModR/M byte; the address of the "
|
|
|
|
|
"operand is encoded in the instruction; no base register, index register, "
|
|
|
|
|
"or scaling factor can be applied (for example, far JMP (EA))."},
|
|
|
|
|
{"C", "CRn",
|
|
|
|
|
"The reg field of the ModR/M byte selects a control register (only MOV "
|
|
|
|
|
"(0F20, 0F22))."},
|
|
|
|
|
{"D", "DRn",
|
|
|
|
|
"The reg field of the ModR/M byte selects a debug register (only MOV "
|
|
|
|
|
"(0F21, 0F23))."},
|
|
|
|
|
|
|
|
|
|
{"I", "imm",
|
|
|
|
|
"Immediate data. The operand value is encoded in subsequent bytes of the "
|
|
|
|
|
"instruction."},
|
|
|
|
|
|
|
|
|
|
{"J", "rel",
|
|
|
|
|
"The instruction contains a relative offset to be added to the "
|
|
|
|
|
"instruction pointer register (for example, JMP (E9), LOOP))."},
|
|
|
|
|
|
|
|
|
|
{"E", "r/m",
|
|
|
|
|
"A ModR/M byte follows the opcode and specifies the operand. The "
|
|
|
|
|
"operand is either a general-purpose register or a memory address. "
|
|
|
|
|
"If it is a memory address, the address is computed from a segment "
|
|
|
|
|
"register and any of the following values: a base register, an index "
|
|
|
|
|
"register, a scaling factor, or a displacement."},
|
|
|
|
|
|
|
|
|
|
{"G", "r",
|
|
|
|
|
"The reg field of the ModR/M byte selects a general register (for "
|
|
|
|
|
"example, AX (000))."},
|
|
|
|
|
|
|
|
|
|
{"H", "r",
|
|
|
|
|
"The r/m field of the ModR/M byte always selects a general register, "
|
|
|
|
|
"regardless of the mod field (for example, MOV (0F20))."},
|
|
|
|
|
|
|
|
|
|
{"M", "rm",
|
|
|
|
|
"The ModR/M byte may refer only to memory: mod != 11bin (BOUND, LEA, "
|
|
|
|
|
"CALLF, JMPF, LES, LDS, LSS, LFS, LGS, CMPXCHG8B, CMPXCHG16B, F20FF0 "
|
|
|
|
|
"LDDQU)."},
|
|
|
|
|
|
|
|
|
|
{"N", "mm",
|
|
|
|
|
"The R/M field of the ModR/M byte selects a packed quadword MMX "
|
|
|
|
|
"technology register."},
|
|
|
|
|
|
|
|
|
|
{"O", "moffs",
|
|
|
|
|
"The instruction has no ModR/M byte; the offset of the operand is coded "
|
|
|
|
|
"as a word, double word or quad word (depending on address size "
|
|
|
|
|
"attribute) in the instruction. No base register, index register, or "
|
|
|
|
|
"scaling factor can be applied (only MOV (A0, A1, A2, A3))."},
|
|
|
|
|
|
|
|
|
|
{"P", "mm",
|
|
|
|
|
"The reg field of the ModR/M byte selects a packed quadword MMX "
|
|
|
|
|
"technology register."},
|
|
|
|
|
|
|
|
|
|
{"Q", "mm/m64",
|
|
|
|
|
"A ModR/M byte follows the opcode and specifies the operand. The "
|
|
|
|
|
"operand is either an MMX technology register or a memory address. "
|
|
|
|
|
"If it is a memory address, the address is computed from a segment "
|
|
|
|
|
"register and any of the following values: a base register, an index "
|
|
|
|
|
"register, a scaling factor, and a displacement."},
|
|
|
|
|
|
|
|
|
|
{"R", "r",
|
|
|
|
|
"The mod field of the ModR/M byte may refer only to a general "
|
|
|
|
|
"register (only MOV (0F20-0F24, 0F26))."},
|
|
|
|
|
|
|
|
|
|
{"S", "Sreg",
|
|
|
|
|
"The reg field of the ModR/M byte selects a segment register (only MOV "
|
|
|
|
|
"(8C, 8E))."},
|
|
|
|
|
|
|
|
|
|
{"T", "TRn",
|
|
|
|
|
"The reg field of the ModR/M byte selects a test register (only MOV "
|
|
|
|
|
"(0F24, 0F26))."},
|
|
|
|
|
|
|
|
|
|
{"U", "xmm",
|
|
|
|
|
"The R/M field of the ModR/M byte selects a 128-bit XMM register."},
|
|
|
|
|
|
|
|
|
|
{"sr", "32real",
|
|
|
|
|
"Single-real. Only x87 FPU instructions (for example, FADD)."},
|
|
|
|
|
{"dr", "64real",
|
|
|
|
|
"Double-real. Only x87 FPU instructions (for example, FADD)."},
|
|
|
|
|
{"er", "80real",
|
|
|
|
|
"Extended-real. Only x87 FPU instructions (for example, FLD)."},
|
|
|
|
|
{"e", "14/28", "x87 FPU environment (for example, FSTENV)."},
|
|
|
|
|
|
|
|
|
|
{"V", "xmm",
|
|
|
|
|
"The reg field of the ModR/M byte selects a 128-bit XMM register."},
|
|
|
|
|
|
|
|
|
|
{"W", "xmm/m",
|
|
|
|
|
"A ModR/M byte follows the opcode and specifies the operand. The operand "
|
|
|
|
|
"is either a 128-bit XMM register or a memory address. If it is a memory "
|
|
|
|
|
"address, the address is computed from a segment register and any of the "
|
|
|
|
|
"following values: a base register, an index register, a scaling factor, "
|
|
|
|
|
"and a displacement"},
|
|
|
|
|
|
|
|
|
|
{"X", "m",
|
|
|
|
|
"Memory addressed by the DS:eSI or by RSI (only MOVS, CMPS, OUTS, "
|
|
|
|
|
"and LODS). In 64-bit mode, only 64-bit (RSI) and 32-bit (ESI) "
|
|
|
|
|
"address sizes are supported. In non-64-bit modes, only 32-bit (ESI) "
|
|
|
|
|
"and 16-bit (SI) address sizes are supported."},
|
|
|
|
|
|
|
|
|
|
{"Y", "m",
|
|
|
|
|
"Memory addressed by the ES:eDI or by RDI (only MOVS, CMPS, INS, "
|
|
|
|
|
"STOS, and SCAS). In 64-bit mode, only 64-bit (RDI) and 32-bit (EDI) "
|
|
|
|
|
"address sizes are supported. In non-64-bit modes, only 32-bit (EDI) "
|
|
|
|
|
"and 16-bit (DI) address sizes are supported. The implicit ES "
|
|
|
|
|
"segment register cannot be overriden by a segment prefix."},
|
|
|
|
|
|
|
|
|
|
{"Z", "r",
|
|
|
|
|
"The instruction has no ModR/M byte; the three least-significant "
|
|
|
|
|
"bits of the opcode byte selects a general-purpose register."},
|
|
|
|
|
|
|
|
|
|
{"F", "-", "rFLAGS register."},
|
|
|
|
|
|
|
|
|
|
{"si", "32real",
|
|
|
|
|
"Doubleword integer register (e. g., eax). (unused even by Intel?)"},
|
|
|
|
|
{"ss", "-",
|
|
|
|
|
"Scalar element of a 128-bit packed single-precision floating data."},
|
|
|
|
|
{"stx", "512", "x87 FPU and SIMD state (FXSAVE and FXRSTOR)."},
|
|
|
|
|
{"st", "94/108", "x87 FPU state (for example, FSAVE)."},
|
|
|
|
|
{"s", "-",
|
|
|
|
|
"6-byte pseudo-descriptor, or 10-byte pseudo-descriptor in 64-bit mode "
|
|
|
|
|
"(for example, SGDT)."},
|
|
|
|
|
|
|
|
|
|
{"bcd", "80dec",
|
|
|
|
|
"Packed-BCD. Only x87 FPU instructions (for example, FBLD)."},
|
|
|
|
|
{"bsq", "", "(Byte, sign-extended to 64 bits.)"},
|
|
|
|
|
{"bss", "8",
|
|
|
|
|
"Byte, sign-extended to the size of the stack pointer (for example, PUSH "
|
|
|
|
|
"(6A))."},
|
|
|
|
|
{"bs", "8", "Byte, sign-extended to the size of the destination operand."},
|
|
|
|
|
|
|
|
|
|
{"a", "16/32&16/32",
|
|
|
|
|
"Two one-word operands in memory or two double-word operands in memory, "
|
|
|
|
|
"depending on operand-size attribute (only BOUND)."},
|
|
|
|
|
{"b", "8", "Byte, regardless of operand-size attribute."},
|
|
|
|
|
{"c", "?",
|
|
|
|
|
"Byte or word, depending on operand-size attribute. (unused even by "
|
|
|
|
|
"Intel?)"},
|
|
|
|
|
{"dqp", "32/64",
|
|
|
|
|
"Doubleword, or quadword, promoted by REX.W in 64-bit mode (for example, "
|
|
|
|
|
"MOVSXD)."},
|
|
|
|
|
{"dq", "128",
|
|
|
|
|
"Double-quadword, regardless of operand-size attribute (for example, "
|
|
|
|
|
"CMPXCHG16B)."},
|
|
|
|
|
{"ds", "32",
|
|
|
|
|
"Doubleword, sign-extended to 64 bits (for example, CALL (E8)."},
|
|
|
|
|
{"di", "32int",
|
|
|
|
|
"Doubleword-integer. Only x87 FPU instructions (for example, FIADD)."},
|
|
|
|
|
{"d", "32", "Doubleword, regardless of operand-size attribute."},
|
|
|
|
|
{"qi", "64int",
|
|
|
|
|
"Qword-integer. Only x87 FPU instructions (for example, FILD)."},
|
|
|
|
|
{"qp", "64", "Quadword, promoted by REX.W (for example, IRETQ)."},
|
|
|
|
|
{"q", "64",
|
|
|
|
|
"Quadword, regardless of operand-size attribute (for example, CALL (FF "
|
|
|
|
|
"/2))."},
|
|
|
|
|
|
|
|
|
|
{"v", "16/32",
|
|
|
|
|
"Word or doubleword, depending on operand-size attribute (for example, "
|
|
|
|
|
"INC (40), PUSH (50))."},
|
|
|
|
|
|
|
|
|
|
{"wi", "16int",
|
|
|
|
|
"Word-integer. Only x87 FPU instructions (for example, FIADD)."},
|
|
|
|
|
{"w", "16",
|
|
|
|
|
"Word, regardless of operand-size attribute (for example, ENTER)."},
|
|
|
|
|
|
|
|
|
|
{"pi", "-", "Quadword MMX technology data."},
|
|
|
|
|
{"psq", "-", "64-bit packed single-precision floating-point data."},
|
|
|
|
|
{"pd", "-", "128-bit packed double-precision floating-point data."},
|
|
|
|
|
{"ps", "-", "128-bit packed single-precision floating-point data."},
|
|
|
|
|
{"ptp", "16:16/32/64",
|
|
|
|
|
"32-bit or 48-bit pointer, depending on operand-size attribute, or 80-bit "
|
|
|
|
|
"far pointer, promoted by REX.W in 64-bit mode (for example, "
|
|
|
|
|
"CALLF (FF /3))."},
|
|
|
|
|
{"p", "16:16/32",
|
|
|
|
|
"32-bit or 48-bit pointer, depending on operand-size attribute (for "
|
|
|
|
|
"example, CALLF (9A)."},
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void HandleOperand(const char *op) {
|
|
|
|
|
int i;
|
|
|
|
|
bool found;
|
|
|
|
|
while (*op) {
|
|
|
|
|
found = false;
|
|
|
|
|
for (i = 0; i < ARRAYLEN(kDescriptors); ++i) {
|
|
|
|
|
if (startswith(op, kDescriptors[i].prefix)) {
|
|
|
|
|
found = true;
|
|
|
|
|
op += strlen(kDescriptors[i].prefix);
|
|
|
|
|
if (succinct_) {
|
|
|
|
|
printf("%s ", kDescriptors[i].succinct);
|
|
|
|
|
} else if (!isempty(kDescriptors[i].succinct) &&
|
|
|
|
|
strcmp(kDescriptors[i].succinct, "-") != 0 &&
|
|
|
|
|
strcmp(kDescriptors[i].succinct, "?") != 0) {
|
|
|
|
|
printf("%s (%s): %s\n", kDescriptors[i].prefix,
|
|
|
|
|
kDescriptors[i].succinct, kDescriptors[i].description);
|
|
|
|
|
} else {
|
|
|
|
|
printf("%s: %s\n", kDescriptors[i].prefix,
|
|
|
|
|
kDescriptors[i].description);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!found) {
|
|
|
|
|
printf("%c?", *op);
|
|
|
|
|
if (succinct_) {
|
|
|
|
|
printf(" ");
|
|
|
|
|
} else {
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
|
|
|
|
op++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
|
int i;
|
|
|
|
|
GetOpts(argc, argv);
|
|
|
|
|
for (i = optind; i < argc; ++i) {
|
|
|
|
|
HandleOperand(argv[i]);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|