3ef13e253f0ddc56e23887ff56aee11075144798
[oota-llvm.git] / lib / Support / PathV2.cpp
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
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
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
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
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
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
161 const_iterator end(StringRef path) {
162   const_iterator i;
163   i.Path      = path;
164   i.Position  = path.size();
165   return i;
166 }
167
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
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
251 bool const_iterator::operator==(const const_iterator &RHS) const {
252   return Path.begin() == RHS.Path.begin() &&
253          Position == RHS.Position;
254 }
255
256 bool const_iterator::operator!=(const const_iterator &RHS) const {
257   return !(*this == RHS);
258 }
259
260 ptrdiff_t const_iterator::operator-(const const_iterator &RHS) const {
261   return Position - RHS.Position;
262 }
263
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
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
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
347 const StringRef relative_path(StringRef path) {
348   StringRef root = root_path(path);
349   return root.substr(root.size());
350 }
351
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
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
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
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
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
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
452 const StringRef filename(StringRef path) {
453   return *(--end(path));
454 }
455
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
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
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
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
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
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
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
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
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
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
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
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
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
592 bool is_relative(const Twine &path) {
593   return !is_absolute(path);
594 }
595
596 } // end namespace path
597
598 namespace fs {
599
600 error_code make_absolute(SmallVectorImpl<char> &path) {
601   StringRef p(path.data(), path.size());
602
603   bool rootName      = path::has_root_name(p),
604        rootDirectory = path::has_root_directory(p);
605
606   // Already absolute.
607   if (rootName && rootDirectory)
608     return error_code::success();
609
610   // All of the following conditions will need the current directory.
611   SmallString<128> current_dir;
612   if (error_code ec = current_path(current_dir)) return ec;
613
614   // Relative path. Prepend the current directory.
615   if (!rootName && !rootDirectory) {
616     // Append path to the current directory.
617     path::append(current_dir, p);
618     // Set path to the result.
619     path.swap(current_dir);
620     return error_code::success();
621   }
622
623   if (!rootName && rootDirectory) {
624     StringRef cdrn = path::root_name(current_dir);
625     SmallString<128> curDirRootName(cdrn.begin(), cdrn.end());
626     path::append(curDirRootName, p);
627     // Set path to the result.
628     path.swap(curDirRootName);
629     return error_code::success();
630   }
631
632   if (rootName && !rootDirectory) {
633     StringRef pRootName      = path::root_name(p);
634     StringRef bRootDirectory = path::root_directory(current_dir);
635     StringRef bRelativePath  = path::relative_path(current_dir);
636     StringRef pRelativePath  = path::relative_path(p);
637
638     SmallString<128> res;
639     path::append(res, pRootName, bRootDirectory, bRelativePath, pRelativePath);
640     path.swap(res);
641     return error_code::success();
642   }
643
644   llvm_unreachable("All rootName and rootDirectory combinations should have "
645                    "occurred above!");
646 }
647
648 error_code create_directories(const Twine &path, bool &existed) {
649   SmallString<128> path_storage;
650   StringRef p = path.toStringRef(path_storage);
651
652   StringRef parent = path::parent_path(p);
653   bool parent_exists;
654
655   if (error_code ec = fs::exists(parent, parent_exists)) return ec;
656
657   if (!parent_exists)
658     if (error_code ec = create_directories(parent, existed)) return ec;
659
660   return create_directory(p, existed);
661 }
662
663 bool exists(file_status status) {
664   return status_known(status) && status.type() != file_type::file_not_found;
665 }
666
667 bool status_known(file_status s) {
668   return s.type() != file_type::status_error;
669 }
670
671 bool is_directory(file_status status) {
672   return status.type() == file_type::directory_file;
673 }
674
675 error_code is_directory(const Twine &path, bool &result) {
676   file_status st;
677   if (error_code ec = status(path, st))
678     return ec;
679   result = is_directory(st);
680   return error_code::success();
681 }
682
683 bool is_regular_file(file_status status) {
684   return status.type() == file_type::regular_file;
685 }
686
687 error_code is_regular_file(const Twine &path, bool &result) {
688   file_status st;
689   if (error_code ec = status(path, st))
690     return ec;
691   result = is_regular_file(st);
692   return error_code::success();
693 }
694
695 bool is_symlink(file_status status) {
696   return status.type() == file_type::symlink_file;
697 }
698
699 error_code is_symlink(const Twine &path, bool &result) {
700   file_status st;
701   if (error_code ec = status(path, st))
702     return ec;
703   result = is_symlink(st);
704   return error_code::success();
705 }
706
707 bool is_other(file_status status) {
708   return exists(status) &&
709          !is_regular_file(status) &&
710          !is_directory(status) &&
711          !is_symlink(status);
712 }
713
714 void directory_entry::replace_filename(const Twine &filename, file_status st) {
715   SmallString<128> path(Path.begin(), Path.end());
716   path::remove_filename(path);
717   path::append(path, filename);
718   Path = path.str();
719   Status = st;
720 }
721
722 error_code has_magic(const Twine &path, const Twine &magic, bool &result) {
723   SmallString<32>  MagicStorage;
724   StringRef Magic = magic.toStringRef(MagicStorage);
725   SmallString<32> Buffer;
726
727   if (error_code ec = get_magic(path, Magic.size(), Buffer)) {
728     if (ec == errc::value_too_large) {
729       // Magic.size() > file_size(Path).
730       result = false;
731       return error_code::success();
732     }
733     return ec;
734   }
735
736   result = Magic == Buffer;
737   return error_code::success();
738 }
739
740 /// @brief Identify the magic in magic.
741 file_magic identify_magic(StringRef magic) {
742   switch ((unsigned char)magic[0]) {
743     case 0xDE:  // 0x0B17C0DE = BC wraper
744       if (magic[1] == (char)0xC0 && magic[2] == (char)0x17 &&
745           magic[3] == (char)0x0B)
746         return file_magic::bitcode;
747       break;
748     case 'B':
749       if (magic[1] == 'C' && magic[2] == (char)0xC0 && magic[3] == (char)0xDE)
750         return file_magic::bitcode;
751       break;
752     case '!':
753       if (magic.size() >= 8)
754         if (memcmp(magic.data(),"!<arch>\n",8) == 0)
755           return file_magic::archive;
756       break;
757
758     case '\177':
759       if (magic[1] == 'E' && magic[2] == 'L' && magic[3] == 'F') {
760         if (magic.size() >= 18 && magic[17] == 0)
761           switch (magic[16]) {
762             default: break;
763             case 1: return file_magic::elf_relocatable;
764             case 2: return file_magic::elf_executable;
765             case 3: return file_magic::elf_shared_object;
766             case 4: return file_magic::elf_core;
767           }
768       }
769       break;
770
771     case 0xCA:
772       if (magic[1] == char(0xFE) && magic[2] == char(0xBA) &&
773           magic[3] == char(0xBE)) {
774         // This is complicated by an overlap with Java class files.
775         // See the Mach-O section in /usr/share/file/magic for details.
776         if (magic.size() >= 8 && magic[7] < 43)
777           // FIXME: Universal Binary of any type.
778           return file_magic::macho_dynamically_linked_shared_lib;
779       }
780       break;
781
782       // The two magic numbers for mach-o are:
783       // 0xfeedface - 32-bit mach-o
784       // 0xfeedfacf - 64-bit mach-o
785     case 0xFE:
786     case 0xCE:
787     case 0xCF: {
788       uint16_t type = 0;
789       if (magic[0] == char(0xFE) && magic[1] == char(0xED) &&
790           magic[2] == char(0xFA) &&
791           (magic[3] == char(0xCE) || magic[3] == char(0xCF))) {
792         /* Native endian */
793         if (magic.size() >= 16) type = magic[14] << 8 | magic[15];
794       } else if ((magic[0] == char(0xCE) || magic[0] == char(0xCF)) &&
795                  magic[1] == char(0xFA) && magic[2] == char(0xED) &&
796                  magic[3] == char(0xFE)) {
797         /* Reverse endian */
798         if (magic.size() >= 14) type = magic[13] << 8 | magic[12];
799       }
800       switch (type) {
801         default: break;
802         case 1: return file_magic::macho_object;
803         case 2: return file_magic::macho_executable;
804         case 3: return file_magic::macho_fixed_virtual_memory_shared_lib;
805         case 4: return file_magic::macho_core;
806         case 5: return file_magic::macho_preload_executabl;
807         case 6: return file_magic::macho_dynamically_linked_shared_lib;
808         case 7: return file_magic::macho_dynamic_linker;
809         case 8: return file_magic::macho_bundle;
810         case 9: return file_magic::macho_dynamic_linker;
811         case 10: return file_magic::macho_dsym_companion;
812       }
813       break;
814     }
815     case 0xF0: // PowerPC Windows
816     case 0x83: // Alpha 32-bit
817     case 0x84: // Alpha 64-bit
818     case 0x66: // MPS R4000 Windows
819     case 0x50: // mc68K
820     case 0x4c: // 80386 Windows
821       if (magic[1] == 0x01)
822         return file_magic::coff_object;
823
824     case 0x90: // PA-RISC Windows
825     case 0x68: // mc68K Windows
826       if (magic[1] == 0x02)
827         return file_magic::coff_object;
828       break;
829
830     case 0x4d: // Possible MS-DOS stub on Windows PE file
831       if (magic[1] == 0x5a) {
832         uint32_t off =
833           *reinterpret_cast<const support::ulittle32_t*>(magic.data() + 0x3c);
834         // PE/COFF file, either EXE or DLL.
835         if (off < magic.size() && memcmp(magic.data() + off, "PE\0\0",4) == 0)
836           return file_magic::pecoff_executable;
837       }
838       break;
839
840     case 0x64: // x86-64 Windows.
841       if (magic[1] == char(0x86))
842         return file_magic::coff_object;
843       break;
844
845     default:
846       break;
847   }
848   return file_magic::unknown;
849 }
850
851 error_code identify_magic(const Twine &path, file_magic &result) {
852   SmallString<32> Magic;
853   error_code ec = get_magic(path, Magic.capacity(), Magic);
854   if (ec && ec != errc::value_too_large)
855     return ec;
856
857   result = identify_magic(Magic);
858   return error_code::success();
859 }
860
861 namespace {
862 error_code remove_all_r(StringRef path, file_type ft, uint32_t &count) {
863   if (ft == file_type::directory_file) {
864     // This code would be a lot better with exceptions ;/.
865     error_code ec;
866     directory_iterator i(path, ec);
867     if (ec) return ec;
868     for (directory_iterator e; i != e; i.increment(ec)) {
869       if (ec) return ec;
870       file_status st;
871       if (error_code ec = i->status(st)) return ec;
872       if (error_code ec = remove_all_r(i->path(), st.type(), count)) return ec;
873     }
874     bool obviously_this_exists;
875     if (error_code ec = remove(path, obviously_this_exists)) return ec;
876     assert(obviously_this_exists);
877     ++count; // Include the directory itself in the items removed.
878   } else {
879     bool obviously_this_exists;
880     if (error_code ec = remove(path, obviously_this_exists)) return ec;
881     assert(obviously_this_exists);
882     ++count;
883   }
884
885   return error_code::success();
886 }
887 } // end unnamed namespace
888
889 error_code remove_all(const Twine &path, uint32_t &num_removed) {
890   SmallString<128> path_storage;
891   StringRef p = path.toStringRef(path_storage);
892
893   file_status fs;
894   if (error_code ec = status(path, fs))
895     return ec;
896   num_removed = 0;
897   return remove_all_r(p, fs.type(), num_removed);
898 }
899
900 error_code directory_entry::status(file_status &result) const {
901   return fs::status(Path, result);
902 }
903
904 } // end namespace fs
905 } // end namespace sys
906 } // end namespace llvm
907
908 // Include the truly platform-specific parts.
909 #if defined(LLVM_ON_UNIX)
910 #include "Unix/PathV2.inc"
911 #endif
912 #if defined(LLVM_ON_WIN32)
913 #include "Windows/PathV2.inc"
914 #endif