[llvm-commits] CVS: llvm/lib/System/Unix/Path.cpp
Reid Spencer
reid at x10sys.com
Thu Dec 23 22:29:53 PST 2004
Changes in directory llvm/lib/System/Unix:
Path.cpp updated: 1.27 -> 1.28
---
Log message:
For PR351: http://llvm.cs.uiuc.edu/PR351 :
Merge implementations of isValid and GetTemporaryDirectory into this file. There
is not any point having the operating system specific files for such little
variation between the Unix family of systems.
---
Diffs of the changes: (+89 -0)
Index: llvm/lib/System/Unix/Path.cpp
diff -u llvm/lib/System/Unix/Path.cpp:1.27 llvm/lib/System/Unix/Path.cpp:1.28
--- llvm/lib/System/Unix/Path.cpp:1.27 Mon Dec 20 21:27:08 2004
+++ llvm/lib/System/Unix/Path.cpp Fri Dec 24 00:29:42 2004
@@ -22,6 +22,7 @@
#include <fcntl.h>
#include <utime.h>
#include <dirent.h>
+#include <time.h>
namespace llvm {
using namespace sys;
@@ -36,6 +37,21 @@
ThrowErrno(unverified_path + ": path is not valid");
}
+bool
+Path::isValid() const {
+ if (path.empty())
+ return false;
+ else if (path.length() >= MAXPATHLEN)
+ return false;
+#if defined(HAVE_REALPATH)
+ char pathname[MAXPATHLEN];
+ if (0 == realpath(path.c_str(), pathname))
+ if (errno != EACCES && errno != EIO && errno != ENOENT && errno != ENOTDIR)
+ return false;
+#endif
+ return true;
+}
+
Path
Path::GetRootDirectory() {
Path result;
@@ -43,6 +59,79 @@
return result;
}
+Path
+Path::GetTemporaryDirectory() {
+#if defined(HAVE_MKDTEMP)
+ // The best way is with mkdtemp but that's not available on many systems,
+ // Linux and FreeBSD have it. Others probably won't.
+ char pathname[MAXPATHLEN];
+ strcpy(pathname,"/tmp/llvm_XXXXXX");
+ if (0 == mkdtemp(pathname))
+ ThrowErrno(std::string(pathname) + ": Can't create temporary directory");
+ Path result;
+ result.setDirectory(pathname);
+ assert(result.isValid() && "mkdtemp didn't create a valid pathname!");
+ return result;
+#elif defined(HAVE_MKSTEMP)
+ // If no mkdtemp is available, mkstemp can be used to create a temporary file
+ // which is then removed and created as a directory. We prefer this over
+ // mktemp because of mktemp's inherent security and threading risks. We still
+ // have a slight race condition from the time the temporary file is created to
+ // the time it is re-created as a directoy.
+ char pathname[MAXPATHLEN];
+ strcpy(pathname, "/tmp/llvm_XXXXXX");
+ int fd = 0;
+ if (-1 == (fd = mkstemp(pathname)))
+ ThrowErrno(std::string(pathname) + ": Can't create temporary directory");
+ ::close(fd);
+ ::unlink(pathname); // start race condition, ignore errors
+ if (-1 == ::mkdir(pathname, S_IRWXU)) // end race condition
+ ThrowErrno(std::string(pathname) + ": Can't create temporary directory");
+ Path result;
+ result.setDirectory(pathname);
+ assert(result.isValid() && "mkstemp didn't create a valid pathname!");
+ return result;
+#elif defined(HAVE_MKTEMP)
+ // If a system doesn't have mkdtemp(3) or mkstemp(3) but it does have
+ // mktemp(3) then we'll assume that system (e.g. AIX) has a reasonable
+ // implementation of mktemp(3) and doesn't follow BSD 4.3's lead of replacing
+ // the XXXXXX with the pid of the process and a letter. That leads to only
+ // twenty six temporary files that can be generated.
+ char pathname[MAXPATHLEN];
+ strcpy(pathname, "/tmp/llvm_XXXXXX");
+ char *TmpName = ::mktemp(pathname);
+ if (TmpName == 0)
+ throw std::string(TmpName) + ": Can't create unique directory name";
+ if (-1 == ::mkdir(TmpName, S_IRWXU))
+ ThrowErrno(std::string(TmpName) + ": Can't create temporary directory");
+ Path result;
+ result.setDirectory(TmpName);
+ assert(result.isValid() && "mktemp didn't create a valid pathname!");
+ return result;
+#else
+ // This is the worst case implementation. tempnam(3) leaks memory unless its
+ // on an SVID2 (or later) system. On BSD 4.3 it leaks. tmpnam(3) has thread
+ // issues. The mktemp(3) function doesn't have enough variability in the
+ // temporary name generated. So, we provide our own implementation that
+ // increments an integer from a random number seeded by the current time. This
+ // should be sufficiently unique that we don't have many collisions between
+ // processes. Generally LLVM processes don't run very long and don't use very
+ // many temporary files so this shouldn't be a big issue for LLVM.
+ static time_t num = ::time(0);
+ char pathname[MAXPATHLEN];
+ do {
+ num++;
+ sprintf(pathname, "/tmp/llvm_%010u", unsigned(num));
+ } while ( 0 == access(pathname, F_OK ) );
+ if (-1 == ::mkdir(pathname, S_IRWXU))
+ ThrowErrno(std::string(pathname) + ": Can't create temporary directory");
+ Path result;
+ result.setDirectory(pathname);
+ assert(result.isValid() && "mkstemp didn't create a valid pathname!");
+ return result;
+#endif
+}
+
static void getPathList(const char*path, std::vector<sys::Path>& Paths) {
const char* at = path;
const char* delim = strchr(at, ':');
More information about the llvm-commits
mailing list