123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //==- include/llvm/CodeGen/AccelTable.h - Accelerator Tables -----*- C++ -*-==//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- /// \file
- /// This file contains support for writing accelerator tables.
- ///
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_CODEGEN_ACCELTABLE_H
- #define LLVM_CODEGEN_ACCELTABLE_H
- #include "llvm/ADT/ArrayRef.h"
- #include "llvm/ADT/SmallVector.h"
- #include "llvm/ADT/StringMap.h"
- #include "llvm/ADT/StringRef.h"
- #include "llvm/BinaryFormat/Dwarf.h"
- #include "llvm/CodeGen/DIE.h"
- #include "llvm/CodeGen/DwarfStringPoolEntry.h"
- #include "llvm/MC/MCSymbol.h"
- #include "llvm/Support/Allocator.h"
- #include "llvm/Support/DJB.h"
- #include "llvm/Support/Debug.h"
- #include "llvm/Support/Format.h"
- #include "llvm/Support/raw_ostream.h"
- #include <cstddef>
- #include <cstdint>
- #include <vector>
- /// \file
- /// The DWARF and Apple accelerator tables are an indirect hash table optimized
- /// for null lookup rather than access to known data. The Apple accelerator
- /// tables are a precursor of the newer DWARF v5 accelerator tables. Both
- /// formats share common design ideas.
- ///
- /// The Apple accelerator table are output into an on-disk format that looks
- /// like this:
- ///
- /// .------------------.
- /// | HEADER |
- /// |------------------|
- /// | BUCKETS |
- /// |------------------|
- /// | HASHES |
- /// |------------------|
- /// | OFFSETS |
- /// |------------------|
- /// | DATA |
- /// `------------------'
- ///
- /// The header contains a magic number, version, type of hash function,
- /// the number of buckets, total number of hashes, and room for a special struct
- /// of data and the length of that struct.
- ///
- /// The buckets contain an index (e.g. 6) into the hashes array. The hashes
- /// section contains all of the 32-bit hash values in contiguous memory, and the
- /// offsets contain the offset into the data area for the particular hash.
- ///
- /// For a lookup example, we could hash a function name and take it modulo the
- /// number of buckets giving us our bucket. From there we take the bucket value
- /// as an index into the hashes table and look at each successive hash as long
- /// as the hash value is still the same modulo result (bucket value) as earlier.
- /// If we have a match we look at that same entry in the offsets table and grab
- /// the offset in the data for our final match.
- ///
- /// The DWARF v5 accelerator table consists of zero or more name indices that
- /// are output into an on-disk format that looks like this:
- ///
- /// .------------------.
- /// | HEADER |
- /// |------------------|
- /// | CU LIST |
- /// |------------------|
- /// | LOCAL TU LIST |
- /// |------------------|
- /// | FOREIGN TU LIST |
- /// |------------------|
- /// | HASH TABLE |
- /// |------------------|
- /// | NAME TABLE |
- /// |------------------|
- /// | ABBREV TABLE |
- /// |------------------|
- /// | ENTRY POOL |
- /// `------------------'
- ///
- /// For the full documentation please refer to the DWARF 5 standard.
- ///
- ///
- /// This file defines the class template AccelTable, which is represents an
- /// abstract view of an Accelerator table, without any notion of an on-disk
- /// layout. This class is parameterized by an entry type, which should derive
- /// from AccelTableData. This is the type of individual entries in the table,
- /// and it should store the data necessary to emit them. AppleAccelTableData is
- /// the base class for Apple Accelerator Table entries, which have a uniform
- /// structure based on a sequence of Atoms. There are different sub-classes
- /// derived from AppleAccelTable, which differ in the set of Atoms and how they
- /// obtain their values.
- ///
- /// An Apple Accelerator Table can be serialized by calling emitAppleAccelTable
- /// function.
- namespace llvm {
- class AsmPrinter;
- class DwarfCompileUnit;
- class DwarfDebug;
- /// Interface which the different types of accelerator table data have to
- /// conform. It serves as a base class for different values of the template
- /// argument of the AccelTable class template.
- class AccelTableData {
- public:
- virtual ~AccelTableData() = default;
- bool operator<(const AccelTableData &Other) const {
- return order() < Other.order();
- }
- // Subclasses should implement:
- // static uint32_t hash(StringRef Name);
- #ifndef NDEBUG
- virtual void print(raw_ostream &OS) const = 0;
- #endif
- protected:
- virtual uint64_t order() const = 0;
- };
- /// A base class holding non-template-dependant functionality of the AccelTable
- /// class. Clients should not use this class directly but rather instantiate
- /// AccelTable with a type derived from AccelTableData.
- class AccelTableBase {
- public:
- using HashFn = uint32_t(StringRef);
- /// Represents a group of entries with identical name (and hence, hash value).
- struct HashData {
- DwarfStringPoolEntryRef Name;
- uint32_t HashValue;
- std::vector<AccelTableData *> Values;
- MCSymbol *Sym;
- HashData(DwarfStringPoolEntryRef Name, HashFn *Hash)
- : Name(Name), HashValue(Hash(Name.getString())) {}
- #ifndef NDEBUG
- void print(raw_ostream &OS) const;
- void dump() const { print(dbgs()); }
- #endif
- };
- using HashList = std::vector<HashData *>;
- using BucketList = std::vector<HashList>;
- protected:
- /// Allocator for HashData and Values.
- BumpPtrAllocator Allocator;
- using StringEntries = StringMap<HashData, BumpPtrAllocator &>;
- StringEntries Entries;
- HashFn *Hash;
- uint32_t BucketCount;
- uint32_t UniqueHashCount;
- HashList Hashes;
- BucketList Buckets;
- void computeBucketCount();
- AccelTableBase(HashFn *Hash) : Entries(Allocator), Hash(Hash) {}
- public:
- void finalize(AsmPrinter *Asm, StringRef Prefix);
- ArrayRef<HashList> getBuckets() const { return Buckets; }
- uint32_t getBucketCount() const { return BucketCount; }
- uint32_t getUniqueHashCount() const { return UniqueHashCount; }
- uint32_t getUniqueNameCount() const { return Entries.size(); }
- #ifndef NDEBUG
- void print(raw_ostream &OS) const;
- void dump() const { print(dbgs()); }
- #endif
- AccelTableBase(const AccelTableBase &) = delete;
- void operator=(const AccelTableBase &) = delete;
- };
- /// This class holds an abstract representation of an Accelerator Table,
- /// consisting of a sequence of buckets, each bucket containint a sequence of
- /// HashData entries. The class is parameterized by the type of entries it
- /// holds. The type template parameter also defines the hash function to use for
- /// hashing names.
- template <typename DataT> class AccelTable : public AccelTableBase {
- public:
- AccelTable() : AccelTableBase(DataT::hash) {}
- template <typename... Types>
- void addName(DwarfStringPoolEntryRef Name, Types &&... Args);
- };
- template <typename AccelTableDataT>
- template <typename... Types>
- void AccelTable<AccelTableDataT>::addName(DwarfStringPoolEntryRef Name,
- Types &&... Args) {
- assert(Buckets.empty() && "Already finalized!");
- // If the string is in the list already then add this die to the list
- // otherwise add a new one.
- auto Iter = Entries.try_emplace(Name.getString(), Name, Hash).first;
- assert(Iter->second.Name == Name);
- Iter->second.Values.push_back(
- new (Allocator) AccelTableDataT(std::forward<Types>(Args)...));
- }
- /// A base class for different implementations of Data classes for Apple
- /// Accelerator Tables. The columns in the table are defined by the static Atoms
- /// variable defined on the subclasses.
- class AppleAccelTableData : public AccelTableData {
- public:
- /// An Atom defines the form of the data in an Apple accelerator table.
- /// Conceptually it is a column in the accelerator consisting of a type and a
- /// specification of the form of its data.
- struct Atom {
- /// Atom Type.
- const uint16_t Type;
- /// DWARF Form.
- const uint16_t Form;
- constexpr Atom(uint16_t Type, uint16_t Form) : Type(Type), Form(Form) {}
- #ifndef NDEBUG
- void print(raw_ostream &OS) const;
- void dump() const { print(dbgs()); }
- #endif
- };
- // Subclasses should define:
- // static constexpr Atom Atoms[];
- virtual void emit(AsmPrinter *Asm) const = 0;
- static uint32_t hash(StringRef Buffer) { return djbHash(Buffer); }
- };
- /// The Data class implementation for DWARF v5 accelerator table. Unlike the
- /// Apple Data classes, this class is just a DIE wrapper, and does not know to
- /// serialize itself. The complete serialization logic is in the
- /// emitDWARF5AccelTable function.
- class DWARF5AccelTableData : public AccelTableData {
- public:
- static uint32_t hash(StringRef Name) { return caseFoldingDjbHash(Name); }
- DWARF5AccelTableData(const DIE &Die) : Die(Die) {}
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- const DIE &getDie() const { return Die; }
- uint64_t getDieOffset() const { return Die.getOffset(); }
- unsigned getDieTag() const { return Die.getTag(); }
- protected:
- const DIE &Die;
- uint64_t order() const override { return Die.getOffset(); }
- };
- class DWARF5AccelTableStaticData : public AccelTableData {
- public:
- static uint32_t hash(StringRef Name) { return caseFoldingDjbHash(Name); }
- DWARF5AccelTableStaticData(uint64_t DieOffset, unsigned DieTag,
- unsigned CUIndex)
- : DieOffset(DieOffset), DieTag(DieTag), CUIndex(CUIndex) {}
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- uint64_t getDieOffset() const { return DieOffset; }
- unsigned getDieTag() const { return DieTag; }
- unsigned getCUIndex() const { return CUIndex; }
- protected:
- uint64_t DieOffset;
- unsigned DieTag;
- unsigned CUIndex;
- uint64_t order() const override { return DieOffset; }
- };
- void emitAppleAccelTableImpl(AsmPrinter *Asm, AccelTableBase &Contents,
- StringRef Prefix, const MCSymbol *SecBegin,
- ArrayRef<AppleAccelTableData::Atom> Atoms);
- /// Emit an Apple Accelerator Table consisting of entries in the specified
- /// AccelTable. The DataT template parameter should be derived from
- /// AppleAccelTableData.
- template <typename DataT>
- void emitAppleAccelTable(AsmPrinter *Asm, AccelTable<DataT> &Contents,
- StringRef Prefix, const MCSymbol *SecBegin) {
- static_assert(std::is_convertible<DataT *, AppleAccelTableData *>::value, "");
- emitAppleAccelTableImpl(Asm, Contents, Prefix, SecBegin, DataT::Atoms);
- }
- void emitDWARF5AccelTable(AsmPrinter *Asm,
- AccelTable<DWARF5AccelTableData> &Contents,
- const DwarfDebug &DD,
- ArrayRef<std::unique_ptr<DwarfCompileUnit>> CUs);
- void emitDWARF5AccelTable(
- AsmPrinter *Asm, AccelTable<DWARF5AccelTableStaticData> &Contents,
- ArrayRef<MCSymbol *> CUs,
- llvm::function_ref<unsigned(const DWARF5AccelTableStaticData &)>
- getCUIndexForEntry);
- /// Accelerator table data implementation for simple Apple accelerator tables
- /// with just a DIE reference.
- class AppleAccelTableOffsetData : public AppleAccelTableData {
- public:
- AppleAccelTableOffsetData(const DIE &D) : Die(D) {}
- void emit(AsmPrinter *Asm) const override;
- static constexpr Atom Atoms[] = {
- Atom(dwarf::DW_ATOM_die_offset, dwarf::DW_FORM_data4)};
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- protected:
- uint64_t order() const override { return Die.getOffset(); }
- const DIE &Die;
- };
- /// Accelerator table data implementation for Apple type accelerator tables.
- class AppleAccelTableTypeData : public AppleAccelTableOffsetData {
- public:
- AppleAccelTableTypeData(const DIE &D) : AppleAccelTableOffsetData(D) {}
- void emit(AsmPrinter *Asm) const override;
- static constexpr Atom Atoms[] = {
- Atom(dwarf::DW_ATOM_die_offset, dwarf::DW_FORM_data4),
- Atom(dwarf::DW_ATOM_die_tag, dwarf::DW_FORM_data2),
- Atom(dwarf::DW_ATOM_type_flags, dwarf::DW_FORM_data1)};
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- };
- /// Accelerator table data implementation for simple Apple accelerator tables
- /// with a DIE offset but no actual DIE pointer.
- class AppleAccelTableStaticOffsetData : public AppleAccelTableData {
- public:
- AppleAccelTableStaticOffsetData(uint32_t Offset) : Offset(Offset) {}
- void emit(AsmPrinter *Asm) const override;
- static constexpr Atom Atoms[] = {
- Atom(dwarf::DW_ATOM_die_offset, dwarf::DW_FORM_data4)};
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- protected:
- uint64_t order() const override { return Offset; }
- uint32_t Offset;
- };
- /// Accelerator table data implementation for type accelerator tables with
- /// a DIE offset but no actual DIE pointer.
- class AppleAccelTableStaticTypeData : public AppleAccelTableStaticOffsetData {
- public:
- AppleAccelTableStaticTypeData(uint32_t Offset, uint16_t Tag,
- bool ObjCClassIsImplementation,
- uint32_t QualifiedNameHash)
- : AppleAccelTableStaticOffsetData(Offset),
- QualifiedNameHash(QualifiedNameHash), Tag(Tag),
- ObjCClassIsImplementation(ObjCClassIsImplementation) {}
- void emit(AsmPrinter *Asm) const override;
- static constexpr Atom Atoms[] = {
- Atom(dwarf::DW_ATOM_die_offset, dwarf::DW_FORM_data4),
- Atom(dwarf::DW_ATOM_die_tag, dwarf::DW_FORM_data2),
- Atom(5, dwarf::DW_FORM_data1), Atom(6, dwarf::DW_FORM_data4)};
- #ifndef NDEBUG
- void print(raw_ostream &OS) const override;
- #endif
- protected:
- uint64_t order() const override { return Offset; }
- uint32_t QualifiedNameHash;
- uint16_t Tag;
- bool ObjCClassIsImplementation;
- };
- } // end namespace llvm
- #endif // LLVM_CODEGEN_ACCELTABLE_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|