2017-08-12 16:48:01 +00:00
|
|
|
/*
|
2020-08-29 13:27:11 +00:00
|
|
|
* Copyright (C) 2009-2018 Apple Inc. All rights reserved.
|
2017-08-12 16:48:01 +00:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
|
|
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
|
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
|
|
|
|
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#if ENABLE(ASSEMBLER)
|
|
|
|
|
|
|
|
#define DUMP_LINK_STATISTICS 0
|
|
|
|
#define DUMP_CODE 0
|
|
|
|
|
|
|
|
#define GLOBAL_THUNK_ID reinterpret_cast<void*>(static_cast<intptr_t>(-1))
|
|
|
|
#define REGEXP_CODE_ID reinterpret_cast<void*>(static_cast<intptr_t>(-2))
|
|
|
|
#define CSS_CODE_ID reinterpret_cast<void*>(static_cast<intptr_t>(-3))
|
|
|
|
|
|
|
|
#include "JITCompilationEffort.h"
|
|
|
|
#include "MacroAssembler.h"
|
2020-08-29 13:27:11 +00:00
|
|
|
#include "MacroAssemblerCodeRef.h"
|
2017-08-12 16:48:01 +00:00
|
|
|
#include <wtf/DataLog.h>
|
|
|
|
#include <wtf/FastMalloc.h>
|
|
|
|
#include <wtf/Noncopyable.h>
|
|
|
|
|
|
|
|
namespace JSC {
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
namespace Wasm {
|
|
|
|
enum class CompilationMode : uint8_t;
|
|
|
|
}
|
|
|
|
|
2017-08-12 16:48:01 +00:00
|
|
|
class CodeBlock;
|
|
|
|
|
|
|
|
// LinkBuffer:
|
|
|
|
//
|
|
|
|
// This class assists in linking code generated by the macro assembler, once code generation
|
|
|
|
// has been completed, and the code has been copied to is final location in memory. At this
|
|
|
|
// time pointers to labels within the code may be resolved, and relative offsets to external
|
|
|
|
// addresses may be fixed.
|
|
|
|
//
|
|
|
|
// Specifically:
|
|
|
|
// * Jump objects may be linked to external targets,
|
|
|
|
// * The address of Jump objects may taken, such that it can later be relinked.
|
|
|
|
// * The return address of a Call may be acquired.
|
|
|
|
// * The address of a Label pointing into the code may be resolved.
|
|
|
|
// * The value referenced by a DataLabel may be set.
|
|
|
|
//
|
|
|
|
class LinkBuffer {
|
|
|
|
WTF_MAKE_NONCOPYABLE(LinkBuffer); WTF_MAKE_FAST_ALLOCATED;
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag> using CodePtr = MacroAssemblerCodePtr<tag>;
|
|
|
|
template<PtrTag tag> using CodeRef = MacroAssemblerCodeRef<tag>;
|
2017-08-12 16:48:01 +00:00
|
|
|
typedef MacroAssembler::Label Label;
|
|
|
|
typedef MacroAssembler::Jump Jump;
|
|
|
|
typedef MacroAssembler::PatchableJump PatchableJump;
|
|
|
|
typedef MacroAssembler::JumpList JumpList;
|
|
|
|
typedef MacroAssembler::Call Call;
|
|
|
|
typedef MacroAssembler::DataLabelCompact DataLabelCompact;
|
|
|
|
typedef MacroAssembler::DataLabel32 DataLabel32;
|
|
|
|
typedef MacroAssembler::DataLabelPtr DataLabelPtr;
|
|
|
|
typedef MacroAssembler::ConvertibleLoadLabel ConvertibleLoadLabel;
|
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
|
|
|
typedef MacroAssembler::LinkRecord LinkRecord;
|
|
|
|
typedef MacroAssembler::JumpLinkType JumpLinkType;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
public:
|
2018-01-03 05:16:05 +00:00
|
|
|
LinkBuffer(MacroAssembler& macroAssembler, void* ownerUID, JITCompilationEffort effort = JITCompilationMustSucceed)
|
2017-08-12 16:48:01 +00:00
|
|
|
: m_size(0)
|
|
|
|
, m_didAllocate(false)
|
|
|
|
#ifndef NDEBUG
|
|
|
|
, m_completed(false)
|
|
|
|
#endif
|
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
UNUSED_PARAM(ownerUID);
|
|
|
|
linkCode(macroAssembler, effort);
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
LinkBuffer(MacroAssembler& macroAssembler, MacroAssemblerCodePtr<tag> code, size_t size, JITCompilationEffort effort = JITCompilationMustSucceed, bool shouldPerformBranchCompaction = true)
|
2017-08-12 16:48:01 +00:00
|
|
|
: m_size(size)
|
|
|
|
, m_didAllocate(false)
|
|
|
|
#ifndef NDEBUG
|
|
|
|
, m_completed(false)
|
|
|
|
#endif
|
2020-08-29 13:27:11 +00:00
|
|
|
, m_code(code.template retagged<LinkBufferPtrTag>())
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
|
|
|
m_shouldPerformBranchCompaction = shouldPerformBranchCompaction;
|
|
|
|
#else
|
|
|
|
UNUSED_PARAM(shouldPerformBranchCompaction);
|
|
|
|
#endif
|
2022-10-23 02:55:20 +00:00
|
|
|
linkCode(macroAssembler, effort);
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
~LinkBuffer()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool didFailToAllocate() const
|
|
|
|
{
|
|
|
|
return !m_didAllocate;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValid() const
|
|
|
|
{
|
|
|
|
return !didFailToAllocate();
|
|
|
|
}
|
2022-10-23 02:55:20 +00:00
|
|
|
|
|
|
|
void setIsJumpIsland()
|
|
|
|
{
|
|
|
|
#if ASSERT_ENABLED
|
|
|
|
m_isJumpIsland = true;
|
|
|
|
#endif
|
|
|
|
}
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
// These methods are used to link or set values at code generation time.
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag, typename Func, typename = std::enable_if_t<std::is_function<typename std::remove_pointer<Func>::type>::value>>
|
|
|
|
void link(Call call, Func funcName)
|
|
|
|
{
|
|
|
|
FunctionPtr<tag> function(funcName);
|
|
|
|
link(call, function);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<PtrTag tag>
|
|
|
|
void link(Call call, FunctionPtr<tag> function)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
ASSERT(call.isFlagSet(Call::Linkable));
|
|
|
|
call.m_label = applyOffset(call.m_label);
|
|
|
|
MacroAssembler::linkCall(code(), call, function);
|
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
void link(Call call, CodeLocationLabel<tag> label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
link(call, FunctionPtr<tag>(label));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
void link(Jump jump, CodeLocationLabel<tag> label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
jump.m_label = applyOffset(jump.m_label);
|
|
|
|
MacroAssembler::linkJump(code(), jump, label);
|
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
void link(const JumpList& list, CodeLocationLabel<tag> label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
for (const Jump& jump : list.jumps())
|
|
|
|
link(jump, label);
|
|
|
|
}
|
|
|
|
|
|
|
|
void patch(DataLabelPtr label, void* value)
|
|
|
|
{
|
|
|
|
AssemblerLabel target = applyOffset(label.m_label);
|
|
|
|
MacroAssembler::linkPointer(code(), target, value);
|
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
void patch(DataLabelPtr label, CodeLocationLabel<tag> value)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
AssemblerLabel target = applyOffset(label.m_label);
|
2020-08-29 13:27:11 +00:00
|
|
|
MacroAssembler::linkPointer(code(), target, value);
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// These methods are used to obtain handles to allow the code to be relinked / repatched later.
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationLabel<tag> entrypoint()
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
return CodeLocationLabel<tag>(tagCodePtr<tag>(code()));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationCall<tag> locationOf(Call call)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
ASSERT(call.isFlagSet(Call::Linkable));
|
|
|
|
ASSERT(!call.isFlagSet(Call::Near));
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationCall<tag>(getLinkerAddress<tag>(applyOffset(call.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationNearCall<tag> locationOfNearCall(Call call)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
|
|
|
ASSERT(call.isFlagSet(Call::Linkable));
|
|
|
|
ASSERT(call.isFlagSet(Call::Near));
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationNearCall<tag>(getLinkerAddress<tag>(applyOffset(call.m_label)),
|
2017-08-12 16:48:01 +00:00
|
|
|
call.isFlagSet(Call::Tail) ? NearCallMode::Tail : NearCallMode::Regular);
|
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationLabel<tag> locationOf(PatchableJump jump)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationLabel<tag>(getLinkerAddress<tag>(applyOffset(jump.m_jump.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationLabel<tag> locationOf(Label label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationLabel<tag>(getLinkerAddress<tag>(applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationDataLabelPtr<tag> locationOf(DataLabelPtr label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationDataLabelPtr<tag>(getLinkerAddress<tag>(applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationDataLabel32<tag> locationOf(DataLabel32 label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationDataLabel32<tag>(getLinkerAddress<tag>(applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationDataLabelCompact<tag> locationOf(DataLabelCompact label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationDataLabelCompact<tag>(getLinkerAddress<tag>(applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeLocationConvertibleLoad<tag> locationOf(ConvertibleLoadLabel label)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return CodeLocationConvertibleLoad<tag>(getLinkerAddress<tag>(applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This method obtains the return address of the call, given as an offset from
|
|
|
|
// the start of the code.
|
|
|
|
unsigned returnAddressOffset(Call call)
|
|
|
|
{
|
|
|
|
call.m_label = applyOffset(call.m_label);
|
|
|
|
return MacroAssembler::getLinkerCallReturnOffset(call);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t offsetOf(Label label)
|
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return applyOffset(label.m_label).offset();
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned offsetOf(PatchableJump jump)
|
|
|
|
{
|
2022-10-23 02:55:20 +00:00
|
|
|
return applyOffset(jump.m_jump.m_label).offset();
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Upon completion of all patching 'FINALIZE_CODE()' should be called once to
|
|
|
|
// complete generation of the code. Alternatively, call
|
|
|
|
// finalizeCodeWithoutDisassembly() directly if you have your own way of
|
|
|
|
// displaying disassembly.
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
template<PtrTag tag>
|
|
|
|
CodeRef<tag> finalizeCodeWithoutDisassembly()
|
|
|
|
{
|
|
|
|
return finalizeCodeWithoutDisassemblyImpl().template retagged<tag>();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<PtrTag tag, typename... Args>
|
|
|
|
CodeRef<tag> finalizeCodeWithDisassembly(bool dumpDisassembly, const char* format, Args... args)
|
2017-08-12 16:48:01 +00:00
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
ALLOW_NONLITERAL_FORMAT_BEGIN
|
|
|
|
IGNORE_WARNINGS_BEGIN("format-security")
|
|
|
|
return finalizeCodeWithDisassemblyImpl(dumpDisassembly, format, args...).template retagged<tag>();
|
|
|
|
IGNORE_WARNINGS_END
|
|
|
|
ALLOW_NONLITERAL_FORMAT_END
|
|
|
|
}
|
|
|
|
|
|
|
|
template<PtrTag tag>
|
|
|
|
CodePtr<tag> trampolineAt(Label label)
|
|
|
|
{
|
|
|
|
return CodePtr<tag>(MacroAssembler::AssemblerType_T::getRelocatedAddress(code(), applyOffset(label.m_label)));
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void* debugAddress()
|
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
return m_code.dataLocation();
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t size() const { return m_size; }
|
|
|
|
|
|
|
|
bool wasAlreadyDisassembled() const { return m_alreadyDisassembled; }
|
|
|
|
void didAlreadyDisassemble() { m_alreadyDisassembled = true; }
|
|
|
|
|
|
|
|
private:
|
2020-08-29 13:27:11 +00:00
|
|
|
JS_EXPORT_PRIVATE CodeRef<LinkBufferPtrTag> finalizeCodeWithoutDisassemblyImpl();
|
|
|
|
JS_EXPORT_PRIVATE CodeRef<LinkBufferPtrTag> finalizeCodeWithDisassemblyImpl(bool dumpDisassembly, const char* format, ...) WTF_ATTRIBUTE_PRINTF(3, 4);
|
|
|
|
|
2017-08-12 16:48:01 +00:00
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
|
|
|
int executableOffsetFor(int location)
|
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
// Returning 0 in this case works because at location <
|
|
|
|
// sizeof(int32_t), no compaction could have happened before this
|
|
|
|
// point as the assembler could not have placed a branch instruction
|
|
|
|
// within this space that required compaction.
|
|
|
|
if (location < static_cast<int>(sizeof(int32_t)))
|
2017-08-12 16:48:01 +00:00
|
|
|
return 0;
|
|
|
|
return bitwise_cast<int32_t*>(m_assemblerStorage.buffer())[location / sizeof(int32_t) - 1];
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
template <typename T> T applyOffset(T src)
|
|
|
|
{
|
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
2022-10-23 02:55:20 +00:00
|
|
|
src = src.labelAtOffset(-executableOffsetFor(src.offset()));
|
2017-08-12 16:48:01 +00:00
|
|
|
#endif
|
|
|
|
return src;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep this private! - the underlying code should only be obtained externally via finalizeCode().
|
|
|
|
void* code()
|
|
|
|
{
|
2020-08-29 13:27:11 +00:00
|
|
|
return m_code.dataLocation();
|
2017-08-12 16:48:01 +00:00
|
|
|
}
|
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
void allocate(MacroAssembler&, JITCompilationEffort);
|
|
|
|
|
|
|
|
template<PtrTag tag, typename T>
|
|
|
|
void* getLinkerAddress(T src)
|
|
|
|
{
|
|
|
|
void *code = this->code();
|
|
|
|
void* address = MacroAssembler::getLinkerAddress<tag>(code, src);
|
|
|
|
RELEASE_ASSERT(code <= untagCodePtr<tag>(address) && untagCodePtr<tag>(address) <= static_cast<char*>(code) + size());
|
|
|
|
return address;
|
|
|
|
}
|
2017-08-12 16:48:01 +00:00
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
JS_EXPORT_PRIVATE void linkCode(MacroAssembler&, JITCompilationEffort);
|
2017-08-12 16:48:01 +00:00
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
|
|
|
template <typename InstructionType>
|
2022-10-23 02:55:20 +00:00
|
|
|
void copyCompactAndLinkCode(MacroAssembler&, JITCompilationEffort);
|
2017-08-12 16:48:01 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
void performFinalization();
|
|
|
|
|
|
|
|
#if DUMP_LINK_STATISTICS
|
|
|
|
static void dumpLinkStatistics(void* code, size_t initialSize, size_t finalSize);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DUMP_CODE
|
|
|
|
static void dumpCode(void* code, size_t);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
RefPtr<ExecutableMemoryHandle> m_executableMemory;
|
|
|
|
size_t m_size;
|
|
|
|
#if ENABLE(BRANCH_COMPACTION)
|
|
|
|
AssemblerData m_assemblerStorage;
|
2022-10-23 02:55:20 +00:00
|
|
|
#if CPU(ARM64E)
|
|
|
|
AssemblerData m_assemblerHashesStorage;
|
|
|
|
#endif
|
2017-08-12 16:48:01 +00:00
|
|
|
bool m_shouldPerformBranchCompaction { true };
|
|
|
|
#endif
|
|
|
|
bool m_didAllocate;
|
|
|
|
#ifndef NDEBUG
|
|
|
|
bool m_completed;
|
2022-10-23 02:55:20 +00:00
|
|
|
#endif
|
|
|
|
#if ASSERT_ENABLED
|
|
|
|
bool m_isJumpIsland { false };
|
2017-08-12 16:48:01 +00:00
|
|
|
#endif
|
|
|
|
bool m_alreadyDisassembled { false };
|
2020-08-29 13:27:11 +00:00
|
|
|
MacroAssemblerCodePtr<LinkBufferPtrTag> m_code;
|
2017-08-12 16:48:01 +00:00
|
|
|
Vector<RefPtr<SharedTask<void(LinkBuffer&)>>> m_linkTasks;
|
|
|
|
};
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
#if OS(LINUX)
|
|
|
|
#define FINALIZE_CODE_IF(condition, linkBufferReference, resultPtrTag, ...) \
|
2017-08-12 16:48:01 +00:00
|
|
|
(UNLIKELY((condition)) \
|
2020-08-29 13:27:11 +00:00
|
|
|
? (linkBufferReference).finalizeCodeWithDisassembly<resultPtrTag>(true, __VA_ARGS__) \
|
|
|
|
: (UNLIKELY(JSC::Options::logJITCodeForPerf()) \
|
|
|
|
? (linkBufferReference).finalizeCodeWithDisassembly<resultPtrTag>(false, __VA_ARGS__) \
|
|
|
|
: (linkBufferReference).finalizeCodeWithoutDisassembly<resultPtrTag>()))
|
|
|
|
#else
|
|
|
|
#define FINALIZE_CODE_IF(condition, linkBufferReference, resultPtrTag, ...) \
|
|
|
|
(UNLIKELY((condition)) \
|
|
|
|
? (linkBufferReference).finalizeCodeWithDisassembly<resultPtrTag>(true, __VA_ARGS__) \
|
|
|
|
: (linkBufferReference).finalizeCodeWithoutDisassembly<resultPtrTag>())
|
|
|
|
#endif
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
bool shouldDumpDisassemblyFor(CodeBlock*);
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
#define FINALIZE_CODE_FOR(codeBlock, linkBufferReference, resultPtrTag, ...) \
|
|
|
|
FINALIZE_CODE_IF((shouldDumpDisassemblyFor(codeBlock) || Options::asyncDisassembly()), linkBufferReference, resultPtrTag, __VA_ARGS__)
|
2017-08-12 16:48:01 +00:00
|
|
|
|
|
|
|
// Use this to finalize code, like so:
|
|
|
|
//
|
2020-08-29 13:27:11 +00:00
|
|
|
// CodeRef code = FINALIZE_CODE(linkBuffer, tag, "my super thingy number %d", number);
|
2017-08-12 16:48:01 +00:00
|
|
|
//
|
|
|
|
// Which, in disassembly mode, will print:
|
|
|
|
//
|
|
|
|
// Generated JIT code for my super thingy number 42:
|
|
|
|
// Code at [0x123456, 0x234567]:
|
|
|
|
// 0x123456: mov $0, 0
|
|
|
|
// 0x12345a: ret
|
|
|
|
//
|
|
|
|
// ... and so on.
|
|
|
|
//
|
2020-08-29 13:27:11 +00:00
|
|
|
// Note that the format string and print arguments are only evaluated when dumpDisassembly
|
2017-08-12 16:48:01 +00:00
|
|
|
// is true, so you can hide expensive disassembly-only computations inside there.
|
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
#define FINALIZE_CODE(linkBufferReference, resultPtrTag, ...) \
|
|
|
|
FINALIZE_CODE_IF((JSC::Options::asyncDisassembly() || JSC::Options::dumpDisassembly()), linkBufferReference, resultPtrTag, __VA_ARGS__)
|
|
|
|
|
|
|
|
#define FINALIZE_DFG_CODE(linkBufferReference, resultPtrTag, ...) \
|
|
|
|
FINALIZE_CODE_IF((JSC::Options::asyncDisassembly() || JSC::Options::dumpDisassembly() || Options::dumpDFGDisassembly()), linkBufferReference, resultPtrTag, __VA_ARGS__)
|
2017-08-12 16:48:01 +00:00
|
|
|
|
2020-08-29 13:27:11 +00:00
|
|
|
#define FINALIZE_REGEXP_CODE(linkBufferReference, resultPtrTag, dataLogFArgumentsForHeading) \
|
|
|
|
FINALIZE_CODE_IF(JSC::Options::asyncDisassembly() || JSC::Options::dumpDisassembly() || Options::dumpRegExpDisassembly(), linkBufferReference, resultPtrTag, dataLogFArgumentsForHeading)
|
2017-08-12 16:48:01 +00:00
|
|
|
|
2022-10-23 02:55:20 +00:00
|
|
|
bool shouldDumpDisassemblyFor(Wasm::CompilationMode);
|
|
|
|
|
|
|
|
#define FINALIZE_WASM_CODE(linkBufferReference, resultPtrTag, ...) \
|
|
|
|
FINALIZE_CODE_IF((JSC::Options::asyncDisassembly() || JSC::Options::dumpDisassembly() || Options::dumpWasmDisassembly()), linkBufferReference, resultPtrTag, __VA_ARGS__)
|
|
|
|
|
|
|
|
#define FINALIZE_WASM_CODE_FOR_MODE(mode, linkBufferReference, resultPtrTag, ...) \
|
|
|
|
FINALIZE_CODE_IF(shouldDumpDisassemblyFor(mode), linkBufferReference, resultPtrTag, __VA_ARGS__)
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-08-12 16:48:01 +00:00
|
|
|
} // namespace JSC
|
|
|
|
|
|
|
|
#endif // ENABLE(ASSEMBLER)
|