2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-11-07 09:06:35 +01:00
|
|
|
* MODULE: thd.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* 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-29 03:45:09 +01:00
|
|
|
*
|
|
|
|
* 2002.10.28 Sean Leyne - Completed removal of obsolete "DGUX" port
|
|
|
|
*
|
2002-10-30 07:40:58 +01:00
|
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2004-04-29 00:43:34 +02:00
|
|
|
#include <stdio.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include "../jrd/thd.h"
|
|
|
|
#include "../jrd/isc.h"
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../jrd/os/thd_priority.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
#include "../jrd/gdsassert.h"
|
2004-07-14 23:56:19 +02:00
|
|
|
#include "../common/classes/fb_tls.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#include <process.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2002-08-26 14:18:16 +02:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-11-01 14:21:17 +01:00
|
|
|
#ifdef SOLARIS
|
|
|
|
#include <thread.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
2004-06-08 15:41:08 +02:00
|
|
|
// THE SOLE PURPOSE OF THE FOLLOWING DECLARATION IS TO ALLOW THE VMS KIT TO
|
|
|
|
// COMPILE. IT IS NOT CORRECT AND MUST BE REMOVED AT SOME POINT.
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData* gdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#ifdef VMS
|
|
|
|
#define THREAD_MUTEXES_DEFINED
|
|
|
|
int THD_mutex_destroy(MUTX_T * mutex)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ m u t e x _ d e s t r o y ( V M S )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int THD_mutex_init(MUTX_T * mutex)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ m u t e x _ i n i t ( V M S )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return ISC_mutex_init(mutex, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int THD_mutex_lock(MUTX_T * mutex)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ m u t e x _ l o c k ( V M S )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return ISC_mutex_lock(mutex);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int THD_mutex_unlock(MUTX_T * mutex)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ m u t e x _ u n l o c k ( V M S )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return ISC_mutex_unlock(mutex);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-06-08 15:41:08 +02:00
|
|
|
#endif //VMS
|
|
|
|
#endif //NOT_USED_OR_REPLACED
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../common/classes/locks.h"
|
|
|
|
#include "../common/classes/rwlock.h"
|
|
|
|
Firebird::Mutex ib_mutex;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-07-14 23:56:19 +02:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
TLS_DECLARE (void*, tSpecific);
|
2004-08-30 20:11:08 +02:00
|
|
|
TLS_DECLARE (ThreadData*, tData);
|
2004-07-14 23:56:19 +02:00
|
|
|
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int API_ROUTINE gds__thread_start(
|
2004-07-02 12:47:03 +02:00
|
|
|
ThreadEntryPoint* entrypoint,
|
2004-06-08 15:41:08 +02:00
|
|
|
void *arg,
|
|
|
|
int priority, int flags, void *thd_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* g d s _ $ t h r e a d _ s t a r t
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* Start a thread.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int rc = 0;
|
|
|
|
try {
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData::start(entrypoint, arg, priority, flags, thd_id);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-06-08 15:41:08 +02:00
|
|
|
catch(const Firebird::status_exception& status) {
|
|
|
|
rc = status.value()[1];
|
|
|
|
}
|
|
|
|
return rc;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
FB_THREAD_ID ThreadData::getId(void)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ g e t _ t h r e a d _ i d
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* Get platform's notion of a thread ID.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2004-06-08 15:41:08 +02:00
|
|
|
FB_THREAD_ID id = 1;
|
2003-03-24 15:41:42 +01:00
|
|
|
#ifdef WIN_NT
|
2004-06-08 15:41:08 +02:00
|
|
|
id = GetCurrentThreadId();
|
2003-03-24 15:41:42 +01:00
|
|
|
#endif
|
2004-06-08 15:41:08 +02:00
|
|
|
#ifdef SOLARIS_MT
|
|
|
|
id = thr_self();
|
2003-10-30 10:49:40 +01:00
|
|
|
#endif
|
2004-05-22 04:11:56 +02:00
|
|
|
#ifdef USE_POSIX_THREADS
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
/* The following is just a temp. decision.
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
#ifdef HP10
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
id = (FB_THREAD_ID) (pthread_self().field1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
id = (FB_THREAD_ID) pthread_self();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#endif /* HP10 */
|
2004-06-08 15:41:08 +02:00
|
|
|
#endif /* USE_POSIX_THREADS */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return id;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData* ThreadData::getSpecific(void)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ g e t _ s p e c i f i c
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* Gets thread specific data and returns
|
|
|
|
* a pointer to it.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2004-06-08 15:41:08 +02:00
|
|
|
return TLS_GET(tData);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::getSpecificData(void **t_data)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ g e t s p e c i f i c _ d a t a
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* return the previously stored t_data.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
*t_data = TLS_GET(tSpecific);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int THD_wlck_lock(WLCK_T* wlock, WLCK_type type)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ w l c k _ l o c k
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
switch(type)
|
|
|
|
{
|
|
|
|
case WLCK_read:
|
|
|
|
#ifdef DEBUG_THREAD
|
|
|
|
fprintf(stderr, "calling rwlock_rdlock %x\n", wlock);
|
|
|
|
#endif
|
|
|
|
wlock->rwLock.beginRead();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WLCK_write:
|
|
|
|
#ifdef DEBUG_THREAD
|
|
|
|
fprintf(stderr, "calling rwlock_wrlock %x\n", wlock);
|
|
|
|
#endif
|
|
|
|
wlock->rwLock.beginWrite();
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifdef DEV_BUILD
|
|
|
|
default:
|
|
|
|
fb_assert(false);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
wlock->type = type;
|
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
int THD_wlck_unlock(WLCK_T* wlock)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-06-08 15:41:08 +02:00
|
|
|
* T H D _ w l c k _ u n l o c k
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#ifdef DEBUG_THREAD
|
|
|
|
fprintf(stderr, "calling rwlock_unlock %x\n", wlock);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch(wlock->type)
|
|
|
|
{
|
|
|
|
case WLCK_read:
|
|
|
|
wlock->rwLock.endRead();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WLCK_write:
|
|
|
|
wlock->rwLock.endWrite();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::putSpecific()
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ p u t _ s p e c i f i c
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
threadDataPriorContext = TLS_GET(tData);
|
2004-06-08 15:41:08 +02:00
|
|
|
TLS_SET(tData, this);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::putSpecificData(void *t_data)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ p u t s p e c i f i c _ d a t a
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* Store the passed t_data
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
TLS_SET(tSpecific, t_data);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::restoreSpecific()
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e s t o r e _ s p e c i f i c
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData* current_context = getSpecific();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
TLS_SET(tData, current_context->threadDataPriorContext);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
int THD_rec_mutex_destroy(REC_MUTX_T * rec_mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e c _ m u t e x _ d e s t r o y ( S U P E R S E R V E R )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int THD_rec_mutex_init(REC_MUTX_T * rec_mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e c _ m u t e x _ i n i t ( S U P E R S E R V E R )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
rec_mutex->rec_mutx_id = 0;
|
|
|
|
rec_mutex->rec_mutx_count = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int THD_rec_mutex_lock(REC_MUTX_T * rec_mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e c _ m u t e x _ l o c k ( S U P E R S E R V E R )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
if (rec_mutex->rec_mutx_id == ThreadData::getId())
|
2001-05-23 15:26:42 +02:00
|
|
|
rec_mutex->rec_mutx_count++;
|
|
|
|
else {
|
|
|
|
if (ret = THD_mutex_lock(rec_mutex->rec_mutx_mtx))
|
|
|
|
return ret;
|
2004-08-30 20:11:08 +02:00
|
|
|
rec_mutex->rec_mutx_id = ThreadData::getId();
|
2001-05-23 15:26:42 +02:00
|
|
|
rec_mutex->rec_mutx_count = 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int THD_rec_mutex_unlock(REC_MUTX_T * rec_mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e c _ m u t e x _ u n l o c k ( S U P E R S E R V E R )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
if (rec_mutex->rec_mutx_id != ThreadData::getId())
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
rec_mutex->rec_mutx_count--;
|
|
|
|
|
|
|
|
if (rec_mutex->rec_mutx_count)
|
|
|
|
return 0;
|
|
|
|
else {
|
|
|
|
rec_mutex->rec_mutx_id = 0;
|
|
|
|
return THD_mutex_unlock(rec_mutex->rec_mutx_mtx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define THREAD_SUSPEND_DEFINED
|
|
|
|
int THD_resume(THD_T thread)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e s u m e ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Resume execution of a thread that has been
|
|
|
|
* suspended.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (ResumeThread(thread) == 0xFFFFFFFF)
|
|
|
|
return GetLastError();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int THD_suspend(THD_T thread)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ s u s p e n d ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Suspend execution of a thread.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (SuspendThread(thread) == 0xFFFFFFFF)
|
|
|
|
return GetLastError();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef THREAD_SUSPEND_DEFINED
|
|
|
|
int THD_resume(THD_T thread)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ r e s u m e ( G e n e r i c )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Resume execution of a thread that has been
|
|
|
|
* suspended.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int THD_suspend(THD_T thread)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ s u s p e n d ( G e n e r i c )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Suspend execution of a thread.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
void THD_sleep(ULONG milliseconds)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ s l e e p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Thread sleeps for requested number
|
|
|
|
* of milliseconds.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef WIN_NT
|
|
|
|
SleepEx(milliseconds, FALSE);
|
|
|
|
#else
|
2003-03-24 15:41:42 +01:00
|
|
|
|
|
|
|
#ifdef ANY_THREADING
|
2004-01-03 11:59:52 +01:00
|
|
|
event_t timer;
|
|
|
|
event_t* timer_ptr = &timer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ISC_event_init(&timer, 0, 0);
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG count = ISC_event_clear(&timer);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &timer_ptr, &count, milliseconds * 1000, NULL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_fini(&timer);
|
|
|
|
#else /* !ANY_THREADING */
|
|
|
|
int seconds;
|
|
|
|
|
|
|
|
/* Insure that process sleeps some amount of time. */
|
|
|
|
|
|
|
|
if (!(seconds = milliseconds / 1000))
|
|
|
|
++seconds;
|
|
|
|
|
|
|
|
/* Feedback unslept time due to premature wakeup from signals. */
|
|
|
|
|
|
|
|
while (seconds = sleep(seconds));
|
|
|
|
|
2003-03-24 15:41:42 +01:00
|
|
|
#endif /* !ANY_THREADING */
|
|
|
|
|
|
|
|
#endif /* !WIN_NT */
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void THD_yield(void)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T H D _ y i e l d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Thread relinquishes the processor.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef ANY_THREADING
|
|
|
|
|
2004-05-22 04:11:56 +02:00
|
|
|
#ifdef USE_POSIX_THREADS
|
2001-05-23 15:26:42 +02:00
|
|
|
/* use sched_yield() instead of pthread_yield(). Because pthread_yield()
|
|
|
|
is not part of the (final) POSIX 1003.1c standard. Several drafts of
|
|
|
|
the standard contained pthread_yield(), but then the POSIX guys
|
|
|
|
discovered it was redundant with sched_yield() and dropped it.
|
|
|
|
So, just use sched_yield() instead. POSIX systems on which
|
|
|
|
sched_yield() is available define _POSIX_PRIORITY_SCHEDULING
|
2001-07-12 07:46:06 +02:00
|
|
|
in <unistd.h>. Darwin defined _POSIX_THREAD_PRIORITY_SCHEDULING
|
|
|
|
instead of _POSIX_PRIORITY_SCHEDULING.
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
2001-07-12 07:46:06 +02:00
|
|
|
#if (defined _POSIX_PRIORITY_SCHEDULING || defined _POSIX_THREAD_PRIORITY_SCHEDULING)
|
2001-05-23 15:26:42 +02:00
|
|
|
sched_yield();
|
|
|
|
#else
|
|
|
|
pthread_yield();
|
|
|
|
#endif /* _POSIX_PRIORITY_SCHEDULING */
|
|
|
|
#endif
|
|
|
|
|
2004-06-15 11:02:14 +02:00
|
|
|
#ifdef SOLARIS_MT
|
2001-05-23 15:26:42 +02:00
|
|
|
thr_yield();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2003-05-14 10:09:47 +02:00
|
|
|
SleepEx(0, FALSE);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#endif /* ANY_THREADING */
|
|
|
|
}
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#ifdef THREAD_PSCHED
|
|
|
|
static THREAD_ENTRY_DECLARE threadStart(THREAD_ENTRY_PARAM arg) {
|
|
|
|
ThreadPriorityScheduler* tps =
|
|
|
|
reinterpret_cast<ThreadPriorityScheduler*>(arg);
|
|
|
|
try {
|
|
|
|
tps->run();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-06-08 15:41:08 +02:00
|
|
|
catch (...) {
|
|
|
|
tps->detach();
|
|
|
|
throw;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-06-08 15:41:08 +02:00
|
|
|
tps->detach();
|
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#define THREAD_ENTRYPOINT threadStart
|
|
|
|
#define THREAD_ARG tps
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#else //THREAD_PSCHED
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
// due to same names of parameters for various ThreadData::start(...),
|
2004-06-08 15:41:08 +02:00
|
|
|
// we may use common macro for various platforms
|
|
|
|
#define THREAD_ENTRYPOINT reinterpret_cast<THREAD_ENTRY_RETURN (THREAD_ENTRY_CALL *) (THREAD_ENTRY_PARAM)>(routine)
|
|
|
|
#define THREAD_ARG reinterpret_cast<THREAD_ENTRY_PARAM>(arg)
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#endif //THREAD_PSCHED
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef ANY_THREADING
|
2004-05-22 04:11:56 +02:00
|
|
|
#ifdef USE_POSIX_THREADS
|
2001-05-23 15:26:42 +02:00
|
|
|
#define START_THREAD
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::start(ThreadEntryPoint* routine,
|
2004-06-08 15:41:08 +02:00
|
|
|
void *arg,
|
|
|
|
int priority_arg,
|
|
|
|
int flags,
|
|
|
|
void *thd_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* 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;
|
|
|
|
|
2003-03-02 18:47:22 +01:00
|
|
|
#if ( !defined HP10 && !defined LINUX && !defined FREEBSD )
|
2004-06-08 15:41:08 +02:00
|
|
|
state = pthread_attr_init(&pattr);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (state)
|
2004-06-08 15:41:08 +02:00
|
|
|
Firebird::system_call_failed::raise("pthread_attr_init", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
// Do not make thread bound for superserver/client
|
2001-05-23 15:26:42 +02:00
|
|
|
#if (!defined (SUPERCLIENT) && !defined (SUPERSERVER))
|
|
|
|
pthread_attr_setscope(&pattr, PTHREAD_SCOPE_SYSTEM);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
pthread_attr_setdetachstate(&pattr, PTHREAD_CREATE_DETACHED);
|
2004-06-08 15:41:08 +02:00
|
|
|
state = pthread_create(&thread, &pattr, THREAD_ENTRYPOINT, THREAD_ARG);
|
2001-05-23 15:26:42 +02:00
|
|
|
pthread_attr_destroy(&pattr);
|
2004-06-08 15:41:08 +02:00
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_create", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#else
|
2003-03-02 18:47:22 +01:00
|
|
|
#if ( defined LINUX || defined FREEBSD )
|
2004-06-08 15:41:08 +02:00
|
|
|
|
|
|
|
if (state = pthread_create(&thread, NULL, THREAD_ENTRYPOINT, THREAD_ARG))
|
|
|
|
Firebird::system_call_failed::raise("pthread_create", state);
|
|
|
|
if (state = pthread_detach(thread))
|
|
|
|
Firebird::system_call_failed::raise("pthread_detach", state);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
2004-06-08 15:41:08 +02:00
|
|
|
|
|
|
|
long stack_size;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
state = pthread_attr_create(&pattr);
|
2004-06-08 15:41:08 +02:00
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_attr_create", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* The default HP's stack size is too small. HP's documentation
|
|
|
|
says it is "machine specific". My test showed it was less
|
|
|
|
than 64K. We definitly need more stack to be able to execute
|
|
|
|
concurrently many (at least 100) copies of the same request
|
|
|
|
(like, for example in case of recursive stored prcedure).
|
|
|
|
The following code sets threads stack size up to 256K if the
|
|
|
|
default stack size is less than this number
|
|
|
|
*/
|
|
|
|
stack_size = pthread_attr_getstacksize(pattr);
|
|
|
|
if (stack_size == -1)
|
2004-06-08 15:41:08 +02:00
|
|
|
Firebird::system_call_failed::raise("pthread_attr_getstacksize");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (stack_size < 0x40000L) {
|
|
|
|
state = pthread_attr_setstacksize(&pattr, 0x40000L);
|
2004-06-08 15:41:08 +02:00
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_attr_setstacksize", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* HP's Posix threads implementation does not support
|
|
|
|
bound attribute. It just a user level library.
|
|
|
|
*/
|
2004-06-08 15:41:08 +02:00
|
|
|
state = pthread_create(&thread, pattr, THREAD_ENTRYPOINT, THREAD_ARG);
|
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_create", state);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
state = pthread_detach(&thread);
|
2004-06-08 15:41:08 +02:00
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_detach", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
state = pthread_attr_delete(&pattr);
|
2004-06-08 15:41:08 +02:00
|
|
|
if (state)
|
|
|
|
Firebird::system_call_failed::raise("pthread_attr_delete", state);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#endif /* linux */
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
2004-05-22 04:11:56 +02:00
|
|
|
#endif /* USE_POSIX_THREADS */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef SOLARIS_MT
|
|
|
|
#define START_THREAD
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::start(ThreadEntryPoint* routine,
|
2004-06-08 15:41:08 +02:00
|
|
|
void *arg,
|
|
|
|
int priority_arg,
|
|
|
|
int flags,
|
|
|
|
void *thd_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t h r e a d _ s t a r t ( S o l a r i s )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a new thread. Return 0 if successful,
|
|
|
|
* status if not.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int rval;
|
|
|
|
thread_t thread_id;
|
|
|
|
sigset_t new_mask, orig_mask;
|
|
|
|
|
2003-08-28 15:16:03 +02:00
|
|
|
sigfillset(&new_mask);
|
|
|
|
sigdelset(&new_mask, SIGALRM);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (rval = thr_sigsetmask(SIG_SETMASK, &new_mask, &orig_mask))
|
2004-06-08 15:41:08 +02:00
|
|
|
Firebird::system_call_failed::raise("thr_sigsetmask", rval);
|
2001-05-23 15:26:42 +02:00
|
|
|
#if (defined SUPERCLIENT || defined SUPERSERVER)
|
2004-06-15 11:02:14 +02:00
|
|
|
rval = thr_create(NULL, 0, THREAD_ENTRYPOINT, THREAD_ARG, THR_DETACHED | THR_NEW_LWP,
|
|
|
|
&thread_id);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
rval =
|
2004-06-15 11:02:14 +02:00
|
|
|
thr_create(NULL, 0, THREAD_ENTRYPOINT, THREAD_ARG, THR_DETACHED | THR_BOUND,
|
|
|
|
&thread_id);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-08-28 15:16:03 +02:00
|
|
|
thr_sigsetmask(SIG_SETMASK, &orig_mask, NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
if (rval)
|
|
|
|
Firebird::system_call_failed::raise("thr_create", rval);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define START_THREAD
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::start(ThreadEntryPoint* routine,
|
2004-06-08 15:41:08 +02:00
|
|
|
void *arg,
|
|
|
|
int priority_arg,
|
|
|
|
int flags,
|
|
|
|
void *thd_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
HANDLE handle;
|
|
|
|
DWORD thread_id;
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2004-06-08 15:41:08 +02:00
|
|
|
#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 long real_handle =
|
|
|
|
_beginthreadex(NULL, 0, THREAD_ENTRYPOINT, THREAD_ARG, CREATE_SUSPENDED,
|
|
|
|
reinterpret_cast <unsigned *>(&thread_id));
|
|
|
|
if (!real_handle)
|
|
|
|
{
|
|
|
|
Firebird::system_call_failed::raise("_beginthreadex", GetLastError());
|
|
|
|
}
|
|
|
|
handle = reinterpret_cast<HANDLE>(real_handle);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
SetThreadPriority(handle, priority);
|
2004-06-08 15:41:08 +02:00
|
|
|
|
2002-11-22 09:48:13 +01:00
|
|
|
if (! (flags & THREAD_wait))
|
2004-06-08 15:41:08 +02:00
|
|
|
{
|
2002-11-22 09:48:13 +01:00
|
|
|
ResumeThread(handle);
|
2004-06-08 15:41:08 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (thd_id)
|
2004-06-08 15:41:08 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
*(HANDLE *) thd_id = handle;
|
2004-06-08 15:41:08 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2004-06-08 15:41:08 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
CloseHandle(handle);
|
2004-06-08 15:41:08 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef ANY_THREADING
|
|
|
|
#ifdef VMS
|
2004-05-22 04:11:56 +02:00
|
|
|
#ifndef USE_POSIX_THREADS
|
2001-05-23 15:26:42 +02:00
|
|
|
#define START_THREAD
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t h r e a d _ s t a r t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a new thread. Return 0 if successful,
|
|
|
|
* status if not. This routine is coded in macro.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef START_THREAD
|
2004-08-30 20:11:08 +02:00
|
|
|
void ThreadData::start(ThreadEntryPoint* routine,
|
2004-06-08 15:41:08 +02:00
|
|
|
void *arg,
|
|
|
|
int priority_arg,
|
|
|
|
int flags,
|
|
|
|
void *thd_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t h r e a d _ s t a r t ( G e n e r i c )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-06-08 15:41:08 +02:00
|
|
|
* Wrong attempt to start a new thread.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|