aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorGreg Stark <stark@mit.edu>2010-02-14 17:50:39 +0000
committerGreg Stark <stark@mit.edu>2010-02-14 17:50:39 +0000
commit1534023ff0bd836ec851f780fa3fa1289ecb752f (patch)
tree35e54593307079fee2aedfc6b257f1bf3f59794b /src
parentc53917dc66bb1df640636f96d7803e8ace2889d3 (diff)
downloadpostgresql-1534023ff0bd836ec851f780fa3fa1289ecb752f.tar.gz
postgresql-1534023ff0bd836ec851f780fa3fa1289ecb752f.zip
Make CREATE DATABASE safe against losing whole files by fsyncing the
directory and not just the individual files. Back-patch to 8.1 -- before that we just called "cp -r" and never fsynced anything anyways.
Diffstat (limited to 'src')
-rw-r--r--src/port/copydir.c23
1 files changed, 22 insertions, 1 deletions
diff --git a/src/port/copydir.c b/src/port/copydir.c
index f12732cf212..4cc6e4e0b6f 100644
--- a/src/port/copydir.c
+++ b/src/port/copydir.c
@@ -11,7 +11,7 @@
* as a service.
*
* IDENTIFICATION
- * $PostgreSQL: pgsql/src/port/copydir.c,v 1.18.2.1 2008/03/31 01:32:17 tgl Exp $
+ * $PostgreSQL: pgsql/src/port/copydir.c,v 1.18.2.2 2010/02/14 17:50:39 stark Exp $
*
*-------------------------------------------------------------------------
*/
@@ -50,6 +50,7 @@ copydir(char *fromdir, char *todir, bool recurse)
{
DIR *xldir;
struct dirent *xlde;
+ int dirfd;
char fromfile[MAXPGPATH];
char tofile[MAXPGPATH];
@@ -91,6 +92,26 @@ copydir(char *fromdir, char *todir, bool recurse)
}
FreeDir(xldir);
+
+ /*
+ * fsync the directory to make sure not just the data but also the
+ * new directory file entries have reached the disk. While needed
+ * by most filesystems, the window got bigger with newer ones like
+ * ext4.
+ */
+ dirfd = BasicOpenFile(todir,
+ O_RDONLY | PG_BINARY,
+ S_IRUSR | S_IWUSR);
+ if(dirfd == -1)
+ ereport(ERROR,
+ (errcode_for_file_access(),
+ errmsg("could not open directory for fsync \"%s\": %m", todir)));
+
+ if(pg_fsync(dirfd) == -1)
+ ereport(ERROR,
+ (errcode_for_file_access(),
+ errmsg("could not fsync directory \"%s\": %m", todir)));
+ close(dirfd);
}
/*