2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: isc_sync.c
|
|
|
|
* DESCRIPTION: General purpose but non-user 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-02-16 03:50:01 +01:00
|
|
|
*
|
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "XENIX" port
|
2002-02-16 04:05:21 +01:00
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "DELTA" port
|
2002-02-16 04:27:33 +01:00
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "IMP" port
|
2002-02-16 03:50:01 +01:00
|
|
|
*
|
2002-02-23 23:15:24 +01:00
|
|
|
* 2002-02-23 Sean Leyne - Code Cleanup, removed old M88K and NCR3000 port
|
|
|
|
*
|
2002-10-28 05:42:54 +01:00
|
|
|
* 2002.10.27 Sean Leyne - Completed removal of obsolete "DG_X86" port
|
2002-10-28 05:44:59 +01:00
|
|
|
* 2002.10.27 Sean Leyne - Completed removal of obsolete "M88K" port
|
2002-10-28 05:42:54 +01:00
|
|
|
*
|
2002-10-29 03:45:09 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Completed removal of obsolete "DGUX" port
|
2002-10-29 04:17:45 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
|
2002-10-29 04:31:20 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "SGI" port
|
2002-10-29 03:45:09 +01:00
|
|
|
*
|
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
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef SHLIB_DEFS
|
|
|
|
#define LOCAL_SHLIB_DEFS
|
|
|
|
#endif
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/ib_stdio.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2003-03-04 04:42:23 +01:00
|
|
|
#ifdef SOLARIS
|
|
|
|
#ifndef DEV_BUILD
|
|
|
|
#define NDEBUG // Turn off assert() macros
|
|
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
|
|
#endif
|
|
|
|
|
2002-07-29 17:37:59 +02:00
|
|
|
#ifdef HAVE_SETJMP_H
|
2001-12-24 03:51:06 +01:00
|
|
|
#include <setjmp.h>
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef HP10
|
|
|
|
#include <sys/pstat.h>
|
|
|
|
#endif
|
|
|
|
|
2002-09-27 02:20:12 +02:00
|
|
|
#include "../jrd/jrd_time.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/common.h"
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "gen/codes.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/isc.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#include "../jrd/isc_i_proto.h"
|
|
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
#include "../jrd/file_params.h"
|
|
|
|
#include "../jrd/gdsassert.h"
|
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/sch_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
2001-12-24 03:51:06 +01:00
|
|
|
#include "../jrd/thd_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-04-02 13:22:30 +02:00
|
|
|
#if defined(SIG_RESTART) || defined(UNIX)
|
2001-05-23 15:26:42 +02:00
|
|
|
static USHORT inhibit_restart;
|
2003-04-02 13:22:30 +02:00
|
|
|
#endif
|
|
|
|
#ifndef REQUESTER
|
2001-05-23 15:26:42 +02:00
|
|
|
static int process_id;
|
2003-04-02 13:22:30 +02:00
|
|
|
#endif
|
|
|
|
#ifdef UNIX
|
2001-05-23 15:26:42 +02:00
|
|
|
static UCHAR *next_shared_memory;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* VMS Specific Stuff */
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
|
|
|
|
#include <rms.h>
|
|
|
|
#include <descrip.h>
|
|
|
|
#include <ssdef.h>
|
|
|
|
#include <jpidef.h>
|
|
|
|
#include <prvdef.h>
|
|
|
|
#include <secdef.h>
|
|
|
|
#include <lckdef.h>
|
|
|
|
#include "../jrd/lnmdef.h"
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
#include "../jrd/prv_m_bypass.h"
|
|
|
|
#endif /* of ifdef VMS */
|
|
|
|
|
|
|
|
|
|
|
|
/* Unix specific stuff */
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/file.h>
|
|
|
|
|
2002-11-04 10:04:18 +01:00
|
|
|
#ifdef HAVE_SIGNAL_H
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
2002-11-04 10:04:18 +01:00
|
|
|
|
|
|
|
#ifdef HAVE_SYS_SIGNAL_H
|
|
|
|
#include <sys/signal.h>
|
2002-11-01 14:21:17 +01:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#include <errno.h>
|
2002-08-22 12:48:26 +02:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <unistd.h>
|
2002-08-22 12:48:26 +02:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <sys/ipc.h>
|
|
|
|
#include <sys/shm.h>
|
|
|
|
#include <sys/sem.h>
|
|
|
|
|
|
|
|
#ifndef O_RDWR
|
|
|
|
#include <fcntl.h>
|
|
|
|
#endif
|
|
|
|
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <sys/mman.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define FTOK_KEY 15
|
|
|
|
#define PRIV 0666
|
|
|
|
#define LOCAL_SEMAPHORES 4
|
|
|
|
|
|
|
|
#ifndef GDS_RELAY
|
|
|
|
#define GDS_RELAY "/bin/gds_relay"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SHMEM_DELTA
|
|
|
|
#define SHMEM_DELTA (1 << 22)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SIGURG
|
|
|
|
#define SIGURG SIGINT
|
|
|
|
#endif
|
|
|
|
|
2002-11-21 11:20:07 +01:00
|
|
|
#ifndef HAVE_SEMUN
|
2001-05-23 15:26:42 +02:00
|
|
|
union semun {
|
|
|
|
int val;
|
|
|
|
struct semid_ds *buf;
|
|
|
|
ushort *array;
|
|
|
|
};
|
|
|
|
#endif
|
2002-11-21 11:20:07 +01:00
|
|
|
#endif /* UNIX */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-07-06 07:32:02 +02:00
|
|
|
#ifdef HAVE_SYS_PARAM_H
|
2002-07-02 11:49:19 +02:00
|
|
|
#include <sys/param.h>
|
2002-07-06 07:32:02 +02:00
|
|
|
#endif
|
|
|
|
|
2002-10-16 10:38:04 +02:00
|
|
|
#ifndef HAVE_GETPAGESIZE
|
|
|
|
static size_t getpagesize(void) {return PAGESIZE;}
|
|
|
|
#endif
|
2002-07-02 11:49:19 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Windows NT */
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
|
|
|
|
#include <process.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
static void error(STATUS *, TEXT *, STATUS);
|
2003-03-01 13:29:22 +01:00
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
static void alarm_handler(void);
|
2001-05-23 15:26:42 +02:00
|
|
|
static SLONG find_key(STATUS *, TEXT *);
|
|
|
|
static SLONG init_semaphores(STATUS *, SLONG, int);
|
2003-03-01 13:29:22 +01:00
|
|
|
static BOOLEAN semaphore_wait_isc_sync(int, int, int *);
|
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#if defined(UNIX) && defined(SUPERSERVER)
|
|
|
|
static void longjmp_sig_handler(int);
|
|
|
|
#endif // UNIX && SUPERSERVER
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
static int event_test(WAIT *);
|
|
|
|
static BOOLEAN mutex_test(MTX);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(WIN_NT)
|
2002-11-16 16:54:21 +01:00
|
|
|
static void make_object_name(TEXT *, const TEXT *, const TEXT *);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef sigvector
|
|
|
|
#ifndef hpux
|
|
|
|
#define sigvector sigvec
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SV_INTERRUPT
|
|
|
|
#define SV_INTERRUPT 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
#if defined FREEBSD || defined NETBSD || defined DARWIN
|
2001-05-23 15:26:42 +02:00
|
|
|
#define sigset signal
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SHLIB_DEFS
|
|
|
|
#define sprintf (*_libgds_sprintf)
|
|
|
|
#define strlen (*_libgds_strlen)
|
|
|
|
#define strcmp (*_libgds_strcmp)
|
|
|
|
#define strcpy (*_libgds_strcpy)
|
|
|
|
#define _iob (*_libgds__iob)
|
|
|
|
#define shmdt (*_libgds_shmdt)
|
|
|
|
#define ib_fprintf (*_libgds_fprintf)
|
|
|
|
#define errno (*_libgds_errno)
|
|
|
|
#define ib_fopen (*_libgds_fopen)
|
|
|
|
#define ib_fclose (*_libgds_fclose)
|
|
|
|
#define open (*_libgds_open)
|
|
|
|
#define semctl (*_libgds_semctl)
|
|
|
|
#define semop (*_libgds_semop)
|
|
|
|
#define umask (*_libgds_umask)
|
|
|
|
#define close (*_libgds_close)
|
|
|
|
#define _ctype (*_libgds__ctype)
|
|
|
|
#define sigvector (*_libgds_sigvec)
|
|
|
|
#define pause (*_libgds_pause)
|
|
|
|
#define lockf (*_libgds_lockf)
|
|
|
|
#define shmget (*_libgds_shmget)
|
|
|
|
#define shmat (*_libgds_shmat)
|
|
|
|
#define shmctl (*_libgds_shmctl)
|
|
|
|
#define ftok (*_libgds_ftok)
|
|
|
|
#define semget (*_libgds_semget)
|
|
|
|
#define sigset (*_libgds_sigset)
|
|
|
|
#define sbrk (*_libgds_sbrk)
|
|
|
|
#define setitimer (*_libgds_setitimer)
|
|
|
|
#define alarm (*_libgds_alarm)
|
|
|
|
#define sigprocmask (*_libgds_sigprocmask)
|
|
|
|
#define sigsuspend (*_libgds_sigsuspend)
|
|
|
|
#define sigaddset (*_libgds_sigaddset)
|
|
|
|
|
|
|
|
extern int sprintf();
|
|
|
|
extern int strlen();
|
|
|
|
extern int strcmp();
|
|
|
|
extern SCHAR *strcpy();
|
|
|
|
extern IB_FILE _iob[];
|
|
|
|
extern int shmdt();
|
|
|
|
extern int ib_fprintf();
|
|
|
|
extern int errno;
|
|
|
|
extern IB_FILE *ib_fopen();
|
|
|
|
extern int ib_fclose();
|
|
|
|
extern int open();
|
|
|
|
extern int semctl();
|
|
|
|
extern int semop();
|
|
|
|
extern mode_t umask();
|
|
|
|
extern int close();
|
|
|
|
extern SCHAR _ctype[];
|
|
|
|
extern int sigvector();
|
|
|
|
extern int pause();
|
|
|
|
extern int lockf();
|
|
|
|
extern int shmget();
|
|
|
|
extern int shmctl();
|
|
|
|
extern key_t ftok();
|
|
|
|
extern int semget();
|
|
|
|
extern void (*sigset()) ();
|
|
|
|
extern SCHAR *sbrk();
|
|
|
|
extern int setitimer();
|
|
|
|
extern int alarm();
|
|
|
|
extern SCHAR *shmat();
|
|
|
|
extern int sigprocmask();
|
|
|
|
extern int sigsuspend();
|
|
|
|
extern int sigaddset();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SIG_RESTART
|
|
|
|
BOOLEAN ISC_check_restart(void)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ c h e c k _ r e s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return a flag that indicats whether
|
|
|
|
* or not to restart an interrupted
|
|
|
|
* system call.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return (inhibit_restart) ? FALSE : TRUE;
|
|
|
|
}
|
|
|
|
#endif /* SIG_RESTART */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SOLARIS_MT
|
|
|
|
#define EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
for (; count > 0; --count, ++events, ++values)
|
|
|
|
if ((*events)->event_count >= *values) {
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: FALSE (eg something to report)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: TRUE (eg nothing happened yet)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG ret;
|
|
|
|
|
|
|
|
mutex_lock(event->event_mutex);
|
|
|
|
ret = event->event_count + 1;
|
|
|
|
mutex_unlock(event->event_mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ISC_event_fini(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ f i n i ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Discard an event object.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Inter-Process event's are destroyed only */
|
|
|
|
if (event->event_semid == -1) {
|
|
|
|
mutex_destroy(event->event_mutex);
|
|
|
|
cond_destroy(event->event_semnum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_init(EVENT event, int semid, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG key, n;
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
event->event_count = 0;
|
|
|
|
|
|
|
|
if (!semnum) {
|
|
|
|
/* Prepare an Inter-Thread event block */
|
|
|
|
event->event_semid = -1;
|
|
|
|
mutex_init(event->event_mutex, USYNC_THREAD, NULL);
|
|
|
|
cond_init(event->event_semnum, USYNC_THREAD, NULL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Prepare an Inter-Process event block */
|
|
|
|
event->event_semid = semid;
|
|
|
|
mutex_init(event->event_mutex, USYNC_PROCESS, NULL);
|
|
|
|
cond_init(event->event_semnum, USYNC_PROCESS, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* For Solaris, we use cond_broadcast rather than cond_signal so that
|
|
|
|
all waiters on the event are notified and awakened */
|
|
|
|
|
|
|
|
mutex_lock(event->event_mutex);
|
|
|
|
++event->event_count;
|
|
|
|
ret = cond_broadcast(event->event_semnum);
|
|
|
|
mutex_unlock(event->event_mutex);
|
|
|
|
if (ret)
|
|
|
|
gds__log("ISC_event_post: cond_broadcast failed with errno = %d",
|
|
|
|
ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_wait(SSHORT count,
|
|
|
|
EVENT* events,
|
|
|
|
SLONG* values,
|
|
|
|
SLONG micro_seconds,
|
2002-11-01 14:21:17 +01:00
|
|
|
FPTR_VOID timeout_handler,
|
2001-05-23 15:26:42 +02:00
|
|
|
void* handler_arg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event. If timeout limit specified, return
|
|
|
|
* anyway after the timeout even if no event has
|
|
|
|
* happened. If returning due to timeout, return
|
2002-11-14 09:33:08 +01:00
|
|
|
* FB_FAILURE else return FB_SUCCESS.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
timestruc_t timer;
|
|
|
|
|
|
|
|
/* While the API for ISC_event_wait allows for a list of events
|
|
|
|
we never actually make use of it. This implementation wont
|
|
|
|
support it anyway as Solaris doesn't provide a "wait for one
|
|
|
|
of a series of conditions" function */
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(count == 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If we're not blocked, the rest is a gross waste of time */
|
|
|
|
|
|
|
|
if (!ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Set up timers if a timeout period was specified. */
|
|
|
|
|
|
|
|
if (micro_seconds > 0) {
|
|
|
|
timer.tv_sec = time(NULL);
|
|
|
|
timer.tv_sec += micro_seconds / 1000000;
|
|
|
|
timer.tv_nsec = 1000 * (micro_seconds % 1000000);
|
|
|
|
}
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
int ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
mutex_lock((*events)->event_mutex);
|
|
|
|
for (;;) {
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Solaris cond_wait & cond_timedwait calls atomically release
|
|
|
|
the mutex and start a wait. The mutex is reacquired before the
|
|
|
|
call returns. */
|
|
|
|
|
|
|
|
if (micro_seconds > 0)
|
|
|
|
ret =
|
|
|
|
cond_timedwait((*events)->event_semnum,
|
|
|
|
(*events)->event_mutex, &timer);
|
|
|
|
else
|
|
|
|
ret = cond_wait((*events)->event_semnum, (*events)->event_mutex);
|
|
|
|
if (micro_seconds > 0 && (ret == ETIME)) {
|
|
|
|
|
|
|
|
/* The timer expired - see if the event occured and return
|
2002-11-14 09:33:08 +01:00
|
|
|
FB_SUCCESS or FB_FAILURE accordingly. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock((*events)->event_mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif /* SOLARIS_MT */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef POSIX_THREADS
|
|
|
|
#define EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
for (; count > 0; --count, ++events, ++values)
|
|
|
|
if ((*events)->event_count >= *values) {
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: FALSE (eg something to report)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: TRUE (eg nothing happened yet)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG ret;
|
|
|
|
|
|
|
|
pthread_mutex_lock(event->event_mutex);
|
|
|
|
ret = event->event_count + 1;
|
|
|
|
pthread_mutex_unlock(event->event_mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ISC_event_fini(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ f i n i ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Discard an event object.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Inter-Thread event's are destroyed only */
|
|
|
|
if (event->event_semid == -1) {
|
|
|
|
pthread_mutex_destroy(event->event_mutex);
|
|
|
|
pthread_cond_destroy(event->event_semnum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_init(EVENT event, int semid, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG key, n;
|
|
|
|
union semun arg;
|
|
|
|
pthread_mutexattr_t mattr;
|
|
|
|
pthread_condattr_t cattr;
|
|
|
|
|
|
|
|
event->event_count = 0;
|
|
|
|
|
|
|
|
if (!semnum) {
|
|
|
|
/* Prepare an Inter-Thread event block */
|
|
|
|
event->event_semid = -1;
|
|
|
|
|
2002-02-16 04:05:21 +01:00
|
|
|
/* Default attribute objects initialize sync. primitives
|
2001-05-23 15:26:42 +02:00
|
|
|
to be used to sync thread within one process only.
|
|
|
|
*/
|
|
|
|
#ifdef HP10
|
|
|
|
pthread_mutex_init(event->event_mutex, pthread_mutexattr_default);
|
|
|
|
pthread_cond_init(event->event_semnum, pthread_condattr_default);
|
|
|
|
#else
|
|
|
|
pthread_mutex_init(event->event_mutex, NULL);
|
|
|
|
pthread_cond_init(event->event_semnum, NULL);
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Prepare an Inter-Process event block */
|
|
|
|
event->event_semid = semid;
|
|
|
|
|
|
|
|
/* NOTE: HP's Posix threads implementation does not support thread
|
|
|
|
synchronization in different processes. Thus the following
|
|
|
|
fragment is just a temp. decision we could be use for super-
|
|
|
|
server (until we are to implement local IPC using shared
|
|
|
|
memory in which case we need interprocess thread sync.
|
|
|
|
*/
|
|
|
|
#ifdef HP10
|
|
|
|
pthread_mutex_init(event->event_mutex, pthread_mutexattr_default);
|
|
|
|
pthread_cond_init(event->event_semnum, pthread_condattr_default);
|
|
|
|
#else
|
2002-07-02 11:49:19 +02:00
|
|
|
/* RITTER - added HP11 to the preprocessor condition below */
|
2003-03-02 18:47:22 +01:00
|
|
|
#if (defined LINUX || defined DARWIN || defined HP11 || defined FREEBSD)
|
2001-05-23 15:26:42 +02:00
|
|
|
pthread_mutex_init(event->event_mutex, NULL);
|
|
|
|
pthread_cond_init(event->event_semnum, NULL);
|
|
|
|
#else
|
|
|
|
pthread_mutexattr_init(&mattr);
|
|
|
|
pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED);
|
|
|
|
pthread_mutex_init(event->event_mutex, &mattr);
|
|
|
|
pthread_mutexattr_destroy(&mattr);
|
|
|
|
|
|
|
|
pthread_condattr_init(&cattr);
|
|
|
|
pthread_condattr_setpshared(&cattr, PTHREAD_PROCESS_SHARED);
|
|
|
|
pthread_cond_init(event->event_semnum, &cattr);
|
|
|
|
pthread_condattr_destroy(&cattr);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
pthread_mutex_lock(event->event_mutex);
|
|
|
|
++event->event_count;
|
|
|
|
ret = pthread_cond_broadcast(event->event_semnum);
|
|
|
|
pthread_mutex_unlock(event->event_mutex);
|
|
|
|
if (ret)
|
|
|
|
#ifdef HP10
|
|
|
|
|
|
|
|
{
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(ret == -1);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log
|
|
|
|
("ISC_event_post: pthread_cond_broadcast failed with errno = %d",
|
|
|
|
errno);
|
|
|
|
return errno;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
gds__log
|
|
|
|
("ISC_event_post: pthread_cond_broadcast failed with errno = %d",
|
|
|
|
ret);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_wait(
|
|
|
|
SSHORT count,
|
|
|
|
EVENT * events,
|
|
|
|
SLONG * values,
|
|
|
|
SLONG micro_seconds,
|
2001-12-24 03:51:06 +01:00
|
|
|
FPTR_VOID timeout_handler, void *handler_arg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event. If timeout limit specified, return
|
|
|
|
* anyway after the timeout even if no event has
|
|
|
|
* happened. If returning due to timeout, return
|
2002-11-14 09:33:08 +01:00
|
|
|
* FB_FAILURE else return FB_SUCCESS.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
struct timespec timer;
|
|
|
|
|
|
|
|
/* While the API for ISC_event_wait allows for a list of events
|
|
|
|
we never actually make use of it. This implementation wont
|
|
|
|
support it anyway as Solaris doesn't provide a "wait for one
|
|
|
|
of a series of conditions" function */
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(count == 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If we're not blocked, the rest is a gross waste of time */
|
|
|
|
|
|
|
|
if (!ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Set up timers if a timeout period was specified. */
|
|
|
|
|
|
|
|
if (micro_seconds > 0) {
|
|
|
|
timer.tv_sec = time(NULL);
|
|
|
|
timer.tv_sec += micro_seconds / 1000000;
|
|
|
|
timer.tv_nsec = 1000 * (micro_seconds % 1000000);
|
|
|
|
}
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
pthread_mutex_lock((*events)->event_mutex);
|
|
|
|
for (;;) {
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The Posix pthread_cond_wait & pthread_cond_timedwait calls
|
|
|
|
atomically release the mutex and start a wait.
|
|
|
|
The mutex is reacquired before the call returns.
|
|
|
|
*/
|
|
|
|
if (micro_seconds > 0)
|
|
|
|
ret =
|
|
|
|
pthread_cond_timedwait((*events)->event_semnum,
|
|
|
|
(*events)->event_mutex, &timer);
|
|
|
|
else
|
|
|
|
ret =
|
|
|
|
pthread_cond_wait((*events)->event_semnum,
|
|
|
|
(*events)->event_mutex);
|
|
|
|
|
|
|
|
#ifdef HP10
|
|
|
|
if (micro_seconds > 0 && (ret == -1) && (errno == EAGAIN))
|
|
|
|
#else
|
2002-07-02 11:49:19 +02:00
|
|
|
/* RITTER - added HP11 to the preprocessor condition below */
|
2003-03-02 18:47:22 +01:00
|
|
|
#if (defined LINUX || defined DARWIN || defined HP11 || defined FREEBSD)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (micro_seconds > 0 && (ret == ETIMEDOUT))
|
|
|
|
#else
|
|
|
|
if (micro_seconds > 0 && (ret == ETIME))
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
|
|
|
|
/* The timer expired - see if the event occured and return
|
2002-11-14 09:33:08 +01:00
|
|
|
FB_SUCCESS or FB_FAILURE accordingly. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock((*events)->event_mutex);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif /* POSIX_THREADS */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
#ifndef EVENTS
|
|
|
|
#define EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
for (; count > 0; --count, ++events, ++values)
|
|
|
|
if ((*events)->event_count >= *values) {
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: FALSE (eg something to report)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-02-10 01:03:54 +01:00
|
|
|
#ifdef DEBUG_ISC_SYNC
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_printf("ISC_event_blocked: TRUE (eg nothing happened yet)\n");
|
|
|
|
ib_fflush(ib_stdout);
|
|
|
|
#endif
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
if (event->event_semid != -1) {
|
|
|
|
arg.val = 1;
|
|
|
|
ret = semctl(event->event_semid, event->event_semnum, SETVAL, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (event->event_count + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ISC_event_fini(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ f i n i ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Discard an event object.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_init(EVENT event, int semid, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-12-10 12:53:53 +01:00
|
|
|
SLONG n;
|
2001-05-23 15:26:42 +02:00
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
event->event_count = 0;
|
|
|
|
|
|
|
|
if (!semnum) {
|
|
|
|
event->event_semid = -1;
|
|
|
|
event->event_semnum = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
event->event_semid = semid;
|
|
|
|
event->event_semnum = semnum;
|
|
|
|
arg.val = 0;
|
|
|
|
n = semctl(semid, semnum, SETVAL, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
++event->event_count;
|
|
|
|
|
|
|
|
while (event->event_semid != -1) {
|
|
|
|
arg.val = 0;
|
|
|
|
ret = semctl(event->event_semid, event->event_semnum, SETVAL, arg);
|
|
|
|
if (ret != -1)
|
|
|
|
return 0;
|
|
|
|
if (!SYSCALL_INTERRUPTED(errno)) {
|
|
|
|
gds__log("ISC_event_post: semctl failed with errno = %d", errno);
|
|
|
|
return errno;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_wait(
|
|
|
|
SSHORT count,
|
|
|
|
EVENT * events,
|
|
|
|
SLONG * values,
|
|
|
|
SLONG micro_seconds,
|
2001-07-12 07:46:06 +02:00
|
|
|
FPTR_VOID timeout_handler, void *handler_arg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event. If timeout limit specified, return
|
|
|
|
* anyway after the timeout even if no event has
|
|
|
|
* happened. If returning due to timeout, return
|
2002-11-14 09:33:08 +01:00
|
|
|
* FB_FAILURE else return FB_SUCCESS.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int ret;
|
|
|
|
sigset_t mask, oldmask;
|
|
|
|
EVENT *event;
|
|
|
|
int semid, i;
|
|
|
|
int semnums[16], *semnum;
|
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
SLONG user_timer;
|
|
|
|
void *user_handler;
|
|
|
|
#else
|
|
|
|
struct itimerval user_timer;
|
2002-08-22 12:48:26 +02:00
|
|
|
#ifndef HAVE_SIGACTION
|
2001-05-23 15:26:42 +02:00
|
|
|
struct sigvec user_handler;
|
|
|
|
#else
|
|
|
|
struct sigaction user_handler;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* If we're not blocked, the rest is a gross waste of time */
|
|
|
|
|
|
|
|
if (!ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If this is a local semaphore, don't sweat the semaphore non-sense */
|
|
|
|
|
|
|
|
if ((*events)->event_semid == -1) {
|
|
|
|
++inhibit_restart;
|
|
|
|
sigprocmask(SIG_BLOCK, NULL, &oldmask);
|
|
|
|
mask = oldmask;
|
|
|
|
sigaddset(&mask, SIGUSR1);
|
|
|
|
sigaddset(&mask, SIGUSR2);
|
|
|
|
sigaddset(&mask, SIGURG);
|
|
|
|
sigprocmask(SIG_BLOCK, &mask, NULL);
|
|
|
|
for (;;) {
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
|
|
|
--inhibit_restart;
|
|
|
|
sigprocmask(SIG_SETMASK, &oldmask, NULL);
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
sigsuspend(&oldmask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only the internal event work is available in the SHRLIB version of pipe server
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Set up for a semaphore operation */
|
|
|
|
|
|
|
|
semid = (int) (*events)->event_semid;
|
|
|
|
|
|
|
|
/* Collect the semaphore numbers in an array */
|
|
|
|
|
|
|
|
for (i = 0, event = events, semnum = semnums; i < count; i++)
|
|
|
|
*semnum++ = (*event++)->event_semnum;
|
|
|
|
|
|
|
|
/* Set up timers if a timeout period was specified. */
|
|
|
|
|
|
|
|
if (micro_seconds > 0) {
|
|
|
|
if (!timeout_handler)
|
|
|
|
timeout_handler = alarm_handler;
|
|
|
|
|
|
|
|
ISC_set_timer(micro_seconds, timeout_handler, handler_arg,
|
2001-07-12 07:46:06 +02:00
|
|
|
(SLONG*)&user_timer, (void**)&user_handler);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Go into wait loop */
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
|
|
|
if (micro_seconds <= 0)
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
2001-07-12 07:46:06 +02:00
|
|
|
(void) semaphore_wait_isc_sync(count, semid, semnums);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (micro_seconds > 0) {
|
2002-11-14 09:33:08 +01:00
|
|
|
/* semaphore_wait_isc_sync() routine may return FB_SUCCESS if our timeout
|
2002-02-16 04:05:21 +01:00
|
|
|
handler poked the semaphore. So make sure that the event
|
2001-05-23 15:26:42 +02:00
|
|
|
actually happened. If it didn't, indicate failure. */
|
|
|
|
|
|
|
|
if (ISC_event_blocked(count, events, values))
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2002-11-14 09:33:08 +01:00
|
|
|
ret = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cancel the handler. We only get here if a timeout was specified. */
|
|
|
|
|
2001-07-12 07:46:06 +02:00
|
|
|
ISC_reset_timer(timeout_handler, handler_arg, (SLONG*)&user_timer, (void**)&user_handler);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif /* EVENTS */
|
|
|
|
#endif /* UNIX */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
#define EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
for (; count; --count, events++, values++)
|
|
|
|
if ((*events)->event_count >= *values)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return event->event_count + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_init(EVENT event, int semid, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
gds__wake_init();
|
|
|
|
event->event_count = 0;
|
|
|
|
event->event_pid = getpid();
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int status;
|
|
|
|
|
|
|
|
++event->event_count;
|
|
|
|
ISC_wake(event->event_pid);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_wait(
|
|
|
|
SSHORT count,
|
|
|
|
EVENT * events,
|
|
|
|
SLONG * values,
|
|
|
|
SLONG micro_seconds,
|
|
|
|
void (*timeout_handler) (), void *handler_arg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
WAIT wait;
|
|
|
|
|
|
|
|
if (!ISC_event_blocked(count, events, values))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
wait.wait_count = count;
|
|
|
|
wait.wait_events = events;
|
|
|
|
wait.wait_values = values;
|
|
|
|
gds__thread_wait(event_test, &wait);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
|
|
|
|
#define EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
for (; count > 0; --count, ++events, ++values)
|
2002-11-16 16:54:21 +01:00
|
|
|
{
|
|
|
|
EVENT pEvent = *events;
|
|
|
|
if (pEvent->event_shared) {
|
|
|
|
pEvent = pEvent->event_shared;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-11-16 16:54:21 +01:00
|
|
|
if (pEvent->event_count >= *values) {
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-11-16 16:54:21 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG DLL_EXPORT ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
ResetEvent((HANDLE) event->event_handle);
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
EVENT pEvent = event;
|
|
|
|
if (pEvent->event_shared) {
|
|
|
|
pEvent = pEvent->event_shared;
|
|
|
|
}
|
|
|
|
return pEvent->event_count + 1;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ISC_event_fini(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ f i n i ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Discard an event object.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
CloseHandle((HANDLE) event->event_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_init(EVENT event, int type, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
event->event_pid = process_id = getpid();
|
|
|
|
event->event_count = 0;
|
|
|
|
event->event_type = type;
|
|
|
|
event->event_shared = NULL;
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
event->event_handle = ISC_make_signal(TRUE, TRUE, process_id, type);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
return (event->event_handle) ? TRUE : FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_event_init_shared(
|
2002-11-16 16:54:21 +01:00
|
|
|
EVENT lcl_event,
|
|
|
|
int type,
|
|
|
|
TEXT * name,
|
|
|
|
EVENT shr_event,
|
|
|
|
USHORT init_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t _ s h a r e d ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT event_name[MAXPATHLEN], type_name[16];
|
|
|
|
|
|
|
|
lcl_event->event_pid = process_id = getpid();
|
|
|
|
lcl_event->event_count = 0;
|
|
|
|
lcl_event->event_type = type;
|
|
|
|
lcl_event->event_shared = shr_event;
|
|
|
|
|
|
|
|
sprintf(type_name, "_event%d", type);
|
|
|
|
make_object_name(event_name, name, type_name);
|
|
|
|
if (!
|
|
|
|
(lcl_event->event_handle =
|
|
|
|
CreateEvent(ISC_get_security_desc(), TRUE, FALSE,
|
|
|
|
event_name))) return FALSE;
|
|
|
|
|
|
|
|
if (init_flag) {
|
|
|
|
shr_event->event_pid = 0;
|
|
|
|
shr_event->event_count = 0;
|
|
|
|
shr_event->event_type = type;
|
|
|
|
shr_event->event_handle = NULL;
|
|
|
|
shr_event->event_shared = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (!event->event_shared)
|
|
|
|
++event->event_count;
|
|
|
|
else
|
|
|
|
++event->event_shared->event_count;
|
|
|
|
|
|
|
|
if (event->event_pid != process_id)
|
|
|
|
ISC_kill(event->event_pid, event->event_type, event->event_handle);
|
|
|
|
else
|
|
|
|
SetEvent((HANDLE) event->event_handle);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_wait(
|
2002-11-16 16:54:21 +01:00
|
|
|
SSHORT count,
|
|
|
|
EVENT * events,
|
|
|
|
SLONG * values,
|
|
|
|
SLONG micro_seconds,
|
|
|
|
void (*timeout_handler) (),
|
|
|
|
void *handler_arg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
EVENT *ptr, *end;
|
|
|
|
HANDLE handles[16], *handle_ptr;
|
|
|
|
|
|
|
|
/* If we're not blocked, the rest is a gross waste of time */
|
|
|
|
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ptr = events, end = events + count, handle_ptr = handles; ptr < end;) {
|
|
|
|
*handle_ptr++ = (*ptr++)->event_handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Go into wait loop */
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
DWORD timeout = (micro_seconds > 0) ? micro_seconds / 1000 : INFINITE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
if (!ISC_event_blocked(count, events, values)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const DWORD status =
|
|
|
|
WaitForMultipleObjects((DWORD) count, handles, TRUE, timeout);
|
|
|
|
|
2003-04-03 15:52:42 +02:00
|
|
|
if (!((status >= WAIT_OBJECT_0) && (status < WAIT_OBJECT_0 + (DWORD) count))) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // WIN_NT
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
#ifndef EVENTS
|
|
|
|
int ISC_event_blocked(USHORT count, EVENT * events, SLONG * values)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ b l o c k e d ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If a wait would block, return TRUE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG DLL_EXPORT ISC_event_clear(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ c l e a r ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear an event preparatory to waiting on it. The order of
|
|
|
|
* battle for event synchronization is:
|
|
|
|
*
|
|
|
|
* 1. Clear event.
|
|
|
|
* 2. Test data structure for event already completed
|
|
|
|
* 3. Wait on event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0L;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_init(EVENT event, int semid, int semnum)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ i n i t ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare an event object for use. Return FALSE if not
|
|
|
|
* supported.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_post(EVENT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ p o s t ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post an event to wake somebody else up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_event_wait(
|
|
|
|
SSHORT count,
|
|
|
|
EVENT * events,
|
|
|
|
SLONG * values,
|
|
|
|
SLONG micro_seconds,
|
|
|
|
void (*timeout_handler) (), void *handler_arg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ e v e n t _ w a i t ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait on an event.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#ifdef UNIX
|
|
|
|
void ISC_exception_post(ULONG sig_num, TEXT * err_msg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 04:05:21 +01:00
|
|
|
* I S C _ e x c e p t i o n _ p o s t ( U N I X )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-02-16 04:05:21 +01:00
|
|
|
* When we got a sync exception, fomulate the error code
|
2001-05-23 15:26:42 +02:00
|
|
|
* write it to the log file, and abort.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT *log_msg;
|
|
|
|
|
|
|
|
if (!SCH_thread_enter_check())
|
|
|
|
THREAD_ENTER;
|
|
|
|
|
|
|
|
if (err_msg)
|
|
|
|
log_msg = (TEXT *) gds__alloc(strlen(err_msg) + 256);
|
|
|
|
|
|
|
|
switch (sig_num) {
|
|
|
|
case SIGSEGV:
|
|
|
|
sprintf(log_msg, "%s Segmentation Fault.\n"
|
|
|
|
"\t\tThe code attempted to access memory\n"
|
|
|
|
"\t\twithout privilege to do so.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case SIGBUS:
|
|
|
|
sprintf(log_msg, "%s Bus Error.\n"
|
|
|
|
"\t\tThe code caused a system bus error.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case SIGILL:
|
|
|
|
|
|
|
|
sprintf(log_msg, "%s Illegal Instruction.\n"
|
|
|
|
"\t\tThe code attempted to perfrom an\n"
|
|
|
|
"\t\tillegal operation."
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGFPE:
|
|
|
|
sprintf(log_msg, "%s Floating Point Error.\n"
|
|
|
|
"\t\tThe code caused an arithmetic exception\n"
|
|
|
|
"\t\tor floating point exception."
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
sprintf(log_msg, "%s Unknown Exception.\n"
|
|
|
|
"\t\tException number %d."
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg, sig_num);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err_msg) {
|
|
|
|
gds__log(log_msg);
|
|
|
|
gds__free(log_msg);
|
|
|
|
}
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
#endif /* UNIX */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
ULONG ISC_exception_post(ULONG except_code, TEXT * err_msg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 04:05:21 +01:00
|
|
|
* I S C _ e x c e p t i o n _ p o s t ( W I N _ N T )
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-02-16 04:05:21 +01:00
|
|
|
* When we got a sync exception, fomulate the error code
|
2001-05-23 15:26:42 +02:00
|
|
|
* write it to the log file, and abort. Note: We can not
|
|
|
|
* actually call "abort" since in windows this will cause
|
|
|
|
* a dialog to appear stating the obvious! Since on NT we
|
|
|
|
* would not get a core file, there is actually no difference
|
2002-02-16 04:05:21 +01:00
|
|
|
* between abort() and exit(3).
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-16 16:54:21 +01:00
|
|
|
ULONG result;
|
|
|
|
bool is_critical = true;
|
|
|
|
|
|
|
|
if (!SCH_thread_enter_check ())
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
2002-11-16 16:54:21 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
|
|
|
|
|
|
|
if (!err_msg)
|
|
|
|
{
|
|
|
|
err_msg = "";
|
|
|
|
}
|
|
|
|
|
2003-02-12 19:47:00 +01:00
|
|
|
TEXT* log_msg = (TEXT*) gds__alloc(strlen(err_msg) + 256);
|
|
|
|
log_msg[0] = '\0';
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (except_code) {
|
|
|
|
case EXCEPTION_ACCESS_VIOLATION:
|
|
|
|
sprintf(log_msg, "%s Access violation.\n"
|
|
|
|
"\t\tThe code attempted to access a virtual\n"
|
|
|
|
"\t\taddress without privilege to do so.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_DATATYPE_MISALIGNMENT:
|
|
|
|
sprintf(log_msg, "%s Datatype misalignment.\n"
|
|
|
|
"\t\tThe attempted to read or write a value\n"
|
|
|
|
"\t\tthat was not stored on a memory boundary.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_ARRAY_BOUNDS_EXCEEDED:
|
|
|
|
sprintf(log_msg, "%s Array bounds exceeded.\n"
|
|
|
|
"\t\tThe code attempted to access an array\n"
|
|
|
|
"\t\telement that is out of bounds.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_DENORMAL_OPERAND:
|
|
|
|
sprintf(log_msg, "%s Float denormal operand.\n"
|
|
|
|
"\t\tOne of the floating-point operands is too\n"
|
|
|
|
"\t\tsmall to represent as a standard floating-point\n"
|
|
|
|
"\t\tvalue.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_DIVIDE_BY_ZERO:
|
|
|
|
sprintf(log_msg, "%s Floating-point divide by zero.\n"
|
|
|
|
"\t\tThe code attempted to divide a floating-point\n"
|
|
|
|
"\t\tvalue by a floating-point divisor of zero.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_INEXACT_RESULT:
|
|
|
|
sprintf(log_msg, "%s Floating-point inexact result.\n"
|
|
|
|
"\t\tThe result of a floating-point operation cannot\n"
|
|
|
|
"\t\tbe represented exactly as a decimal fraction.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_INVALID_OPERATION:
|
|
|
|
sprintf(log_msg, "%s Floating-point invalid operand.\n"
|
|
|
|
"\t\tAn indeterminant error occurred during a\n"
|
|
|
|
"\t\tfloating-point operation.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_OVERFLOW:
|
|
|
|
sprintf(log_msg, "%s Floating-point overflow.\n"
|
|
|
|
"\t\tThe exponent of a floating-point operation\n"
|
|
|
|
"\t\tis greater than the magnitude allowed.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_STACK_CHECK:
|
|
|
|
sprintf(log_msg, "%s Floating-point stack check.\n"
|
|
|
|
"\t\tThe stack overflowed or underflowed as the\n"
|
|
|
|
"result of a floating-point operation.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_FLT_UNDERFLOW:
|
|
|
|
sprintf(log_msg, "%s Floating-point underflow.\n"
|
|
|
|
"\t\tThe exponent of a floating-point operation\n"
|
|
|
|
"\t\tis less than the magnitude allowed.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_INT_DIVIDE_BY_ZERO:
|
|
|
|
sprintf(log_msg, "%s Integer divide by zero.\n"
|
|
|
|
"\t\tThe code attempted to divide an integer value\n"
|
|
|
|
"\t\tby an integer divisor of zero.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_INT_OVERFLOW:
|
|
|
|
sprintf(log_msg, "%s Interger overflow.\n"
|
|
|
|
"\t\tThe result of an integer operation caused the\n"
|
|
|
|
"\t\tmost significant bit of the result to carry.\n"
|
2002-07-02 11:49:19 +02:00
|
|
|
"\tThis exception will cause the Firebird server\n"
|
2001-05-23 15:26:42 +02:00
|
|
|
"\tto terminate abnormally.", err_msg);
|
|
|
|
break;
|
|
|
|
case EXCEPTION_STACK_OVERFLOW:
|
|
|
|
ERR_post(isc_exception_stack_overflow, 0);
|
|
|
|
/* This will never be called, but to be safe it's here */
|
2002-11-16 16:54:21 +01:00
|
|
|
result = EXCEPTION_CONTINUE_EXECUTION;
|
|
|
|
is_critical = false;
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case EXCEPTION_BREAKPOINT:
|
|
|
|
case EXCEPTION_SINGLE_STEP:
|
|
|
|
case EXCEPTION_NONCONTINUABLE_EXCEPTION:
|
|
|
|
case EXCEPTION_INVALID_DISPOSITION:
|
|
|
|
case EXCEPTION_PRIV_INSTRUCTION:
|
|
|
|
case EXCEPTION_IN_PAGE_ERROR:
|
|
|
|
case EXCEPTION_ILLEGAL_INSTRUCTION:
|
|
|
|
case EXCEPTION_GUARD_PAGE:
|
|
|
|
/* Pass these exception on to someone else,
|
|
|
|
probably the OS or the debugger, since there
|
|
|
|
isn't a dam thing we can do with them */
|
2002-11-16 16:54:21 +01:00
|
|
|
result = EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
is_critical = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2002-11-16 16:54:21 +01:00
|
|
|
default:
|
|
|
|
/* If we've catched our own software exception,
|
|
|
|
continue rewinding the stack to properly handle it
|
|
|
|
and deliver an error information to the client side */
|
|
|
|
if (tdbb->tdbb_status_vector[0] == 1 && tdbb->tdbb_status_vector[1] > 0)
|
|
|
|
{
|
|
|
|
result = EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
is_critical = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sprintf (log_msg, "%s An exception occurred that does\n"
|
|
|
|
"\t\tnot have a description. Exception number %X.\n"
|
|
|
|
"\tThis exception will cause the Firebird server\n"
|
|
|
|
"\tto terminate abnormally.", err_msg, except_code);
|
|
|
|
}
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-11-16 16:54:21 +01:00
|
|
|
|
2003-03-12 14:20:23 +01:00
|
|
|
if (is_critical)
|
|
|
|
{
|
|
|
|
gds__log(log_msg);
|
|
|
|
}
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
gds__free(log_msg);
|
|
|
|
|
|
|
|
if (is_critical)
|
|
|
|
{
|
|
|
|
exit(3);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return result;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* WIN_NT */
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
void *ISC_make_signal(
|
2002-11-16 16:54:21 +01:00
|
|
|
BOOLEAN create_flag,
|
|
|
|
BOOLEAN manual_reset,
|
|
|
|
int process_id,
|
|
|
|
int signal_number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a k e _ s i g n a l ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create or open a Windows/NT event.
|
|
|
|
* Use the signal number and process id
|
|
|
|
* in naming the object.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (!signal_number)
|
|
|
|
return CreateEvent(NULL, manual_reset, FALSE, NULL);
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
TEXT event_name[64];
|
|
|
|
sprintf(event_name, "_interbase_process%u_signal%d", process_id, signal_number);
|
|
|
|
|
|
|
|
HANDLE hEvent;
|
|
|
|
if (create_flag) {
|
|
|
|
hEvent = CreateEvent(ISC_get_security_desc(), manual_reset, FALSE, event_name);
|
|
|
|
} else {
|
|
|
|
hEvent = OpenEvent(EVENT_ALL_ACCESS, TRUE, event_name);
|
|
|
|
}
|
|
|
|
return hEvent;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
#define ISC_MAP_FILE_DEFINED
|
|
|
|
UCHAR *ISC_map_file(STATUS * status_vector,
|
|
|
|
TEXT * filename,
|
|
|
|
void (*init_routine) (void *, struct sh_mem *, int),
|
|
|
|
void *init_arg, SLONG length, SH_MEM shmem_data)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a p _ f i l e ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map a given file. If we are the first (i.e. only)
|
|
|
|
* process to map the file, call a given initialization
|
|
|
|
* routine (if given) or punt (leaving the file unmapped).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG inadr[2], retadr[2], flags;
|
|
|
|
TEXT section[64], *p, *q, expanded_filename[MAXPATHLEN], temp[MAXPATHLEN],
|
|
|
|
hostname[64];
|
|
|
|
STATUS status;
|
|
|
|
struct FAB fab;
|
|
|
|
struct XABPRO xab;
|
|
|
|
struct dsc$descriptor_s desc;
|
|
|
|
|
|
|
|
if (length < 0)
|
|
|
|
length = -length;
|
|
|
|
|
|
|
|
if (length == 0) {
|
|
|
|
/* Must be able to handle case where zero length passed in. */
|
|
|
|
|
|
|
|
ib_fprintf(ib_stderr, "Unimplemented feature in ISC_map_file.\n");
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
gds__prefix(temp, filename);
|
|
|
|
sprintf(expanded_filename, temp,
|
|
|
|
ISC_get_host(hostname, sizeof(hostname)));
|
|
|
|
|
|
|
|
/* Find section name */
|
|
|
|
|
|
|
|
q = expanded_filename;
|
|
|
|
|
|
|
|
for (p = expanded_filename; *p; p++)
|
|
|
|
if (*p == ':' || *p == ']')
|
|
|
|
q = p + 1;
|
|
|
|
|
|
|
|
for (p = section; *q && *q != '.';)
|
|
|
|
*p++ = *q++;
|
|
|
|
|
|
|
|
*p = 0;
|
|
|
|
|
|
|
|
/* Setup to open the file */
|
|
|
|
|
|
|
|
fab = cc$rms_fab;
|
|
|
|
fab.fab$l_fna = expanded_filename;
|
|
|
|
fab.fab$b_fns = strlen(expanded_filename);
|
|
|
|
fab.fab$l_fop = FAB$M_UFO;
|
|
|
|
fab.fab$l_alq = length / 512;
|
|
|
|
fab.fab$b_fac = FAB$M_UPD | FAB$M_PUT;
|
|
|
|
fab.fab$b_shr = FAB$M_SHRGET | FAB$M_SHRPUT | FAB$M_UPI;
|
|
|
|
fab.fab$b_rfm = FAB$C_UDF;
|
|
|
|
|
|
|
|
/* Setup to create or map the file */
|
|
|
|
|
|
|
|
inadr[0] = inadr[1] = 0;
|
|
|
|
ISC_make_desc(section, &desc, 0);
|
|
|
|
flags = SEC$M_GBL | SEC$M_EXPREG | SEC$M_WRT |
|
|
|
|
((shmem_data->sh_mem_system_flag) ? 0 : SEC$M_SYSGBL);
|
|
|
|
|
|
|
|
if (init_routine) {
|
|
|
|
/* If we're a server, start by opening file.
|
|
|
|
If we can't open it, create it. */
|
|
|
|
|
|
|
|
status = sys$open(&fab);
|
|
|
|
|
|
|
|
if (!(status & 1)) {
|
|
|
|
fab.fab$l_xab = &xab;
|
|
|
|
xab = cc$rms_xabpro;
|
|
|
|
xab.xab$w_pro =
|
|
|
|
(XAB$M_NOEXE << XAB$V_SYS) | (XAB$M_NOEXE << XAB$V_OWN) |
|
|
|
|
(XAB$M_NOEXE << XAB$V_GRP) | (XAB$M_NODEL << XAB$V_GRP) |
|
|
|
|
(XAB$M_NOEXE << XAB$V_WLD) | (XAB$M_NODEL << XAB$V_WLD);
|
|
|
|
|
|
|
|
status = sys$create(&fab);
|
|
|
|
|
|
|
|
if (!(status & 1)) {
|
|
|
|
error(status_vector, "sys$create", status);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create and map section */
|
|
|
|
|
|
|
|
status = sys$crmpsc(inadr, retadr, 0, /* acmode */
|
|
|
|
flags, /* flags */
|
|
|
|
&desc, /* gsdnam */
|
|
|
|
0, /* ident */
|
|
|
|
0, /* relpag */
|
|
|
|
fab.fab$l_stv, /* chan */
|
|
|
|
length / 512, 0, /* vbm */
|
|
|
|
0, /* prot */
|
|
|
|
0); /* pfc */
|
|
|
|
|
|
|
|
if (!(status & 1)) {
|
|
|
|
if (status == SS$_CREATED)
|
|
|
|
sys$deltva(retadr, 0, 0);
|
|
|
|
sys$dassgn(fab.fab$l_stv);
|
|
|
|
error(status_vector, "sys$crmpsc", status);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* We're not a server, just map the global section */
|
|
|
|
|
|
|
|
status = sys$mgblsc(inadr, retadr, 0, /* acmode */
|
|
|
|
flags, /* flags */
|
|
|
|
&desc, /* gsdnam */
|
|
|
|
0, /* ident */
|
|
|
|
0); /* relpag */
|
|
|
|
|
|
|
|
if (!(status & 1)) {
|
|
|
|
error(status_vector, "sys$mgblsc", status);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
shmem_data->sh_mem_address = retadr[0];
|
|
|
|
shmem_data->sh_mem_length_mapped = length;
|
|
|
|
shmem_data->sh_mem_retadr[0] = retadr[0];
|
|
|
|
shmem_data->sh_mem_retadr[0] = retadr[1];
|
|
|
|
shmem_data->sh_mem_channel = fab.fab$l_stv;
|
|
|
|
strcpy(shmem_data->sh_mem_filename, expanded_filename);
|
|
|
|
if (init_routine)
|
|
|
|
(*init_routine) (init_arg, shmem_data, status == SS$_CREATED);
|
|
|
|
|
|
|
|
return (UCHAR *) retadr[0];
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define ISC_MAP_FILE_DEFINED
|
|
|
|
UCHAR *ISC_map_file(STATUS * status_vector,
|
|
|
|
TEXT * filename,
|
|
|
|
void (*init_routine) (void *, struct sh_mem *, int),
|
|
|
|
void *init_arg, SLONG length, SH_MEM shmem_data)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a p _ f i l e ( U N I X - m m a p )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map a given file. If we are the first (i.e. only)
|
|
|
|
* process to map the file, call a given initialization
|
|
|
|
* routine (if given) or punt (leaving the file unmapped).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT expanded_filename[MAXPATHLEN], hostname[64];
|
|
|
|
TEXT tmp[MAXPATHLEN];
|
|
|
|
TEXT init_filename[MAXPATHLEN]; /* to hold the complete filename
|
|
|
|
of the init file. */
|
|
|
|
UCHAR *address;
|
|
|
|
SLONG key, semid;
|
2002-12-10 12:53:53 +01:00
|
|
|
int oldmask, fd;
|
2001-05-23 15:26:42 +02:00
|
|
|
int fd_init; /* filedecr. for the init file */
|
|
|
|
USHORT trunc_flag;
|
|
|
|
struct stat file_stat;
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifndef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
struct flock lock;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
sprintf(expanded_filename, filename,
|
|
|
|
ISC_get_host(hostname, sizeof(hostname)));
|
|
|
|
|
2002-02-16 04:05:21 +01:00
|
|
|
/* make the complete filename for the init file this file is to be used as a
|
2001-05-23 15:26:42 +02:00
|
|
|
master lock to eliminate possible race conditions with just a single file
|
2002-02-16 04:05:21 +01:00
|
|
|
locking. The race condition is caused as the conversion of a EXCLUSIVE
|
2001-05-23 15:26:42 +02:00
|
|
|
lock to a SHARED lock is not atomic*/
|
|
|
|
|
|
|
|
gds__prefix_lock(tmp, INIT_FILE);
|
|
|
|
sprintf(init_filename, tmp, hostname); /* already have the hostname! */
|
|
|
|
|
|
|
|
oldmask = umask(0);
|
|
|
|
if (length < 0) {
|
|
|
|
length = -length;
|
|
|
|
trunc_flag = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
trunc_flag = TRUE;
|
|
|
|
|
|
|
|
/* Produce shared memory key for file */
|
|
|
|
|
|
|
|
if (!(key = find_key(status_vector, expanded_filename))) {
|
|
|
|
umask(oldmask);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* open the init lock file */
|
|
|
|
fd_init = open(init_filename, O_RDWR | O_CREAT, 0666);
|
|
|
|
if (fd_init == -1) {
|
|
|
|
error(status_vector, "open", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifndef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* get an exclusive lock on the INIT file with a block */
|
|
|
|
lock.l_type = F_WRLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
if (fcntl(fd_init, F_SETLKW, &lock) == -1) {
|
|
|
|
error(status_vector, "fcntl", errno);
|
|
|
|
close(fd_init);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/* get an flock exclusive on the INIT file with blocking */
|
|
|
|
if (flock(fd_init, LOCK_EX)) {
|
|
|
|
/* we failed to get an exclusive lock return back */
|
|
|
|
error(status_vector, "flock", errno);
|
|
|
|
close(fd_init);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* open the file to be inited */
|
|
|
|
fd = open(expanded_filename, O_RDWR | O_CREAT, 0666);
|
|
|
|
umask(oldmask);
|
|
|
|
|
|
|
|
if (fd == -1) {
|
|
|
|
error(status_vector, "open", errno);
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* unlock init file */
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
close(fd_init);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length == 0) {
|
|
|
|
/* Get and use the existing length of the shared segment */
|
|
|
|
|
|
|
|
if (fstat(fd, &file_stat) == -1) {
|
|
|
|
error(status_vector, "fstat", errno);
|
|
|
|
close(fd);
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* unlock init file */
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
close(fd_init); /* while we are at it close the init file also */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
length = file_stat.st_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
address =
|
|
|
|
(UCHAR *) mmap(0, length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
|
|
|
|
|
|
|
|
if ((U_IPTR) address == -1) {
|
|
|
|
error(status_vector, "mmap", errno);
|
|
|
|
close(fd);
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* unlock init file */
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
close(fd_init);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get semaphore for mutex */
|
|
|
|
|
|
|
|
#if !(defined SOLARIS_MT || defined POSIX_THREADS)
|
|
|
|
if (shmem_data->sh_mem_semaphores &&
|
|
|
|
(semid =
|
|
|
|
init_semaphores(status_vector, key,
|
|
|
|
shmem_data->sh_mem_semaphores)) < 0) {
|
|
|
|
close(fd);
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* unlock init file */
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
close(fd_init);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
shmem_data->sh_mem_address = address;
|
|
|
|
shmem_data->sh_mem_length_mapped = length;
|
|
|
|
|
|
|
|
#if !(defined SOLARIS_MT || defined POSIX_THREADS)
|
|
|
|
shmem_data->sh_mem_mutex_arg = semid;
|
|
|
|
#else
|
|
|
|
shmem_data->sh_mem_mutex_arg = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
shmem_data->sh_mem_handle = fd;
|
|
|
|
|
|
|
|
/* Try to get an exclusive lock on the lock file. This will
|
|
|
|
fail if somebody else has the exclusive lock */
|
|
|
|
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!flock(fd, LOCK_EX | LOCK_NB))
|
|
|
|
{
|
|
|
|
if (!init_routine) {
|
|
|
|
/* unlock both files */
|
|
|
|
flock(fd, LOCK_UN);
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_WRLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
if (fcntl(fd, F_SETLK, &lock) != -1) {
|
|
|
|
if (!init_routine) {
|
|
|
|
/* unlock the file and the init file to release the other process */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd, F_SETLK, &lock);
|
|
|
|
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
munmap((char *) address, length);
|
|
|
|
close(fd);
|
|
|
|
close(fd_init);
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (trunc_flag)
|
|
|
|
ftruncate(fd, length);
|
|
|
|
(*init_routine) (init_arg, shmem_data, TRUE);
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
if (flock(fd, LOCK_SH)) {
|
|
|
|
error(status_vector, "flock", errno);
|
|
|
|
flock(fd, LOCK_UN);
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_RDLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
if (fcntl(fd, F_SETLK, &lock) == -1)
|
|
|
|
{
|
|
|
|
error(status_vector, "fcntl", errno);
|
|
|
|
/* unlock the file */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd, F_SETLK, &lock);
|
|
|
|
|
|
|
|
/* unlock the init file to release the other process */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
munmap((char *) address, length);
|
|
|
|
|
|
|
|
close(fd_init);
|
|
|
|
close(fd);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
if (flock(fd, LOCK_SH)) {
|
|
|
|
error(status_vector, "flock", errno);
|
|
|
|
flock(fd, LOCK_UN);
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
lock.l_type = F_RDLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
if (fcntl(fd, F_SETLK, &lock) == -1) {
|
|
|
|
error(status_vector, "fcntl", errno);
|
|
|
|
/* unlock the file */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd, F_SETLK, &lock);
|
|
|
|
|
|
|
|
/* unlock the init file to release the other process */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
munmap((char *) address, length);
|
|
|
|
close(fd_init);
|
|
|
|
close(fd);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (init_routine)
|
|
|
|
(*init_routine) (init_arg, shmem_data, FALSE);
|
|
|
|
}
|
|
|
|
|
2002-09-11 13:30:50 +02:00
|
|
|
#ifdef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
/* unlock the init file to release the other process */
|
|
|
|
flock(fd_init, LOCK_UN);
|
|
|
|
#else
|
|
|
|
/* unlock the init file to release the other process */
|
|
|
|
lock.l_type = F_UNLCK;
|
|
|
|
lock.l_whence = 0;
|
|
|
|
lock.l_start = 0;
|
|
|
|
lock.l_len = 0;
|
|
|
|
fcntl(fd_init, F_SETLK, &lock);
|
|
|
|
#endif
|
|
|
|
/* close the init file_decriptor */
|
|
|
|
close(fd_init);
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifndef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define ISC_MAP_FILE_DEFINED
|
|
|
|
UCHAR *ISC_map_file(STATUS * status_vector,
|
|
|
|
TEXT * filename,
|
|
|
|
void (*init_routine) (void *, struct sh_mem *, int),
|
|
|
|
void *init_arg, SLONG length, SH_MEM shmem_data)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a p _ f i l e ( U N I X - s h m a t )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map a given file. If we are the first (i.e. only)
|
|
|
|
* process to map the file, call a given initialization
|
|
|
|
* routine (if given) or punt (leaving the file unmapped).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT expanded_filename[512], hostname[64];
|
|
|
|
UCHAR *address;
|
|
|
|
SSHORT count;
|
|
|
|
int init_flag, oldmask;
|
|
|
|
SLONG key, shmid, semid;
|
|
|
|
struct shmid_ds buf;
|
|
|
|
IB_FILE *fp;
|
|
|
|
|
|
|
|
#ifdef NOHOSTNAME
|
|
|
|
strcpy(expanded_filename, filename);
|
|
|
|
#else
|
|
|
|
sprintf(expanded_filename, filename,
|
|
|
|
ISC_get_host(hostname, sizeof(hostname)));
|
|
|
|
#endif
|
|
|
|
oldmask = umask(0);
|
|
|
|
init_flag = FALSE;
|
|
|
|
if (length < 0)
|
|
|
|
length = -length;
|
|
|
|
|
|
|
|
/* Produce shared memory key for file */
|
|
|
|
|
|
|
|
if (!(key = find_key(status_vector, expanded_filename))) {
|
|
|
|
umask(oldmask);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Write shared memory key into expanded_filename file */
|
|
|
|
|
|
|
|
fp = ib_fopen(expanded_filename, "w");
|
|
|
|
umask(oldmask);
|
|
|
|
|
|
|
|
if (!fp) {
|
|
|
|
error(status_vector, "ib_fopen", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(fp, "%ld", key);
|
|
|
|
|
|
|
|
/* Get an exclusive lock on the file until the initialization process
|
|
|
|
is complete. That way potential race conditions are avoided. */
|
|
|
|
|
2002-11-12 08:59:19 +01:00
|
|
|
#ifndef HAVE_FLOCK
|
2001-05-23 15:26:42 +02:00
|
|
|
if (lockf(ib_fileno(fp), F_LOCK, 0)) {
|
|
|
|
error(status_vector, "lockf", errno);
|
|
|
|
#else
|
|
|
|
if (flock(ib_fileno(fp), LOCK_EX)) {
|
|
|
|
error(status_vector, "flock", errno);
|
|
|
|
#endif
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create the shared memory region if it doesn't already exist. */
|
|
|
|
|
|
|
|
if ((shmid = shmget(key, length, IPC_CREAT | PRIV)) == -1)
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
if (errno == EINVAL) {
|
|
|
|
/* There are two cases when shmget() returns EINVAL error:
|
|
|
|
|
|
|
|
- "length" is less than the system-imposed minimum or
|
|
|
|
greater than the system-imposed maximum.
|
|
|
|
|
|
|
|
- A shared memory identifier exists for "key" but the
|
|
|
|
size of the segment associated with it is less
|
|
|
|
than "length" and "length" is not equal to zero.
|
|
|
|
|
2002-02-16 04:05:21 +01:00
|
|
|
Let's find out what the problem is by getting the
|
2001-05-23 15:26:42 +02:00
|
|
|
system-imposed limits.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HP10
|
|
|
|
struct pst_ipcinfo pst;
|
|
|
|
|
|
|
|
if (pstat_getipc(&pst, sizeof(struct pst_ipcinfo), 1, 0) == -1) {
|
|
|
|
error(status_vector, "pstat_getipc", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((length < pst.psi_shmmin) || (length > pst.psi_shmmax)) {
|
|
|
|
/* If pstat_getipc did not return an error "errno"
|
|
|
|
is still EINVAL, exactly what we want.
|
|
|
|
*/
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif /* HP10 */
|
|
|
|
|
|
|
|
/* If we are here then the shared memory segment already
|
|
|
|
exists and the "length" we specified in shmget() is
|
2002-02-16 04:05:21 +01:00
|
|
|
bigger than the size of the existing segment.
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
Because the segment has to exist at this point the
|
|
|
|
following shmget() does not have IPC_CREAT flag set.
|
|
|
|
|
|
|
|
We need to get shmid to delete the segment. Because we
|
|
|
|
do not know the size of the existing segment the easiest
|
|
|
|
way to get shmid is to attach to the segment with zero
|
|
|
|
length
|
|
|
|
*/
|
|
|
|
if ((shmid = shmget(key, 0, PRIV)) == -1) {
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shmctl(shmid, IPC_RMID, &buf) == -1) {
|
|
|
|
error(status_vector, "shmctl/IPC_RMID", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have just deleted shared memory segment and current
|
|
|
|
code fragment is an atomic operation (we are holding an
|
|
|
|
exclusive lock on the "isc_lock1.<machine>" file), so
|
|
|
|
we use IPC_EXCL flag to get an error if by some miracle
|
|
|
|
the sagment with the same key is already exists
|
|
|
|
*/
|
|
|
|
if ((shmid = shmget(key, length, IPC_CREAT | IPC_EXCL | PRIV)) ==
|
|
|
|
-1) {
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else /* if errno != EINVAL) */
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
{
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
/* If we are here there are two possibilities:
|
|
|
|
|
|
|
|
1. we mapped the shared memory segment of the "length" size;
|
|
|
|
2. we mapped the segment of the size less than "length" (but
|
|
|
|
not zero length and bigger than system-imposed minimum);
|
|
|
|
|
|
|
|
We want shared memory segment exactly of the "length" size.
|
|
|
|
Let's find out what the size of the segment is and if it is
|
|
|
|
bigger than length, we remove it and create new one with the
|
|
|
|
size "length".
|
|
|
|
Also, if "length" is zero (that means we have already mapped
|
2002-02-16 04:05:21 +01:00
|
|
|
the existing segment with the zero size) remap the segment
|
|
|
|
with the existing size
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
if (shmctl(shmid, IPC_STAT, &buf) == -1) {
|
|
|
|
error(status_vector, "shmctl/IPC_STAT", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(length <= buf.shm_segsz);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (length < buf.shm_segsz)
|
|
|
|
if (length) {
|
|
|
|
if (shmctl(shmid, IPC_RMID, &buf) == -1) {
|
|
|
|
error(status_vector, "shmctl/IPC_RMID", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((shmid = shmget(key, length, IPC_CREAT | IPC_EXCL | PRIV)) ==
|
|
|
|
-1) {
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
length = buf.shm_segsz;
|
|
|
|
if ((shmid = shmget(key, length, PRIV)) == -1) {
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else /* !SUPERSERVER */
|
|
|
|
|
|
|
|
if (length == 0) {
|
2002-02-16 04:05:21 +01:00
|
|
|
/* Use the existing length. This should not happen for the
|
2001-05-23 15:26:42 +02:00
|
|
|
very first attachment to the shared memory. */
|
|
|
|
|
|
|
|
if (shmctl(shmid, IPC_STAT, &buf) == -1) {
|
|
|
|
error(status_vector, "shmctl/IPC_STAT", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
length = buf.shm_segsz;
|
|
|
|
|
|
|
|
/* Now remap with the new-found length */
|
|
|
|
|
|
|
|
if ((shmid = shmget(key, length, PRIV)) == -1) {
|
|
|
|
error(status_vector, "shmget", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SHMEM_PICKY
|
|
|
|
if (!next_shared_memory)
|
|
|
|
next_shared_memory = (UCHAR *) sbrk(0) + SHMEM_DELTA;
|
|
|
|
address = (UCHAR *) shmat(shmid, next_shared_memory, SHM_RND);
|
|
|
|
if ((U_IPTR) address != -1)
|
|
|
|
#ifndef SYSV_SHMEM
|
|
|
|
next_shared_memory = address + length;
|
|
|
|
#else
|
|
|
|
next_shared_memory = address + length + SHMLBA;
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
address = (UCHAR *) shmat(shmid, NULL, 0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if ((U_IPTR) address == -1) {
|
|
|
|
error(status_vector, "shmat", errno);
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shmctl(shmid, IPC_STAT, &buf) == -1) {
|
|
|
|
error(status_vector, "shmctl/IPC_STAT", errno);
|
|
|
|
shmdt(address);
|
|
|
|
next_shared_memory -= length;
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get semaphore for mutex */
|
|
|
|
|
|
|
|
#ifndef POSIX_THREADS
|
|
|
|
if (shmem_data->sh_mem_semaphores &&
|
|
|
|
(semid =
|
|
|
|
init_semaphores(status_vector, key,
|
|
|
|
shmem_data->sh_mem_semaphores)) < 0) {
|
|
|
|
shmdt(address);
|
|
|
|
next_shared_memory -= length;
|
|
|
|
ib_fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif /* POSIX_THREADS */
|
|
|
|
|
|
|
|
/* If we're the only one with shared memory mapped, see if
|
|
|
|
we can initialize it. If we can't, return failure. */
|
|
|
|
|
|
|
|
if (buf.shm_nattch == 1) {
|
|
|
|
if (!init_routine) {
|
|
|
|
shmdt(address);
|
|
|
|
next_shared_memory -= length;
|
|
|
|
ib_fclose(fp);
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
buf.shm_perm.mode = 0666;
|
|
|
|
shmctl(shmid, IPC_SET, &buf);
|
|
|
|
init_flag = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
shmem_data->sh_mem_address = address;
|
|
|
|
shmem_data->sh_mem_length_mapped = length;
|
|
|
|
|
|
|
|
#ifndef POSIX_THREADS
|
|
|
|
shmem_data->sh_mem_mutex_arg = semid;
|
|
|
|
#else
|
|
|
|
shmem_data->sh_mem_mutex_arg = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
shmem_data->sh_mem_handle = shmid;
|
|
|
|
|
|
|
|
if (init_routine)
|
|
|
|
(*init_routine) (init_arg, shmem_data, init_flag);
|
|
|
|
|
|
|
|
/* When the mapped file is closed here, the lock we applied for
|
|
|
|
synchronization will be released. */
|
|
|
|
|
|
|
|
ib_fclose(fp);
|
|
|
|
|
|
|
|
return address;
|
|
|
|
}
|
2002-08-26 14:14:55 +02:00
|
|
|
#endif // !HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#endif // UNIX
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define ISC_MAP_FILE_DEFINED
|
2002-11-16 16:54:21 +01:00
|
|
|
UCHAR *DLL_EXPORT ISC_map_file(
|
|
|
|
STATUS * status_vector,
|
|
|
|
TEXT * filename,
|
2003-02-12 13:51:07 +01:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2002-11-16 16:54:21 +01:00
|
|
|
// MUST of course match header.
|
|
|
|
FPTR_VOID init_routine,
|
2003-02-12 13:51:07 +01:00
|
|
|
#else
|
|
|
|
// TMN: Parameter is in errors!
|
|
|
|
void (*init_routine) (void *, struct sh_mem *, int),
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2002-11-16 16:54:21 +01:00
|
|
|
void *init_arg,
|
|
|
|
SLONG length,
|
|
|
|
SH_MEM shmem_data)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a p _ f i l e ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map a given file. If we are the first (i.e. only)
|
|
|
|
* process to map the file, call a given initialization
|
|
|
|
* routine (if given) or punt (leaving the file unmapped).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT expanded_filename[MAXPATHLEN], hostname[64], *p;
|
|
|
|
TEXT map_file[MAXPATHLEN];
|
2002-11-16 16:54:21 +01:00
|
|
|
HANDLE file_handle, event_handle;
|
|
|
|
int init_flag;
|
2001-05-23 15:26:42 +02:00
|
|
|
DWORD ret_event, fdw_create;
|
|
|
|
int retry_count = 0;
|
|
|
|
|
|
|
|
/* To maintain compatibility with the FAT file system we will truncate
|
|
|
|
the host name to 8 characters. Heaven help us if this creates
|
|
|
|
an ambiguity. */
|
|
|
|
|
|
|
|
|
|
|
|
/* retry to attach to mmapped file if the process initializing
|
|
|
|
* dies during initialization.
|
|
|
|
*/
|
|
|
|
|
|
|
|
retry:
|
|
|
|
retry_count++;
|
|
|
|
|
|
|
|
ISC_get_host(hostname, sizeof(hostname));
|
|
|
|
hostname[8] = 0;
|
|
|
|
sprintf(map_file, filename, hostname);
|
|
|
|
|
|
|
|
if (length < 0)
|
|
|
|
length = -length;
|
|
|
|
|
|
|
|
file_handle = CreateFile(map_file,
|
2002-11-16 16:54:21 +01:00
|
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL,
|
|
|
|
OPEN_ALWAYS,
|
|
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
|
|
NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (file_handle == INVALID_HANDLE_VALUE) {
|
|
|
|
error(status_vector, "CreateFile", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if file already exists */
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
const bool file_exists = (GetLastError() == ERROR_ALREADY_EXISTS);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Create an event that can be used to determine if someone has already
|
|
|
|
initialized shared memory. */
|
|
|
|
|
|
|
|
make_object_name(expanded_filename, filename, "_event");
|
|
|
|
if (!ISC_is_WinNT())
|
|
|
|
event_handle =
|
|
|
|
CreateMutex(ISC_get_security_desc(), TRUE, expanded_filename);
|
|
|
|
else
|
|
|
|
event_handle =
|
|
|
|
CreateEvent(ISC_get_security_desc(), TRUE, FALSE,
|
|
|
|
expanded_filename);
|
|
|
|
if (!event_handle) {
|
|
|
|
if (!ISC_is_WinNT())
|
|
|
|
error(status_vector, "CreateMutex", GetLastError());
|
|
|
|
else
|
|
|
|
error(status_vector, "CreateEvent", GetLastError());
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
init_flag = (GetLastError() == ERROR_ALREADY_EXISTS) ? FALSE : TRUE;
|
|
|
|
|
|
|
|
if (init_flag && !init_routine) {
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2002-11-16 16:54:21 +01:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length == 0) {
|
|
|
|
/* Get and use the existing length of the shared segment */
|
|
|
|
|
|
|
|
if ((length = GetFileSize(file_handle, NULL)) == -1) {
|
|
|
|
error(status_vector, "GetFileSize", GetLastError());
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* All but the initializer will wait until the event is set. That
|
|
|
|
* is done after initialization is complete.
|
|
|
|
* Close the file and wait for the event to be set or time out.
|
|
|
|
* The file may be truncated.
|
|
|
|
*/
|
|
|
|
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
|
|
|
|
if (!init_flag) {
|
|
|
|
/* Wait for 10 seconds. Then retry */
|
|
|
|
|
|
|
|
if (!ISC_is_WinNT()) {
|
|
|
|
ret_event = WaitForSingleObject(event_handle, 10000);
|
|
|
|
ReleaseMutex(event_handle);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ret_event = WaitForSingleObject(event_handle, 10000);
|
|
|
|
|
|
|
|
/* If we timed out, just retry. It is possible that the
|
|
|
|
* process doing the initialization died before setting the
|
|
|
|
* event.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (ret_event == WAIT_TIMEOUT) {
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
if (retry_count > 10) {
|
|
|
|
error(status_vector, "WaitForSingleObject", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (init_flag && file_exists)
|
|
|
|
fdw_create = TRUNCATE_EXISTING | OPEN_ALWAYS;
|
|
|
|
else
|
|
|
|
fdw_create = OPEN_ALWAYS;
|
|
|
|
|
|
|
|
file_handle = CreateFile(map_file,
|
2002-11-16 16:54:21 +01:00
|
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL,
|
|
|
|
fdw_create,
|
|
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
|
|
NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (file_handle == INVALID_HANDLE_VALUE) {
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
error(status_vector, "CreateFile", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a file mapping object that will be used to make remapping possible.
|
|
|
|
The current length of real mapped file and its name are saved in it. */
|
|
|
|
|
|
|
|
make_object_name(expanded_filename, filename, "_mapping");
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
HANDLE header_obj = CreateFileMapping ((HANDLE) -1,
|
|
|
|
ISC_get_security_desc(),
|
|
|
|
PAGE_READWRITE,
|
|
|
|
0,
|
|
|
|
2 * sizeof (SLONG),
|
|
|
|
expanded_filename);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (header_obj == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
SLONG* header_address =
|
|
|
|
(SLONG*) MapViewOfFile(header_obj, FILE_MAP_WRITE, 0, 0, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (header_address == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
CloseHandle(header_obj);
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set or get the true length of the file depending on whether or not
|
|
|
|
we are the first user. */
|
|
|
|
|
|
|
|
if (init_flag) {
|
|
|
|
header_address[0] = length;
|
|
|
|
header_address[1] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
length = header_address[0];
|
|
|
|
|
|
|
|
/* Create the real file mapping object. */
|
|
|
|
|
|
|
|
for (p = expanded_filename; *p; p++);
|
2003-04-01 23:59:31 +02:00
|
|
|
sprintf(p, "%"SLONGFORMAT, header_address[1]);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
HANDLE file_obj =
|
|
|
|
CreateFileMapping(file_handle,
|
|
|
|
ISC_get_security_desc(),
|
|
|
|
PAGE_READWRITE,
|
|
|
|
0,
|
|
|
|
length,
|
|
|
|
expanded_filename);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (file_obj == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
UnmapViewOfFile(header_address);
|
|
|
|
CloseHandle(header_obj);
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
UCHAR* address =
|
|
|
|
(UCHAR*) MapViewOfFile(file_obj, FILE_MAP_WRITE, 0, 0, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (address == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
CloseHandle(file_obj);
|
|
|
|
UnmapViewOfFile(header_address);
|
|
|
|
CloseHandle(header_obj);
|
|
|
|
CloseHandle(event_handle);
|
|
|
|
CloseHandle(file_handle);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p = 0;
|
|
|
|
|
|
|
|
shmem_data->sh_mem_address = address;
|
|
|
|
shmem_data->sh_mem_length_mapped = length;
|
|
|
|
shmem_data->sh_mem_handle = file_handle;
|
|
|
|
shmem_data->sh_mem_object = file_obj;
|
|
|
|
shmem_data->sh_mem_interest = event_handle;
|
|
|
|
shmem_data->sh_mem_hdr_object = header_obj;
|
|
|
|
shmem_data->sh_mem_hdr_address = header_address;
|
|
|
|
strcpy(shmem_data->sh_mem_name, expanded_filename);
|
|
|
|
|
|
|
|
if (init_routine)
|
2002-11-16 16:54:21 +01:00
|
|
|
// Lie a bit to make it compile...
|
|
|
|
reinterpret_cast < void (*) (void *, sh_mem *, int) >
|
|
|
|
(*init_routine) (init_arg, shmem_data, init_flag);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (init_flag) {
|
|
|
|
FlushViewOfFile(address, 0);
|
|
|
|
if (!ISC_is_WinNT())
|
|
|
|
ReleaseMutex(event_handle);
|
|
|
|
else
|
|
|
|
SetEvent(event_handle);
|
|
|
|
if (SetFilePointer
|
|
|
|
(shmem_data->sh_mem_handle, length, NULL,
|
|
|
|
FILE_BEGIN) == 0xFFFFFFFF
|
|
|
|
|| !SetEndOfFile(shmem_data->sh_mem_handle)
|
|
|
|
|| !FlushViewOfFile(shmem_data->sh_mem_address, 0)) {
|
|
|
|
error(status_vector, "SetFilePointer", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
#ifndef ISC_MAP_FILE_DEFINED
|
|
|
|
UCHAR *ISC_map_file(STATUS * status_vector,
|
|
|
|
TEXT * filename,
|
|
|
|
void (*init_routine) (void *, struct sh_mem *, int),
|
|
|
|
void *init_arg, SLONG length, SH_MEM shmem_data)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m a p _ f i l e ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map a given file. If we are the first (i.e. only)
|
|
|
|
* process to map the file, call a given initialization
|
|
|
|
* routine (if given) or punt (leaving the file unmapped).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define ISC_MAP_OBJECT_DEFINED
|
|
|
|
UCHAR *ISC_map_object(STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data,
|
|
|
|
SLONG object_offset, SLONG object_length)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 04:05:21 +01:00
|
|
|
* I S C _ m a p _ o b j e c t
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to map an object given a file mapping.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
UCHAR *address;
|
|
|
|
int fd;
|
|
|
|
SLONG page_size, start, end, length;
|
|
|
|
|
|
|
|
/* Get system page size as this is the unit of mapping. */
|
|
|
|
|
2002-02-23 23:15:24 +01:00
|
|
|
#ifdef SOLARIS
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((page_size = sysconf(_SC_PAGESIZE)) == -1) {
|
|
|
|
error(status_vector, "sysconf", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if ((page_size = (int) getpagesize()) == -1) {
|
|
|
|
error(status_vector, "getpagesize", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Compute the start and end page-aligned offsets which
|
|
|
|
contain the object being mapped. */
|
|
|
|
|
|
|
|
start = (object_offset / page_size) * page_size;
|
|
|
|
end = (((object_offset + object_length) / page_size) + 1) * page_size;
|
|
|
|
length = end - start;
|
|
|
|
fd = shmem_data->sh_mem_handle;
|
|
|
|
|
|
|
|
address =
|
|
|
|
(UCHAR *) mmap(0, length, PROT_READ | PROT_WRITE, MAP_SHARED, fd,
|
|
|
|
start);
|
|
|
|
|
|
|
|
if ((U_IPTR) address == -1) {
|
|
|
|
error(status_vector, "mmap", errno);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return the virtual address of the mapped object. */
|
|
|
|
|
|
|
|
return (address + (object_offset - start));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define ISC_UNMAP_OBJECT_DEFINED
|
|
|
|
BOOLEAN ISC_unmap_object(STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data,
|
|
|
|
UCHAR ** object_pointer, SLONG object_length)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 04:05:21 +01:00
|
|
|
* I S C _ u n m a p _ o b j e c t
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to unmap an object given a file mapping.
|
|
|
|
* Zero the object pointer after a successful unmap.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG page_size, length;
|
|
|
|
UCHAR *start, *end;
|
|
|
|
|
|
|
|
/* Get system page size as this is the unit of mapping. */
|
|
|
|
|
2002-02-23 23:15:24 +01:00
|
|
|
#ifdef SOLARIS
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((page_size = sysconf(_SC_PAGESIZE)) == -1) {
|
|
|
|
error(status_vector, "sysconf", errno);
|
2002-09-26 11:26:40 +02:00
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
if ((page_size = (int) getpagesize()) == -1) {
|
|
|
|
error(status_vector, "getpagesize", errno);
|
2002-09-26 11:26:40 +02:00
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Compute the start and end page-aligned addresses which
|
|
|
|
contain the mapped object. */
|
|
|
|
|
|
|
|
start = (UCHAR *) ((U_IPTR) * object_pointer & ~(page_size - 1));
|
|
|
|
end =
|
|
|
|
(UCHAR
|
|
|
|
*) ((U_IPTR) ((*object_pointer + object_length) +
|
|
|
|
(page_size - 1)) & ~(page_size - 1));
|
|
|
|
length = end - start;
|
|
|
|
|
|
|
|
if (munmap((char *) start, length) == -1) {
|
|
|
|
error(status_vector, "munmap", errno);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*object_pointer = 0;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
#define MUTEX
|
|
|
|
int ISC_mutex_init(MTX mutex, SLONG event_flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
mutex->mtx_use_count = 0;
|
|
|
|
mutex->mtx_event_count[0] = 0;
|
|
|
|
mutex->mtx_event_count[1] = event_flag;
|
|
|
|
mutex->mtx_wait = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG bit, status;
|
|
|
|
|
|
|
|
bit = 0;
|
|
|
|
++mutex->mtx_wait;
|
|
|
|
|
|
|
|
if (lib$bbssi(&bit, mutex->mtx_event_count))
|
|
|
|
for (;;) {
|
|
|
|
if (!lib$bbssi(&bit, mutex->mtx_event_count))
|
|
|
|
break;
|
|
|
|
gds__thread_wait(mutex_test, mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
--mutex->mtx_wait;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG status, bit;
|
|
|
|
|
|
|
|
bit = 0;
|
|
|
|
lib$bbcci(&bit, mutex->mtx_event_count);
|
|
|
|
#ifndef __ALPHA
|
|
|
|
sys$wake(0, 0);
|
|
|
|
#else
|
|
|
|
THREAD_wakeup();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SOLARIS_MT
|
|
|
|
#define MUTEX
|
|
|
|
int ISC_mutex_init(MTX mutex, SLONG semaphore)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return mutex_init(mutex->mtx_mutex, USYNC_PROCESS, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
state = mutex_lock(mutex->mtx_mutex);
|
|
|
|
if (!state)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(state))
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock_cond(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k _ c o n d ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Conditionally sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
state = mutex_trylock(mutex->mtx_mutex);
|
|
|
|
if (!state)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(state))
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( S O L A R I S _ M T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
|
|
|
|
for (;;) {
|
2002-02-16 04:05:21 +01:00
|
|
|
/* Note use of undocumented lwp_mutex_unlock call
|
2001-05-23 15:26:42 +02:00
|
|
|
* due to Solaris 2.4 bug */
|
|
|
|
state = _lwp_mutex_unlock(mutex->mtx_mutex);
|
|
|
|
if (!state)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(state))
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif /* SOLARIS_MT */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef POSIX_THREADS
|
|
|
|
#define MUTEX
|
|
|
|
int ISC_mutex_init(MTX mutex, SLONG semaphore)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
|
2002-07-02 11:49:19 +02:00
|
|
|
/* RITTER - replaced HP10 with HPUX in the line below */
|
2003-03-02 18:47:22 +01:00
|
|
|
#if (!defined HPUX && !defined LINUX && !defined DARWIN && !defined FREEBSD)
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
pthread_mutexattr_t mattr;
|
|
|
|
|
|
|
|
state = pthread_mutexattr_init(&mattr);
|
|
|
|
if (state)
|
|
|
|
return state;
|
|
|
|
pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED);
|
|
|
|
state = pthread_mutex_init(mutex->mtx_mutex, &mattr);
|
|
|
|
pthread_mutexattr_destroy(&mattr);
|
|
|
|
return state;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
/* NOTE: HP's Posix threads implementation does not support thread
|
|
|
|
synchronization in different processes. Thus the following
|
|
|
|
fragment is just a temp. decision we could be use for super-
|
|
|
|
server (until we are to implement local IPC using shared
|
|
|
|
memory in which case we need interprocess thread sync.
|
|
|
|
*/
|
2002-10-12 06:16:32 +02:00
|
|
|
/* RITTER - added HP11 */
|
2003-03-02 18:47:22 +01:00
|
|
|
#if (defined LINUX || defined DARWIN || defined HP11 || defined FREEBSD)
|
2001-05-23 15:26:42 +02:00
|
|
|
return pthread_mutex_init(mutex->mtx_mutex, NULL);
|
|
|
|
#else
|
|
|
|
state = pthread_mutex_init(mutex->mtx_mutex, pthread_mutexattr_default);
|
|
|
|
if (!state)
|
|
|
|
return 0;
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(state == -1); /* if state is not 0, it should be -1 */
|
2001-05-23 15:26:42 +02:00
|
|
|
return errno;
|
|
|
|
|
|
|
|
#endif /* linux */
|
|
|
|
#endif /* HP10 */
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef HP10
|
|
|
|
|
|
|
|
int state;
|
|
|
|
|
|
|
|
state = pthread_mutex_lock(mutex->mtx_mutex);
|
|
|
|
if (!state)
|
|
|
|
return 0;
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(state == -1); /* if state is not 0, it should be -1 */
|
2001-05-23 15:26:42 +02:00
|
|
|
return errno;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
return pthread_mutex_lock(mutex->mtx_mutex);
|
|
|
|
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock_cond(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k _ c o n d ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Conditionally sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef HP10
|
|
|
|
|
|
|
|
int state;
|
|
|
|
|
|
|
|
state = pthread_mutex_trylock(mutex->mtx_mutex);
|
|
|
|
|
|
|
|
/* HP's interpretation of return codes is different than Solaris
|
|
|
|
(and Posix Standard?). Usually in case of error they return
|
|
|
|
-1 and set errno to whatever error number is.
|
|
|
|
pthread_mutex_trylock() is a special case:
|
|
|
|
|
|
|
|
return errno description
|
|
|
|
1 Success
|
|
|
|
0 mutex has alreary been locked. Could not get it
|
|
|
|
-1 EINVAL invalid value of mutex
|
|
|
|
*/
|
|
|
|
if (!state)
|
|
|
|
return -99; /* we did not get the mutex for it had already */
|
|
|
|
/* been locked, let's return something which is */
|
|
|
|
/* not zero and negative (errno values are positive) */
|
|
|
|
if (state == 1)
|
|
|
|
return 0;
|
|
|
|
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(state == -1); /* if state is not 0 or 1, it should be -1 */
|
2001-05-23 15:26:42 +02:00
|
|
|
return errno;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
return pthread_mutex_trylock(mutex->mtx_mutex);
|
|
|
|
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( P O S I X _ T H R E A D S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef HP10
|
|
|
|
|
|
|
|
int state;
|
|
|
|
|
|
|
|
state = pthread_mutex_unlock(mutex->mtx_mutex);
|
|
|
|
if (!state)
|
|
|
|
return 0;
|
2003-03-01 00:46:34 +01:00
|
|
|
assert(state == -1); /* if state is not 0, it should be -1 */
|
2001-05-23 15:26:42 +02:00
|
|
|
return errno;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
return pthread_mutex_unlock(mutex->mtx_mutex);
|
|
|
|
|
|
|
|
#endif /* HP10 */
|
|
|
|
}
|
|
|
|
#endif /* POSIX_THREADS */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
#ifndef MUTEX
|
|
|
|
#define MUTEX
|
|
|
|
int ISC_mutex_init(MTX mutex, SLONG semaphore)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
mutex->mtx_semid = semaphore;
|
|
|
|
mutex->mtx_semnum = 0;
|
|
|
|
|
|
|
|
arg.val = 1;
|
|
|
|
state = semctl((int) semaphore, 0, SETVAL, arg);
|
|
|
|
if (state == -1)
|
|
|
|
return errno;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
struct sembuf sop;
|
|
|
|
|
|
|
|
sop.sem_num = mutex->mtx_semnum;
|
|
|
|
sop.sem_op = -1;
|
|
|
|
sop.sem_flg = SEM_UNDO;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
state = semop(mutex->mtx_semid, &sop, 1);
|
|
|
|
if (state != -1)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(errno))
|
|
|
|
return errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock_cond(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k _ c o n d ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Conditionally sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
struct sembuf sop;
|
|
|
|
|
|
|
|
sop.sem_num = mutex->mtx_semnum;
|
|
|
|
sop.sem_op = -1;
|
|
|
|
sop.sem_flg = SEM_UNDO | IPC_NOWAIT;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
state = semop(mutex->mtx_semid, &sop, 1);
|
|
|
|
if (state != -1)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(errno))
|
|
|
|
return errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( U N I X )
|
|
|
|
* not NeXT
|
|
|
|
* not SOLARIS
|
|
|
|
* not POSIX_THREADS
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int state;
|
|
|
|
struct sembuf sop;
|
|
|
|
|
|
|
|
sop.sem_num = mutex->mtx_semnum;
|
|
|
|
sop.sem_op = 1;
|
|
|
|
sop.sem_flg = SEM_UNDO;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
state = semop(mutex->mtx_semid, &sop, 1);
|
|
|
|
if (state != -1)
|
|
|
|
break;
|
|
|
|
if (!SYSCALL_INTERRUPTED(errno))
|
|
|
|
return errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define MUTEX
|
|
|
|
int DLL_EXPORT ISC_mutex_init(MTX mutex, TEXT * mutex_name)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-16 16:54:21 +01:00
|
|
|
char name_buffer[MAXPATHLEN];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
make_object_name(name_buffer, mutex_name, "_mutex");
|
|
|
|
mutex->mtx_handle =
|
|
|
|
CreateMutex(ISC_get_security_desc(), FALSE, name_buffer);
|
|
|
|
|
|
|
|
return (mutex->mtx_handle) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
const DWORD status = WaitForSingleObject(mutex->mtx_handle, INFINITE);
|
2001-05-23 15:26:42 +02:00
|
|
|
return (!status || status == WAIT_ABANDONED) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_mutex_lock_cond(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k _ c o n d ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Conditionally sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
const DWORD status = WaitForSingleObject (mutex->mtx_handle, 0L);
|
2001-05-23 15:26:42 +02:00
|
|
|
return (!status || status == WAIT_ABANDONED) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DLL_EXPORT ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return !ReleaseMutex(mutex->mtx_handle);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef MUTEX
|
|
|
|
int ISC_mutex_init(MTX mutex, SLONG dummy)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ i n i t ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_lock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ l o c k ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sieze a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ISC_mutex_unlock(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ m u t e x _ u n l o c k ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a mutex.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define ISC_REMAP_FILE_DEFINED
|
|
|
|
UCHAR *ISC_remap_file(STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data, SLONG new_length, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ r e m a p _ f i l e ( U N I X - m m a p )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to re-map a given file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
UCHAR *address;
|
|
|
|
|
|
|
|
if (flag)
|
|
|
|
ftruncate(shmem_data->sh_mem_handle, new_length);
|
|
|
|
|
|
|
|
address =
|
|
|
|
(UCHAR *) mmap(0, new_length, PROT_READ | PROT_WRITE, MAP_SHARED,
|
|
|
|
shmem_data->sh_mem_handle, 0);
|
|
|
|
if ((U_IPTR) address == -1)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
munmap((char *) shmem_data->sh_mem_address,
|
|
|
|
shmem_data->sh_mem_length_mapped);
|
|
|
|
|
|
|
|
shmem_data->sh_mem_address = address;
|
|
|
|
shmem_data->sh_mem_length_mapped = new_length;
|
|
|
|
|
|
|
|
return address;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define ISC_REMAP_FILE_DEFINED
|
|
|
|
UCHAR *DLL_EXPORT ISC_remap_file(STATUS * status_vector,
|
2002-11-16 16:54:21 +01:00
|
|
|
SH_MEM shmem_data,
|
|
|
|
SLONG new_length,
|
|
|
|
USHORT flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ r e m a p _ f i l e ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to re-map a given file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (flag)
|
|
|
|
if (SetFilePointer
|
|
|
|
(shmem_data->sh_mem_handle, new_length, NULL,
|
|
|
|
FILE_BEGIN) == 0xFFFFFFFF
|
|
|
|
|| !SetEndOfFile(shmem_data->sh_mem_handle)
|
|
|
|
|| !FlushViewOfFile(shmem_data->sh_mem_address, 0)) {
|
|
|
|
error(status_vector, "SetFilePointer", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the remap file exists, remap does not occur correctly.
|
2002-02-16 04:05:21 +01:00
|
|
|
* The file number is local to the process and when it is
|
2001-05-23 15:26:42 +02:00
|
|
|
* incremented and a new filename is created, that file may
|
|
|
|
* already exist. In that case, the file is not expanded.
|
|
|
|
* This will happen when the file is expanded more than once
|
|
|
|
* by concurrently running processes.
|
2002-02-16 04:05:21 +01:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
* The problem will be fixed by making sure that a new file name
|
|
|
|
* is generated with the mapped file is created.
|
|
|
|
*/
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
HANDLE file_obj;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
while (1) {
|
2002-11-16 16:54:21 +01:00
|
|
|
TEXT expanded_filename[MAXPATHLEN];
|
2003-04-01 23:59:31 +02:00
|
|
|
sprintf(expanded_filename, "%s%"SLONGFORMAT, shmem_data->sh_mem_name,
|
2001-05-23 15:26:42 +02:00
|
|
|
shmem_data->sh_mem_hdr_address[1] + 1);
|
|
|
|
|
|
|
|
file_obj = CreateFileMapping(shmem_data->sh_mem_handle,
|
2002-11-16 16:54:21 +01:00
|
|
|
ISC_get_security_desc(),
|
|
|
|
PAGE_READWRITE,
|
|
|
|
0,
|
|
|
|
new_length,
|
|
|
|
expanded_filename);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!((GetLastError() == ERROR_ALREADY_EXISTS) && flag))
|
|
|
|
break;
|
|
|
|
|
|
|
|
CloseHandle(file_obj);
|
|
|
|
shmem_data->sh_mem_hdr_address[1]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (file_obj == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
LPVOID address = MapViewOfFile(file_obj, FILE_MAP_WRITE, 0, 0, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (address == NULL) {
|
|
|
|
error(status_vector, "CreateFileMapping", GetLastError());
|
|
|
|
CloseHandle(file_obj);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flag) {
|
|
|
|
shmem_data->sh_mem_hdr_address[0] = new_length;
|
|
|
|
shmem_data->sh_mem_hdr_address[1]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
UnmapViewOfFile(shmem_data->sh_mem_address);
|
|
|
|
CloseHandle(shmem_data->sh_mem_object);
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
shmem_data->sh_mem_address = reinterpret_cast<UCHAR*>(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
shmem_data->sh_mem_length_mapped = new_length;
|
|
|
|
shmem_data->sh_mem_object = file_obj;
|
|
|
|
|
2002-11-16 16:54:21 +01:00
|
|
|
return reinterpret_cast<UCHAR*>(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef ISC_REMAP_FILE_DEFINED
|
|
|
|
UCHAR *DLL_EXPORT ISC_remap_file(STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data,
|
|
|
|
SLONG new_length, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ r e m a p _ f i l e ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to re-map a given file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if (defined UNIX)
|
|
|
|
void ISC_reset_timer(
|
2002-09-24 21:53:31 +02:00
|
|
|
FPTR_VOID timeout_handler,
|
2001-05-23 15:26:42 +02:00
|
|
|
void *timeout_arg,
|
|
|
|
SLONG * client_timer, void **client_handler)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ r e s e t _ t i m e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear a previously established timer and restore
|
|
|
|
* the previous context.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifndef SYSV_SIGNALS
|
|
|
|
struct itimerval internal_timer;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ISC_signal_cancel(SIGALRM, timeout_handler, timeout_arg);
|
|
|
|
|
|
|
|
/* Cancel the timer, then restore the previous handler and alarm */
|
|
|
|
|
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
alarm(0);
|
|
|
|
sigset(SIGALRM, *client_handler);
|
|
|
|
alarm(*client_timer);
|
|
|
|
#else
|
|
|
|
timerclear(&internal_timer.it_interval);
|
|
|
|
timerclear(&internal_timer.it_value);
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer, NULL);
|
2002-08-22 12:48:26 +02:00
|
|
|
#ifndef HAVE_SIGACTION
|
2001-05-23 15:26:42 +02:00
|
|
|
sigvector(SIGALRM, client_handler, NULL);
|
|
|
|
#else
|
2001-07-12 07:46:06 +02:00
|
|
|
sigaction(SIGALRM, (struct sigaction*)client_handler, NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2001-07-12 07:46:06 +02:00
|
|
|
setitimer(ITIMER_REAL, (itimerval*)client_timer, NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if (defined UNIX)
|
|
|
|
void ISC_set_timer(
|
|
|
|
SLONG micro_seconds,
|
2001-07-12 07:46:06 +02:00
|
|
|
FPTR_VOID timeout_handler,
|
2001-05-23 15:26:42 +02:00
|
|
|
void *timeout_arg,
|
|
|
|
SLONG * client_timer, void **client_handler)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ s e t _ t i m e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set a timer for the specified amount of time.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-12-10 12:53:53 +01:00
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
void *d2;
|
|
|
|
SLONG d1;
|
|
|
|
#else
|
2001-05-23 15:26:42 +02:00
|
|
|
struct itimerval internal_timer;
|
2002-08-22 12:48:26 +02:00
|
|
|
#ifndef HAVE_SIGACTION
|
2001-05-23 15:26:42 +02:00
|
|
|
struct sigvec internal_handler;
|
|
|
|
#else
|
|
|
|
struct sigaction internal_handler;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Start by cancelling any existing timer */
|
|
|
|
|
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
if (!client_timer)
|
|
|
|
client_timer = &d1;
|
|
|
|
*client_timer = alarm(0);
|
|
|
|
#else
|
|
|
|
timerclear(&internal_timer.it_interval);
|
|
|
|
timerclear(&internal_timer.it_value);
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer,
|
|
|
|
(struct itimerval *) client_timer);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Now clear the signal handler while saving the existing one */
|
|
|
|
|
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
if (!client_handler)
|
|
|
|
client_handler = &d2;
|
|
|
|
*client_handler = (void *) sigset(SIGALRM, SIG_DFL);
|
|
|
|
#else
|
2002-08-22 12:48:26 +02:00
|
|
|
#ifndef HAVE_SIGACTION
|
2001-05-23 15:26:42 +02:00
|
|
|
internal_handler.sv_handler = SIG_DFL;
|
|
|
|
internal_handler.sv_mask = 0;
|
|
|
|
internal_handler.sv_flags = SV_INTERRUPT;
|
|
|
|
sigvector(SIGALRM, &internal_handler, (struct sigvec *) client_handler);
|
|
|
|
#else
|
2002-08-22 12:48:26 +02:00
|
|
|
internal_handler.sa_handler = (SIG_FPTR) SIG_DFL;
|
2001-05-23 15:26:42 +02:00
|
|
|
memset(&internal_handler.sa_mask, 0, sizeof(internal_handler.sa_mask));
|
|
|
|
internal_handler.sa_flags = SA_RESTART;
|
|
|
|
sigaction(SIGALRM, &internal_handler,
|
|
|
|
(struct sigaction *) client_handler);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!micro_seconds)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Next set the new alarm handler and finally set the new timer */
|
|
|
|
|
|
|
|
ISC_signal(SIGALRM, timeout_handler, timeout_arg);
|
|
|
|
|
|
|
|
#ifdef SYSV_SIGNALS
|
|
|
|
if (micro_seconds < 1000000)
|
|
|
|
alarm(1);
|
|
|
|
else
|
|
|
|
alarm(micro_seconds / 1000000);
|
|
|
|
#else
|
|
|
|
if (micro_seconds < 1000000)
|
|
|
|
internal_timer.it_value.tv_usec = micro_seconds;
|
|
|
|
else {
|
|
|
|
internal_timer.it_value.tv_sec = micro_seconds / 1000000;
|
|
|
|
internal_timer.it_value.tv_usec = micro_seconds % 1000000;
|
|
|
|
}
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer, NULL);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#ifdef UNIX
|
|
|
|
void ISC_sync_signals_set()
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ s y n c _ s i g n a l s _ s e t( U N I X )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set all the synchronous signals for a particular thread
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
sigset(SIGILL, longjmp_sig_handler);
|
|
|
|
sigset(SIGFPE, longjmp_sig_handler);
|
|
|
|
sigset(SIGBUS, longjmp_sig_handler);
|
|
|
|
sigset(SIGSEGV, longjmp_sig_handler);
|
|
|
|
}
|
|
|
|
#endif /* UNIX */
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#ifdef UNIX
|
|
|
|
void ISC_sync_signals_reset()
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ s y n c _ s i g n a l s _ r e s e t ( U N I X )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Reset all the synchronous signals for a particular thread
|
|
|
|
* to default.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
sigset(SIGILL, SIG_DFL);
|
|
|
|
sigset(SIGFPE, SIG_DFL);
|
|
|
|
sigset(SIGBUS, SIG_DFL);
|
|
|
|
sigset(SIGSEGV, SIG_DFL);
|
|
|
|
}
|
|
|
|
#endif /* UNIX */
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifdef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define UNMAP_FILE
|
|
|
|
void ISC_unmap_file(STATUS * status_vector, SH_MEM shmem_data, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ u n m a p _ f i l e ( U N I X - m m a p )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Unmap a given file. Depending upon the flag,
|
|
|
|
* get rid of the semaphore and/or truncate the file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
munmap((char *) shmem_data->sh_mem_address,
|
|
|
|
shmem_data->sh_mem_length_mapped);
|
|
|
|
|
|
|
|
if (flag & ISC_SEM_REMOVE)
|
|
|
|
semctl(shmem_data->sh_mem_mutex_arg, 0, IPC_RMID, arg);
|
|
|
|
if (flag & ISC_MEM_REMOVE)
|
|
|
|
ftruncate(shmem_data->sh_mem_handle, 0L);
|
|
|
|
close(shmem_data->sh_mem_handle);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2002-08-26 14:14:55 +02:00
|
|
|
#ifndef HAVE_MMAP
|
2001-05-23 15:26:42 +02:00
|
|
|
#define UNMAP_FILE
|
|
|
|
void ISC_unmap_file(STATUS * status_vector, SH_MEM shmem_data, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ u n m a p _ f i l e ( U N I X - s h m a t )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Detach from the shared memory. Depending upon the flag,
|
|
|
|
* get rid of the semaphore and/or get rid of shared memory.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
struct shmid_ds buf;
|
|
|
|
union semun arg;
|
|
|
|
|
|
|
|
shmdt(shmem_data->sh_mem_address);
|
|
|
|
if (flag & ISC_SEM_REMOVE)
|
|
|
|
semctl(shmem_data->sh_mem_mutex_arg, 0, IPC_RMID, arg);
|
|
|
|
if (flag & ISC_MEM_REMOVE)
|
|
|
|
shmctl(shmem_data->sh_mem_handle, IPC_RMID, &buf);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define UNMAP_FILE
|
|
|
|
void DLL_EXPORT ISC_unmap_file(
|
2002-11-16 16:54:21 +01:00
|
|
|
STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data,
|
|
|
|
USHORT flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ u n m a p _ f i l e ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Detach from the shared memory. Depending upon the flag,
|
|
|
|
* get rid of the semaphore and/or get rid of shared memory.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
CloseHandle(shmem_data->sh_mem_interest);
|
|
|
|
CloseHandle((HANDLE) shmem_data->sh_mem_mutex_arg);
|
|
|
|
UnmapViewOfFile(shmem_data->sh_mem_address);
|
|
|
|
CloseHandle(shmem_data->sh_mem_object);
|
|
|
|
if (flag & ISC_MEM_REMOVE)
|
2002-11-16 16:54:21 +01:00
|
|
|
if (SetFilePointer(shmem_data->sh_mem_handle, 0, NULL, FILE_BEGIN) != 0xFFFFFFFF)
|
|
|
|
SetEndOfFile(shmem_data->sh_mem_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
CloseHandle(shmem_data->sh_mem_handle);
|
|
|
|
UnmapViewOfFile(shmem_data->sh_mem_hdr_address);
|
|
|
|
CloseHandle(shmem_data->sh_mem_hdr_object);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef UNMAP_FILE
|
|
|
|
void DLL_EXPORT ISC_unmap_file(
|
|
|
|
STATUS * status_vector,
|
|
|
|
SH_MEM shmem_data, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I S C _ u n m a p _ f i l e ( G E N E R I C )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* unmap a given file or shared memory.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_unavailable;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
static void alarm_handler(void)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a l a r m _ h a n d l e r ( U N I X )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-02-16 04:05:21 +01:00
|
|
|
* Handle an alarm clock interrupt.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static void error(STATUS * status_vector, TEXT * string, STATUS status)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We've encountered an error, report it.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
*status_vector++ = gds_arg_gds;
|
2001-07-29 19:42:23 +02:00
|
|
|
*status_vector++ = gds_sys_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = gds_arg_string;
|
|
|
|
*status_vector++ = (STATUS) string;
|
|
|
|
*status_vector++ = SYS_ARG;
|
|
|
|
*status_vector++ = status;
|
|
|
|
*status_vector++ = gds_arg_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
static int event_test(WAIT * wait)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e v e n t _ t e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Callback routine from thread package for VMS. Returns
|
|
|
|
* TRUE if wait is satified, otherwise FALSE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return !ISC_event_blocked(wait->wait_count, wait->wait_events,
|
|
|
|
wait->wait_values);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
static SLONG find_key(STATUS * status_vector, TEXT * filename)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 04:05:21 +01:00
|
|
|
* f i n d _ k e y
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Find the semaphore/shared memory key for a file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
int fd;
|
2002-11-21 11:20:07 +01:00
|
|
|
key_t key;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Produce shared memory key for file */
|
|
|
|
|
|
|
|
if ((key = ftok(filename, FTOK_KEY)) == -1) {
|
|
|
|
if ((fd = open(filename, O_RDWR | O_CREAT | O_TRUNC, PRIV)) == -1) {
|
|
|
|
error(status_vector, "open", errno);
|
2002-09-26 11:26:40 +02:00
|
|
|
return 0L;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
if ((key = ftok(filename, FTOK_KEY)) == -1) {
|
|
|
|
error(status_vector, "ftok", errno);
|
2002-09-26 11:26:40 +02:00
|
|
|
return 0L;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
|
|
|
static SLONG init_semaphores(
|
|
|
|
STATUS * status_vector,
|
|
|
|
SLONG key, int semaphores)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n i t _ s e m a p h o r e s ( U N I X )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create or find a block of semaphores.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG semid;
|
|
|
|
|
|
|
|
semid = semget(key, semaphores, IPC_CREAT | IPC_EXCL | PRIV);
|
|
|
|
|
|
|
|
if (semid == -1) {
|
|
|
|
semid = semget(key, semaphores, PRIV);
|
|
|
|
if (semid == -1)
|
|
|
|
error(status_vector, "semget", errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
return semid;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#ifdef UNIX
|
|
|
|
void longjmp_sig_handler(int sig_num)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l o n g j m p _ s i g _ h a n d l e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* The generic signal handler for all signals in a thread.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TDBB tdbb;
|
|
|
|
|
|
|
|
/* Note: we can only do this since we know that we
|
|
|
|
will only be going to JRD, specifically fun and blf.
|
2002-02-16 04:05:21 +01:00
|
|
|
If we were to make this generic, we would need to
|
2001-05-23 15:26:42 +02:00
|
|
|
actally hang the sigsetjmp menber off of THDD, and
|
|
|
|
make sure that it is set properly for all sub-systems. */
|
|
|
|
|
|
|
|
tdbb = GET_THREAD_DATA;
|
|
|
|
|
2002-07-29 17:37:59 +02:00
|
|
|
siglongjmp(tdbb->tdbb_sigsetjmp, sig_num);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif /* UNIX */
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
static BOOLEAN mutex_test(MTX mutex)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* m u t e x _ t e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Callback routine from thread package for VMS. Returns
|
|
|
|
* TRUE if mutex has been granted, otherwise FALSE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return (mutex->mtx_event_count[0] & 1) ? FALSE : TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
static void make_object_name(
|
2002-11-16 16:54:21 +01:00
|
|
|
TEXT * buffer,
|
|
|
|
const TEXT * object_name,
|
|
|
|
const TEXT * object_type)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* m a k e _ o b j e c t _ n a m e ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create an object name from a name and type.
|
|
|
|
* Also replace the file separator with "_".
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-16 16:54:21 +01:00
|
|
|
char hostname[64], *p, c;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
sprintf(buffer, object_name, ISC_get_host(hostname, sizeof(hostname)));
|
|
|
|
for (p = buffer; c = *p; p++)
|
|
|
|
if (c == '/' || c == '\\' || c == ':')
|
|
|
|
*p = '_';
|
|
|
|
strcpy(p, object_type);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef UNIX
|
2001-07-12 07:46:06 +02:00
|
|
|
static BOOLEAN semaphore_wait_isc_sync(int count, int semid, int *semnums)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2001-07-12 07:46:06 +02:00
|
|
|
* s e m a p h o r e _ w a i t _ i s c _ s y n c
|
|
|
|
*
|
|
|
|
* (formerly known as: s e m a p h o r e _ w a i t)
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-11-14 09:33:08 +01:00
|
|
|
* Wait on the given semaphores. Return FB_FAILURE if
|
2001-05-23 15:26:42 +02:00
|
|
|
* interrupted (including timeout) before any
|
2002-11-14 09:33:08 +01:00
|
|
|
* semaphore was poked else return FB_SUCCESS.
|
2002-02-16 04:05:21 +01:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
**************************************/
|
|
|
|
int i, ret;
|
|
|
|
struct sembuf semops[16], *semptr;
|
|
|
|
|
|
|
|
for (semptr = semops, i = 0; i < count; ++semptr, i++) {
|
|
|
|
semptr->sem_op = 0;
|
|
|
|
semptr->sem_flg = 0;
|
|
|
|
semptr->sem_num = *semnums++;
|
|
|
|
}
|
|
|
|
ret = semop(semid, semops, count);
|
|
|
|
|
|
|
|
if (ret == -1 && SYSCALL_INTERRUPTED(errno))
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-03-11 17:34:08 +01:00
|
|
|
} // extern "C"
|
|
|
|
|