123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- #pragma once
- #ifdef __GNUC__
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wunused-parameter"
- #endif
- //===- LineTable.h ----------------------------------------------*- 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
- //
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_DEBUGINFO_GSYM_LINETABLE_H
- #define LLVM_DEBUGINFO_GSYM_LINETABLE_H
- #include "llvm/DebugInfo/GSYM/LineEntry.h"
- #include "llvm/Support/Error.h"
- #include <cstdint>
- #include <vector>
- namespace llvm {
- namespace gsym {
- struct FunctionInfo;
- class FileWriter;
- /// LineTable class contains deserialized versions of line tables for each
- /// function's address ranges.
- ///
- /// When saved to disk, the line table is encoded using a modified version of
- /// the DWARF line tables that only tracks address to source file and line.
- ///
- /// ENCODING
- ///
- /// The line table starts with a small prolog that contains the following
- /// values:
- ///
- /// ENCODING NAME DESCRIPTION
- /// ======== =========== ====================================================
- /// SLEB MinDelta The min line delta for special opcodes that advance
- /// the address and line number.
- /// SLEB MaxDelta The max line delta for single byte opcodes that
- /// advance the address and line number.
- /// ULEB FirstLine The value of the first source line number to
- /// initialize the LineEntry with.
- ///
- /// Once these prolog items are read, we initialize a LineEntry struct with
- /// the start address of the function from the FunctionInfo's address range,
- /// a default file index of 1, and the line number set to "FirstLine" from
- /// the prolog above:
- ///
- /// LineEntry Row(BaseAddr, 1, FirstLine);
- ///
- /// The line table state machine is now initialized and ready to be parsed.
- /// The stream that follows this encodes the line entries in a compact
- /// form. Some opcodes cause "Row" to be modified and some opcodes may also
- /// push "Row" onto the end of the "LineTable.Lines" vector. The end result
- /// is a vector of LineEntry structs that is sorted in ascending address
- /// order.
- ///
- /// NORMAL OPCODES
- ///
- /// The opcodes 0 through 3 are normal in opcodes. Their encoding and
- /// descriptions are listed below:
- ///
- /// ENCODING ENUMERATION VALUE DESCRIPTION
- /// ======== ================ ===== ========================================
- /// LTOC_EndSequence 0x00 Parsing is done.
- /// ULEB LTOC_SetFile 0x01 Row.File = ULEB
- /// ULEB LTOC_AdvancePC 0x02 Row.Addr += ULEB, push "Row".
- /// SLEB LTOC_AdvanceLine 0x03 Row.Line += SLEB
- /// LTOC_FirstSpecial 0x04 First special opcode (see SPECIAL
- /// OPCODES below).
- ///
- /// SPECIAL OPCODES
- ///
- /// Opcodes LTOC_FirstSpecial through 255 are special opcodes that always
- /// increment both the Row.Addr and Row.Line and push "Row" onto the
- /// LineEntry.Lines array. They do this by using some of the bits to
- /// increment/decrement the source line number, and some of the bits to
- /// increment the address. Line numbers can go up or down when making line
- /// tables, where addresses always only increase since line tables are sorted
- /// by address.
- ///
- /// In order to calculate the amount to increment the line and address for
- /// these special opcodes, we calculate the number of values reserved for the
- /// line increment/decrement using the "MinDelta" and "MaxDelta" from the
- /// prolog:
- ///
- /// const int64_t LineRange = MaxDelta - MinDelta + 1;
- ///
- /// Then we can adjust the opcode to not include any of the normal opcodes:
- ///
- /// const uint8_t AdjustedOp = Opcode - LTOC_FirstSpecial;
- ///
- /// And we can calculate the line offset, and address offset:
- ///
- /// const int64_t LineDelta = MinDelta + (AdjustedOp % LineRange);
- /// const uint64_t AddrDelta = (AdjustedOp / LineRange);
- ///
- /// And use these to modify our "Row":
- ///
- /// Row.Line += LineDelta;
- /// Row.Addr += AddrDelta;
- ///
- /// And push a row onto the line table:
- ///
- /// Lines.push_back(Row);
- ///
- /// This is verify similar to the way that DWARF encodes its line tables. The
- /// only difference is the DWARF line tables have more normal opcodes and the
- /// "Row" contains more members, like source column number, bools for end of
- /// prologue, beginnging of epilogue, is statement and many others. There are
- /// also more complex rules that happen for the extra normal opcodes. By
- /// leaving these extra opcodes out, we leave more bits for the special
- /// opcodes that allows us to encode line tables in fewer bytes than standard
- /// DWARF encodings.
- ///
- /// Opcodes that will push "Row" onto the LineEntry.Lines include the
- /// LTOC_AdvancePC opcode and all special opcodes. All other opcodes
- /// only modify the current "Row", or cause the line table to end.
- class LineTable {
- typedef std::vector<gsym::LineEntry> Collection;
- Collection Lines; ///< All line entries in the line table.
- public:
- /// Lookup a single address within a line table's data.
- ///
- /// Clients have the option to decode an entire line table using
- /// LineTable::decode() or just find a single matching entry using this
- /// function. The benefit of using this function is that parsed LineEntry
- /// objects that do not match will not be stored in an array. This will avoid
- /// memory allocation costs and parsing can stop once a match has been found.
- ///
- /// \param Data The binary stream to read the data from. This object must
- /// have the data for the LineTable object starting at offset zero. The data
- /// can contain more data than needed.
- ///
- /// \param BaseAddr The base address to use when decoding the line table.
- /// This will be the FunctionInfo's start address and will be used to
- /// initialize the line table row prior to parsing any opcodes.
- ///
- /// \returns An LineEntry object if a match is found, error otherwise.
- static Expected<LineEntry> lookup(DataExtractor &Data, uint64_t BaseAddr,
- uint64_t Addr);
- /// Decode an LineTable object from a binary data stream.
- ///
- /// \param Data The binary stream to read the data from. This object must
- /// have the data for the LineTable object starting at offset zero. The data
- /// can contain more data than needed.
- ///
- /// \param BaseAddr The base address to use when decoding the line table.
- /// This will be the FunctionInfo's start address and will be used to
- /// initialize the line table row prior to parsing any opcodes.
- ///
- /// \returns An LineTable or an error describing the issue that was
- /// encountered during decoding.
- static llvm::Expected<LineTable> decode(DataExtractor &Data,
- uint64_t BaseAddr);
- /// Encode this LineTable object into FileWriter stream.
- ///
- /// \param O The binary stream to write the data to at the current file
- /// position.
- ///
- /// \param BaseAddr The base address to use when decoding the line table.
- /// This will be the FunctionInfo's start address.
- ///
- /// \returns An error object that indicates success or failure or the
- /// encoding process.
- llvm::Error encode(FileWriter &O, uint64_t BaseAddr) const;
- bool empty() const { return Lines.empty(); }
- void clear() { Lines.clear(); }
- /// Return the first line entry if the line table isn't empty.
- ///
- /// \returns An optional line entry with the first line entry if the line
- /// table isn't empty, or std::nullopt if the line table is emtpy.
- std::optional<LineEntry> first() const {
- if (Lines.empty())
- return std::nullopt;
- return Lines.front();
- }
- /// Return the last line entry if the line table isn't empty.
- ///
- /// \returns An optional line entry with the last line entry if the line
- /// table isn't empty, or std::nullopt if the line table is emtpy.
- std::optional<LineEntry> last() const {
- if (Lines.empty())
- return std::nullopt;
- return Lines.back();
- }
- void push(const LineEntry &LE) {
- Lines.push_back(LE);
- }
- size_t isValid() const {
- return !Lines.empty();
- }
- size_t size() const {
- return Lines.size();
- }
- LineEntry &get(size_t i) {
- assert(i < Lines.size());
- return Lines[i];
- }
- const LineEntry &get(size_t i) const {
- assert(i < Lines.size());
- return Lines[i];
- }
- LineEntry &operator[](size_t i) {
- return get(i);
- }
- const LineEntry &operator[](size_t i) const {
- return get(i);
- }
- bool operator==(const LineTable &RHS) const {
- return Lines == RHS.Lines;
- }
- bool operator!=(const LineTable &RHS) const {
- return Lines != RHS.Lines;
- }
- bool operator<(const LineTable &RHS) const {
- const auto LHSSize = Lines.size();
- const auto RHSSize = RHS.Lines.size();
- if (LHSSize == RHSSize)
- return Lines < RHS.Lines;
- return LHSSize < RHSSize;
- }
- Collection::const_iterator begin() const { return Lines.begin(); }
- Collection::const_iterator end() const { return Lines.end(); }
- };
- raw_ostream &operator<<(raw_ostream &OS, const gsym::LineTable <);
- } // namespace gsym
- } // namespace llvm
- #endif // LLVM_DEBUGINFO_GSYM_LINETABLE_H
- #ifdef __GNUC__
- #pragma GCC diagnostic pop
- #endif
|