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
);