mirror of
https://github.com/asmjit/asmjit.git
synced 2025-12-17 04:24:37 +03:00
* Reworked register operands - all vector registers are now
platform::Vec deriving from UniVec (universal vector operand),
additionally, there is no platform::Reg, instead asmjit::Reg
provides all necessary features to make it a base register for
each target architecture
* Reworked casting between registers - now architecture agnostic
names are preferred - use Gp32 instead of Gpd or GpW, Gp64
instead of Gpq and GpX, etc...
* Reworked vector registers and their names - architecture
agnostic naming is now preferred Vec32, Vec64, Vec128, etc...
* Reworked naming conventions used across AsmJit - for clarity
Identifiers are now prefixed with the type, like sectionId(),
labelId(), etc...
* Reworked how Zone and ZoneAllocator are used across AsmJit,
prefering Zone in most cases and ZoneAllocator only for
containers - this change alone achieves around 5% better
performance of Builder and Compiler
* Reworked LabelEntry - decreased the size of the base entry
to 16 bytes for anonymous and unnamed labels. Avoided an
indirection when using labelEntries() - LabelEntry is now
a value and not a pointer
* Renamed LabelLink to Fixup
* Added a new header <asmjit/host.h> which would include
<asmjit/core.h> + target tools for the host architecture,
if enabled and supported
* Added new AArch64 instructions (BTI, CSSC, CHKFEAT)
* Added a mvn_ alternative of mvn instruction (fix for Windows
ARM64 SDK)
* Added more AArch64 CPU features to CpuInfo
* Added better support for Apple CPU detection (Apple M3, M4)
* Added a new benchmarking tool asmjit_bench_overhead, which
benchmarks the overhead of CodeHolder::init()/reset() and
creating/attaching emitters to it. Thanks to the benchmark the
most common code-paths were optimized
* Added a new benchmarking tool asmjit_bench_regalloc, which
aims to benchmark the cost and complexity of register allocation.
* Renamed asmjit_test_perf to asmjit_bench_codegen to make it
clear what is a test and what is a benchmark
109 lines
2.8 KiB
C++
109 lines
2.8 KiB
C++
// This file is part of AsmJit project <https://asmjit.com>
|
|
//
|
|
// See <asmjit/core.h> or LICENSE.md for license and copyright information
|
|
// SPDX-License-Identifier: Zlib
|
|
|
|
#ifndef ASMJIT_TEST_ASSEMBLER_H_INCLUDED
|
|
#define ASMJIT_TEST_ASSEMBLER_H_INCLUDED
|
|
|
|
#include <asmjit/core.h>
|
|
#include <stdio.h>
|
|
|
|
struct TestSettings {
|
|
bool verbose;
|
|
bool validate;
|
|
};
|
|
|
|
template<typename AssemblerType>
|
|
class AssemblerTester {
|
|
public:
|
|
asmjit::Environment env {};
|
|
asmjit::CodeHolder code {};
|
|
AssemblerType assembler {};
|
|
asmjit::Label L0 {};
|
|
const TestSettings& settings;
|
|
|
|
size_t passed {};
|
|
size_t count {};
|
|
|
|
AssemblerTester(asmjit::Arch arch, const TestSettings& settings) noexcept
|
|
: env(arch),
|
|
settings(settings) {
|
|
prepare();
|
|
}
|
|
|
|
void printHeader(const char* archName) noexcept {
|
|
printf("%s assembler tests:\n", archName);
|
|
}
|
|
|
|
void printSummary() noexcept {
|
|
printf(" Passed: %zu / %zu tests\n\n", passed, count);
|
|
}
|
|
|
|
bool didPass() const noexcept { return passed == count; }
|
|
|
|
void prepare() noexcept {
|
|
code.reset();
|
|
code.init(env, 0);
|
|
code.attach(&assembler);
|
|
L0 = assembler.newLabel();
|
|
|
|
if (settings.validate)
|
|
assembler.addDiagnosticOptions(asmjit::DiagnosticOptions::kValidateAssembler);
|
|
}
|
|
|
|
ASMJIT_NOINLINE bool testValidInstruction(const char* s, const char* expectedOpcode, asmjit::Error err = asmjit::kErrorOk) noexcept {
|
|
count++;
|
|
|
|
if (err) {
|
|
printf(" !! %s\n"
|
|
" <%s>\n", s, asmjit::DebugUtils::errorAsString(err));
|
|
prepare();
|
|
return false;
|
|
}
|
|
|
|
asmjit::String encodedOpcode;
|
|
asmjit::Section* text = code.textSection();
|
|
|
|
encodedOpcode.appendHex(text->data(), text->bufferSize());
|
|
if (encodedOpcode != expectedOpcode) {
|
|
printf(" !! [%s] <- %s\n"
|
|
" [%s] (Expected)\n", encodedOpcode.data(), s, expectedOpcode);
|
|
prepare();
|
|
return false;
|
|
}
|
|
|
|
if (settings.verbose)
|
|
printf(" OK [%s] <- %s\n", encodedOpcode.data(), s);
|
|
|
|
passed++;
|
|
prepare();
|
|
return true;
|
|
}
|
|
|
|
ASMJIT_NOINLINE bool testInvalidInstruction(const char* s, asmjit::Error expectedError, asmjit::Error err) noexcept {
|
|
count++;
|
|
|
|
if (err == asmjit::kErrorOk) {
|
|
printf(" !! %s passed, but should have failed with <%s> error\n", s, asmjit::DebugUtils::errorAsString(expectedError));
|
|
prepare();
|
|
return false;
|
|
}
|
|
|
|
if (err != asmjit::kErrorOk) {
|
|
printf(" !! %s failed with <%s>, but should have failed with <%s>\n", s, asmjit::DebugUtils::errorAsString(err), asmjit::DebugUtils::errorAsString(expectedError));
|
|
prepare();
|
|
return false;
|
|
}
|
|
|
|
if (settings.verbose)
|
|
printf(" OK [%s] <- %s\n", asmjit::DebugUtils::errorAsString(err), s);
|
|
|
|
passed++;
|
|
prepare();
|
|
return true;
|
|
}
|
|
};
|
|
|
|
#endif // ASMJIT_TEST_ASSEMBLER_H_INCLUDED
|