/**************************************************************************** ** ** Copyright (C) 2019 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the tools applications of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ /* clangcodeparser.cpp */ #include "clangcodeparser.h" #include "access.h" #include "classnode.h" #include "codechunk.h" #include "config.h" #include "enumnode.h" #include "functionnode.h" #include "namespacenode.h" #include "propertynode.h" #include "qdocdatabase.h" #include "typedefnode.h" #include "utilities.h" #include "variablenode.h" #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE // We're printing diagnostics in ClangCodeParser::printDiagnostics, // so avoid clang itself printing them. static const auto kClangDontDisplayDiagnostics = 0; static CXTranslationUnit_Flags flags_ = static_cast(0); static CXIndex index_ = nullptr; QByteArray ClangCodeParser::s_fn; constexpr const char *fnDummyFileName = "/fn_dummyfile.cpp"; #ifndef QT_NO_DEBUG_STREAM template static QDebug operator<<(QDebug debug, const std::vector &v) { QDebugStateSaver saver(debug); debug.noquote(); debug.nospace(); const size_t size = v.size(); debug << "std::vector<>[" << size << "]("; for (size_t i = 0; i < size; ++i) { if (i) debug << ", "; debug << v[i]; } debug << ')'; return debug; } #endif // !QT_NO_DEBUG_STREAM /*! Call clang_visitChildren on the given cursor with the lambda as a callback T can be any functor that is callable with a CXCursor parameter and returns a CXChildVisitResult (in other word compatible with function */ template bool visitChildrenLambda(CXCursor cursor, T &&lambda) { CXCursorVisitor visitor = [](CXCursor c, CXCursor, CXClientData client_data) -> CXChildVisitResult { return (*static_cast(client_data))(c); }; return clang_visitChildren(cursor, visitor, &lambda); } /*! convert a CXString to a QString, and dispose the CXString */ static QString fromCXString(CXString &&string) { QString ret = QString::fromUtf8(clang_getCString(string)); clang_disposeString(string); return ret; } static QString templateDecl(CXCursor cursor); /*! Returns a list of template parameters at \a cursor. */ static QStringList getTemplateParameters(CXCursor cursor) { QStringList parameters; visitChildrenLambda(cursor, [¶meters](CXCursor cur) { QString name = fromCXString(clang_getCursorSpelling(cur)); QString type; switch (clang_getCursorKind(cur)) { case CXCursor_TemplateTypeParameter: type = QStringLiteral("typename"); break; case CXCursor_NonTypeTemplateParameter: type = fromCXString(clang_getTypeSpelling(clang_getCursorType(cur))); // Hack: Omit QtPrivate template parameters from public documentation if (type.startsWith(QLatin1String("QtPrivate"))) return CXChildVisit_Continue; break; case CXCursor_TemplateTemplateParameter: type = templateDecl(cur) + QLatin1String(" class"); break; default: return CXChildVisit_Continue; } if (!name.isEmpty()) name.prepend(QLatin1Char(' ')); parameters << type + name; return CXChildVisit_Continue; }); return parameters; } /*! Gets the template declaration at specified \a cursor. */ static QString templateDecl(CXCursor cursor) { QStringList params = getTemplateParameters(cursor); return QLatin1String("template <") + params.join(QLatin1String(", ")) + QLatin1Char('>'); } /*! convert a CXSourceLocation to a qdoc Location */ static Location fromCXSourceLocation(CXSourceLocation location) { unsigned int line, column; CXString file; clang_getPresumedLocation(location, &file, &line, &column); Location l(fromCXString(std::move(file))); l.setColumnNo(column); l.setLineNo(line); return l; } /*! convert a CX_CXXAccessSpecifier to Node::Access */ static Access fromCX_CXXAccessSpecifier(CX_CXXAccessSpecifier spec) { switch (spec) { case CX_CXXPrivate: return Access::Private; case CX_CXXProtected: return Access::Protected; case CX_CXXPublic: return Access::Public; default: return Access::Public; } } /*! Returns the spelling in the file for a source range */ static QString getSpelling(CXSourceRange range) { auto start = clang_getRangeStart(range); auto end = clang_getRangeEnd(range); CXFile file1, file2; unsigned int offset1, offset2; clang_getFileLocation(start, &file1, nullptr, nullptr, &offset1); clang_getFileLocation(end, &file2, nullptr, nullptr, &offset2); if (file1 != file2 || offset2 <= offset1) return QString(); QFile file(fromCXString(clang_getFileName(file1))); if (!file.open(QFile::ReadOnly)) { if (file.fileName() == fnDummyFileName) return QString::fromUtf8(ClangCodeParser::fn().mid(offset1, offset2 - offset1)); return QString(); } file.seek(offset1); return QString::fromUtf8(file.read(offset2 - offset1)); } /*! Returns the function name from a given cursor representing a function declaration. This is usually clang_getCursorSpelling, but not for the conversion function in which case it is a bit more complicated */ QString functionName(CXCursor cursor) { if (clang_getCursorKind(cursor) == CXCursor_ConversionFunction) { // For a CXCursor_ConversionFunction we don't want the spelling which would be something // like "operator type-parameter-0-0" or "operator unsigned int". we want the actual name as // spelled; QString type = fromCXString(clang_getTypeSpelling(clang_getCursorResultType(cursor))); if (type.isEmpty()) return fromCXString(clang_getCursorSpelling(cursor)); return QLatin1String("operator ") + type; } QString name = fromCXString(clang_getCursorSpelling(cursor)); // Remove template stuff from constructor and destructor but not from operator< auto ltLoc = name.indexOf('<'); if (ltLoc > 0 && !name.startsWith("operator<")) name = name.left(ltLoc); return name; } /*! Reconstruct the qualified path name of a function that is being overridden. */ static QString reconstructQualifiedPathForCursor(CXCursor cur) { QString path; auto kind = clang_getCursorKind(cur); while (!clang_isInvalid(kind) && kind != CXCursor_TranslationUnit) { switch (kind) { case CXCursor_Namespace: case CXCursor_StructDecl: case CXCursor_ClassDecl: case CXCursor_UnionDecl: case CXCursor_ClassTemplate: path.prepend("::"); path.prepend(fromCXString(clang_getCursorSpelling(cur))); break; case CXCursor_FunctionDecl: case CXCursor_FunctionTemplate: case CXCursor_CXXMethod: case CXCursor_Constructor: case CXCursor_Destructor: case CXCursor_ConversionFunction: path = functionName(cur); break; default: break; } cur = clang_getCursorSemanticParent(cur); kind = clang_getCursorKind(cur); } return path; } /*! Find the node from the QDocDatabase \a qdb that corrseponds to the declaration represented by the cursor \a cur, if it exists. */ static Node *findNodeForCursor(QDocDatabase *qdb, CXCursor cur) { auto kind = clang_getCursorKind(cur); if (clang_isInvalid(kind)) return nullptr; if (kind == CXCursor_TranslationUnit) return qdb->primaryTreeRoot(); Node *p = findNodeForCursor(qdb, clang_getCursorSemanticParent(cur)); if (p == nullptr) return nullptr; if (!p->isAggregate()) return nullptr; auto parent = static_cast(p); QString name = fromCXString(clang_getCursorSpelling(cur)); switch (kind) { case CXCursor_Namespace: return parent->findNonfunctionChild(name, &Node::isNamespace); case CXCursor_StructDecl: case CXCursor_ClassDecl: case CXCursor_UnionDecl: case CXCursor_ClassTemplate: return parent->findNonfunctionChild(name, &Node::isClassNode); case CXCursor_FunctionDecl: case CXCursor_FunctionTemplate: case CXCursor_CXXMethod: case CXCursor_Constructor: case CXCursor_Destructor: case CXCursor_ConversionFunction: { NodeVector candidates; parent->findChildren(functionName(cur), candidates); if (candidates.isEmpty()) return nullptr; CXType funcType = clang_getCursorType(cur); auto numArg = clang_getNumArgTypes(funcType); bool isVariadic = clang_isFunctionTypeVariadic(funcType); QVarLengthArray args; for (Node *candidate : qAsConst(candidates)) { if (!candidate->isFunction(Node::CPP)) continue; auto fn = static_cast(candidate); const Parameters ¶meters = fn->parameters(); const int actualArg = numArg - parameters.isPrivateSignal(); if (parameters.count() != actualArg + isVariadic) continue; if (fn->isConst() != bool(clang_CXXMethod_isConst(cur))) continue; if (isVariadic && parameters.last().type() != QLatin1String("...")) continue; bool different = false; for (int i = 0; i < actualArg; ++i) { CXType argType = clang_getArgType(funcType, i); if (args.size() <= i) args.append(fromCXString(clang_getTypeSpelling(argType))); QString recordedType = parameters.at(i).type(); QString typeSpelling = args.at(i); auto p = parent; while (p && recordedType != typeSpelling) { QString parentScope = p->name() + QLatin1String("::"); recordedType.remove(parentScope); typeSpelling.remove(parentScope); p = p->parent(); } different = recordedType != typeSpelling; // Retry with a canonical type spelling if (different && (argType.kind == CXType_Typedef || argType.kind == CXType_Elaborated)) { QStringView canonicalType = parameters.at(i).canonicalType(); if (!canonicalType.isEmpty()) { different = canonicalType != fromCXString(clang_getTypeSpelling(clang_getCanonicalType(argType))); } } if (different) { break; } } if (!different) return fn; } return nullptr; } case CXCursor_EnumDecl: return parent->findNonfunctionChild(name, &Node::isEnumType); case CXCursor_FieldDecl: case CXCursor_VarDecl: return parent->findNonfunctionChild(name, &Node::isVariable); case CXCursor_TypedefDecl: return parent->findNonfunctionChild(name, &Node::isTypedef); default: return nullptr; } } static void setOverridesForFunction(FunctionNode *fn, CXCursor cursor) { CXCursor *overridden; unsigned int numOverridden = 0; clang_getOverriddenCursors(cursor, &overridden, &numOverridden); for (uint i = 0; i < numOverridden; ++i) { QString path = reconstructQualifiedPathForCursor(overridden[i]); if (!path.isEmpty()) { fn->setOverride(true); fn->setOverridesThis(path); break; } } clang_disposeOverriddenCursors(overridden); } class ClangVisitor { public: ClangVisitor(QDocDatabase *qdb, const QHash &allHeaders) : qdb_(qdb), parent_(qdb->primaryTreeRoot()), allHeaders_(allHeaders) { } QDocDatabase *qdocDB() { return qdb_; } CXChildVisitResult visitChildren(CXCursor cursor) { auto ret = visitChildrenLambda(cursor, [&](CXCursor cur) { auto loc = clang_getCursorLocation(cur); if (clang_Location_isFromMainFile(loc)) return visitSource(cur, loc); CXFile file; clang_getFileLocation(loc, &file, nullptr, nullptr, nullptr); bool isInteresting = false; auto it = isInterestingCache_.find(file); if (it != isInterestingCache_.end()) { isInteresting = *it; } else { QFileInfo fi(fromCXString(clang_getFileName(file))); // Match by file name in case of PCH/installed headers isInteresting = allHeaders_.contains(fi.fileName()); isInterestingCache_[file] = isInteresting; } if (isInteresting) { return visitHeader(cur, loc); } return CXChildVisit_Continue; }); return ret ? CXChildVisit_Break : CXChildVisit_Continue; } /* Not sure about all the possibilities, when the cursor location is not in the main file. */ CXChildVisitResult visitFnArg(CXCursor cursor, Node **fnNode, bool &ignoreSignature) { auto ret = visitChildrenLambda(cursor, [&](CXCursor cur) { auto loc = clang_getCursorLocation(cur); if (clang_Location_isFromMainFile(loc)) return visitFnSignature(cur, loc, fnNode, ignoreSignature); return CXChildVisit_Continue; }); return ret ? CXChildVisit_Break : CXChildVisit_Continue; } Node *nodeForCommentAtLocation(CXSourceLocation loc, CXSourceLocation nextCommentLoc); private: /*! SimpleLoc represents a simple location in the main source file, which can be used as a key in a QMap. */ struct SimpleLoc { unsigned int line, column; friend bool operator<(const SimpleLoc &a, const SimpleLoc &b) { return a.line != b.line ? a.line < b.line : a.column < b.column; } }; /*! \variable ClangVisitor::declMap_ Map of all the declarations in the source file so we can match them with a documentation comment. */ QMap declMap_; QDocDatabase *qdb_; Aggregate *parent_; bool m_friendDecl { false }; // true if currently visiting a friend declaration const QHash allHeaders_; QHash isInterestingCache_; // doing a canonicalFilePath is slow, so keep a cache. /*! Returns true if the symbol should be ignored for the documentation. */ bool ignoredSymbol(const QString &symbolName) { if (symbolName == QLatin1String("QPrivateSignal")) return true; // Ignore functions generated by property macros if (symbolName.startsWith("_qt_property_")) return true; return false; } /*! The type parameters do not need to be fully qualified This function removes the ClassName:: if needed. example: 'QLinkedList::iterator' -> 'iterator' */ QString adjustTypeName(const QString &typeName) { auto parent = parent_->parent(); if (parent && parent->isClassNode()) { QStringView typeNameConstRemoved(typeName); if (typeNameConstRemoved.startsWith(QLatin1String("const "))) typeNameConstRemoved = typeNameConstRemoved.mid(6); auto parentName = parent->fullName(); if (typeNameConstRemoved.startsWith(parentName) && typeNameConstRemoved.mid(parentName.size(), 2) == QLatin1String("::")) { QString result = typeName; result.remove(typeName.indexOf(typeNameConstRemoved), parentName.size() + 2); return result; } } return typeName; } CXChildVisitResult visitSource(CXCursor cursor, CXSourceLocation loc); CXChildVisitResult visitHeader(CXCursor cursor, CXSourceLocation loc); CXChildVisitResult visitFnSignature(CXCursor cursor, CXSourceLocation loc, Node **fnNode, bool &ignoreSignature); void processFunction(FunctionNode *fn, CXCursor cursor); bool parseProperty(const QString &spelling, const Location &loc); void readParameterNamesAndAttributes(FunctionNode *fn, CXCursor cursor); Aggregate *getSemanticParent(CXCursor cursor); }; /*! Visits a cursor in the .cpp file. This fills the declMap_ */ CXChildVisitResult ClangVisitor::visitSource(CXCursor cursor, CXSourceLocation loc) { auto kind = clang_getCursorKind(cursor); if (clang_isDeclaration(kind)) { SimpleLoc l; clang_getPresumedLocation(loc, nullptr, &l.line, &l.column); declMap_.insert(l, cursor); return CXChildVisit_Recurse; } return CXChildVisit_Continue; } /*! If the semantic and lexical parent cursors of \a cursor are not the same, find the Aggregate node for the semantic parent cursor and return it. Otherwise return the current parent. */ Aggregate *ClangVisitor::getSemanticParent(CXCursor cursor) { CXCursor sp = clang_getCursorSemanticParent(cursor); CXCursor lp = clang_getCursorLexicalParent(cursor); if (!clang_equalCursors(sp, lp) && clang_isDeclaration(clang_getCursorKind(sp))) { Node *spn = findNodeForCursor(qdb_, sp); if (spn && spn->isAggregate()) { return static_cast(spn); } } return parent_; } CXChildVisitResult ClangVisitor::visitFnSignature(CXCursor cursor, CXSourceLocation, Node **fnNode, bool &ignoreSignature) { switch (clang_getCursorKind(cursor)) { case CXCursor_Namespace: return CXChildVisit_Recurse; case CXCursor_FunctionDecl: case CXCursor_FunctionTemplate: case CXCursor_CXXMethod: case CXCursor_Constructor: case CXCursor_Destructor: case CXCursor_ConversionFunction: { ignoreSignature = false; if (ignoredSymbol(functionName(cursor))) { *fnNode = nullptr; ignoreSignature = true; } else { *fnNode = findNodeForCursor(qdb_, cursor); if (*fnNode) { if ((*fnNode)->isFunction(Node::CPP)) { FunctionNode *fn = static_cast(*fnNode); readParameterNamesAndAttributes(fn, cursor); } } else { // Possibly an implicitly generated special member QString name = functionName(cursor); if (ignoredSymbol(name)) return CXChildVisit_Continue; Aggregate *semanticParent = getSemanticParent(cursor); if (semanticParent && semanticParent->isClass()) { auto *candidate = new FunctionNode(nullptr, name); processFunction(candidate, cursor); if (!candidate->isSpecialMemberFunction()) { delete candidate; return CXChildVisit_Continue; } candidate->setDefault(true); semanticParent->addChild(*fnNode = candidate); } } } break; } default: break; } return CXChildVisit_Continue; } CXChildVisitResult ClangVisitor::visitHeader(CXCursor cursor, CXSourceLocation loc) { auto kind = clang_getCursorKind(cursor); QString templateString; switch (kind) { case CXCursor_TypeAliasTemplateDecl: case CXCursor_TypeAliasDecl: { QString aliasDecl = getSpelling(clang_getCursorExtent(cursor)).simplified(); QStringList typeAlias = aliasDecl.split(QLatin1Char('=')); if (typeAlias.size() == 2) { typeAlias[0] = typeAlias[0].trimmed(); const QLatin1String usingString("using "); int usingPos = typeAlias[0].indexOf(usingString); if (usingPos != -1) { if (kind == CXCursor_TypeAliasTemplateDecl) templateString = typeAlias[0].left(usingPos).trimmed(); typeAlias[0].remove(0, usingPos + usingString.size()); typeAlias[0] = typeAlias[0].split(QLatin1Char(' ')).first(); typeAlias[1] = typeAlias[1].trimmed(); TypeAliasNode *ta = new TypeAliasNode(parent_, typeAlias[0], typeAlias[1]); ta->setAccess(fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor))); ta->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); ta->setTemplateDecl(templateString); } } return CXChildVisit_Continue; } case CXCursor_StructDecl: case CXCursor_UnionDecl: if (fromCXString(clang_getCursorSpelling(cursor)).isEmpty()) // anonymous struct or union return CXChildVisit_Continue; Q_FALLTHROUGH(); case CXCursor_ClassTemplate: templateString = templateDecl(cursor); Q_FALLTHROUGH(); case CXCursor_ClassDecl: { if (!clang_isCursorDefinition(cursor)) return CXChildVisit_Continue; if (findNodeForCursor(qdb_, cursor)) // Was already parsed, probably in another TU return CXChildVisit_Continue; QString className = fromCXString(clang_getCursorSpelling(cursor)); Aggregate *semanticParent = getSemanticParent(cursor); if (semanticParent && semanticParent->findNonfunctionChild(className, &Node::isClassNode)) { return CXChildVisit_Continue; } CXCursorKind actualKind = (kind == CXCursor_ClassTemplate) ? clang_getTemplateCursorKind(cursor) : kind; Node::NodeType type = Node::Class; if (actualKind == CXCursor_StructDecl) type = Node::Struct; else if (actualKind == CXCursor_UnionDecl) type = Node::Union; ClassNode *classe = new ClassNode(type, semanticParent, className); classe->setAccess(fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor))); classe->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); if (kind == CXCursor_ClassTemplate) classe->setTemplateDecl(templateString); QScopedValueRollback setParent(parent_, classe); return visitChildren(cursor); } case CXCursor_CXXBaseSpecifier: { if (!parent_->isClassNode()) return CXChildVisit_Continue; auto access = fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor)); auto type = clang_getCursorType(cursor); auto baseCursor = clang_getTypeDeclaration(type); auto baseNode = findNodeForCursor(qdb_, baseCursor); auto classe = static_cast(parent_); if (baseNode == nullptr || !baseNode->isClassNode()) { QString bcName = reconstructQualifiedPathForCursor(baseCursor); classe->addUnresolvedBaseClass( access, bcName.split(QLatin1String("::"), Qt::SkipEmptyParts), bcName); return CXChildVisit_Continue; } auto baseClasse = static_cast(baseNode); classe->addResolvedBaseClass(access, baseClasse); return CXChildVisit_Continue; } case CXCursor_Namespace: { QString namespaceName = fromCXString(clang_getCursorDisplayName(cursor)); NamespaceNode *ns = nullptr; if (parent_) ns = static_cast( parent_->findNonfunctionChild(namespaceName, &Node::isNamespace)); if (!ns) { ns = new NamespaceNode(parent_, namespaceName); ns->setAccess(Access::Public); ns->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); } QScopedValueRollback setParent(parent_, ns); return visitChildren(cursor); } case CXCursor_FunctionTemplate: templateString = templateDecl(cursor); Q_FALLTHROUGH(); case CXCursor_FunctionDecl: case CXCursor_CXXMethod: case CXCursor_Constructor: case CXCursor_Destructor: case CXCursor_ConversionFunction: { if (findNodeForCursor(qdb_, cursor)) // Was already parsed, probably in another TU return CXChildVisit_Continue; QString name = functionName(cursor); if (ignoredSymbol(name)) return CXChildVisit_Continue; FunctionNode *fn = new FunctionNode(parent_, name); CXSourceRange range = clang_Cursor_getCommentRange(cursor); if (!clang_Range_isNull(range)) { QString comment = getSpelling(range); if (comment.startsWith("//!")) { int tag = comment.indexOf(QChar('[')); if (tag > 0) { int end = comment.indexOf(QChar(']'), tag); if (end > 0) fn->setTag(comment.mid(tag, 1 + end - tag)); } } } processFunction(fn, cursor); fn->setTemplateDecl(templateString); return CXChildVisit_Continue; } #if CINDEX_VERSION >= 36 case CXCursor_FriendDecl: { QScopedValueRollback setFriend(m_friendDecl, true); // Visit the friend functions return visitChildren(cursor); } #endif case CXCursor_EnumDecl: { EnumNode *en = static_cast(findNodeForCursor(qdb_, cursor)); if (en && en->items().count()) return CXChildVisit_Continue; // Was already parsed, probably in another TU QString enumTypeName = fromCXString(clang_getCursorSpelling(cursor)); if (enumTypeName.isEmpty()) { enumTypeName = "anonymous"; if (parent_ && (parent_->isClassNode() || parent_->isNamespace())) { Node *n = parent_->findNonfunctionChild(enumTypeName, &Node::isEnumType); if (n) en = static_cast(n); } } if (!en) { en = new EnumNode(parent_, enumTypeName, clang_EnumDecl_isScoped(cursor)); en->setAccess(fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor))); en->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); } // Enum values visitChildrenLambda(cursor, [&](CXCursor cur) { if (clang_getCursorKind(cur) != CXCursor_EnumConstantDecl) return CXChildVisit_Continue; QString value; visitChildrenLambda(cur, [&](CXCursor cur) { if (clang_isExpression(clang_getCursorKind(cur))) { value = getSpelling(clang_getCursorExtent(cur)); return CXChildVisit_Break; } return CXChildVisit_Continue; }); if (value.isEmpty()) { QLatin1String hex("0x"); if (!en->items().isEmpty() && en->items().last().value().startsWith(hex)) { value = hex + QString::number(clang_getEnumConstantDeclValue(cur), 16); } else { value = QString::number(clang_getEnumConstantDeclValue(cur)); } } en->addItem(EnumItem(fromCXString(clang_getCursorSpelling(cur)), value)); return CXChildVisit_Continue; }); return CXChildVisit_Continue; } case CXCursor_FieldDecl: case CXCursor_VarDecl: { if (findNodeForCursor(qdb_, cursor)) // Was already parsed, probably in another TU return CXChildVisit_Continue; auto access = fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor)); auto var = new VariableNode(parent_, fromCXString(clang_getCursorSpelling(cursor))); var->setAccess(access); var->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); var->setLeftType(fromCXString(clang_getTypeSpelling(clang_getCursorType(cursor)))); var->setStatic(kind == CXCursor_VarDecl && parent_->isClassNode()); return CXChildVisit_Continue; } case CXCursor_TypedefDecl: { if (findNodeForCursor(qdb_, cursor)) // Was already parsed, probably in another TU return CXChildVisit_Continue; TypedefNode *td = new TypedefNode(parent_, fromCXString(clang_getCursorSpelling(cursor))); td->setAccess(fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor))); td->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); // Search to see if this is a Q_DECLARE_FLAGS (if the type is QFlags) visitChildrenLambda(cursor, [&](CXCursor cur) { if (clang_getCursorKind(cur) != CXCursor_TemplateRef || fromCXString(clang_getCursorSpelling(cur)) != QLatin1String("QFlags")) return CXChildVisit_Continue; // Found QFlags visitChildrenLambda(cursor, [&](CXCursor cur) { if (clang_getCursorKind(cur) != CXCursor_TypeRef) return CXChildVisit_Continue; auto *en = findNodeForCursor(qdb_, clang_getTypeDeclaration(clang_getCursorType(cur))); if (en && en->isEnumType()) static_cast(en)->setFlagsType(td); return CXChildVisit_Break; }); return CXChildVisit_Break; }); return CXChildVisit_Continue; } default: if (clang_isDeclaration(kind) && parent_->isClassNode()) { // may be a property macro or a static_assert // which is not exposed from the clang API parseProperty(getSpelling(clang_getCursorExtent(cursor)), fromCXSourceLocation(loc)); } return CXChildVisit_Continue; } } void ClangVisitor::readParameterNamesAndAttributes(FunctionNode *fn, CXCursor cursor) { Parameters ¶meters = fn->parameters(); // Visit the parameters and attributes int i = 0; visitChildrenLambda(cursor, [&](CXCursor cur) { auto kind = clang_getCursorKind(cur); if (kind == CXCursor_AnnotateAttr) { QString annotation = fromCXString(clang_getCursorDisplayName(cur)); if (annotation == QLatin1String("qt_slot")) { fn->setMetaness(FunctionNode::Slot); } else if (annotation == QLatin1String("qt_signal")) { fn->setMetaness(FunctionNode::Signal); } if (annotation == QLatin1String("qt_invokable")) fn->setInvokable(true); } else if (kind == CXCursor_CXXOverrideAttr) { fn->setOverride(true); } else if (kind == CXCursor_ParmDecl) { if (i >= parameters.count()) return CXChildVisit_Break; // Attributes comes before parameters so we can break. QString name = fromCXString(clang_getCursorSpelling(cur)); if (!name.isEmpty()) { parameters[i].setName(name); // Find the default value visitChildrenLambda(cur, [&](CXCursor cur) { if (clang_isExpression(clang_getCursorKind(cur))) { QString defaultValue = getSpelling(clang_getCursorExtent(cur)); if (defaultValue.startsWith('=')) // In some cases, the = is part of the range. defaultValue = QStringView{defaultValue}.mid(1).trimmed().toString(); if (defaultValue.isEmpty()) defaultValue = QStringLiteral("..."); parameters[i].setDefaultValue(defaultValue); return CXChildVisit_Break; } return CXChildVisit_Continue; }); ++i; } } return CXChildVisit_Continue; }); } void ClangVisitor::processFunction(FunctionNode *fn, CXCursor cursor) { CXCursorKind kind = clang_getCursorKind(cursor); CXType funcType = clang_getCursorType(cursor); fn->setAccess(fromCX_CXXAccessSpecifier(clang_getCXXAccessSpecifier(cursor))); fn->setLocation(fromCXSourceLocation(clang_getCursorLocation(cursor))); if (kind == CXCursor_Constructor // a constructor template is classified as CXCursor_FunctionTemplate || (kind == CXCursor_FunctionTemplate && fn->name() == parent_->name())) fn->setMetaness(FunctionNode::Ctor); else if (kind == CXCursor_Destructor) fn->setMetaness(FunctionNode::Dtor); else fn->setReturnType(adjustTypeName( fromCXString(clang_getTypeSpelling(clang_getResultType(funcType))))); fn->setStatic(clang_CXXMethod_isStatic(cursor)); fn->setConst(clang_CXXMethod_isConst(cursor)); fn->setVirtualness(!clang_CXXMethod_isVirtual(cursor) ? FunctionNode::NonVirtual : clang_CXXMethod_isPureVirtual(cursor) ? FunctionNode::PureVirtual : FunctionNode::NormalVirtual); CXRefQualifierKind refQualKind = clang_Type_getCXXRefQualifier(funcType); if (refQualKind == CXRefQualifier_LValue) fn->setRef(true); else if (refQualKind == CXRefQualifier_RValue) fn->setRefRef(true); // For virtual functions, determine what it overrides // (except for destructor for which we do not want to classify as overridden) if (!fn->isNonvirtual() && kind != CXCursor_Destructor) setOverridesForFunction(fn, cursor); int numArg = clang_getNumArgTypes(funcType); Parameters ¶meters = fn->parameters(); parameters.clear(); parameters.reserve(numArg); for (int i = 0; i < numArg; ++i) { CXType argType = clang_getArgType(funcType, i); if (fn->isCtor()) { if (fromCXString(clang_getTypeSpelling(clang_getPointeeType(argType))) == fn->name()) { if (argType.kind == CXType_RValueReference) fn->setMetaness(FunctionNode::MCtor); else if (argType.kind == CXType_LValueReference) fn->setMetaness(FunctionNode::CCtor); } } else if ((kind == CXCursor_CXXMethod) && (fn->name() == QLatin1String("operator="))) { if (argType.kind == CXType_RValueReference) fn->setMetaness(FunctionNode::MAssign); else if (argType.kind == CXType_LValueReference) fn->setMetaness(FunctionNode::CAssign); } parameters.append(adjustTypeName(fromCXString(clang_getTypeSpelling(argType)))); if (argType.kind == CXType_Typedef || argType.kind == CXType_Elaborated) { parameters.last().setCanonicalType(fromCXString( clang_getTypeSpelling(clang_getCanonicalType(argType)))); } } if (parameters.count() > 0) { if (parameters.last().type().endsWith(QLatin1String("::QPrivateSignal"))) { parameters.pop_back(); // remove the QPrivateSignal argument parameters.setPrivateSignal(); } } if (clang_isFunctionTypeVariadic(funcType)) parameters.append(QStringLiteral("...")); readParameterNamesAndAttributes(fn, cursor); // Friend functions are not members if (m_friendDecl) fn->setRelatedNonmember(true); } bool ClangVisitor::parseProperty(const QString &spelling, const Location &loc) { if (!spelling.startsWith(QLatin1String("Q_PROPERTY")) && !spelling.startsWith(QLatin1String("QDOC_PROPERTY")) && !spelling.startsWith(QLatin1String("Q_OVERRIDE"))) return false; int lpIdx = spelling.indexOf(QChar('(')); int rpIdx = spelling.lastIndexOf(QChar(')')); if (lpIdx <= 0 || rpIdx <= lpIdx) return false; QString signature = spelling.mid(lpIdx + 1, rpIdx - lpIdx - 1); signature = signature.simplified(); QStringList part; part = signature.split(QChar(' ')); if (part.first() == QLatin1String("enum")) part.takeFirst(); // QTBUG-80027 if (part.size() < 2) return false; QString type = part.at(0); QString name = part.at(1); if (name.front() == QChar('*')) { type.append(QChar('*')); name.remove(0, 1); } auto *property = new PropertyNode(parent_, name); property->setAccess(Access::Public); property->setLocation(loc); property->setDataType(type); int i = 2; while (i < part.size()) { QString key = part.at(i++); // Keywords with no associated values if (key == "CONSTANT") { property->setConstant(); } else if (key == "FINAL") { property->setFinal(); } else if (key == "REQUIRED") { property->setRequired(); } if (i < part.size()) { QString value = part.at(i++); if (key == "READ") { qdb_->addPropertyFunction(property, value, PropertyNode::Getter); } else if (key == "WRITE") { qdb_->addPropertyFunction(property, value, PropertyNode::Setter); property->setWritable(true); } else if (key == "STORED") { property->setStored(value.toLower() == "true"); } else if (key == "DESIGNABLE") { QString v = value.toLower(); if (v == "true") property->setDesignable(true); else if (v == "false") property->setDesignable(false); else { property->setDesignable(false); property->setRuntimeDesFunc(value); } } else if (key == "BINDABLE") { property->setPropertyType(PropertyNode::Bindable); } else if (key == "RESET") { qdb_->addPropertyFunction(property, value, PropertyNode::Resetter); } else if (key == "NOTIFY") { qdb_->addPropertyFunction(property, value, PropertyNode::Notifier); } else if (key == "REVISION") { int revision; bool ok; revision = value.toInt(&ok); if (ok) property->setRevision(revision); else loc.warning(QStringLiteral("Invalid revision number: %1").arg(value)); } else if (key == "SCRIPTABLE") { QString v = value.toLower(); if (v == "true") property->setScriptable(true); else if (v == "false") property->setScriptable(false); else { property->setScriptable(false); property->setRuntimeScrFunc(value); } } } } return true; } /*! Given a comment at location \a loc, return a Node for this comment \a nextCommentLoc is the location of the next comment so the declaration must be inbetween. Returns nullptr if no suitable declaration was found between the two comments. */ Node *ClangVisitor::nodeForCommentAtLocation(CXSourceLocation loc, CXSourceLocation nextCommentLoc) { ClangVisitor::SimpleLoc docloc; clang_getPresumedLocation(loc, nullptr, &docloc.line, &docloc.column); auto decl_it = declMap_.upperBound(docloc); if (decl_it == declMap_.end()) return nullptr; unsigned int declLine = decl_it.key().line; unsigned int nextCommentLine; clang_getPresumedLocation(nextCommentLoc, nullptr, &nextCommentLine, nullptr); if (nextCommentLine < declLine) return nullptr; // there is another comment before the declaration, ignore it. // make sure the previous decl was finished. if (decl_it != declMap_.begin()) { CXSourceLocation prevDeclEnd = clang_getRangeEnd(clang_getCursorExtent(*(std::prev(decl_it)))); unsigned int prevDeclLine; clang_getPresumedLocation(prevDeclEnd, nullptr, &prevDeclLine, nullptr); if (prevDeclLine >= docloc.line) { // The previous declaration was still going. This is only valid if the previous // declaration is a parent of the next declaration. auto parent = clang_getCursorLexicalParent(*decl_it); if (!clang_equalCursors(parent, *(std::prev(decl_it)))) return nullptr; } } auto *node = findNodeForCursor(qdb_, *decl_it); // borrow the parameter name from the definition if (node && node->isFunction(Node::CPP)) readParameterNamesAndAttributes(static_cast(node), *decl_it); return node; } /*! The destructor is trivial. */ ClangCodeParser::~ClangCodeParser() { // nothing. } /*! Get the include paths from the qdoc configuration database \a config. Call the initializeParser() in the base class. Get the defines list from the qdocconf database. */ void ClangCodeParser::initializeParser() { Config &config = Config::instance(); m_version = config.getString(CONFIG_VERSION); const auto args = config.getStringList(CONFIG_INCLUDEPATHS); QSet seen; m_includePaths.clear(); // Remove empty paths and duplicates and add -I and canonicalize if necessary for (const auto &p : args) { QByteArray option; QString rawpath; if (p.startsWith(QLatin1String("-I")) || p.startsWith(QLatin1String("-F"))) { rawpath = p.mid(2).trimmed(); option = p.left(2).toUtf8(); } else if (p.startsWith(QLatin1String("-isystem"))) { rawpath = p.mid(8).trimmed(); option = "-isystem"; } else { rawpath = p; option = "-I"; } if (rawpath.isEmpty() || seen.contains(rawpath)) continue; seen.insert(rawpath); QByteArray path(rawpath.toUtf8()); QFileInfo fi(QDir::current(), rawpath); if (fi.exists()) path = fi.canonicalFilePath().toUtf8(); path.prepend(option); m_includePaths.append(path); } CppCodeParser::initializeParser(); m_pchFileDir.reset(nullptr); m_allHeaders.clear(); m_pchName.clear(); m_defines.clear(); QSet accepted; { const QStringList tmpDefines = config.getStringList(CONFIG_CLANGDEFINES); for (const QString &def : tmpDefines) { if (!accepted.contains(def)) { QByteArray tmp("-D"); tmp.append(def.toUtf8()); m_defines.append(tmp.constData()); accepted.insert(def); } } } { const QStringList tmpDefines = config.getStringList(CONFIG_DEFINES); for (const QString &def : tmpDefines) { if (!accepted.contains(def) && !def.contains(QChar('*'))) { QByteArray tmp("-D"); tmp.append(def.toUtf8()); m_defines.append(tmp.constData()); accepted.insert(def); } } } qCDebug(lcQdoc).nospace() << __FUNCTION__ << " Clang v" << CINDEX_VERSION_MAJOR << '.' << CINDEX_VERSION_MINOR; } /*! */ void ClangCodeParser::terminateParser() { CppCodeParser::terminateParser(); } /*! */ QString ClangCodeParser::language() { return "Clang"; } /*! Returns a list of extensions for header files. */ QStringList ClangCodeParser::headerFileNameFilter() { return QStringList() << "*.ch" << "*.h" << "*.h++" << "*.hh" << "*.hpp" << "*.hxx"; } /*! Returns a list of extensions for source files, i.e. not header files. */ QStringList ClangCodeParser::sourceFileNameFilter() { return QStringList() << "*.c++" << "*.cc" << "*.cpp" << "*.cxx" << "*.mm"; } /*! Parse the C++ header file identified by \a filePath and add the parsed contents to the database. The \a location is used for reporting errors. */ void ClangCodeParser::parseHeaderFile(const Location & /*location*/, const QString &filePath) { QFileInfo fi(filePath); m_allHeaders.insert(fi.fileName(), fi.canonicalPath()); } static const char *defaultArgs_[] = { "-std=c++17", #ifndef Q_OS_WIN "-fPIC", #else "-fms-compatibility-version=19", #endif "-DQ_QDOC", "-DQ_CLANG_QDOC", "-DQT_DISABLE_DEPRECATED_BEFORE=0", "-DQT_ANNOTATE_CLASS(type,...)=static_assert(sizeof(#__VA_ARGS__),#type);", "-DQT_ANNOTATE_CLASS2(type,a1,a2)=static_assert(sizeof(#a1,#a2),#type);", "-DQT_ANNOTATE_FUNCTION(a)=__attribute__((annotate(#a)))", "-DQT_ANNOTATE_ACCESS_SPECIFIER(a)=__attribute__((annotate(#a)))", "-Wno-constant-logical-operand", "-Wno-macro-redefined", "-Wno-nullability-completeness", "-fvisibility=default", "-ferror-limit=0", "-I" CLANG_RESOURCE_DIR }; /*! Load the default arguments and the defines into \a args. Clear \a args first. */ void ClangCodeParser::getDefaultArgs() { m_args.clear(); m_args.insert(m_args.begin(), std::begin(defaultArgs_), std::end(defaultArgs_)); // Add the defines from the qdocconf file. for (const auto &p : qAsConst(m_defines)) m_args.push_back(p.constData()); } static QList includePathsFromHeaders(const QHash &allHeaders) { QList result; for (auto it = allHeaders.cbegin(); it != allHeaders.cend(); ++it) { const QByteArray path = "-I" + it.value().toLatin1(); const QByteArray parent = "-I" + QDir::cleanPath(it.value() + QLatin1String("/../")).toLatin1(); if (!result.contains(path)) result.append(path); if (!result.contains(parent)) result.append(parent); } return result; } /*! Load the include paths into \a moreArgs. If no include paths were provided, try to guess reasonable include paths. */ void ClangCodeParser::getMoreArgs() { if (m_includePaths.isEmpty()) { /* The include paths provided are inadequate. Make a list of reasonable places to look for include files and use that list instead. */ qCWarning(lcQdoc) << "No include paths passed to qdoc; guessing reasonable include paths"; auto forest = qdb_->searchOrder(); QByteArray version = qdb_->version().toUtf8(); QString basicIncludeDir = QDir::cleanPath(QString(Config::installDir + "/../include")); m_moreArgs += "-I" + basicIncludeDir.toLatin1(); m_moreArgs += includePathsFromHeaders(m_allHeaders); } else { m_moreArgs = m_includePaths; } } /*! Building the PCH must be possible when there are no .cpp files, so it is moved here to its own member function, and it is called after the list of header files is complete. */ void ClangCodeParser::buildPCH() { if (!m_pchFileDir && !moduleHeader().isEmpty()) { m_pchFileDir.reset(new QTemporaryDir(QDir::tempPath() + QLatin1String("/qdoc_pch"))); if (m_pchFileDir->isValid()) { const QByteArray module = moduleHeader().toUtf8(); QByteArray header; QByteArray privateHeaderDir; qCDebug(lcQdoc) << "Build and visit PCH for" << moduleHeader(); // A predicate for std::find_if() to locate a path to the module's header // (e.g. QtGui/QtGui) to be used as pre-compiled header struct FindPredicate { enum SearchType { Any, Module, Private }; QByteArray &candidate_; const QByteArray &module_; SearchType type_; FindPredicate(QByteArray &candidate, const QByteArray &module, SearchType type = Any) : candidate_(candidate), module_(module), type_(type) { } bool operator()(const QByteArray &p) const { if (type_ != Any && !p.endsWith(module_)) return false; candidate_ = p + "/"; switch (type_) { case Any: case Module: candidate_.append(module_); break; case Private: candidate_.append("private"); break; default: break; } if (p.startsWith("-I")) candidate_ = candidate_.mid(2); return QFile::exists(QString::fromUtf8(candidate_)); } }; // First, search for an include path that contains the module name, then any path QByteArray candidate; auto it = std::find_if(m_includePaths.begin(), m_includePaths.end(), FindPredicate(candidate, module, FindPredicate::Module)); if (it == m_includePaths.end()) it = std::find_if(m_includePaths.begin(), m_includePaths.end(), FindPredicate(candidate, module, FindPredicate::Any)); if (it != m_includePaths.end()) header = candidate; // Find the path to module's private headers - currently unused it = std::find_if(m_includePaths.begin(), m_includePaths.end(), FindPredicate(candidate, module, FindPredicate::Private)); if (it != m_includePaths.end()) privateHeaderDir = candidate; if (header.isEmpty()) { qWarning() << "(qdoc) Could not find the module header in include paths for module" << module << " (include paths: " << m_includePaths << ")"; qWarning() << " Artificial module header built from header dirs in qdocconf " "file"; } m_args.push_back("-xc++"); CXTranslationUnit tu; QString tmpHeader = m_pchFileDir->path() + "/" + module; QFile tmpHeaderFile(tmpHeader); if (tmpHeaderFile.open(QIODevice::Text | QIODevice::WriteOnly)) { QTextStream out(&tmpHeaderFile); if (header.isEmpty()) { for (auto it = m_allHeaders.constKeyValueBegin(); it != m_allHeaders.constKeyValueEnd(); ++it) { if (!(*it).first.endsWith(QLatin1String("_p.h")) && !(*it).first.startsWith(QLatin1String("moc_"))) { QString line = QLatin1String("#include \"") + (*it).second + QLatin1String("/") + (*it).first + QLatin1String("\""); out << line << "\n"; } } } else { QFileInfo headerFile(header); if (!headerFile.exists()) { qWarning() << "Could not find module header file" << header; return; } out << QLatin1String("#include \"") + header + QLatin1String("\""); } tmpHeaderFile.close(); } CXErrorCode err = clang_parseTranslationUnit2(index_, tmpHeader.toLatin1().data(), m_args.data(), static_cast(m_args.size()), nullptr, 0, flags_ | CXTranslationUnit_ForSerialization, &tu); qCDebug(lcQdoc) << __FUNCTION__ << "clang_parseTranslationUnit2(" << tmpHeader << m_args << ") returns" << err; printDiagnostics(tu); if (!err && tu) { m_pchName = m_pchFileDir->path().toUtf8() + "/" + module + ".pch"; auto error = clang_saveTranslationUnit(tu, m_pchName.constData(), clang_defaultSaveOptions(tu)); if (error) { qCCritical(lcQdoc) << "Could not save PCH file for" << moduleHeader(); m_pchName.clear(); } else { // Visit the header now, as token from pre-compiled header won't be visited // later CXCursor cur = clang_getTranslationUnitCursor(tu); ClangVisitor visitor(qdb_, m_allHeaders); visitor.visitChildren(cur); qCDebug(lcQdoc) << "PCH built and visited for" << moduleHeader(); } clang_disposeTranslationUnit(tu); } else { m_pchFileDir->remove(); qCCritical(lcQdoc) << "Could not create PCH file for " << moduleHeader(); } m_args.pop_back(); // remove the "-xc++"; } } } /*! Precompile the header files for the current module. */ void ClangCodeParser::precompileHeaders() { getDefaultArgs(); getMoreArgs(); for (const auto &p : qAsConst(m_moreArgs)) m_args.push_back(p.constData()); flags_ = static_cast(CXTranslationUnit_Incomplete | CXTranslationUnit_SkipFunctionBodies | CXTranslationUnit_KeepGoing); index_ = clang_createIndex(1, kClangDontDisplayDiagnostics); buildPCH(); clang_disposeIndex(index_); } static float getUnpatchedVersion(QString t) { if (t.count(QChar('.')) > 1) t.truncate(t.lastIndexOf(QChar('.'))); return t.toFloat(); } /*! Get ready to parse the C++ cpp file identified by \a filePath and add its parsed contents to the database. \a location is used for reporting errors. Call matchDocsAndStuff() to do all the parsing and tree building. */ void ClangCodeParser::parseSourceFile(const Location & /*location*/, const QString &filePath) { /* The set of open namespaces is cleared before parsing each source file. The word "source" here means cpp file. */ qdb_->clearOpenNamespaces(); currentFile_ = filePath; flags_ = static_cast(CXTranslationUnit_Incomplete | CXTranslationUnit_SkipFunctionBodies | CXTranslationUnit_KeepGoing); index_ = clang_createIndex(1, kClangDontDisplayDiagnostics); getDefaultArgs(); if (!m_pchName.isEmpty() && !filePath.endsWith(".mm")) { m_args.push_back("-w"); m_args.push_back("-include-pch"); m_args.push_back(m_pchName.constData()); } getMoreArgs(); for (const auto &p : qAsConst(m_moreArgs)) m_args.push_back(p.constData()); CXTranslationUnit tu; CXErrorCode err = clang_parseTranslationUnit2(index_, filePath.toLocal8Bit(), m_args.data(), static_cast(m_args.size()), nullptr, 0, flags_, &tu); qCDebug(lcQdoc) << __FUNCTION__ << "clang_parseTranslationUnit2(" << filePath << m_args << ") returns" << err; printDiagnostics(tu); if (err || !tu) { qWarning() << "(qdoc) Could not parse source file" << filePath << " error code:" << err; clang_disposeIndex(index_); return; } CXCursor tuCur = clang_getTranslationUnitCursor(tu); ClangVisitor visitor(qdb_, m_allHeaders); visitor.visitChildren(tuCur); CXToken *tokens; unsigned int numTokens = 0; const QSet &commands = topicCommands() + metaCommands(); clang_tokenize(tu, clang_getCursorExtent(tuCur), &tokens, &numTokens); for (unsigned int i = 0; i < numTokens; ++i) { if (clang_getTokenKind(tokens[i]) != CXToken_Comment) continue; QString comment = fromCXString(clang_getTokenSpelling(tu, tokens[i])); if (!comment.startsWith("/*!")) continue; auto commentLoc = clang_getTokenLocation(tu, tokens[i]); auto loc = fromCXSourceLocation(commentLoc); auto end_loc = fromCXSourceLocation(clang_getRangeEnd(clang_getTokenExtent(tu, tokens[i]))); Doc::trimCStyleComment(loc, comment); // Doc constructor parses the comment. Doc doc(loc, end_loc, comment, commands, topicCommands()); if (hasTooManyTopics(doc)) continue; DocList docs; QString topic; NodeList nodes; const TopicList &topics = doc.topicsUsed(); if (!topics.isEmpty()) topic = topics[0].topic; if (topic.isEmpty()) { Node *n = nullptr; if (i + 1 < numTokens) { // Try to find the next declaration. CXSourceLocation nextCommentLoc = commentLoc; while (i + 2 < numTokens && clang_getTokenKind(tokens[i + 1]) != CXToken_Comment) ++i; // already skip all the tokens that are not comments nextCommentLoc = clang_getTokenLocation(tu, tokens[i + 1]); n = visitor.nodeForCommentAtLocation(commentLoc, nextCommentLoc); } if (n) { nodes.append(n); docs.append(doc); } else if (CodeParser::isWorthWarningAbout(doc)) { bool future = false; if (doc.metaCommandsUsed().contains(COMMAND_SINCE)) { QString sinceVersion = doc.metaCommandArgs(COMMAND_SINCE)[0].first; if (getUnpatchedVersion(sinceVersion) > getUnpatchedVersion(m_version)) future = true; } if (!future) { doc.location().warning( QStringLiteral("Cannot tie this documentation to anything"), QStringLiteral("qdoc found a /*! ... */ comment, but there was no " "topic command (e.g., '\\%1', '\\%2') in the " "comment and no function definition following " "the comment.") .arg(COMMAND_FN) .arg(COMMAND_PAGE)); } } } else { // Store the namespace scope from lexical parents of the comment m_namespaceScope.clear(); CXCursor cur = clang_getCursor(tu, commentLoc); while (true) { CXCursorKind kind = clang_getCursorKind(cur); if (clang_isTranslationUnit(kind) || clang_isInvalid(kind)) break; if (kind == CXCursor_Namespace) m_namespaceScope << fromCXString(clang_getCursorSpelling(cur)); cur = clang_getCursorLexicalParent(cur); } processTopicArgs(doc, topic, nodes, docs); } processMetaCommands(nodes, docs); } clang_disposeTokens(tu, tokens, numTokens); clang_disposeTranslationUnit(tu); clang_disposeIndex(index_); m_namespaceScope.clear(); s_fn.clear(); } /*! Use clang to parse the function signature from a function command. \a location is used for reporting errors. \a fnArg is the string to parse. It is always a function decl. */ Node *ClangCodeParser::parseFnArg(const Location &location, const QString &fnArg) { Node *fnNode = nullptr; /* If the \fn command begins with a tag, then don't try to parse the \fn command with clang. Use the tag to search for the correct function node. It is an error if it can not be found. Return 0 in that case. */ if (fnArg.startsWith('[')) { int end = fnArg.indexOf(QChar(']', 0)); if (end > 1) { QString tag = fnArg.left(end + 1); fnNode = qdb_->findFunctionNodeForTag(tag); if (!fnNode) { location.error( QStringLiteral("tag \\fn %1 not used in any include file in current module") .arg(tag)); } else { /* The function node was found. Use the formal parameter names from the \FN command, because they will be the names used in the documentation. */ FunctionNode *fn = static_cast(fnNode); QStringList leftParenSplit = fnArg.split('('); if (leftParenSplit.size() > 1) { QStringList rightParenSplit = leftParenSplit[1].split(')'); if (rightParenSplit.size() > 0) { QString params = rightParenSplit[0]; if (!params.isEmpty()) { QStringList commaSplit = params.split(','); Parameters ¶meters = fn->parameters(); if (parameters.count() == commaSplit.size()) { for (int i = 0; i < parameters.count(); ++i) { QStringList blankSplit = commaSplit[i].split(' '); if (blankSplit.size() > 0) { QString pName = blankSplit.last(); int j = 0; while (j < pName.length() && !pName.at(j).isLetter()) ++j; if (j > 0) pName = pName.mid(j); if (!pName.isEmpty() && pName != parameters[i].name()) parameters[i].setName(pName); } } } } } } } } return fnNode; } CXTranslationUnit_Flags flags = static_cast( CXTranslationUnit_Incomplete | CXTranslationUnit_SkipFunctionBodies | CXTranslationUnit_KeepGoing); CXIndex index = clang_createIndex(1, kClangDontDisplayDiagnostics); std::vector args(std::begin(defaultArgs_), std::end(defaultArgs_)); // Add the defines from the qdocconf file. for (const auto &p : qAsConst(m_defines)) args.push_back(p.constData()); if (!m_pchName.isEmpty()) { args.push_back("-w"); args.push_back("-include-pch"); args.push_back(m_pchName.constData()); } CXTranslationUnit tu; s_fn.clear(); for (const auto &ns : qAsConst(m_namespaceScope)) s_fn.prepend("namespace " + ns.toUtf8() + " {"); s_fn += fnArg.toUtf8(); if (!s_fn.endsWith(";")) s_fn += "{ }"; s_fn.append(m_namespaceScope.size(), '}'); const char *dummyFileName = fnDummyFileName; CXUnsavedFile unsavedFile { dummyFileName, s_fn.constData(), static_cast(s_fn.size()) }; CXErrorCode err = clang_parseTranslationUnit2(index, dummyFileName, args.data(), args.size(), &unsavedFile, 1, flags, &tu); qCDebug(lcQdoc) << __FUNCTION__ << "clang_parseTranslationUnit2(" << dummyFileName << args << ") returns" << err; printDiagnostics(tu); if (err || !tu) { location.error(QStringLiteral("clang could not parse \\fn %1").arg(fnArg)); clang_disposeTranslationUnit(tu); clang_disposeIndex(index); return fnNode; } else { /* Always visit the tu if one is constructed, because it might be possible to find the correct node, even if clang detected diagnostics. Only bother to report the diagnostics if they stop us finding the node. */ CXCursor cur = clang_getTranslationUnitCursor(tu); ClangVisitor visitor(qdb_, m_allHeaders); bool ignoreSignature = false; visitor.visitFnArg(cur, &fnNode, ignoreSignature); /* If the visitor couldn't find a FunctionNode for the signature, then print the clang diagnostics if there were any. */ if (fnNode == nullptr) { unsigned diagnosticCount = clang_getNumDiagnostics(tu); const auto &config = Config::instance(); if (diagnosticCount > 0 && (!config.preparing() || config.singleExec())) { bool report = true; QStringList signature = fnArg.split(QChar('(')); if (signature.size() > 1) { QStringList qualifiedName = signature.at(0).split(QChar(' ')); qualifiedName = qualifiedName.last().split(QLatin1String("::")); if (qualifiedName.size() > 1) { QString qualifier = qualifiedName.at(0); int i = 0; while (qualifier.size() > i && !qualifier.at(i).isLetter()) qualifier[i++] = QChar(' '); if (i > 0) qualifier = qualifier.simplified(); ClassNode *cn = qdb_->findClassNode(QStringList(qualifier)); if (cn && cn->isInternal()) report = false; } } if (report) { location.warning( QStringLiteral("clang couldn't find function when parsing \\fn %1").arg(fnArg)); } } } } clang_disposeTranslationUnit(tu); clang_disposeIndex(index); return fnNode; } void ClangCodeParser::printDiagnostics(const CXTranslationUnit &translationUnit) const { if (!lcQdocClang().isDebugEnabled()) return; static const auto displayOptions = CXDiagnosticDisplayOptions::CXDiagnostic_DisplaySourceLocation | CXDiagnosticDisplayOptions::CXDiagnostic_DisplayColumn | CXDiagnosticDisplayOptions::CXDiagnostic_DisplayOption; for (unsigned i = 0, numDiagnostics = clang_getNumDiagnostics(translationUnit); i < numDiagnostics; ++i) { auto diagnostic = clang_getDiagnostic(translationUnit, i); auto formattedDiagnostic = clang_formatDiagnostic(diagnostic, displayOptions); qCDebug(lcQdocClang) << clang_getCString(formattedDiagnostic); clang_disposeString(formattedDiagnostic); } } QT_END_NAMESPACE