mirror of
https://github.com/asmjit/asmjit.git
synced 2025-12-16 20:17:05 +03:00
* Denested src folder to root, renamed testing to asmjit-testing
* Refactored how headers are included into <asmjit/...> form. This
is necessary as compilers would never simplify a path once a ..
appears in include directory - then paths such as ../core/../core
appeared in asserts, which was ugly
* Moved support utilities into asmjit/support/... (still included
by asmjit/core.h for convenience and compatibility)
* Added CMakePresets.json for making it easy to develop AsmJit
* Reworked CMakeLists to be shorter and use CMake option(),
etc... This simplifies it and makes it using more standard
features
* ASMJIT_EMBED now creates asmjit_embed INTERFACE library,
which is accessible via asmjit::asmjit target - this simplifies
embedding and makes it the same as library targets from a CMake
perspective
* Removed ASMJIT_DEPS - this is now provided by cmake target
aliases - 'asmjit::asmjit' so users should not need this variable
* Changed meaning of ASMJIT_LIBS - this now contains only AsmJit
dependencies without asmjit::asmjit target alias. Don't rely on
ASMJIT_LIBS anymore as it's only used internally
* Removed ASMJIT_NO_DEPRECATED option - AsmJit is not going
to provide controllable deprecations in the future
* Removed ASMJIT_NO_VALIDATION in favor of ASMJIT_NO_INTROSPECTION,
which now controls query, features, and validation API presence
* Removed ASMJIT_DIR option - it was never really needed
* Removed AMX_TRANSPOSE feature from instruction database (X86).
Intel has removed it as well, so it's a feature that won't
be siliconized
118 lines
3.0 KiB
C++
118 lines
3.0 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_PERF_H_INCLUDED
|
|
#define ASMJIT_TEST_PERF_H_INCLUDED
|
|
|
|
#include <asmjit/core.h>
|
|
|
|
#include <asmjit-testing/commons/asmjitutils.h>
|
|
#include <asmjit-testing/commons/performancetimer.h>
|
|
|
|
namespace asmjit_perf_utils {
|
|
|
|
class TestErrorHandler : public asmjit::ErrorHandler {
|
|
void handle_error(asmjit::Error err, const char* message, asmjit::BaseEmitter* origin) override {
|
|
(void)err;
|
|
(void)origin;
|
|
printf("ERROR: %s\n", message);
|
|
abort();
|
|
}
|
|
};
|
|
|
|
#ifndef ASMJIT_NO_BUILDER
|
|
template<typename BuilderT, typename FuncT>
|
|
static uint32_t calculate_instruction_count(asmjit::CodeHolder& code, asmjit::Arch arch, const FuncT& func) noexcept {
|
|
BuilderT builder;
|
|
TestErrorHandler eh;
|
|
|
|
asmjit::Environment env(arch);
|
|
code.init(env);
|
|
code.set_error_handler(&eh);
|
|
code.attach(&builder);
|
|
func(builder);
|
|
|
|
uint32_t count = 0;
|
|
asmjit::BaseNode* node = builder.first_node();
|
|
|
|
while (node) {
|
|
count += uint32_t(node->is_inst());
|
|
node = node->next();
|
|
}
|
|
|
|
code.reset();
|
|
return count;
|
|
}
|
|
#endif
|
|
|
|
static inline double calculate_mbps(double duration_us, uint64_t output_size) noexcept {
|
|
if (duration_us == 0)
|
|
return 0.0;
|
|
|
|
double bytes_total = double(output_size);
|
|
return (bytes_total * 1000000) / (duration_us * 1024 * 1024);
|
|
}
|
|
|
|
static inline double calculate_mips(double duration, uint64_t instruction_count) noexcept {
|
|
if (duration == 0)
|
|
return 0.0;
|
|
|
|
return double(instruction_count) * 1000000.0 / (duration * 1e6);
|
|
}
|
|
|
|
template<typename EmitterT, typename FuncT>
|
|
static void bench(asmjit::CodeHolder& code, asmjit::Arch arch, uint32_t num_iterations, const char* test_name, uint32_t instruction_count, const FuncT& func) noexcept {
|
|
EmitterT emitter;
|
|
TestErrorHandler eh;
|
|
|
|
const char* arch_name = asmjit_arch_as_string(arch);
|
|
const char* emitter_name =
|
|
emitter.is_assembler() ? "Assembler" :
|
|
emitter.is_compiler() ? "Compiler" :
|
|
emitter.is_builder() ? "Builder" : "Unknown";
|
|
|
|
uint64_t code_size = 0;
|
|
asmjit::Environment env(arch);
|
|
|
|
PerformanceTimer timer;
|
|
double duration = std::numeric_limits<double>::infinity();
|
|
|
|
code.init(env);
|
|
code.set_error_handler(&eh);
|
|
code.attach(&emitter);
|
|
|
|
for (uint32_t r = 0; r < num_iterations; r++) {
|
|
code_size = 0;
|
|
|
|
timer.start();
|
|
func(emitter);
|
|
code_size += code.code_size();
|
|
|
|
code.reinit();
|
|
timer.stop();
|
|
|
|
duration = asmjit::Support::min(duration, timer.duration() * 1000);
|
|
}
|
|
|
|
printf(" [%-7s] %-9s %-16s | CodeSize:%5llu [B] | Time:%7.3f [us]", arch_name, emitter_name, test_name, (unsigned long long)code_size, duration);
|
|
if (code_size) {
|
|
printf(" | Speed:%7.1f [MiB/s]", calculate_mbps(duration, code_size));
|
|
}
|
|
else {
|
|
printf(" | Speed: N/A ");
|
|
}
|
|
|
|
if (instruction_count) {
|
|
printf(", %8.1f [MInst/s]", calculate_mips(duration, instruction_count));
|
|
}
|
|
|
|
printf("\n");
|
|
code.reset();
|
|
}
|
|
|
|
} // {asmjit_perf_utils}
|
|
|
|
#endif // ASMJIT_TEST_PERF_H_INCLUDED
|