Files
swift-mirror/include/swift/AST/ASTNode.h
2025-07-29 18:37:48 +01:00

156 lines
4.6 KiB
C++

//===--- ASTNode.h - Swift Language ASTs ------------------------*- C++ -*-===//
//
// 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
//
//===----------------------------------------------------------------------===//
//
// This file defines the ASTNode, which is a union of Stmt, Expr, Decl,
// Pattern, TypeRepr, StmtCondition, and CaseLabelItem.
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_AST_AST_NODE_H
#define SWIFT_AST_AST_NODE_H
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/PointerUnion.h"
#include "swift/Basic/Debug.h"
#include "swift/Basic/SourceManager.h"
#include "swift/AST/TypeAlignments.h"
namespace llvm {
class raw_ostream;
}
namespace swift {
class Expr;
class Stmt;
class Decl;
class Pattern;
class TypeRepr;
class DeclContext;
class SourceLoc;
class SourceRange;
class ASTWalker;
class StmtConditionElement;
class CaseLabelItem;
enum class ExprKind : uint8_t;
enum class DeclKind : uint8_t;
enum class PatternKind : uint8_t;
enum class StmtKind;
namespace detail {
using ASTNodeBase =
llvm::PointerUnion<Expr *, Stmt *, Decl *, Pattern *, TypeRepr *,
StmtConditionElement *, CaseLabelItem *>;
} // end namespace detail
struct ASTNode : public detail::ASTNodeBase {
using Base = detail::ASTNodeBase;
// Inherit the constructors from PointerUnion.
using PointerUnion::PointerUnion;
// These are needed for lldb.
ASTNode(Expr *E) : PointerUnion(E) {}
ASTNode(Stmt *S) : PointerUnion(S) {}
ASTNode(Decl *D) : PointerUnion(D) {}
ASTNode(Pattern *P) : PointerUnion(P) {}
ASTNode(TypeRepr *T) : PointerUnion(T) {}
ASTNode(StmtConditionElement *S) : PointerUnion(S) {}
ASTNode(CaseLabelItem *C) : PointerUnion(C) {}
SourceRange getSourceRange() const;
/// Return the location of the start of the statement.
SourceLoc getStartLoc() const;
/// Return the location of the end of the statement.
SourceLoc getEndLoc() const;
void walk(ASTWalker &Walker);
void walk(ASTWalker &&walker) { walk(walker); }
/// get the underlying entity as a decl context if it is one,
/// otherwise, return nullptr;
DeclContext *getAsDeclContext() const;
/// Provides some utilities to decide detailed node kind.
#define FUNC(T) bool is##T(T##Kind Kind) const;
FUNC(Stmt)
FUNC(Expr)
FUNC(Decl)
FUNC(Pattern)
#undef FUNC
static inline ASTNode getFromOpaqueValue(void *ptr) {
ASTNode result;
result.Val = decltype(result.Val)::getFromOpaqueValue(ptr);
return result;
}
SWIFT_DEBUG_DUMP;
void dump(llvm::raw_ostream &OS, unsigned Indent = 0) const;
/// Whether the AST node is implicit.
bool isImplicit() const;
friend llvm::hash_code hash_value(ASTNode N) {
return llvm::hash_value(N.getOpaqueValue());
}
};
/// Find the outermost range that \p range was originally generated from.
/// Returns an invalid source range if \p range wasn't generated from a macro.
SourceRange getUnexpandedMacroRange(const SourceManager &SM,
SourceRange range);
} // namespace swift
namespace llvm {
using swift::ASTNode;
/// `isa`, `dyn_cast`, `cast` for `ASTNode`.
template <typename To>
struct CastInfo<To, ASTNode> : public CastInfo<To, ASTNode::Base> {};
template <typename To>
struct CastInfo<To, const ASTNode>
: public CastInfo<To, const ASTNode::Base> {};
template <> struct DenseMapInfo<ASTNode> {
static inline ASTNode getEmptyKey() {
return DenseMapInfo<swift::Expr *>::getEmptyKey();
}
static inline ASTNode getTombstoneKey() {
return DenseMapInfo<swift::Expr *>::getTombstoneKey();
}
static unsigned getHashValue(const ASTNode Val) {
return DenseMapInfo<void *>::getHashValue(Val.getOpaqueValue());
}
static bool isEqual(const ASTNode LHS, const ASTNode RHS) {
return LHS.getOpaqueValue() == RHS.getOpaqueValue();
}
};
// A ASTNode is "pointer like".
template <>
struct PointerLikeTypeTraits<ASTNode> {
public:
static inline void *getAsVoidPointer(ASTNode N) {
return (void *)N.getOpaqueValue();
}
static inline ASTNode getFromVoidPointer(void *P) {
return ASTNode::getFromOpaqueValue(P);
}
enum { NumLowBitsAvailable = swift::TypeAlignInBits };
};
}
#endif // LLVM_SWIFT_AST_AST_NODE_H