2012-12-13 21:59:55 +08:00
|
|
|
//===--- ASTDumper.cpp - Dumping implementation for ASTs ------------------===//
|
2007-08-09 06:51:59 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-30 03:59:25 +08:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-08-09 06:51:59 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2012-12-13 21:59:55 +08:00
|
|
|
// This file implements the AST dump methods, which dump out the
|
2007-08-09 06:51:59 +08:00
|
|
|
// AST in a form that exposes type details and other fields.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-07-05 01:04:04 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2013-01-08 01:53:08 +08:00
|
|
|
#include "clang/AST/Attr.h"
|
2013-01-14 22:07:11 +08:00
|
|
|
#include "clang/AST/CommentVisitor.h"
|
2009-02-04 03:21:40 +08:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2012-12-20 10:09:13 +08:00
|
|
|
#include "clang/AST/DeclVisitor.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/AST/StmtVisitor.h"
|
2012-12-20 10:09:13 +08:00
|
|
|
#include "clang/Basic/Module.h"
|
2007-08-30 14:17:34 +08:00
|
|
|
#include "clang/Basic/SourceManager.h"
|
2009-12-03 17:13:13 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2007-08-09 06:51:59 +08:00
|
|
|
using namespace clang;
|
2013-01-14 22:07:11 +08:00
|
|
|
using namespace clang::comments;
|
2007-08-09 06:51:59 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
2012-12-13 21:59:55 +08:00
|
|
|
// ASTDumper Visitor
|
2007-08-09 06:51:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
namespace {
|
2013-01-26 09:31:20 +08:00
|
|
|
// Colors used for various parts of the AST dump
|
|
|
|
|
|
|
|
struct TerminalColor {
|
|
|
|
raw_ostream::Colors Color;
|
|
|
|
bool Bold;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Decl kind names (VarDecl, FunctionDecl, etc)
|
|
|
|
static const TerminalColor DeclKindNameColor = { raw_ostream::GREEN, true };
|
|
|
|
// Attr names (CleanupAttr, GuardedByAttr, etc)
|
|
|
|
static const TerminalColor AttrColor = { raw_ostream::BLUE, true };
|
|
|
|
// Statement names (DeclStmt, ImplicitCastExpr, etc)
|
|
|
|
static const TerminalColor StmtColor = { raw_ostream::MAGENTA, true };
|
|
|
|
// Comment names (FullComment, ParagraphComment, TextComment, etc)
|
|
|
|
static const TerminalColor CommentColor = { raw_ostream::YELLOW, true };
|
|
|
|
|
|
|
|
// Type names (int, float, etc, plus user defined types)
|
|
|
|
static const TerminalColor TypeColor = { raw_ostream::GREEN, false };
|
|
|
|
|
|
|
|
// Pointer address
|
|
|
|
static const TerminalColor AddressColor = { raw_ostream::YELLOW, false };
|
|
|
|
// Source locations
|
|
|
|
static const TerminalColor LocationColor = { raw_ostream::YELLOW, false };
|
|
|
|
|
|
|
|
// lvalue/xvalue
|
|
|
|
static const TerminalColor ValueKindColor = { raw_ostream::CYAN, false };
|
|
|
|
// bitfield/objcproperty/objcsubscript/vectorcomponent
|
|
|
|
static const TerminalColor ObjectKindColor = { raw_ostream::CYAN, false };
|
|
|
|
|
|
|
|
// Null statements
|
|
|
|
static const TerminalColor NullColor = { raw_ostream::BLUE, false };
|
|
|
|
|
|
|
|
// CastKind from CastExpr's
|
|
|
|
static const TerminalColor CastColor = { raw_ostream::RED, false };
|
|
|
|
|
|
|
|
// Value of the statement
|
|
|
|
static const TerminalColor ValueColor = { raw_ostream::CYAN, true };
|
|
|
|
// Decl names
|
|
|
|
static const TerminalColor DeclNameColor = { raw_ostream::CYAN, true };
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
// Indents ( `, -. | )
|
|
|
|
static const TerminalColor IndentColor = { raw_ostream::BLUE, false };
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
class ASTDumper
|
2013-02-01 20:35:51 +08:00
|
|
|
: public ConstDeclVisitor<ASTDumper>, public ConstStmtVisitor<ASTDumper>,
|
2013-01-14 22:07:11 +08:00
|
|
|
public ConstCommentVisitor<ASTDumper> {
|
2011-07-23 18:55:15 +08:00
|
|
|
raw_ostream &OS;
|
2013-01-14 22:07:11 +08:00
|
|
|
const CommandTraits *Traits;
|
|
|
|
const SourceManager *SM;
|
2012-11-07 08:33:12 +08:00
|
|
|
bool IsFirstLine;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
// Indicates whether more child are expected at the current tree depth
|
|
|
|
enum IndentType { IT_Child, IT_LastChild };
|
|
|
|
|
|
|
|
/// Indents[i] indicates if another child exists at level i.
|
|
|
|
/// Used by Indent() to print the tree structure.
|
|
|
|
llvm::SmallVector<IndentType, 32> Indents;
|
|
|
|
|
|
|
|
/// Indicates that more children will be needed at this indent level.
|
|
|
|
/// If true, prevents lastChild() from marking the node as the last child.
|
|
|
|
/// This is used when there are multiple collections of children to be
|
|
|
|
/// dumped as well as during conditional node dumping.
|
|
|
|
bool MoreChildren;
|
|
|
|
|
2012-12-11 23:28:09 +08:00
|
|
|
/// Keep track of the last location we print out so that we can
|
|
|
|
/// print out deltas from then on out.
|
2007-08-30 14:17:34 +08:00
|
|
|
const char *LastLocFilename;
|
|
|
|
unsigned LastLocLine;
|
2009-05-30 04:38:28 +08:00
|
|
|
|
2013-01-14 22:07:11 +08:00
|
|
|
/// The \c FullComment parent of the comment being dumped.
|
|
|
|
const FullComment *FC;
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
bool ShowColors;
|
|
|
|
|
2012-11-07 08:33:12 +08:00
|
|
|
class IndentScope {
|
2012-12-13 21:59:55 +08:00
|
|
|
ASTDumper &Dumper;
|
2013-01-31 09:44:26 +08:00
|
|
|
// Preserve the Dumper's MoreChildren value from the previous IndentScope
|
|
|
|
bool MoreChildren;
|
2012-11-07 08:33:12 +08:00
|
|
|
public:
|
2012-12-13 21:59:55 +08:00
|
|
|
IndentScope(ASTDumper &Dumper) : Dumper(Dumper) {
|
2013-01-31 09:44:26 +08:00
|
|
|
MoreChildren = Dumper.hasMoreChildren();
|
|
|
|
Dumper.setMoreChildren(false);
|
2012-11-07 08:33:12 +08:00
|
|
|
Dumper.indent();
|
|
|
|
}
|
|
|
|
~IndentScope() {
|
2013-01-31 09:44:26 +08:00
|
|
|
Dumper.setMoreChildren(MoreChildren);
|
2012-11-07 08:33:12 +08:00
|
|
|
Dumper.unindent();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
class ColorScope {
|
|
|
|
ASTDumper &Dumper;
|
|
|
|
public:
|
|
|
|
ColorScope(ASTDumper &Dumper, TerminalColor Color)
|
|
|
|
: Dumper(Dumper) {
|
|
|
|
if (Dumper.ShowColors)
|
|
|
|
Dumper.OS.changeColor(Color.Color, Color.Bold);
|
|
|
|
}
|
|
|
|
~ColorScope() {
|
|
|
|
if (Dumper.ShowColors)
|
|
|
|
Dumper.OS.resetColor();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
public:
|
2013-01-14 22:07:11 +08:00
|
|
|
ASTDumper(raw_ostream &OS, const CommandTraits *Traits,
|
|
|
|
const SourceManager *SM)
|
2013-01-31 10:04:38 +08:00
|
|
|
: OS(OS), Traits(Traits), SM(SM), IsFirstLine(true), MoreChildren(false),
|
|
|
|
LastLocFilename(""), LastLocLine(~0U), FC(0),
|
2013-01-26 09:31:20 +08:00
|
|
|
ShowColors(SM && SM->getDiagnostics().getShowColors()) { }
|
|
|
|
|
|
|
|
ASTDumper(raw_ostream &OS, const CommandTraits *Traits,
|
|
|
|
const SourceManager *SM, bool ShowColors)
|
2013-01-31 10:04:38 +08:00
|
|
|
: OS(OS), Traits(Traits), SM(SM), IsFirstLine(true), MoreChildren(false),
|
|
|
|
LastLocFilename(""), LastLocLine(~0U),
|
2013-01-31 09:44:26 +08:00
|
|
|
ShowColors(ShowColors) { }
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-12-13 21:59:55 +08:00
|
|
|
~ASTDumper() {
|
2012-11-07 08:33:12 +08:00
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void dumpDecl(const Decl *D);
|
|
|
|
void dumpStmt(const Stmt *S);
|
2013-01-14 22:07:11 +08:00
|
|
|
void dumpFullComment(const FullComment *C);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
// Formatting
|
2012-12-11 23:28:09 +08:00
|
|
|
void indent();
|
|
|
|
void unindent();
|
2013-01-31 09:44:26 +08:00
|
|
|
void lastChild();
|
|
|
|
bool hasMoreChildren();
|
|
|
|
void setMoreChildren(bool Value);
|
|
|
|
|
|
|
|
// Utilities
|
2012-12-20 10:09:13 +08:00
|
|
|
void dumpPointer(const void *Ptr);
|
|
|
|
void dumpSourceRange(SourceRange R);
|
2012-12-11 23:28:09 +08:00
|
|
|
void dumpLocation(SourceLocation Loc);
|
2012-12-20 10:09:13 +08:00
|
|
|
void dumpBareType(QualType T);
|
2012-12-11 23:28:09 +08:00
|
|
|
void dumpType(QualType T);
|
2012-12-20 19:08:38 +08:00
|
|
|
void dumpBareDeclRef(const Decl *Node);
|
2012-12-20 20:23:54 +08:00
|
|
|
void dumpDeclRef(const Decl *Node, const char *Label = 0);
|
2012-12-20 19:08:38 +08:00
|
|
|
void dumpName(const NamedDecl *D);
|
2013-01-31 09:44:26 +08:00
|
|
|
bool hasNodes(const DeclContext *DC);
|
2012-12-20 19:08:38 +08:00
|
|
|
void dumpDeclContext(const DeclContext *DC);
|
2013-01-08 01:53:08 +08:00
|
|
|
void dumpAttr(const Attr *A);
|
2012-12-20 10:09:13 +08:00
|
|
|
|
|
|
|
// C++ Utilities
|
|
|
|
void dumpAccessSpecifier(AccessSpecifier AS);
|
2012-12-20 19:08:38 +08:00
|
|
|
void dumpCXXCtorInitializer(const CXXCtorInitializer *Init);
|
|
|
|
void dumpTemplateParameters(const TemplateParameterList *TPL);
|
2012-12-20 10:09:13 +08:00
|
|
|
void dumpTemplateArgumentListInfo(const TemplateArgumentListInfo &TALI);
|
|
|
|
void dumpTemplateArgumentLoc(const TemplateArgumentLoc &A);
|
|
|
|
void dumpTemplateArgumentList(const TemplateArgumentList &TAL);
|
|
|
|
void dumpTemplateArgument(const TemplateArgument &A,
|
|
|
|
SourceRange R = SourceRange());
|
|
|
|
|
|
|
|
// Decls
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitLabelDecl(const LabelDecl *D);
|
|
|
|
void VisitTypedefDecl(const TypedefDecl *D);
|
|
|
|
void VisitEnumDecl(const EnumDecl *D);
|
|
|
|
void VisitRecordDecl(const RecordDecl *D);
|
|
|
|
void VisitEnumConstantDecl(const EnumConstantDecl *D);
|
|
|
|
void VisitIndirectFieldDecl(const IndirectFieldDecl *D);
|
|
|
|
void VisitFunctionDecl(const FunctionDecl *D);
|
|
|
|
void VisitFieldDecl(const FieldDecl *D);
|
|
|
|
void VisitVarDecl(const VarDecl *D);
|
|
|
|
void VisitFileScopeAsmDecl(const FileScopeAsmDecl *D);
|
|
|
|
void VisitImportDecl(const ImportDecl *D);
|
2012-12-20 10:09:13 +08:00
|
|
|
|
|
|
|
// C++ Decls
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitNamespaceDecl(const NamespaceDecl *D);
|
|
|
|
void VisitUsingDirectiveDecl(const UsingDirectiveDecl *D);
|
|
|
|
void VisitNamespaceAliasDecl(const NamespaceAliasDecl *D);
|
|
|
|
void VisitTypeAliasDecl(const TypeAliasDecl *D);
|
|
|
|
void VisitTypeAliasTemplateDecl(const TypeAliasTemplateDecl *D);
|
|
|
|
void VisitCXXRecordDecl(const CXXRecordDecl *D);
|
|
|
|
void VisitStaticAssertDecl(const StaticAssertDecl *D);
|
|
|
|
void VisitFunctionTemplateDecl(const FunctionTemplateDecl *D);
|
|
|
|
void VisitClassTemplateDecl(const ClassTemplateDecl *D);
|
2012-12-20 10:09:13 +08:00
|
|
|
void VisitClassTemplateSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassTemplateSpecializationDecl *D);
|
2012-12-20 10:09:13 +08:00
|
|
|
void VisitClassTemplatePartialSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassTemplatePartialSpecializationDecl *D);
|
2012-12-20 10:09:13 +08:00
|
|
|
void VisitClassScopeFunctionSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassScopeFunctionSpecializationDecl *D);
|
|
|
|
void VisitTemplateTypeParmDecl(const TemplateTypeParmDecl *D);
|
|
|
|
void VisitNonTypeTemplateParmDecl(const NonTypeTemplateParmDecl *D);
|
|
|
|
void VisitTemplateTemplateParmDecl(const TemplateTemplateParmDecl *D);
|
|
|
|
void VisitUsingDecl(const UsingDecl *D);
|
|
|
|
void VisitUnresolvedUsingTypenameDecl(const UnresolvedUsingTypenameDecl *D);
|
|
|
|
void VisitUnresolvedUsingValueDecl(const UnresolvedUsingValueDecl *D);
|
|
|
|
void VisitUsingShadowDecl(const UsingShadowDecl *D);
|
|
|
|
void VisitLinkageSpecDecl(const LinkageSpecDecl *D);
|
|
|
|
void VisitAccessSpecDecl(const AccessSpecDecl *D);
|
|
|
|
void VisitFriendDecl(const FriendDecl *D);
|
2012-12-20 10:09:13 +08:00
|
|
|
|
|
|
|
// ObjC Decls
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitObjCIvarDecl(const ObjCIvarDecl *D);
|
|
|
|
void VisitObjCMethodDecl(const ObjCMethodDecl *D);
|
|
|
|
void VisitObjCCategoryDecl(const ObjCCategoryDecl *D);
|
|
|
|
void VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *D);
|
|
|
|
void VisitObjCProtocolDecl(const ObjCProtocolDecl *D);
|
|
|
|
void VisitObjCInterfaceDecl(const ObjCInterfaceDecl *D);
|
|
|
|
void VisitObjCImplementationDecl(const ObjCImplementationDecl *D);
|
|
|
|
void VisitObjCCompatibleAliasDecl(const ObjCCompatibleAliasDecl *D);
|
|
|
|
void VisitObjCPropertyDecl(const ObjCPropertyDecl *D);
|
|
|
|
void VisitObjCPropertyImplDecl(const ObjCPropertyImplDecl *D);
|
|
|
|
void VisitBlockDecl(const BlockDecl *D);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-30 09:00:35 +08:00
|
|
|
// Stmts.
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitStmt(const Stmt *Node);
|
|
|
|
void VisitDeclStmt(const DeclStmt *Node);
|
|
|
|
void VisitAttributedStmt(const AttributedStmt *Node);
|
|
|
|
void VisitLabelStmt(const LabelStmt *Node);
|
|
|
|
void VisitGotoStmt(const GotoStmt *Node);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-30 09:00:35 +08:00
|
|
|
// Exprs
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitExpr(const Expr *Node);
|
|
|
|
void VisitCastExpr(const CastExpr *Node);
|
|
|
|
void VisitDeclRefExpr(const DeclRefExpr *Node);
|
|
|
|
void VisitPredefinedExpr(const PredefinedExpr *Node);
|
|
|
|
void VisitCharacterLiteral(const CharacterLiteral *Node);
|
|
|
|
void VisitIntegerLiteral(const IntegerLiteral *Node);
|
|
|
|
void VisitFloatingLiteral(const FloatingLiteral *Node);
|
|
|
|
void VisitStringLiteral(const StringLiteral *Str);
|
|
|
|
void VisitUnaryOperator(const UnaryOperator *Node);
|
|
|
|
void VisitUnaryExprOrTypeTraitExpr(const UnaryExprOrTypeTraitExpr *Node);
|
|
|
|
void VisitMemberExpr(const MemberExpr *Node);
|
|
|
|
void VisitExtVectorElementExpr(const ExtVectorElementExpr *Node);
|
|
|
|
void VisitBinaryOperator(const BinaryOperator *Node);
|
|
|
|
void VisitCompoundAssignOperator(const CompoundAssignOperator *Node);
|
|
|
|
void VisitAddrLabelExpr(const AddrLabelExpr *Node);
|
|
|
|
void VisitBlockExpr(const BlockExpr *Node);
|
|
|
|
void VisitOpaqueValueExpr(const OpaqueValueExpr *Node);
|
2007-08-30 09:00:35 +08:00
|
|
|
|
|
|
|
// C++
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitCXXNamedCastExpr(const CXXNamedCastExpr *Node);
|
|
|
|
void VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *Node);
|
|
|
|
void VisitCXXThisExpr(const CXXThisExpr *Node);
|
|
|
|
void VisitCXXFunctionalCastExpr(const CXXFunctionalCastExpr *Node);
|
|
|
|
void VisitCXXConstructExpr(const CXXConstructExpr *Node);
|
|
|
|
void VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *Node);
|
|
|
|
void VisitExprWithCleanups(const ExprWithCleanups *Node);
|
|
|
|
void VisitUnresolvedLookupExpr(const UnresolvedLookupExpr *Node);
|
|
|
|
void dumpCXXTemporary(const CXXTemporary *Temporary);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-30 09:00:35 +08:00
|
|
|
// ObjC
|
2013-02-01 20:35:51 +08:00
|
|
|
void VisitObjCAtCatchStmt(const ObjCAtCatchStmt *Node);
|
|
|
|
void VisitObjCEncodeExpr(const ObjCEncodeExpr *Node);
|
|
|
|
void VisitObjCMessageExpr(const ObjCMessageExpr *Node);
|
|
|
|
void VisitObjCBoxedExpr(const ObjCBoxedExpr *Node);
|
|
|
|
void VisitObjCSelectorExpr(const ObjCSelectorExpr *Node);
|
|
|
|
void VisitObjCProtocolExpr(const ObjCProtocolExpr *Node);
|
|
|
|
void VisitObjCPropertyRefExpr(const ObjCPropertyRefExpr *Node);
|
|
|
|
void VisitObjCSubscriptRefExpr(const ObjCSubscriptRefExpr *Node);
|
|
|
|
void VisitObjCIvarRefExpr(const ObjCIvarRefExpr *Node);
|
|
|
|
void VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *Node);
|
2013-01-14 22:07:11 +08:00
|
|
|
|
|
|
|
// Comments.
|
|
|
|
const char *getCommandName(unsigned CommandID);
|
|
|
|
void dumpComment(const Comment *C);
|
|
|
|
|
|
|
|
// Inline comments.
|
|
|
|
void visitTextComment(const TextComment *C);
|
|
|
|
void visitInlineCommandComment(const InlineCommandComment *C);
|
|
|
|
void visitHTMLStartTagComment(const HTMLStartTagComment *C);
|
|
|
|
void visitHTMLEndTagComment(const HTMLEndTagComment *C);
|
|
|
|
|
|
|
|
// Block comments.
|
|
|
|
void visitBlockCommandComment(const BlockCommandComment *C);
|
|
|
|
void visitParamCommandComment(const ParamCommandComment *C);
|
|
|
|
void visitTParamCommandComment(const TParamCommandComment *C);
|
|
|
|
void visitVerbatimBlockComment(const VerbatimBlockComment *C);
|
|
|
|
void visitVerbatimBlockLineComment(const VerbatimBlockLineComment *C);
|
|
|
|
void visitVerbatimLineComment(const VerbatimLineComment *C);
|
2007-08-09 06:51:59 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2007-08-30 14:17:34 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Utilities
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
// Print out the appropriate tree structure using the Indents vector.
|
|
|
|
// Example of tree and the Indents vector at each level.
|
|
|
|
// A { }
|
|
|
|
// |-B { IT_Child }
|
|
|
|
// | `-C { IT_Child, IT_LastChild }
|
|
|
|
// `-D { IT_LastChild }
|
|
|
|
// |-E { IT_LastChild, IT_Child }
|
|
|
|
// `-F { IT_LastChild, IT_LastChild }
|
|
|
|
// Type non-last element, last element
|
|
|
|
// IT_Child "| " "|-"
|
|
|
|
// IT_LastChild " " "`-"
|
2012-12-13 21:59:55 +08:00
|
|
|
void ASTDumper::indent() {
|
2012-12-11 23:28:09 +08:00
|
|
|
if (IsFirstLine)
|
|
|
|
IsFirstLine = false;
|
|
|
|
else
|
|
|
|
OS << "\n";
|
2013-01-31 09:44:26 +08:00
|
|
|
|
|
|
|
ColorScope Color(*this, IndentColor);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (llvm::SmallVector<IndentType, 32>::const_iterator I = Indents.begin(),
|
|
|
|
E = Indents.end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
switch (*I) {
|
2013-01-31 10:04:38 +08:00
|
|
|
case IT_Child:
|
|
|
|
if (I == E - 1)
|
|
|
|
OS << "|-";
|
|
|
|
else
|
|
|
|
OS << "| ";
|
|
|
|
continue;
|
|
|
|
case IT_LastChild:
|
|
|
|
if (I == E - 1)
|
|
|
|
OS << "`-";
|
|
|
|
else
|
|
|
|
OS << " ";
|
|
|
|
continue;
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2013-01-31 10:04:38 +08:00
|
|
|
llvm_unreachable("Invalid IndentType");
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
|
|
|
Indents.push_back(IT_Child);
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
|
|
|
|
2012-12-13 21:59:55 +08:00
|
|
|
void ASTDumper::unindent() {
|
2013-01-31 09:44:26 +08:00
|
|
|
Indents.pop_back();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Call before each potential last child node is to be dumped. If MoreChildren
|
|
|
|
// is false, then this is the last child, otherwise treat as a regular node.
|
|
|
|
void ASTDumper::lastChild() {
|
|
|
|
if (!hasMoreChildren())
|
|
|
|
Indents.back() = IT_LastChild;
|
|
|
|
}
|
|
|
|
|
|
|
|
// MoreChildren should be set before calling another function that may print
|
|
|
|
// additional nodes to prevent conflicting final child nodes.
|
|
|
|
bool ASTDumper::hasMoreChildren() {
|
|
|
|
return MoreChildren;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::setMoreChildren(bool Value) {
|
|
|
|
MoreChildren = Value;
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::dumpPointer(const void *Ptr) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, AddressColor);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ' << Ptr;
|
|
|
|
}
|
|
|
|
|
2012-12-13 21:59:55 +08:00
|
|
|
void ASTDumper::dumpLocation(SourceLocation Loc) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, LocationColor);
|
2009-01-16 15:00:02 +08:00
|
|
|
SourceLocation SpellingLoc = SM->getSpellingLoc(Loc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-30 14:17:34 +08:00
|
|
|
// The general format we print out is filename:line:col, but we drop pieces
|
|
|
|
// that haven't changed since the last loc printed.
|
2009-01-27 15:57:44 +08:00
|
|
|
PresumedLoc PLoc = SM->getPresumedLoc(SpellingLoc);
|
|
|
|
|
2010-11-12 15:15:47 +08:00
|
|
|
if (PLoc.isInvalid()) {
|
|
|
|
OS << "<invalid sloc>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-27 15:57:44 +08:00
|
|
|
if (strcmp(PLoc.getFilename(), LastLocFilename) != 0) {
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << PLoc.getFilename() << ':' << PLoc.getLine()
|
|
|
|
<< ':' << PLoc.getColumn();
|
2009-01-27 15:57:44 +08:00
|
|
|
LastLocFilename = PLoc.getFilename();
|
|
|
|
LastLocLine = PLoc.getLine();
|
|
|
|
} else if (PLoc.getLine() != LastLocLine) {
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << "line" << ':' << PLoc.getLine()
|
|
|
|
<< ':' << PLoc.getColumn();
|
2009-01-27 15:57:44 +08:00
|
|
|
LastLocLine = PLoc.getLine();
|
2007-08-30 14:17:34 +08:00
|
|
|
} else {
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << "col" << ':' << PLoc.getColumn();
|
2007-08-30 14:17:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::dumpSourceRange(SourceRange R) {
|
2007-08-30 14:17:34 +08:00
|
|
|
// Can't translate locations if a SourceManager isn't available.
|
2012-12-11 23:28:09 +08:00
|
|
|
if (!SM)
|
|
|
|
return;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " <";
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpLocation(R.getBegin());
|
2007-10-17 06:36:42 +08:00
|
|
|
if (R.getBegin() != R.getEnd()) {
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << ", ";
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpLocation(R.getEnd());
|
2007-08-30 14:17:34 +08:00
|
|
|
}
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << ">";
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2007-08-30 14:17:34 +08:00
|
|
|
// <t2.c:123:421[blah], t2.c:412:321>
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::dumpBareType(QualType T) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, TypeColor);
|
|
|
|
|
2012-12-11 23:28:09 +08:00
|
|
|
SplitQualType T_split = T.split();
|
|
|
|
OS << "'" << QualType::getAsString(T_split) << "'";
|
|
|
|
|
|
|
|
if (!T.isNull()) {
|
|
|
|
// If the type is sugared, also dump a (shallow) desugared type.
|
|
|
|
SplitQualType D_split = T.getSplitDesugaredType();
|
|
|
|
if (T_split != D_split)
|
|
|
|
OS << ":'" << QualType::getAsString(D_split) << "'";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::dumpType(QualType T) {
|
|
|
|
OS << ' ';
|
|
|
|
dumpBareType(T);
|
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpBareDeclRef(const Decl *D) {
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, DeclKindNameColor);
|
|
|
|
OS << D->getDeclKindName();
|
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpPointer(D);
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, DeclNameColor);
|
2013-05-15 05:04:00 +08:00
|
|
|
OS << " '" << ND->getDeclName() << '\'';
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
if (const ValueDecl *VD = dyn_cast<ValueDecl>(D))
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpType(VD->getType());
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpDeclRef(const Decl *D, const char *Label) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (!D)
|
|
|
|
return;
|
|
|
|
|
|
|
|
IndentScope Indent(*this);
|
|
|
|
if (Label)
|
|
|
|
OS << Label << ' ';
|
|
|
|
dumpBareDeclRef(D);
|
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpName(const NamedDecl *ND) {
|
2013-01-26 09:31:20 +08:00
|
|
|
if (ND->getDeclName()) {
|
|
|
|
ColorScope Color(*this, DeclNameColor);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ' << ND->getNameAsString();
|
2013-01-26 09:31:20 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
bool ASTDumper::hasNodes(const DeclContext *DC) {
|
|
|
|
if (!DC)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return DC->decls_begin() != DC->decls_end();
|
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpDeclContext(const DeclContext *DC) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (!DC)
|
|
|
|
return;
|
|
|
|
for (DeclContext::decl_iterator I = DC->decls_begin(), E = DC->decls_end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
DeclContext::decl_iterator Next = I;
|
|
|
|
++Next;
|
|
|
|
if (Next == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-01-08 01:53:08 +08:00
|
|
|
void ASTDumper::dumpAttr(const Attr *A) {
|
|
|
|
IndentScope Indent(*this);
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, AttrColor);
|
|
|
|
switch (A->getKind()) {
|
2013-01-08 01:53:08 +08:00
|
|
|
#define ATTR(X) case attr::X: OS << #X; break;
|
|
|
|
#include "clang/Basic/AttrList.inc"
|
2013-01-26 09:31:20 +08:00
|
|
|
default: llvm_unreachable("unexpected attribute kind");
|
|
|
|
}
|
|
|
|
OS << "Attr";
|
2013-01-08 01:53:08 +08:00
|
|
|
}
|
|
|
|
dumpPointer(A);
|
|
|
|
dumpSourceRange(A->getRange());
|
|
|
|
#include "clang/AST/AttrDump.inc"
|
|
|
|
}
|
|
|
|
|
2013-02-07 09:35:44 +08:00
|
|
|
static Decl *getPreviousDeclImpl(...) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
static const Decl *getPreviousDeclImpl(const Redeclarable<T> *D) {
|
|
|
|
return D->getPreviousDecl();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the previous declaration in the redeclaration chain for a declaration.
|
|
|
|
static const Decl *getPreviousDecl(const Decl *D) {
|
|
|
|
switch (D->getKind()) {
|
|
|
|
#define DECL(DERIVED, BASE) \
|
|
|
|
case Decl::DERIVED: \
|
|
|
|
return getPreviousDeclImpl(cast<DERIVED##Decl>(D));
|
|
|
|
#define ABSTRACT_DECL(DECL)
|
|
|
|
#include "clang/AST/DeclNodes.inc"
|
|
|
|
}
|
|
|
|
llvm_unreachable("Decl that isn't part of DeclNodes.inc!");
|
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// C++ Utilities
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void ASTDumper::dumpAccessSpecifier(AccessSpecifier AS) {
|
|
|
|
switch (AS) {
|
|
|
|
case AS_none:
|
|
|
|
break;
|
|
|
|
case AS_public:
|
|
|
|
OS << "public";
|
|
|
|
break;
|
|
|
|
case AS_protected:
|
|
|
|
OS << "protected";
|
|
|
|
break;
|
|
|
|
case AS_private:
|
|
|
|
OS << "private";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpCXXCtorInitializer(const CXXCtorInitializer *Init) {
|
2012-12-20 10:09:13 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "CXXCtorInitializer";
|
|
|
|
if (Init->isAnyMemberInitializer()) {
|
|
|
|
OS << ' ';
|
|
|
|
dumpBareDeclRef(Init->getAnyMember());
|
|
|
|
} else {
|
|
|
|
dumpType(QualType(Init->getBaseClass(), 0));
|
|
|
|
}
|
|
|
|
dumpStmt(Init->getInit());
|
|
|
|
}
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
void ASTDumper::dumpTemplateParameters(const TemplateParameterList *TPL) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (!TPL)
|
|
|
|
return;
|
|
|
|
|
2012-12-20 19:08:38 +08:00
|
|
|
for (TemplateParameterList::const_iterator I = TPL->begin(), E = TPL->end();
|
2012-12-20 10:09:13 +08:00
|
|
|
I != E; ++I)
|
|
|
|
dumpDecl(*I);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpTemplateArgumentListInfo(
|
|
|
|
const TemplateArgumentListInfo &TALI) {
|
2013-01-31 09:44:26 +08:00
|
|
|
for (unsigned i = 0, e = TALI.size(); i < e; ++i) {
|
|
|
|
if (i + 1 == e)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpTemplateArgumentLoc(TALI[i]);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpTemplateArgumentLoc(const TemplateArgumentLoc &A) {
|
|
|
|
dumpTemplateArgument(A.getArgument(), A.getSourceRange());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpTemplateArgumentList(const TemplateArgumentList &TAL) {
|
|
|
|
for (unsigned i = 0, e = TAL.size(); i < e; ++i)
|
|
|
|
dumpTemplateArgument(TAL[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpTemplateArgument(const TemplateArgument &A, SourceRange R) {
|
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "TemplateArgument";
|
|
|
|
if (R.isValid())
|
|
|
|
dumpSourceRange(R);
|
|
|
|
|
|
|
|
switch (A.getKind()) {
|
|
|
|
case TemplateArgument::Null:
|
|
|
|
OS << " null";
|
|
|
|
break;
|
|
|
|
case TemplateArgument::Type:
|
|
|
|
OS << " type";
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpType(A.getAsType());
|
|
|
|
break;
|
|
|
|
case TemplateArgument::Declaration:
|
|
|
|
OS << " decl";
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(A.getAsDecl());
|
|
|
|
break;
|
|
|
|
case TemplateArgument::NullPtr:
|
|
|
|
OS << " nullptr";
|
|
|
|
break;
|
|
|
|
case TemplateArgument::Integral:
|
2012-12-20 19:08:38 +08:00
|
|
|
OS << " integral " << A.getAsIntegral();
|
2012-12-20 10:09:13 +08:00
|
|
|
break;
|
|
|
|
case TemplateArgument::Template:
|
|
|
|
OS << " template ";
|
|
|
|
A.getAsTemplate().dump(OS);
|
|
|
|
break;
|
|
|
|
case TemplateArgument::TemplateExpansion:
|
|
|
|
OS << " template expansion";
|
|
|
|
A.getAsTemplateOrTemplatePattern().dump(OS);
|
|
|
|
break;
|
|
|
|
case TemplateArgument::Expression:
|
|
|
|
OS << " expr";
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(A.getAsExpr());
|
|
|
|
break;
|
|
|
|
case TemplateArgument::Pack:
|
|
|
|
OS << " pack";
|
|
|
|
for (TemplateArgument::pack_iterator I = A.pack_begin(), E = A.pack_end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpTemplateArgument(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
break;
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-12-11 23:28:09 +08:00
|
|
|
// Decl dumping methods.
|
2007-08-09 06:51:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::dumpDecl(const Decl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
|
|
|
|
if (!D) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, NullColor);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << "<<<NULL>>>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, DeclKindNameColor);
|
|
|
|
OS << D->getDeclKindName() << "Decl";
|
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpPointer(D);
|
2013-02-07 09:35:44 +08:00
|
|
|
if (D->getLexicalDeclContext() != D->getDeclContext())
|
|
|
|
OS << " parent " << cast<Decl>(D->getDeclContext());
|
|
|
|
if (const Decl *Prev = getPreviousDecl(D))
|
|
|
|
OS << " prev " << Prev;
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpSourceRange(D->getSourceRange());
|
2013-01-31 09:44:26 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
bool HasAttrs = D->attr_begin() != D->attr_end();
|
2013-05-21 13:24:00 +08:00
|
|
|
const FullComment *Comment =
|
|
|
|
D->getASTContext().getLocalCommentForDeclUncached(D);
|
2013-01-31 09:44:26 +08:00
|
|
|
// Decls within functions are visited by the body
|
|
|
|
bool HasDeclContext = !isa<FunctionDecl>(*D) && !isa<ObjCMethodDecl>(*D) &&
|
|
|
|
hasNodes(dyn_cast<DeclContext>(D));
|
|
|
|
|
2013-05-21 13:24:00 +08:00
|
|
|
setMoreChildren(HasAttrs || Comment || HasDeclContext);
|
2013-02-01 20:35:51 +08:00
|
|
|
ConstDeclVisitor<ASTDumper>::Visit(D);
|
2013-01-31 09:44:26 +08:00
|
|
|
|
2013-05-21 13:24:00 +08:00
|
|
|
setMoreChildren(Comment || HasDeclContext);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (Decl::attr_iterator I = D->attr_begin(), E = D->attr_end();
|
|
|
|
I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
|
|
|
dumpAttr(*I);
|
2013-01-08 01:53:08 +08:00
|
|
|
}
|
2013-01-31 09:44:26 +08:00
|
|
|
|
|
|
|
setMoreChildren(HasDeclContext);
|
|
|
|
lastChild();
|
2013-05-21 13:24:00 +08:00
|
|
|
dumpFullComment(Comment);
|
2013-01-31 09:44:26 +08:00
|
|
|
|
|
|
|
setMoreChildren(false);
|
|
|
|
if (HasDeclContext)
|
2013-02-07 09:35:44 +08:00
|
|
|
dumpDeclContext(cast<DeclContext>(D));
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitLabelDecl(const LabelDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitTypedefDecl(const TypedefDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getUnderlyingType());
|
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitEnumDecl(const EnumDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (D->isScoped()) {
|
|
|
|
if (D->isScopedUsingClassTag())
|
|
|
|
OS << " class";
|
|
|
|
else
|
|
|
|
OS << " struct";
|
|
|
|
}
|
|
|
|
dumpName(D);
|
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
|
|
|
if (D->isFixed())
|
|
|
|
dumpType(D->getIntegerType());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitRecordDecl(const RecordDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ' << D->getKindName();
|
|
|
|
dumpName(D);
|
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitEnumConstantDecl(const EnumConstantDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
2013-02-01 20:35:51 +08:00
|
|
|
if (const Expr *Init = D->getInitExpr()) {
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(Init);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitIndirectFieldDecl(const IndirectFieldDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
|
|
|
for (IndirectFieldDecl::chain_iterator I = D->chain_begin(),
|
2013-02-01 20:35:51 +08:00
|
|
|
E = D->chain_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFunctionDecl(const FunctionDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
|
|
|
|
2013-04-04 03:27:57 +08:00
|
|
|
StorageClass SC = D->getStorageClass();
|
2012-12-20 10:09:13 +08:00
|
|
|
if (SC != SC_None)
|
|
|
|
OS << ' ' << VarDecl::getStorageClassSpecifierString(SC);
|
|
|
|
if (D->isInlineSpecified())
|
|
|
|
OS << " inline";
|
|
|
|
if (D->isVirtualAsWritten())
|
|
|
|
OS << " virtual";
|
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
|
|
|
|
|
|
|
if (D->isPure())
|
|
|
|
OS << " pure";
|
|
|
|
else if (D->isDeletedAsWritten())
|
|
|
|
OS << " delete";
|
|
|
|
|
2013-05-17 10:09:46 +08:00
|
|
|
if (const FunctionProtoType *FPT = D->getType()->getAs<FunctionProtoType>()) {
|
|
|
|
FunctionProtoType::ExtProtoInfo EPI = FPT->getExtProtoInfo();
|
|
|
|
switch (EPI.ExceptionSpecType) {
|
|
|
|
default: break;
|
|
|
|
case EST_Unevaluated:
|
|
|
|
OS << " noexcept-unevaluated " << EPI.ExceptionSpecDecl;
|
|
|
|
break;
|
|
|
|
case EST_Uninstantiated:
|
|
|
|
OS << " noexcept-uninstantiated " << EPI.ExceptionSpecTemplate;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
bool OldMoreChildren = hasMoreChildren();
|
|
|
|
const FunctionTemplateSpecializationInfo *FTSI =
|
|
|
|
D->getTemplateSpecializationInfo();
|
|
|
|
bool HasTemplateSpecialization = FTSI;
|
|
|
|
|
|
|
|
bool HasNamedDecls = D->getDeclsInPrototypeScope().begin() !=
|
|
|
|
D->getDeclsInPrototypeScope().end();
|
|
|
|
|
|
|
|
bool HasFunctionDecls = D->param_begin() != D->param_end();
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
const CXXConstructorDecl *C = dyn_cast<CXXConstructorDecl>(D);
|
2013-01-31 09:44:26 +08:00
|
|
|
bool HasCtorInitializers = C && C->init_begin() != C->init_end();
|
|
|
|
|
|
|
|
bool HasDeclarationBody = D->doesThisDeclarationHaveABody();
|
|
|
|
|
|
|
|
setMoreChildren(OldMoreChildren || HasNamedDecls || HasFunctionDecls ||
|
|
|
|
HasCtorInitializers || HasDeclarationBody);
|
|
|
|
if (HasTemplateSpecialization) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpTemplateArgumentList(*FTSI->TemplateArguments);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(OldMoreChildren || HasFunctionDecls ||
|
|
|
|
HasCtorInitializers || HasDeclarationBody);
|
2013-01-13 03:30:44 +08:00
|
|
|
for (ArrayRef<NamedDecl *>::iterator
|
2012-12-20 10:09:13 +08:00
|
|
|
I = D->getDeclsInPrototypeScope().begin(),
|
2013-01-31 09:44:26 +08:00
|
|
|
E = D->getDeclsInPrototypeScope().end(); I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(OldMoreChildren || HasCtorInitializers || HasDeclarationBody);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (FunctionDecl::param_const_iterator I = D->param_begin(),
|
|
|
|
E = D->param_end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
setMoreChildren(OldMoreChildren || HasDeclarationBody);
|
|
|
|
if (HasCtorInitializers)
|
2012-12-20 10:09:13 +08:00
|
|
|
for (CXXConstructorDecl::init_const_iterator I = C->init_begin(),
|
2013-02-01 20:35:51 +08:00
|
|
|
E = C->init_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpCXXCtorInitializer(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(OldMoreChildren);
|
|
|
|
if (HasDeclarationBody) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getBody());
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFieldDecl(const FieldDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
|
|
|
if (D->isMutable())
|
|
|
|
OS << " mutable";
|
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
2013-01-31 09:44:26 +08:00
|
|
|
|
|
|
|
bool OldMoreChildren = hasMoreChildren();
|
|
|
|
bool IsBitField = D->isBitField();
|
|
|
|
Expr *Init = D->getInClassInitializer();
|
|
|
|
bool HasInit = Init;
|
|
|
|
|
|
|
|
setMoreChildren(OldMoreChildren || HasInit);
|
|
|
|
if (IsBitField) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getBitWidth());
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
|
|
|
setMoreChildren(OldMoreChildren);
|
|
|
|
if (HasInit) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(Init);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitVarDecl(const VarDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
2013-04-04 03:27:57 +08:00
|
|
|
StorageClass SC = D->getStorageClass();
|
2012-12-20 10:09:13 +08:00
|
|
|
if (SC != SC_None)
|
|
|
|
OS << ' ' << VarDecl::getStorageClassSpecifierString(SC);
|
2013-04-13 10:43:54 +08:00
|
|
|
switch (D->getTLSKind()) {
|
|
|
|
case VarDecl::TLS_None: break;
|
|
|
|
case VarDecl::TLS_Static: OS << " tls"; break;
|
|
|
|
case VarDecl::TLS_Dynamic: OS << " tls_dynamic"; break;
|
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
if (D->isModulePrivate())
|
|
|
|
OS << " __module_private__";
|
|
|
|
if (D->isNRVOVariable())
|
|
|
|
OS << " nrvo";
|
2013-01-31 09:44:26 +08:00
|
|
|
if (D->hasInit()) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getInit());
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFileScopeAsmDecl(const FileScopeAsmDecl *D) {
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getAsmString());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitImportDecl(const ImportDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ' << D->getImportedModule()->getFullModuleName();
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// C++ Declarations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitNamespaceDecl(const NamespaceDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
if (D->isInline())
|
|
|
|
OS << " inline";
|
|
|
|
if (!D->isOriginalNamespace())
|
|
|
|
dumpDeclRef(D->getOriginalNamespace(), "original");
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUsingDirectiveDecl(const UsingDirectiveDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
dumpBareDeclRef(D->getNominatedNamespace());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitNamespaceAliasDecl(const NamespaceAliasDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpDeclRef(D->getAliasedNamespace());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitTypeAliasDecl(const TypeAliasDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getUnderlyingType());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitTypeAliasTemplateDecl(const TypeAliasTemplateDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpTemplateParameters(D->getTemplateParameters());
|
|
|
|
dumpDecl(D->getTemplatedDecl());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXRecordDecl(const CXXRecordDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
VisitRecordDecl(D);
|
|
|
|
if (!D->isCompleteDefinition())
|
|
|
|
return;
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
for (CXXRecordDecl::base_class_const_iterator I = D->bases_begin(),
|
|
|
|
E = D->bases_end();
|
|
|
|
I != E; ++I) {
|
2012-12-20 10:09:13 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
if (I->isVirtual())
|
|
|
|
OS << "virtual ";
|
|
|
|
dumpAccessSpecifier(I->getAccessSpecifier());
|
|
|
|
dumpType(I->getType());
|
|
|
|
if (I->isPackExpansion())
|
|
|
|
OS << "...";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitStaticAssertDecl(const StaticAssertDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getAssertExpr());
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getMessage());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFunctionTemplateDecl(const FunctionTemplateDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpTemplateParameters(D->getTemplateParameters());
|
|
|
|
dumpDecl(D->getTemplatedDecl());
|
2013-02-14 21:20:36 +08:00
|
|
|
for (FunctionTemplateDecl::spec_iterator I = D->spec_begin(),
|
|
|
|
E = D->spec_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
FunctionTemplateDecl::spec_iterator Next = I;
|
|
|
|
++Next;
|
|
|
|
if (Next == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
switch (I->getTemplateSpecializationKind()) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ImplicitInstantiation:
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
2013-02-22 06:01:10 +08:00
|
|
|
if (D == D->getCanonicalDecl())
|
|
|
|
dumpDecl(*I);
|
|
|
|
else
|
|
|
|
dumpDeclRef(*I);
|
2012-12-20 10:09:13 +08:00
|
|
|
break;
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
dumpDeclRef(*I);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitClassTemplateDecl(const ClassTemplateDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpTemplateParameters(D->getTemplateParameters());
|
2013-01-31 09:44:26 +08:00
|
|
|
|
2013-02-14 21:20:36 +08:00
|
|
|
ClassTemplateDecl::spec_iterator I = D->spec_begin();
|
|
|
|
ClassTemplateDecl::spec_iterator E = D->spec_end();
|
2013-02-01 20:35:51 +08:00
|
|
|
if (I == E)
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(D->getTemplatedDecl());
|
2013-02-01 20:35:51 +08:00
|
|
|
for (; I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
ClassTemplateDecl::spec_iterator Next = I;
|
|
|
|
++Next;
|
|
|
|
if (Next == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
switch (I->getTemplateSpecializationKind()) {
|
|
|
|
case TSK_Undeclared:
|
|
|
|
case TSK_ImplicitInstantiation:
|
2013-02-22 06:01:10 +08:00
|
|
|
if (D == D->getCanonicalDecl())
|
|
|
|
dumpDecl(*I);
|
|
|
|
else
|
|
|
|
dumpDeclRef(*I);
|
2012-12-20 10:09:13 +08:00
|
|
|
break;
|
|
|
|
case TSK_ExplicitSpecialization:
|
|
|
|
case TSK_ExplicitInstantiationDeclaration:
|
|
|
|
case TSK_ExplicitInstantiationDefinition:
|
|
|
|
dumpDeclRef(*I);
|
|
|
|
break;
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::VisitClassTemplateSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassTemplateSpecializationDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
VisitCXXRecordDecl(D);
|
|
|
|
dumpTemplateArgumentList(D->getTemplateArgs());
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
void ASTDumper::VisitClassTemplatePartialSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassTemplatePartialSpecializationDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
VisitClassTemplateSpecializationDecl(D);
|
|
|
|
dumpTemplateParameters(D->getTemplateParameters());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::VisitClassScopeFunctionSpecializationDecl(
|
2013-02-01 20:35:51 +08:00
|
|
|
const ClassScopeFunctionSpecializationDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getSpecialization());
|
|
|
|
if (D->hasExplicitTemplateArgs())
|
|
|
|
dumpTemplateArgumentListInfo(D->templateArgs());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitTemplateTypeParmDecl(const TemplateTypeParmDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (D->wasDeclaredWithTypename())
|
|
|
|
OS << " typename";
|
|
|
|
else
|
|
|
|
OS << " class";
|
|
|
|
if (D->isParameterPack())
|
|
|
|
OS << " ...";
|
|
|
|
dumpName(D);
|
|
|
|
if (D->hasDefaultArgument())
|
|
|
|
dumpType(D->getDefaultArgument());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitNonTypeTemplateParmDecl(const NonTypeTemplateParmDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpType(D->getType());
|
|
|
|
if (D->isParameterPack())
|
|
|
|
OS << " ...";
|
|
|
|
dumpName(D);
|
|
|
|
if (D->hasDefaultArgument())
|
|
|
|
dumpStmt(D->getDefaultArgument());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitTemplateTemplateParmDecl(
|
|
|
|
const TemplateTemplateParmDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (D->isParameterPack())
|
|
|
|
OS << " ...";
|
|
|
|
dumpName(D);
|
|
|
|
dumpTemplateParameters(D->getTemplateParameters());
|
|
|
|
if (D->hasDefaultArgument())
|
|
|
|
dumpTemplateArgumentLoc(D->getDefaultArgument());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUsingDecl(const UsingDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
D->getQualifier()->print(OS, D->getASTContext().getPrintingPolicy());
|
|
|
|
OS << D->getNameAsString();
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUnresolvedUsingTypenameDecl(
|
|
|
|
const UnresolvedUsingTypenameDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
D->getQualifier()->print(OS, D->getASTContext().getPrintingPolicy());
|
|
|
|
OS << D->getNameAsString();
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUnresolvedUsingValueDecl(const UnresolvedUsingValueDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
D->getQualifier()->print(OS, D->getASTContext().getPrintingPolicy());
|
|
|
|
OS << D->getNameAsString();
|
|
|
|
dumpType(D->getType());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUsingShadowDecl(const UsingShadowDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
dumpBareDeclRef(D->getTargetDecl());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitLinkageSpecDecl(const LinkageSpecDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
switch (D->getLanguage()) {
|
|
|
|
case LinkageSpecDecl::lang_c: OS << " C"; break;
|
|
|
|
case LinkageSpecDecl::lang_cxx: OS << " C++"; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitAccessSpecDecl(const AccessSpecDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << ' ';
|
|
|
|
dumpAccessSpecifier(D->getAccess());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFriendDecl(const FriendDecl *D) {
|
2013-02-07 09:35:44 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
if (TypeSourceInfo *T = D->getFriendType())
|
|
|
|
dumpType(T->getType());
|
|
|
|
else
|
|
|
|
dumpDecl(D->getFriendDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Obj-C Declarations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCIvarDecl(const ObjCIvarDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
|
|
|
if (D->getSynthesize())
|
|
|
|
OS << " synthesize";
|
|
|
|
|
|
|
|
switch (D->getAccessControl()) {
|
|
|
|
case ObjCIvarDecl::None:
|
|
|
|
OS << " none";
|
|
|
|
break;
|
|
|
|
case ObjCIvarDecl::Private:
|
|
|
|
OS << " private";
|
|
|
|
break;
|
|
|
|
case ObjCIvarDecl::Protected:
|
|
|
|
OS << " protected";
|
|
|
|
break;
|
|
|
|
case ObjCIvarDecl::Public:
|
|
|
|
OS << " public";
|
|
|
|
break;
|
|
|
|
case ObjCIvarDecl::Package:
|
|
|
|
OS << " package";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCMethodDecl(const ObjCMethodDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
if (D->isInstanceMethod())
|
|
|
|
OS << " -";
|
|
|
|
else
|
|
|
|
OS << " +";
|
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getResultType());
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
bool OldMoreChildren = hasMoreChildren();
|
|
|
|
bool IsVariadic = D->isVariadic();
|
|
|
|
bool HasBody = D->hasBody();
|
|
|
|
|
|
|
|
setMoreChildren(OldMoreChildren || IsVariadic || HasBody);
|
|
|
|
if (D->isThisDeclarationADefinition()) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclContext(D);
|
2013-01-31 09:44:26 +08:00
|
|
|
} else {
|
2013-02-01 20:35:51 +08:00
|
|
|
for (ObjCMethodDecl::param_const_iterator I = D->param_begin(),
|
|
|
|
E = D->param_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(*I);
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(OldMoreChildren || HasBody);
|
|
|
|
if (IsVariadic) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "...";
|
|
|
|
}
|
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(OldMoreChildren);
|
|
|
|
if (HasBody) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getBody());
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCCategoryDecl(const ObjCCategoryDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpDeclRef(D->getClassInterface());
|
2013-01-31 09:44:26 +08:00
|
|
|
if (D->protocol_begin() == D->protocol_end())
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getImplementation());
|
|
|
|
for (ObjCCategoryDecl::protocol_iterator I = D->protocol_begin(),
|
2013-02-01 20:35:51 +08:00
|
|
|
E = D->protocol_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpDeclRef(D->getClassInterface());
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getCategoryDecl());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCProtocolDecl(const ObjCProtocolDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
for (ObjCProtocolDecl::protocol_iterator I = D->protocol_begin(),
|
2013-02-01 20:35:51 +08:00
|
|
|
E = D->protocol_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCInterfaceDecl(const ObjCInterfaceDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpDeclRef(D->getSuperClass(), "super");
|
2013-01-31 09:44:26 +08:00
|
|
|
if (D->protocol_begin() == D->protocol_end())
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getImplementation());
|
|
|
|
for (ObjCInterfaceDecl::protocol_iterator I = D->protocol_begin(),
|
2013-02-01 20:35:51 +08:00
|
|
|
E = D->protocol_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCImplementationDecl(const ObjCImplementationDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpDeclRef(D->getSuperClass(), "super");
|
2013-01-31 09:44:26 +08:00
|
|
|
if (D->init_begin() == D->init_end())
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getClassInterface());
|
2013-02-01 20:35:51 +08:00
|
|
|
for (ObjCImplementationDecl::init_const_iterator I = D->init_begin(),
|
|
|
|
E = D->init_end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpCXXCtorInitializer(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCCompatibleAliasDecl(const ObjCCompatibleAliasDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getClassInterface());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCPropertyDecl(const ObjCPropertyDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D);
|
|
|
|
dumpType(D->getType());
|
|
|
|
|
|
|
|
if (D->getPropertyImplementation() == ObjCPropertyDecl::Required)
|
|
|
|
OS << " required";
|
|
|
|
else if (D->getPropertyImplementation() == ObjCPropertyDecl::Optional)
|
|
|
|
OS << " optional";
|
|
|
|
|
|
|
|
ObjCPropertyDecl::PropertyAttributeKind Attrs = D->getPropertyAttributes();
|
|
|
|
if (Attrs != ObjCPropertyDecl::OBJC_PR_noattr) {
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_readonly)
|
|
|
|
OS << " readonly";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_assign)
|
|
|
|
OS << " assign";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_readwrite)
|
|
|
|
OS << " readwrite";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_retain)
|
|
|
|
OS << " retain";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_copy)
|
|
|
|
OS << " copy";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_nonatomic)
|
|
|
|
OS << " nonatomic";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_atomic)
|
|
|
|
OS << " atomic";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_weak)
|
|
|
|
OS << " weak";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_strong)
|
|
|
|
OS << " strong";
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_unsafe_unretained)
|
|
|
|
OS << " unsafe_unretained";
|
2013-01-31 09:44:26 +08:00
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_getter) {
|
|
|
|
if (!(Attrs & ObjCPropertyDecl::OBJC_PR_setter))
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getGetterMethodDecl(), "getter");
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
|
|
|
if (Attrs & ObjCPropertyDecl::OBJC_PR_setter) {
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getSetterMethodDecl(), "setter");
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCPropertyImplDecl(const ObjCPropertyImplDecl *D) {
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpName(D->getPropertyDecl());
|
|
|
|
if (D->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize)
|
|
|
|
OS << " synthesize";
|
|
|
|
else
|
|
|
|
OS << " dynamic";
|
|
|
|
dumpDeclRef(D->getPropertyDecl());
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDeclRef(D->getPropertyIvarDecl());
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitBlockDecl(const BlockDecl *D) {
|
|
|
|
for (BlockDecl::param_const_iterator I = D->param_begin(), E = D->param_end();
|
2012-12-20 10:09:13 +08:00
|
|
|
I != E; ++I)
|
|
|
|
dumpDecl(*I);
|
|
|
|
|
|
|
|
if (D->isVariadic()) {
|
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "...";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (D->capturesCXXThis()) {
|
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "capture this";
|
|
|
|
}
|
2013-02-01 20:35:51 +08:00
|
|
|
for (BlockDecl::capture_iterator I = D->capture_begin(), E = D->capture_end();
|
|
|
|
I != E; ++I) {
|
2012-12-20 10:09:13 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
OS << "capture";
|
|
|
|
if (I->isByRef())
|
|
|
|
OS << " byref";
|
|
|
|
if (I->isNested())
|
|
|
|
OS << " nested";
|
|
|
|
if (I->getVariable()) {
|
|
|
|
OS << ' ';
|
|
|
|
dumpBareDeclRef(I->getVariable());
|
|
|
|
}
|
|
|
|
if (I->hasCopyExpr())
|
|
|
|
dumpStmt(I->getCopyExpr());
|
|
|
|
}
|
2013-01-31 09:44:26 +08:00
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpStmt(D->getBody());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2012-12-11 23:20:44 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-12-11 23:28:09 +08:00
|
|
|
// Stmt dumping methods.
|
2012-12-11 23:20:44 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::dumpStmt(const Stmt *S) {
|
2012-12-11 23:20:44 +08:00
|
|
|
IndentScope Indent(*this);
|
|
|
|
|
|
|
|
if (!S) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, NullColor);
|
2012-12-11 23:20:44 +08:00
|
|
|
OS << "<<<NULL>>>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
if (const DeclStmt *DS = dyn_cast<DeclStmt>(S)) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitDeclStmt(DS);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-05-15 15:37:26 +08:00
|
|
|
setMoreChildren(!S->children().empty());
|
2013-02-01 20:35:51 +08:00
|
|
|
ConstStmtVisitor<ASTDumper>::Visit(S);
|
2013-01-31 09:44:26 +08:00
|
|
|
setMoreChildren(false);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (Stmt::const_child_range CI = S->children(); CI; ++CI) {
|
|
|
|
Stmt::const_child_range Next = CI;
|
2013-01-31 09:44:26 +08:00
|
|
|
++Next;
|
|
|
|
if (!Next)
|
|
|
|
lastChild();
|
2012-12-11 23:20:44 +08:00
|
|
|
dumpStmt(*CI);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2012-12-11 23:20:44 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitStmt(const Stmt *Node) {
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, StmtColor);
|
|
|
|
OS << Node->getStmtClassName();
|
|
|
|
}
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpPointer(Node);
|
|
|
|
dumpSourceRange(Node->getSourceRange());
|
2012-12-11 23:20:44 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitDeclStmt(const DeclStmt *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitStmt(Node);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (DeclStmt::const_decl_iterator I = Node->decl_begin(),
|
|
|
|
E = Node->decl_end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2007-12-12 14:59:42 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitAttributedStmt(const AttributedStmt *Node) {
|
2013-01-08 01:53:08 +08:00
|
|
|
VisitStmt(Node);
|
2013-02-01 20:35:51 +08:00
|
|
|
for (ArrayRef<const Attr *>::iterator I = Node->getAttrs().begin(),
|
|
|
|
E = Node->getAttrs().end();
|
|
|
|
I != E; ++I) {
|
2013-01-31 09:44:26 +08:00
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2013-01-08 01:53:08 +08:00
|
|
|
dumpAttr(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2013-01-08 01:53:08 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitLabelStmt(const LabelStmt *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitStmt(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " '" << Node->getName() << "'";
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitGotoStmt(const GotoStmt *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitStmt(Node);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " '" << Node->getLabel()->getName() << "'";
|
|
|
|
dumpPointer(Node->getLabel());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
2012-12-11 23:28:09 +08:00
|
|
|
// Expr dumping methods.
|
2007-08-09 06:51:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitExpr(const Expr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitStmt(Node);
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpType(Node->getType());
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, ValueKindColor);
|
|
|
|
switch (Node->getValueKind()) {
|
|
|
|
case VK_RValue:
|
|
|
|
break;
|
|
|
|
case VK_LValue:
|
|
|
|
OS << " lvalue";
|
|
|
|
break;
|
|
|
|
case VK_XValue:
|
|
|
|
OS << " xvalue";
|
|
|
|
break;
|
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, ObjectKindColor);
|
|
|
|
switch (Node->getObjectKind()) {
|
|
|
|
case OK_Ordinary:
|
|
|
|
break;
|
|
|
|
case OK_BitField:
|
|
|
|
OS << " bitfield";
|
|
|
|
break;
|
|
|
|
case OK_ObjCProperty:
|
|
|
|
OS << " objcproperty";
|
|
|
|
break;
|
|
|
|
case OK_ObjCSubscript:
|
|
|
|
OS << " objcsubscript";
|
|
|
|
break;
|
|
|
|
case OK_VectorComponent:
|
|
|
|
OS << " vectorcomponent";
|
|
|
|
break;
|
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
}
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
static void dumpBasePath(raw_ostream &OS, const CastExpr *Node) {
|
2010-08-07 14:22:56 +08:00
|
|
|
if (Node->path_empty())
|
2010-04-25 03:06:50 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
OS << " (";
|
|
|
|
bool First = true;
|
2013-02-01 20:35:51 +08:00
|
|
|
for (CastExpr::path_const_iterator I = Node->path_begin(),
|
|
|
|
E = Node->path_end();
|
|
|
|
I != E; ++I) {
|
2010-04-25 03:06:50 +08:00
|
|
|
const CXXBaseSpecifier *Base = *I;
|
|
|
|
if (!First)
|
|
|
|
OS << " -> ";
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2010-04-25 03:06:50 +08:00
|
|
|
const CXXRecordDecl *RD =
|
|
|
|
cast<CXXRecordDecl>(Base->getType()->getAs<RecordType>()->getDecl());
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2010-04-25 03:06:50 +08:00
|
|
|
if (Base->isVirtual())
|
|
|
|
OS << "virtual ";
|
|
|
|
OS << RD->getName();
|
|
|
|
First = false;
|
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2010-04-25 03:06:50 +08:00
|
|
|
OS << ')';
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCastExpr(const CastExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2013-01-26 09:31:20 +08:00
|
|
|
OS << " <";
|
|
|
|
{
|
|
|
|
ColorScope Color(*this, CastColor);
|
|
|
|
OS << Node->getCastKindName();
|
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpBasePath(OS, Node);
|
2010-04-25 03:06:50 +08:00
|
|
|
OS << ">";
|
2009-08-23 07:33:40 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitDeclRefExpr(const DeclRefExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2007-09-11 01:32:55 +08:00
|
|
|
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " ";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpBareDeclRef(Node->getDecl());
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
if (Node->getDecl() != Node->getFoundDecl()) {
|
|
|
|
OS << " (";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpBareDeclRef(Node->getFoundDecl());
|
Add an optional field attached to a DeclRefExpr which points back to the
Decl actually found via name lookup & overload resolution when that Decl
is different from the ValueDecl which is actually referenced by the
expression.
This can be used by AST consumers to correctly attribute references to
the spelling location of a using declaration, and otherwise gain insight
into the name resolution performed by Clang.
The public interface to DRE is kept as narrow as possible: we provide
a getFoundDecl() which always returns a NamedDecl, either the ValueDecl
referenced or the new, more precise NamedDecl if present. This way AST
clients can code against getFoundDecl without know when exactly the AST
has a split representation.
For an example of the data this provides consider:
% cat x.cc
namespace N1 {
struct S {};
void f(const S&);
}
void test(N1::S s) {
f(s);
using N1::f;
f(s);
}
% ./bin/clang -fsyntax-only -Xclang -ast-dump x.cc
[...]
void test(N1::S s) (CompoundStmt 0x5b02010 <x.cc:5:20, line:9:1>
(CallExpr 0x5b01df0 <line:6:3, col:6> 'void'
(ImplicitCastExpr 0x5b01dd8 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01d80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)'))
(ImplicitCastExpr 0x5b01e20 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01d58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S')))
(DeclStmt 0x5b01ee0 <line:7:3, col:14>
0x5b01e40 "UsingN1::;")
(CallExpr 0x5b01fc8 <line:8:3, col:6> 'void'
(ImplicitCastExpr 0x5b01fb0 <col:3> 'void (*)(const struct N1::S &)' <FunctionToPointerDecay>
(DeclRefExpr 0x5b01f80 <col:3> 'void (const struct N1::S &)' lvalue Function 0x5b01a20 'f' 'void (const struct N1::S &)' (UsingShadow 0x5b01ea0 'f')))
(ImplicitCastExpr 0x5b01ff8 <col:5> 'const struct N1::S' lvalue <NoOp>
(DeclRefExpr 0x5b01f58 <col:5> 'N1::S':'struct N1::S' lvalue ParmVar 0x5b01b60 's' 'N1::S':'struct N1::S'))))
Now we can tell that the second call is 'using' (no pun intended) the using
declaration, and *which* using declaration it sees. Without this, we can
mistake calls that go through using declarations for ADL calls, and have no way
to attribute names looked up with using declarations to the appropriate
UsingDecl.
llvm-svn: 130670
2011-05-02 07:48:14 +08:00
|
|
|
OS << ")";
|
|
|
|
}
|
2011-02-07 18:33:21 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUnresolvedLookupExpr(const UnresolvedLookupExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-12 05:50:11 +08:00
|
|
|
OS << " (";
|
2012-12-11 23:28:09 +08:00
|
|
|
if (!Node->requiresADL())
|
|
|
|
OS << "no ";
|
2010-04-17 17:33:03 +08:00
|
|
|
OS << "ADL) = '" << Node->getName() << '\'';
|
2009-12-12 05:50:11 +08:00
|
|
|
|
|
|
|
UnresolvedLookupExpr::decls_iterator
|
|
|
|
I = Node->decls_begin(), E = Node->decls_end();
|
2012-12-11 23:28:09 +08:00
|
|
|
if (I == E)
|
|
|
|
OS << " empty";
|
2009-12-12 05:50:11 +08:00
|
|
|
for (; I != E; ++I)
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpPointer(*I);
|
2009-12-12 05:50:11 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCIvarRefExpr(const ObjCIvarRefExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2008-03-12 21:19:12 +08:00
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, DeclKindNameColor);
|
|
|
|
OS << " " << Node->getDecl()->getDeclKindName() << "Decl";
|
|
|
|
}
|
|
|
|
OS << "='" << *Node->getDecl() << "'";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpPointer(Node->getDecl());
|
2008-05-24 06:01:24 +08:00
|
|
|
if (Node->isFreeIvar())
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " isFreeIvar";
|
2008-03-12 21:19:12 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitPredefinedExpr(const PredefinedExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2007-08-09 06:51:59 +08:00
|
|
|
switch (Node->getIdentType()) {
|
2011-09-23 13:06:16 +08:00
|
|
|
default: llvm_unreachable("unknown case");
|
2009-12-03 17:13:13 +08:00
|
|
|
case PredefinedExpr::Func: OS << " __func__"; break;
|
|
|
|
case PredefinedExpr::Function: OS << " __FUNCTION__"; break;
|
2012-12-11 23:28:09 +08:00
|
|
|
case PredefinedExpr::LFunction: OS << " L__FUNCTION__"; break;
|
2009-12-03 17:13:13 +08:00
|
|
|
case PredefinedExpr::PrettyFunction: OS << " __PRETTY_FUNCTION__";break;
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCharacterLiteral(const CharacterLiteral *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, ValueColor);
|
2011-11-04 07:56:23 +08:00
|
|
|
OS << " " << Node->getValue();
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitIntegerLiteral(const IntegerLiteral *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2007-08-09 06:51:59 +08:00
|
|
|
|
|
|
|
bool isSigned = Node->getType()->isSignedIntegerType();
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, ValueColor);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << Node->getValue().toString(10, isSigned);
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitFloatingLiteral(const FloatingLiteral *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, ValueColor);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << Node->getValueAsApproximateDouble();
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2007-08-26 11:42:43 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitStringLiteral(const StringLiteral *Str) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Str);
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, ValueColor);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " ";
|
2012-06-14 04:25:24 +08:00
|
|
|
Str->outputString(OS);
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2007-08-30 09:00:35 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUnaryOperator(const UnaryOperator *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << (Node->isPostfix() ? "postfix" : "prefix")
|
|
|
|
<< " '" << UnaryOperator::getOpcodeStr(Node->getOpcode()) << "'";
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitUnaryExprOrTypeTraitExpr(
|
|
|
|
const UnaryExprOrTypeTraitExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2011-03-12 03:24:49 +08:00
|
|
|
switch(Node->getKind()) {
|
|
|
|
case UETT_SizeOf:
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " sizeof";
|
2011-03-12 03:24:49 +08:00
|
|
|
break;
|
|
|
|
case UETT_AlignOf:
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " alignof";
|
2011-03-12 03:24:49 +08:00
|
|
|
break;
|
|
|
|
case UETT_VecStep:
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " vec_step";
|
2011-03-12 03:24:49 +08:00
|
|
|
break;
|
|
|
|
}
|
2008-11-12 01:56:53 +08:00
|
|
|
if (Node->isArgumentType())
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpType(Node->getArgumentType());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2007-08-10 01:35:30 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitMemberExpr(const MemberExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " " << (Node->isArrow() ? "->" : ".") << *Node->getMemberDecl();
|
|
|
|
dumpPointer(Node->getMemberDecl());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitExtVectorElementExpr(const ExtVectorElementExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << Node->getAccessor().getNameStart();
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitBinaryOperator(const BinaryOperator *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " '" << BinaryOperator::getOpcodeStr(Node->getOpcode()) << "'";
|
2007-08-25 10:00:02 +08:00
|
|
|
}
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCompoundAssignOperator(
|
|
|
|
const CompoundAssignOperator *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " '" << BinaryOperator::getOpcodeStr(Node->getOpcode())
|
|
|
|
<< "' ComputeLHSTy=";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpBareType(Node->getComputationLHSType());
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " ComputeResultTy=";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpBareType(Node->getComputationResultType());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitBlockExpr(const BlockExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpDecl(Node->getBlockDecl());
|
2011-02-07 18:33:21 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitOpaqueValueExpr(const OpaqueValueExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2011-11-06 17:01:30 +08:00
|
|
|
|
2013-01-31 09:44:26 +08:00
|
|
|
if (Expr *Source = Node->getSourceExpr()) {
|
|
|
|
lastChild();
|
2012-12-11 23:20:44 +08:00
|
|
|
dumpStmt(Source);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2011-11-06 17:01:30 +08:00
|
|
|
}
|
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
// GNU extensions.
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitAddrLabelExpr(const AddrLabelExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << " " << Node->getLabel()->getName();
|
|
|
|
dumpPointer(Node->getLabel());
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2007-08-10 02:03:18 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// C++ Expressions
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-08-09 06:51:59 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXNamedCastExpr(const CXXNamedCastExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-11 23:28:09 +08:00
|
|
|
OS << " " << Node->getCastName()
|
2009-12-03 17:13:13 +08:00
|
|
|
<< "<" << Node->getTypeAsWritten().getAsString() << ">"
|
2010-04-25 03:06:50 +08:00
|
|
|
<< " <" << Node->getCastKindName();
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpBasePath(OS, Node);
|
2010-04-25 03:06:50 +08:00
|
|
|
OS << ">";
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXBoolLiteralExpr(const CXXBoolLiteralExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << (Node->getValue() ? "true" : "false");
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXThisExpr(const CXXThisExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " this";
|
2008-11-04 22:56:14 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXFunctionalCastExpr(const CXXFunctionalCastExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2011-09-03 01:38:59 +08:00
|
|
|
OS << " functional cast to " << Node->getTypeAsWritten().getAsString()
|
|
|
|
<< " <" << Node->getCastKindName() << ">";
|
2008-10-28 03:41:14 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXConstructExpr(const CXXConstructExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2010-02-03 03:03:45 +08:00
|
|
|
CXXConstructorDecl *Ctor = Node->getConstructor();
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpType(Ctor->getType());
|
2009-08-12 08:21:52 +08:00
|
|
|
if (Node->isElidable())
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " elidable";
|
2010-08-07 14:38:55 +08:00
|
|
|
if (Node->requiresZeroInitialization())
|
|
|
|
OS << " zeroing";
|
2009-08-12 08:21:52 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitCXXBindTemporaryExpr(const CXXBindTemporaryExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " ";
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpCXXTemporary(Node->getTemporary());
|
2009-08-12 08:21:52 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitExprWithCleanups(const ExprWithCleanups *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-20 10:09:13 +08:00
|
|
|
for (unsigned i = 0, e = Node->getNumObjects(); i != e; ++i)
|
|
|
|
dumpDeclRef(Node->getObject(i), "cleanup");
|
2009-08-12 08:21:52 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::dumpCXXTemporary(const CXXTemporary *Temporary) {
|
2012-12-20 10:09:13 +08:00
|
|
|
OS << "(CXXTemporary";
|
|
|
|
dumpPointer(Temporary);
|
|
|
|
OS << ")";
|
2009-08-12 08:21:52 +08:00
|
|
|
}
|
|
|
|
|
2007-08-22 01:43:55 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Obj-C Expressions
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCMessageExpr(const ObjCMessageExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " selector=" << Node->getSelector().getAsString();
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
switch (Node->getReceiverKind()) {
|
|
|
|
case ObjCMessageExpr::Instance:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjCMessageExpr::Class:
|
|
|
|
OS << " class=";
|
2012-12-20 10:09:13 +08:00
|
|
|
dumpBareType(Node->getClassReceiver());
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjCMessageExpr::SuperInstance:
|
|
|
|
OS << " super (instance)";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjCMessageExpr::SuperClass:
|
|
|
|
OS << " super (class)";
|
|
|
|
break;
|
|
|
|
}
|
2008-03-01 06:04:05 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCBoxedExpr(const ObjCBoxedExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-05-11 04:02:31 +08:00
|
|
|
OS << " selector=" << Node->getBoxingMethod()->getSelector().getAsString();
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCAtCatchStmt(const ObjCAtCatchStmt *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitStmt(Node);
|
2013-02-01 20:35:51 +08:00
|
|
|
if (const VarDecl *CatchParam = Node->getCatchParamDecl())
|
2012-12-11 23:20:44 +08:00
|
|
|
dumpDecl(CatchParam);
|
2012-12-20 10:09:13 +08:00
|
|
|
else
|
2010-04-24 06:50:49 +08:00
|
|
|
OS << " catch all";
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCEncodeExpr(const ObjCEncodeExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-12-11 23:28:09 +08:00
|
|
|
dumpType(Node->getEncodedType());
|
2007-08-22 23:14:15 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCSelectorExpr(const ObjCSelectorExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-12-03 17:13:13 +08:00
|
|
|
OS << " " << Node->getSelector().getAsString();
|
2007-10-17 04:40:23 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCProtocolExpr(const ObjCProtocolExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-12-11 23:28:09 +08:00
|
|
|
OS << ' ' << *Node->getProtocol();
|
2007-10-18 00:58:11 +08:00
|
|
|
}
|
2008-08-30 13:35:15 +08:00
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCPropertyRefExpr(const ObjCPropertyRefExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2010-12-02 09:19:52 +08:00
|
|
|
if (Node->isImplicitProperty()) {
|
2010-12-23 03:46:35 +08:00
|
|
|
OS << " Kind=MethodRef Getter=\"";
|
|
|
|
if (Node->getImplicitPropertyGetter())
|
|
|
|
OS << Node->getImplicitPropertyGetter()->getSelector().getAsString();
|
|
|
|
else
|
|
|
|
OS << "(null)";
|
|
|
|
|
|
|
|
OS << "\" Setter=\"";
|
2010-12-02 09:19:52 +08:00
|
|
|
if (ObjCMethodDecl *Setter = Node->getImplicitPropertySetter())
|
|
|
|
OS << Setter->getSelector().getAsString();
|
|
|
|
else
|
|
|
|
OS << "(null)";
|
|
|
|
OS << "\"";
|
|
|
|
} else {
|
2011-10-15 02:45:37 +08:00
|
|
|
OS << " Kind=PropertyRef Property=\"" << *Node->getExplicitProperty() <<'"';
|
2010-12-02 09:19:52 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-10-15 00:04:05 +08:00
|
|
|
if (Node->isSuperReceiver())
|
|
|
|
OS << " super";
|
2012-03-30 08:19:18 +08:00
|
|
|
|
|
|
|
OS << " Messaging=";
|
|
|
|
if (Node->isMessagingGetter() && Node->isMessagingSetter())
|
|
|
|
OS << "Getter&Setter";
|
|
|
|
else if (Node->isMessagingGetter())
|
|
|
|
OS << "Getter";
|
|
|
|
else if (Node->isMessagingSetter())
|
|
|
|
OS << "Setter";
|
2008-11-04 22:56:14 +08:00
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCSubscriptRefExpr(const ObjCSubscriptRefExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Node->isArraySubscriptRefExpr())
|
|
|
|
OS << " Kind=ArraySubscript GetterForArray=\"";
|
|
|
|
else
|
|
|
|
OS << " Kind=DictionarySubscript GetterForDictionary=\"";
|
|
|
|
if (Node->getAtIndexMethodDecl())
|
|
|
|
OS << Node->getAtIndexMethodDecl()->getSelector().getAsString();
|
|
|
|
else
|
|
|
|
OS << "(null)";
|
2012-12-11 23:28:09 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Node->isArraySubscriptRefExpr())
|
|
|
|
OS << "\" SetterForArray=\"";
|
|
|
|
else
|
|
|
|
OS << "\" SetterForDictionary=\"";
|
|
|
|
if (Node->setAtIndexMethodDecl())
|
|
|
|
OS << Node->setAtIndexMethodDecl()->getSelector().getAsString();
|
|
|
|
else
|
|
|
|
OS << "(null)";
|
|
|
|
}
|
|
|
|
|
2013-02-01 20:35:51 +08:00
|
|
|
void ASTDumper::VisitObjCBoolLiteralExpr(const ObjCBoolLiteralExpr *Node) {
|
2012-12-11 23:20:44 +08:00
|
|
|
VisitExpr(Node);
|
2012-03-07 04:05:56 +08:00
|
|
|
OS << " " << (Node->getValue() ? "__objc_yes" : "__objc_no");
|
|
|
|
}
|
|
|
|
|
2013-01-14 22:07:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Comments
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
const char *ASTDumper::getCommandName(unsigned CommandID) {
|
|
|
|
if (Traits)
|
|
|
|
return Traits->getCommandInfo(CommandID)->Name;
|
|
|
|
const CommandInfo *Info = CommandTraits::getBuiltinCommandInfo(CommandID);
|
|
|
|
if (Info)
|
|
|
|
return Info->Name;
|
|
|
|
return "<not a builtin command>";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpFullComment(const FullComment *C) {
|
|
|
|
if (!C)
|
|
|
|
return;
|
|
|
|
|
|
|
|
FC = C;
|
|
|
|
dumpComment(C);
|
|
|
|
FC = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::dumpComment(const Comment *C) {
|
|
|
|
IndentScope Indent(*this);
|
|
|
|
|
|
|
|
if (!C) {
|
2013-01-26 09:31:20 +08:00
|
|
|
ColorScope Color(*this, NullColor);
|
2013-01-14 22:07:11 +08:00
|
|
|
OS << "<<<NULL>>>";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
{
|
|
|
|
ColorScope Color(*this, CommentColor);
|
|
|
|
OS << C->getCommentKindName();
|
|
|
|
}
|
2013-01-14 22:07:11 +08:00
|
|
|
dumpPointer(C);
|
|
|
|
dumpSourceRange(C->getSourceRange());
|
|
|
|
ConstCommentVisitor<ASTDumper>::visit(C);
|
|
|
|
for (Comment::child_iterator I = C->child_begin(), E = C->child_end();
|
2013-01-31 09:44:26 +08:00
|
|
|
I != E; ++I) {
|
|
|
|
if (I + 1 == E)
|
|
|
|
lastChild();
|
2013-01-14 22:07:11 +08:00
|
|
|
dumpComment(*I);
|
2013-01-31 09:44:26 +08:00
|
|
|
}
|
2013-01-14 22:07:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitTextComment(const TextComment *C) {
|
|
|
|
OS << " Text=\"" << C->getText() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitInlineCommandComment(const InlineCommandComment *C) {
|
|
|
|
OS << " Name=\"" << getCommandName(C->getCommandID()) << "\"";
|
|
|
|
switch (C->getRenderKind()) {
|
|
|
|
case InlineCommandComment::RenderNormal:
|
|
|
|
OS << " RenderNormal";
|
|
|
|
break;
|
|
|
|
case InlineCommandComment::RenderBold:
|
|
|
|
OS << " RenderBold";
|
|
|
|
break;
|
|
|
|
case InlineCommandComment::RenderMonospaced:
|
|
|
|
OS << " RenderMonospaced";
|
|
|
|
break;
|
|
|
|
case InlineCommandComment::RenderEmphasized:
|
|
|
|
OS << " RenderEmphasized";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = C->getNumArgs(); i != e; ++i)
|
|
|
|
OS << " Arg[" << i << "]=\"" << C->getArgText(i) << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitHTMLStartTagComment(const HTMLStartTagComment *C) {
|
|
|
|
OS << " Name=\"" << C->getTagName() << "\"";
|
|
|
|
if (C->getNumAttrs() != 0) {
|
|
|
|
OS << " Attrs: ";
|
|
|
|
for (unsigned i = 0, e = C->getNumAttrs(); i != e; ++i) {
|
|
|
|
const HTMLStartTagComment::Attribute &Attr = C->getAttr(i);
|
|
|
|
OS << " \"" << Attr.Name << "=\"" << Attr.Value << "\"";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (C->isSelfClosing())
|
|
|
|
OS << " SelfClosing";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitHTMLEndTagComment(const HTMLEndTagComment *C) {
|
|
|
|
OS << " Name=\"" << C->getTagName() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitBlockCommandComment(const BlockCommandComment *C) {
|
|
|
|
OS << " Name=\"" << getCommandName(C->getCommandID()) << "\"";
|
|
|
|
for (unsigned i = 0, e = C->getNumArgs(); i != e; ++i)
|
|
|
|
OS << " Arg[" << i << "]=\"" << C->getArgText(i) << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitParamCommandComment(const ParamCommandComment *C) {
|
|
|
|
OS << " " << ParamCommandComment::getDirectionAsString(C->getDirection());
|
|
|
|
|
|
|
|
if (C->isDirectionExplicit())
|
|
|
|
OS << " explicitly";
|
|
|
|
else
|
|
|
|
OS << " implicitly";
|
|
|
|
|
|
|
|
if (C->hasParamName()) {
|
|
|
|
if (C->isParamIndexValid())
|
|
|
|
OS << " Param=\"" << C->getParamName(FC) << "\"";
|
|
|
|
else
|
|
|
|
OS << " Param=\"" << C->getParamNameAsWritten() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (C->isParamIndexValid())
|
|
|
|
OS << " ParamIndex=" << C->getParamIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitTParamCommandComment(const TParamCommandComment *C) {
|
|
|
|
if (C->hasParamName()) {
|
|
|
|
if (C->isPositionValid())
|
|
|
|
OS << " Param=\"" << C->getParamName(FC) << "\"";
|
|
|
|
else
|
|
|
|
OS << " Param=\"" << C->getParamNameAsWritten() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (C->isPositionValid()) {
|
|
|
|
OS << " Position=<";
|
|
|
|
for (unsigned i = 0, e = C->getDepth(); i != e; ++i) {
|
|
|
|
OS << C->getIndex(i);
|
|
|
|
if (i != e - 1)
|
|
|
|
OS << ", ";
|
|
|
|
}
|
|
|
|
OS << ">";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitVerbatimBlockComment(const VerbatimBlockComment *C) {
|
|
|
|
OS << " Name=\"" << getCommandName(C->getCommandID()) << "\""
|
|
|
|
" CloseName=\"" << C->getCloseName() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitVerbatimBlockLineComment(
|
|
|
|
const VerbatimBlockLineComment *C) {
|
|
|
|
OS << " Text=\"" << C->getText() << "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void ASTDumper::visitVerbatimLineComment(const VerbatimLineComment *C) {
|
|
|
|
OS << " Text=\"" << C->getText() << "\"";
|
|
|
|
}
|
|
|
|
|
2012-12-20 10:09:13 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Decl method implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void Decl::dump() const {
|
|
|
|
dump(llvm::errs());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Decl::dump(raw_ostream &OS) const {
|
2013-01-14 22:07:11 +08:00
|
|
|
ASTDumper P(OS, &getASTContext().getCommentCommandTraits(),
|
|
|
|
&getASTContext().getSourceManager());
|
2013-02-01 20:35:51 +08:00
|
|
|
P.dumpDecl(this);
|
2012-12-20 10:09:13 +08:00
|
|
|
}
|
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
void Decl::dumpColor() const {
|
|
|
|
ASTDumper P(llvm::errs(), &getASTContext().getCommentCommandTraits(),
|
|
|
|
&getASTContext().getSourceManager(), /*ShowColors*/true);
|
2013-02-01 20:35:51 +08:00
|
|
|
P.dumpDecl(this);
|
2013-01-26 09:31:20 +08:00
|
|
|
}
|
2007-08-09 06:51:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Stmt method implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2007-08-30 14:17:34 +08:00
|
|
|
void Stmt::dump(SourceManager &SM) const {
|
2010-08-09 18:54:31 +08:00
|
|
|
dump(llvm::errs(), SM);
|
|
|
|
}
|
|
|
|
|
2011-07-23 18:55:15 +08:00
|
|
|
void Stmt::dump(raw_ostream &OS, SourceManager &SM) const {
|
2013-01-14 22:07:11 +08:00
|
|
|
ASTDumper P(OS, 0, &SM);
|
2013-02-01 20:35:51 +08:00
|
|
|
P.dumpStmt(this);
|
2007-08-30 08:40:08 +08:00
|
|
|
}
|
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
void Stmt::dump() const {
|
2013-01-14 22:07:11 +08:00
|
|
|
ASTDumper P(llvm::errs(), 0, 0);
|
2013-02-01 20:35:51 +08:00
|
|
|
P.dumpStmt(this);
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2013-01-14 22:07:11 +08:00
|
|
|
|
2013-01-26 09:31:20 +08:00
|
|
|
void Stmt::dumpColor() const {
|
|
|
|
ASTDumper P(llvm::errs(), 0, 0, /*ShowColors*/true);
|
2013-02-01 20:35:51 +08:00
|
|
|
P.dumpStmt(this);
|
2013-01-26 09:31:20 +08:00
|
|
|
}
|
|
|
|
|
2013-01-14 22:07:11 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Comment method implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
void Comment::dump() const {
|
|
|
|
dump(llvm::errs(), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Comment::dump(const ASTContext &Context) const {
|
|
|
|
dump(llvm::errs(), &Context.getCommentCommandTraits(),
|
|
|
|
&Context.getSourceManager());
|
|
|
|
}
|
|
|
|
|
2013-01-15 20:20:21 +08:00
|
|
|
void Comment::dump(raw_ostream &OS, const CommandTraits *Traits,
|
2013-01-14 22:07:11 +08:00
|
|
|
const SourceManager *SM) const {
|
|
|
|
const FullComment *FC = dyn_cast<FullComment>(this);
|
|
|
|
ASTDumper D(OS, Traits, SM);
|
|
|
|
D.dumpFullComment(FC);
|
|
|
|
}
|
2013-01-26 09:31:20 +08:00
|
|
|
|
|
|
|
void Comment::dumpColor() const {
|
|
|
|
const FullComment *FC = dyn_cast<FullComment>(this);
|
|
|
|
ASTDumper D(llvm::errs(), 0, 0, /*ShowColors*/true);
|
|
|
|
D.dumpFullComment(FC);
|
|
|
|
}
|