Changeset 219647 in webkit
- Timestamp:
- Jul 18, 2017, 9:53:59 PM (8 years ago)
- Location:
- trunk/Source
- Files:
-
- 14 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/Source/JavaScriptCore/ChangeLog
r219644 r219647 1 2017-07-18 Yusuke Suzuki <utatane.tea@gmail.com> 2 3 WTF::Thread should have the threads stack bounds. 4 https://bugs.webkit.org/show_bug.cgi?id=173975 5 6 Reviewed by Mark Lam. 7 8 There is a site in JSC that try to walk another thread's stack. 9 Currently, stack bounds are stored in WTFThreadData which is located 10 in TLS. Thus, only the thread itself can access its own WTFThreadData. 11 We workaround this situation by holding StackBounds in MachineThread in JSC, 12 but StackBounds should be put in WTF::Thread instead. 13 14 This patch adds StackBounds to WTF::Thread. StackBounds information is tightly 15 coupled with Thread. Thus putting it in WTF::Thread is natural choice. 16 17 * heap/MachineStackMarker.cpp: 18 (JSC::MachineThreads::MachineThread::MachineThread): 19 (JSC::MachineThreads::MachineThread::captureStack): 20 * heap/MachineStackMarker.h: 21 (JSC::MachineThreads::MachineThread::stackBase): 22 (JSC::MachineThreads::MachineThread::stackEnd): 23 * runtime/VMTraps.cpp: 24 1 25 2017-07-18 Andy Estes <aestes@apple.com> 2 26 -
trunk/Source/JavaScriptCore/heap/MachineStackMarker.cpp
r219595 r219647 239 239 : m_thread(WTF::Thread::current()) 240 240 { 241 auto stackBounds = wtfThreadData().stack();242 m_stackBase = stackBounds.origin();243 m_stackEnd = stackBounds.end();244 241 } 245 242 … … 302 299 std::pair<void*, size_t> MachineThreads::MachineThread::captureStack(void* stackTop) 303 300 { 304 char* begin = reinterpret_cast_ptr<char*>( m_stackBase);301 char* begin = reinterpret_cast_ptr<char*>(stackBase()); 305 302 char* end = bitwise_cast<char*>(WTF::roundUpToMultipleOf<sizeof(void*)>(reinterpret_cast<uintptr_t>(stackTop))); 306 303 ASSERT(begin >= end); … … 309 306 ASSERT(WTF::roundUpToMultipleOf<sizeof(void*)>(reinterpret_cast<uintptr_t>(endWithRedZone)) == reinterpret_cast<uintptr_t>(endWithRedZone)); 310 307 311 if (endWithRedZone < m_stackEnd)312 endWithRedZone = reinterpret_cast_ptr<char*>( m_stackEnd);308 if (endWithRedZone < stackEnd()) 309 endWithRedZone = reinterpret_cast_ptr<char*>(stackEnd()); 313 310 314 311 std::swap(begin, endWithRedZone); -
trunk/Source/JavaScriptCore/heap/MachineStackMarker.h
r219427 r219647 73 73 74 74 WTF::ThreadIdentifier threadID() const { return m_thread->id(); } 75 void* stackBase() const { return m_ stackBase; }76 void* stackEnd() const { return m_ stackEnd; }75 void* stackBase() const { return m_thread->stack().origin(); } 76 void* stackEnd() const { return m_thread->stack().end(); } 77 77 78 78 Ref<WTF::Thread> m_thread; 79 void* m_stackBase;80 void* m_stackEnd;81 79 MachineThread* m_next { nullptr }; 82 80 MachineThread* m_prev { nullptr }; … … 105 103 CurrentThreadState stateName; \ 106 104 stateName.stackTop = &stateName; \ 107 stateName.stackOrigin = wtfThreadData().stack().origin(); \105 stateName.stackOrigin = Thread::current().stack().origin(); \ 108 106 ALLOCATE_AND_GET_REGISTER_STATE(stateName ## _registerState); \ 109 107 stateName.registerState = &stateName ## _registerState -
trunk/Source/JavaScriptCore/runtime/VMTraps.cpp
r219427 r219647 268 268 269 269 Thread& thread = *ownerThread->get(); 270 StackBounds stackBounds = StackBounds::emptyBounds(); 271 { 272 // FIXME: We need to use the machine threads because it is the only non-TLS source 273 // for the stack bounds of this thread. We should keep in on the WTF::Thread instead. 274 // see: https://bugs.webkit.org/show_bug.cgi?id=173975 275 MachineThreads& machineThreads = vm.heap.machineThreads(); 276 auto machineThreadsLock = tryHoldLock(machineThreads.getLock()); 277 if (!machineThreadsLock) 278 return; // Try again later. 279 280 auto& threadList = machineThreads.threadsListHead(machineThreadsLock); 281 for (MachineThreads::MachineThread* machineThread = threadList.head(); machineThread; machineThread = machineThread->next()) { 282 if (machineThread->m_thread.get() == thread) 283 stackBounds = StackBounds(machineThread->stackBase(), machineThread->stackEnd()); 284 } 285 RELEASE_ASSERT(!stackBounds.isEmpty()); 286 } 287 288 vm.traps().tryInstallTrapBreakpoints(context, stackBounds); 270 vm.traps().tryInstallTrapBreakpoints(context, thread.stack()); 289 271 }); 290 272 } -
trunk/Source/WTF/ChangeLog
r219644 r219647 1 2017-07-18 Yusuke Suzuki <utatane.tea@gmail.com> 2 3 WTF::Thread should have the threads stack bounds. 4 https://bugs.webkit.org/show_bug.cgi?id=173975 5 6 Reviewed by Mark Lam. 7 8 We move StackBounds from WTFThreadData to WTF::Thread. 9 One important thing is that we should make valid StackBounds 10 visible to Thread::create() caller. When the caller get 11 WTF::Thread from Thread::create(), this WTF::Thread should 12 have a valid StackBounds. But StackBounds information can be 13 retrived only in the WTF::Thread's thread itself. 14 15 We also clean up WTF::initializeThreading. StringImpl::empty() 16 is now statically initialized by using constexpr constructor. 17 Thus we do not need to call StringImpl::empty() explicitly here. 18 And WTF::initializeThreading() does not have any main thread 19 affinity right now in all the platforms. So we fix the comment 20 in Threading.h. Then, now, WTF::initializeThreading() is called 21 in UI thread when using Web thread in iOS. 22 23 * wtf/StackBounds.h: 24 (WTF::StackBounds::emptyBounds): 25 (WTF::StackBounds::StackBounds): 26 * wtf/Threading.cpp: 27 (WTF::threadEntryPoint): 28 (WTF::Thread::create): 29 (WTF::Thread::currentMayBeNull): 30 (WTF::Thread::initialize): 31 (WTF::initializeThreading): 32 * wtf/Threading.h: 33 (WTF::Thread::stack): 34 * wtf/ThreadingPthreads.cpp: 35 (WTF::Thread::initializeCurrentThreadEvenIfNonWTFCreated): 36 (WTF::Thread::current): 37 (WTF::initializeCurrentThreadEvenIfNonWTFCreated): Deleted. 38 (WTF::Thread::currentMayBeNull): Deleted. 39 * wtf/ThreadingWin.cpp: 40 (WTF::Thread::initializeCurrentThreadEvenIfNonWTFCreated): 41 (WTF::Thread::initializeCurrentThreadInternal): 42 (WTF::Thread::current): 43 * wtf/WTFThreadData.cpp: 44 (WTF::WTFThreadData::WTFThreadData): 45 * wtf/WTFThreadData.h: 46 (WTF::WTFThreadData::stack): 47 1 48 2017-07-18 Andy Estes <aestes@apple.com> 2 49 -
trunk/Source/WTF/wtf/StackBounds.h
r219427 r219647 41 41 42 42 public: 43 static StackBounds emptyBounds() { return StackBounds(); }43 static constexpr StackBounds emptyBounds() { return StackBounds(); } 44 44 45 45 static StackBounds currentThreadStackBounds() … … 128 128 129 129 private: 130 StackBounds()131 : m_origin( 0)132 , m_bound( 0)130 constexpr StackBounds() 131 : m_origin(nullptr) 132 , m_bound(nullptr) 133 133 { 134 134 } -
trunk/Source/WTF/wtf/Threading.cpp
r219427 r219647 47 47 namespace WTF { 48 48 49 enum class Stage { 50 Start, Initialized 51 }; 52 49 53 struct NewThreadContext { 50 WTF_MAKE_FAST_ALLOCATED;51 public:52 54 const char* name; 53 55 Function<void()> entryPoint; 54 Mutex creationMutex; 56 Stage stage; 57 Mutex mutex; 58 ThreadCondition condition; 55 59 }; 56 60 … … 85 89 { 86 90 NewThreadContext* context = static_cast<NewThreadContext*>(contextData); 91 Function<void()> entryPoint; 92 { 93 // Block until our creating thread has completed any extra setup work, including establishing ThreadIdentifier. 94 MutexLocker locker(context->mutex); 87 95 88 // Block until our creating thread has completed any extra setup work, including 89 // establishing ThreadIdentifier. 90 { 91 MutexLocker locker(context->creationMutex); 96 Thread::initializeCurrentThreadInternal(context->name); 97 entryPoint = WTFMove(context->entryPoint); 98 99 // Ack completion of initialization to the creating thread. 100 context->stage = Stage::Initialized; 101 context->condition.signal(); 92 102 } 93 94 Thread::initializeCurrentThreadInternal(context->name);95 96 auto entryPoint = WTFMove(context->entryPoint);97 98 // Delete the context before starting the thread.99 delete context;100 103 101 104 entryPoint(); … … 104 107 RefPtr<Thread> Thread::create(const char* name, Function<void()>&& entryPoint) 105 108 { 106 NewThreadContext * context = new NewThreadContext { name, WTFMove(entryPoint), { } };109 NewThreadContext context { name, WTFMove(entryPoint), Stage::Start, { }, { } }; 107 110 108 // Prevent the thread body from executing until we've established the thread identifier. 109 MutexLocker locker(context->creationMutex); 111 MutexLocker locker(context.mutex); 112 RefPtr<Thread> result = Thread::createInternal(threadEntryPoint, &context, name); 113 // After establishing Thread, release the mutex and wait for completion of initialization. 114 while (context.stage != Stage::Initialized) 115 context.condition.wait(context.mutex); 110 116 111 return Thread::createInternal(threadEntryPoint, context, name); 117 return result; 118 } 119 120 Thread* Thread::currentMayBeNull() 121 { 122 ThreadHolder* data = ThreadHolder::current(); 123 if (data) 124 return &data->thread(); 125 return nullptr; 126 } 127 128 void Thread::initialize() 129 { 130 m_stack = StackBounds::currentThreadStackBounds(); 112 131 } 113 132 … … 167 186 std::call_once(initializeKey, [] { 168 187 ThreadHolder::initializeOnce(); 169 // StringImpl::empty() does not construct its static string in a threadsafe fashion,170 // so ensure it has been initialized from here.171 StringImpl::empty();172 188 initializeRandomNumberGenerator(); 173 189 wtfThreadData(); -
trunk/Source/WTF/wtf/Threading.h
r219427 r219647 39 39 #include <wtf/PlatformRegisters.h> 40 40 #include <wtf/RefPtr.h> 41 #include <wtf/StackBounds.h> 41 42 #include <wtf/ThreadSafeRefCounted.h> 42 43 … … 68 69 // Returns Thread object. 69 70 WTF_EXPORT_PRIVATE static Thread& current(); 70 WTF_EXPORT_PRIVATEstatic Thread* currentMayBeNull();71 static Thread* currentMayBeNull(); 71 72 72 73 // Returns ThreadIdentifier directly. It is useful if the user only cares about identity … … 109 110 // Helpful for platforms where the thread name must be set from within the thread. 110 111 static void initializeCurrentThreadInternal(const char* threadName); 112 static void initializeCurrentThreadEvenIfNonWTFCreated(); 111 113 112 114 WTF_EXPORT_PRIVATE void dump(PrintStream& out) const; … … 126 128 static void initializePlatformThreading(); 127 129 130 const StackBounds& stack() const 131 { 132 return m_stack; 133 } 134 128 135 #if OS(DARWIN) 129 136 mach_port_t machThread() { return m_platformThread; } … … 141 148 void establish(HANDLE, ThreadIdentifier); 142 149 #endif 150 void initialize(); 143 151 144 152 #if USE(PTHREADS) && !OS(DARWIN) … … 172 180 ThreadIdentifier m_id { 0 }; 173 181 JoinableState m_joinableState { Joinable }; 182 StackBounds m_stack { StackBounds::emptyBounds() }; 174 183 bool m_didExit { false }; 175 184 #if USE(PTHREADS) … … 191 200 }; 192 201 193 // This function must be called from the main thread. It is safe to call it repeatedly. 194 // Darwin is an exception to this rule: it is OK to call it from any thread, the only 195 // requirement is that the calls are not reentrant. 202 // This function can be called from any threads. 196 203 WTF_EXPORT_PRIVATE void initializeThreading(); 197 204 -
trunk/Source/WTF/wtf/ThreadingPthreads.cpp
r219595 r219647 188 188 } 189 189 190 static void initializeCurrentThreadEvenIfNonWTFCreated() 191 { 190 void Thread::initializeCurrentThreadEvenIfNonWTFCreated() 191 { 192 Thread::current().initialize(); 192 193 #if !OS(DARWIN) 193 194 sigset_t mask; … … 299 300 } 300 301 301 Thread* Thread::currentMayBeNull()302 {303 ThreadHolder* data = ThreadHolder::current();304 if (data)305 return &data->thread();306 return nullptr;307 }308 309 302 Thread& Thread::current() 310 303 { … … 313 306 314 307 // Not a WTF-created thread, ThreadIdentifier is not established yet. 315 Ref Ptr<Thread> thread = adoptRef(new Thread());308 Ref<Thread> thread = adoptRef(*new Thread()); 316 309 thread->establish(pthread_self()); 317 ThreadHolder::initialize( *thread);310 ThreadHolder::initialize(thread.get()); 318 311 initializeCurrentThreadEvenIfNonWTFCreated(); 319 return *thread;312 return thread.get(); 320 313 } 321 314 -
trunk/Source/WTF/wtf/ThreadingWin.cpp
r219427 r219647 117 117 } 118 118 119 void Thread::initializeCurrentThreadEvenIfNonWTFCreated() 120 { 121 Thread::current().initialize(); 122 } 123 119 124 // MS_VC_EXCEPTION, THREADNAME_INFO, and setThreadNameInternal all come from <http://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx>. 120 125 static const DWORD MS_VC_EXCEPTION = 0x406D1388; … … 146 151 } 147 152 #endif 153 initializeCurrentThreadEvenIfNonWTFCreated(); 148 154 } 149 155 … … 262 268 Thread& Thread::current() 263 269 { 264 ThreadHolder* data = ThreadHolder::current(); 265 if (data) 266 return data->thread(); 270 if (Thread* current = currentMayBeNull()) 271 return *current; 267 272 268 273 // Not a WTF-created thread, ThreadIdentifier is not established yet. … … 275 280 thread->establish(handle, currentID()); 276 281 ThreadHolder::initialize(thread.get(), Thread::currentID()); 282 initializeCurrentThreadEvenIfNonWTFCreated(); 277 283 return thread.get(); 278 284 } -
trunk/Source/WTF/wtf/WTFThreadData.cpp
r219427 r219647 41 41 42 42 WTFThreadData::WTFThreadData() 43 : m_apiData(0) 44 , m_currentAtomicStringTable(0) 45 , m_defaultAtomicStringTable(0) 46 , m_atomicStringTableDestructor(0) 47 , m_stackBounds(StackBounds::currentThreadStackBounds()) 48 #if ENABLE(STACK_STATS) 49 , m_stackStats() 50 #endif 51 , m_savedStackPointerAtVMEntry(0) 43 : m_stackBounds(StackBounds::currentThreadStackBounds()) 52 44 , m_savedLastStackTop(stack().origin()) 53 45 { -
trunk/Source/WTF/wtf/WTFThreadData.h
r219427 r219647 60 60 const StackBounds& stack() 61 61 { 62 // We need to always get a fresh StackBounds from the OS due to how fibers work.63 // See https://bugs.webkit.org/show_bug.cgi?id=10241164 #if OS(WINDOWS)65 m_stackBounds = StackBounds::currentThreadStackBounds();66 #endif67 62 return m_stackBounds; 68 63 } … … 95 90 } 96 91 97 void* m_apiData ;92 void* m_apiData { nullptr }; 98 93 99 94 private: 100 AtomicStringTable* m_currentAtomicStringTable ;101 AtomicStringTable* m_defaultAtomicStringTable ;102 AtomicStringTableDestructor m_atomicStringTableDestructor ;95 AtomicStringTable* m_currentAtomicStringTable { nullptr }; 96 AtomicStringTable* m_defaultAtomicStringTable { nullptr }; 97 AtomicStringTableDestructor m_atomicStringTableDestructor { nullptr }; 103 98 104 99 StackBounds m_stackBounds; … … 106 101 StackStats::PerThreadStats m_stackStats; 107 102 #endif 108 void* m_savedStackPointerAtVMEntry ;103 void* m_savedStackPointerAtVMEntry { nullptr }; 109 104 void* m_savedLastStackTop; 110 105 -
trunk/Source/WebCore/ChangeLog
r219646 r219647 1 2017-07-18 Yusuke Suzuki <utatane.tea@gmail.com> 2 3 WTF::Thread should have the threads stack bounds. 4 https://bugs.webkit.org/show_bug.cgi?id=173975 5 6 Reviewed by Mark Lam. 7 8 When creating WebThread, we first allocate WebCore::ThreadGlobalData in UI thread 9 and share it with WebThread. 10 The problem is that WebCore::ThreadGlobalData has CachedResourceRequestInitiators. 11 It allocates AtomicString, which requires WTFThreadData. 12 13 In this patch, we call WTF::initializeThreading() before allocating WebCore::ThreadGlobalData. 14 And we also call AtomicString::init() before calling WebCore::ThreadGlobalData since 15 WebCore::ThreadGlobalData allocates AtomicString. 16 17 * platform/ios/wak/WebCoreThread.mm: 18 (StartWebThread): 19 1 20 2017-07-18 Myles C. Maxfield <mmaxfield@apple.com> 2 21 -
trunk/Source/WebCore/platform/ios/wak/WebCoreThread.mm
r219427 r219647 703 703 webThreadStarted = TRUE; 704 704 705 // ThreadGlobalData touches AtomicString, which requires WTFThreadData and Threading initialization. 706 WTF::initializeThreading(); 707 708 // Initialize AtomicString on the main thread. 709 WTF::AtomicString::init(); 710 705 711 // Initialize ThreadGlobalData on the main UI thread so that the WebCore thread 706 712 // can later set it's thread-specific data to point to the same objects. 707 713 WebCore::ThreadGlobalData& unused = WebCore::threadGlobalData(); 708 714 (void)unused; 709 710 // Initialize AtomicString on the main thread.711 WTF::AtomicString::init();712 715 713 716 RunLoop::initializeMainRunLoop();
Note:
See TracChangeset
for help on using the changeset viewer.