188 lines
5.5 KiB
C++
188 lines
5.5 KiB
C++
// This file is part of AsmJit project <https://asmjit.com>
|
|
//
|
|
// See asmjit.h or LICENSE.md for license and copyright information
|
|
// SPDX-License-Identifier: Zlib
|
|
|
|
#include <asmjit/core.h>
|
|
|
|
#if !defined(ASMJIT_NO_X86)
|
|
#include <asmjit/x86.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
|
|
using namespace asmjit;
|
|
|
|
namespace {
|
|
|
|
#if !defined(ASMJIT_NO_X86)
|
|
static char accessLetter(bool r, bool w) noexcept {
|
|
return r && w ? 'X' : r ? 'R' : w ? 'W' : '_';
|
|
}
|
|
|
|
static void printInfo(Arch arch, const BaseInst& inst, const Operand_* operands, size_t opCount) {
|
|
StringTmp<512> sb;
|
|
|
|
// Read & Write Information
|
|
// ------------------------
|
|
|
|
InstRWInfo rw;
|
|
InstAPI::queryRWInfo(arch, inst, operands, opCount, &rw);
|
|
|
|
#ifndef ASMJIT_NO_LOGGING
|
|
Formatter::formatInstruction(sb, FormatFlags::kNone, nullptr, arch, inst, operands, opCount);
|
|
#else
|
|
sb.append("<Logging-Not-Available>");
|
|
#endif
|
|
sb.append("\n");
|
|
|
|
sb.append(" Operands:\n");
|
|
for (uint32_t i = 0; i < rw.opCount(); i++) {
|
|
const OpRWInfo& op = rw.operand(i);
|
|
|
|
sb.appendFormat(" [%u] Op=%c Read=%016llX Write=%016llX Extend=%016llX",
|
|
i,
|
|
accessLetter(op.isRead(), op.isWrite()),
|
|
op.readByteMask(),
|
|
op.writeByteMask(),
|
|
op.extendByteMask());
|
|
|
|
if (op.isMemBaseUsed()) {
|
|
sb.appendFormat(" Base=%c", accessLetter(op.isMemBaseRead(), op.isMemBaseWrite()));
|
|
if (op.isMemBasePreModify())
|
|
sb.appendFormat(" <PRE>");
|
|
if (op.isMemBasePostModify())
|
|
sb.appendFormat(" <POST>");
|
|
}
|
|
|
|
if (op.isMemIndexUsed()) {
|
|
sb.appendFormat(" Index=%c", accessLetter(op.isMemIndexRead(), op.isMemIndexWrite()));
|
|
}
|
|
|
|
sb.append("\n");
|
|
}
|
|
|
|
// CPU Flags (Read/Write)
|
|
// ----------------------
|
|
|
|
if ((rw.readFlags() | rw.writeFlags()) != CpuRWFlags::kNone) {
|
|
sb.append(" Flags: \n");
|
|
|
|
struct FlagMap {
|
|
CpuRWFlags flag;
|
|
char name[4];
|
|
};
|
|
|
|
static const FlagMap flagMap[] = {
|
|
{ CpuRWFlags::kX86_CF, "CF" },
|
|
{ CpuRWFlags::kX86_OF, "OF" },
|
|
{ CpuRWFlags::kX86_SF, "SF" },
|
|
{ CpuRWFlags::kX86_ZF, "ZF" },
|
|
{ CpuRWFlags::kX86_AF, "AF" },
|
|
{ CpuRWFlags::kX86_PF, "PF" },
|
|
{ CpuRWFlags::kX86_DF, "DF" },
|
|
{ CpuRWFlags::kX86_IF, "IF" },
|
|
{ CpuRWFlags::kX86_AC, "AC" },
|
|
{ CpuRWFlags::kX86_C0, "C0" },
|
|
{ CpuRWFlags::kX86_C1, "C1" },
|
|
{ CpuRWFlags::kX86_C2, "C2" },
|
|
{ CpuRWFlags::kX86_C3, "C3" }
|
|
};
|
|
|
|
sb.append(" ");
|
|
for (uint32_t f = 0; f < 13; f++) {
|
|
char c = accessLetter((rw.readFlags() & flagMap[f].flag) != CpuRWFlags::kNone,
|
|
(rw.writeFlags() & flagMap[f].flag) != CpuRWFlags::kNone);
|
|
if (c != '_')
|
|
sb.appendFormat("%s=%c ", flagMap[f].name, c);
|
|
}
|
|
|
|
sb.append("\n");
|
|
}
|
|
|
|
// CPU Features
|
|
// ------------
|
|
|
|
CpuFeatures features;
|
|
InstAPI::queryFeatures(arch, inst, operands, opCount, &features);
|
|
|
|
#ifndef ASMJIT_NO_LOGGING
|
|
if (!features.empty()) {
|
|
sb.append(" Features:\n");
|
|
sb.append(" ");
|
|
|
|
bool first = true;
|
|
CpuFeatures::Iterator it(features.iterator());
|
|
while (it.hasNext()) {
|
|
uint32_t featureId = uint32_t(it.next());
|
|
if (!first)
|
|
sb.append(" & ");
|
|
Formatter::formatFeature(sb, arch, featureId);
|
|
first = false;
|
|
}
|
|
sb.append("\n");
|
|
}
|
|
#endif
|
|
|
|
printf("%s\n", sb.data());
|
|
}
|
|
|
|
template<typename... Args>
|
|
static void printInfoSimple(Arch arch,InstId instId, InstOptions options, Args&&... args) {
|
|
BaseInst inst(instId);
|
|
inst.addOptions(options);
|
|
Operand_ opArray[] = { std::forward<Args>(args)... };
|
|
printInfo(arch, inst, opArray, sizeof...(args));
|
|
}
|
|
|
|
template<typename... Args>
|
|
static void printInfoExtra(Arch arch, InstId instId, InstOptions options, const BaseReg& extraReg, Args&&... args) {
|
|
BaseInst inst(instId);
|
|
inst.addOptions(options);
|
|
inst.setExtraReg(extraReg);
|
|
Operand_ opArray[] = { std::forward<Args>(args)... };
|
|
printInfo(arch, inst, opArray, sizeof...(args));
|
|
}
|
|
#endif // !ASMJIT_NO_X86
|
|
|
|
static void testX86Arch() {
|
|
#if !defined(ASMJIT_NO_X86)
|
|
using namespace x86;
|
|
Arch arch = Arch::kX64;
|
|
|
|
printInfoSimple(arch, Inst::kIdAdd, InstOptions::kNone, eax, ebx);
|
|
printInfoSimple(arch, Inst::kIdLods, InstOptions::kNone, eax, dword_ptr(rsi));
|
|
|
|
printInfoSimple(arch, Inst::kIdPshufd, InstOptions::kNone, xmm0, xmm1, imm(0));
|
|
printInfoSimple(arch, Inst::kIdPabsb, InstOptions::kNone, mm1, mm2);
|
|
printInfoSimple(arch, Inst::kIdPabsb, InstOptions::kNone, xmm1, xmm2);
|
|
printInfoSimple(arch, Inst::kIdPextrw, InstOptions::kNone, eax, mm1, imm(0));
|
|
printInfoSimple(arch, Inst::kIdPextrw, InstOptions::kNone, eax, xmm1, imm(0));
|
|
printInfoSimple(arch, Inst::kIdPextrw, InstOptions::kNone, ptr(rax), xmm1, imm(0));
|
|
|
|
printInfoSimple(arch, Inst::kIdVpdpbusd, InstOptions::kNone, xmm0, xmm1, xmm2);
|
|
printInfoSimple(arch, Inst::kIdVpdpbusd, InstOptions::kX86_Vex, xmm0, xmm1, xmm2);
|
|
|
|
printInfoSimple(arch, Inst::kIdVaddpd, InstOptions::kNone, ymm0, ymm1, ymm2);
|
|
printInfoSimple(arch, Inst::kIdVaddpd, InstOptions::kNone, ymm0, ymm30, ymm31);
|
|
printInfoSimple(arch, Inst::kIdVaddpd, InstOptions::kNone, zmm0, zmm1, zmm2);
|
|
|
|
printInfoExtra(arch, Inst::kIdVaddpd, InstOptions::kNone, k1, zmm0, zmm1, zmm2);
|
|
printInfoExtra(arch, Inst::kIdVaddpd, InstOptions::kX86_ZMask, k1, zmm0, zmm1, zmm2);
|
|
#endif // !ASMJIT_NO_X86
|
|
}
|
|
|
|
} // {anonymous}
|
|
|
|
int main() {
|
|
printf("AsmJit Instruction Info Test-Suite v%u.%u.%u\n",
|
|
unsigned((ASMJIT_LIBRARY_VERSION >> 16) ),
|
|
unsigned((ASMJIT_LIBRARY_VERSION >> 8) & 0xFF),
|
|
unsigned((ASMJIT_LIBRARY_VERSION ) & 0xFF));
|
|
printf("\n");
|
|
|
|
testX86Arch();
|
|
|
|
return 0;
|
|
}
|