* Use low-level unix I/O interface since we're on Unix.
[oota-llvm.git] / lib / System / Unix / Path.cpp
index c9333decbfd64e1c573961b47dc511ffa1822b6b..70cc4f01032f2253e502ecc98a7120718f87c7c9 100644 (file)
 //===----------------------------------------------------------------------===//
 
 #include <llvm/Config/config.h>
+#include <llvm/Config/alloca.h>
 #include "Unix.h"
 #include <sys/stat.h>
 #include <fcntl.h>
 #include <fstream>
+#include <utime.h>
+#include <dirent.h>
 
 namespace llvm {
 using namespace sys;
@@ -30,7 +33,7 @@ Path::Path(std::string unverified_path)
 {
   if (unverified_path.empty())
     return;
-  if (this->is_valid()) 
+  if (this->isValid()) 
     return;
   // oops, not valid.
   path.clear();
@@ -40,28 +43,28 @@ Path::Path(std::string unverified_path)
 Path
 Path::GetRootDirectory() {
   Path result;
-  result.set_directory("/");
+  result.setDirectory("/");
   return result;
 }
 
 static inline bool IsLibrary(Path& path, const std::string& basename) {
-  if (path.append_file(std::string("lib") + basename)) {
-    if (path.append_suffix(Path::GetDLLSuffix()) && path.readable())
+  if (path.appendFile(std::string("lib") + basename)) {
+    if (path.appendSuffix(Path::GetDLLSuffix()) && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("a") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("a") && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("o") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("o") && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("bc") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("bc") && path.readable())
       return true;
-  } else if (path.elide_file() && path.append_file(basename)) {
-    if (path.append_suffix(Path::GetDLLSuffix()) && path.readable())
+  } else if (path.elideFile() && path.appendFile(basename)) {
+    if (path.appendSuffix(Path::GetDLLSuffix()) && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("a") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("a") && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("o") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("o") && path.readable())
       return true;
-    else if (path.elide_suffix() && path.append_suffix("bc") && path.readable())
+    else if (path.elideSuffix() && path.appendSuffix("bc") && path.readable())
       return true;
   }
   path.clear();
@@ -76,20 +79,20 @@ Path::GetLibraryPath(const std::string& basename,
   // Try the paths provided
   for (std::vector<std::string>::const_iterator I = LibPaths.begin(),
        E = LibPaths.end(); I != E; ++I ) {
-    if (result.set_directory(*I) && IsLibrary(result,basename))
+    if (result.setDirectory(*I) && IsLibrary(result,basename))
       return result;
   }
 
   // Try the LLVM lib directory in the LLVM install area
-  if (result.set_directory(LLVM_LIBDIR) && IsLibrary(result,basename))
+  if (result.setDirectory(LLVM_LIBDIR) && IsLibrary(result,basename))
     return result;
 
   // Try /usr/lib
-  if (result.set_directory("/usr/lib/") && IsLibrary(result,basename))
+  if (result.setDirectory("/usr/lib/") && IsLibrary(result,basename))
     return result;
 
   // Try /lib
-  if (result.set_directory("/lib/") && IsLibrary(result,basename))
+  if (result.setDirectory("/lib/") && IsLibrary(result,basename))
     return result;
 
   // Can't find it, give up and return invalid path.
@@ -115,7 +118,7 @@ Path::GetLLVMDefaultConfigDir() {
 Path 
 Path::GetLLVMConfigDir() {
   Path result;
-  if (result.set_directory(LLVM_ETCDIR))
+  if (result.setDirectory(LLVM_ETCDIR))
     return result;
   return GetLLVMDefaultConfigDir();
 }
@@ -125,24 +128,24 @@ Path::GetUserHomeDirectory() {
   const char* home = getenv("HOME");
   if (home) {
     Path result;
-    if (result.set_directory(home))
+    if (result.setDirectory(home))
       return result;
   }
   return GetRootDirectory();
 }
 
 bool
-Path::is_file() const {
-  return (is_valid() && path[path.length()-1] != '/');
+Path::isFile() const {
+  return (isValid() && path[path.length()-1] != '/');
 }
 
 bool
-Path::is_directory() const {
-  return (is_valid() && path[path.length()-1] == '/');
+Path::isDirectory() const {
+  return (isValid() && path[path.length()-1] == '/');
 }
 
 std::string
-Path::get_basename() const {
+Path::getBasename() const {
   // Find the last slash
   size_t slash = path.rfind('/');
   if (slash == std::string::npos)
@@ -153,27 +156,53 @@ Path::get_basename() const {
   return path.substr(slash, path.rfind('.'));
 }
 
-bool Path::has_magic_number(const std::string &Magic) const {
+bool Path::hasMagicNumber(const std::string &Magic) const {
   size_t len = Magic.size();
-  char buf[ 1 + len];
-  std::ifstream f(path.c_str());
-  f.read(buf, len);
+  assert(len < 1024 && "Request for magic string too long");
+  char* buf = (char*) alloca(1 + len);
+  int fd = ::open(path.c_str(),O_RDONLY);
+  if (fd < 0)
+    return false;
+  if (0 != ::read(fd, buf, len))
+    return false;
+  close(fd);
   buf[len] = '\0';
   return Magic == buf;
 }
 
+bool Path::getMagicNumber(std::string& Magic, unsigned len) const {
+  if (!isFile())
+    return false;
+  assert(len < 1024 && "Request for magic string too long");
+  char* buf = (char*) alloca(1 + len);
+  int fd = ::open(path.c_str(),O_RDONLY);
+  if (fd < 0)
+    return false;
+  if (0 != ::read(fd, buf, len))
+    return false;
+  close(fd);
+  buf[len] = '\0';
+  Magic = buf;
+  return true;
+}
+
 bool 
-Path::is_bytecode_file() const {
-  if (readable()) {
-    return has_magic_number("llvm");
-  }
-  return false;
+Path::isBytecodeFile() const {
+  char buffer[ 4];
+  buffer[0] = 0;
+  std::ifstream f(path.c_str());
+  f.read(buffer, 4);
+  if (f.bad())
+    ThrowErrno("can't read file signature");
+
+  return (buffer[0] == 'l' && buffer[1] == 'l' && buffer[2] == 'v' &&
+      (buffer[3] == 'c' || buffer[3] == 'm'));
 }
 
 bool
-Path::is_archive() const {
+Path::isArchive() const {
   if (readable()) {
-    return has_magic_number("!<arch>\012");
+    return hasMagicNumber("!<arch>\012");
   }
   return false;
 }
@@ -220,8 +249,51 @@ Path::getLast() const {
   return path.substr(pos+1);
 }
 
+void
+Path::getStatusInfo(StatusInfo& info) const {
+  struct stat buf;
+  if (0 != stat(path.c_str(), &buf)) {
+    ThrowErrno(std::string("Can't get status: ")+path);
+  }
+  info.fileSize = buf.st_size;
+  info.modTime.fromEpochTime(buf.st_mtime);
+  info.mode = buf.st_mode;
+  info.user = buf.st_uid;
+  info.group = buf.st_gid;
+  info.isDir = S_ISDIR(buf.st_mode);
+  if (info.isDir && path[path.length()-1] != '/')
+    path += '/';
+}
+
+bool
+Path::getDirectoryContents(std::set<Path>& result) const {
+  if (!isDirectory())
+    return false;
+  DIR* direntries = ::opendir(path.c_str());
+  if (direntries == 0)
+    ThrowErrno(path + ": can't open directory");
+
+  result.clear();
+  struct dirent* de = ::readdir(direntries);
+  while (de != 0) {
+    if (de->d_name[0] != '.') {
+      Path aPath(path + (const char*)de->d_name);
+      struct stat buf;
+      if (0 != stat(aPath.path.c_str(), &buf))
+        ThrowErrno(aPath.path + ": can't get status");
+      if (S_ISDIR(buf.st_mode))
+        aPath.path += "/";
+      result.insert(aPath);
+    }
+    de = ::readdir(direntries);
+  }
+  
+  closedir(direntries);
+  return true;
+}
+
 bool
-Path::set_directory(const std::string& a_path) {
+Path::setDirectory(const std::string& a_path) {
   if (a_path.size() == 0)
     return false;
   Path save(*this);
@@ -229,7 +301,7 @@ Path::set_directory(const std::string& a_path) {
   size_t last = a_path.size() -1;
   if (last != 0 && a_path[last] != '/')
     path += '/';
-  if (!is_valid()) {
+  if (!isValid()) {
     path = save.path;
     return false;
   }
@@ -237,7 +309,7 @@ Path::set_directory(const std::string& a_path) {
 }
 
 bool
-Path::set_file(const std::string& a_path) {
+Path::setFile(const std::string& a_path) {
   if (a_path.size() == 0)
     return false;
   Path save(*this);
@@ -246,7 +318,7 @@ Path::set_file(const std::string& a_path) {
   while (last > 0 && a_path[last] == '/')
     last--;
   path.erase(last+1);
-  if (!is_valid()) {
+  if (!isValid()) {
     path = save.path;
     return false;
   }
@@ -254,13 +326,13 @@ Path::set_file(const std::string& a_path) {
 }
 
 bool
-Path::append_directory(const std::string& dir) {
-  if (is_file()) 
+Path::appendDirectory(const std::string& dir) {
+  if (isFile()) 
     return false;
   Path save(*this);
   path += dir;
   path += "/";
-  if (!is_valid()) {
+  if (!isValid()) {
     path = save.path;
     return false;
   }
@@ -268,8 +340,8 @@ Path::append_directory(const std::string& dir) {
 }
 
 bool
-Path::elide_directory() {
-  if (is_file()) 
+Path::elideDirectory() {
+  if (isFile()) 
     return false;
   size_t slashpos = path.rfind('/',path.size());
   if (slashpos == 0 || slashpos == std::string::npos)
@@ -283,12 +355,12 @@ Path::elide_directory() {
 }
 
 bool
-Path::append_file(const std::string& file) {
-  if (!is_directory()) 
+Path::appendFile(const std::string& file) {
+  if (!isDirectory()) 
     return false;
   Path save(*this);
   path += file;
-  if (!is_valid()) {
+  if (!isValid()) {
     path = save.path;
     return false;
   }
@@ -296,8 +368,8 @@ Path::append_file(const std::string& file) {
 }
 
 bool
-Path::elide_file() {
-  if (is_directory()) 
+Path::elideFile() {
+  if (isDirectory()) 
     return false;
   size_t slashpos = path.rfind('/',path.size());
   if (slashpos == std::string::npos)
@@ -307,13 +379,13 @@ Path::elide_file() {
 }
 
 bool
-Path::append_suffix(const std::string& suffix) {
-  if (is_directory()) 
+Path::appendSuffix(const std::string& suffix) {
+  if (isDirectory()) 
     return false;
   Path save(*this);
   path.append(".");
   path.append(suffix);
-  if (!is_valid()) {
+  if (!isValid()) {
     path = save.path;
     return false;
   }
@@ -321,8 +393,8 @@ Path::append_suffix(const std::string& suffix) {
 }
 
 bool 
-Path::elide_suffix() {
-  if (is_directory()) return false;
+Path::elideSuffix() {
+  if (isDirectory()) return false;
   size_t dotpos = path.rfind('.',path.size());
   size_t slashpos = path.rfind('/',path.size());
   if (slashpos != std::string::npos && dotpos != std::string::npos &&
@@ -335,9 +407,9 @@ Path::elide_suffix() {
 
 
 bool
-Path::create_directory( bool create_parents) {
+Path::createDirectory( bool create_parents) {
   // Make sure we're dealing with a directory
-  if (!is_directory()) return false;
+  if (!isDirectory()) return false;
 
   // Get a writeable copy of the path name
   char pathname[MAXPATHLEN];
@@ -347,6 +419,8 @@ Path::create_directory( bool create_parents) {
   int lastchar = path.length() - 1 ; 
   if (pathname[lastchar] == '/') 
     pathname[lastchar] = 0;
+  else 
+    pathname[lastchar+1] = 0;
 
   // If we're supposed to create intermediate directories
   if ( create_parents ) {
@@ -362,33 +436,54 @@ Path::create_directory( bool create_parents) {
         if (0 != mkdir(pathname, S_IRWXU | S_IRWXG))
           ThrowErrno(std::string(pathname) + ": Can't create directory");
       char* save = next;
-      next = strchr(pathname,'/');
+      next = strchr(next+1,'/');
       *save = '/';
     }
-  } else if (0 != mkdir(pathname, S_IRWXU | S_IRWXG)) {
-    ThrowErrno(std::string(pathname) + ": Can't create directory");
   } 
+
+  if (0 != access(pathname, F_OK | R_OK))
+    if (0 != mkdir(pathname, S_IRWXU | S_IRWXG))
+      ThrowErrno(std::string(pathname) + ": Can't create directory");
   return true;
 }
 
 bool
-Path::create_file() {
+Path::createFile() {
   // Make sure we're dealing with a file
-  if (!is_file()) return false; 
+  if (!isFile()) return false; 
 
   // Create the file
   int fd = ::creat(path.c_str(), S_IRUSR | S_IWUSR);
   if (fd < 0)
-    ThrowErrno(std::string(path.c_str()) + ": Can't create file");
+    ThrowErrno(path + ": Can't create file");
   ::close(fd);
 
   return true;
 }
 
 bool
-Path::destroy_directory(bool remove_contents) {
+Path::createTemporaryFile() {
+  // Make sure we're dealing with a file
+  if (!isFile()) return false;
+
+  // Append the filename filler
+  char pathname[MAXPATHLEN];
+  path.copy(pathname,MAXPATHLEN);
+  pathname[path.length()] = 0;
+  strcat(pathname,"XXXXXX");
+  int fd = ::mkstemp(pathname);
+  if (fd < 0) {
+    ThrowErrno(path + ": Can't create temporary file");
+  }
+  path = pathname;
+  ::close(fd);
+  return true;
+}
+
+bool
+Path::destroyDirectory(bool remove_contents) {
   // Make sure we're dealing with a directory
-  if (!is_directory()) return false;
+  if (!isDirectory()) return false;
 
   // If it doesn't exist, we're done.
   if (!exists()) return true;
@@ -405,6 +500,8 @@ Path::destroy_directory(bool remove_contents) {
     int lastchar = path.length() - 1 ; 
     if (pathname[lastchar] == '/') 
       pathname[lastchar] = 0;
+    else
+      pathname[lastchar+1] = 0;
     if ( 0 != rmdir(pathname))
       ThrowErrno(std::string(pathname) + ": Can't destroy directory");
   }
@@ -412,10 +509,31 @@ Path::destroy_directory(bool remove_contents) {
 }
 
 bool
-Path::destroy_file() {
-  if (!is_file()) return false;
+Path::destroyFile() {
+  if (!isFile()) return false;
   if (0 != unlink(path.c_str()))
-    ThrowErrno(std::string(path.c_str()) + ": Can't destroy file");
+    ThrowErrno(path + ": Can't destroy file");
+  return true;
+}
+
+bool
+Path::renameFile(const Path& newName) {
+  if (!isFile()) return false;
+  if (0 != rename(path.c_str(), newName.c_str()))
+    ThrowErrno(std::string("can't rename ") + path + " as " + newName.get());
+  return true;
+}
+
+bool
+Path::setStatusInfo(const StatusInfo& si) const {
+  if (!isFile()) return false;
+  struct utimbuf utb;
+  utb.actime = si.modTime.toPosixTime();
+  utb.modtime = utb.actime;
+  if (0 != ::utime(path.c_str(),&utb))
+    ThrowErrno(path + ": can't set file modification time");
+  if (0 != ::chmod(path.c_str(),si.mode))
+    ThrowErrno(path + ": can't set mode");
   return true;
 }