NativeExecutable.cpp 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. /*
  2. * Copyright (c) 2023, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Bytecode/Interpreter.h>
  7. #include <LibJS/JIT/NativeExecutable.h>
  8. #include <LibJS/Runtime/VM.h>
  9. #include <LibX86/Disassembler.h>
  10. #include <sys/mman.h>
  11. namespace JS::JIT {
  12. NativeExecutable::NativeExecutable(void* code, size_t size)
  13. : m_code(code)
  14. , m_size(size)
  15. {
  16. }
  17. NativeExecutable::~NativeExecutable()
  18. {
  19. munmap(m_code, m_size);
  20. }
  21. void NativeExecutable::run(VM& vm) const
  22. {
  23. typedef void (*JITCode)(VM&, Value* registers, Value* locals);
  24. ((JITCode)m_code)(vm,
  25. vm.bytecode_interpreter().registers().data(),
  26. vm.running_execution_context().local_variables.data());
  27. }
  28. void NativeExecutable::dump_disassembly() const
  29. {
  30. #if ARCH(X86_64)
  31. auto const* code_bytes = static_cast<u8 const*>(m_code);
  32. auto stream = X86::SimpleInstructionStream { code_bytes, m_size };
  33. auto disassembler = X86::Disassembler(stream);
  34. while (true) {
  35. auto offset = stream.offset();
  36. auto virtual_offset = bit_cast<size_t>(m_code) + offset;
  37. auto insn = disassembler.next();
  38. if (!insn.has_value())
  39. break;
  40. StringBuilder builder;
  41. builder.appendff("{:p} ", virtual_offset);
  42. auto length = insn.value().length();
  43. for (size_t i = 0; i < 7; i++) {
  44. if (i < length)
  45. builder.appendff("{:02x} ", code_bytes[offset + i]);
  46. else
  47. builder.append(" "sv);
  48. }
  49. builder.append(" "sv);
  50. builder.append(insn.value().to_deprecated_string(virtual_offset, nullptr));
  51. dbgln("{}", builder.string_view());
  52. for (size_t bytes_printed = 7; bytes_printed < length; bytes_printed += 7) {
  53. builder.clear();
  54. builder.appendff("{:p} ", virtual_offset + bytes_printed);
  55. for (size_t i = bytes_printed; i < bytes_printed + 7 && i < length; i++)
  56. builder.appendff(" {:02x}", code_bytes[offset + i]);
  57. dbgln("{}", builder.string_view());
  58. }
  59. }
  60. #endif
  61. }
  62. }