2 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
3 * Copyright (C) 2007 Justin Haygood (jhaygood@reaktix.com)
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. Neither the name of Apple Inc. ("Apple") nor the names of
15 * its contributors may be used to endorse or promote products derived
16 * from this software without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
19 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
22 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #include "Threading.h"
35 #include "CurrentTime.h"
37 #include "MainThread.h"
38 #include "RandomNumberSeed.h"
45 static Mutex
* atomicallyInitializedStaticMutex
;
47 static Mutex
& threadMapMutex()
53 void initializeThreading()
55 if (!g_thread_supported())
57 ASSERT(g_thread_supported());
59 if (!atomicallyInitializedStaticMutex
) {
60 atomicallyInitializedStaticMutex
= new Mutex
;
62 initializeRandomNumberGenerator();
66 void lockAtomicallyInitializedStaticMutex()
68 ASSERT(atomicallyInitializedStaticMutex
);
69 atomicallyInitializedStaticMutex
->lock();
72 void unlockAtomicallyInitializedStaticMutex()
74 atomicallyInitializedStaticMutex
->unlock();
77 static HashMap
<ThreadIdentifier
, GThread
*>& threadMap()
79 static HashMap
<ThreadIdentifier
, GThread
*> map
;
83 static ThreadIdentifier
identifierByGthreadHandle(GThread
*& thread
)
85 MutexLocker
locker(threadMapMutex());
87 HashMap
<ThreadIdentifier
, GThread
*>::iterator i
= threadMap().begin();
88 for (; i
!= threadMap().end(); ++i
) {
89 if (i
->second
== thread
)
96 static ThreadIdentifier
establishIdentifierForThread(GThread
*& thread
)
98 ASSERT(!identifierByGthreadHandle(thread
));
100 MutexLocker
locker(threadMapMutex());
102 static ThreadIdentifier identifierCount
= 1;
104 threadMap().add(identifierCount
, thread
);
106 return identifierCount
++;
109 static GThread
* threadForIdentifier(ThreadIdentifier id
)
111 MutexLocker
locker(threadMapMutex());
113 return threadMap().get(id
);
116 static void clearThreadForIdentifier(ThreadIdentifier id
)
118 MutexLocker
locker(threadMapMutex());
120 ASSERT(threadMap().contains(id
));
122 threadMap().remove(id
);
125 ThreadIdentifier
createThreadInternal(ThreadFunction entryPoint
, void* data
, const char*)
128 if (!(thread
= g_thread_create(entryPoint
, data
, TRUE
, 0))) {
129 LOG_ERROR("Failed to create thread at entry point %p with data %p", entryPoint
, data
);
133 ThreadIdentifier threadID
= establishIdentifierForThread(thread
);
137 void initializeCurrentThreadInternal(const char*)
141 int waitForThreadCompletion(ThreadIdentifier threadID
, void** result
)
145 GThread
* thread
= threadForIdentifier(threadID
);
147 void* joinResult
= g_thread_join(thread
);
149 *result
= joinResult
;
151 clearThreadForIdentifier(threadID
);
155 void detachThread(ThreadIdentifier
)
159 ThreadIdentifier
currentThread()
161 GThread
* currentThread
= g_thread_self();
162 if (ThreadIdentifier id
= identifierByGthreadHandle(currentThread
))
164 return establishIdentifierForThread(currentThread
);
168 : m_mutex(g_mutex_new())
178 g_mutex_lock(m_mutex
.get());
181 bool Mutex::tryLock()
183 return g_mutex_trylock(m_mutex
.get());
188 g_mutex_unlock(m_mutex
.get());
191 ThreadCondition::ThreadCondition()
192 : m_condition(g_cond_new())
196 ThreadCondition::~ThreadCondition()
200 void ThreadCondition::wait(Mutex
& mutex
)
202 g_cond_wait(m_condition
.get(), mutex
.impl().get());
205 bool ThreadCondition::timedWait(Mutex
& mutex
, double absoluteTime
)
207 // Time is in the past - return right away.
208 if (absoluteTime
< currentTime())
211 // Time is too far in the future for g_cond_timed_wait - wait forever.
212 if (absoluteTime
> INT_MAX
) {
217 int timeSeconds
= static_cast<int>(absoluteTime
);
218 int timeMicroseconds
= static_cast<int>((absoluteTime
- timeSeconds
) * 1000000.0);
221 targetTime
.tv_sec
= timeSeconds
;
222 targetTime
.tv_usec
= timeMicroseconds
;
224 return g_cond_timed_wait(m_condition
.get(), mutex
.impl().get(), &targetTime
);
227 void ThreadCondition::signal()
229 g_cond_signal(m_condition
.get());
232 void ThreadCondition::broadcast()
234 g_cond_broadcast(m_condition
.get());
240 #endif // !USE(PTHREADS)