8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 05:23:02 +01:00
firebird-mirror/src/jrd/ThreadStart.cpp
2009-06-27 06:23:36 +00:00

346 lines
8.1 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: ThreadData.cpp
* DESCRIPTION: Thread support routines
*
* The contents of this file are subject to the Interbase Public
* License Version 1.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy
* of the License at http://www.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
* 2002.10.28 Sean Leyne - Completed removal of obsolete "DGUX" port
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
* Alex Peshkov
*/
#include "firebird.h"
#include <stdio.h>
#include <errno.h>
#include "../jrd/common.h"
#include "../jrd/ThreadStart.h"
#include "../jrd/os/thd_priority.h"
#include "../jrd/gds_proto.h"
#include "../jrd/isc_s_proto.h"
#include "../jrd/gdsassert.h"
#ifdef WIN_NT
#include <process.h>
#include <windows.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#include "../common/classes/locks.h"
#include "../common/classes/rwlock.h"
int API_ROUTINE gds__thread_start(ThreadEntryPoint* entrypoint,
void* arg,
int priority,
int /*flags*/,
void* thd_id)
{
/**************************************
*
* g d s _ _ t h r e a d _ s t a r t
*
**************************************
*
* Functional description
* Start a thread.
*
**************************************/
int rc = 0;
try {
ThreadStart::start(entrypoint, arg, priority, thd_id);
}
catch (const Firebird::status_exception& status) {
rc = status.value()[1];
}
return rc;
}
namespace
{
#ifdef THREAD_PSCHED
THREAD_ENTRY_DECLARE threadStart(THREAD_ENTRY_PARAM arg)
{
fb_assert(arg);
Firebird::MemoryPool::setContextPool(getDefaultMemoryPool());
{
ThreadPriorityScheduler* tps = static_cast<ThreadPriorityScheduler*>(arg);
try {
tps->run();
}
catch (...) {
tps->detach();
throw;
}
tps->detach();
return 0;
}
}
#define THREAD_ENTRYPOINT threadStart
#define THREAD_ARG tps
#else // THREAD_PSCHED
// due to same names of parameters for various ThreadData::start(...),
// we may use common macro for various platforms
#define THREAD_ENTRYPOINT threadStart
#define THREAD_ARG static_cast<THREAD_ENTRY_PARAM> (FB_NEW(*getDefaultMemoryPool()) \
ThreadArgs(reinterpret_cast<THREAD_ENTRY_RETURN (THREAD_ENTRY_CALL *) (THREAD_ENTRY_PARAM)>(routine), \
static_cast<THREAD_ENTRY_PARAM>(arg)))
class ThreadArgs
{
public:
typedef THREAD_ENTRY_RETURN (THREAD_ENTRY_CALL *Routine) (THREAD_ENTRY_PARAM);
typedef THREAD_ENTRY_PARAM Arg;
private:
Routine routine;
Arg arg;
public:
ThreadArgs(Routine r, Arg a) : routine(r), arg(a) { }
ThreadArgs(const ThreadArgs& t) : routine(t.routine), arg(t.arg) { }
void run() { routine(arg); }
private:
ThreadArgs& operator=(const ThreadArgs&);
};
THREAD_ENTRY_DECLARE threadStart(THREAD_ENTRY_PARAM arg)
{
fb_assert(arg);
Firebird::ContextPoolHolder mainThreadContext(getDefaultMemoryPool());
ThreadArgs localArgs(*static_cast<ThreadArgs*>(arg));
delete static_cast<ThreadArgs*>(arg);
localArgs.run();
return 0;
}
#endif //THREAD_PSCHED
} // anonymous namespace
#ifdef USE_POSIX_THREADS
#define START_THREAD
void ThreadStart::start(ThreadEntryPoint* routine,
void* arg,
int priority_arg,
void* thd_id)
{
/**************************************
*
* t h r e a d _ s t a r t ( P O S I X )
*
**************************************
*
* Functional description
* Start a new thread. Return 0 if successful,
* status if not.
*
**************************************/
pthread_t thread;
pthread_attr_t pattr;
int state;
#if defined (LINUX) || defined (FREEBSD)
if (state = pthread_create(&thread, NULL, THREAD_ENTRYPOINT, THREAD_ARG))
Firebird::system_call_failed::raise("pthread_create", state);
if (!thd_id)
{
if (state = pthread_detach(thread))
Firebird::system_call_failed::raise("pthread_detach", state);
}
#else
state = pthread_attr_init(&pattr);
if (state)
Firebird::system_call_failed::raise("pthread_attr_init", state);
#ifdef _AIX
// adjust stack size for AIX
// For AIX 32-bit compiled applications, the default stacksize is 96 KB,
// see <pthread.h>. For 64-bit compiled applications, the default stacksize
// is 192 KB. This is too small - see HP-UX note above
size_t stack_size;
state = pthread_attr_getstacksize(&pattr, &stack_size);
if (state)
Firebird::system_call_failed::raise("pthread_attr_getstacksize");
if (stack_size < 0x40000L)
{
state = pthread_attr_setstacksize(&pattr, 0x40000L);
if (state)
Firebird::system_call_failed::raise("pthread_attr_setstacksize", state);
}
#endif // _AIX
state = pthread_attr_setscope(&pattr, PTHREAD_SCOPE_SYSTEM);
if (state)
Firebird::system_call_failed::raise("pthread_attr_setscope", state);
if (!thd_id)
{
state = pthread_attr_setdetachstate(&pattr, PTHREAD_CREATE_DETACHED);
if (state)
Firebird::system_call_failed::raise("pthread_attr_setdetachstate", state);
}
state = pthread_create(&thread, &pattr, THREAD_ENTRYPOINT, THREAD_ARG);
int state2 = pthread_attr_destroy(&pattr);
if (state)
Firebird::system_call_failed::raise("pthread_create", state);
if (state2)
Firebird::system_call_failed::raise("pthread_attr_destroy", state2);
#endif
if (thd_id)
{
*static_cast<pthread_t*>(thd_id) = thread;
}
}
void THD_wait_for_completion(ThreadHandle& thread)
{
int state = pthread_join(thread, NULL);
if (state)
Firebird::system_call_failed::raise("pthread_join", state);
}
#endif /* USE_POSIX_THREADS */
#ifdef WIN_NT
#define START_THREAD
void ThreadStart::start(ThreadEntryPoint* routine,
void* arg,
int priority_arg,
void* thd_id)
{
/**************************************
*
* t h r e a d _ s t a r t ( W I N _ N T )
*
**************************************
*
* Functional description
* Start a new thread. Return 0 if successful,
* status if not.
*
**************************************/
int priority;
switch (priority_arg)
{
case THREAD_critical:
priority = THREAD_PRIORITY_TIME_CRITICAL;
break;
case THREAD_high:
priority = THREAD_PRIORITY_HIGHEST;
break;
case THREAD_medium_high:
priority = THREAD_PRIORITY_ABOVE_NORMAL;
break;
case THREAD_medium:
priority = THREAD_PRIORITY_NORMAL;
break;
case THREAD_medium_low:
priority = THREAD_PRIORITY_BELOW_NORMAL;
break;
case THREAD_low:
default:
priority = THREAD_PRIORITY_LOWEST;
break;
}
#ifdef THREAD_PSCHED
ThreadPriorityScheduler::Init();
ThreadPriorityScheduler* tps = FB_NEW(*getDefaultMemoryPool())
ThreadPriorityScheduler(routine, arg, ThreadPriorityScheduler::adjustPriority(priority));
#endif // THREAD_PSCHED
/* I have changed the CreateThread here to _beginthreadex() as using
* CreateThread() can lead to memory leaks caused by C-runtime library.
* Advanced Windows by Richter pg. # 109. */
unsigned thread_id;
unsigned long real_handle =
_beginthreadex(NULL, 0, THREAD_ENTRYPOINT, THREAD_ARG, CREATE_SUSPENDED, &thread_id);
if (!real_handle)
{
Firebird::system_call_failed::raise("_beginthreadex", GetLastError());
}
HANDLE handle = reinterpret_cast<HANDLE>(real_handle);
SetThreadPriority(handle, priority);
ResumeThread(handle);
if (thd_id)
{
*static_cast<HANDLE*>(thd_id) = handle;
}
else
{
CloseHandle(handle);
}
}
void THD_wait_for_completion(ThreadHandle& handle)
{
WaitForSingleObject(handle, INFINITE);
}
#endif // WIN_NT
#ifndef START_THREAD
void ThreadStart::start(ThreadEntryPoint* routine,
void* arg,
int priority_arg,
void* thd_id)
{
/**************************************
*
* t h r e a d _ s t a r t ( G e n e r i c )
*
**************************************
*
* Functional description
* Wrong attempt to start a new thread.
*
**************************************/
}
void THD_wait_for_completion(ThreadHandle&)
{
}
#endif // START_THREAD