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/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)});
}
|