xref: /src/contrib/llvm-project/clang/lib/InstallAPI/HeaderFile.cpp (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
1ac9a064cSDimitry Andric //===- HeaderFile.cpp ------------------------------------------*- C++ -*-===//
2ac9a064cSDimitry Andric //
3ac9a064cSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4ac9a064cSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5ac9a064cSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6ac9a064cSDimitry Andric //
7ac9a064cSDimitry Andric //===----------------------------------------------------------------------===//
8ac9a064cSDimitry Andric 
9ac9a064cSDimitry Andric #include "clang/InstallAPI/HeaderFile.h"
10ac9a064cSDimitry Andric #include "llvm/TextAPI/Utils.h"
11ac9a064cSDimitry Andric 
12ac9a064cSDimitry Andric using namespace llvm;
13ac9a064cSDimitry Andric namespace clang::installapi {
14ac9a064cSDimitry Andric 
getFrameworkIncludeRule()15ac9a064cSDimitry Andric llvm::Regex HeaderFile::getFrameworkIncludeRule() {
16ac9a064cSDimitry Andric   return llvm::Regex("/(.+)\\.framework/(.+)?Headers/(.+)");
17ac9a064cSDimitry Andric }
18ac9a064cSDimitry Andric 
createIncludeHeaderName(const StringRef FullPath)19ac9a064cSDimitry Andric std::optional<std::string> createIncludeHeaderName(const StringRef FullPath) {
20ac9a064cSDimitry Andric   // Headers in usr(/local)*/include.
21ac9a064cSDimitry Andric   std::string Pattern = "/include/";
22ac9a064cSDimitry Andric   auto PathPrefix = FullPath.find(Pattern);
23ac9a064cSDimitry Andric   if (PathPrefix != StringRef::npos) {
24ac9a064cSDimitry Andric     PathPrefix += Pattern.size();
25ac9a064cSDimitry Andric     return FullPath.drop_front(PathPrefix).str();
26ac9a064cSDimitry Andric   }
27ac9a064cSDimitry Andric 
28ac9a064cSDimitry Andric   // Framework Headers.
29ac9a064cSDimitry Andric   SmallVector<StringRef, 4> Matches;
30ac9a064cSDimitry Andric   HeaderFile::getFrameworkIncludeRule().match(FullPath, &Matches);
31ac9a064cSDimitry Andric   // Returned matches are always in stable order.
32ac9a064cSDimitry Andric   if (Matches.size() != 4)
33ac9a064cSDimitry Andric     return std::nullopt;
34ac9a064cSDimitry Andric 
35ac9a064cSDimitry Andric   return Matches[1].drop_front(Matches[1].rfind('/') + 1).str() + "/" +
36ac9a064cSDimitry Andric          Matches[3].str();
37ac9a064cSDimitry Andric }
38ac9a064cSDimitry Andric 
isHeaderFile(StringRef Path)39ac9a064cSDimitry Andric bool isHeaderFile(StringRef Path) {
40ac9a064cSDimitry Andric   return StringSwitch<bool>(sys::path::extension(Path))
41ac9a064cSDimitry Andric       .Cases(".h", ".H", ".hh", ".hpp", ".hxx", true)
42ac9a064cSDimitry Andric       .Default(false);
43ac9a064cSDimitry Andric }
44ac9a064cSDimitry Andric 
enumerateFiles(FileManager & FM,StringRef Directory)45ac9a064cSDimitry Andric llvm::Expected<PathSeq> enumerateFiles(FileManager &FM, StringRef Directory) {
46ac9a064cSDimitry Andric   PathSeq Files;
47ac9a064cSDimitry Andric   std::error_code EC;
48ac9a064cSDimitry Andric   auto &FS = FM.getVirtualFileSystem();
49ac9a064cSDimitry Andric   for (llvm::vfs::recursive_directory_iterator i(FS, Directory, EC), ie;
50ac9a064cSDimitry Andric        i != ie; i.increment(EC)) {
51ac9a064cSDimitry Andric     if (EC)
52ac9a064cSDimitry Andric       return errorCodeToError(EC);
53ac9a064cSDimitry Andric 
54ac9a064cSDimitry Andric     // Skip files that do not exist. This usually happens for broken symlinks.
55ac9a064cSDimitry Andric     if (FS.status(i->path()) == std::errc::no_such_file_or_directory)
56ac9a064cSDimitry Andric       continue;
57ac9a064cSDimitry Andric 
58ac9a064cSDimitry Andric     StringRef Path = i->path();
59ac9a064cSDimitry Andric     if (isHeaderFile(Path))
60ac9a064cSDimitry Andric       Files.emplace_back(Path);
61ac9a064cSDimitry Andric   }
62ac9a064cSDimitry Andric 
63ac9a064cSDimitry Andric   return Files;
64ac9a064cSDimitry Andric }
65ac9a064cSDimitry Andric 
HeaderGlob(StringRef GlobString,Regex && Rule,HeaderType Type)66ac9a064cSDimitry Andric HeaderGlob::HeaderGlob(StringRef GlobString, Regex &&Rule, HeaderType Type)
67ac9a064cSDimitry Andric     : GlobString(GlobString), Rule(std::move(Rule)), Type(Type) {}
68ac9a064cSDimitry Andric 
match(const HeaderFile & Header)69ac9a064cSDimitry Andric bool HeaderGlob::match(const HeaderFile &Header) {
70ac9a064cSDimitry Andric   if (Header.getType() != Type)
71ac9a064cSDimitry Andric     return false;
72ac9a064cSDimitry Andric 
73ac9a064cSDimitry Andric   bool Match = Rule.match(Header.getPath());
74ac9a064cSDimitry Andric   if (Match)
75ac9a064cSDimitry Andric     FoundMatch = true;
76ac9a064cSDimitry Andric   return Match;
77ac9a064cSDimitry Andric }
78ac9a064cSDimitry Andric 
create(StringRef GlobString,HeaderType Type)79ac9a064cSDimitry Andric Expected<std::unique_ptr<HeaderGlob>> HeaderGlob::create(StringRef GlobString,
80ac9a064cSDimitry Andric                                                          HeaderType Type) {
81ac9a064cSDimitry Andric   auto Rule = MachO::createRegexFromGlob(GlobString);
82ac9a064cSDimitry Andric   if (!Rule)
83ac9a064cSDimitry Andric     return Rule.takeError();
84ac9a064cSDimitry Andric 
85ac9a064cSDimitry Andric   return std::make_unique<HeaderGlob>(GlobString, std::move(*Rule), Type);
86ac9a064cSDimitry Andric }
87ac9a064cSDimitry Andric 
88ac9a064cSDimitry Andric } // namespace clang::installapi
89