forked from OSchip/llvm-project
1176 lines
33 KiB
C++
1176 lines
33 KiB
C++
//===- unittests/AST/CommentParser.cpp ------ Comment parser tests --------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Basic/SourceManager.h"
|
|
#include "clang/Basic/FileManager.h"
|
|
#include "clang/Basic/Diagnostic.h"
|
|
#include "clang/AST/Comment.h"
|
|
#include "clang/AST/CommentLexer.h"
|
|
#include "clang/AST/CommentParser.h"
|
|
#include "clang/AST/CommentSema.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/Support/Allocator.h"
|
|
#include <vector>
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
using namespace llvm;
|
|
using namespace clang;
|
|
|
|
namespace clang {
|
|
namespace comments {
|
|
|
|
namespace {
|
|
|
|
const bool DEBUG = true;
|
|
|
|
class CommentParserTest : public ::testing::Test {
|
|
protected:
|
|
CommentParserTest()
|
|
: FileMgr(FileMgrOpts),
|
|
DiagID(new DiagnosticIDs()),
|
|
Diags(DiagID, new IgnoringDiagConsumer()),
|
|
SourceMgr(Diags, FileMgr) {
|
|
}
|
|
|
|
FileSystemOptions FileMgrOpts;
|
|
FileManager FileMgr;
|
|
IntrusiveRefCntPtr<DiagnosticIDs> DiagID;
|
|
DiagnosticsEngine Diags;
|
|
SourceManager SourceMgr;
|
|
llvm::BumpPtrAllocator Allocator;
|
|
|
|
FullComment *parseString(const char *Source);
|
|
};
|
|
|
|
FullComment *CommentParserTest::parseString(const char *Source) {
|
|
MemoryBuffer *Buf = MemoryBuffer::getMemBuffer(Source);
|
|
FileID File = SourceMgr.createFileIDForMemBuffer(Buf);
|
|
SourceLocation Begin = SourceMgr.getLocForStartOfFile(File);
|
|
|
|
comments::Lexer L(Begin, CommentOptions(),
|
|
Source, Source + strlen(Source));
|
|
|
|
comments::Sema S(Allocator, SourceMgr, Diags);
|
|
comments::Parser P(L, S, Allocator, SourceMgr, Diags);
|
|
comments::FullComment *FC = P.parseFullComment();
|
|
|
|
if (DEBUG) {
|
|
llvm::errs() << "=== Source:\n" << Source << "\n=== AST:\n";
|
|
FC->dump(SourceMgr);
|
|
}
|
|
|
|
Token Tok;
|
|
L.lex(Tok);
|
|
if (Tok.is(tok::eof))
|
|
return FC;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
::testing::AssertionResult HasChildCount(const Comment *C, size_t Count) {
|
|
if (!C)
|
|
return ::testing::AssertionFailure() << "Comment is NULL";
|
|
|
|
if (Count != C->child_count())
|
|
return ::testing::AssertionFailure()
|
|
<< "Count = " << Count
|
|
<< ", child_count = " << C->child_count();
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
template <typename T>
|
|
::testing::AssertionResult GetChildAt(const Comment *C,
|
|
size_t Idx,
|
|
T *&Child) {
|
|
if (!C)
|
|
return ::testing::AssertionFailure() << "Comment is NULL";
|
|
|
|
if (Idx >= C->child_count())
|
|
return ::testing::AssertionFailure()
|
|
<< "Idx out of range. Idx = " << Idx
|
|
<< ", child_count = " << C->child_count();
|
|
|
|
Comment::child_iterator I = C->child_begin() + Idx;
|
|
Comment *CommentChild = *I;
|
|
if (!CommentChild)
|
|
return ::testing::AssertionFailure() << "Child is NULL";
|
|
|
|
Child = dyn_cast<T>(CommentChild);
|
|
if (!Child)
|
|
return ::testing::AssertionFailure()
|
|
<< "Child is not of requested type, but a "
|
|
<< CommentChild->getCommentKindName();
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasTextAt(const Comment *C,
|
|
size_t Idx,
|
|
StringRef Text) {
|
|
TextComment *TC;
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, TC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualText = TC->getText();
|
|
if (ActualText != Text)
|
|
return ::testing::AssertionFailure()
|
|
<< "TextComment has text \"" << ActualText.str() << "\", "
|
|
"expected \"" << Text.str() << "\"";
|
|
|
|
if (TC->hasTrailingNewline())
|
|
return ::testing::AssertionFailure()
|
|
<< "TextComment has a trailing newline";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasTextWithNewlineAt(const Comment *C,
|
|
size_t Idx,
|
|
StringRef Text) {
|
|
TextComment *TC;
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, TC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualText = TC->getText();
|
|
if (ActualText != Text)
|
|
return ::testing::AssertionFailure()
|
|
<< "TextComment has text \"" << ActualText.str() << "\", "
|
|
"expected \"" << Text.str() << "\"";
|
|
|
|
if (!TC->hasTrailingNewline())
|
|
return ::testing::AssertionFailure()
|
|
<< "TextComment has no trailing newline";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasBlockCommandAt(const Comment *C,
|
|
size_t Idx,
|
|
BlockCommandComment *&BCC,
|
|
StringRef Name,
|
|
ParagraphComment *&Paragraph) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, BCC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualName = BCC->getCommandName();
|
|
if (ActualName != Name)
|
|
return ::testing::AssertionFailure()
|
|
<< "BlockCommandComment has name \"" << ActualName.str() << "\", "
|
|
"expected \"" << Name.str() << "\"";
|
|
|
|
Paragraph = BCC->getParagraph();
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasParamCommandAt(
|
|
const Comment *C,
|
|
size_t Idx,
|
|
ParamCommandComment *&PCC,
|
|
StringRef CommandName,
|
|
ParamCommandComment::PassDirection Direction,
|
|
bool IsDirectionExplicit,
|
|
StringRef ParamName,
|
|
ParagraphComment *&Paragraph) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, PCC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualCommandName = PCC->getCommandName();
|
|
if (ActualCommandName != CommandName)
|
|
return ::testing::AssertionFailure()
|
|
<< "ParamCommandComment has name \"" << ActualCommandName.str() << "\", "
|
|
"expected \"" << CommandName.str() << "\"";
|
|
|
|
if (PCC->getDirection() != Direction)
|
|
return ::testing::AssertionFailure()
|
|
<< "ParamCommandComment has direction " << PCC->getDirection() << ", "
|
|
"expected " << Direction;
|
|
|
|
if (PCC->isDirectionExplicit() != IsDirectionExplicit)
|
|
return ::testing::AssertionFailure()
|
|
<< "ParamCommandComment has "
|
|
<< (PCC->isDirectionExplicit() ? "explicit" : "implicit")
|
|
<< " direction, "
|
|
"expected " << (IsDirectionExplicit ? "explicit" : "implicit");
|
|
|
|
StringRef ActualParamName = PCC->getParamName();
|
|
if (ActualParamName != ParamName)
|
|
return ::testing::AssertionFailure()
|
|
<< "ParamCommandComment has name \"" << ActualParamName.str() << "\", "
|
|
"expected \"" << ParamName.str() << "\"";
|
|
|
|
Paragraph = PCC->getParagraph();
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasInlineCommandAt(const Comment *C,
|
|
size_t Idx,
|
|
InlineCommandComment *&ICC,
|
|
StringRef Name) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, ICC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualName = ICC->getCommandName();
|
|
if (ActualName != Name)
|
|
return ::testing::AssertionFailure()
|
|
<< "InlineCommandComment has name \"" << ActualName.str() << "\", "
|
|
"expected \"" << Name.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
struct NoArgs {};
|
|
|
|
::testing::AssertionResult HasInlineCommandAt(const Comment *C,
|
|
size_t Idx,
|
|
InlineCommandComment *&ICC,
|
|
StringRef Name,
|
|
NoArgs) {
|
|
::testing::AssertionResult AR = HasInlineCommandAt(C, Idx, ICC, Name);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (ICC->getNumArgs() != 0)
|
|
return ::testing::AssertionFailure()
|
|
<< "InlineCommandComment has " << ICC->getNumArgs() << " arg(s), "
|
|
"expected 0";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasInlineCommandAt(const Comment *C,
|
|
size_t Idx,
|
|
InlineCommandComment *&ICC,
|
|
StringRef Name,
|
|
StringRef Arg) {
|
|
::testing::AssertionResult AR = HasInlineCommandAt(C, Idx, ICC, Name);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (ICC->getNumArgs() != 1)
|
|
return ::testing::AssertionFailure()
|
|
<< "InlineCommandComment has " << ICC->getNumArgs() << " arg(s), "
|
|
"expected 1";
|
|
|
|
StringRef ActualArg = ICC->getArgText(0);
|
|
if (ActualArg != Arg)
|
|
return ::testing::AssertionFailure()
|
|
<< "InlineCommandComment has argument \"" << ActualArg.str() << "\", "
|
|
"expected \"" << Arg.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasHTMLStartTagAt(const Comment *C,
|
|
size_t Idx,
|
|
HTMLStartTagComment *&HST,
|
|
StringRef TagName) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, HST);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualTagName = HST->getTagName();
|
|
if (ActualTagName != TagName)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment has name \"" << ActualTagName.str() << "\", "
|
|
"expected \"" << TagName.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
struct SelfClosing {};
|
|
|
|
::testing::AssertionResult HasHTMLStartTagAt(const Comment *C,
|
|
size_t Idx,
|
|
HTMLStartTagComment *&HST,
|
|
StringRef TagName,
|
|
SelfClosing) {
|
|
::testing::AssertionResult AR = HasHTMLStartTagAt(C, Idx, HST, TagName);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (!HST->isSelfClosing())
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment is not self-closing";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
|
|
struct NoAttrs {};
|
|
|
|
::testing::AssertionResult HasHTMLStartTagAt(const Comment *C,
|
|
size_t Idx,
|
|
HTMLStartTagComment *&HST,
|
|
StringRef TagName,
|
|
NoAttrs) {
|
|
::testing::AssertionResult AR = HasHTMLStartTagAt(C, Idx, HST, TagName);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (HST->isSelfClosing())
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment is self-closing";
|
|
|
|
if (HST->getNumAttrs() != 0)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment has " << HST->getNumAttrs() << " attr(s), "
|
|
"expected 0";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasHTMLStartTagAt(const Comment *C,
|
|
size_t Idx,
|
|
HTMLStartTagComment *&HST,
|
|
StringRef TagName,
|
|
StringRef AttrName,
|
|
StringRef AttrValue) {
|
|
::testing::AssertionResult AR = HasHTMLStartTagAt(C, Idx, HST, TagName);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (HST->isSelfClosing())
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment is self-closing";
|
|
|
|
if (HST->getNumAttrs() != 1)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment has " << HST->getNumAttrs() << " attr(s), "
|
|
"expected 1";
|
|
|
|
StringRef ActualName = HST->getAttr(0).Name;
|
|
if (ActualName != AttrName)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment has attr \"" << ActualName.str() << "\", "
|
|
"expected \"" << AttrName.str() << "\"";
|
|
|
|
StringRef ActualValue = HST->getAttr(0).Value;
|
|
if (ActualValue != AttrValue)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLStartTagComment has attr value \"" << ActualValue.str() << "\", "
|
|
"expected \"" << AttrValue.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasHTMLEndTagAt(const Comment *C,
|
|
size_t Idx,
|
|
HTMLEndTagComment *&HET,
|
|
StringRef TagName) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, HET);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualTagName = HET->getTagName();
|
|
if (ActualTagName != TagName)
|
|
return ::testing::AssertionFailure()
|
|
<< "HTMLEndTagComment has name \"" << ActualTagName.str() << "\", "
|
|
"expected \"" << TagName.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasVerbatimBlockAt(const Comment *C,
|
|
size_t Idx,
|
|
VerbatimBlockComment *&VBC,
|
|
StringRef Name) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, VBC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualName = VBC->getCommandName();
|
|
if (ActualName != Name)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has name \"" << ActualName.str() << "\", "
|
|
"expected \"" << Name.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
struct NoLines {};
|
|
|
|
::testing::AssertionResult HasVerbatimBlockAt(const Comment *C,
|
|
size_t Idx,
|
|
VerbatimBlockComment *&VBC,
|
|
StringRef Name,
|
|
NoLines) {
|
|
::testing::AssertionResult AR = HasVerbatimBlockAt(C, Idx, VBC, Name);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (VBC->getNumLines() != 0)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has " << VBC->getNumLines() << " lines(s), "
|
|
"expected 0";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasVerbatimBlockAt(const Comment *C,
|
|
size_t Idx,
|
|
VerbatimBlockComment *&VBC,
|
|
StringRef Name,
|
|
StringRef Line0) {
|
|
::testing::AssertionResult AR = HasVerbatimBlockAt(C, Idx, VBC, Name);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (VBC->getNumLines() != 1)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has " << VBC->getNumLines() << " lines(s), "
|
|
"expected 1";
|
|
|
|
StringRef ActualLine0 = VBC->getText(0);
|
|
if (ActualLine0 != Line0)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has lines[0] \"" << ActualLine0.str() << "\", "
|
|
"expected \"" << Line0.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasVerbatimBlockAt(const Comment *C,
|
|
size_t Idx,
|
|
VerbatimBlockComment *&VBC,
|
|
StringRef Name,
|
|
StringRef Line0,
|
|
StringRef Line1) {
|
|
::testing::AssertionResult AR = HasVerbatimBlockAt(C, Idx, VBC, Name);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
if (VBC->getNumLines() != 2)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has " << VBC->getNumLines() << " lines(s), "
|
|
"expected 2";
|
|
|
|
StringRef ActualLine0 = VBC->getText(0);
|
|
if (ActualLine0 != Line0)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has lines[0] \"" << ActualLine0.str() << "\", "
|
|
"expected \"" << Line0.str() << "\"";
|
|
|
|
StringRef ActualLine1 = VBC->getText(1);
|
|
if (ActualLine1 != Line1)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimBlockComment has lines[1] \"" << ActualLine1.str() << "\", "
|
|
"expected \"" << Line1.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
::testing::AssertionResult HasVerbatimLineAt(const Comment *C,
|
|
size_t Idx,
|
|
VerbatimLineComment *&VLC,
|
|
StringRef Name,
|
|
StringRef Text) {
|
|
::testing::AssertionResult AR = GetChildAt(C, Idx, VLC);
|
|
if (!AR)
|
|
return AR;
|
|
|
|
StringRef ActualName = VLC->getCommandName();
|
|
if (ActualName != Name)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimLineComment has name \"" << ActualName.str() << "\", "
|
|
"expected \"" << Name.str() << "\"";
|
|
|
|
StringRef ActualText = VLC->getText();
|
|
if (ActualText != Text)
|
|
return ::testing::AssertionFailure()
|
|
<< "VerbatimLineComment has text \"" << ActualText.str() << "\", "
|
|
"expected \"" << Text.str() << "\"";
|
|
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
|
|
TEST_F(CommentParserTest, Basic1) {
|
|
const char *Source = "//";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 0));
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Basic2) {
|
|
const char *Source = "// Meow";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Meow"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Basic3) {
|
|
const char *Source =
|
|
"// Aaa\n"
|
|
"// Bbb";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextWithNewlineAt(PC, 0, " Aaa"));
|
|
ASSERT_TRUE(HasTextAt(PC, 1, " Bbb"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Paragraph1) {
|
|
const char *Sources[] = {
|
|
"// Aaa\n"
|
|
"//\n"
|
|
"// Bbb",
|
|
|
|
"// Aaa\n"
|
|
"//\n"
|
|
"//\n"
|
|
"// Bbb",
|
|
};
|
|
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Aaa"));
|
|
}
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 1, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Bbb"));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Paragraph2) {
|
|
const char *Source =
|
|
"// \\brief Aaa\n"
|
|
"//\n"
|
|
"// Bbb";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 3));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
BlockCommandComment *BCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasBlockCommandAt(FC, 1, BCC, "brief", PC));
|
|
|
|
ASSERT_TRUE(GetChildAt(BCC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Aaa"));
|
|
}
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 2, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Bbb"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Paragraph3) {
|
|
const char *Source = "// \\brief \\author";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 3));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
BlockCommandComment *BCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasBlockCommandAt(FC, 1, BCC, "brief", PC));
|
|
|
|
ASSERT_TRUE(GetChildAt(BCC, 0, PC));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
BlockCommandComment *BCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasBlockCommandAt(FC, 2, BCC, "author", PC));
|
|
|
|
ASSERT_TRUE(GetChildAt(BCC, 0, PC));
|
|
ASSERT_TRUE(HasChildCount(PC, 0));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, Paragraph4) {
|
|
const char *Source =
|
|
"// \\brief Aaa\n"
|
|
"// Bbb \\author\n"
|
|
"// Ccc";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 3));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
BlockCommandComment *BCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasBlockCommandAt(FC, 1, BCC, "brief", PC));
|
|
|
|
ASSERT_TRUE(GetChildAt(BCC, 0, PC));
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextWithNewlineAt(PC, 0, " Aaa"));
|
|
ASSERT_TRUE(HasTextAt(PC, 1, " Bbb "));
|
|
}
|
|
{
|
|
BlockCommandComment *BCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasBlockCommandAt(FC, 2, BCC, "author", PC));
|
|
|
|
ASSERT_TRUE(GetChildAt(BCC, 0, PC));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " Ccc"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, ParamCommand1) {
|
|
const char *Source =
|
|
"// \\param aaa\n"
|
|
"// \\param [in] aaa\n"
|
|
"// \\param [out] aaa\n"
|
|
"// \\param [in,out] aaa\n"
|
|
"// \\param [in, out] aaa\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 6));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
ParamCommandComment *PCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasParamCommandAt(FC, 1, PCC, "param",
|
|
ParamCommandComment::In,
|
|
/* IsDirectionExplicit = */ false,
|
|
"aaa", PC));
|
|
ASSERT_TRUE(HasChildCount(PCC, 1));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
ParamCommandComment *PCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasParamCommandAt(FC, 2, PCC, "param",
|
|
ParamCommandComment::In,
|
|
/* IsDirectionExplicit = */ true,
|
|
"aaa", PC));
|
|
ASSERT_TRUE(HasChildCount(PCC, 1));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
ParamCommandComment *PCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasParamCommandAt(FC, 3, PCC, "param",
|
|
ParamCommandComment::Out,
|
|
/* IsDirectionExplicit = */ true,
|
|
"aaa", PC));
|
|
ASSERT_TRUE(HasChildCount(PCC, 1));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
ParamCommandComment *PCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasParamCommandAt(FC, 4, PCC, "param",
|
|
ParamCommandComment::InOut,
|
|
/* IsDirectionExplicit = */ true,
|
|
"aaa", PC));
|
|
ASSERT_TRUE(HasChildCount(PCC, 1));
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
ParamCommandComment *PCC;
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(HasParamCommandAt(FC, 5, PCC, "param",
|
|
ParamCommandComment::InOut,
|
|
/* IsDirectionExplicit = */ true,
|
|
"aaa", PC));
|
|
ASSERT_TRUE(HasChildCount(PCC, 1));
|
|
ASSERT_TRUE(HasChildCount(PC, 0));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, InlineCommand1) {
|
|
const char *Source = "// \\c";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
InlineCommandComment *ICC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasInlineCommandAt(PC, 1, ICC, "c", NoArgs()));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, InlineCommand2) {
|
|
const char *Source = "// \\c ";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
InlineCommandComment *ICC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 3));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasInlineCommandAt(PC, 1, ICC, "c", NoArgs()));
|
|
ASSERT_TRUE(HasTextAt(PC, 2, " "));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, InlineCommand3) {
|
|
const char *Source = "// \\c aaa\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
InlineCommandComment *ICC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasInlineCommandAt(PC, 1, ICC, "c", "aaa"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, InlineCommand4) {
|
|
const char *Source = "// \\c aaa bbb";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
InlineCommandComment *ICC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 3));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasInlineCommandAt(PC, 1, ICC, "c", "aaa"));
|
|
ASSERT_TRUE(HasTextAt(PC, 2, " bbb"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, InlineCommand5) {
|
|
const char *Source = "// \\unknown aaa\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
InlineCommandComment *ICC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 3));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasInlineCommandAt(PC, 1, ICC, "unknown", NoArgs()));
|
|
ASSERT_TRUE(HasTextAt(PC, 2, " aaa"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML1) {
|
|
const char *Sources[] = {
|
|
"// <a",
|
|
"// <a>",
|
|
"// <a >"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLStartTagComment *HST;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLStartTagAt(PC, 1, HST, "a", NoAttrs()));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML2) {
|
|
const char *Sources[] = {
|
|
"// <br/>",
|
|
"// <br />"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLStartTagComment *HST;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLStartTagAt(PC, 1, HST, "br", SelfClosing()));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML3) {
|
|
const char *Sources[] = {
|
|
"// <a href",
|
|
"// <a href ",
|
|
"// <a href>",
|
|
"// <a href >",
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLStartTagComment *HST;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLStartTagAt(PC, 1, HST, "a", "href", ""));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML4) {
|
|
const char *Sources[] = {
|
|
"// <a href=\"bbb\"",
|
|
"// <a href=\"bbb\">",
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLStartTagComment *HST;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLStartTagAt(PC, 1, HST, "a", "href", "bbb"));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML5) {
|
|
const char *Sources[] = {
|
|
"// </a",
|
|
"// </a>",
|
|
"// </a >"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLEndTagComment *HET;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 2));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLEndTagAt(PC, 1, HET, "a"));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, HTML6) {
|
|
const char *Source =
|
|
"// <pre>\n"
|
|
"// Aaa\n"
|
|
"// Bbb\n"
|
|
"// </pre>\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
HTMLStartTagComment *HST;
|
|
HTMLEndTagComment *HET;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 6));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
ASSERT_TRUE(HasHTMLStartTagAt(PC, 1, HST, "pre", NoAttrs()));
|
|
ASSERT_TRUE(HasTextWithNewlineAt(PC, 2, " Aaa"));
|
|
ASSERT_TRUE(HasTextWithNewlineAt(PC, 3, " Bbb"));
|
|
ASSERT_TRUE(HasTextAt(PC, 4, " "));
|
|
ASSERT_TRUE(HasHTMLEndTagAt(PC, 5, HET, "pre"));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock1) {
|
|
const char *Source = "// \\verbatim\\endverbatim\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimBlockComment *VCC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 1, VCC, "verbatim", NoLines()));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock2) {
|
|
const char *Source = "// \\verbatim Aaa \\endverbatim\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimBlockComment *VBC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 1, VBC, "verbatim", " Aaa "));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock3) {
|
|
const char *Source =
|
|
"//\\verbatim\n"
|
|
"//\\endverbatim\n";
|
|
|
|
FullComment *FC = parseString(Source);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
VerbatimBlockComment *VBC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 0, VBC, "verbatim", NoLines()));
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock4) {
|
|
const char *Sources[] = {
|
|
"//\\verbatim\n"
|
|
"// Aaa\n"
|
|
"//\\endverbatim\n",
|
|
|
|
"/*\\verbatim\n"
|
|
" * Aaa\n"
|
|
" *\\endverbatim*/"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 1));
|
|
|
|
{
|
|
VerbatimBlockComment *VBC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 0, VBC, "verbatim", " Aaa"));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock5) {
|
|
const char *Sources[] = {
|
|
"// \\verbatim\n"
|
|
"// Aaa\n"
|
|
"// \\endverbatim\n",
|
|
|
|
"/* \\verbatim\n"
|
|
" * Aaa\n"
|
|
" * \\endverbatim*/"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimBlockComment *VBC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 1, VBC, "verbatim", " Aaa", " "));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimBlock6) {
|
|
const char *Sources[] = {
|
|
"// \\verbatim\n"
|
|
"// Aaa\n"
|
|
"//\n"
|
|
"// Bbb\n"
|
|
"// \\endverbatim\n",
|
|
|
|
"/* \\verbatim\n"
|
|
" * Aaa\n"
|
|
" *\n"
|
|
" * Bbb\n"
|
|
" * \\endverbatim*/"
|
|
};
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimBlockComment *VBC;
|
|
ASSERT_TRUE(HasVerbatimBlockAt(FC, 1, VBC, "verbatim"));
|
|
ASSERT_EQ(4U, VBC->getNumLines());
|
|
ASSERT_EQ(" Aaa", VBC->getText(0));
|
|
ASSERT_EQ("", VBC->getText(1));
|
|
ASSERT_EQ(" Bbb", VBC->getText(2));
|
|
ASSERT_EQ(" ", VBC->getText(3));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimLine1) {
|
|
const char *Sources[] = {
|
|
"// \\fn",
|
|
"// \\fn\n"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimLineComment *VLC;
|
|
ASSERT_TRUE(HasVerbatimLineAt(FC, 1, VLC, "fn", ""));
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_F(CommentParserTest, VerbatimLine2) {
|
|
const char *Sources[] = {
|
|
"/// \\fn void *foo(const char *zzz = \"\\$\");\n//",
|
|
"/** \\fn void *foo(const char *zzz = \"\\$\");*/"
|
|
};
|
|
|
|
for (size_t i = 0, e = array_lengthof(Sources); i != e; i++) {
|
|
FullComment *FC = parseString(Sources[i]);
|
|
ASSERT_TRUE(HasChildCount(FC, 2));
|
|
|
|
{
|
|
ParagraphComment *PC;
|
|
ASSERT_TRUE(GetChildAt(FC, 0, PC));
|
|
|
|
ASSERT_TRUE(HasChildCount(PC, 1));
|
|
ASSERT_TRUE(HasTextAt(PC, 0, " "));
|
|
}
|
|
{
|
|
VerbatimLineComment *VLC;
|
|
ASSERT_TRUE(HasVerbatimLineAt(FC, 1, VLC, "fn",
|
|
" void *foo(const char *zzz = \"\\$\");"));
|
|
}
|
|
}
|
|
}
|
|
|
|
} // unnamed namespace
|
|
|
|
} // end namespace comments
|
|
} // end namespace clang
|
|
|