blob: 7ff61e0a27f15eae0760fca7dac1be19fb6b105c [file] [log] [blame]
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +00001//===--- CommentDumper.cpp - Dumping implementation for Comment ASTs ------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "clang/AST/CommentVisitor.h"
11#include "llvm/Support/raw_ostream.h"
12
13namespace clang {
14namespace comments {
15
16namespace {
17class CommentDumper: public comments::ConstCommentVisitor<CommentDumper> {
18 raw_ostream &OS;
19 SourceManager *SM;
20 unsigned IndentLevel;
21
22public:
23 CommentDumper(raw_ostream &OS, SourceManager *SM) :
24 OS(OS), SM(SM), IndentLevel(0)
25 { }
26
27 void dumpIndent() const {
28 for (unsigned i = 1, e = IndentLevel; i < e; ++i)
29 OS << " ";
30 }
31
32 void dumpLocation(SourceLocation Loc) {
33 if (SM)
34 Loc.print(OS, *SM);
35 }
36
37 void dumpSourceRange(const Comment *C);
38
39 void dumpComment(const Comment *C);
40
41 void dumpSubtree(const Comment *C);
42
43 // Inline content.
44 void visitTextComment(const TextComment *C);
45 void visitInlineCommandComment(const InlineCommandComment *C);
Dmitri Gribenko3f38bf22012-07-13 00:44:24 +000046 void visitHTMLStartTagComment(const HTMLStartTagComment *C);
47 void visitHTMLEndTagComment(const HTMLEndTagComment *C);
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +000048
49 // Block content.
50 void visitParagraphComment(const ParagraphComment *C);
51 void visitBlockCommandComment(const BlockCommandComment *C);
52 void visitParamCommandComment(const ParamCommandComment *C);
53 void visitVerbatimBlockComment(const VerbatimBlockComment *C);
54 void visitVerbatimBlockLineComment(const VerbatimBlockLineComment *C);
55 void visitVerbatimLineComment(const VerbatimLineComment *C);
56
57 void visitFullComment(const FullComment *C);
58};
59
60void CommentDumper::dumpSourceRange(const Comment *C) {
61 if (!SM)
62 return;
63
64 SourceRange SR = C->getSourceRange();
65
66 OS << " <";
67 dumpLocation(SR.getBegin());
68 if (SR.getBegin() != SR.getEnd()) {
69 OS << ", ";
70 dumpLocation(SR.getEnd());
71 }
72 OS << ">";
73}
74
75void CommentDumper::dumpComment(const Comment *C) {
76 dumpIndent();
77 OS << "(" << C->getCommentKindName()
78 << " " << (void *) C;
79 dumpSourceRange(C);
80}
81
82void CommentDumper::dumpSubtree(const Comment *C) {
83 ++IndentLevel;
84 if (C) {
85 visit(C);
86 for (Comment::child_iterator I = C->child_begin(),
87 E = C->child_end();
88 I != E; ++I) {
89 OS << '\n';
90 dumpSubtree(*I);
91 }
92 OS << ')';
93 } else {
94 dumpIndent();
95 OS << "<<<NULL>>>";
96 }
97 --IndentLevel;
98}
99
100void CommentDumper::visitTextComment(const TextComment *C) {
101 dumpComment(C);
102
103 OS << " Text=\"" << C->getText() << "\"";
104}
105
106void CommentDumper::visitInlineCommandComment(const InlineCommandComment *C) {
107 dumpComment(C);
108
109 for (unsigned i = 0, e = C->getArgCount(); i != e; ++i)
110 OS << " Arg[" << i << "]=\"" << C->getArgText(i) << "\"";
111}
112
Dmitri Gribenko3f38bf22012-07-13 00:44:24 +0000113void CommentDumper::visitHTMLStartTagComment(const HTMLStartTagComment *C) {
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +0000114 dumpComment(C);
115
116 OS << " Name=\"" << C->getTagName() << "\"";
117 if (C->getAttrCount() != 0) {
118 OS << " Attrs: ";
119 for (unsigned i = 0, e = C->getAttrCount(); i != e; ++i) {
Dmitri Gribenko3f38bf22012-07-13 00:44:24 +0000120 const HTMLStartTagComment::Attribute &Attr = C->getAttr(i);
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +0000121 OS << " \"" << Attr.Name << "=\"" << Attr.Value << "\"";
122 }
123 }
Dmitri Gribenkoa5ef44f2012-07-11 21:38:39 +0000124 if (C->isSelfClosing())
125 OS << " SelfClosing";
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +0000126}
127
Dmitri Gribenko3f38bf22012-07-13 00:44:24 +0000128void CommentDumper::visitHTMLEndTagComment(const HTMLEndTagComment *C) {
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +0000129 dumpComment(C);
130
131 OS << " Name=\"" << C->getTagName() << "\"";
132}
133
134void CommentDumper::visitParagraphComment(const ParagraphComment *C) {
135 dumpComment(C);
136}
137
138void CommentDumper::visitBlockCommandComment(const BlockCommandComment *C) {
139 dumpComment(C);
140
141 OS << " Name=\"" << C->getCommandName() << "\"";
142}
143
144void CommentDumper::visitParamCommandComment(const ParamCommandComment *C) {
145 dumpComment(C);
146
Dmitri Gribenkoa5ef44f2012-07-11 21:38:39 +0000147 OS << " " << ParamCommandComment::getDirectionAsString(C->getDirection());
Dmitri Gribenko8d3ba232012-07-06 00:28:32 +0000148
149 if (C->isDirectionExplicit())
150 OS << " explicitly";
151 else
152 OS << " implicitly";
153
154 if (C->hasParamName()) {
155 OS << " Param=\"" << C->getParamName() << "\"";
156 }
157}
158
159void CommentDumper::visitVerbatimBlockComment(const VerbatimBlockComment *C) {
160 dumpComment(C);
161
162 OS << " Name=\"" << C->getCommandName() << "\""
163 " CloseName=\"" << C->getCloseName() << "\"";
164}
165
166void CommentDumper::visitVerbatimBlockLineComment(const VerbatimBlockLineComment *C) {
167 dumpComment(C);
168
169 OS << " Text=\"" << C->getText() << "\"";
170}
171
172void CommentDumper::visitVerbatimLineComment(const VerbatimLineComment *C) {
173 dumpComment(C);
174
175 OS << " Text=\"" << C->getText() << "\"";
176}
177
178void CommentDumper::visitFullComment(const FullComment *C) {
179 dumpComment(C);
180}
181
182} // unnamed namespace
183
184void Comment::dump() const {
185 CommentDumper D(llvm::errs(), NULL);
186 D.dumpSubtree(this);
187 llvm::errs() << '\n';
188}
189
190void Comment::dump(SourceManager &SM) const {
191 CommentDumper D(llvm::errs(), &SM);
192 D.dumpSubtree(this);
193 llvm::errs() << '\n';
194}
195
196} // end namespace comments
197} // end namespace clang
198