2 * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved.
3 * Copyright (C) 2007 Eric Seidel <eric@webkit.org>
4 * Copyright (C) 2009 Acision BV. All rights reserved.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
23 #include "MachineStackMarker.h"
25 #include "ConservativeRoots.h"
28 #include "JSCInlines.h"
32 #include <wtf/StdLibExtras.h>
36 #include <mach/mach_init.h>
37 #include <mach/mach_port.h>
38 #include <mach/task.h>
39 #include <mach/thread_act.h>
40 #include <mach/vm_map.h>
58 #if HAVE(PTHREAD_NP_H)
59 #include <pthread_np.h>
62 #if USE(PTHREADS) && !OS(WINDOWS) && !OS(DARWIN)
72 static inline void swapIfBackwards(void*& begin
, void*& end
)
77 std::swap(begin
, end
);
85 typedef mach_port_t PlatformThread
;
87 typedef HANDLE PlatformThread
;
89 typedef pthread_t PlatformThread
;
90 static const int SigThreadSuspendResume
= SIGUSR2
;
92 #if defined(SA_RESTART)
93 static void pthreadSignalHandlerSuspendResume(int)
96 sigemptyset(&signalSet
);
97 sigaddset(&signalSet
, SigThreadSuspendResume
);
98 sigsuspend(&signalSet
);
103 class MachineThreads::Thread
{
104 WTF_MAKE_FAST_ALLOCATED
;
106 Thread(const PlatformThread
& platThread
, void* base
)
107 : platformThread(platThread
)
110 #if USE(PTHREADS) && !OS(WINDOWS) && !OS(DARWIN) && defined(SA_RESTART)
111 // if we have SA_RESTART, enable SIGUSR2 debugging mechanism
112 struct sigaction action
;
113 action
.sa_handler
= pthreadSignalHandlerSuspendResume
;
114 sigemptyset(&action
.sa_mask
);
115 action
.sa_flags
= SA_RESTART
;
116 sigaction(SigThreadSuspendResume
, &action
, 0);
120 sigaddset(&mask
, SigThreadSuspendResume
);
121 pthread_sigmask(SIG_UNBLOCK
, &mask
, 0);
126 PlatformThread platformThread
;
130 MachineThreads::MachineThreads(Heap
* heap
)
131 : m_registeredThreads(0)
132 , m_threadSpecific(0)
140 MachineThreads::~MachineThreads()
142 if (m_threadSpecific
)
143 threadSpecificKeyDelete(m_threadSpecific
);
145 MutexLocker
registeredThreadsLock(m_registeredThreadsMutex
);
146 for (Thread
* t
= m_registeredThreads
; t
;) {
147 Thread
* next
= t
->next
;
153 static inline PlatformThread
getCurrentPlatformThread()
156 return pthread_mach_thread_np(pthread_self());
158 return GetCurrentThread();
160 return pthread_self();
164 static inline bool equalThread(const PlatformThread
& first
, const PlatformThread
& second
)
166 #if OS(DARWIN) || OS(WINDOWS)
167 return first
== second
;
169 return !!pthread_equal(first
, second
);
171 #error Need a way to compare threads on this platform
175 void MachineThreads::makeUsableFromMultipleThreads()
177 if (m_threadSpecific
)
180 threadSpecificKeyCreate(&m_threadSpecific
, removeThread
);
183 void MachineThreads::addCurrentThread()
185 ASSERT(!m_heap
->vm()->hasExclusiveThread() || m_heap
->vm()->exclusiveThread() == std::this_thread::get_id());
187 if (!m_threadSpecific
|| threadSpecificGet(m_threadSpecific
))
190 threadSpecificSet(m_threadSpecific
, this);
191 Thread
* thread
= new Thread(getCurrentPlatformThread(), wtfThreadData().stack().origin());
193 MutexLocker
lock(m_registeredThreadsMutex
);
195 thread
->next
= m_registeredThreads
;
196 m_registeredThreads
= thread
;
199 void MachineThreads::removeThread(void* p
)
202 static_cast<MachineThreads
*>(p
)->removeCurrentThread();
205 void MachineThreads::removeCurrentThread()
207 PlatformThread currentPlatformThread
= getCurrentPlatformThread();
209 MutexLocker
lock(m_registeredThreadsMutex
);
211 if (equalThread(currentPlatformThread
, m_registeredThreads
->platformThread
)) {
212 Thread
* t
= m_registeredThreads
;
213 m_registeredThreads
= m_registeredThreads
->next
;
216 Thread
* last
= m_registeredThreads
;
218 for (t
= m_registeredThreads
->next
; t
; t
= t
->next
) {
219 if (equalThread(t
->platformThread
, currentPlatformThread
)) {
220 last
->next
= t
->next
;
225 ASSERT(t
); // If t is NULL, we never found ourselves in the list.
231 #define REGISTER_BUFFER_ALIGNMENT __attribute__ ((aligned (sizeof(void*))))
233 #define REGISTER_BUFFER_ALIGNMENT
236 void MachineThreads::gatherFromCurrentThread(ConservativeRoots
& conservativeRoots
, JITStubRoutineSet
& jitStubRoutines
, CodeBlockSet
& codeBlocks
, void* stackCurrent
)
238 // setjmp forces volatile registers onto the stack
239 jmp_buf registers REGISTER_BUFFER_ALIGNMENT
;
241 #pragma warning(push)
242 #pragma warning(disable: 4611)
249 void* registersBegin
= ®isters
;
250 void* registersEnd
= reinterpret_cast<void*>(roundUpToMultipleOf
<sizeof(void*)>(reinterpret_cast<uintptr_t>(®isters
+ 1)));
251 swapIfBackwards(registersBegin
, registersEnd
);
252 conservativeRoots
.add(registersBegin
, registersEnd
, jitStubRoutines
, codeBlocks
);
254 void* stackBegin
= stackCurrent
;
255 void* stackEnd
= wtfThreadData().stack().origin();
256 swapIfBackwards(stackBegin
, stackEnd
);
257 conservativeRoots
.add(stackBegin
, stackEnd
, jitStubRoutines
, codeBlocks
);
260 static inline void suspendThread(const PlatformThread
& platformThread
)
263 thread_suspend(platformThread
);
265 SuspendThread(platformThread
);
267 pthread_kill(platformThread
, SigThreadSuspendResume
);
269 #error Need a way to suspend threads on this platform
273 static inline void resumeThread(const PlatformThread
& platformThread
)
276 thread_resume(platformThread
);
278 ResumeThread(platformThread
);
280 pthread_kill(platformThread
, SigThreadSuspendResume
);
282 #error Need a way to resume threads on this platform
286 typedef unsigned long usword_t
; // word size, assumed to be either 32 or 64 bit
291 typedef i386_thread_state_t PlatformThreadRegisters
;
293 typedef x86_thread_state64_t PlatformThreadRegisters
;
295 typedef ppc_thread_state_t PlatformThreadRegisters
;
297 typedef ppc_thread_state64_t PlatformThreadRegisters
;
299 typedef arm_thread_state_t PlatformThreadRegisters
;
301 typedef arm_thread_state64_t PlatformThreadRegisters
;
303 #error Unknown Architecture
307 typedef CONTEXT PlatformThreadRegisters
;
309 typedef pthread_attr_t PlatformThreadRegisters
;
311 #error Need a thread register struct for this platform
314 static size_t getPlatformThreadRegisters(const PlatformThread
& platformThread
, PlatformThreadRegisters
& regs
)
319 unsigned user_count
= sizeof(regs
)/sizeof(int);
320 thread_state_flavor_t flavor
= i386_THREAD_STATE
;
322 unsigned user_count
= x86_THREAD_STATE64_COUNT
;
323 thread_state_flavor_t flavor
= x86_THREAD_STATE64
;
325 unsigned user_count
= PPC_THREAD_STATE_COUNT
;
326 thread_state_flavor_t flavor
= PPC_THREAD_STATE
;
328 unsigned user_count
= PPC_THREAD_STATE64_COUNT
;
329 thread_state_flavor_t flavor
= PPC_THREAD_STATE64
;
331 unsigned user_count
= ARM_THREAD_STATE_COUNT
;
332 thread_state_flavor_t flavor
= ARM_THREAD_STATE
;
334 unsigned user_count
= ARM_THREAD_STATE64_COUNT
;
335 thread_state_flavor_t flavor
= ARM_THREAD_STATE64
;
337 #error Unknown Architecture
340 kern_return_t result
= thread_get_state(platformThread
, flavor
, (thread_state_t
)®s
, &user_count
);
341 if (result
!= KERN_SUCCESS
) {
342 WTFReportFatalError(__FILE__
, __LINE__
, WTF_PRETTY_FUNCTION
,
343 "JavaScript garbage collection failed because thread_get_state returned an error (%d). This is probably the result of running inside Rosetta, which is not supported.", result
);
346 return user_count
* sizeof(usword_t
);
350 regs
.ContextFlags
= CONTEXT_INTEGER
| CONTEXT_CONTROL
;
351 GetThreadContext(platformThread
, ®s
);
352 return sizeof(CONTEXT
);
354 pthread_attr_init(®s
);
355 #if HAVE(PTHREAD_NP_H) || OS(NETBSD)
357 // e.g. on FreeBSD 5.4, neundorf@kde.org
358 pthread_attr_get_np(platformThread
, ®s
);
361 // FIXME: this function is non-portable; other POSIX systems may have different np alternatives
362 pthread_getattr_np(platformThread
, ®s
);
366 #error Need a way to get thread registers on this platform
370 static inline void* otherThreadStackPointer(const PlatformThreadRegisters
& regs
)
377 return reinterpret_cast<void*>(regs
.__esp
);
379 return reinterpret_cast<void*>(regs
.__rsp
);
380 #elif CPU(PPC) || CPU(PPC64)
381 return reinterpret_cast<void*>(regs
.__r1
);
383 return reinterpret_cast<void*>(regs
.__sp
);
385 return reinterpret_cast<void*>(regs
.__sp
);
387 #error Unknown Architecture
390 #else // !__DARWIN_UNIX03
393 return reinterpret_cast<void*>(regs
.esp
);
395 return reinterpret_cast<void*>(regs
.rsp
);
396 #elif CPU(PPC) || CPU(PPC64)
397 return reinterpret_cast<void*>(regs
.r1
);
399 #error Unknown Architecture
402 #endif // __DARWIN_UNIX03
408 return reinterpret_cast<void*>((uintptr_t) regs
.Sp
);
410 return reinterpret_cast<void*>((uintptr_t) regs
.IntSp
);
412 return reinterpret_cast<void*>((uintptr_t) regs
.Esp
);
414 return reinterpret_cast<void*>((uintptr_t) regs
.Rsp
);
416 #error Unknown Architecture
421 size_t stackSize
= 0;
424 int rc
= pthread_stackseg_np(pthread_self(), &ss
);
425 stackBase
= (void*)((size_t) ss
.ss_sp
- ss
.ss_size
);
426 stackSize
= ss
.ss_size
;
428 int rc
= pthread_attr_getstack(®s
, &stackBase
, &stackSize
);
430 (void)rc
; // FIXME: Deal with error code somehow? Seems fatal.
432 return static_cast<char*>(stackBase
) + stackSize
;
434 #error Need a way to get the stack pointer for another thread on this platform
438 static void freePlatformThreadRegisters(PlatformThreadRegisters
& regs
)
440 #if USE(PTHREADS) && !OS(WINDOWS) && !OS(DARWIN)
441 pthread_attr_destroy(®s
);
447 void MachineThreads::gatherFromOtherThread(ConservativeRoots
& conservativeRoots
, Thread
* thread
, JITStubRoutineSet
& jitStubRoutines
, CodeBlockSet
& codeBlocks
)
449 PlatformThreadRegisters regs
;
450 size_t regSize
= getPlatformThreadRegisters(thread
->platformThread
, regs
);
452 conservativeRoots
.add(static_cast<void*>(®s
), static_cast<void*>(reinterpret_cast<char*>(®s
) + regSize
), jitStubRoutines
, codeBlocks
);
454 void* stackPointer
= otherThreadStackPointer(regs
);
455 void* stackBase
= thread
->stackBase
;
456 swapIfBackwards(stackPointer
, stackBase
);
457 stackPointer
= reinterpret_cast<void*>(WTF::roundUpToMultipleOf
<sizeof(void*)>(reinterpret_cast<uintptr_t>(stackPointer
)));
458 conservativeRoots
.add(stackPointer
, stackBase
, jitStubRoutines
, codeBlocks
);
460 freePlatformThreadRegisters(regs
);
463 void MachineThreads::gatherConservativeRoots(ConservativeRoots
& conservativeRoots
, JITStubRoutineSet
& jitStubRoutines
, CodeBlockSet
& codeBlocks
, void* stackCurrent
)
465 gatherFromCurrentThread(conservativeRoots
, jitStubRoutines
, codeBlocks
, stackCurrent
);
467 if (m_threadSpecific
) {
468 PlatformThread currentPlatformThread
= getCurrentPlatformThread();
470 MutexLocker
lock(m_registeredThreadsMutex
);
473 // Forbid malloc during the gather phase. The gather phase suspends
474 // threads, so a malloc during gather would risk a deadlock with a
475 // thread that had been suspended while holding the malloc lock.
478 for (Thread
* thread
= m_registeredThreads
; thread
; thread
= thread
->next
) {
479 if (!equalThread(thread
->platformThread
, currentPlatformThread
))
480 suspendThread(thread
->platformThread
);
483 // It is safe to access the registeredThreads list, because we earlier asserted that locks are being held,
484 // and since this is a shared heap, they are real locks.
485 for (Thread
* thread
= m_registeredThreads
; thread
; thread
= thread
->next
) {
486 if (!equalThread(thread
->platformThread
, currentPlatformThread
))
487 gatherFromOtherThread(conservativeRoots
, thread
, jitStubRoutines
, codeBlocks
);
490 for (Thread
* thread
= m_registeredThreads
; thread
; thread
= thread
->next
) {
491 if (!equalThread(thread
->platformThread
, currentPlatformThread
))
492 resumeThread(thread
->platformThread
);