Location via proxy:   [ UP ]  
[Report a bug]   [Manage cookies]                
Add mkdtemp() to libpgport.
authorNoah Misch <noah@leadboat.com>
Sat, 14 Jun 2014 13:41:13 +0000 (09:41 -0400)
committerNoah Misch <noah@leadboat.com>
Sat, 14 Jun 2014 13:41:18 +0000 (09:41 -0400)
This function is pervasive on free software operating systems; import
NetBSD's implementation.  Back-patch to 8.4, like the commit that will
harness it.

configure
configure.in
src/include/pg_config.h.in
src/include/pg_config.h.win32
src/include/port.h
src/port/mkdtemp.c [new file with mode: 0644]
src/tools/msvc/Mkvcbuild.pm

index dbc069e47ca769e56df4af9f6a03f865f5e659d5..3c00d11d8a9e2aca1cd66a3751c0f2d4ecf1acf9 100755 (executable)
--- a/configure
+++ b/configure
@@ -20646,7 +20646,8 @@ fi
 
 
 
-for ac_func in crypt erand48 getopt getrusage inet_aton random rint srandom strdup strerror strlcat strlcpy strtol strtoul
+
+for ac_func in crypt erand48 getopt getrusage inet_aton mkdtemp random rint srandom strdup strerror strlcat strlcpy strtol strtoul
 do
 as_ac_var=`$as_echo "ac_cv_func_$ac_func" | $as_tr_sh`
 { $as_echo "$as_me:$LINENO: checking for $ac_func" >&5
index 36eef2c12976ee097650c06061cb7ed1d9e4f595..70c7360a42b759e88ccdee5dabd28c156388d2b4 100644 (file)
@@ -1325,7 +1325,7 @@ else
   AC_CHECK_FUNCS([fpclass fp_class fp_class_d class], [break])
 fi
 
-AC_REPLACE_FUNCS([crypt erand48 getopt getrusage inet_aton random rint srandom strdup strerror strlcat strlcpy strtol strtoul])
+AC_REPLACE_FUNCS([crypt erand48 getopt getrusage inet_aton mkdtemp random rint srandom strdup strerror strlcat strlcpy strtol strtoul])
 
 case $host_os in
 
index f901f82676b7135e3c151dae90b96fff552d1fe8..3ce73698ae955f8029d175f0fdff71f820d5ef5e 100644 (file)
 /* Define to 1 if the system has the type `MINIDUMP_TYPE'. */
 #undef HAVE_MINIDUMP_TYPE
 
+/* Define to 1 if you have the `mkdtemp' function. */
+#undef HAVE_MKDTEMP
+
 /* Define to 1 if you have the <netinet/in.h> header file. */
 #undef HAVE_NETINET_IN_H
 
index 13a65c7a2b5bb5c2128a53b7eee6a88019482ab3..c43e91a860befff032ec3a61b7c60f0c5924ffba 100644 (file)
 /* Define to 1 if the system has the type `MINIDUMP_TYPE'. */
 #define HAVE_MINIDUMP_TYPE 1
 
+/* Define to 1 if you have the `mkdtemp' function. */
+/* #undef HAVE_MKDTEMP */
+
 /* Define to 1 if you have the <netinet/in.h> header file. */
 #define HAVE_NETINET_IN_H 1
 
index 8563cd6158bae90c17578ab024a24bd95b4b2fc9..2dd50c9478b04e1db18569aafa0c124fb33ec106 100644 (file)
@@ -497,4 +497,7 @@ extern int  pg_check_dir(const char *dir);
 /* port/pgmkdirp.c */
 extern int pg_mkdir_p(char *path, int omode);
 
+/* port/mkdtemp.c */
+extern char *mkdtemp(char *path);
+
 #endif   /* PG_PORT_H */
diff --git a/src/port/mkdtemp.c b/src/port/mkdtemp.c
new file mode 100644 (file)
index 0000000..a5e991f
--- /dev/null
@@ -0,0 +1,293 @@
+/*-------------------------------------------------------------------------
+ *
+ * mkdtemp.c
+ *   create a mode-0700 temporary directory
+ *
+ * Portions Copyright (c) 1996-2014, PostgreSQL Global Development Group
+ *
+ *
+ * IDENTIFICATION
+ *   src/port/mkdtemp.c
+ *
+ * This code was taken from NetBSD to provide an implementation for platforms
+ * that lack it.  (Among compatibly-licensed implementations, the OpenBSD
+ * version better resists denial-of-service attacks.  However, it has a
+ * cryptographic dependency.)  The NetBSD copyright terms follow.
+ *-------------------------------------------------------------------------
+ */
+
+#include "c.h"
+
+#define _DIAGASSERT(x) do {} while (0)
+
+
+/* $NetBSD: gettemp.c,v 1.17 2014/01/21 19:09:48 seanb Exp $   */
+
+/*
+ * Copyright (c) 1987, 1993
+ * The Regents of the University of California.  All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *   notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *   notice, this list of conditions and the following disclaimer in the
+ *   documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of the University nor the names of its contributors
+ *   may be used to endorse or promote products derived from this software
+ *   without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#if HAVE_NBTOOL_CONFIG_H
+#include "nbtool_config.h"
+#endif
+
+#if !HAVE_NBTOOL_CONFIG_H || !HAVE_MKSTEMP || !HAVE_MKDTEMP
+
+#ifdef NOT_POSTGRESQL
+#include <sys/cdefs.h>
+#if defined(LIBC_SCCS) && !defined(lint)
+#if 0
+static char sccsid[] = "@(#)mktemp.c   8.1 (Berkeley) 6/4/93";
+#else
+__RCSID("$NetBSD: gettemp.c,v 1.17 2014/01/21 19:09:48 seanb Exp $");
+#endif
+#endif   /* LIBC_SCCS and not lint */
+#endif
+
+#include <sys/types.h>
+#include <sys/stat.h>
+
+#include <assert.h>
+#include <ctype.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+
+#ifdef NOT_POSTGRESQL
+#if HAVE_NBTOOL_CONFIG_H
+#define GETTEMP        __nbcompat_gettemp
+#else
+#include "reentrant.h"
+#include "local.h"
+#define GETTEMP        __gettemp
+#endif
+#endif
+
+static int
+GETTEMP(char *path, int *doopen, int domkdir)
+{
+   char       *start,
+              *trv;
+   struct stat sbuf;
+   u_int       pid;
+
+   /*
+    * To guarantee multiple calls generate unique names even if the file is
+    * not created. 676 different possibilities with 7 or more X's, 26 with 6
+    * or less.
+    */
+   static char xtra[2] = "aa";
+   int         xcnt = 0;
+
+   _DIAGASSERT(path != NULL);
+   /* doopen may be NULL */
+
+   pid = getpid();
+
+   /* Move to end of path and count trailing X's. */
+   for (trv = path; *trv; ++trv)
+       if (*trv == 'X')
+           xcnt++;
+       else
+           xcnt = 0;
+
+   /* Use at least one from xtra.  Use 2 if more than 6 X's. */
+   if (xcnt > 0)
+   {
+       *--trv = xtra[0];
+       xcnt--;
+   }
+   if (xcnt > 5)
+   {
+       *--trv = xtra[1];
+       xcnt--;
+   }
+
+   /* Set remaining X's to pid digits with 0's to the left. */
+   for (; xcnt > 0; xcnt--)
+   {
+       *--trv = (pid % 10) + '0';
+       pid /= 10;
+   }
+
+   /* update xtra for next call. */
+   if (xtra[0] != 'z')
+       xtra[0]++;
+   else
+   {
+       xtra[0] = 'a';
+       if (xtra[1] != 'z')
+           xtra[1]++;
+       else
+           xtra[1] = 'a';
+   }
+
+   /*
+    * check the target directory; if you have six X's and it doesn't exist
+    * this runs for a *very* long time.
+    */
+   for (start = trv + 1;; --trv)
+   {
+       if (trv <= path)
+           break;
+       if (*trv == '/')
+       {
+           int         e;
+
+           *trv = '\0';
+           e = stat(path, &sbuf);
+           *trv = '/';
+           if (e == -1)
+               return doopen == NULL && !domkdir;
+           if (!S_ISDIR(sbuf.st_mode))
+           {
+               errno = ENOTDIR;
+               return doopen == NULL && !domkdir;
+           }
+           break;
+       }
+   }
+
+   for (;;)
+   {
+       if (doopen)
+       {
+           if ((*doopen =
+                open(path, O_CREAT | O_EXCL | O_RDWR, 0600)) >= 0)
+               return 1;
+           if (errno != EEXIST)
+               return 0;
+       }
+       else if (domkdir)
+       {
+           if (mkdir(path, 0700) >= 0)
+               return 1;
+           if (errno != EEXIST)
+               return 0;
+       }
+       else if (lstat(path, &sbuf))
+           return errno == ENOENT ? 1 : 0;
+
+       /* tricky little algorithm for backward compatibility */
+       for (trv = start;;)
+       {
+           if (!*trv)
+               return 0;
+           if (*trv == 'z')
+               *trv++ = 'a';
+           else
+           {
+               if (isdigit((unsigned char) *trv))
+                   *trv = 'a';
+               else
+                   ++* trv;
+               break;
+           }
+       }
+   }
+   /* NOTREACHED */
+}
+
+#endif   /* !HAVE_NBTOOL_CONFIG_H || !HAVE_MKSTEMP ||
+                                * !HAVE_MKDTEMP */
+
+
+/* $NetBSD: mkdtemp.c,v 1.11 2012/03/15 18:22:30 christos Exp $    */
+
+/*
+ * Copyright (c) 1987, 1993
+ * The Regents of the University of California.  All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *   notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *   notice, this list of conditions and the following disclaimer in the
+ *   documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of the University nor the names of its contributors
+ *   may be used to endorse or promote products derived from this software
+ *   without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#if HAVE_NBTOOL_CONFIG_H
+#include "nbtool_config.h"
+#endif
+
+#if !HAVE_NBTOOL_CONFIG_H || !HAVE_MKDTEMP
+
+#ifdef NOT_POSTGRESQL
+
+#include <sys/cdefs.h>
+#if defined(LIBC_SCCS) && !defined(lint)
+#if 0
+static char sccsid[] = "@(#)mktemp.c   8.1 (Berkeley) 6/4/93";
+#else
+__RCSID("$NetBSD: mkdtemp.c,v 1.11 2012/03/15 18:22:30 christos Exp $");
+#endif
+#endif   /* LIBC_SCCS and not lint */
+
+#if HAVE_NBTOOL_CONFIG_H
+#define GETTEMP        __nbcompat_gettemp
+#else
+#include <assert.h>
+#include <errno.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include "reentrant.h"
+#include "local.h"
+#define GETTEMP        __gettemp
+#endif
+
+#endif
+
+char *
+mkdtemp(char *path)
+{
+   _DIAGASSERT(path != NULL);
+
+   return GETTEMP(path, NULL, 1) ? path : NULL;
+}
+
+#endif   /* !HAVE_NBTOOL_CONFIG_H || !HAVE_MKDTEMP */
index 7883587f9546bdd4d3fc7cc310d17dc0eae8fda0..5c36f319682b861586052624b5a9f3e30ff1881e 100644 (file)
@@ -55,7 +55,7 @@ sub mkvcbuild
       snprintf.c strlcat.c strlcpy.c dirmod.c exec.c noblock.c path.c
       pgcheckdir.c pgmkdirp.c pgsleep.c pgstrcasecmp.c qsort.c qsort_arg.c
       sprompt.c thread.c getopt.c getopt_long.c dirent.c rint.c win32env.c
-      win32error.c win32setlocale.c);
+      win32error.c win32setlocale.c mkdtemp.c);
 
     $libpgport = $solution->AddProject('libpgport','lib','misc');
     $libpgport->AddDefine('FRONTEND');