NT4/private/crt32/lowio/locking.c
2020-09-30 17:12:29 +02:00

188 lines
5.0 KiB
C

/***
*locking.c - OS/2 file locking function
*
* Copyright (c) 1989-1992, Microsoft Corporation. All rights reserved.
*
*Purpose:
* Defined the _locking() function - file locking and unlocking
*
*Revision History:
* 06-09-89 PHG Module created, based on asm version
* 08-10-89 JCR Changed DOS32FILELOCKS to DOS32SETFILELOCKS
* 03-12-90 GJF Made calling type _CALLTYPE2 (for now), added #include
* <cruntime.h> and fixed the copyright. Also, cleaned up
* the formatting a bit.
* 04-03-90 GJF Now _CALLTYPE1.
* 07-24-90 SBM Removed '32' from API names
* 08-14-90 SBM Compiles cleanly with -W3
* 09-28-90 GJF New-style function declarator.
* 12-04-90 GJF Appended Win32 version onto the source with #ifdef-s.
* It is enough different that there is little point in
* trying to more closely merge the two versions.
* 12-04-90 SRW Changed to include <oscalls.h> instead of <doscalls.h>
* 12-06-90 SRW Changed to use _osfile and _osfhnd instead of _osfinfo
* 01-16-91 GJF ANSI naming.
* 02-07-91 SRW Changed to call _get_osfhandle [_WIN32_]
* 12-05-91 GJF Fixed usage of [Un]LockFile APIs [_WIN32_].
* 02-13-92 GJF Replaced _nfile by _nhandle for Win32.
* 05-06-92 SRW WIN32 LockFile API changed. [_WIN32_].
*
*******************************************************************************/
#include <cruntime.h>
#include <oscalls.h>
#include <errno.h>
#include <sys\locking.h>
#include <io.h>
#include <stdlib.h>
#include <internal.h>
#include <os2dll.h>
/***
*int _locking(fh,lmode,nbytes) - file record locking function
*
*Purpose:
* Locks or unlocks nbytes of a specified file
*
* Multi-thread - Must lock/unlock the file handle to prevent
* other threads from working on the file at the same time as us.
* [NOTE: We do NOT release the lock during the 1 second delays
* since some other thread could get in and do something to the
* file. The DOSFILELOCK call locks out other processes, not
* threads, so there is no multi-thread deadlock at the DOS file
* locking level.]
*
*Entry:
* int fh - file handle
* int lmode - locking mode:
* _LK_LOCK/_LK_RLCK -> lock, retry 10 times
* _LK_NBLCK/_LK_N_BRLCK -> lock, don't retry
* _LK_UNLCK -> unlock
* long nbytes - number of bytes to lock/unlock
*
*Exit:
* returns 0 if successful
* returns -1 and sets errno if unsuccessful
*
*Exceptions:
*
*******************************************************************************/
int _CALLTYPE1 _locking (
int fh,
int lmode,
long nbytes
)
{
ULONG dosretval; /* OS/2 return code */
LONG lockoffset;
int retry; /* retry count */
/* validate file handle */
#ifdef _WIN32_
if ( (unsigned)fh >= (unsigned)_nhandle ) {
#else
if ((unsigned)fh >= (unsigned)_nfile) {
#endif
/* fh out of range */
errno = EBADF;
_doserrno = 0; /* not an OS/2 error */
return -1;
}
_lock_fh(fh); /* acquire file handle lock */
/* obtain current position in file by calling _lseek */
/* Use _lseek_lk as we already own lock */
lockoffset = _lseek_lk(fh, 0L, 1);
if (lockoffset == -1) {
_unlock_fh(fh);
return -1;
}
/* set retry count based on mode */
if (lmode == _LK_LOCK || lmode == _LK_RLCK)
retry = 9; /* retry 9 times */
else
retry = 0; /* don't retry */
/* ask OS/2 to lock the file until success or retry count finished */
/* note that the only error possible is a locking violation, since */
/* an invalid handle would have already failed above */
for (;;) {
#ifdef _CRUISER_
FILELOCK filelock; /* OS/2 file locking structure */
filelock.lOffset = lockoffset;
filelock.lRange = nbytes;
if (lmode == _LK_UNLCK)
dosretval = DOSSETFILELOCKS(fh, &filelock, NULL);
else
dosretval = DOSSETFILELOCKS(fh, NULL, &filelock);
if (retry <= 0 || dosretval == 0)
break; /* exit loop on success or retry exhausted */
DOSSLEEP(1000); /* wait 1 sec until try again */
#else /* ndef _CRUISER_ */
#ifdef _WIN32_
dosretval = 0;
if (lmode == _LK_UNLCK) {
if ( !(UnlockFile((HANDLE)_get_osfhandle(fh),
lockoffset,
0L,
nbytes,
0L))
)
dosretval = GetLastError();
} else {
if ( !(LockFile((HANDLE)_get_osfhandle(fh),
lockoffset,
0L,
nbytes,
0L))
)
dosretval = GetLastError();
}
if (retry <= 0 || dosretval == 0)
break; /* exit loop on success or retry exhausted */
Sleep(1000L);
#else /* ndef _WIN32_ */
#error ERROR - ONLY CRUISER OR WIN32 TARGET SUPPORTED!
#endif /* _WIN32_ */
#endif /* _CRUISER_ */
--retry;
}
_unlock_fh(fh); /* release the file handle lock */
if (dosretval != 0) {
/* OS/2 error occurred -- file was already locked; if a
blocking call, then return EDEADLOCK, otherwise map
error normally */
if (lmode == _LK_LOCK || lmode == _LK_RLCK) {
errno = EDEADLOCK;
_doserrno = dosretval;
}
else {
_dosmaperr(dosretval);
}
return -1;
}
else
return 0;
}