/* * PROGRAM: JRD Access Method * MODULE: thd.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 * */ #include "firebird.h" #include #include #include "../jrd/common.h" #include "../jrd/thd.h" #include "../jrd/isc.h" #include "../jrd/os/thd_priority.h" #include "../jrd/gds_proto.h" #include "../jrd/isc_s_proto.h" #include "../jrd/gdsassert.h" #include "../common/classes/fb_tls.h" #ifdef WIN_NT #include #include #endif #ifdef HAVE_UNISTD_H #include #endif #ifdef SOLARIS_MT #include #include #endif #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 * **************************************/ return 0; } 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 * **************************************/ if (rec_mutex->rec_mutx_id == ThreadData::getId()) { rec_mutex->rec_mutx_count++; } else { try { rec_mutex->rec_mutx_mtx->enter(); } catch (const Firebird::system_call_failed& e) { return e.getErrorCode(); } rec_mutex->rec_mutx_id = ThreadData::getId(); 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 * **************************************/ if (rec_mutex->rec_mutx_id != ThreadData::getId()) return FB_FAILURE; rec_mutex->rec_mutx_count--; if (rec_mutex->rec_mutx_count == 0) { rec_mutex->rec_mutx_id = 0; try { rec_mutex->rec_mutx_mtx->leave(); } catch (const Firebird::system_call_failed& e) { return e.getErrorCode(); } } return 0; } #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 #ifdef MULTI_THREAD event_t timer; event_t* timer_ptr = &timer; ISC_event_init(&timer, 0, 0); SLONG count = ISC_event_clear(&timer); ISC_event_wait(1, &timer_ptr, &count, milliseconds * 1000, NULL, 0); ISC_event_fini(&timer); #else // MULTI_THREAD 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)); #endif // MULTI_THREAD #endif // WIN_NT } void THD_yield(void) { /************************************** * * T H D _ y i e l d * ************************************** * * Functional description * Thread relinquishes the processor. * **************************************/ #ifdef MULTI_THREAD #ifdef USE_POSIX_THREADS /* 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 in . Darwin defined _POSIX_THREAD_PRIORITY_SCHEDULING instead of _POSIX_PRIORITY_SCHEDULING. */ #if (defined _POSIX_PRIORITY_SCHEDULING || defined _POSIX_THREAD_PRIORITY_SCHEDULING) sched_yield(); #else pthread_yield(); #endif // _POSIX_PRIORITY_SCHEDULING #endif #ifdef SOLARIS_MT thr_yield(); #endif #ifdef WIN_NT SleepEx(0, FALSE); #endif #endif // MULTI_THREAD }