38#include <mach-o/dyld.h>
40#if __has_include(<sys/clonefile.h>)
41#include <sys/clonefile.h>
43#elif defined(__FreeBSD__)
45#if __FreeBSD_version >= 1300057
48#include <machine/elf.h>
51#elif defined(__DragonFly__)
69#define PATH_MAX _XOPEN_PATH_MAX
74#if !defined(__APPLE__) && !defined(__OpenBSD__) && !defined(__FreeBSD__) && \
75 !defined(__linux__) && !defined(__FreeBSD_kernel__) && !defined(_AIX) && \
76 !defined(__managarm__)
77#include <sys/statvfs.h>
78#define STATVFS statvfs
79#define FSTATVFS fstatvfs
80#define STATVFS_F_FRSIZE(vfs) vfs.f_frsize
82#if defined(__OpenBSD__) || defined(__FreeBSD__)
85#elif defined(__linux__) || defined(__managarm__)
86#if defined(HAVE_LINUX_MAGIC_H)
87#include <linux/magic.h>
89#if defined(HAVE_LINUX_NFS_FS_H)
90#include <linux/nfs_fs.h>
92#if defined(HAVE_LINUX_SMB_H)
98#include <sys/statfs.h>
104#include <sys/vmount.h>
106#include <sys/mount.h>
108#define STATVFS statfs
109#define FSTATVFS fstatfs
110#define STATVFS_F_FRSIZE(vfs) static_cast<uint64_t>(vfs.f_bsize)
113#if defined(__NetBSD__) || defined(__DragonFly__) || defined(__GNU__) || \
115#define STATVFS_F_FLAG(vfs) (vfs).f_flag
117#define STATVFS_F_FLAG(vfs) (vfs).f_flags
128#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || \
129 defined(__FreeBSD_kernel__) || defined(__linux__) || \
130 defined(__CYGWIN__) || defined(__DragonFly__) || defined(_AIX) || \
131 defined(__GNU__) || \
132 (defined(__sun__) && defined(__svr4__) || defined(__HAIKU__)) || \
133 defined(__managarm__)
134static int test_dir(
char ret[
PATH_MAX],
const char *dir,
const char *bin) {
138 int chars = snprintf(fullpath,
PATH_MAX,
"%s/%s", dir, bin);
143 if (!realpath(fullpath, ret))
145 if (stat(fullpath, &sb) != 0)
151static char *getprogpath(
char ret[
PATH_MAX],
const char *bin) {
157 if (test_dir(ret,
"/", bin) == 0)
163 if (strchr(bin,
'/')) {
167 if (test_dir(ret, cwd, bin) == 0)
174 if ((pv = getenv(
"PATH")) ==
nullptr)
176 char *s = strdup(pv);
180 for (
char *t = strtok_r(s,
":", &state); t !=
nullptr;
181 t = strtok_r(
nullptr,
":", &state)) {
182 if (test_dir(ret, t, bin) == 0) {
197#if defined(__APPLE__)
202 uint32_t
size =
sizeof(exe_path);
203 if (_NSGetExecutablePath(exe_path, &
size) == 0) {
205 if (realpath(exe_path, link_path))
208#elif defined(__FreeBSD__)
214#if __FreeBSD_version >= 1300057
215 if (elf_aux_info(AT_EXECPATH, exe_path,
sizeof(exe_path)) == 0) {
217 if (realpath(exe_path, link_path))
224 char **
p = ::environ;
228 for (Elf_Auxinfo *aux = (Elf_Auxinfo *)p; aux->a_type != AT_NULL; aux++) {
229 if (aux->a_type == AT_EXECPATH) {
231 if (realpath((
char *)aux->a_un.a_ptr, link_path))
237 if (getprogpath(exe_path, argv0) != NULL)
239#elif defined(_AIX) || defined(__DragonFly__) || defined(__FreeBSD_kernel__) || \
241 const char *curproc =
"/proc/curproc/file";
244 ssize_t len =
::readlink(curproc, exe_path,
sizeof(exe_path));
248 len = std::min(len, ssize_t(
sizeof(exe_path) - 1));
249 exe_path[len] =
'\0';
254 if (getprogpath(exe_path, argv0) != NULL)
256#elif defined(__linux__) || defined(__CYGWIN__) || defined(__gnu_hurd__) || \
257 defined(__managarm__)
259 const char *aPath =
"/proc/self/exe";
262 ssize_t len =
::readlink(aPath, exe_path,
sizeof(exe_path));
268 len = std::min(len, ssize_t(
sizeof(exe_path) - 1));
269 exe_path[len] =
'\0';
275#if _POSIX_VERSION >= 200112 || defined(__GLIBC__)
276 if (
char *
real_path = realpath(exe_path,
nullptr)) {
277 std::string ret = std::string(
real_path);
288 if (getprogpath(exe_path, argv0))
290#elif defined(__OpenBSD__) || defined(__HAIKU__)
293 if (getprogpath(exe_path, argv0) != NULL)
295#elif defined(__sun__) && defined(__svr4__)
297 const char *aPath =
"/proc/self/execname";
299 int fd = open(aPath, O_RDONLY);
302 if (
read(fd, exe_path,
sizeof(exe_path)) < 0)
307 if (getprogpath(exe_path, argv0) != NULL)
309#elif defined(__MVS__)
312 char exe_path[PS_PATHBLEN];
313 pid_t pid = getpid();
315 memset(&buf, 0,
sizeof(buf));
316 buf.ps_pathptr = exe_path;
317 buf.ps_pathlen =
sizeof(exe_path);
320 if ((token = w_getpsent(token, &buf,
sizeof(buf))) <= 0)
322 if (buf.ps_pid != pid)
329#elif defined(HAVE_DLOPEN)
332 int err = dladdr(MainAddr, &DLInfo);
339 if (realpath(DLInfo.dli_fname, link_path))
342#error GetMainExecutable is not implemented on this host yet.
356 return UniqueID(fs_st_dev, fs_st_ino);
361ErrorOr<space_info>
disk_space(
const Twine &Path) {
363 if (::STATVFS(
const_cast<char *
>(
Path.str().c_str()), &Vfs))
365 auto FrSize = STATVFS_F_FRSIZE(Vfs);
367 SpaceInfo.
capacity =
static_cast<uint64_t
>(Vfs.f_blocks) * FrSize;
368 SpaceInfo.free =
static_cast<uint64_t
>(Vfs.f_bfree) * FrSize;
369 SpaceInfo.available =
static_cast<uint64_t
>(Vfs.f_bavail) * FrSize;
373std::error_code
current_path(SmallVectorImpl<char> &result) {
378 const char *pwd = ::getenv(
"PWD");
379 llvm::sys::fs::file_status PWDStatus, DotStatus;
384 result.
append(pwd, pwd + strlen(pwd));
385 return std::error_code();
391 if (::getcwd(result.
data(), result.
size()) ==
nullptr) {
393 if (errno != ENOMEM) {
405 return std::error_code();
411 SmallString<128> path_storage;
414 if (::chdir(
p.begin()) == -1)
417 return std::error_code();
422 SmallString<128> path_storage;
425 if (::mkdir(
p.begin(), Perms) == -1) {
426 if (errno != EEXIST || !IgnoreExisting)
430 return std::error_code();
433std::error_code
create_symlink(
const Twine &to,
const Twine &from) {
435 SmallString<128> from_storage;
436 SmallString<128> to_storage;
440 if (::symlink(t.
begin(),
f.begin()) == -1)
443 return std::error_code();
446std::error_code
create_link(
const Twine &to,
const Twine &from) {
455 SmallString<128> from_storage;
456 SmallString<128> to_storage;
460 if (::link(t.
begin(),
f.begin()) == -1)
463 return std::error_code();
466std::error_code
remove(
const Twine &path,
bool IgnoreNonExisting) {
467 SmallString<128> path_storage;
468 StringRef
p = path.toNullTerminatedStringRef(path_storage);
471 if (lstat(
p.begin(), &buf) != 0) {
472 if (errno != ENOENT || !IgnoreNonExisting)
474 return std::error_code();
482 if (!S_ISREG(buf.st_mode) && !S_ISDIR(buf.st_mode) && !S_ISLNK(buf.st_mode))
486 if (errno != ENOENT || !IgnoreNonExisting)
490 return std::error_code();
493static bool is_local_impl(
struct STATVFS &Vfs) {
494#if defined(__linux__) || defined(__GNU__) || defined(__managarm__)
495#ifndef NFS_SUPER_MAGIC
496#define NFS_SUPER_MAGIC 0x6969
498#ifndef SMB_SUPER_MAGIC
499#define SMB_SUPER_MAGIC 0x517B
501#ifndef CIFS_MAGIC_NUMBER
502#define CIFS_MAGIC_NUMBER 0xFF534D42
504#if defined(__GNU__) && ((__GLIBC__ < 2) || ((__GLIBC__ == 2) && (__GLIBC_MINOR__ < 39)))
505 switch ((uint32_t)Vfs.__f_type) {
507 switch ((uint32_t)Vfs.f_type) {
509 case NFS_SUPER_MAGIC:
510 case SMB_SUPER_MAGIC:
511 case CIFS_MAGIC_NUMBER:
516#elif defined(__CYGWIN__)
519#elif defined(__Fuchsia__)
522#elif defined(__EMSCRIPTEN__)
525#elif defined(__HAIKU__)
531 StringRef fstype(Vfs.f_basetype);
533 return fstype !=
"nfs";
538 size_t BufSize = 2048
u;
539 std::unique_ptr<char[]> Buf;
542 Buf = std::make_unique<char[]>(BufSize);
543 Ret = mntctl(MCTL_QUERY, BufSize, Buf.get());
546 BufSize = *
reinterpret_cast<unsigned int *
>(Buf.get());
555 char *CurObjPtr = Buf.get();
557 struct vmount *Vp =
reinterpret_cast<struct vmount *
>(CurObjPtr);
558 static_assert(
sizeof(Vfs.f_fsid) ==
sizeof(Vp->vmt_fsid),
559 "fsid length mismatch");
560 if (
memcmp(&Vfs.f_fsid, &Vp->vmt_fsid,
sizeof Vfs.f_fsid) == 0)
561 return (Vp->vmt_flags & MNT_REMOTE) == 0;
563 CurObjPtr += Vp->vmt_length;
568#elif defined(__MVS__)
573 return !!(STATVFS_F_FLAG(Vfs) & MNT_LOCAL);
577std::error_code
is_local(
const Twine &Path,
bool &Result) {
581 if (::STATVFS(
const_cast<char *
>(
Path.str().c_str()), &Vfs))
584 Result = is_local_impl(Vfs);
585 return std::error_code();
588std::error_code
is_local(
int FD,
bool &Result) {
592 if (::FSTATVFS(FD, &Vfs))
595 Result = is_local_impl(Vfs);
596 return std::error_code();
599std::error_code
rename(
const Twine &from,
const Twine &to) {
601 SmallString<128> from_storage;
602 SmallString<128> to_storage;
609 return std::error_code();
615 if (::ftruncate(FD,
Size) == -1)
618 return std::error_code();
641 SmallString<128> PathStorage;
642 StringRef
P =
Path.toNullTerminatedStringRef(PathStorage);
650 if (0 != stat(
P.begin(), &buf))
652 if (!S_ISREG(buf.st_mode))
656 return std::error_code();
667 return A.fs_st_dev ==
B.fs_st_dev &&
A.fs_st_ino ==
B.fs_st_ino;
670std::error_code
equivalent(
const Twine &
A,
const Twine &
B,
bool &result) {
674 if (std::error_code ec =
status(
A, fsA))
676 if (std::error_code ec =
status(
B, fsB))
679 return std::error_code();
682static void expandTildeExpr(SmallVectorImpl<char> &Path) {
683 StringRef PathStr(
Path.begin(),
Path.size());
684 if (PathStr.empty() || !PathStr.starts_with(
"~"))
687 PathStr = PathStr.drop_front();
690 StringRef Remainder = PathStr.substr(Expr.
size() + 1);
691 SmallString<128> Storage;
700 Path[0] = Storage[0];
707 std::unique_ptr<char[]> Buf;
708 long BufSize = sysconf(_SC_GETPW_R_SIZE_MAX);
711 Buf = std::make_unique<char[]>(BufSize);
714 struct passwd *
Entry =
nullptr;
715 getpwnam_r(
User.c_str(), &Pwd, Buf.get(), BufSize, &Entry);
717 if (!Entry || !
Entry->pw_dir) {
728void expand_tilde(
const Twine &path, SmallVectorImpl<char> &dest) {
734 expandTildeExpr(dest);
740 else if (S_ISREG(
Mode))
742 else if (S_ISBLK(
Mode))
744 else if (S_ISCHR(
Mode))
746 else if (S_ISFIFO(
Mode))
748 else if (S_ISSOCK(
Mode))
750 else if (S_ISLNK(
Mode))
755static std::error_code fillStatus(
int StatRet,
const struct stat &Status,
766 uint32_t atime_nsec, mtime_nsec;
767#if defined(HAVE_STRUCT_STAT_ST_MTIMESPEC_TV_NSEC)
768 atime_nsec = Status.st_atimespec.tv_nsec;
769 mtime_nsec = Status.st_mtimespec.tv_nsec;
770#elif defined(HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC)
771 atime_nsec = Status.st_atim.tv_nsec;
772 mtime_nsec = Status.st_mtim.tv_nsec;
774 atime_nsec = mtime_nsec = 0;
779 Status.st_nlink, Status.st_ino, Status.st_atime,
780 atime_nsec, Status.st_mtime, mtime_nsec, Status.st_uid,
781 Status.st_gid, Status.st_size);
783 return std::error_code();
789 SmallString<128> PathStorage;
790 StringRef
P =
Path.toNullTerminatedStringRef(PathStorage);
793 int StatRet = (Follow ? ::stat : ::lstat)(
P.begin(), &Status);
794 return fillStatus(StatRet, Status, Result);
801 int StatRet = ::fstat(FD, &Status);
802 return fillStatus(StatRet, Status, Result);
808 unsigned Mask = ::umask(0);
814 SmallString<128> PathStorage;
815 StringRef
P =
Path.toNullTerminatedStringRef(PathStorage);
817 if (::chmod(
P.begin(), Permissions))
819 return std::error_code();
823 if (::fchmod(FD, Permissions))
825 return std::error_code();
830#if defined(HAVE_FUTIMENS)
834 if (::futimens(FD, Times))
836 return std::error_code();
837#elif defined(HAVE_FUTIMES)
840 std::chrono::time_point_cast<std::chrono::microseconds>(AccessTime));
842 sys::toTimeVal(std::chrono::time_point_cast<std::chrono::microseconds>(
844 if (::futimes(FD, Times))
846 return std::error_code();
847#elif defined(__MVS__)
849 memset(&Attr, 0,
sizeof(Attr));
850 Attr.att_atimechg = 1;
852 Attr.att_mtimechg = 1;
854 if (::__fchattr(FD, &Attr,
sizeof(Attr)) != 0)
856 return std::error_code();
858#warning Missing futimes() and futimens()
863std::error_code mapped_file_region::init(
int FD, uint64_t
Offset,
867 int flags = (Mode ==
readwrite) ? MAP_SHARED : MAP_PRIVATE;
868 int prot = (Mode ==
readonly) ? PROT_READ : (PROT_READ | PROT_WRITE);
869#if defined(MAP_NORESERVE)
870 flags |= MAP_NORESERVE;
872#if defined(__APPLE__)
883#if defined(MAP_RESILIENT_CODESIGN)
884 flags |= MAP_RESILIENT_CODESIGN;
886#if defined(MAP_RESILIENT_MEDIA)
887 flags |= MAP_RESILIENT_MEDIA;
892 Mapping = ::mmap(
nullptr, Size, prot, flags, FD,
Offset);
893 if (Mapping == MAP_FAILED)
895 return std::error_code();
899 uint64_t offset, std::error_code &ec)
906 copyFrom(mapped_file_region());
909void mapped_file_region::unmapImpl() {
911 ::munmap(Mapping,
Size);
914std::error_code mapped_file_region::sync()
const {
915 if (
int Res = ::msync(Mapping,
Size, MS_SYNC))
916 return std::error_code(Res, std::generic_category());
917 return std::error_code();
920void mapped_file_region::dontNeedImpl() {
924#if defined(__MVS__) || defined(_AIX)
926#elif defined(POSIX_MADV_DONTNEED)
927 ::posix_madvise(Mapping,
Size, POSIX_MADV_DONTNEED);
929 ::madvise(Mapping,
Size, MADV_DONTNEED);
933void mapped_file_region::willNeedImpl() {
937#if defined(__MVS__) || defined(_AIX)
939#elif defined(POSIX_MADV_WILLNEED)
940 ::posix_madvise(Mapping,
Size, POSIX_MADV_WILLNEED);
942 ::madvise(Mapping,
Size, MADV_WILLNEED);
948std::error_code detail::directory_iterator_construct(detail::DirIterState &it,
950 bool follow_symlinks) {
954 DIR *directory = ::opendir(path_null.c_str());
958 it.IterationHandle =
reinterpret_cast<intptr_t
>(directory);
961 it.CurrentEntry = directory_entry(path_null.str(), follow_symlinks);
965std::error_code detail::directory_iterator_destruct(detail::DirIterState &it) {
966 if (it.IterationHandle)
967 ::closedir(
reinterpret_cast<DIR *
>(it.IterationHandle));
968 it.IterationHandle = 0;
969 it.CurrentEntry = directory_entry();
970 return std::error_code();
973static file_type direntType(dirent *Entry) {
980 return typeForMode(DTTOIF(Entry->d_type));
983 return file_type::type_unknown;
987std::error_code detail::directory_iterator_increment(detail::DirIterState &It) {
991 dirent *CurDir = ::readdir(
reinterpret_cast<DIR *
>(It.IterationHandle));
992 if (CurDir ==
nullptr && errno != 0) {
994 }
else if (CurDir !=
nullptr) {
996 if ((
Name.size() == 1 && Name[0] ==
'.') ||
997 (
Name.size() == 2 && Name[0] ==
'.' && Name[1] ==
'.'))
999 It.CurrentEntry.replace_filename(Name, direntType(CurDir));
1004 return std::error_code();
1011 if (
auto EC =
fs::status(Path, s, FollowSymlinks))
1020#if !defined(__FreeBSD__) && !defined(__OpenBSD__)
1021#define TRY_PROC_SELF_FD
1024#if !defined(F_GETPATH) && defined(TRY_PROC_SELF_FD)
1025static bool hasProcSelfFD() {
1028 static const bool Result = (::access(
"/proc/self/fd", R_OK) == 0);
1033static int nativeOpenFlags(CreationDisposition Disp, OpenFlags Flags,
1038 else if (
Access == FA_Write)
1040 else if (
Access == (FA_Read | FA_Write))
1045 if (Flags & OF_Append)
1048 if (Disp == CD_CreateNew) {
1051 }
else if (Disp == CD_CreateAlways) {
1054 }
else if (Disp == CD_OpenAlways) {
1056 }
else if (Disp == CD_OpenExisting) {
1064 if (Flags & OF_Append)
1069 if (!(Flags & OF_ChildInherit))
1077 CreationDisposition Disp, FileAccess
Access,
1078 OpenFlags Flags,
unsigned Mode) {
1087 auto Open = [&]() { return ::open(
P.begin(), OpenFlags,
Mode); };
1091 if (!(Flags & OF_ChildInherit)) {
1092 int r = fcntl(ResultFD, F_SETFD, FD_CLOEXEC);
1094 assert(r == 0 &&
"fcntl(F_SETFD, FD_CLOEXEC) failed");
1152 if ((Flags & OF_Append) && lseek(ResultFD, 0, SEEK_END) == -1)
1155 if (fstat(ResultFD, &Stat) == -1)
1157 if (S_ISREG(Stat.st_mode)) {
1158 bool DoSetTag = (
Access &
FA_Write) && (Disp != CD_OpenExisting) &&
1159 !Stat.st_tag.ft_txtflag && !Stat.st_tag.ft_ccsid &&
1161 if (Flags & OF_Text) {
1162 if (
auto EC = llvm::enableAutoConversion(ResultFD))
1165 if (
auto EC = llvm::setzOSFileTag(ResultFD,
CCSID_IBM_1047,
true))
1169 if (
auto EC = llvm::disableAutoConversion(ResultFD))
1172 if (
auto EC = llvm::setzOSFileTag(ResultFD, FT_BINARY,
false))
1179 return std::error_code();
1183 FileAccess
Access, OpenFlags Flags,
1199 std::error_code
EC =
1200 openFile(Name, ResultFD, CD_OpenExisting, FA_Read, Flags, 0666);
1206 return std::error_code();
1208#if defined(F_GETPATH)
1212 if (::fcntl(ResultFD, F_GETPATH, Buffer) != -1)
1213 RealPath->
append(Buffer, Buffer + strlen(Buffer));
1216#if defined(TRY_PROC_SELF_FD)
1217 if (hasProcSelfFD()) {
1219 snprintf(ProcPath,
sizeof(ProcPath),
"/proc/self/fd/%d", ResultFD);
1220 ssize_t CharCount = ::readlink(ProcPath, Buffer,
sizeof(Buffer));
1222 RealPath->
append(Buffer, Buffer + CharCount);
1229 if (::realpath(
P.begin(), Buffer) !=
nullptr)
1230 RealPath->
append(Buffer, Buffer + strlen(Buffer));
1231#if defined(TRY_PROC_SELF_FD)
1235 return std::error_code();
1256#if defined(__APPLE__)
1257 size_t Size = std::min<size_t>(Buf.
size(), INT32_MAX);
1266#if defined(__MVS__) || defined(_AIX)
1268 if (fstat(FD, &
Status) == -1)
1270 if (S_ISDIR(
Status.st_mode))
1280#if defined(__APPLE__)
1281 size_t Size = std::min<size_t>(Buf.
size(), INT32_MAX);
1289 if (lseek(FD,
Offset, SEEK_SET) == -1)
1300 auto Start = std::chrono::steady_clock::now();
1304 memset(&Lock, 0,
sizeof(Lock));
1306 case LockKind::Exclusive:
1307 Lock.l_type = F_WRLCK;
1309 case LockKind::Shared:
1310 Lock.l_type = F_RDLCK;
1313 Lock.l_whence = SEEK_SET;
1316 if (::fcntl(FD, F_SETLK, &Lock) != -1)
1317 return std::error_code();
1319 if (Error != EACCES && Error != EAGAIN)
1320 return std::error_code(Error, std::generic_category());
1324 }
while (std::chrono::steady_clock::now() < End);
1328std::error_code
lockFile(
int FD, LockKind Kind) {
1330 memset(&Lock, 0,
sizeof(Lock));
1332 case LockKind::Exclusive:
1333 Lock.l_type = F_WRLCK;
1335 case LockKind::Shared:
1336 Lock.l_type = F_RDLCK;
1339 Lock.l_whence = SEEK_SET;
1342 if (::fcntl(FD, F_SETLKW, &Lock) != -1)
1343 return std::error_code();
1349 Lock.l_type = F_UNLCK;
1350 Lock.l_whence = SEEK_SET;
1353 if (::fcntl(FD, F_SETLK, &Lock) != -1)
1354 return std::error_code();
1366template <
typename T>
1367static std::error_code remove_directories_impl(
const T &Entry,
1368 bool IgnoreErrors) {
1370 directory_iterator Begin(Entry, EC,
false);
1371 directory_iterator End;
1372 while (Begin != End) {
1373 auto &Item = *Begin;
1377 EC = remove_directories_impl(Item, IgnoreErrors);
1378 if (EC && !IgnoreErrors)
1383 if (EC && !IgnoreErrors)
1385 }
else if (!IgnoreErrors) {
1389 Begin.increment(EC);
1390 if (EC && !IgnoreErrors)
1393 return std::error_code();
1397 auto EC = remove_directories_impl(
path, IgnoreErrors);
1398 if (EC && !IgnoreErrors)
1401 if (EC && !IgnoreErrors)
1403 return std::error_code();
1407 bool expand_tilde) {
1411 if (
path.isTriviallyEmpty())
1412 return std::error_code();
1416 path.toVector(Storage);
1417 expandTildeExpr(Storage);
1424 if (::realpath(
P.begin(), Buffer) ==
nullptr)
1426 dest.
append(Buffer, Buffer + strlen(Buffer));
1427 return std::error_code();
1445 size_t BufSize = std::max(std::size_t{32}, dest.
capacity());
1448 ssize_t
Len = ::readlink(
P.begin(), dest.
data(), dest.
size());
1451 if (
static_cast<size_t>(Len) < BufSize) {
1453 return std::error_code();
1461 auto FChown = [&]() { return ::fchown(FD,
Owner, Group); };
1465 return std::error_code();
1473 std::unique_ptr<char[]> Buf;
1474 char *RequestedDir = getenv(
"HOME");
1475 if (!RequestedDir) {
1476 long BufSize = sysconf(_SC_GETPW_R_SIZE_MAX);
1479 Buf = std::make_unique<char[]>(BufSize);
1481 struct passwd *pw =
nullptr;
1482 getpwuid_r(getuid(), &Pwd, Buf.get(), BufSize, &pw);
1483 if (pw && pw->pw_dir)
1484 RequestedDir = pw->pw_dir;
1490 result.
append(RequestedDir, RequestedDir + strlen(RequestedDir));
1494static bool getDarwinConfDir(
bool TempDir, SmallVectorImpl<char> &Result) {
1495#if defined(_CS_DARWIN_USER_TEMP_DIR) && defined(_CS_DARWIN_USER_CACHE_DIR)
1498 int ConfName = TempDir ? _CS_DARWIN_USER_TEMP_DIR : _CS_DARWIN_USER_CACHE_DIR;
1499 size_t ConfLen = confstr(ConfName,
nullptr, 0);
1503 ConfLen = confstr(ConfName,
Result.data(),
Result.size());
1504 }
while (ConfLen > 0 && ConfLen !=
Result.size());
1522 append(result,
"Library",
"Preferences");
1528 if (
const char *RequestedDir = getenv(
"XDG_CONFIG_HOME")) {
1530 result.
append(RequestedDir, RequestedDir + strlen(RequestedDir));
1538 append(result,
".config");
1544 if (getDarwinConfDir(
false , result)) {
1550 if (
const char *RequestedDir = getenv(
"XDG_CACHE_HOME")) {
1552 result.
append(RequestedDir, RequestedDir + strlen(RequestedDir));
1559 append(result,
".cache");
1563static const char *getEnvTempDir() {
1566 const char *EnvironmentVariables[] = {
"TMPDIR",
"TMP",
"TEMP",
"TEMPDIR"};
1567 for (
const char *Env : EnvironmentVariables) {
1568 if (
const char *Dir = std::getenv(Env))
1575static const char *getDefaultTempDir(
bool ErasedOnReboot) {
1589 if (ErasedOnReboot) {
1591 if (
const char *RequestedDir = getEnvTempDir()) {
1592 Result.append(RequestedDir, RequestedDir + strlen(RequestedDir));
1597 if (getDarwinConfDir(ErasedOnReboot, Result))
1600 const char *RequestedDir = getDefaultTempDir(ErasedOnReboot);
1601 Result.append(RequestedDir, RequestedDir + strlen(RequestedDir));
1614std::error_code
copy_file(
const Twine &From,
const Twine &To) {
1615 std::string FromS = From.
str();
1616 std::string ToS = To.
str();
1617#if __has_builtin(__builtin_available)
1618 if (__builtin_available(macos 10.12, *)) {
1625 if (!clonefile(FromS.c_str(), ToS.c_str(), 0))
1626 return std::error_code();
1636 return std::error_code(Errno, std::generic_category());
1645 if (!copyfile(FromS.c_str(), ToS.c_str(), NULL, COPYFILE_DATA))
1646 return std::error_code();
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
static ManagedStatic< DebugCounterOwner > Owner
amode Optimize addressing mode
std::unique_ptr< MemoryBuffer > openFile(const Twine &Path)
Merge contiguous icmps into a memcmp
static cl::opt< RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode > Mode("regalloc-enable-advisor", cl::Hidden, cl::init(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default), cl::desc("Enable regalloc advisor mode"), cl::values(clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default, "default", "Default"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Release, "release", "precompiled"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Development, "development", "for training")))
LLVM_ABI const file_t kInvalidFile
Represents the result of a call to sys::fs::status().
size_t size() const
size - Get the array size.
Represents either an error or a value T.
std::error_code getError() const
MutableArrayRef - Represent a mutable reference to an array (0 or more elements consecutively in memo...
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void resize_for_overwrite(size_type N)
Like resize, but T is POD, the new values won't be initialized.
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void truncate(size_type N)
Like resize, but requires that N is less than size().
pointer data()
Return a pointer to the vector's buffer, even if empty().
StringRef - Represent a constant reference to a string, i.e.
std::string str() const
str - Get the contents as an std::string.
constexpr bool empty() const
empty - Check if the string is empty.
constexpr size_t size() const
size - Get the string size.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
LLVM_ABI std::string str() const
Return the twine contents as a std::string.
LLVM_ABI StringRef toNullTerminatedStringRef(SmallVectorImpl< char > &Out) const
This returns the twine as a single null terminated StringRef if it can be represented as such.
bool isTriviallyEmpty() const
Check if this twine is trivially empty; a false return value does not necessarily mean the twine is e...
LLVM_ABI void toVector(SmallVectorImpl< char > &Out) const
Append the concatenated string into the given SmallString or SmallVector.
static LLVM_ABI std::error_code SafelyCloseFileDescriptor(int FD)
static unsigned getPageSizeEstimate()
Get the process's estimated page size.
uint32_t fs_st_mtime_nsec
uint32_t fs_st_atime_nsec
LLVM_ABI TimePoint getLastModificationTime() const
The file modification time as reported from the underlying file system.
LLVM_ABI TimePoint getLastAccessedTime() const
The file access time as reported from the underlying file system.
Represents the result of a call to sys::fs::status().
LLVM_ABI uint32_t getLinkCount() const
LLVM_ABI UniqueID getUniqueID() const
mapped_file_region()=default
@ readonly
May only access map via const_data as read only.
@ readwrite
May access map via data and modify it. Written to path.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
constexpr std::underlying_type_t< E > Mask()
Get a bitmask with 1s in all places up to the high-order bit of E's largest value.
initializer< Ty > init(const Ty &Val)
@ User
could "use" a pointer
value_type read(const void *memory, endianness endian)
Read a value of a particular endianness from memory.
LLVM_ABI std::error_code directory_iterator_destruct(DirIterState &)
LLVM_ABI std::error_code directory_iterator_increment(DirIterState &)
LLVM_ABI std::error_code readlink(const Twine &path, SmallVectorImpl< char > &output)
Read the target of a symbolic link.
LLVM_ABI bool can_execute(const Twine &Path)
Can we execute this file?
LLVM_ABI std::error_code closeFile(file_t &F)
Close the file object.
LLVM_ABI std::error_code rename(const Twine &from, const Twine &to)
Rename from to to.
LLVM_ABI std::error_code create_hard_link(const Twine &to, const Twine &from)
Create a hard link from from to to, or return an error.
LLVM_ABI const file_t kInvalidFile
LLVM_ABI std::error_code access(const Twine &Path, AccessMode Mode)
Can the file be accessed?
LLVM_ABI ErrorOr< space_info > disk_space(const Twine &Path)
Get disk space usage information.
LLVM_ABI Expected< size_t > readNativeFile(file_t FileHandle, MutableArrayRef< char > Buf)
Reads Buf.size() bytes from FileHandle into Buf.
LLVM_ABI unsigned getUmask()
Get file creation mode mask of the process.
LLVM_ABI bool exists(const basic_file_status &status)
Does file exist?
LLVM_ABI Expected< file_t > openNativeFile(const Twine &Name, CreationDisposition Disp, FileAccess Access, OpenFlags Flags, unsigned Mode=0666)
Opens a file with the specified creation disposition, access mode, and flags and returns a platform-s...
LLVM_ABI file_t getStdoutHandle()
Return an open handle to standard out.
file_type
An enumeration for the file system's view of the type.
LLVM_ABI std::error_code create_link(const Twine &to, const Twine &from)
Create a link from from to to.
LLVM_ABI std::error_code create_symlink(const Twine &to, const Twine &from)
Create a symbolic link from from to to.
LLVM_ABI void expand_tilde(const Twine &path, SmallVectorImpl< char > &output)
Expands ~ expressions to the user's home directory.
LLVM_ABI std::error_code lockFile(int FD, LockKind Kind=LockKind::Exclusive)
Lock the file.
LLVM_ABI std::error_code remove(const Twine &path, bool IgnoreNonExisting=true)
Remove path.
LLVM_ABI std::error_code set_current_path(const Twine &path)
Set the current path.
LLVM_ABI std::error_code real_path(const Twine &path, SmallVectorImpl< char > &output, bool expand_tilde=false)
Collapse all .
@ CD_OpenAlways
CD_OpenAlways - When opening a file:
LLVM_ABI Expected< size_t > readNativeFileSlice(file_t FileHandle, MutableArrayRef< char > Buf, uint64_t Offset)
Reads Buf.size() bytes from FileHandle at offset Offset into Buf.
LLVM_ABI std::error_code changeFileOwnership(int FD, uint32_t Owner, uint32_t Group)
Change ownership of a file.
LLVM_ABI std::string getMainExecutable(const char *argv0, void *MainExecAddr)
Return the path to the main executable, given the value of argv[0] from program startup and the addre...
LLVM_ABI Expected< file_t > openNativeFileForRead(const Twine &Name, OpenFlags Flags=OF_None, SmallVectorImpl< char > *RealPath=nullptr)
Opens the file with the given name in a read-only mode, returning its open file descriptor.
LLVM_ABI bool status_known(const basic_file_status &s)
Is status available?
LLVM_ABI std::error_code resize_file_sparse(int FD, uint64_t Size)
Resize path to size with sparse files explicitly enabled.
LLVM_ABI std::error_code copy_file(const Twine &From, const Twine &To)
Copy the contents of From to To.
LLVM_ABI std::error_code tryLockFile(int FD, std::chrono::milliseconds Timeout=std::chrono::milliseconds(0), LockKind Kind=LockKind::Exclusive)
Try to locks the file during the specified time.
LLVM_ABI std::error_code current_path(SmallVectorImpl< char > &result)
Get the current path.
LLVM_ABI std::error_code resize_file(int FD, uint64_t Size)
Resize path to size.
LLVM_ABI std::error_code status(const Twine &path, file_status &result, bool follow=true)
Get file status as if by POSIX stat().
LLVM_ABI std::error_code create_directory(const Twine &path, bool IgnoreExisting=true, perms Perms=owner_all|group_all)
Create the directory in path.
LLVM_ABI std::error_code is_local(const Twine &path, bool &result)
Is the file mounted on a local filesystem?
LLVM_ABI std::error_code openFileForRead(const Twine &Name, int &ResultFD, OpenFlags Flags=OF_None, SmallVectorImpl< char > *RealPath=nullptr)
Opens the file with the given name in a read-only mode, returning its open file descriptor.
LLVM_ABI std::error_code remove_directories(const Twine &path, bool IgnoreErrors=true)
Recursively delete a directory.
LLVM_ABI bool equivalent(file_status A, file_status B)
Do file_status's represent the same thing?
LLVM_ABI file_t getStderrHandle()
Return an open handle to standard error.
LLVM_ABI std::error_code setLastAccessAndModificationTime(int FD, TimePoint<> AccessTime, TimePoint<> ModificationTime)
Set the file modification and access time.
LLVM_ABI file_t getStdinHandle()
Return an open handle to standard in.
LLVM_ABI std::error_code unlockFile(int FD)
Unlock the file.
LLVM_ABI std::error_code setPermissions(const Twine &Path, perms Permissions)
Set file permissions.
LLVM_ABI bool is_directory(const basic_file_status &status)
Does status represent a directory?
LLVM_ABI bool cache_directory(SmallVectorImpl< char > &result)
Get the directory where installed packages should put their machine-local cache, e....
LLVM_ABI bool user_config_directory(SmallVectorImpl< char > &result)
Get the directory where packages should read user-specific configurations.
LLVM_ABI void system_temp_directory(bool erasedOnReboot, SmallVectorImpl< char > &result)
Get the typical temporary directory for the system, e.g., "/var/tmp" or "C:/TEMP".
LLVM_ABI bool is_absolute(const Twine &path, Style style=Style::native)
Is path absolute?
LLVM_ABI void append(SmallVectorImpl< char > &path, const Twine &a, const Twine &b="", const Twine &c="", const Twine &d="")
Append to path.
LLVM_ABI bool home_directory(SmallVectorImpl< char > &result)
Get the user's home directory.
LLVM_ABI bool is_separator(char value, Style style=Style::native)
Check whether the given char is a path separator on the host OS.
void violationIfEnabled()
ScopedSetting scopedDisable()
decltype(auto) RetryAfterSignal(const FailT &Fail, const Fun &F, const Args &... As)
std::chrono::time_point< std::chrono::system_clock, D > TimePoint
A time point on the system clock.
TimePoint< std::chrono::seconds > toTimePoint(std::time_t T)
Convert a std::time_t to a TimePoint.
struct timespec toTimeSpec(TimePoint<> TP)
Convert a time point to struct timespec.
struct timeval toTimeVal(TimePoint< std::chrono::microseconds > TP)
Convert a time point to struct timeval.
std::time_t toTimeT(TimePoint<> TP)
Convert a TimePoint to std::time_t.
This is an optimization pass for GlobalISel generic memory operations.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
std::error_code make_error_code(BitcodeError E)
@ no_such_file_or_directory
@ operation_not_permitted
@ Timeout
Reached timeout while waiting for the owner to release the lock.
LLVM_ABI Error errorCodeToError(std::error_code EC)
Helper for converting an std::error_code to a Error.
std::error_code errnoAsErrorCode()
Helper to get errno as an std::error_code.
space_info - Self explanatory.