-
Notifications
You must be signed in to change notification settings - Fork 10.5k
/
Copy pathSILDebugInfoExpression.cpp
87 lines (77 loc) · 3.16 KB
/
SILDebugInfoExpression.cpp
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
//===-------- SILDebugInfoExpression.cpp - DIExpression for SIL -----------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
///
/// \file
/// This file contains the table used by SILDIExprInfo to map from
/// SILDIExprOperator to supplement information like the operator string.
///
//===----------------------------------------------------------------------===//
#include "swift/Basic/Assertions.h"
#include "swift/SIL/SILDebugInfoExpression.h"
#include <unordered_map>
using namespace swift;
namespace std {
// This is, unfortunately, a workaround for an ancient bug in libstdc++:
// https://gcc.gnu.org/bugzilla/show_bug.cgi?id=60970
// Which prevented std::hash<T> users from having drop-in support for
// enum class types.
template <> struct hash<SILDIExprOperator> {
size_t operator()(SILDIExprOperator V) const noexcept {
return std::hash<unsigned>{}(static_cast<unsigned>(V));
}
};
} // end namespace std
const SILDIExprInfo *SILDIExprInfo::get(SILDIExprOperator Op) {
static const std::unordered_map<SILDIExprOperator, SILDIExprInfo> Infos = {
{SILDIExprOperator::Fragment,
{"op_fragment", {SILDIExprElement::DeclKind}}},
{SILDIExprOperator::TupleFragment,
{"op_tuple_fragment",
{SILDIExprElement::TypeKind, SILDIExprElement::ConstIntKind}}},
{SILDIExprOperator::Dereference, {"op_deref", {}}},
{SILDIExprOperator::Plus, {"op_plus", {}}},
{SILDIExprOperator::Minus, {"op_minus", {}}},
{SILDIExprOperator::ConstUInt,
{"op_constu", {SILDIExprElement::ConstIntKind}}},
{SILDIExprOperator::ConstSInt,
{"op_consts", {SILDIExprElement::ConstIntKind}}}};
return Infos.count(Op) ? &Infos.at(Op) : nullptr;
}
void SILDebugInfoExpression::op_iterator::increment() {
if (Remain.empty()) {
// Effectively making this an end iterator
Current = {};
return;
}
const SILDIExprElement &NextHead = Remain[0];
SILDIExprOperator Op = NextHead.getAsOperator();
if (const auto *ExprInfo = SILDIExprInfo::get(Op)) {
auto NewSlice = Remain.take_front(ExprInfo->OperandKinds.size() + 1);
Current = SILDIExprOperand(NewSlice.data(), NewSlice.size());
if (Remain.size() >= Current.size())
Remain = Remain.drop_front(Current.size());
}
}
SILDebugInfoExpression SILDebugInfoExpression::createFragment(VarDecl *Field) {
assert(Field);
return SILDebugInfoExpression(
{SILDIExprElement::createOperator(SILDIExprOperator::Fragment),
SILDIExprElement::createDecl(Field)});
}
SILDebugInfoExpression
SILDebugInfoExpression::createTupleFragment(TupleType *TypePtr, unsigned Idx) {
assert(TypePtr);
return SILDebugInfoExpression(
{SILDIExprElement::createOperator(SILDIExprOperator::TupleFragment),
SILDIExprElement::createType(TypePtr),
SILDIExprElement::createConstInt(Idx)});
}