2 * Copyright (C) 2012, 2013 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "JITThunks.h"
31 #include "Executable.h"
34 #include "JSCInlines.h"
38 JITThunks::JITThunks()
39 : m_hostFunctionStubMap(std::make_unique
<HostFunctionStubMap
>())
43 JITThunks::~JITThunks()
47 MacroAssemblerCodePtr
JITThunks::ctiNativeCall(VM
* vm
)
50 return MacroAssemblerCodePtr::createLLIntCodePtr(llint_native_call_trampoline
);
51 return ctiStub(vm
, nativeCallGenerator
).code();
54 MacroAssemblerCodePtr
JITThunks::ctiNativeConstruct(VM
* vm
)
57 return MacroAssemblerCodePtr::createLLIntCodePtr(llint_native_construct_trampoline
);
58 return ctiStub(vm
, nativeConstructGenerator
).code();
61 MacroAssemblerCodePtr
JITThunks::ctiNativeTailCall(VM
* vm
)
63 ASSERT(vm
->canUseJIT());
64 return ctiStub(vm
, nativeTailCallGenerator
).code();
67 MacroAssemblerCodeRef
JITThunks::ctiStub(VM
* vm
, ThunkGenerator generator
)
69 Locker
locker(m_lock
);
70 CTIStubMap::AddResult entry
= m_ctiStubMap
.add(generator
, MacroAssemblerCodeRef());
71 if (entry
.isNewEntry
) {
72 // Compilation thread can only retrieve existing entries.
73 ASSERT(!isCompilationThread());
74 entry
.iterator
->value
= generator(vm
);
76 return entry
.iterator
->value
;
79 void JITThunks::finalize(Handle
<Unknown
> handle
, void*)
81 auto* nativeExecutable
= jsCast
<NativeExecutable
*>(handle
.get().asCell());
82 weakRemove(*m_hostFunctionStubMap
, std::make_pair(nativeExecutable
->function(), nativeExecutable
->constructor()), nativeExecutable
);
85 NativeExecutable
* JITThunks::hostFunctionStub(VM
* vm
, NativeFunction function
, NativeFunction constructor
)
87 ASSERT(!isCompilationThread());
89 if (NativeExecutable
* nativeExecutable
= m_hostFunctionStubMap
->get(std::make_pair(function
, constructor
)))
90 return nativeExecutable
;
92 NativeExecutable
* nativeExecutable
= NativeExecutable::create(
94 adoptRef(new NativeJITCode(JIT::compileCTINativeCall(vm
, function
), JITCode::HostCallThunk
)),
96 adoptRef(new NativeJITCode(MacroAssemblerCodeRef::createSelfManagedCodeRef(ctiNativeConstruct(vm
)), JITCode::HostCallThunk
)),
97 constructor
, NoIntrinsic
);
98 weakAdd(*m_hostFunctionStubMap
, std::make_pair(function
, constructor
), Weak
<NativeExecutable
>(nativeExecutable
, this));
99 return nativeExecutable
;
102 NativeExecutable
* JITThunks::hostFunctionStub(VM
* vm
, NativeFunction function
, ThunkGenerator generator
, Intrinsic intrinsic
)
104 ASSERT(!isCompilationThread());
105 ASSERT(vm
->canUseJIT());
107 if (NativeExecutable
* nativeExecutable
= m_hostFunctionStubMap
->get(std::make_pair(function
, &callHostFunctionAsConstructor
)))
108 return nativeExecutable
;
110 RefPtr
<JITCode
> forCall
;
112 MacroAssemblerCodeRef entry
= generator(vm
);
113 forCall
= adoptRef(new DirectJITCode(entry
, entry
.code(), JITCode::HostCallThunk
));
115 forCall
= adoptRef(new NativeJITCode(JIT::compileCTINativeCall(vm
, function
), JITCode::HostCallThunk
));
117 RefPtr
<JITCode
> forConstruct
= adoptRef(new NativeJITCode(MacroAssemblerCodeRef::createSelfManagedCodeRef(ctiNativeConstruct(vm
)), JITCode::HostCallThunk
));
119 NativeExecutable
* nativeExecutable
= NativeExecutable::create(*vm
, forCall
, function
, forConstruct
, callHostFunctionAsConstructor
, intrinsic
);
120 weakAdd(*m_hostFunctionStubMap
, std::make_pair(function
, &callHostFunctionAsConstructor
), Weak
<NativeExecutable
>(nativeExecutable
, this));
121 return nativeExecutable
;
124 void JITThunks::clearHostFunctionStubs()
126 m_hostFunctionStubMap
= nullptr;
131 #endif // ENABLE(JIT)