mirror of
https://github.com/postgrespro/pg_probackup.git
synced 2025-01-09 14:45:47 +02:00
4df4b1644f
git-svn-id: http://pg-rman.googlecode.com/svn/trunk@28 182aca00-e38e-11de-a668-6fd11605f5ce
194 lines
3.4 KiB
C
194 lines
3.4 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* pgut-port.c
|
|
*
|
|
* Copyright (c) 2009-2010, NIPPON TELEGRAPH AND TELEPHONE CORPORATION
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "c.h"
|
|
#include "pgut-port.h"
|
|
|
|
#undef flock
|
|
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
|
|
#ifdef WIN32
|
|
|
|
#include <winioctl.h>
|
|
|
|
#define REPARSE_DATA_SIZE 1024
|
|
|
|
/* same layout as REPARSE_DATA_BUFFER, which is defined only in old winnt.h */
|
|
typedef struct REPARSE_DATA
|
|
{
|
|
ULONG ReparseTag;
|
|
WORD ReparseDataLength;
|
|
WORD Reserved;
|
|
union
|
|
{
|
|
struct
|
|
{
|
|
WORD SubstituteNameOffset;
|
|
WORD SubstituteNameLength;
|
|
WORD PrintNameOffset;
|
|
WORD PrintNameLength;
|
|
ULONG Flags;
|
|
WCHAR PathBuffer[1];
|
|
} Symlink;
|
|
struct
|
|
{
|
|
WORD SubstituteNameOffset;
|
|
WORD SubstituteNameLength;
|
|
WORD PrintNameOffset;
|
|
WORD PrintNameLength;
|
|
WCHAR PathBuffer[1];
|
|
} Mount;
|
|
struct
|
|
{
|
|
BYTE DataBuffer[REPARSE_DATA_SIZE];
|
|
} Generic;
|
|
};
|
|
} REPARSE_DATA;
|
|
|
|
ssize_t
|
|
readlink(const char *path, char *target, size_t size)
|
|
{
|
|
HANDLE handle;
|
|
DWORD attr;
|
|
REPARSE_DATA data;
|
|
DWORD datasize;
|
|
PCWSTR wpath;
|
|
int wlen;
|
|
int r;
|
|
|
|
attr = GetFileAttributes(path);
|
|
if (attr == INVALID_FILE_ATTRIBUTES)
|
|
{
|
|
_dosmaperr(GetLastError());
|
|
return -1;
|
|
}
|
|
if ((attr & FILE_ATTRIBUTE_REPARSE_POINT) == 0)
|
|
{
|
|
errno = EINVAL; /* not a symlink */
|
|
return -1;
|
|
}
|
|
|
|
handle = CreateFileA(path, 0,
|
|
FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, NULL,
|
|
OPEN_EXISTING,
|
|
FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, NULL);
|
|
if (handle == INVALID_HANDLE_VALUE)
|
|
{
|
|
_dosmaperr(GetLastError());
|
|
return -1;
|
|
}
|
|
|
|
wpath = NULL;
|
|
if (DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, NULL, 0,
|
|
&data, sizeof(data), &datasize, NULL))
|
|
{
|
|
switch (data.ReparseTag)
|
|
{
|
|
case IO_REPARSE_TAG_MOUNT_POINT:
|
|
{
|
|
wpath = data.Mount.PathBuffer + data.Mount.SubstituteNameOffset;
|
|
wlen = data.Mount.SubstituteNameLength;
|
|
break;
|
|
}
|
|
case IO_REPARSE_TAG_SYMLINK:
|
|
{
|
|
wpath = data.Symlink.PathBuffer + data.Symlink.SubstituteNameOffset;
|
|
wlen = data.Symlink.SubstituteNameLength;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (wpath == NULL)
|
|
r = -1;
|
|
else
|
|
{
|
|
if (wcsncmp(wpath, L"\\??\\", 4) == 0 ||
|
|
wcsncmp(wpath, L"\\\\?\\", 4) == 0)
|
|
{
|
|
wpath += 4;
|
|
wlen -= 4;
|
|
}
|
|
r = WideCharToMultiByte(CP_ACP, 0, wpath, wlen, target, size, NULL, NULL);
|
|
}
|
|
|
|
CloseHandle(handle);
|
|
return r;
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef PGUT_FLOCK
|
|
|
|
#ifdef WIN32
|
|
int
|
|
pgut_flock(int fd, int operation)
|
|
{
|
|
BOOL ret;
|
|
HANDLE handle = (HANDLE) _get_osfhandle(fd);
|
|
DWORD lo = 0;
|
|
DWORD hi = 0;
|
|
|
|
if (operation & LOCK_UN)
|
|
{
|
|
ret = UnlockFileEx(handle, 0, lo, hi, NULL);
|
|
}
|
|
else
|
|
{
|
|
DWORD flags = 0;
|
|
if (operation & LOCK_EX)
|
|
flags |= LOCKFILE_EXCLUSIVE_LOCK;
|
|
if (operation & LOCK_NB)
|
|
flags |= LOCKFILE_FAIL_IMMEDIATELY;
|
|
ret = LockFileEx(handle, flags, 0, lo, hi, NULL);
|
|
}
|
|
|
|
if (!ret)
|
|
{
|
|
_dosmaperr(GetLastError());
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#else
|
|
|
|
int
|
|
pgut_flock(int fd, int operation)
|
|
{
|
|
struct flock lck;
|
|
int cmd;
|
|
|
|
memset(&lck, 0, sizeof(lck));
|
|
lck.l_whence = SEEK_SET;
|
|
lck.l_start = 0;
|
|
lck.l_len = 0;
|
|
lck.l_pid = getpid();
|
|
|
|
if (operation & LOCK_UN)
|
|
lck.l_type = F_UNLCK;
|
|
else if (operation & LOCK_EX)
|
|
lck.l_type = F_WRLCK;
|
|
else
|
|
lck.l_type = F_RDLCK;
|
|
|
|
if (operation & LOCK_NB)
|
|
cmd = F_SETLK;
|
|
else
|
|
cmd = F_SETLKW;
|
|
|
|
return fcntl(fd, cmd, &lck);
|
|
}
|
|
#endif
|
|
|
|
#endif
|