Commit 1f67078e authored by Heikki Linnakangas's avatar Heikki Linnakangas

Add OpenTransientFile, with automatic cleanup at end-of-xact.

Files opened with BasicOpenFile or PathNameOpenFile are not automatically
cleaned up on error. That puts unnecessary burden on callers that only want
to keep the file open for a short time. There is AllocateFile, but that
returns a buffered FILE * stream, which in many cases is not the nicest API
to work with. So add function called OpenTransientFile, which returns a
unbuffered fd that's cleaned up like the FILE* returned by AllocateFile().

This plugs a few rare fd leaks in error cases:

1. copy_file() - fixed by by using OpenTransientFile instead of BasicOpenFile
2. XLogFileInit() - fixed by adding close() calls to the error cases. Can't
   use OpenTransientFile here because the fd is supposed to persist over
   transaction boundaries.
3. lo_import/lo_export - fixed by using OpenTransientFile instead of
   PathNameOpenFile.

In addition to plugging those leaks, this replaces many BasicOpenFile() calls
with OpenTransientFile() that were not leaking, because the code meticulously
closed the file on error. That wasn't strictly necessary, but IMHO it's good
for robustness.

The same leaks exist in older versions, but given the rarity of the issues,
I'm not backpatching this. Not yet, anyway - it might be good to backpatch
later, after this mechanism has had some more testing in master branch.
parent 53299429
...@@ -531,7 +531,7 @@ SlruInternalWritePage(SlruCtl ctl, int slotno, SlruFlush fdata) ...@@ -531,7 +531,7 @@ SlruInternalWritePage(SlruCtl ctl, int slotno, SlruFlush fdata)
int i; int i;
for (i = 0; i < fdata->num_files; i++) for (i = 0; i < fdata->num_files; i++)
close(fdata->fd[i]); CloseTransientFile(fdata->fd[i]);
} }
/* Re-acquire control lock and update page state */ /* Re-acquire control lock and update page state */
...@@ -593,7 +593,7 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno) ...@@ -593,7 +593,7 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno)
* SlruPhysicalWritePage). Hence, if we are InRecovery, allow the case * SlruPhysicalWritePage). Hence, if we are InRecovery, allow the case
* where the file doesn't exist, and return zeroes instead. * where the file doesn't exist, and return zeroes instead.
*/ */
fd = BasicOpenFile(path, O_RDWR | PG_BINARY, S_IRUSR | S_IWUSR); fd = OpenTransientFile(path, O_RDWR | PG_BINARY, S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
{ {
if (errno != ENOENT || !InRecovery) if (errno != ENOENT || !InRecovery)
...@@ -614,7 +614,7 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno) ...@@ -614,7 +614,7 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno)
{ {
slru_errcause = SLRU_SEEK_FAILED; slru_errcause = SLRU_SEEK_FAILED;
slru_errno = errno; slru_errno = errno;
close(fd); CloseTransientFile(fd);
return false; return false;
} }
...@@ -623,11 +623,11 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno) ...@@ -623,11 +623,11 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno)
{ {
slru_errcause = SLRU_READ_FAILED; slru_errcause = SLRU_READ_FAILED;
slru_errno = errno; slru_errno = errno;
close(fd); CloseTransientFile(fd);
return false; return false;
} }
if (close(fd)) if (CloseTransientFile(fd))
{ {
slru_errcause = SLRU_CLOSE_FAILED; slru_errcause = SLRU_CLOSE_FAILED;
slru_errno = errno; slru_errno = errno;
...@@ -740,7 +740,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata) ...@@ -740,7 +740,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata)
* don't use O_EXCL or O_TRUNC or anything like that. * don't use O_EXCL or O_TRUNC or anything like that.
*/ */
SlruFileName(ctl, path, segno); SlruFileName(ctl, path, segno);
fd = BasicOpenFile(path, O_RDWR | O_CREAT | PG_BINARY, fd = OpenTransientFile(path, O_RDWR | O_CREAT | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
{ {
...@@ -773,7 +773,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata) ...@@ -773,7 +773,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata)
slru_errcause = SLRU_SEEK_FAILED; slru_errcause = SLRU_SEEK_FAILED;
slru_errno = errno; slru_errno = errno;
if (!fdata) if (!fdata)
close(fd); CloseTransientFile(fd);
return false; return false;
} }
...@@ -786,7 +786,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata) ...@@ -786,7 +786,7 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata)
slru_errcause = SLRU_WRITE_FAILED; slru_errcause = SLRU_WRITE_FAILED;
slru_errno = errno; slru_errno = errno;
if (!fdata) if (!fdata)
close(fd); CloseTransientFile(fd);
return false; return false;
} }
...@@ -800,11 +800,11 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata) ...@@ -800,11 +800,11 @@ SlruPhysicalWritePage(SlruCtl ctl, int pageno, int slotno, SlruFlush fdata)
{ {
slru_errcause = SLRU_FSYNC_FAILED; slru_errcause = SLRU_FSYNC_FAILED;
slru_errno = errno; slru_errno = errno;
close(fd); CloseTransientFile(fd);
return false; return false;
} }
if (close(fd)) if (CloseTransientFile(fd))
{ {
slru_errcause = SLRU_CLOSE_FAILED; slru_errcause = SLRU_CLOSE_FAILED;
slru_errno = errno; slru_errno = errno;
...@@ -1078,7 +1078,7 @@ SimpleLruFlush(SlruCtl ctl, bool checkpoint) ...@@ -1078,7 +1078,7 @@ SimpleLruFlush(SlruCtl ctl, bool checkpoint)
ok = false; ok = false;
} }
if (close(fdata.fd[i])) if (CloseTransientFile(fdata.fd[i]))
{ {
slru_errcause = SLRU_CLOSE_FAILED; slru_errcause = SLRU_CLOSE_FAILED;
slru_errno = errno; slru_errno = errno;
......
...@@ -244,7 +244,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -244,7 +244,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
unlink(tmppath); unlink(tmppath);
/* do not use get_sync_bit() here --- want to fsync only at end of fill */ /* do not use get_sync_bit() here --- want to fsync only at end of fill */
fd = BasicOpenFile(tmppath, O_RDWR | O_CREAT | O_EXCL, fd = OpenTransientFile(tmppath, O_RDWR | O_CREAT | O_EXCL,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
ereport(ERROR, ereport(ERROR,
...@@ -262,7 +262,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -262,7 +262,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
else else
TLHistoryFilePath(path, parentTLI); TLHistoryFilePath(path, parentTLI);
srcfd = BasicOpenFile(path, O_RDONLY, 0); srcfd = OpenTransientFile(path, O_RDONLY, 0);
if (srcfd < 0) if (srcfd < 0)
{ {
if (errno != ENOENT) if (errno != ENOENT)
...@@ -304,7 +304,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -304,7 +304,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
errmsg("could not write to file \"%s\": %m", tmppath))); errmsg("could not write to file \"%s\": %m", tmppath)));
} }
} }
close(srcfd); CloseTransientFile(srcfd);
} }
/* /*
...@@ -345,7 +345,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -345,7 +345,7 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync file \"%s\": %m", tmppath))); errmsg("could not fsync file \"%s\": %m", tmppath)));
if (close(fd)) if (CloseTransientFile(fd))
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close file \"%s\": %m", tmppath))); errmsg("could not close file \"%s\": %m", tmppath)));
......
...@@ -970,15 +970,10 @@ EndPrepare(GlobalTransaction gxact) ...@@ -970,15 +970,10 @@ EndPrepare(GlobalTransaction gxact)
/* /*
* Create the 2PC state file. * Create the 2PC state file.
*
* Note: because we use BasicOpenFile(), we are responsible for ensuring
* the FD gets closed in any error exit path. Once we get into the
* critical section, though, it doesn't matter since any failure causes
* PANIC anyway.
*/ */
TwoPhaseFilePath(path, xid); TwoPhaseFilePath(path, xid);
fd = BasicOpenFile(path, fd = OpenTransientFile(path,
O_CREAT | O_EXCL | O_WRONLY | PG_BINARY, O_CREAT | O_EXCL | O_WRONLY | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
...@@ -995,7 +990,7 @@ EndPrepare(GlobalTransaction gxact) ...@@ -995,7 +990,7 @@ EndPrepare(GlobalTransaction gxact)
COMP_CRC32(statefile_crc, record->data, record->len); COMP_CRC32(statefile_crc, record->data, record->len);
if ((write(fd, record->data, record->len)) != record->len) if ((write(fd, record->data, record->len)) != record->len)
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not write two-phase state file: %m"))); errmsg("could not write two-phase state file: %m")));
...@@ -1012,7 +1007,7 @@ EndPrepare(GlobalTransaction gxact) ...@@ -1012,7 +1007,7 @@ EndPrepare(GlobalTransaction gxact)
if ((write(fd, &bogus_crc, sizeof(pg_crc32))) != sizeof(pg_crc32)) if ((write(fd, &bogus_crc, sizeof(pg_crc32))) != sizeof(pg_crc32))
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not write two-phase state file: %m"))); errmsg("could not write two-phase state file: %m")));
...@@ -1021,7 +1016,7 @@ EndPrepare(GlobalTransaction gxact) ...@@ -1021,7 +1016,7 @@ EndPrepare(GlobalTransaction gxact)
/* Back up to prepare for rewriting the CRC */ /* Back up to prepare for rewriting the CRC */
if (lseek(fd, -((off_t) sizeof(pg_crc32)), SEEK_CUR) < 0) if (lseek(fd, -((off_t) sizeof(pg_crc32)), SEEK_CUR) < 0)
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not seek in two-phase state file: %m"))); errmsg("could not seek in two-phase state file: %m")));
...@@ -1061,13 +1056,13 @@ EndPrepare(GlobalTransaction gxact) ...@@ -1061,13 +1056,13 @@ EndPrepare(GlobalTransaction gxact)
/* write correct CRC and close file */ /* write correct CRC and close file */
if ((write(fd, &statefile_crc, sizeof(pg_crc32))) != sizeof(pg_crc32)) if ((write(fd, &statefile_crc, sizeof(pg_crc32))) != sizeof(pg_crc32))
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not write two-phase state file: %m"))); errmsg("could not write two-phase state file: %m")));
} }
if (close(fd) != 0) if (CloseTransientFile(fd) != 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close two-phase state file: %m"))); errmsg("could not close two-phase state file: %m")));
...@@ -1144,7 +1139,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings) ...@@ -1144,7 +1139,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings)
TwoPhaseFilePath(path, xid); TwoPhaseFilePath(path, xid);
fd = BasicOpenFile(path, O_RDONLY | PG_BINARY, 0); fd = OpenTransientFile(path, O_RDONLY | PG_BINARY, 0);
if (fd < 0) if (fd < 0)
{ {
if (give_warnings) if (give_warnings)
...@@ -1163,7 +1158,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings) ...@@ -1163,7 +1158,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings)
*/ */
if (fstat(fd, &stat)) if (fstat(fd, &stat))
{ {
close(fd); CloseTransientFile(fd);
if (give_warnings) if (give_warnings)
ereport(WARNING, ereport(WARNING,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -1177,14 +1172,14 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings) ...@@ -1177,14 +1172,14 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings)
sizeof(pg_crc32)) || sizeof(pg_crc32)) ||
stat.st_size > MaxAllocSize) stat.st_size > MaxAllocSize)
{ {
close(fd); CloseTransientFile(fd);
return NULL; return NULL;
} }
crc_offset = stat.st_size - sizeof(pg_crc32); crc_offset = stat.st_size - sizeof(pg_crc32);
if (crc_offset != MAXALIGN(crc_offset)) if (crc_offset != MAXALIGN(crc_offset))
{ {
close(fd); CloseTransientFile(fd);
return NULL; return NULL;
} }
...@@ -1195,7 +1190,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings) ...@@ -1195,7 +1190,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings)
if (read(fd, buf, stat.st_size) != stat.st_size) if (read(fd, buf, stat.st_size) != stat.st_size)
{ {
close(fd); CloseTransientFile(fd);
if (give_warnings) if (give_warnings)
ereport(WARNING, ereport(WARNING,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -1205,7 +1200,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings) ...@@ -1205,7 +1200,7 @@ ReadTwoPhaseFile(TransactionId xid, bool give_warnings)
return NULL; return NULL;
} }
close(fd); CloseTransientFile(fd);
hdr = (TwoPhaseFileHeader *) buf; hdr = (TwoPhaseFileHeader *) buf;
if (hdr->magic != TWOPHASE_MAGIC || hdr->total_len != stat.st_size) if (hdr->magic != TWOPHASE_MAGIC || hdr->total_len != stat.st_size)
...@@ -1469,7 +1464,7 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len) ...@@ -1469,7 +1464,7 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len)
TwoPhaseFilePath(path, xid); TwoPhaseFilePath(path, xid);
fd = BasicOpenFile(path, fd = OpenTransientFile(path,
O_CREAT | O_TRUNC | O_WRONLY | PG_BINARY, O_CREAT | O_TRUNC | O_WRONLY | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
...@@ -1481,14 +1476,14 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len) ...@@ -1481,14 +1476,14 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len)
/* Write content and CRC */ /* Write content and CRC */
if (write(fd, content, len) != len) if (write(fd, content, len) != len)
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not write two-phase state file: %m"))); errmsg("could not write two-phase state file: %m")));
} }
if (write(fd, &statefile_crc, sizeof(pg_crc32)) != sizeof(pg_crc32)) if (write(fd, &statefile_crc, sizeof(pg_crc32)) != sizeof(pg_crc32))
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not write two-phase state file: %m"))); errmsg("could not write two-phase state file: %m")));
...@@ -1500,13 +1495,13 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len) ...@@ -1500,13 +1495,13 @@ RecreateTwoPhaseFile(TransactionId xid, void *content, int len)
*/ */
if (pg_fsync(fd) != 0) if (pg_fsync(fd) != 0)
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync two-phase state file: %m"))); errmsg("could not fsync two-phase state file: %m")));
} }
if (close(fd) != 0) if (CloseTransientFile(fd) != 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close two-phase state file: %m"))); errmsg("could not close two-phase state file: %m")));
...@@ -1577,7 +1572,7 @@ CheckPointTwoPhase(XLogRecPtr redo_horizon) ...@@ -1577,7 +1572,7 @@ CheckPointTwoPhase(XLogRecPtr redo_horizon)
TwoPhaseFilePath(path, xid); TwoPhaseFilePath(path, xid);
fd = BasicOpenFile(path, O_RDWR | PG_BINARY, 0); fd = OpenTransientFile(path, O_RDWR | PG_BINARY, 0);
if (fd < 0) if (fd < 0)
{ {
if (errno == ENOENT) if (errno == ENOENT)
...@@ -1596,14 +1591,14 @@ CheckPointTwoPhase(XLogRecPtr redo_horizon) ...@@ -1596,14 +1591,14 @@ CheckPointTwoPhase(XLogRecPtr redo_horizon)
if (pg_fsync(fd) != 0) if (pg_fsync(fd) != 0)
{ {
close(fd); CloseTransientFile(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync two-phase state file \"%s\": %m", errmsg("could not fsync two-phase state file \"%s\": %m",
path))); path)));
} }
if (close(fd) != 0) if (CloseTransientFile(fd) != 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close two-phase state file \"%s\": %m", errmsg("could not close two-phase state file \"%s\": %m",
......
...@@ -2246,6 +2246,16 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock) ...@@ -2246,6 +2246,16 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock)
unlink(tmppath); unlink(tmppath);
/*
* Allocate a buffer full of zeros. This is done before opening the file
* so that we don't leak the file descriptor if palloc fails.
*
* Note: palloc zbuffer, instead of just using a local char array, to
* ensure it is reasonably well-aligned; this may save a few cycles
* transferring data to the kernel.
*/
zbuffer = (char *) palloc0(XLOG_BLCKSZ);
/* do not use get_sync_bit() here --- want to fsync only at end of fill */ /* do not use get_sync_bit() here --- want to fsync only at end of fill */
fd = BasicOpenFile(tmppath, O_RDWR | O_CREAT | O_EXCL | PG_BINARY, fd = BasicOpenFile(tmppath, O_RDWR | O_CREAT | O_EXCL | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
...@@ -2262,12 +2272,7 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock) ...@@ -2262,12 +2272,7 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock)
* fsync below) that all the indirect blocks are down on disk. Therefore, * fsync below) that all the indirect blocks are down on disk. Therefore,
* fdatasync(2) or O_DSYNC will be sufficient to sync future writes to the * fdatasync(2) or O_DSYNC will be sufficient to sync future writes to the
* log file. * log file.
*
* Note: palloc zbuffer, instead of just using a local char array, to
* ensure it is reasonably well-aligned; this may save a few cycles
* transferring data to the kernel.
*/ */
zbuffer = (char *) palloc0(XLOG_BLCKSZ);
for (nbytes = 0; nbytes < XLogSegSize; nbytes += XLOG_BLCKSZ) for (nbytes = 0; nbytes < XLogSegSize; nbytes += XLOG_BLCKSZ)
{ {
errno = 0; errno = 0;
...@@ -2279,6 +2284,9 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock) ...@@ -2279,6 +2284,9 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock)
* If we fail to make the file, delete it to release disk space * If we fail to make the file, delete it to release disk space
*/ */
unlink(tmppath); unlink(tmppath);
close(fd);
/* if write didn't set errno, assume problem is no disk space */ /* if write didn't set errno, assume problem is no disk space */
errno = save_errno ? save_errno : ENOSPC; errno = save_errno ? save_errno : ENOSPC;
...@@ -2290,9 +2298,12 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock) ...@@ -2290,9 +2298,12 @@ XLogFileInit(XLogSegNo logsegno, bool *use_existent, bool use_lock)
pfree(zbuffer); pfree(zbuffer);
if (pg_fsync(fd) != 0) if (pg_fsync(fd) != 0)
{
close(fd);
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync file \"%s\": %m", tmppath))); errmsg("could not fsync file \"%s\": %m", tmppath)));
}
if (close(fd)) if (close(fd))
ereport(ERROR, ereport(ERROR,
...@@ -2363,7 +2374,7 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno) ...@@ -2363,7 +2374,7 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno)
* Open the source file * Open the source file
*/ */
XLogFilePath(path, srcTLI, srcsegno); XLogFilePath(path, srcTLI, srcsegno);
srcfd = BasicOpenFile(path, O_RDONLY | PG_BINARY, 0); srcfd = OpenTransientFile(path, O_RDONLY | PG_BINARY, 0);
if (srcfd < 0) if (srcfd < 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -2377,7 +2388,7 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno) ...@@ -2377,7 +2388,7 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno)
unlink(tmppath); unlink(tmppath);
/* do not use get_sync_bit() here --- want to fsync only at end of fill */ /* do not use get_sync_bit() here --- want to fsync only at end of fill */
fd = BasicOpenFile(tmppath, O_RDWR | O_CREAT | O_EXCL | PG_BINARY, fd = OpenTransientFile(tmppath, O_RDWR | O_CREAT | O_EXCL | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
ereport(ERROR, ereport(ERROR,
...@@ -2423,12 +2434,12 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno) ...@@ -2423,12 +2434,12 @@ XLogFileCopy(XLogSegNo destsegno, TimeLineID srcTLI, XLogSegNo srcsegno)
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync file \"%s\": %m", tmppath))); errmsg("could not fsync file \"%s\": %m", tmppath)));
if (close(fd)) if (CloseTransientFile(fd))
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close file \"%s\": %m", tmppath))); errmsg("could not close file \"%s\": %m", tmppath)));
close(srcfd); CloseTransientFile(srcfd);
/* /*
* Now move the segment into place with its final name. * Now move the segment into place with its final name.
......
...@@ -442,7 +442,7 @@ lo_import_with_oid(PG_FUNCTION_ARGS) ...@@ -442,7 +442,7 @@ lo_import_with_oid(PG_FUNCTION_ARGS)
static Oid static Oid
lo_import_internal(text *filename, Oid lobjOid) lo_import_internal(text *filename, Oid lobjOid)
{ {
File fd; int fd;
int nbytes, int nbytes,
tmp PG_USED_FOR_ASSERTS_ONLY; tmp PG_USED_FOR_ASSERTS_ONLY;
char buf[BUFSIZE]; char buf[BUFSIZE];
...@@ -464,7 +464,7 @@ lo_import_internal(text *filename, Oid lobjOid) ...@@ -464,7 +464,7 @@ lo_import_internal(text *filename, Oid lobjOid)
* open the file to be read in * open the file to be read in
*/ */
text_to_cstring_buffer(filename, fnamebuf, sizeof(fnamebuf)); text_to_cstring_buffer(filename, fnamebuf, sizeof(fnamebuf));
fd = PathNameOpenFile(fnamebuf, O_RDONLY | PG_BINARY, S_IRWXU); fd = OpenTransientFile(fnamebuf, O_RDONLY | PG_BINARY, S_IRWXU);
if (fd < 0) if (fd < 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -481,7 +481,7 @@ lo_import_internal(text *filename, Oid lobjOid) ...@@ -481,7 +481,7 @@ lo_import_internal(text *filename, Oid lobjOid)
*/ */
lobj = inv_open(oid, INV_WRITE, fscxt); lobj = inv_open(oid, INV_WRITE, fscxt);
while ((nbytes = FileRead(fd, buf, BUFSIZE)) > 0) while ((nbytes = read(fd, buf, BUFSIZE)) > 0)
{ {
tmp = inv_write(lobj, buf, nbytes); tmp = inv_write(lobj, buf, nbytes);
Assert(tmp == nbytes); Assert(tmp == nbytes);
...@@ -494,7 +494,7 @@ lo_import_internal(text *filename, Oid lobjOid) ...@@ -494,7 +494,7 @@ lo_import_internal(text *filename, Oid lobjOid)
fnamebuf))); fnamebuf)));
inv_close(lobj); inv_close(lobj);
FileClose(fd); CloseTransientFile(fd);
return oid; return oid;
} }
...@@ -508,7 +508,7 @@ lo_export(PG_FUNCTION_ARGS) ...@@ -508,7 +508,7 @@ lo_export(PG_FUNCTION_ARGS)
{ {
Oid lobjId = PG_GETARG_OID(0); Oid lobjId = PG_GETARG_OID(0);
text *filename = PG_GETARG_TEXT_PP(1); text *filename = PG_GETARG_TEXT_PP(1);
File fd; int fd;
int nbytes, int nbytes,
tmp; tmp;
char buf[BUFSIZE]; char buf[BUFSIZE];
...@@ -540,7 +540,7 @@ lo_export(PG_FUNCTION_ARGS) ...@@ -540,7 +540,7 @@ lo_export(PG_FUNCTION_ARGS)
*/ */
text_to_cstring_buffer(filename, fnamebuf, sizeof(fnamebuf)); text_to_cstring_buffer(filename, fnamebuf, sizeof(fnamebuf));
oumask = umask(S_IWGRP | S_IWOTH); oumask = umask(S_IWGRP | S_IWOTH);
fd = PathNameOpenFile(fnamebuf, O_CREAT | O_WRONLY | O_TRUNC | PG_BINARY, fd = OpenTransientFile(fnamebuf, O_CREAT | O_WRONLY | O_TRUNC | PG_BINARY,
S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH); S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
umask(oumask); umask(oumask);
if (fd < 0) if (fd < 0)
...@@ -554,7 +554,7 @@ lo_export(PG_FUNCTION_ARGS) ...@@ -554,7 +554,7 @@ lo_export(PG_FUNCTION_ARGS)
*/ */
while ((nbytes = inv_read(lobj, buf, BUFSIZE)) > 0) while ((nbytes = inv_read(lobj, buf, BUFSIZE)) > 0)
{ {
tmp = FileWrite(fd, buf, nbytes); tmp = write(fd, buf, nbytes);
if (tmp != nbytes) if (tmp != nbytes)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -562,7 +562,7 @@ lo_export(PG_FUNCTION_ARGS) ...@@ -562,7 +562,7 @@ lo_export(PG_FUNCTION_ARGS)
fnamebuf))); fnamebuf)));
} }
FileClose(fd); CloseTransientFile(fd);
inv_close(lobj); inv_close(lobj);
PG_RETURN_INT32(1); PG_RETURN_INT32(1);
......
...@@ -162,13 +162,13 @@ copy_file(char *fromfile, char *tofile) ...@@ -162,13 +162,13 @@ copy_file(char *fromfile, char *tofile)
/* /*
* Open the files * Open the files
*/ */
srcfd = BasicOpenFile(fromfile, O_RDONLY | PG_BINARY, 0); srcfd = OpenTransientFile(fromfile, O_RDONLY | PG_BINARY, 0);
if (srcfd < 0) if (srcfd < 0)
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not open file \"%s\": %m", fromfile))); errmsg("could not open file \"%s\": %m", fromfile)));
dstfd = BasicOpenFile(tofile, O_RDWR | O_CREAT | O_EXCL | PG_BINARY, dstfd = OpenTransientFile(tofile, O_RDWR | O_CREAT | O_EXCL | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (dstfd < 0) if (dstfd < 0)
ereport(ERROR, ereport(ERROR,
...@@ -209,12 +209,12 @@ copy_file(char *fromfile, char *tofile) ...@@ -209,12 +209,12 @@ copy_file(char *fromfile, char *tofile)
(void) pg_flush_data(dstfd, offset, nbytes); (void) pg_flush_data(dstfd, offset, nbytes);
} }
if (close(dstfd)) if (CloseTransientFile(dstfd))
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close file \"%s\": %m", tofile))); errmsg("could not close file \"%s\": %m", tofile)));
close(srcfd); CloseTransientFile(srcfd);
pfree(buffer); pfree(buffer);
} }
...@@ -238,11 +238,11 @@ fsync_fname(char *fname, bool isdir) ...@@ -238,11 +238,11 @@ fsync_fname(char *fname, bool isdir)
* cases here * cases here
*/ */
if (!isdir) if (!isdir)
fd = BasicOpenFile(fname, fd = OpenTransientFile(fname,
O_RDWR | PG_BINARY, O_RDWR | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
else else
fd = BasicOpenFile(fname, fd = OpenTransientFile(fname,
O_RDONLY | PG_BINARY, O_RDONLY | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
...@@ -263,7 +263,7 @@ fsync_fname(char *fname, bool isdir) ...@@ -263,7 +263,7 @@ fsync_fname(char *fname, bool isdir)
/* Some OSs don't allow us to fsync directories at all */ /* Some OSs don't allow us to fsync directories at all */
if (returncode != 0 && isdir && errno == EBADF) if (returncode != 0 && isdir && errno == EBADF)
{ {
close(fd); CloseTransientFile(fd);
return; return;
} }
...@@ -272,5 +272,5 @@ fsync_fname(char *fname, bool isdir) ...@@ -272,5 +272,5 @@ fsync_fname(char *fname, bool isdir)
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not fsync file \"%s\": %m", fname))); errmsg("could not fsync file \"%s\": %m", fname)));
close(fd); CloseTransientFile(fd);
} }
...@@ -30,11 +30,29 @@ ...@@ -30,11 +30,29 @@
* routines (e.g., open(2) and fopen(3)) themselves. Otherwise, we * routines (e.g., open(2) and fopen(3)) themselves. Otherwise, we
* may find ourselves short of real file descriptors anyway. * may find ourselves short of real file descriptors anyway.
* *
* This file used to contain a bunch of stuff to support RAID levels 0 * INTERFACE ROUTINES
* (jbod), 1 (duplex) and 5 (xor parity). That stuff is all gone *
* because the parallel query processing code that called it is all * PathNameOpenFile and OpenTemporaryFile are used to open virtual files.
* gone. If you really need it you could get it from the original * A File opened with OpenTemporaryFile is automatically deleted when the
* POSTGRES source. * File is closed, either explicitly or implicitly at end of transaction or
* process exit. PathNameOpenFile is intended for files that are held open
* for a long time, like relation files. It is the caller's responsibility
* to close them, there is no automatic mechanism in fd.c for that.
*
* AllocateFile, AllocateDir and OpenTransientFile are wrappers around
* fopen(3), opendir(3), and open(2), respectively. They behave like the
* corresponding native functions, except that the handle is registered with
* the current subtransaction, and will be automatically closed at abort.
* These are intended for short operations like reading a configuration file.
* and there is a fixed limit on the number files that can be open using these
* functions at any one time.
*
* Finally, BasicOpenFile is a just thin wrapper around open() that can
* release file descriptors in use by the virtual file descriptors if
* necessary. There is no automatic cleanup of file descriptors returned by
* BasicOpenFile, it is solely the caller's responsibility to close the file
* descriptor by calling close(2).
*
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -94,11 +112,11 @@ int max_files_per_process = 1000; ...@@ -94,11 +112,11 @@ int max_files_per_process = 1000;
/* /*
* Maximum number of file descriptors to open for either VFD entries or * Maximum number of file descriptors to open for either VFD entries or
* AllocateFile/AllocateDir operations. This is initialized to a conservative * AllocateFile/AllocateDir/OpenTransientFile operations. This is initialized
* value, and remains that way indefinitely in bootstrap or standalone-backend * to a conservative value, and remains that way indefinitely in bootstrap or
* cases. In normal postmaster operation, the postmaster calls * standalone-backend cases. In normal postmaster operation, the postmaster
* set_max_safe_fds() late in initialization to update the value, and that * calls set_max_safe_fds() late in initialization to update the value, and
* value is then inherited by forked subprocesses. * that value is then inherited by forked subprocesses.
* *
* Note: the value of max_files_per_process is taken into account while * Note: the value of max_files_per_process is taken into account while
* setting this variable, and so need not be tested separately. * setting this variable, and so need not be tested separately.
...@@ -171,10 +189,10 @@ static bool have_xact_temporary_files = false; ...@@ -171,10 +189,10 @@ static bool have_xact_temporary_files = false;
static uint64 temporary_files_size = 0; static uint64 temporary_files_size = 0;
/* /*
* List of stdio FILEs and <dirent.h> DIRs opened with AllocateFile * List of OS handles opened with AllocateFile, AllocateDir and
* and AllocateDir. * OpenTransientFile.
* *
* Since we don't want to encourage heavy use of AllocateFile or AllocateDir, * Since we don't want to encourage heavy use of those functions,
* it seems OK to put a pretty small maximum limit on the number of * it seems OK to put a pretty small maximum limit on the number of
* simultaneously allocated descs. * simultaneously allocated descs.
*/ */
...@@ -183,7 +201,8 @@ static uint64 temporary_files_size = 0; ...@@ -183,7 +201,8 @@ static uint64 temporary_files_size = 0;
typedef enum typedef enum
{ {
AllocateDescFile, AllocateDescFile,
AllocateDescDir AllocateDescDir,
AllocateDescRawFD
} AllocateDescKind; } AllocateDescKind;
typedef struct typedef struct
...@@ -193,6 +212,7 @@ typedef struct ...@@ -193,6 +212,7 @@ typedef struct
{ {
FILE *file; FILE *file;
DIR *dir; DIR *dir;
int fd;
} desc; } desc;
SubTransactionId create_subid; SubTransactionId create_subid;
} AllocateDesc; } AllocateDesc;
...@@ -1523,8 +1543,49 @@ TryAgain: ...@@ -1523,8 +1543,49 @@ TryAgain:
return NULL; return NULL;
} }
/*
* Like AllocateFile, but returns an unbuffered fd like open(2)
*/
int
OpenTransientFile(FileName fileName, int fileFlags, int fileMode)
{
int fd;
DO_DB(elog(LOG, "OpenTransientFile: Allocated %d (%s)",
numAllocatedDescs, fileName));
/*
* The test against MAX_ALLOCATED_DESCS prevents us from overflowing
* allocatedFiles[]; the test against max_safe_fds prevents BasicOpenFile
* from hogging every one of the available FDs, which'd lead to infinite
* looping.
*/
if (numAllocatedDescs >= MAX_ALLOCATED_DESCS ||
numAllocatedDescs >= max_safe_fds - 1)
elog(ERROR, "exceeded MAX_ALLOCATED_DESCS while trying to open file \"%s\"",
fileName);
fd = BasicOpenFile(fileName, fileFlags, fileMode);
if (fd >= 0)
{
AllocateDesc *desc = &allocatedDescs[numAllocatedDescs];
desc->kind = AllocateDescRawFD;
desc->desc.fd = fd;
desc->create_subid = GetCurrentSubTransactionId();
numAllocatedDescs++;
return fd;
}
return -1; /* failure */
}
/* /*
* Free an AllocateDesc of either type. * Free an AllocateDesc of any type.
* *
* The argument *must* point into the allocatedDescs[] array. * The argument *must* point into the allocatedDescs[] array.
*/ */
...@@ -1542,6 +1603,9 @@ FreeDesc(AllocateDesc *desc) ...@@ -1542,6 +1603,9 @@ FreeDesc(AllocateDesc *desc)
case AllocateDescDir: case AllocateDescDir:
result = closedir(desc->desc.dir); result = closedir(desc->desc.dir);
break; break;
case AllocateDescRawFD:
result = close(desc->desc.fd);
break;
default: default:
elog(ERROR, "AllocateDesc kind not recognized"); elog(ERROR, "AllocateDesc kind not recognized");
result = 0; /* keep compiler quiet */ result = 0; /* keep compiler quiet */
...@@ -1583,6 +1647,33 @@ FreeFile(FILE *file) ...@@ -1583,6 +1647,33 @@ FreeFile(FILE *file)
return fclose(file); return fclose(file);
} }
/*
* Close a file returned by OpenTransientFile.
*
* Note we do not check close's return value --- it is up to the caller
* to handle close errors.
*/
int
CloseTransientFile(int fd)
{
int i;
DO_DB(elog(LOG, "CloseTransientFile: Allocated %d", numAllocatedDescs));
/* Remove fd from list of allocated files, if it's present */
for (i = numAllocatedDescs; --i >= 0;)
{
AllocateDesc *desc = &allocatedDescs[i];
if (desc->kind == AllocateDescRawFD && desc->desc.fd == fd)
return FreeDesc(desc);
}
/* Only get here if someone passes us a file not in allocatedDescs */
elog(WARNING, "fd passed to CloseTransientFile was not obtained from OpenTransientFile");
return close(fd);
}
/* /*
* Routines that want to use <dirent.h> (ie, DIR*) should use AllocateDir * Routines that want to use <dirent.h> (ie, DIR*) should use AllocateDir
...@@ -1874,7 +1965,7 @@ AtProcExit_Files(int code, Datum arg) ...@@ -1874,7 +1965,7 @@ AtProcExit_Files(int code, Datum arg)
* exiting. If that's the case, we should remove all temporary files; if * exiting. If that's the case, we should remove all temporary files; if
* that's not the case, we are being called for transaction commit/abort * that's not the case, we are being called for transaction commit/abort
* and should only remove transaction-local temp files. In either case, * and should only remove transaction-local temp files. In either case,
* also clean up "allocated" stdio files and dirs. * also clean up "allocated" stdio files, dirs and fds.
*/ */
static void static void
CleanupTempFiles(bool isProcExit) CleanupTempFiles(bool isProcExit)
...@@ -1916,7 +2007,7 @@ CleanupTempFiles(bool isProcExit) ...@@ -1916,7 +2007,7 @@ CleanupTempFiles(bool isProcExit)
have_xact_temporary_files = false; have_xact_temporary_files = false;
} }
/* Clean up "allocated" stdio files and dirs. */ /* Clean up "allocated" stdio files, dirs and fds. */
while (numAllocatedDescs > 0) while (numAllocatedDescs > 0)
FreeDesc(&allocatedDescs[0]); FreeDesc(&allocatedDescs[0]);
} }
......
...@@ -401,14 +401,14 @@ mdunlinkfork(RelFileNodeBackend rnode, ForkNumber forkNum, bool isRedo) ...@@ -401,14 +401,14 @@ mdunlinkfork(RelFileNodeBackend rnode, ForkNumber forkNum, bool isRedo)
/* truncate(2) would be easier here, but Windows hasn't got it */ /* truncate(2) would be easier here, but Windows hasn't got it */
int fd; int fd;
fd = BasicOpenFile(path, O_RDWR | PG_BINARY, 0); fd = OpenTransientFile(path, O_RDWR | PG_BINARY, 0);
if (fd >= 0) if (fd >= 0)
{ {
int save_errno; int save_errno;
ret = ftruncate(fd, 0); ret = ftruncate(fd, 0);
save_errno = errno; save_errno = errno;
close(fd); CloseTransientFile(fd);
errno = save_errno; errno = save_errno;
} }
else else
......
...@@ -588,7 +588,8 @@ load_relmap_file(bool shared) ...@@ -588,7 +588,8 @@ load_relmap_file(bool shared)
} }
/* Read data ... */ /* Read data ... */
fd = BasicOpenFile(mapfilename, O_RDONLY | PG_BINARY, S_IRUSR | S_IWUSR); fd = OpenTransientFile(mapfilename,
O_RDONLY | PG_BINARY, S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
ereport(FATAL, ereport(FATAL,
(errcode_for_file_access(), (errcode_for_file_access(),
...@@ -608,7 +609,7 @@ load_relmap_file(bool shared) ...@@ -608,7 +609,7 @@ load_relmap_file(bool shared)
errmsg("could not read relation mapping file \"%s\": %m", errmsg("could not read relation mapping file \"%s\": %m",
mapfilename))); mapfilename)));
close(fd); CloseTransientFile(fd);
/* check for correct magic number, etc */ /* check for correct magic number, etc */
if (map->magic != RELMAPPER_FILEMAGIC || if (map->magic != RELMAPPER_FILEMAGIC ||
...@@ -672,12 +673,6 @@ write_relmap_file(bool shared, RelMapFile *newmap, ...@@ -672,12 +673,6 @@ write_relmap_file(bool shared, RelMapFile *newmap,
/* /*
* Open the target file. We prefer to do this before entering the * Open the target file. We prefer to do this before entering the
* critical section, so that an open() failure need not force PANIC. * critical section, so that an open() failure need not force PANIC.
*
* Note: since we use BasicOpenFile, we are nominally responsible for
* ensuring the fd is closed on error. In practice, this isn't important
* because either an error happens inside the critical section, or we are
* in bootstrap or WAL replay; so an error past this point is always fatal
* anyway.
*/ */
if (shared) if (shared)
{ {
...@@ -692,7 +687,7 @@ write_relmap_file(bool shared, RelMapFile *newmap, ...@@ -692,7 +687,7 @@ write_relmap_file(bool shared, RelMapFile *newmap,
realmap = &local_map; realmap = &local_map;
} }
fd = BasicOpenFile(mapfilename, fd = OpenTransientFile(mapfilename,
O_WRONLY | O_CREAT | PG_BINARY, O_WRONLY | O_CREAT | PG_BINARY,
S_IRUSR | S_IWUSR); S_IRUSR | S_IWUSR);
if (fd < 0) if (fd < 0)
...@@ -753,7 +748,7 @@ write_relmap_file(bool shared, RelMapFile *newmap, ...@@ -753,7 +748,7 @@ write_relmap_file(bool shared, RelMapFile *newmap,
errmsg("could not fsync relation mapping file \"%s\": %m", errmsg("could not fsync relation mapping file \"%s\": %m",
mapfilename))); mapfilename)));
if (close(fd)) if (CloseTransientFile(fd))
ereport(ERROR, ereport(ERROR,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close relation mapping file \"%s\": %m", errmsg("could not close relation mapping file \"%s\": %m",
......
...@@ -16,13 +16,13 @@ ...@@ -16,13 +16,13 @@
* calls: * calls:
* *
* File {Close, Read, Write, Seek, Tell, Sync} * File {Close, Read, Write, Seek, Tell, Sync}
* {File Name Open, Allocate, Free} File * {Path Name Open, Allocate, Free} File
* *
* These are NOT JUST RENAMINGS OF THE UNIX ROUTINES. * These are NOT JUST RENAMINGS OF THE UNIX ROUTINES.
* Use them for all file activity... * Use them for all file activity...
* *
* File fd; * File fd;
* fd = FilePathOpenFile("foo", O_RDONLY, 0600); * fd = PathNameOpenFile("foo", O_RDONLY, 0600);
* *
* AllocateFile(); * AllocateFile();
* FreeFile(); * FreeFile();
...@@ -33,7 +33,8 @@ ...@@ -33,7 +33,8 @@
* no way for them to share kernel file descriptors with other files. * no way for them to share kernel file descriptors with other files.
* *
* Likewise, use AllocateDir/FreeDir, not opendir/closedir, to allocate * Likewise, use AllocateDir/FreeDir, not opendir/closedir, to allocate
* open directories (DIR*). * open directories (DIR*), and OpenTransientFile/CloseTransient File for an
* unbuffered file descriptor.
*/ */
#ifndef FD_H #ifndef FD_H
#define FD_H #define FD_H
...@@ -84,6 +85,10 @@ extern DIR *AllocateDir(const char *dirname); ...@@ -84,6 +85,10 @@ extern DIR *AllocateDir(const char *dirname);
extern struct dirent *ReadDir(DIR *dir, const char *dirname); extern struct dirent *ReadDir(DIR *dir, const char *dirname);
extern int FreeDir(DIR *dir); extern int FreeDir(DIR *dir);
/* Operations to allow use of a plain kernel FD, with automatic cleanup */
extern int OpenTransientFile(FileName fileName, int fileFlags, int fileMode);
extern int CloseTransientFile(int fd);
/* If you've really really gotta have a plain kernel FD, use this */ /* If you've really really gotta have a plain kernel FD, use this */
extern int BasicOpenFile(FileName fileName, int fileFlags, int fileMode); extern int BasicOpenFile(FileName fileName, int fileFlags, int fileMode);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment