about summary refs log tree commit diff
path: root/src/libstddjb
diff options
context:
space:
mode:
authorLaurent Bercot <ska-skaware@skarnet.org>2017-09-27 20:16:01 +0000
committerLaurent Bercot <ska-skaware@skarnet.org>2017-09-27 20:16:01 +0000
commitc365aeb0721d3749bb5f94f09a3ad97ec6131805 (patch)
tree0a63fa2d7c499b6e835d5665438f1f67cd7db494 /src/libstddjb
parent1728f7ed8e96a03dd6a75c9668d394562ed63a59 (diff)
downloadskalibs-c365aeb0721d3749bb5f94f09a3ad97ec6131805.tar.gz
skalibs-c365aeb0721d3749bb5f94f09a3ad97ec6131805.tar.xz
skalibs-c365aeb0721d3749bb5f94f09a3ad97ec6131805.zip
Rewrite openwrite(v)nclose using mkstemp
Diffstat (limited to 'src/libstddjb')
-rw-r--r--src/libstddjb/openwritenclose.c38
-rw-r--r--src/libstddjb/openwritenclose_unsafe.c17
-rw-r--r--src/libstddjb/openwritevnclose.c39
-rw-r--r--src/libstddjb/openwritevnclose_unsafe.c17
-rw-r--r--src/libstddjb/writenclose_unsafe.c18
-rw-r--r--src/libstddjb/writevnclose_unsafe.c20
6 files changed, 123 insertions, 26 deletions
diff --git a/src/libstddjb/openwritenclose.c b/src/libstddjb/openwritenclose.c
new file mode 100644
index 0000000..427e53e
--- /dev/null
+++ b/src/libstddjb/openwritenclose.c
@@ -0,0 +1,38 @@
+/* ISC license. */
+
+#include <string.h>
+#include <errno.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <skalibs/djbunix.h>
+
+#define SUFFIX ":skalibs-openwritenclose:XXXXXX"
+
+int openwritenclose_internal (char const *fn, char const *s, size_t n, dev_t *dev, ino_t *ino, int dosync)
+{
+  dev_t tmpdev ;
+  ino_t tmpino ;
+  size_t fnlen = strlen(fn) ;
+  int fd ;
+  char tmp[fnlen + sizeof(SUFFIX)] ;
+  memcpy(tmp, fn, fnlen) ;
+  memcpy(tmp + fnlen, SUFFIX, sizeof(SUFFIX)) ;
+  fd = mkstemp(tmp) ;
+  if (fd < 0) return 0 ;
+  if (!writenclose_unsafe_internal(fd, s, n, dev ? &tmpdev : 0, ino ? &tmpino : 0, dosync)) goto failclose ;
+  if (rename(tmp, fn) < 0) goto fail ;
+  if (dev) *dev = tmpdev ;
+  if (ino) *ino = tmpino ;
+  return 1 ;
+
+ failclose:
+  fd_close(fd) ;
+ fail:
+  {
+    int e = errno ;
+    unlink(tmp) ;
+    errno = e ;
+  }
+  return 0 ;
+}
diff --git a/src/libstddjb/openwritenclose_unsafe.c b/src/libstddjb/openwritenclose_unsafe.c
index 3aebd97..e5ac260 100644
--- a/src/libstddjb/openwritenclose_unsafe.c
+++ b/src/libstddjb/openwritenclose_unsafe.c
@@ -1,30 +1,21 @@
 /* ISC license. */
 
-#include <sys/stat.h>
 #include <errno.h>
 #include <unistd.h>
-#include <skalibs/allreadwrite.h>
 #include <skalibs/djbunix.h>
 
 int openwritenclose_unsafe_internal (char const *fn, char const *s, size_t len, dev_t *dev, ino_t *ino, int dosync)
 {
-  struct stat st ;
   int fd = open_trunc(fn) ;
   if (fd < 0) return 0 ;
-  if (allwrite(fd, s, len) < len) goto fail ;
-  if ((dev || ino) && (fstat(fd, &st) < 0)) goto fail ;
-  if (dosync && (fd_sync(fd) < 0) && (errno != EINVAL)) goto fail ;
-  fd_close(fd) ;
-  if (dev) *dev = st.st_dev ;
-  if (ino) *ino = st.st_ino ;
-  return 1 ;
-
- fail:
+  if (!writenclose_unsafe_internal(fd, s, len, dev, ino, dosync))
   {
     int e = errno ;
     fd_close(fd) ;
     unlink(fn) ;
     errno = e ;
+    return 0 ;
   }
-  return 0 ;
+  fd_close(fd) ;
+  return 1 ;
 }
diff --git a/src/libstddjb/openwritevnclose.c b/src/libstddjb/openwritevnclose.c
new file mode 100644
index 0000000..01df3e3
--- /dev/null
+++ b/src/libstddjb/openwritevnclose.c
@@ -0,0 +1,39 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+#include <string.h>
+#include <errno.h>
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <skalibs/djbunix.h>
+
+#define SUFFIX ":skalibs-openwritevnclose:XXXXXX"
+
+int openwritevnclose_internal (char const *fn, struct iovec const *v, unsigned int vlen, dev_t *dev, ino_t *ino, int dosync)
+{
+  dev_t tmpdev ;
+  ino_t tmpino ;
+  size_t fnlen = strlen(fn) ;
+  int fd ;
+  char tmp[fnlen + sizeof(SUFFIX)] ;
+  memcpy(tmp, fn, fnlen) ;
+  memcpy(tmp + fnlen, SUFFIX, sizeof(SUFFIX)) ;
+  fd = mkstemp(tmp) ;
+  if (fd < 0) return 0 ;
+  if (!writevnclose_unsafe_internal(fd, v, vlen, dev ? &tmpdev : 0, ino ? &tmpino : 0, dosync)) goto failclose ;
+  if (rename(tmp, fn) < 0) goto fail ;
+  if (dev) *dev = tmpdev ;
+  if (ino) *ino = tmpino ;
+  return 1 ;
+
+ failclose:
+  fd_close(fd) ;
+ fail:
+  {
+    int e = errno ;
+    unlink(tmp) ;
+    errno = e ;
+  }
+  return 0 ;
+}
diff --git a/src/libstddjb/openwritevnclose_unsafe.c b/src/libstddjb/openwritevnclose_unsafe.c
index 4bc7b2b..ddcbfd3 100644
--- a/src/libstddjb/openwritevnclose_unsafe.c
+++ b/src/libstddjb/openwritevnclose_unsafe.c
@@ -1,32 +1,23 @@
 /* ISC license. */
 
 #include <sys/uio.h>
-#include <sys/stat.h>
 #include <errno.h>
 #include <unistd.h>
 #include <skalibs/allreadwrite.h>
-#include <skalibs/siovec.h>
 #include <skalibs/djbunix.h>
 
 int openwritevnclose_unsafe_internal (char const *fn, struct iovec const *v, unsigned int vlen, dev_t *dev, ino_t *ino, int dosync)
 {
-  struct stat st ;
   int fd = open_trunc(fn) ;
   if (fd < 0) return 0 ;
-  if (allwritev(fd, v, vlen) < siovec_len(v, vlen)) goto fail ;
-  if ((dev || ino) && (fstat(fd, &st) < 0)) goto fail ;
-  if (dosync && (fd_sync(fd) < 0) && (errno != EINVAL)) goto fail ;
-  fd_close(fd) ;
-  if (dev) *dev = st.st_dev ;
-  if (ino) *ino = st.st_ino ;
-  return 1 ;
-
- fail:
+  if (!writevnclose_unsafe_internal(fd, v, vlen, dev, ino, dosync))
   {
     int e = errno ;
     fd_close(fd) ;
     unlink(fn) ;
     errno = e ;
+    return 0 ;
   }
-  return 0 ;
+  fd_close(fd) ;
+  return 1 ;
 }
diff --git a/src/libstddjb/writenclose_unsafe.c b/src/libstddjb/writenclose_unsafe.c
new file mode 100644
index 0000000..f71caea
--- /dev/null
+++ b/src/libstddjb/writenclose_unsafe.c
@@ -0,0 +1,18 @@
+/* ISC license. */
+
+#include <sys/stat.h>
+#include <errno.h>
+#include <skalibs/allreadwrite.h>
+#include <skalibs/djbunix.h>
+
+int writenclose_unsafe_internal (int fd, char const *s, size_t len, dev_t *dev, ino_t *ino, int dosync)
+{
+  struct stat st ;
+  if (allwrite(fd, s, len) < len) return 0 ;
+  if ((dev || ino) && (fstat(fd, &st) < 0)) return 0 ;
+  if (dosync && (fd_sync(fd) < 0) && (errno != EINVAL)) return 0 ;
+  fd_close(fd) ;
+  if (dev) *dev = st.st_dev ;
+  if (ino) *ino = st.st_ino ;
+  return 1 ;
+}
diff --git a/src/libstddjb/writevnclose_unsafe.c b/src/libstddjb/writevnclose_unsafe.c
new file mode 100644
index 0000000..9d2348a
--- /dev/null
+++ b/src/libstddjb/writevnclose_unsafe.c
@@ -0,0 +1,20 @@
+/* ISC license. */
+
+#include <sys/uio.h>
+#include <sys/stat.h>
+#include <errno.h>
+#include <skalibs/allreadwrite.h>
+#include <skalibs/siovec.h>
+#include <skalibs/djbunix.h>
+
+int writevnclose_unsafe_internal (int fd, struct iovec const *v, unsigned int vlen, dev_t *dev, ino_t *ino, int dosync)
+{
+  struct stat st ;
+  if (allwritev(fd, v, vlen) < siovec_len(v, vlen)) return 0 ;
+  if ((dev || ino) && (fstat(fd, &st) < 0)) return 0 ;
+  if (dosync && (fd_sync(fd) < 0) && (errno != EINVAL)) return 0 ;
+  fd_close(fd) ;
+  if (dev) *dev = st.st_dev ;
+  if (ino) *ino = st.st_ino ;
+  return 1 ;
+}