blob: 6cec414ee1aa881657aa57c903dec06003dbdc8a [file] [log] [blame]
Chris Lattner7a6ff2b2003-08-01 21:16:14 +00001//===- Support/FileUtilities.cpp - File System Utilities ------------------===//
John Criswellb576c942003-10-20 19:43:21 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file was developed by the LLVM research group and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
Chris Lattner7a6ff2b2003-08-01 21:16:14 +00009//
10// This file implements a family of utility functions which are useful for doing
11// various things with files.
12//
13//===----------------------------------------------------------------------===//
14
Reid Spencer551ccae2004-09-01 22:55:40 +000015#include "llvm/Support/FileUtilities.h"
16#include "llvm/Support/DataTypes.h"
Reid Spencer9d88d1a2004-12-13 17:01:53 +000017#include "llvm/System/Path.h"
Reid Spencer551ccae2004-09-01 22:55:40 +000018#include "llvm/Config/unistd.h"
19#include "llvm/Config/fcntl.h"
20#include "llvm/Config/sys/types.h"
21#include "llvm/Config/sys/stat.h"
22#include "llvm/Config/sys/mman.h"
23#include "llvm/Config/alloca.h"
Alkis Evlogimenos8868e842004-06-05 08:59:43 +000024#include <cerrno>
25#include <cstdio>
Chris Lattner7a6ff2b2003-08-01 21:16:14 +000026#include <fstream>
27#include <iostream>
Chris Lattner2cdd21c2003-12-14 21:35:53 +000028using namespace llvm;
Brian Gaeked0fde302003-11-11 22:41:34 +000029
Chris Lattner7a6ff2b2003-08-01 21:16:14 +000030/// DiffFiles - Compare the two files specified, returning true if they are
31/// different or if there is a file error. If you specify a string to fill in
32/// for the error option, it will set the string to an error message if an error
33/// occurs, allowing the caller to distinguish between a failed diff and a file
34/// system error.
35///
Chris Lattner2cdd21c2003-12-14 21:35:53 +000036bool llvm::DiffFiles(const std::string &FileA, const std::string &FileB,
37 std::string *Error) {
Chris Lattner7a6ff2b2003-08-01 21:16:14 +000038 std::ifstream FileAStream(FileA.c_str());
39 if (!FileAStream) {
40 if (Error) *Error = "Couldn't open file '" + FileA + "'";
41 return true;
42 }
43
44 std::ifstream FileBStream(FileB.c_str());
45 if (!FileBStream) {
46 if (Error) *Error = "Couldn't open file '" + FileB + "'";
47 return true;
48 }
49
50 // Compare the two files...
51 int C1, C2;
52 do {
53 C1 = FileAStream.get();
54 C2 = FileBStream.get();
55 if (C1 != C2) return true;
56 } while (C1 != EOF);
57
58 return false;
59}
60
61
Chris Lattner5bfac5d2004-06-02 00:52:22 +000062/// CopyFile - Copy the specified source file to the specified destination,
63/// overwriting destination if it exists. This returns true on failure.
64///
65bool llvm::CopyFile(const std::string &Dest, const std::string &Src) {
66 FDHandle InFD(open(Src.c_str(), O_RDONLY));
67 if (InFD == -1) return true;
68
69 FileRemover FR(Dest);
70
71 FDHandle OutFD(open(Dest.c_str(), O_WRONLY|O_CREAT, 0666));
72 if (OutFD == -1) return true;
73
74 char Buffer[16*1024];
75 while (ssize_t Amt = read(InFD, Buffer, 16*1024)) {
76 if (Amt == -1) {
77 if (errno != EINTR) return true; // Error reading the file.
78 } else {
79 char *BufPtr = Buffer;
80 while (Amt) {
81 ssize_t AmtWritten = write(OutFD, BufPtr, Amt);
82 if (AmtWritten == -1) {
83 if (errno != EINTR) return true; // Error writing the file.
84 } else {
85 Amt -= AmtWritten;
86 BufPtr += AmtWritten;
87 }
88 }
89 }
90 }
91
92 FR.releaseFile(); // Success!
93 return false;
94}
95
96
Chris Lattner7a6ff2b2003-08-01 21:16:14 +000097/// MoveFileOverIfUpdated - If the file specified by New is different than Old,
98/// or if Old does not exist, move the New file over the Old file. Otherwise,
99/// remove the New file.
100///
Chris Lattner2cdd21c2003-12-14 21:35:53 +0000101void llvm::MoveFileOverIfUpdated(const std::string &New,
102 const std::string &Old) {
Chris Lattner7a6ff2b2003-08-01 21:16:14 +0000103 if (DiffFiles(New, Old)) {
104 if (std::rename(New.c_str(), Old.c_str()))
105 std::cerr << "Error renaming '" << New << "' to '" << Old << "'!\n";
106 } else {
107 std::remove(New.c_str());
108 }
109}
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000110
111/// removeFile - Delete the specified file
112///
Chris Lattner2cdd21c2003-12-14 21:35:53 +0000113void llvm::removeFile(const std::string &Filename) {
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000114 std::remove(Filename.c_str());
115}
116
117/// getUniqueFilename - Return a filename with the specified prefix. If the
118/// file does not exist yet, return it, otherwise add a suffix to make it
119/// unique.
120///
Chris Lattner2cdd21c2003-12-14 21:35:53 +0000121std::string llvm::getUniqueFilename(const std::string &FilenameBase) {
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000122 if (!std::ifstream(FilenameBase.c_str()))
123 return FilenameBase; // Couldn't open the file? Use it!
124
125 // Create a pattern for mkstemp...
126 char *FNBuffer = new char[FilenameBase.size()+8];
127 strcpy(FNBuffer, FilenameBase.c_str());
128 strcpy(FNBuffer+FilenameBase.size(), "-XXXXXX");
129
130 // Agree on a temporary file name to use....
Chris Lattnerfa5fe7c2004-06-07 19:37:24 +0000131#if defined(HAVE_MKSTEMP) && !defined(_MSC_VER)
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000132 int TempFD;
133 if ((TempFD = mkstemp(FNBuffer)) == -1) {
Chris Lattnerfa5fe7c2004-06-07 19:37:24 +0000134 // FIXME: this should return an emtpy string or something and allow the
135 // caller to deal with the error!
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000136 std::cerr << "bugpoint: ERROR: Cannot create temporary file in the current "
137 << " directory!\n";
138 exit(1);
139 }
140
Misha Brukman950971d2003-09-16 15:31:46 +0000141 // We don't need to hold the temp file descriptor... we will trust that no one
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000142 // will overwrite/delete the file while we are working on it...
143 close(TempFD);
Chris Lattnerfa5fe7c2004-06-07 19:37:24 +0000144#else
145 // If we don't have mkstemp, use the old and obsolete mktemp function.
146 if (mktemp(FNBuffer) == 0) {
147 // FIXME: this should return an emtpy string or something and allow the
148 // caller to deal with the error!
149 std::cerr << "bugpoint: ERROR: Cannot create temporary file in the current "
150 << " directory!\n";
151 exit(1);
152 }
153#endif
154
Misha Brukman3d1b0c72003-08-07 21:28:50 +0000155 std::string Result(FNBuffer);
156 delete[] FNBuffer;
157 return Result;
158}
John Criswell6991a032003-09-02 20:14:57 +0000159
Chris Lattner2d6481c2003-12-29 21:35:05 +0000160//===----------------------------------------------------------------------===//
161// FDHandle class implementation
162//
163
Chris Lattner9b448b72003-12-29 21:43:58 +0000164FDHandle::~FDHandle() throw() {
Chris Lattner2d6481c2003-12-29 21:35:05 +0000165 if (FD != -1) close(FD);
166}
167
Chris Lattner9b448b72003-12-29 21:43:58 +0000168FDHandle &FDHandle::operator=(int fd) throw() {
Chris Lattner2d6481c2003-12-29 21:35:05 +0000169 if (FD != -1) close(FD);
170 FD = fd;
171 return *this;
172}
173