1 /* 2 * Copyright (C) 2010 Google Inc. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.google.doclava; 18 19 import java.util.regex.Pattern; 20 import java.util.regex.Matcher; 21 import java.util.ArrayList; 22 import java.util.Arrays; 23 import java.util.HashSet; 24 import java.util.Set; 25 26 public class Comment { 27 static final Pattern FIRST_SENTENCE = 28 Pattern.compile("((.*?)\\.)[ \t\r\n\\<](.*)", Pattern.DOTALL); 29 30 private static final Set<String> KNOWN_TAGS = new HashSet<String>(Arrays.asList(new String[] { 31 "@apiNote", 32 "@author", 33 "@version", 34 //not used by metalava for Android docs (see @apiSince) 35 "@since", 36 //value is an Android API level (set automatically by metalava) 37 "@apiSince", 38 "@deprecated", 39 //value is an Android API level (set automatically by metalava) 40 "@deprecatedSince", 41 "@undeprecate", 42 "@docRoot", 43 "@sdkCurrent", 44 "@inheritDoc", 45 "@more", 46 "@samplecode", 47 "@sample", 48 "@include", 49 "@serial", 50 "@implNote", 51 "@implSpec", 52 "@usesMathJax", 53 })); 54 Comment(String text, ContainerInfo base, SourcePositionInfo sp)55 public Comment(String text, ContainerInfo base, SourcePositionInfo sp) { 56 mText = text; 57 mBase = base; 58 // sp now points to the end of the text, not the beginning! 59 mPosition = SourcePositionInfo.findBeginning(sp, text); 60 } 61 parseCommentTags(String text)62 private void parseCommentTags(String text) { 63 int i = 0; 64 int length = text.length(); 65 while (i < length && isWhitespaceChar(text.charAt(i++))) {} 66 67 if (i <= 0) { 68 return; 69 } 70 71 text = text.substring(i-1); 72 length = text.length(); 73 74 if ("".equals(text)) { 75 return; 76 } 77 78 int start = 0; 79 int end = findStartOfBlock(text, start); 80 81 82 // possible scenarios 83 // main and block(s) 84 // main only (end == -1) 85 // block(s) only (end == 0) 86 87 switch (end) { 88 case -1: // main only 89 parseMainDescription(text, start, length); 90 return; 91 case 0: // block(s) only 92 break; 93 default: // main and block 94 95 // find end of main because end is really the beginning of @ 96 parseMainDescription(text, start, findEndOfMainOrBlock(text, start, end)); 97 break; 98 } 99 100 // parse blocks 101 for (start = end; start < length; start = end) { 102 end = findStartOfBlock(text, start+1); 103 104 if (end == -1) { 105 parseBlock(text, start, length); 106 break; 107 } else { 108 parseBlock(text, start, findEndOfMainOrBlock(text, start, end)); 109 } 110 } 111 112 // for each block 113 // make block parts 114 // end is either next @ at beginning of line or end of text 115 } 116 findEndOfMainOrBlock(String text, int start, int end)117 private int findEndOfMainOrBlock(String text, int start, int end) { 118 for (int i = end-1; i >= start; i--) { 119 if (!isWhitespaceChar(text.charAt(i))) { 120 end = i+1; 121 break; 122 } 123 } 124 return end; 125 } 126 parseMainDescription(String mainDescription, int start, int end)127 private void parseMainDescription(String mainDescription, int start, int end) { 128 if (mainDescription == null) { 129 return; 130 } 131 132 SourcePositionInfo pos = SourcePositionInfo.add(mPosition, mText, 0); 133 while (start < end) { 134 int startOfInlineTag = findStartIndexOfInlineTag(mainDescription, start, end); 135 136 // if there are no more tags 137 if (startOfInlineTag == -1) { 138 tag(null, mainDescription.substring(start, end), true, pos); 139 return; 140 } 141 142 //int endOfInlineTag = mainDescription.indexOf('}', startOfInlineTag); 143 int endOfInlineTag = findEndIndexOfInlineTag(mainDescription, startOfInlineTag, end); 144 145 // if there was only beginning tag 146 if (endOfInlineTag == -1) { 147 // parse all of main as one tag 148 tag(null, mainDescription.substring(start, end), true, pos); 149 return; 150 } 151 152 endOfInlineTag++; // add one to make it a proper ending index 153 154 // do first part without an inline tag - ie, just plaintext 155 tag(null, mainDescription.substring(start, startOfInlineTag), true, pos); 156 157 // parse the rest of this section, the inline tag 158 parseInlineTag(mainDescription, startOfInlineTag, endOfInlineTag, pos); 159 160 // keep going 161 start = endOfInlineTag; 162 } 163 } 164 findStartIndexOfInlineTag(String text, int fromIndex, int toIndex)165 private int findStartIndexOfInlineTag(String text, int fromIndex, int toIndex) { 166 for (int i = fromIndex; i < (toIndex-3); i++) { 167 if (text.charAt(i) == '{' && text.charAt(i+1) == '@' && !isWhitespaceChar(text.charAt(i+2))) { 168 return i; 169 } 170 } 171 172 return -1; 173 } 174 findEndIndexOfInlineTag(String text, int fromIndex, int toIndex)175 private int findEndIndexOfInlineTag(String text, int fromIndex, int toIndex) { 176 int braceDepth = 0; 177 for (int i = fromIndex; i < toIndex; i++) { 178 if (text.charAt(i) == '{') { 179 braceDepth++; 180 } else if (text.charAt(i) == '}') { 181 braceDepth--; 182 if (braceDepth == 0) { 183 return i; 184 } 185 } 186 } 187 188 return -1; 189 } 190 parseInlineTag(String text, int start, int end, SourcePositionInfo pos)191 private void parseInlineTag(String text, int start, int end, SourcePositionInfo pos) { 192 int index = start+1; 193 //int len = text.length(); 194 char c = text.charAt(index); 195 // find the end of the tag name "@something" 196 // need to do something special if we have '}' 197 while (index < end && !isWhitespaceChar(c)) { 198 199 // if this tag has no value, just return with tag name only 200 if (c == '}') { 201 // TODO - should value be "" or null? 202 tag(text.substring(start+1, end), null, true, pos); 203 return; 204 } 205 c = text.charAt(index++); 206 } 207 208 // don't parse things that don't have at least one extra character after @ 209 // probably should be plus 3 210 // TODO - remove this - think it's fixed by change in parseMainDescription 211 if (index == start+3) { 212 return; 213 } 214 215 int endOfFirstPart = index-1; 216 217 // get to beginning of tag value 218 while (index < end && isWhitespaceChar(text.charAt(index++))) {} 219 int startOfSecondPart = index-1; 220 221 // +1 to get rid of opening brace and -1 to get rid of closing brace 222 // maybe i wanna make this more elegant 223 String tagName = text.substring(start+1, endOfFirstPart); 224 String tagText = text.substring(startOfSecondPart, end-1); 225 tag(tagName, tagText, true, pos); 226 } 227 228 229 /** 230 * Finds the index of the start of a new block comment or -1 if there are 231 * no more starts. 232 * @param text The String to search 233 * @param start the index of the String to start searching 234 * @return The index of the start of a new block comment or -1 if there are 235 * no more starts. 236 */ findStartOfBlock(String text, int start)237 private int findStartOfBlock(String text, int start) { 238 // how to detect we're at a new @ 239 // if the chars to the left of it are \r or \n, we're at one 240 // if the chars to the left of it are ' ' or \t, keep looking 241 // otherwise, we're in the middle of a block, keep looking 242 int index = text.indexOf('@', start); 243 244 // no @ in text or index at first position 245 if (index == -1 || 246 (index == 0 && text.length() > 1 && !isWhitespaceChar(text.charAt(index+1)))) { 247 return index; 248 } 249 250 index = getPossibleStartOfBlock(text, index); 251 252 int i = index-1; // start at the character immediately to the left of @ 253 char c; 254 while (i >= 0) { 255 c = text.charAt(i--); 256 257 // found a new block comment because we're at the beginning of a line 258 if (c == '\r' || c == '\n') { 259 return index; 260 } 261 262 // there is a non whitespace character to the left of the @ 263 // before finding a new line, keep searching 264 if (c != ' ' && c != '\t') { 265 index = getPossibleStartOfBlock(text, index+1); 266 i = index-1; 267 } 268 269 // some whitespace character, so keep looking, we might be at a new block comment 270 } 271 272 return -1; 273 } 274 getPossibleStartOfBlock(String text, int index)275 private int getPossibleStartOfBlock(String text, int index) { 276 while (isWhitespaceChar(text.charAt(index+1)) || !isWhitespaceChar(text.charAt(index-1))) { 277 index = text.indexOf('@', index+1); 278 279 if (index == -1 || index == text.length()-1) { 280 return -1; 281 } 282 } 283 284 return index; 285 } 286 parseBlock(String text, int startOfBlock, int endOfBlock)287 private void parseBlock(String text, int startOfBlock, int endOfBlock) { 288 SourcePositionInfo pos = SourcePositionInfo.add(mPosition, mText, startOfBlock); 289 int index = startOfBlock; 290 291 for (char c = text.charAt(index); 292 index < endOfBlock && !isWhitespaceChar(c); c = text.charAt(index++)) {} 293 if (index == startOfBlock+1) { 294 return; 295 } 296 297 int endOfFirstPart = index-1; 298 if (index == endOfBlock) { 299 // TODO - should value be null or "" 300 tag(text.substring(startOfBlock, 301 findEndOfMainOrBlock(text, startOfBlock, index)), "", false, pos); 302 return; 303 } 304 305 306 // get to beginning of tag value 307 while (index < endOfBlock && isWhitespaceChar(text.charAt(index++))) {} 308 int startOfSecondPart = index-1; 309 310 tag(text.substring(startOfBlock, endOfFirstPart), 311 text.substring(startOfSecondPart, endOfBlock), false, pos); 312 } 313 isWhitespaceChar(char c)314 private boolean isWhitespaceChar(char c) { 315 switch (c) { 316 case ' ': 317 case '\r': 318 case '\t': 319 case '\n': 320 return true; 321 } 322 return false; 323 } 324 tag(String name, String text, boolean isInline, SourcePositionInfo pos)325 private void tag(String name, String text, boolean isInline, SourcePositionInfo pos) { 326 /* 327 * String s = isInline ? "inline" : "outofline"; System.out.println("---> " + s + " name=[" + 328 * name + "] text=[" + text + "]"); 329 */ 330 if (name == null) { 331 mInlineTagsList.add(new TextTagInfo("Text", "Text", text, pos)); 332 } else if (name.equals("@param")) { 333 mParamTagsList.add(new ParamTagInfo("@param", "@param", text, mBase, pos)); 334 } else if (name.equals("@apiSince")) { 335 setApiSince(text); 336 } else if (name.equals("@deprecatedSince")) { 337 setDeprecatedSince(text); 338 } else if (name.equals("@see")) { 339 mSeeTagsList.add(new SeeTagInfo("@see", "@see", text, mBase, pos)); 340 } else if (name.equals("@link")) { 341 if (Doclava.DEVSITE_IGNORE_JDLINKS) { 342 TagInfo linkTag = new TextTagInfo(name, name, text, pos); 343 mInlineTagsList.add(linkTag); 344 } else { 345 mInlineTagsList.add(new SeeTagInfo(name, "@see", text, mBase, pos)); 346 } 347 } else if (name.equals("@linkplain")) { 348 mInlineTagsList.add(new SeeTagInfo(name, "@linkplain", text, mBase, pos)); 349 } else if (name.equals("@value")) { 350 mInlineTagsList.add(new SeeTagInfo(name, "@value", text, mBase, pos)); 351 } else if (name.equals("@throws") || name.equals("@exception")) { 352 mThrowsTagsList.add(new ThrowsTagInfo("@throws", "@throws", text, mBase, pos)); 353 } else if (name.equals("@return")) { 354 mReturnTagsList.add(new ParsedTagInfo("@return", "@return", text, mBase, pos)); 355 } else if (name.equals("@deprecated")) { 356 if (text.length() == 0) { 357 Errors.error(Errors.MISSING_COMMENT, pos, "@deprecated tag with no explanatory comment"); 358 text = "No replacement."; 359 } 360 mDeprecatedTagsList.add(new ParsedTagInfo("@deprecated", "@deprecated", text, mBase, pos)); 361 } else if (name.equals("@literal")) { 362 mInlineTagsList.add(new LiteralTagInfo(text, pos)); 363 } else if (name.equals("@code")) { 364 mInlineTagsList.add(new CodeTagInfo(text, pos)); 365 } else if (name.equals("@hide") || name.equals("@removed") 366 || name.equals("@pending") || name.equals("@doconly")) { 367 // nothing 368 } else if (name.equals("@attr")) { 369 AttrTagInfo tag = new AttrTagInfo("@attr", "@attr", text, mBase, pos); 370 mAttrTagsList.add(tag); 371 Comment c = tag.description(); 372 if (c != null) { 373 for (TagInfo t : c.tags()) { 374 mInlineTagsList.add(t); 375 } 376 } 377 } else if (name.equals("@undeprecate")) { 378 mUndeprecateTagsList.add(new TextTagInfo("@undeprecate", "@undeprecate", text, pos)); 379 } else if (name.equals("@include") || name.equals("@sample")) { 380 mInlineTagsList.add(new SampleTagInfo(name, "@include", text, mBase, pos)); 381 } else if (name.equals("@apiNote") || name.equals("@implSpec") || name.equals("@implNote")) { 382 mTagsList.add(new ParsedTagInfo(name, name, text, mBase, pos)); 383 } else if (name.equals("@memberDoc")) { 384 mMemberDocTagsList.add(new ParsedTagInfo("@memberDoc", "@memberDoc", text, mBase, pos)); 385 } else if (name.equals("@paramDoc")) { 386 mParamDocTagsList.add(new ParsedTagInfo("@paramDoc", "@paramDoc", text, mBase, pos)); 387 } else if (name.equals("@returnDoc")) { 388 mReturnDocTagsList.add(new ParsedTagInfo("@returnDoc", "@returnDoc", text, mBase, pos)); 389 } else { 390 boolean known = KNOWN_TAGS.contains(name); 391 if (!known) { 392 known = Doclava.knownTags.contains(name); 393 } 394 if (!known) { 395 Errors.error(Errors.UNKNOWN_TAG, pos == null ? null : new SourcePositionInfo(pos), 396 "Unknown tag: " + name); 397 } 398 TagInfo t = new TextTagInfo(name, name, text, pos); 399 if (isInline) { 400 mInlineTagsList.add(t); 401 } else { 402 mTagsList.add(t); 403 } 404 } 405 } 406 parseBriefTags()407 private void parseBriefTags() { 408 int N = mInlineTagsList.size(); 409 410 // look for "@more" tag, which means that we might go past the first sentence. 411 int more = -1; 412 for (int i = 0; i < N; i++) { 413 if (mInlineTagsList.get(i).name().equals("@more")) { 414 more = i; 415 } 416 } 417 if (more >= 0) { 418 for (int i = 0; i < more; i++) { 419 mBriefTagsList.add(mInlineTagsList.get(i)); 420 } 421 } else { 422 for (int i = 0; i < N; i++) { 423 TagInfo t = mInlineTagsList.get(i); 424 if (t.name().equals("Text")) { 425 Matcher m = FIRST_SENTENCE.matcher(t.text()); 426 if (m.matches()) { 427 String text = m.group(1); 428 TagInfo firstSentenceTag = new TagInfo(t.name(), t.kind(), text, t.position()); 429 mBriefTagsList.add(firstSentenceTag); 430 break; 431 } 432 } 433 mBriefTagsList.add(t); 434 435 } 436 } 437 } 438 tags()439 public TagInfo[] tags() { 440 init(); 441 return mInlineTags; 442 } 443 tags(String name)444 public TagInfo[] tags(String name) { 445 init(); 446 ArrayList<TagInfo> results = new ArrayList<TagInfo>(); 447 int N = mInlineTagsList.size(); 448 for (int i = 0; i < N; i++) { 449 TagInfo t = mInlineTagsList.get(i); 450 if (t.name().equals(name)) { 451 results.add(t); 452 } 453 } 454 return results.toArray(TagInfo.getArray(results.size())); 455 } 456 blockTags()457 public TagInfo[] blockTags() { 458 init(); 459 return mTags; 460 } 461 paramTags()462 public ParamTagInfo[] paramTags() { 463 init(); 464 return mParamTags; 465 } 466 seeTags()467 public SeeTagInfo[] seeTags() { 468 init(); 469 return mSeeTags; 470 } 471 throwsTags()472 public ThrowsTagInfo[] throwsTags() { 473 init(); 474 return mThrowsTags; 475 } 476 returnTags()477 public TagInfo[] returnTags() { 478 init(); 479 return mReturnTags; 480 } 481 deprecatedTags()482 public TagInfo[] deprecatedTags() { 483 init(); 484 return mDeprecatedTags; 485 } 486 undeprecateTags()487 public TagInfo[] undeprecateTags() { 488 init(); 489 return mUndeprecateTags; 490 } 491 attrTags()492 public AttrTagInfo[] attrTags() { 493 init(); 494 return mAttrTags; 495 } 496 briefTags()497 public TagInfo[] briefTags() { 498 init(); 499 return mBriefTags; 500 } 501 memberDocTags()502 public ParsedTagInfo[] memberDocTags() { 503 init(); 504 return mMemberDocTags; 505 } 506 paramDocTags()507 public ParsedTagInfo[] paramDocTags() { 508 init(); 509 return mParamDocTags; 510 } 511 returnDocTags()512 public ParsedTagInfo[] returnDocTags() { 513 init(); 514 return mReturnDocTags; 515 } 516 isHidden()517 public boolean isHidden() { 518 if (mHidden == null) { 519 mHidden = !Doclava.checkLevel(Doclava.SHOW_HIDDEN) && 520 (mText != null) && (mText.indexOf("@hide") >= 0 || mText.indexOf("@pending") >= 0); 521 } 522 return mHidden; 523 } 524 isRemoved()525 public boolean isRemoved() { 526 if (mRemoved == null) { 527 mRemoved = !Doclava.checkLevel(Doclava.SHOW_HIDDEN) && 528 (mText != null) && (mText.indexOf("@removed") >= 0); 529 } 530 531 return mRemoved; 532 } 533 setDeprecatedSince(String since)534 public void setDeprecatedSince(String since) { 535 if (since != null) { 536 since = since.trim(); 537 } 538 mDeprecatedSince = since; 539 } 540 getDeprecatedSince()541 public String getDeprecatedSince() { 542 return mDeprecatedSince; 543 } 544 setApiSince(String since)545 public void setApiSince(String since) { 546 if (since != null) { 547 since = since.trim(); 548 } 549 mApiSince = since; 550 } 551 getApiSince()552 public String getApiSince() { 553 //return the value of @apiSince, an API level in Android 554 return mApiSince; 555 } 556 isDocOnly()557 public boolean isDocOnly() { 558 if (mDocOnly == null) { 559 mDocOnly = (mText != null) && (mText.indexOf("@doconly") >= 0); 560 } 561 return mDocOnly; 562 } 563 isDeprecated()564 public boolean isDeprecated() { 565 if (mDeprecated == null) { 566 mDeprecated = (mText != null) && (mText.indexOf("@deprecated") >= 0); 567 } 568 569 return mDeprecated; 570 } 571 init()572 private void init() { 573 if (!mInitialized) { 574 initImpl(); 575 } 576 } 577 initImpl()578 private void initImpl() { 579 isHidden(); 580 isRemoved(); 581 isDocOnly(); 582 isDeprecated(); 583 584 // Don't bother parsing text if we aren't generating documentation. 585 if (Doclava.parseComments()) { 586 parseCommentTags(mText); 587 parseBriefTags(); 588 } else { 589 // Forces methods to be recognized by findOverriddenMethods in MethodInfo. 590 mInlineTagsList.add(new TextTagInfo("Text", "Text", mText, 591 SourcePositionInfo.add(mPosition, mText, 0))); 592 } 593 594 mText = null; 595 mInitialized = true; 596 597 mInlineTags = mInlineTagsList.toArray(TagInfo.getArray(mInlineTagsList.size())); 598 mTags = mTagsList.toArray(TagInfo.getArray(mTagsList.size())); 599 mParamTags = mParamTagsList.toArray(ParamTagInfo.getArray(mParamTagsList.size())); 600 mSeeTags = mSeeTagsList.toArray(SeeTagInfo.getArray(mSeeTagsList.size())); 601 mThrowsTags = mThrowsTagsList.toArray(ThrowsTagInfo.getArray(mThrowsTagsList.size())); 602 mReturnTags = ParsedTagInfo.joinTags( 603 mReturnTagsList.toArray(ParsedTagInfo.getArray(mReturnTagsList.size()))); 604 mDeprecatedTags = ParsedTagInfo.joinTags( 605 mDeprecatedTagsList.toArray(ParsedTagInfo.getArray(mDeprecatedTagsList.size()))); 606 mUndeprecateTags = mUndeprecateTagsList.toArray(TagInfo.getArray(mUndeprecateTagsList.size())); 607 mAttrTags = mAttrTagsList.toArray(AttrTagInfo.getArray(mAttrTagsList.size())); 608 mBriefTags = mBriefTagsList.toArray(TagInfo.getArray(mBriefTagsList.size())); 609 mMemberDocTags = mMemberDocTagsList.toArray(ParsedTagInfo.getArray(mMemberDocTagsList.size())); 610 mParamDocTags = mParamDocTagsList.toArray(ParsedTagInfo.getArray(mParamDocTagsList.size())); 611 mReturnDocTags = mReturnDocTagsList.toArray(ParsedTagInfo.getArray(mReturnDocTagsList.size())); 612 613 mTagsList = null; 614 mParamTagsList = null; 615 mSeeTagsList = null; 616 mThrowsTagsList = null; 617 mReturnTagsList = null; 618 mDeprecatedTagsList = null; 619 mUndeprecateTagsList = null; 620 mAttrTagsList = null; 621 mBriefTagsList = null; 622 mMemberDocTagsList = null; 623 mParamDocTagsList = null; 624 mReturnDocTagsList = null; 625 } 626 627 boolean mInitialized; 628 Boolean mHidden = null; 629 Boolean mRemoved = null; 630 Boolean mDocOnly = null; 631 Boolean mDeprecated = null; 632 String mDeprecatedSince; 633 String mApiSince; 634 String mText; 635 ContainerInfo mBase; 636 SourcePositionInfo mPosition; 637 int mLine = 1; 638 639 TagInfo[] mInlineTags; 640 TagInfo[] mTags; 641 ParamTagInfo[] mParamTags; 642 SeeTagInfo[] mSeeTags; 643 ThrowsTagInfo[] mThrowsTags; 644 TagInfo[] mBriefTags; 645 TagInfo[] mReturnTags; 646 TagInfo[] mDeprecatedTags; 647 TagInfo[] mUndeprecateTags; 648 AttrTagInfo[] mAttrTags; 649 ParsedTagInfo[] mMemberDocTags; 650 ParsedTagInfo[] mParamDocTags; 651 ParsedTagInfo[] mReturnDocTags; 652 653 ArrayList<TagInfo> mInlineTagsList = new ArrayList<TagInfo>(); 654 ArrayList<TagInfo> mTagsList = new ArrayList<TagInfo>(); 655 ArrayList<ParamTagInfo> mParamTagsList = new ArrayList<ParamTagInfo>(); 656 ArrayList<SeeTagInfo> mSeeTagsList = new ArrayList<SeeTagInfo>(); 657 ArrayList<ThrowsTagInfo> mThrowsTagsList = new ArrayList<ThrowsTagInfo>(); 658 ArrayList<TagInfo> mBriefTagsList = new ArrayList<TagInfo>(); 659 ArrayList<ParsedTagInfo> mReturnTagsList = new ArrayList<ParsedTagInfo>(); 660 ArrayList<ParsedTagInfo> mDeprecatedTagsList = new ArrayList<ParsedTagInfo>(); 661 ArrayList<TagInfo> mUndeprecateTagsList = new ArrayList<TagInfo>(); 662 ArrayList<AttrTagInfo> mAttrTagsList = new ArrayList<AttrTagInfo>(); 663 ArrayList<ParsedTagInfo> mMemberDocTagsList = new ArrayList<ParsedTagInfo>(); 664 ArrayList<ParsedTagInfo> mParamDocTagsList = new ArrayList<ParsedTagInfo>(); 665 ArrayList<ParsedTagInfo> mReturnDocTagsList = new ArrayList<ParsedTagInfo>(); 666 667 } 668