blob: 8d0b5d8a2f1c4a5b451e9e44ac91de2f646c221e [file] [log] [blame]
Rui Ueyama9b55e922017-03-24 00:15:16 +00001//===- Filesystem.cpp -----------------------------------------------------===//
2//
3// The LLVM Linker
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file contains a few utility functions to handle files.
11//
12//===----------------------------------------------------------------------===//
13
14#include "Filesystem.h"
15#include "Config.h"
Bob Haarman4f5c8c22017-10-13 18:22:55 +000016#include "lld/Common/Threads.h"
Rafael Espindolad0929ca2017-11-17 21:40:38 +000017#include "llvm/Config/llvm-config.h"
Rui Ueyama9b55e922017-03-24 00:15:16 +000018#include "llvm/Support/FileOutputBuffer.h"
Rui Ueyama945cd642017-10-05 23:01:11 +000019#include "llvm/Support/FileSystem.h"
Rafael Espindolad0929ca2017-11-17 21:40:38 +000020#if LLVM_ON_UNIX
Rafael Espindola1501a262017-11-07 23:01:37 +000021#include <unistd.h>
22#endif
Rafael Espindolac7889062017-11-07 23:09:54 +000023#include <thread>
Rui Ueyama9b55e922017-03-24 00:15:16 +000024
25using namespace llvm;
26
27using namespace lld;
28using namespace lld::elf;
29
30// Removes a given file asynchronously. This is a performance hack,
31// so remove this when operating systems are improved.
32//
33// On Linux (and probably on other Unix-like systems), unlink(2) is a
34// noticeably slow system call. As of 2016, unlink takes 250
35// milliseconds to remove a 1 GB file on ext4 filesystem on my machine.
36//
37// To create a new result file, we first remove existing file. So, if
38// you repeatedly link a 1 GB program in a regular compile-link-debug
39// cycle, every cycle wastes 250 milliseconds only to remove a file.
40// Since LLD can link a 1 GB binary in about 5 seconds, that waste
41// actually counts.
42//
Rafael Espindola1501a262017-11-07 23:01:37 +000043// This function spawns a background thread to remove the file.
Rui Ueyama9b55e922017-03-24 00:15:16 +000044// The calling thread returns almost immediately.
45void elf::unlinkAsync(StringRef Path) {
Rafael Espindola1501a262017-11-07 23:01:37 +000046// Removing a file is async on windows.
47#if defined(LLVM_ON_WIN32)
48 sys::fs::remove(Path);
49#else
Rafael Espindolafcd74942017-11-07 02:00:51 +000050 if (!ThreadsEnabled || !sys::fs::exists(Path) ||
51 !sys::fs::is_regular_file(Path))
Rui Ueyama9b55e922017-03-24 00:15:16 +000052 return;
53
Rafael Espindola1501a262017-11-07 23:01:37 +000054 // We cannot just remove path from a different thread because we are now going
55 // to create path as a new file.
56 // Instead we open the file and unlink it on this thread. The unlink is fast
57 // since the open fd guarantees that it is not removing the last reference.
58 int FD;
Rui Ueyama1e8813c2017-11-08 04:22:40 +000059 std::error_code EC = sys::fs::openFileForRead(Path, FD);
Rafael Espindola1501a262017-11-07 23:01:37 +000060 sys::fs::remove(Path);
61
62 // close and therefore remove TempPath in background.
Rui Ueyama1e8813c2017-11-08 04:22:40 +000063 if (!EC)
64 std::thread([=] { ::close(FD); }).detach();
Rafael Espindola1501a262017-11-07 23:01:37 +000065#endif
Rui Ueyama9b55e922017-03-24 00:15:16 +000066}
67
Rui Ueyama71630272017-04-26 16:14:46 +000068// Simulate file creation to see if Path is writable.
Rui Ueyama9b55e922017-03-24 00:15:16 +000069//
70// Determining whether a file is writable or not is amazingly hard,
71// and after all the only reliable way of doing that is to actually
72// create a file. But we don't want to do that in this function
73// because LLD shouldn't update any file if it will end in a failure.
Rui Ueyama71630272017-04-26 16:14:46 +000074// We also don't want to reimplement heuristics to determine if a
75// file is writable. So we'll let FileOutputBuffer do the work.
Rui Ueyama9b55e922017-03-24 00:15:16 +000076//
77// FileOutputBuffer doesn't touch a desitnation file until commit()
78// is called. We use that class without calling commit() to predict
79// if the given file is writable.
Rui Ueyama71630272017-04-26 16:14:46 +000080std::error_code elf::tryCreateFile(StringRef Path) {
Rui Ueyamad283eb72017-04-26 16:15:07 +000081 if (Path.empty())
82 return std::error_code();
Rafael Espindola85b8d0c2017-11-08 23:07:32 +000083 if (Path == "-")
84 return std::error_code();
Rafael Espindolaf7a57292017-11-08 01:05:52 +000085 return errorToErrorCode(FileOutputBuffer::create(Path, 1).takeError());
Rui Ueyama9b55e922017-03-24 00:15:16 +000086}