aboutsummaryrefslogtreecommitdiff
path: root/bolt/lib/Core/MCInstUtils.cpp
blob: f505bf73c64eb9c0b8c211d9978cbf740e4d6300 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
//===- bolt/Core/MCInstUtils.cpp ------------------------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//

#include "bolt/Core/MCInstUtils.h"
#include "bolt/Core/BinaryBasicBlock.h"
#include "bolt/Core/BinaryFunction.h"

#include <type_traits>

using namespace llvm;
using namespace llvm::bolt;

// It is assumed in a few places that BinaryBasicBlock stores its instructions
// in a contiguous vector.
using BasicBlockStorageIsVector =
    std::is_same<BinaryBasicBlock::const_iterator,
                 std::vector<MCInst>::const_iterator>;
static_assert(BasicBlockStorageIsVector::value);

MCInstReference MCInstReference::get(const MCInst &Inst,
                                     const BinaryFunction &BF) {
  if (BF.hasCFG()) {
    for (BinaryBasicBlock &BB : BF) {
      for (MCInst &MI : BB)
        if (&MI == &Inst)
          return MCInstReference(BB, Inst);
    }
    llvm_unreachable("Inst is not contained in BF");
  }

  for (auto I = BF.instrs().begin(), E = BF.instrs().end(); I != E; ++I) {
    if (&I->second == &Inst)
      return MCInstReference(BF, I);
  }
  llvm_unreachable("Inst is not contained in BF");
}

uint64_t MCInstReference::computeAddress(const MCCodeEmitter *Emitter) const {
  assert(!empty() && "Taking instruction address by empty reference");

  const BinaryContext &BC = getFunction()->getBinaryContext();
  if (auto *Ref = tryGetRefInBB()) {
    const uint64_t AddressOfBB =
        getFunction()->getAddress() + Ref->BB->getOffset();
    const MCInst *FirstInstInBB = &*Ref->BB->begin();
    const MCInst *ThisInst = &getMCInst();

    // Usage of plain 'const MCInst *' as iterators assumes the instructions
    // are stored in a vector, see BasicBlockStorageIsVector.
    const uint64_t OffsetInBB =
        BC.computeCodeSize(FirstInstInBB, ThisInst, Emitter);

    return AddressOfBB + OffsetInBB;
  }

  auto &Ref = getRefInBF();
  const uint64_t OffsetInBF = Ref.It->first;

  return getFunction()->getAddress() + OffsetInBF;
}

raw_ostream &MCInstReference::print(raw_ostream &OS) const {
  if (const RefInBB *Ref = tryGetRefInBB()) {
    OS << "MCInstBBRef<";
    if (Ref->BB == nullptr)
      OS << "BB:(null)";
    else
      OS << "BB:" << Ref->BB->getName() << ":" << Ref->Index;
    OS << ">";
    return OS;
  }

  const RefInBF &Ref = getRefInBF();
  OS << "MCInstBFRef<";
  if (Ref.BF == nullptr)
    OS << "BF:(null)";
  else
    OS << "BF:" << Ref.BF->getPrintName() << ":" << Ref.It->first;
  OS << ">";
  return OS;
}