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(adoptPtr(new 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 NativeExecutable
* JITThunks::hostFunctionStub(VM
* vm
, NativeFunction function
, NativeFunction constructor
)
81 ASSERT(!isCompilationThread());
83 if (NativeExecutable
* nativeExecutable
= m_hostFunctionStubMap
->get(std::make_pair(function
, constructor
)))
84 return nativeExecutable
;
86 NativeExecutable
* nativeExecutable
= NativeExecutable::create(
88 adoptRef(new NativeJITCode(JIT::compileCTINativeCall(vm
, function
), JITCode::HostCallThunk
)),
90 adoptRef(new NativeJITCode(MacroAssemblerCodeRef::createSelfManagedCodeRef(ctiNativeConstruct(vm
)), JITCode::HostCallThunk
)),
91 constructor
, NoIntrinsic
);
92 weakAdd(*m_hostFunctionStubMap
, std::make_pair(function
, constructor
), Weak
<NativeExecutable
>(nativeExecutable
));
93 return nativeExecutable
;
96 NativeExecutable
* JITThunks::hostFunctionStub(VM
* vm
, NativeFunction function
, ThunkGenerator generator
, Intrinsic intrinsic
)
98 ASSERT(!isCompilationThread());
100 if (NativeExecutable
* nativeExecutable
= m_hostFunctionStubMap
->get(std::make_pair(function
, &callHostFunctionAsConstructor
)))
101 return nativeExecutable
;
103 RefPtr
<JITCode
> forCall
;
105 if (vm
->canUseJIT()) {
106 MacroAssemblerCodeRef entry
= generator(vm
);
107 forCall
= adoptRef(new DirectJITCode(entry
, entry
.code(), JITCode::HostCallThunk
));
110 forCall
= adoptRef(new NativeJITCode(JIT::compileCTINativeCall(vm
, function
), JITCode::HostCallThunk
));
112 RefPtr
<JITCode
> forConstruct
= adoptRef(new NativeJITCode(MacroAssemblerCodeRef::createSelfManagedCodeRef(ctiNativeConstruct(vm
)), JITCode::HostCallThunk
));
114 NativeExecutable
* nativeExecutable
= NativeExecutable::create(*vm
, forCall
, function
, forConstruct
, callHostFunctionAsConstructor
, intrinsic
);
115 weakAdd(*m_hostFunctionStubMap
, std::make_pair(function
, &callHostFunctionAsConstructor
), Weak
<NativeExecutable
>(nativeExecutable
));
116 return nativeExecutable
;
119 void JITThunks::clearHostFunctionStubs()
121 m_hostFunctionStubMap
.clear();
126 #endif // ENABLE(JIT)