// [AsmJit] // Machine Code Generation for C++. // // [License] // Zlib - See LICENSE.md file in the package. #include #include #include #include "./asmjit.h" using namespace asmjit; // Signature of the generated function. typedef void (*SumIntsFunc)(int* dst, const int* a, const int* b); // This function works with both x86::Assembler and x86::Builder. It shows how // `x86::Emitter` can be used to make your code more generic. static void makeRawFunc(x86::Emitter* emitter) noexcept { // Decide which registers will be mapped to function arguments. Try changing // registers of `dst`, `src_a`, and `src_b` and see what happens in function's // prolog and epilog. x86::Gp dst = emitter->zax(); x86::Gp src_a = emitter->zcx(); x86::Gp src_b = emitter->zdx(); // Decide which vector registers to use. We use these to keep the code generic, // you can switch to any other registers when needed. x86::Xmm vec0 = x86::xmm0; x86::Xmm vec1 = x86::xmm1; // Create and initialize `FuncDetail` and `FuncFrame`. FuncDetail func; func.init(FuncSignatureT(CallConv::kIdHost)); FuncFrame frame; frame.init(func); // Make XMM0 and XMM1 dirty. VEC group includes XMM|YMM|ZMM registers. frame.addDirtyRegs(x86::xmm0, x86::xmm1); FuncArgsAssignment args(&func); // Create arguments assignment context. args.assignAll(dst, src_a, src_b); // Assign our registers to arguments. args.updateFuncFrame(frame); // Reflect our args in FuncFrame. frame.finalize(); // Emit prolog and allocate arguments to registers. emitter->emitProlog(frame); emitter->emitArgsAssignment(frame, args); emitter->movdqu(vec0, x86::ptr(src_a)); // Load 4 ints from [src_a] to XMM0. emitter->movdqu(vec1, x86::ptr(src_b)); // Load 4 ints from [src_b] to XMM1. emitter->paddd(vec0, vec1); // Add 4 ints in XMM1 to XMM0. emitter->movdqu(x86::ptr(dst), vec0); // Store the result to [dst]. // Emit epilog and return. emitter->emitEpilog(frame); } // This function works with x86::Compiler, provided for comparison. static void makeCompiledFunc(x86::Compiler* cc) noexcept { x86::Gp dst = cc->newIntPtr(); x86::Gp src_a = cc->newIntPtr(); x86::Gp src_b = cc->newIntPtr(); x86::Xmm vec0 = cc->newXmm(); x86::Xmm vec1 = cc->newXmm(); cc->addFunc(FuncSignatureT(CallConv::kIdHost)); cc->setArg(0, dst); cc->setArg(1, src_a); cc->setArg(2, src_b); cc->movdqu(vec0, x86::ptr(src_a)); cc->movdqu(vec1, x86::ptr(src_b)); cc->paddd(vec0, vec1); cc->movdqu(x86::ptr(dst), vec0); cc->endFunc(); } static uint32_t testFunc(JitRuntime& rt, uint32_t emitterType) noexcept { FileLogger logger(stdout); CodeHolder code; code.init(rt.codeInfo()); code.setLogger(&logger); Error err = kErrorOk; switch (emitterType) { case BaseEmitter::kTypeAssembler: { printf("Using x86::Assembler:\n"); x86::Assembler a(&code); makeRawFunc(a.as()); break; } case BaseEmitter::kTypeBuilder: { printf("Using x86::Builder:\n"); x86::Builder cb(&code); makeRawFunc(cb.as()); err = cb.finalize(); if (err) { printf("x86::Builder::finalize() failed: %s\n", DebugUtils::errorAsString(err)); return 1; } break; } case BaseEmitter::kTypeCompiler: { printf("Using x86::Compiler:\n"); x86::Compiler cc(&code); makeCompiledFunc(&cc); err = cc.finalize(); if (err) { printf("x86::Compiler::finalize() failed: %s\n", DebugUtils::errorAsString(err)); return 1; } break; } } // Add the code generated to the runtime. SumIntsFunc fn; err = rt.add(&fn, &code); if (err) { printf("JitRuntime::add() failed: %s\n", DebugUtils::errorAsString(err)); return 1; } // Execute the generated function. int inA[4] = { 4, 3, 2, 1 }; int inB[4] = { 1, 5, 2, 8 }; int out[4]; fn(out, inA, inB); // Should print {5 8 4 9}. printf("Result = { %d %d %d %d }\n\n", out[0], out[1], out[2], out[3]); rt.release(fn); return !(out[0] == 5 && out[1] == 8 && out[2] == 4 && out[3] == 9); } int main(int argc, char* argv[]) { ASMJIT_UNUSED(argc); ASMJIT_UNUSED(argv); unsigned nFailed = 0; JitRuntime rt; nFailed += testFunc(rt, BaseEmitter::kTypeAssembler); nFailed += testFunc(rt, BaseEmitter::kTypeBuilder); nFailed += testFunc(rt, BaseEmitter::kTypeCompiler); if (!nFailed) printf("[PASSED] All tests passed\n"); else printf("[FAILED] %u %s failed\n", nFailed, nFailed == 1 ? "test" : "tests"); return nFailed ? 1 : 0; }