xref: /llvm-project/clang/lib/InstallAPI/HeaderFile.cpp (revision 487720fc137ac7e97b15ca50ca2fc4a42430a6be)
14c6043deSCyndy Ishida //===- HeaderFile.cpp ------------------------------------------*- C++ -*-===//
24c6043deSCyndy Ishida //
34c6043deSCyndy Ishida // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
44c6043deSCyndy Ishida // See https://llvm.org/LICENSE.txt for license information.
54c6043deSCyndy Ishida // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
64c6043deSCyndy Ishida //
74c6043deSCyndy Ishida //===----------------------------------------------------------------------===//
84c6043deSCyndy Ishida 
94c6043deSCyndy Ishida #include "clang/InstallAPI/HeaderFile.h"
10*487720fcSCyndy Ishida #include "llvm/TextAPI/Utils.h"
114c6043deSCyndy Ishida 
124c6043deSCyndy Ishida using namespace llvm;
134c6043deSCyndy Ishida namespace clang::installapi {
144c6043deSCyndy Ishida 
getFrameworkIncludeRule()154c6043deSCyndy Ishida llvm::Regex HeaderFile::getFrameworkIncludeRule() {
164c6043deSCyndy Ishida   return llvm::Regex("/(.+)\\.framework/(.+)?Headers/(.+)");
174c6043deSCyndy Ishida }
184c6043deSCyndy Ishida 
createIncludeHeaderName(const StringRef FullPath)194c6043deSCyndy Ishida std::optional<std::string> createIncludeHeaderName(const StringRef FullPath) {
204c6043deSCyndy Ishida   // Headers in usr(/local)*/include.
214c6043deSCyndy Ishida   std::string Pattern = "/include/";
224c6043deSCyndy Ishida   auto PathPrefix = FullPath.find(Pattern);
234c6043deSCyndy Ishida   if (PathPrefix != StringRef::npos) {
244c6043deSCyndy Ishida     PathPrefix += Pattern.size();
254c6043deSCyndy Ishida     return FullPath.drop_front(PathPrefix).str();
264c6043deSCyndy Ishida   }
274c6043deSCyndy Ishida 
284c6043deSCyndy Ishida   // Framework Headers.
294c6043deSCyndy Ishida   SmallVector<StringRef, 4> Matches;
304c6043deSCyndy Ishida   HeaderFile::getFrameworkIncludeRule().match(FullPath, &Matches);
314c6043deSCyndy Ishida   // Returned matches are always in stable order.
324c6043deSCyndy Ishida   if (Matches.size() != 4)
334c6043deSCyndy Ishida     return std::nullopt;
344c6043deSCyndy Ishida 
354c6043deSCyndy Ishida   return Matches[1].drop_front(Matches[1].rfind('/') + 1).str() + "/" +
364c6043deSCyndy Ishida          Matches[3].str();
374c6043deSCyndy Ishida }
38*487720fcSCyndy Ishida 
isHeaderFile(StringRef Path)39*487720fcSCyndy Ishida bool isHeaderFile(StringRef Path) {
40*487720fcSCyndy Ishida   return StringSwitch<bool>(sys::path::extension(Path))
41*487720fcSCyndy Ishida       .Cases(".h", ".H", ".hh", ".hpp", ".hxx", true)
42*487720fcSCyndy Ishida       .Default(false);
43*487720fcSCyndy Ishida }
44*487720fcSCyndy Ishida 
enumerateFiles(FileManager & FM,StringRef Directory)45*487720fcSCyndy Ishida llvm::Expected<PathSeq> enumerateFiles(FileManager &FM, StringRef Directory) {
46*487720fcSCyndy Ishida   PathSeq Files;
47*487720fcSCyndy Ishida   std::error_code EC;
48*487720fcSCyndy Ishida   auto &FS = FM.getVirtualFileSystem();
49*487720fcSCyndy Ishida   for (llvm::vfs::recursive_directory_iterator i(FS, Directory, EC), ie;
50*487720fcSCyndy Ishida        i != ie; i.increment(EC)) {
51*487720fcSCyndy Ishida     if (EC)
52*487720fcSCyndy Ishida       return errorCodeToError(EC);
53*487720fcSCyndy Ishida 
54*487720fcSCyndy Ishida     // Skip files that do not exist. This usually happens for broken symlinks.
55*487720fcSCyndy Ishida     if (FS.status(i->path()) == std::errc::no_such_file_or_directory)
56*487720fcSCyndy Ishida       continue;
57*487720fcSCyndy Ishida 
58*487720fcSCyndy Ishida     StringRef Path = i->path();
59*487720fcSCyndy Ishida     if (isHeaderFile(Path))
60*487720fcSCyndy Ishida       Files.emplace_back(Path);
61*487720fcSCyndy Ishida   }
62*487720fcSCyndy Ishida 
63*487720fcSCyndy Ishida   return Files;
64*487720fcSCyndy Ishida }
65*487720fcSCyndy Ishida 
HeaderGlob(StringRef GlobString,Regex && Rule,HeaderType Type)66*487720fcSCyndy Ishida HeaderGlob::HeaderGlob(StringRef GlobString, Regex &&Rule, HeaderType Type)
67*487720fcSCyndy Ishida     : GlobString(GlobString), Rule(std::move(Rule)), Type(Type) {}
68*487720fcSCyndy Ishida 
match(const HeaderFile & Header)69*487720fcSCyndy Ishida bool HeaderGlob::match(const HeaderFile &Header) {
70*487720fcSCyndy Ishida   if (Header.getType() != Type)
71*487720fcSCyndy Ishida     return false;
72*487720fcSCyndy Ishida 
73*487720fcSCyndy Ishida   bool Match = Rule.match(Header.getPath());
74*487720fcSCyndy Ishida   if (Match)
75*487720fcSCyndy Ishida     FoundMatch = true;
76*487720fcSCyndy Ishida   return Match;
77*487720fcSCyndy Ishida }
78*487720fcSCyndy Ishida 
create(StringRef GlobString,HeaderType Type)79*487720fcSCyndy Ishida Expected<std::unique_ptr<HeaderGlob>> HeaderGlob::create(StringRef GlobString,
80*487720fcSCyndy Ishida                                                          HeaderType Type) {
81*487720fcSCyndy Ishida   auto Rule = MachO::createRegexFromGlob(GlobString);
82*487720fcSCyndy Ishida   if (!Rule)
83*487720fcSCyndy Ishida     return Rule.takeError();
84*487720fcSCyndy Ishida 
85*487720fcSCyndy Ishida   return std::make_unique<HeaderGlob>(GlobString, std::move(*Rule), Type);
86*487720fcSCyndy Ishida }
87*487720fcSCyndy Ishida 
884c6043deSCyndy Ishida } // namespace clang::installapi
89