diff options
author | Robert Haas <rhaas@postgresql.org> | 2013-09-04 11:15:00 -0400 |
---|---|---|
committer | Robert Haas <rhaas@postgresql.org> | 2013-09-04 11:15:00 -0400 |
commit | cc52d5b33ff5df29de57dcae9322214cfe9c8464 (patch) | |
tree | b32f5a33953574a45e1bd96e21a50e5caf5c9c41 /src/backend/storage/file/fd.c | |
parent | 0c66a223774dec62edb5281a47e72fe480a8f7aa (diff) | |
download | postgresql-cc52d5b33ff5df29de57dcae9322214cfe9c8464.tar.gz postgresql-cc52d5b33ff5df29de57dcae9322214cfe9c8464.zip |
Expose fsync_fname as a public API.
Andres Freund
Diffstat (limited to 'src/backend/storage/file/fd.c')
-rw-r--r-- | src/backend/storage/file/fd.c | 56 |
1 files changed, 56 insertions, 0 deletions
diff --git a/src/backend/storage/file/fd.c b/src/backend/storage/file/fd.c index 436b901c20d..de4d90205a8 100644 --- a/src/backend/storage/file/fd.c +++ b/src/backend/storage/file/fd.c @@ -385,6 +385,62 @@ pg_flush_data(int fd, off_t offset, off_t amount) /* + * fsync_fname -- fsync a file or directory, handling errors properly + * + * Try to fsync a file or directory. When doing the latter, ignore errors that + * indicate the OS just doesn't allow/require fsyncing directories. + */ +void +fsync_fname(char *fname, bool isdir) +{ + int fd; + int returncode; + + /* + * Some OSs require directories to be opened read-only whereas other + * systems don't allow us to fsync files opened read-only; so we need both + * cases here + */ + if (!isdir) + fd = OpenTransientFile(fname, + O_RDWR | PG_BINARY, + S_IRUSR | S_IWUSR); + else + fd = OpenTransientFile(fname, + O_RDONLY | PG_BINARY, + S_IRUSR | S_IWUSR); + + /* + * Some OSs don't allow us to open directories at all (Windows returns + * EACCES) + */ + if (fd < 0 && isdir && (errno == EISDIR || errno == EACCES)) + return; + + else if (fd < 0) + ereport(ERROR, + (errcode_for_file_access(), + errmsg("could not open file \"%s\": %m", fname))); + + returncode = pg_fsync(fd); + + /* Some OSs don't allow us to fsync directories at all */ + if (returncode != 0 && isdir && errno == EBADF) + { + CloseTransientFile(fd); + return; + } + + if (returncode != 0) + ereport(ERROR, + (errcode_for_file_access(), + errmsg("could not fsync file \"%s\": %m", fname))); + + CloseTransientFile(fd); +} + + +/* * InitFileAccess --- initialize this module during backend startup * * This is called during either normal or standalone backend start. |