Details | Last modification | View Log | RSS feed
| Rev | Author | Line No. | Line |
|---|---|---|---|
| 14 | pmbaty | 1 | //===--- HeaderMap.h - A file that acts like dir of symlinks ----*- C++ -*-===// |
| 2 | // |
||
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
||
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
||
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
||
| 6 | // |
||
| 7 | //===----------------------------------------------------------------------===// |
||
| 8 | // |
||
| 9 | // This file defines the HeaderMap interface. |
||
| 10 | // |
||
| 11 | //===----------------------------------------------------------------------===// |
||
| 12 | |||
| 13 | #ifndef LLVM_CLANG_LEX_HEADERMAP_H |
||
| 14 | #define LLVM_CLANG_LEX_HEADERMAP_H |
||
| 15 | |||
| 16 | #include "clang/Basic/FileManager.h" |
||
| 17 | #include "clang/Basic/LLVM.h" |
||
| 18 | #include "clang/Lex/HeaderMapTypes.h" |
||
| 19 | #include "llvm/ADT/StringMap.h" |
||
| 20 | #include "llvm/Support/Compiler.h" |
||
| 21 | #include "llvm/Support/MemoryBuffer.h" |
||
| 22 | #include <memory> |
||
| 23 | #include <optional> |
||
| 24 | |||
| 25 | namespace clang { |
||
| 26 | |||
| 27 | struct HMapBucket; |
||
| 28 | struct HMapHeader; |
||
| 29 | |||
| 30 | /// Implementation for \a HeaderMap that doesn't depend on \a FileManager. |
||
| 31 | class HeaderMapImpl { |
||
| 32 | std::unique_ptr<const llvm::MemoryBuffer> FileBuffer; |
||
| 33 | bool NeedsBSwap; |
||
| 34 | mutable llvm::StringMap<StringRef> ReverseMap; |
||
| 35 | |||
| 36 | public: |
||
| 37 | HeaderMapImpl(std::unique_ptr<const llvm::MemoryBuffer> File, bool NeedsBSwap) |
||
| 38 | : FileBuffer(std::move(File)), NeedsBSwap(NeedsBSwap) {} |
||
| 39 | |||
| 40 | // Check for a valid header and extract the byte swap. |
||
| 41 | static bool checkHeader(const llvm::MemoryBuffer &File, bool &NeedsByteSwap); |
||
| 42 | |||
| 43 | // Make a call for every Key in the map. |
||
| 44 | template <typename Func> void forEachKey(Func Callback) const { |
||
| 45 | const HMapHeader &Hdr = getHeader(); |
||
| 46 | unsigned NumBuckets = getEndianAdjustedWord(Hdr.NumBuckets); |
||
| 47 | |||
| 48 | for (unsigned Bucket = 0; Bucket < NumBuckets; ++Bucket) { |
||
| 49 | HMapBucket B = getBucket(Bucket); |
||
| 50 | if (B.Key != HMAP_EmptyBucketKey) |
||
| 51 | if (std::optional<StringRef> Key = getString(B.Key)) |
||
| 52 | Callback(*Key); |
||
| 53 | } |
||
| 54 | } |
||
| 55 | |||
| 56 | /// If the specified relative filename is located in this HeaderMap return |
||
| 57 | /// the filename it is mapped to, otherwise return an empty StringRef. |
||
| 58 | StringRef lookupFilename(StringRef Filename, |
||
| 59 | SmallVectorImpl<char> &DestPath) const; |
||
| 60 | |||
| 61 | /// Return the filename of the headermap. |
||
| 62 | StringRef getFileName() const; |
||
| 63 | |||
| 64 | /// Print the contents of this headermap to stderr. |
||
| 65 | void dump() const; |
||
| 66 | |||
| 67 | /// Return key for specifed path. |
||
| 68 | StringRef reverseLookupFilename(StringRef DestPath) const; |
||
| 69 | |||
| 70 | private: |
||
| 71 | unsigned getEndianAdjustedWord(unsigned X) const; |
||
| 72 | const HMapHeader &getHeader() const; |
||
| 73 | HMapBucket getBucket(unsigned BucketNo) const; |
||
| 74 | |||
| 75 | /// Look up the specified string in the string table. If the string index is |
||
| 76 | /// not valid, return std::nullopt. |
||
| 77 | std::optional<StringRef> getString(unsigned StrTabIdx) const; |
||
| 78 | }; |
||
| 79 | |||
| 80 | /// This class represents an Apple concept known as a 'header map'. To the |
||
| 81 | /// \#include file resolution process, it basically acts like a directory of |
||
| 82 | /// symlinks to files. Its advantages are that it is dense and more efficient |
||
| 83 | /// to create and process than a directory of symlinks. |
||
| 84 | class HeaderMap : private HeaderMapImpl { |
||
| 85 | HeaderMap(std::unique_ptr<const llvm::MemoryBuffer> File, bool BSwap) |
||
| 86 | : HeaderMapImpl(std::move(File), BSwap) {} |
||
| 87 | |||
| 88 | public: |
||
| 89 | /// This attempts to load the specified file as a header map. If it doesn't |
||
| 90 | /// look like a HeaderMap, it gives up and returns null. |
||
| 91 | static std::unique_ptr<HeaderMap> Create(const FileEntry *FE, |
||
| 92 | FileManager &FM); |
||
| 93 | |||
| 94 | using HeaderMapImpl::dump; |
||
| 95 | using HeaderMapImpl::forEachKey; |
||
| 96 | using HeaderMapImpl::getFileName; |
||
| 97 | using HeaderMapImpl::lookupFilename; |
||
| 98 | using HeaderMapImpl::reverseLookupFilename; |
||
| 99 | }; |
||
| 100 | |||
| 101 | } // end namespace clang. |
||
| 102 | |||
| 103 | #endif |