• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===-- PathV2.cpp - Implement OS Path Concept ------------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
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 implements the operating system PathV2 API.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "llvm/Support/PathV2.h"
15 #include "llvm/Support/FileSystem.h"
16 #include "llvm/Support/Endian.h"
17 #include "llvm/Support/ErrorHandling.h"
18 #include <cctype>
19 #include <cstdio>
20 #include <cstring>
21 
22 namespace {
23   using llvm::StringRef;
24   using llvm::sys::path::is_separator;
25 
26 #ifdef LLVM_ON_WIN32
27   const char *separators = "\\/";
28   const char  prefered_separator = '\\';
29 #else
30   const char  separators = '/';
31   const char  prefered_separator = '/';
32 #endif
33 
find_first_component(StringRef path)34   StringRef find_first_component(StringRef path) {
35     // Look for this first component in the following order.
36     // * empty (in this case we return an empty string)
37     // * either C: or {//,\\}net.
38     // * {/,\}
39     // * {.,..}
40     // * {file,directory}name
41 
42     if (path.empty())
43       return path;
44 
45 #ifdef LLVM_ON_WIN32
46     // C:
47     if (path.size() >= 2 && std::isalpha(path[0]) && path[1] == ':')
48       return path.substr(0, 2);
49 #endif
50 
51     // //net
52     if ((path.size() > 2) &&
53         is_separator(path[0]) &&
54         path[0] == path[1] &&
55         !is_separator(path[2])) {
56       // Find the next directory separator.
57       size_t end = path.find_first_of(separators, 2);
58       return path.substr(0, end);
59     }
60 
61     // {/,\}
62     if (is_separator(path[0]))
63       return path.substr(0, 1);
64 
65     if (path.startswith(".."))
66       return path.substr(0, 2);
67 
68     if (path[0] == '.')
69       return path.substr(0, 1);
70 
71     // * {file,directory}name
72     size_t end = path.find_first_of(separators, 2);
73     return path.substr(0, end);
74   }
75 
filename_pos(StringRef str)76   size_t filename_pos(StringRef str) {
77     if (str.size() == 2 &&
78         is_separator(str[0]) &&
79         str[0] == str[1])
80       return 0;
81 
82     if (str.size() > 0 && is_separator(str[str.size() - 1]))
83       return str.size() - 1;
84 
85     size_t pos = str.find_last_of(separators, str.size() - 1);
86 
87 #ifdef LLVM_ON_WIN32
88     if (pos == StringRef::npos)
89       pos = str.find_last_of(':', str.size() - 2);
90 #endif
91 
92     if (pos == StringRef::npos ||
93         (pos == 1 && is_separator(str[0])))
94       return 0;
95 
96     return pos + 1;
97   }
98 
root_dir_start(StringRef str)99   size_t root_dir_start(StringRef str) {
100     // case "c:/"
101 #ifdef LLVM_ON_WIN32
102     if (str.size() > 2 &&
103         str[1] == ':' &&
104         is_separator(str[2]))
105       return 2;
106 #endif
107 
108     // case "//"
109     if (str.size() == 2 &&
110         is_separator(str[0]) &&
111         str[0] == str[1])
112       return StringRef::npos;
113 
114     // case "//net"
115     if (str.size() > 3 &&
116         is_separator(str[0]) &&
117         str[0] == str[1] &&
118         !is_separator(str[2])) {
119       return str.find_first_of(separators, 2);
120     }
121 
122     // case "/"
123     if (str.size() > 0 && is_separator(str[0]))
124       return 0;
125 
126     return StringRef::npos;
127   }
128 
parent_path_end(StringRef path)129   size_t parent_path_end(StringRef path) {
130     size_t end_pos = filename_pos(path);
131 
132     bool filename_was_sep = path.size() > 0 && is_separator(path[end_pos]);
133 
134     // Skip separators except for root dir.
135     size_t root_dir_pos = root_dir_start(path.substr(0, end_pos));
136 
137     while(end_pos > 0 &&
138           (end_pos - 1) != root_dir_pos &&
139           is_separator(path[end_pos - 1]))
140       --end_pos;
141 
142     if (end_pos == 1 && root_dir_pos == 0 && filename_was_sep)
143       return StringRef::npos;
144 
145     return end_pos;
146   }
147 } // end unnamed namespace
148 
149 namespace llvm {
150 namespace sys  {
151 namespace path {
152 
begin(StringRef path)153 const_iterator begin(StringRef path) {
154   const_iterator i;
155   i.Path      = path;
156   i.Component = find_first_component(path);
157   i.Position  = 0;
158   return i;
159 }
160 
end(StringRef path)161 const_iterator end(StringRef path) {
162   const_iterator i;
163   i.Path      = path;
164   i.Position  = path.size();
165   return i;
166 }
167 
operator ++()168 const_iterator &const_iterator::operator++() {
169   assert(Position < Path.size() && "Tried to increment past end!");
170 
171   // Increment Position to past the current component
172   Position += Component.size();
173 
174   // Check for end.
175   if (Position == Path.size()) {
176     Component = StringRef();
177     return *this;
178   }
179 
180   // Both POSIX and Windows treat paths that begin with exactly two separators
181   // specially.
182   bool was_net = Component.size() > 2 &&
183     is_separator(Component[0]) &&
184     Component[1] == Component[0] &&
185     !is_separator(Component[2]);
186 
187   // Handle separators.
188   if (is_separator(Path[Position])) {
189     // Root dir.
190     if (was_net
191 #ifdef LLVM_ON_WIN32
192         // c:/
193         || Component.endswith(":")
194 #endif
195         ) {
196       Component = Path.substr(Position, 1);
197       return *this;
198     }
199 
200     // Skip extra separators.
201     while (Position != Path.size() &&
202            is_separator(Path[Position])) {
203       ++Position;
204     }
205 
206     // Treat trailing '/' as a '.'.
207     if (Position == Path.size()) {
208       --Position;
209       Component = ".";
210       return *this;
211     }
212   }
213 
214   // Find next component.
215   size_t end_pos = Path.find_first_of(separators, Position);
216   Component = Path.slice(Position, end_pos);
217 
218   return *this;
219 }
220 
operator --()221 const_iterator &const_iterator::operator--() {
222   // If we're at the end and the previous char was a '/', return '.'.
223   if (Position == Path.size() &&
224       Path.size() > 1 &&
225       is_separator(Path[Position - 1])
226 #ifdef LLVM_ON_WIN32
227       && Path[Position - 2] != ':'
228 #endif
229       ) {
230     --Position;
231     Component = ".";
232     return *this;
233   }
234 
235   // Skip separators unless it's the root directory.
236   size_t root_dir_pos = root_dir_start(Path);
237   size_t end_pos = Position;
238 
239   while(end_pos > 0 &&
240         (end_pos - 1) != root_dir_pos &&
241         is_separator(Path[end_pos - 1]))
242     --end_pos;
243 
244   // Find next separator.
245   size_t start_pos = filename_pos(Path.substr(0, end_pos));
246   Component = Path.slice(start_pos, end_pos);
247   Position = start_pos;
248   return *this;
249 }
250 
operator ==(const const_iterator & RHS) const251 bool const_iterator::operator==(const const_iterator &RHS) const {
252   return Path.begin() == RHS.Path.begin() &&
253          Position == RHS.Position;
254 }
255 
operator !=(const const_iterator & RHS) const256 bool const_iterator::operator!=(const const_iterator &RHS) const {
257   return !(*this == RHS);
258 }
259 
operator -(const const_iterator & RHS) const260 ptrdiff_t const_iterator::operator-(const const_iterator &RHS) const {
261   return Position - RHS.Position;
262 }
263 
root_path(StringRef path)264 const StringRef root_path(StringRef path) {
265   const_iterator b = begin(path),
266                  pos = b,
267                  e = end(path);
268   if (b != e) {
269     bool has_net = b->size() > 2 && is_separator((*b)[0]) && (*b)[1] == (*b)[0];
270     bool has_drive =
271 #ifdef LLVM_ON_WIN32
272       b->endswith(":");
273 #else
274       false;
275 #endif
276 
277     if (has_net || has_drive) {
278       if ((++pos != e) && is_separator((*pos)[0])) {
279         // {C:/,//net/}, so get the first two components.
280         return path.substr(0, b->size() + pos->size());
281       } else {
282         // just {C:,//net}, return the first component.
283         return *b;
284       }
285     }
286 
287     // POSIX style root directory.
288     if (is_separator((*b)[0])) {
289       return *b;
290     }
291   }
292 
293   return StringRef();
294 }
295 
root_name(StringRef path)296 const StringRef root_name(StringRef path) {
297   const_iterator b = begin(path),
298                  e = end(path);
299   if (b != e) {
300     bool has_net = b->size() > 2 && is_separator((*b)[0]) && (*b)[1] == (*b)[0];
301     bool has_drive =
302 #ifdef LLVM_ON_WIN32
303       b->endswith(":");
304 #else
305       false;
306 #endif
307 
308     if (has_net || has_drive) {
309       // just {C:,//net}, return the first component.
310       return *b;
311     }
312   }
313 
314   // No path or no name.
315   return StringRef();
316 }
317 
root_directory(StringRef path)318 const StringRef root_directory(StringRef path) {
319   const_iterator b = begin(path),
320                  pos = b,
321                  e = end(path);
322   if (b != e) {
323     bool has_net = b->size() > 2 && is_separator((*b)[0]) && (*b)[1] == (*b)[0];
324     bool has_drive =
325 #ifdef LLVM_ON_WIN32
326       b->endswith(":");
327 #else
328       false;
329 #endif
330 
331     if ((has_net || has_drive) &&
332         // {C:,//net}, skip to the next component.
333         (++pos != e) && is_separator((*pos)[0])) {
334       return *pos;
335     }
336 
337     // POSIX style root directory.
338     if (!has_net && is_separator((*b)[0])) {
339       return *b;
340     }
341   }
342 
343   // No path or no root.
344   return StringRef();
345 }
346 
relative_path(StringRef path)347 const StringRef relative_path(StringRef path) {
348   StringRef root = root_path(path);
349   return path.substr(root.size());
350 }
351 
append(SmallVectorImpl<char> & path,const Twine & a,const Twine & b,const Twine & c,const Twine & d)352 void append(SmallVectorImpl<char> &path, const Twine &a,
353                                          const Twine &b,
354                                          const Twine &c,
355                                          const Twine &d) {
356   SmallString<32> a_storage;
357   SmallString<32> b_storage;
358   SmallString<32> c_storage;
359   SmallString<32> d_storage;
360 
361   SmallVector<StringRef, 4> components;
362   if (!a.isTriviallyEmpty()) components.push_back(a.toStringRef(a_storage));
363   if (!b.isTriviallyEmpty()) components.push_back(b.toStringRef(b_storage));
364   if (!c.isTriviallyEmpty()) components.push_back(c.toStringRef(c_storage));
365   if (!d.isTriviallyEmpty()) components.push_back(d.toStringRef(d_storage));
366 
367   for (SmallVectorImpl<StringRef>::const_iterator i = components.begin(),
368                                                   e = components.end();
369                                                   i != e; ++i) {
370     bool path_has_sep = !path.empty() && is_separator(path[path.size() - 1]);
371     bool component_has_sep = !i->empty() && is_separator((*i)[0]);
372     bool is_root_name = has_root_name(*i);
373 
374     if (path_has_sep) {
375       // Strip separators from beginning of component.
376       size_t loc = i->find_first_not_of(separators);
377       StringRef c = i->substr(loc);
378 
379       // Append it.
380       path.append(c.begin(), c.end());
381       continue;
382     }
383 
384     if (!component_has_sep && !(path.empty() || is_root_name)) {
385       // Add a separator.
386       path.push_back(prefered_separator);
387     }
388 
389     path.append(i->begin(), i->end());
390   }
391 }
392 
append(SmallVectorImpl<char> & path,const_iterator begin,const_iterator end)393 void append(SmallVectorImpl<char> &path,
394             const_iterator begin, const_iterator end) {
395   for (; begin != end; ++begin)
396     path::append(path, *begin);
397 }
398 
parent_path(StringRef path)399 const StringRef parent_path(StringRef path) {
400   size_t end_pos = parent_path_end(path);
401   if (end_pos == StringRef::npos)
402     return StringRef();
403   else
404     return path.substr(0, end_pos);
405 }
406 
remove_filename(SmallVectorImpl<char> & path)407 void remove_filename(SmallVectorImpl<char> &path) {
408   size_t end_pos = parent_path_end(StringRef(path.begin(), path.size()));
409   if (end_pos != StringRef::npos)
410     path.set_size(end_pos);
411 }
412 
replace_extension(SmallVectorImpl<char> & path,const Twine & extension)413 void replace_extension(SmallVectorImpl<char> &path, const Twine &extension) {
414   StringRef p(path.begin(), path.size());
415   SmallString<32> ext_storage;
416   StringRef ext = extension.toStringRef(ext_storage);
417 
418   // Erase existing extension.
419   size_t pos = p.find_last_of('.');
420   if (pos != StringRef::npos && pos >= filename_pos(p))
421     path.set_size(pos);
422 
423   // Append '.' if needed.
424   if (ext.size() > 0 && ext[0] != '.')
425     path.push_back('.');
426 
427   // Append extension.
428   path.append(ext.begin(), ext.end());
429 }
430 
native(const Twine & path,SmallVectorImpl<char> & result)431 void native(const Twine &path, SmallVectorImpl<char> &result) {
432   // Clear result.
433   result.clear();
434 #ifdef LLVM_ON_WIN32
435   SmallString<128> path_storage;
436   StringRef p = path.toStringRef(path_storage);
437   result.reserve(p.size());
438   for (StringRef::const_iterator i = p.begin(),
439                                  e = p.end();
440                                  i != e;
441                                  ++i) {
442     if (*i == '/')
443       result.push_back('\\');
444     else
445       result.push_back(*i);
446   }
447 #else
448   path.toVector(result);
449 #endif
450 }
451 
filename(StringRef path)452 const StringRef filename(StringRef path) {
453   return *(--end(path));
454 }
455 
stem(StringRef path)456 const StringRef stem(StringRef path) {
457   StringRef fname = filename(path);
458   size_t pos = fname.find_last_of('.');
459   if (pos == StringRef::npos)
460     return fname;
461   else
462     if ((fname.size() == 1 && fname == ".") ||
463         (fname.size() == 2 && fname == ".."))
464       return fname;
465     else
466       return fname.substr(0, pos);
467 }
468 
extension(StringRef path)469 const StringRef extension(StringRef path) {
470   StringRef fname = filename(path);
471   size_t pos = fname.find_last_of('.');
472   if (pos == StringRef::npos)
473     return StringRef();
474   else
475     if ((fname.size() == 1 && fname == ".") ||
476         (fname.size() == 2 && fname == ".."))
477       return StringRef();
478     else
479       return fname.substr(pos);
480 }
481 
is_separator(char value)482 bool is_separator(char value) {
483   switch(value) {
484 #ifdef LLVM_ON_WIN32
485     case '\\': // fall through
486 #endif
487     case '/': return true;
488     default: return false;
489   }
490 }
491 
system_temp_directory(bool erasedOnReboot,SmallVectorImpl<char> & result)492 void system_temp_directory(bool erasedOnReboot, SmallVectorImpl<char> &result) {
493   result.clear();
494 
495   // Check whether the temporary directory is specified by an environment
496   // variable.
497   const char *EnvironmentVariable;
498 #ifdef LLVM_ON_WIN32
499   EnvironmentVariable = "TEMP";
500 #else
501   EnvironmentVariable = "TMPDIR";
502 #endif
503   if (char *RequestedDir = getenv(EnvironmentVariable)) {
504     result.append(RequestedDir, RequestedDir + strlen(RequestedDir));
505     return;
506   }
507 
508   // Fall back to a system default.
509   const char *DefaultResult;
510 #ifdef LLVM_ON_WIN32
511   (void)erasedOnReboot;
512   DefaultResult = "C:\\TEMP";
513 #else
514   if (erasedOnReboot)
515     DefaultResult = "/tmp";
516   else
517     DefaultResult = "/var/tmp";
518 #endif
519   result.append(DefaultResult, DefaultResult + strlen(DefaultResult));
520 }
521 
has_root_name(const Twine & path)522 bool has_root_name(const Twine &path) {
523   SmallString<128> path_storage;
524   StringRef p = path.toStringRef(path_storage);
525 
526   return !root_name(p).empty();
527 }
528 
has_root_directory(const Twine & path)529 bool has_root_directory(const Twine &path) {
530   SmallString<128> path_storage;
531   StringRef p = path.toStringRef(path_storage);
532 
533   return !root_directory(p).empty();
534 }
535 
has_root_path(const Twine & path)536 bool has_root_path(const Twine &path) {
537   SmallString<128> path_storage;
538   StringRef p = path.toStringRef(path_storage);
539 
540   return !root_path(p).empty();
541 }
542 
has_relative_path(const Twine & path)543 bool has_relative_path(const Twine &path) {
544   SmallString<128> path_storage;
545   StringRef p = path.toStringRef(path_storage);
546 
547   return !relative_path(p).empty();
548 }
549 
has_filename(const Twine & path)550 bool has_filename(const Twine &path) {
551   SmallString<128> path_storage;
552   StringRef p = path.toStringRef(path_storage);
553 
554   return !filename(p).empty();
555 }
556 
has_parent_path(const Twine & path)557 bool has_parent_path(const Twine &path) {
558   SmallString<128> path_storage;
559   StringRef p = path.toStringRef(path_storage);
560 
561   return !parent_path(p).empty();
562 }
563 
has_stem(const Twine & path)564 bool has_stem(const Twine &path) {
565   SmallString<128> path_storage;
566   StringRef p = path.toStringRef(path_storage);
567 
568   return !stem(p).empty();
569 }
570 
has_extension(const Twine & path)571 bool has_extension(const Twine &path) {
572   SmallString<128> path_storage;
573   StringRef p = path.toStringRef(path_storage);
574 
575   return !extension(p).empty();
576 }
577 
is_absolute(const Twine & path)578 bool is_absolute(const Twine &path) {
579   SmallString<128> path_storage;
580   StringRef p = path.toStringRef(path_storage);
581 
582   bool rootDir = has_root_directory(p),
583 #ifdef LLVM_ON_WIN32
584        rootName = has_root_name(p);
585 #else
586        rootName = true;
587 #endif
588 
589   return rootDir && rootName;
590 }
591 
is_relative(const Twine & path)592 bool is_relative(const Twine &path) {
593   return !is_absolute(path);
594 }
595 
596 } // end namespace path
597 
598 namespace fs {
599 
make_absolute(SmallVectorImpl<char> & path)600 error_code make_absolute(SmallVectorImpl<char> &path) {
601   StringRef p(path.data(), path.size());
602 
603   bool rootDirectory = path::has_root_directory(p),
604 #ifdef LLVM_ON_WIN32
605        rootName = path::has_root_name(p);
606 #else
607        rootName = true;
608 #endif
609 
610   // Already absolute.
611   if (rootName && rootDirectory)
612     return error_code::success();
613 
614   // All of the following conditions will need the current directory.
615   SmallString<128> current_dir;
616   if (error_code ec = current_path(current_dir)) return ec;
617 
618   // Relative path. Prepend the current directory.
619   if (!rootName && !rootDirectory) {
620     // Append path to the current directory.
621     path::append(current_dir, p);
622     // Set path to the result.
623     path.swap(current_dir);
624     return error_code::success();
625   }
626 
627   if (!rootName && rootDirectory) {
628     StringRef cdrn = path::root_name(current_dir);
629     SmallString<128> curDirRootName(cdrn.begin(), cdrn.end());
630     path::append(curDirRootName, p);
631     // Set path to the result.
632     path.swap(curDirRootName);
633     return error_code::success();
634   }
635 
636   if (rootName && !rootDirectory) {
637     StringRef pRootName      = path::root_name(p);
638     StringRef bRootDirectory = path::root_directory(current_dir);
639     StringRef bRelativePath  = path::relative_path(current_dir);
640     StringRef pRelativePath  = path::relative_path(p);
641 
642     SmallString<128> res;
643     path::append(res, pRootName, bRootDirectory, bRelativePath, pRelativePath);
644     path.swap(res);
645     return error_code::success();
646   }
647 
648   llvm_unreachable("All rootName and rootDirectory combinations should have "
649                    "occurred above!");
650 }
651 
create_directories(const Twine & path,bool & existed)652 error_code create_directories(const Twine &path, bool &existed) {
653   SmallString<128> path_storage;
654   StringRef p = path.toStringRef(path_storage);
655 
656   StringRef parent = path::parent_path(p);
657   if (!parent.empty()) {
658     bool parent_exists;
659     if (error_code ec = fs::exists(parent, parent_exists)) return ec;
660 
661     if (!parent_exists)
662       if (error_code ec = create_directories(parent, existed)) return ec;
663   }
664 
665   return create_directory(p, existed);
666 }
667 
exists(file_status status)668 bool exists(file_status status) {
669   return status_known(status) && status.type() != file_type::file_not_found;
670 }
671 
status_known(file_status s)672 bool status_known(file_status s) {
673   return s.type() != file_type::status_error;
674 }
675 
is_directory(file_status status)676 bool is_directory(file_status status) {
677   return status.type() == file_type::directory_file;
678 }
679 
is_directory(const Twine & path,bool & result)680 error_code is_directory(const Twine &path, bool &result) {
681   file_status st;
682   if (error_code ec = status(path, st))
683     return ec;
684   result = is_directory(st);
685   return error_code::success();
686 }
687 
is_regular_file(file_status status)688 bool is_regular_file(file_status status) {
689   return status.type() == file_type::regular_file;
690 }
691 
is_regular_file(const Twine & path,bool & result)692 error_code is_regular_file(const Twine &path, bool &result) {
693   file_status st;
694   if (error_code ec = status(path, st))
695     return ec;
696   result = is_regular_file(st);
697   return error_code::success();
698 }
699 
is_symlink(file_status status)700 bool is_symlink(file_status status) {
701   return status.type() == file_type::symlink_file;
702 }
703 
is_symlink(const Twine & path,bool & result)704 error_code is_symlink(const Twine &path, bool &result) {
705   file_status st;
706   if (error_code ec = status(path, st))
707     return ec;
708   result = is_symlink(st);
709   return error_code::success();
710 }
711 
is_other(file_status status)712 bool is_other(file_status status) {
713   return exists(status) &&
714          !is_regular_file(status) &&
715          !is_directory(status) &&
716          !is_symlink(status);
717 }
718 
replace_filename(const Twine & filename,file_status st)719 void directory_entry::replace_filename(const Twine &filename, file_status st) {
720   SmallString<128> path(Path.begin(), Path.end());
721   path::remove_filename(path);
722   path::append(path, filename);
723   Path = path.str();
724   Status = st;
725 }
726 
has_magic(const Twine & path,const Twine & magic,bool & result)727 error_code has_magic(const Twine &path, const Twine &magic, bool &result) {
728   SmallString<32>  MagicStorage;
729   StringRef Magic = magic.toStringRef(MagicStorage);
730   SmallString<32> Buffer;
731 
732   if (error_code ec = get_magic(path, Magic.size(), Buffer)) {
733     if (ec == errc::value_too_large) {
734       // Magic.size() > file_size(Path).
735       result = false;
736       return error_code::success();
737     }
738     return ec;
739   }
740 
741   result = Magic == Buffer;
742   return error_code::success();
743 }
744 
745 /// @brief Identify the magic in magic.
identify_magic(StringRef magic)746 file_magic identify_magic(StringRef magic) {
747   if (magic.size() < 4)
748     return file_magic::unknown;
749   switch ((unsigned char)magic[0]) {
750     case 0xDE:  // 0x0B17C0DE = BC wraper
751       if (magic[1] == (char)0xC0 && magic[2] == (char)0x17 &&
752           magic[3] == (char)0x0B)
753         return file_magic::bitcode;
754       break;
755     case 'B':
756       if (magic[1] == 'C' && magic[2] == (char)0xC0 && magic[3] == (char)0xDE)
757         return file_magic::bitcode;
758       break;
759     case '!':
760       if (magic.size() >= 8)
761         if (memcmp(magic.data(),"!<arch>\n",8) == 0)
762           return file_magic::archive;
763       break;
764 
765     case '\177':
766       if (magic[1] == 'E' && magic[2] == 'L' && magic[3] == 'F') {
767         if (magic.size() >= 18 && magic[17] == 0)
768           switch (magic[16]) {
769             default: break;
770             case 1: return file_magic::elf_relocatable;
771             case 2: return file_magic::elf_executable;
772             case 3: return file_magic::elf_shared_object;
773             case 4: return file_magic::elf_core;
774           }
775       }
776       break;
777 
778     case 0xCA:
779       if (magic[1] == char(0xFE) && magic[2] == char(0xBA) &&
780           magic[3] == char(0xBE)) {
781         // This is complicated by an overlap with Java class files.
782         // See the Mach-O section in /usr/share/file/magic for details.
783         if (magic.size() >= 8 && magic[7] < 43)
784           // FIXME: Universal Binary of any type.
785           return file_magic::macho_dynamically_linked_shared_lib;
786       }
787       break;
788 
789       // The two magic numbers for mach-o are:
790       // 0xfeedface - 32-bit mach-o
791       // 0xfeedfacf - 64-bit mach-o
792     case 0xFE:
793     case 0xCE:
794     case 0xCF: {
795       uint16_t type = 0;
796       if (magic[0] == char(0xFE) && magic[1] == char(0xED) &&
797           magic[2] == char(0xFA) &&
798           (magic[3] == char(0xCE) || magic[3] == char(0xCF))) {
799         /* Native endian */
800         if (magic.size() >= 16) type = magic[14] << 8 | magic[15];
801       } else if ((magic[0] == char(0xCE) || magic[0] == char(0xCF)) &&
802                  magic[1] == char(0xFA) && magic[2] == char(0xED) &&
803                  magic[3] == char(0xFE)) {
804         /* Reverse endian */
805         if (magic.size() >= 14) type = magic[13] << 8 | magic[12];
806       }
807       switch (type) {
808         default: break;
809         case 1: return file_magic::macho_object;
810         case 2: return file_magic::macho_executable;
811         case 3: return file_magic::macho_fixed_virtual_memory_shared_lib;
812         case 4: return file_magic::macho_core;
813         case 5: return file_magic::macho_preload_executabl;
814         case 6: return file_magic::macho_dynamically_linked_shared_lib;
815         case 7: return file_magic::macho_dynamic_linker;
816         case 8: return file_magic::macho_bundle;
817         case 9: return file_magic::macho_dynamic_linker;
818         case 10: return file_magic::macho_dsym_companion;
819       }
820       break;
821     }
822     case 0xF0: // PowerPC Windows
823     case 0x83: // Alpha 32-bit
824     case 0x84: // Alpha 64-bit
825     case 0x66: // MPS R4000 Windows
826     case 0x50: // mc68K
827     case 0x4c: // 80386 Windows
828       if (magic[1] == 0x01)
829         return file_magic::coff_object;
830 
831     case 0x90: // PA-RISC Windows
832     case 0x68: // mc68K Windows
833       if (magic[1] == 0x02)
834         return file_magic::coff_object;
835       break;
836 
837     case 0x4d: // Possible MS-DOS stub on Windows PE file
838       if (magic[1] == 0x5a) {
839         uint32_t off =
840           *reinterpret_cast<const support::ulittle32_t*>(magic.data() + 0x3c);
841         // PE/COFF file, either EXE or DLL.
842         if (off < magic.size() && memcmp(magic.data() + off, "PE\0\0",4) == 0)
843           return file_magic::pecoff_executable;
844       }
845       break;
846 
847     case 0x64: // x86-64 Windows.
848       if (magic[1] == char(0x86))
849         return file_magic::coff_object;
850       break;
851 
852     default:
853       break;
854   }
855   return file_magic::unknown;
856 }
857 
identify_magic(const Twine & path,file_magic & result)858 error_code identify_magic(const Twine &path, file_magic &result) {
859   SmallString<32> Magic;
860   error_code ec = get_magic(path, Magic.capacity(), Magic);
861   if (ec && ec != errc::value_too_large)
862     return ec;
863 
864   result = identify_magic(Magic);
865   return error_code::success();
866 }
867 
868 namespace {
remove_all_r(StringRef path,file_type ft,uint32_t & count)869 error_code remove_all_r(StringRef path, file_type ft, uint32_t &count) {
870   if (ft == file_type::directory_file) {
871     // This code would be a lot better with exceptions ;/.
872     error_code ec;
873     directory_iterator i(path, ec);
874     if (ec) return ec;
875     for (directory_iterator e; i != e; i.increment(ec)) {
876       if (ec) return ec;
877       file_status st;
878       if (error_code ec = i->status(st)) return ec;
879       if (error_code ec = remove_all_r(i->path(), st.type(), count)) return ec;
880     }
881     bool obviously_this_exists;
882     if (error_code ec = remove(path, obviously_this_exists)) return ec;
883     assert(obviously_this_exists);
884     ++count; // Include the directory itself in the items removed.
885   } else {
886     bool obviously_this_exists;
887     if (error_code ec = remove(path, obviously_this_exists)) return ec;
888     assert(obviously_this_exists);
889     ++count;
890   }
891 
892   return error_code::success();
893 }
894 } // end unnamed namespace
895 
remove_all(const Twine & path,uint32_t & num_removed)896 error_code remove_all(const Twine &path, uint32_t &num_removed) {
897   SmallString<128> path_storage;
898   StringRef p = path.toStringRef(path_storage);
899 
900   file_status fs;
901   if (error_code ec = status(path, fs))
902     return ec;
903   num_removed = 0;
904   return remove_all_r(p, fs.type(), num_removed);
905 }
906 
status(file_status & result) const907 error_code directory_entry::status(file_status &result) const {
908   return fs::status(Path, result);
909 }
910 
911 } // end namespace fs
912 } // end namespace sys
913 } // end namespace llvm
914 
915 // Include the truly platform-specific parts.
916 #if defined(LLVM_ON_UNIX)
917 #include "Unix/PathV2.inc"
918 #endif
919 #if defined(LLVM_ON_WIN32)
920 #include "Windows/PathV2.inc"
921 #endif
922