Location via proxy:   [ UP ]  
[Report a bug]   [Manage cookies]                
summaryrefslogtreecommitdiff
blob: 627922b600e4739d90a2ab5d1d4cecdea2c0ea61 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*-------------------------------------------------------------------------
 *
 * open.c
 *	   Win32 open() replacement
 *
 *
 * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
 *
 * $PostgreSQL: pgsql/src/port/open.c,v 1.2 2004/04/19 17:42:59 momjian Exp $
 *
 *-------------------------------------------------------------------------
 */

#ifdef WIN32

#include <windows.h>
#include <fcntl.h>
#include <errno.h>
#include <assert.h>

static int
openFlagsToCreateFileFlags(int openFlags)
{
	switch (openFlags & (O_CREAT|O_TRUNC|O_EXCL))
	{
		case 0:
		case O_EXCL:	return OPEN_EXISTING;

		case O_CREAT:	return OPEN_ALWAYS;

		case O_TRUNC:
		case O_TRUNC|O_EXCL:	return TRUNCATE_EXISTING;

		case O_CREAT|O_TRUNC:	return CREATE_ALWAYS;

		case O_CREAT|O_EXCL:
		case O_CREAT|O_TRUNC|O_EXCL:	return CREATE_NEW;
	}

	/* will never get here */
	return 0;
}

/*
 *   - file attribute setting, based on fileMode?
 *   - handle other flags? (eg FILE_FLAG_NO_BUFFERING/FILE_FLAG_WRITE_THROUGH)
 */
int win32_open(const char* fileName, int fileFlags, ...)
{
	int fd;
	HANDLE h;
	SECURITY_ATTRIBUTES sa;

	/* Check that we can handle the request */
	assert((fileFlags & ((O_RDONLY|O_WRONLY|O_RDWR) | O_APPEND	|
						 (O_RANDOM|O_SEQUENTIAL|O_TEMPORARY)	|
						 _O_SHORT_LIVED							|
						 (O_CREAT|O_TRUNC|O_EXCL) | (O_TEXT|O_BINARY))) == fileFlags);

	sa.nLength=sizeof(sa);
	sa.bInheritHandle=TRUE;
	sa.lpSecurityDescriptor=NULL;

	if ((h = CreateFile(fileName,
						/* cannot use O_RDONLY, as it == 0 */
						(fileFlags & O_RDWR)     ? (GENERIC_WRITE | GENERIC_READ) :
						((fileFlags &  O_WRONLY) ?  GENERIC_WRITE : GENERIC_READ),
						(FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE),
						&sa,
						openFlagsToCreateFileFlags(fileFlags),
						FILE_ATTRIBUTE_NORMAL	|
						((fileFlags &  O_RANDOM)		? FILE_FLAG_RANDOM_ACCESS	: 0) |
						((fileFlags &  O_SEQUENTIAL)	? FILE_FLAG_SEQUENTIAL_SCAN	: 0) |
						((fileFlags & _O_SHORT_LIVED)	? FILE_ATTRIBUTE_TEMPORARY	: 0) |
						((fileFlags &  O_TEMPORARY)		? FILE_FLAG_DELETE_ON_CLOSE	: 0),
						NULL)) == INVALID_HANDLE_VALUE)
	{
		switch (GetLastError())
		{
			/* EMFILE, ENFILE should not occur from CreateFile. */
			case ERROR_PATH_NOT_FOUND:
			case ERROR_FILE_NOT_FOUND:	errno = ENOENT; break;
			case ERROR_FILE_EXISTS:		errno = EEXIST; break;
			case ERROR_ACCESS_DENIED:	errno = EACCES; break;
			default:
				errno = EINVAL;
		}
		return -1;
	}

	/* _open_osfhandle will, on error, set errno accordingly */
	if ((fd = _open_osfhandle((long)h,fileFlags&O_APPEND)) < 0 ||
		(fileFlags&(O_TEXT|O_BINARY) && (_setmode(fd,fileFlags&(O_TEXT|O_BINARY)) < 0)))
		CloseHandle(h); /* will not affect errno */
	return fd;
}

#endif