//===- 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