/* * Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "cppparser.h" #include "parserutils.h" #include "../utils.h" #include "qsynedit/syntaxer/cpp.h" #include #include #include #include #include #include static QAtomicInt cppParserCount(0); CppParser::CppParser(QObject *parent) : QObject(parent) { mParserId = cppParserCount.fetchAndAddRelaxed(1); mLanguage = ParserLanguage::CPlusPlus; mSerialCount = 0; updateSerialId(); mUniqId = 0; mParsing = false; //mStatementList ; // owns the objects //mFilesToScan; //mIncludePaths; //mProjectIncludePaths; //mProjectFiles; // mCurrentScope; //mCurrentClassScope; //mSkipList; mParseLocalHeaders = true; mParseGlobalHeaders = true; mLockCount = 0; mIsSystemHeader = false; mIsHeader = false; mIsProjectFile = false; mCppKeywords = CppKeywords; mCppTypeKeywords = CppTypeKeywords; mEnabled = true; internalClear(); //mNamespaces; //mBlockBeginSkips; //mBlockEndSkips; //mInlineNamespaceEndSkips; } CppParser::~CppParser() { while (true) { //wait for all methods finishes running { QMutexLocker locker(&mMutex); if (!mParsing && (mLockCount == 0)) { mParsing = true; break; } } QThread::msleep(50); QCoreApplication* app = QApplication::instance(); app->processEvents(); } //qDebug()<<"-------- parser deleted ------------"; } void CppParser::addHardDefineByLine(const QString &line) { QMutexLocker locker(&mMutex); if (line.startsWith('#')) { mPreprocessor.addHardDefineByLine(line.mid(1).trimmed()); } else { mPreprocessor.addHardDefineByLine(line); } } void CppParser::addIncludePath(const QString &value) { QMutexLocker locker(&mMutex); mPreprocessor.addIncludePath(includeTrailingPathDelimiter(value)); } void CppParser::removeProjectFile(const QString &value) { QMutexLocker locker(&mMutex); mProjectFiles.remove(value); mFilesToScan.remove(value); } void CppParser::addProjectIncludePath(const QString &value) { QMutexLocker locker(&mMutex); mPreprocessor.addProjectIncludePath(includeTrailingPathDelimiter(value)); } void CppParser::clearIncludePaths() { QMutexLocker locker(&mMutex); mPreprocessor.clearIncludePaths(); } void CppParser::clearProjectIncludePaths() { QMutexLocker locker(&mMutex); mPreprocessor.clearProjectIncludePaths(); } void CppParser::clearProjectFiles() { QMutexLocker locker(&mMutex); mProjectFiles.clear(); } QList CppParser::getListOfFunctions(const QString &fileName, const QString &phrase, int line) { QMutexLocker locker(&mMutex); QList result; if (mParsing) return result; PStatement statement = doFindStatementOf(fileName,phrase, line); if (!statement) return result; PStatement parentScope; if (statement->kind == StatementKind::skClass) { parentScope = statement; } else parentScope = statement->parentScope.lock(); if (parentScope && parentScope->kind == StatementKind::skNamespace) { PStatementList namespaceStatementsList = doFindNamespace(parentScope->command); if (namespaceStatementsList) { for (PStatement& namespaceStatement : *namespaceStatementsList) { result.append( getListOfFunctions(fileName,line,statement,namespaceStatement)); } } } else result.append( getListOfFunctions(fileName,line,statement,parentScope) ); return result; } PStatement CppParser::findScopeStatement(const QString &filename, int line) { QMutexLocker locker(&mMutex); if (mParsing) { return PStatement(); } return doFindScopeStatement(filename,line); } PStatement CppParser::doFindScopeStatement(const QString &filename, int line) const { PFileIncludes fileIncludes = mPreprocessor.includesList().value(filename); if (!fileIncludes) return PStatement(); return fileIncludes->scopes.findScopeAtLine(line); } PFileIncludes CppParser::findFileIncludes(const QString &filename, bool deleteIt) { QMutexLocker locker(&mMutex); PFileIncludes fileIncludes = mPreprocessor.includesList().value(filename,PFileIncludes()); if (deleteIt && fileIncludes) mPreprocessor.includesList().remove(filename); return fileIncludes; } QString CppParser::findFirstTemplateParamOf(const QString &fileName, const QString &phrase, const PStatement& currentScope) { QMutexLocker locker(&mMutex); if (mParsing) return ""; return doFindFirstTemplateParamOf(fileName,phrase,currentScope); } PStatement CppParser::findFunctionAt(const QString &fileName, int line) { QMutexLocker locker(&mMutex); PFileIncludes fileIncludes = mPreprocessor.includesList().value(fileName); if (!fileIncludes) return PStatement(); for (PStatement& statement : fileIncludes->statements) { if (statement->kind != StatementKind::skFunction && statement->kind != StatementKind::skConstructor && statement->kind != StatementKind::skDestructor) continue; if (statement->line == line || statement->definitionLine == line) return statement; } return PStatement(); } int CppParser::findLastOperator(const QString &phrase) const { int phraseLength = phrase.length(); int i = phraseLength-1; // Obtain stuff after first operator while (i>=0) { if ((i+1') && (phrase[i] == '-')) return i; else if ((i+1kind == StatementKind::skNamespace) { PStatementList lst = doFindNamespace(parentStatement->fullName); foreach (const PStatement& namespaceStatement, *lst) { statement = findMemberOfStatement(phrase,namespaceStatement); if (statement) break; } } else { statement = findMemberOfStatement(phrase,parentStatement); } if (!statement) return PStatement(); parentStatement = statement; } return statement; } PStatement CppParser::findStatementOf(const QString &fileName, const QString &phrase, int line) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return doFindStatementOf(fileName,phrase,line); } PStatement CppParser::doFindStatementOf(const QString &fileName, const QString &phrase, int line) const { return doFindStatementOf(fileName,phrase,doFindScopeStatement(fileName,line)); } PStatement CppParser::findStatementOf(const QString &fileName, const QString &phrase, const PStatement& currentScope, PStatement &parentScopeType) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return doFindStatementOf(fileName,phrase,currentScope,parentScopeType); } PStatement CppParser::doFindStatementOf(const QString &fileName, const QString &phrase, const PStatement& currentScope, PStatement &parentScopeType) const { PStatement result; parentScopeType = currentScope; //find the start scope statement QString namespaceName, remainder; QString nextScopeWord,operatorToken,memberName; PStatement statement; getFullNamespace(phrase, namespaceName, remainder); if (!namespaceName.isEmpty()) { // (namespace )qualified Name PStatementList namespaceList = mNamespaces.value(namespaceName); if (!namespaceList || namespaceList->isEmpty()) return PStatement(); if (remainder.isEmpty()) return namespaceList->front(); remainder = splitPhrase(remainder,nextScopeWord,operatorToken,memberName); for (PStatement& currentNamespace: *namespaceList) { statement = findMemberOfStatement(nextScopeWord,currentNamespace); if (statement) break; } //not found in namespaces; if (!statement) return PStatement(); // found in namespace } else if ((phrase.length()>2) && (phrase[0]==':') && (phrase[1]==':')) { //global remainder= phrase.mid(2); remainder= splitPhrase(remainder,nextScopeWord,operatorToken,memberName); statement= findMemberOfStatement(nextScopeWord,PStatement()); if (!statement) return PStatement(); } else { //unqualified name parentScopeType = currentScope; remainder = splitPhrase(remainder,nextScopeWord,operatorToken,memberName); statement = findStatementStartingFrom(fileName,nextScopeWord,parentScopeType); if (!statement) return PStatement(); } parentScopeType = currentScope; if (!memberName.isEmpty() && (statement->kind == StatementKind::skTypedef)) { PStatement typeStatement = doFindTypeDefinitionOf(fileName,statement->type, parentScopeType); if (typeStatement) statement = typeStatement; } //using alias like 'using std::vector;' if (statement->kind == StatementKind::skAlias) { statement = doFindAliasedStatement(statement); if (!statement) return PStatement(); } if (statement->kind == StatementKind::skConstructor) { // we need the class, not the construtor statement = statement->parentScope.lock(); if (!statement) return PStatement(); } PStatement lastScopeStatement; QString typeName; PStatement typeStatement; while (!memberName.isEmpty()) { if (statement->kind!=StatementKind::skClass && operatorToken == "::") { return PStatement(); } if (statement->kind == StatementKind::skVariable || statement->kind == StatementKind::skParameter || statement->kind == StatementKind::skFunction) { bool isSTLContainerFunctions = false; if (statement->kind == StatementKind::skFunction){ PStatement parentScope = statement->parentScope.lock(); if (parentScope && STLContainers.contains(parentScope->fullName) && STLElementMethods.contains(statement->command) && lastScopeStatement) { isSTLContainerFunctions = true; PStatement lastScopeParent = lastScopeStatement->parentScope.lock(); typeName=doFindFirstTemplateParamOf(fileName,lastScopeStatement->type, lastScopeParent ); typeStatement=doFindTypeDefinitionOf(fileName, typeName, lastScopeParent ); } else if (parentScope && STLMaps.contains(parentScope->fullName) && STLElementMethods.contains(statement->command) && lastScopeStatement) { isSTLContainerFunctions = true; PStatement lastScopeParent = lastScopeStatement->parentScope.lock(); typeName=doFindTemplateParamOf(fileName,lastScopeStatement->type,1, lastScopeParent ); typeStatement=doFindTypeDefinitionOf(fileName, typeName, lastScopeParent ); } } if (!isSTLContainerFunctions) typeStatement = doFindTypeDefinitionOf(fileName,statement->type, parentScopeType); //it's stl smart pointer if ((typeStatement) && STLPointers.contains(typeStatement->fullName) && (operatorToken == "->")) { PStatement parentScope = statement->parentScope.lock(); typeName=doFindFirstTemplateParamOf(fileName,statement->type, parentScope); typeStatement=doFindTypeDefinitionOf(fileName, typeName,parentScope); } else if ((typeStatement) && STLContainers.contains(typeStatement->fullName) && nextScopeWord.endsWith(']')) { //it's a std container PStatement parentScope = statement->parentScope.lock(); typeName = doFindFirstTemplateParamOf(fileName,statement->type, parentScope); typeStatement = doFindTypeDefinitionOf(fileName, typeName, parentScope); } else if ((typeStatement) && STLMaps.contains(typeStatement->fullName) && nextScopeWord.endsWith(']')) { //it's a std container PStatement parentScope = statement->parentScope.lock(); typeName = doFindFirstTemplateParamOf(fileName,statement->type, parentScope); typeStatement = doFindTypeDefinitionOf(fileName, typeName, parentScope); } lastScopeStatement = statement; if (typeStatement) statement = typeStatement; } else lastScopeStatement = statement; remainder = splitPhrase(remainder,nextScopeWord,operatorToken,memberName); PStatement memberStatement = findMemberOfStatement(nextScopeWord,statement); if (!memberStatement) return PStatement(); parentScopeType=statement; statement = memberStatement; if (!memberName.isEmpty() && (statement->kind == StatementKind::skTypedef)) { PStatement typeStatement = doFindTypeDefinitionOf(fileName,statement->type, parentScopeType); if (typeStatement) statement = typeStatement; } } return statement; } PEvalStatement CppParser::evalExpression( const QString &fileName, QStringList &phraseExpression, const PStatement ¤tScope) { QMutexLocker locker(&mMutex); if (mParsing) return PEvalStatement(); // qDebug()<effectiveTypeStatement && ownerEvalStatement->effectiveTypeStatement->kind == StatementKind::skNamespace) { PStatementList lst = doFindNamespace(ownerEvalStatement->effectiveTypeStatement->fullName); foreach (const PStatement& namespaceStatement, *lst) { PStatement statement = findMemberOfStatement(phrase,namespaceStatement); if (statement) return statement; } return PStatement(); } return findMemberOfStatement(phrase, ownerEvalStatement->effectiveTypeStatement); } } PStatement CppParser::findStatementOf(const QString &fileName, const QStringList &expression, int line) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return doFindStatementOf(fileName,expression,line); } PStatement CppParser::doFindStatementOf(const QString &fileName, const QStringList &expression, int line) const { PStatement statement = doFindStatementOf(fileName,expression,doFindScopeStatement(fileName,line)); if (statement && statement->line != line && statement->definitionLine != line) { PStatement parentStatement = statement->parentScope.lock(); if (parentStatement && (parentStatement->line == line && parentStatement->fileName == fileName)) { return parentStatement; } } return statement; } PStatement CppParser::findAliasedStatement(const PStatement &statement) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return doFindAliasedStatement(statement); } PStatement CppParser::doFindAliasedStatement(const PStatement &statement) const { if (!statement) return PStatement(); QString alias = statement->type; int pos = statement->type.lastIndexOf("::"); if (pos<0) return PStatement(); QString nsName=statement->type.mid(0,pos); QString name = statement->type.mid(pos+2); PStatementList namespaceStatements = doFindNamespace(nsName); if (!namespaceStatements) return PStatement(); foreach (const PStatement& namespaceStatement, *namespaceStatements) { QList resultList = findMembersOfStatement(name,namespaceStatement); foreach(const PStatement& resultStatement,resultList) { if (resultStatement->kind != StatementKind::skAlias) return resultStatement; } } return PStatement(); } PStatement CppParser::findStatementStartingFrom(const QString &fileName, const QString &phrase, const PStatement& startScope) const { PStatement scopeStatement = startScope; // repeat until reach global PStatement result; while (scopeStatement) { //search members of current scope result = findStatementInScope(phrase, scopeStatement); if (result) return result; // not found // search members of all usings (in current scope ) foreach (const QString& namespaceName, scopeStatement->usingList) { result = findStatementInNamespace(phrase,namespaceName); if (result) return result; } scopeStatement = scopeStatement->parentScope.lock(); } // Search all global members result = findMemberOfStatement(phrase,PStatement()); if (result) return result; //Find in all global usings const QSet& fileUsings = internalGetFileUsings(fileName); // add members of all fusings for (const QString& namespaceName:fileUsings) { result = findStatementInNamespace(phrase,namespaceName); if (result) return result; } return PStatement(); } PStatement CppParser::findTypeDefinitionOf(const QString &fileName, const QString &aType, const PStatement& currentClass) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return doFindTypeDefinitionOf(fileName,aType,currentClass); } PStatement CppParser::findTypeDef(const PStatement &statement, const QString &fileName) { QMutexLocker locker(&mMutex); if (mParsing) return PStatement(); return getTypeDef(statement, fileName, ""); } bool CppParser::freeze() { QMutexLocker locker(&mMutex); if (mParsing) return false; mLockCount++; return true; } bool CppParser::freeze(const QString &serialId) { QMutexLocker locker(&mMutex); if (mParsing) return false; if (mSerialId!=serialId) return false; mLockCount++; return true; } QStringList CppParser::getClassesList() { QMutexLocker locker(&mMutex); QStringList list; return list; // fills List with a list of all the known classes QQueue queue; queue.enqueue(PStatement()); while (!queue.isEmpty()) { PStatement statement = queue.dequeue(); StatementMap statementMap = mStatementList.childrenStatements(statement); for (PStatement& child:statementMap) { if (child->kind == StatementKind::skClass) list.append(child->command); if (!child->children.isEmpty()) queue.enqueue(child); } } return list; } QStringList CppParser::getFileDirectIncludes(const QString &filename) { QMutexLocker locker(&mMutex); if (mParsing) return QStringList(); if (filename.isEmpty()) return QStringList(); PFileIncludes fileIncludes = mPreprocessor.includesList().value(filename,PFileIncludes()); if (fileIncludes) { return fileIncludes->directIncludes; } return QStringList(); } QSet CppParser::getFileIncludes(const QString &filename) { QMutexLocker locker(&mMutex); QSet list; if (mParsing) return list; if (filename.isEmpty()) return list; list.insert(filename); PFileIncludes fileIncludes = mPreprocessor.includesList().value(filename,PFileIncludes()); if (fileIncludes) { foreach (const QString& file, fileIncludes->includeFiles.keys()) { list.insert(file); } } return list; } QSet CppParser::getFileUsings(const QString &filename) { QMutexLocker locker(&mMutex); return internalGetFileUsings(filename); } QSet CppParser::internalGetFileUsings(const QString &filename) const { QSet result; if (filename.isEmpty()) return result; if (mParsing) return result; PFileIncludes fileIncludes= mPreprocessor.includesList().value(filename,PFileIncludes()); if (fileIncludes) { foreach (const QString& usingName, fileIncludes->usings) { result.insert(usingName); } foreach (const QString& subFile,fileIncludes->includeFiles.keys()){ PFileIncludes subIncludes = mPreprocessor.includesList().value(subFile,PFileIncludes()); if (subIncludes) { foreach (const QString& usingName, subIncludes->usings) { result.insert(usingName); } } } } return result; } QString CppParser::getHeaderFileName(const QString &relativeTo, const QString &headerName, bool fromNext) { QMutexLocker locker(&mMutex); QString currentDir = includeTrailingPathDelimiter(extractFileDir(relativeTo)); QStringList includes; QStringList projectIncludes; bool found=false; if (fromNext && mPreprocessor.includePaths().contains(currentDir)) { foreach(const QString& s, mPreprocessor.includePathList()) { if (found) { includes.append(s); continue; } else if (s == currentDir) found = true; } projectIncludes = mPreprocessor.projectIncludePathList(); } else if (fromNext && mPreprocessor.projectIncludePaths().contains(currentDir)) { includes = mPreprocessor.includePathList(); foreach(const QString& s, mPreprocessor.projectIncludePathList()) { if (found) { includes.append(s); continue; } else if (s == currentDir) found = true; } } else { includes = mPreprocessor.includePathList(); projectIncludes = mPreprocessor.projectIncludePathList(); } return ::getHeaderFilename(relativeTo, headerName, includes, projectIncludes); } void CppParser::invalidateFile(const QString &fileName) { if (!mEnabled) return; { QMutexLocker locker(&mMutex); if (mParsing || mLockCount>0) return; updateSerialId(); mParsing = true; } QSet files = calculateFilesToBeReparsed(fileName); internalInvalidateFiles(files); mParsing = false; } bool CppParser::isIncludeLine(const QString &line) { QString trimmedLine = line.trimmed(); if ((trimmedLine.length() > 0) && trimmedLine.startsWith('#')) { // it's a preprocessor line if (trimmedLine.mid(1).trimmed().startsWith("include")) return true; } return false; } bool CppParser::isIncludeNextLine(const QString &line) { QString trimmedLine = line.trimmed(); if ((trimmedLine.length() > 0) && trimmedLine.startsWith('#')) { // it's a preprocessor line if (trimmedLine.mid(1).trimmed().startsWith("include_next")) return true; } return false; } bool CppParser::isProjectHeaderFile(const QString &fileName) { QMutexLocker locker(&mMutex); return ::isSystemHeaderFile(fileName,mPreprocessor.projectIncludePaths()); } bool CppParser::isSystemHeaderFile(const QString &fileName) { QMutexLocker locker(&mMutex); return ::isSystemHeaderFile(fileName,mPreprocessor.includePaths()); } void CppParser::parseFile(const QString &fileName, bool inProject, bool onlyIfNotParsed, bool updateView) { if (!mEnabled) return; { QMutexLocker locker(&mMutex); if (mParsing || mLockCount>0) return; updateSerialId(); mParsing = true; if (updateView) emit onBusy(); emit onStartParsing(); } { auto action = finally([&,this]{ mParsing = false; if (updateView) emit onEndParsing(mFilesScannedCount,1); else emit onEndParsing(mFilesScannedCount,0); }); QString fName = fileName; if (onlyIfNotParsed && mPreprocessor.scannedFiles().contains(fName)) return; if (inProject) { QSet filesToReparsed = calculateFilesToBeReparsed(fileName); QStringList files = sortFilesByIncludeRelations(filesToReparsed); internalInvalidateFiles(filesToReparsed); mFilesToScanCount = files.count(); mFilesScannedCount = 0; foreach (const QString& file,files) { mFilesScannedCount++; emit onProgress(file,mFilesToScanCount,mFilesScannedCount); if (!mPreprocessor.scannedFiles().contains(file)) { internalParse(file); } } } else { internalInvalidateFile(fileName); mFilesToScanCount = 1; mFilesScannedCount = 0; mFilesScannedCount++; emit onProgress(fileName,mFilesToScanCount,mFilesScannedCount); internalParse(fileName); } // if (inProject) // mProjectFiles.insert(fileName); // else { // mProjectFiles.remove(fileName); // } // Parse from disk or stream } } void CppParser::parseFileList(bool updateView) { if (!mEnabled) return; { QMutexLocker locker(&mMutex); if (mParsing || mLockCount>0) return; updateSerialId(); mParsing = true; if (updateView) emit onBusy(); emit onStartParsing(); } { auto action = finally([&,this]{ mParsing = false; if (updateView) emit onEndParsing(mFilesScannedCount,1); else emit onEndParsing(mFilesScannedCount,0); }); // Support stopping of parsing when files closes unexpectedly mFilesScannedCount = 0; mFilesToScanCount = mFilesToScan.count(); QStringList files = sortFilesByIncludeRelations(mFilesToScan); // parse header files in the first parse foreach (const QString& file, files) { mFilesScannedCount++; emit onProgress(mCurrentFile,mFilesToScanCount,mFilesScannedCount); if (!mPreprocessor.scannedFiles().contains(file)) { internalParse(file); } } mFilesToScan.clear(); } } void CppParser::parseHardDefines() { QMutexLocker locker(&mMutex); if (mParsing) return; int oldIsSystemHeader = mIsSystemHeader; mIsSystemHeader = true; mParsing=true; { auto action = finally([&,this]{ mParsing = false; mIsSystemHeader=oldIsSystemHeader; }); for (const PDefine& define:mPreprocessor.hardDefines()) { addStatement( PStatement(), // defines don't belong to any scope "", "", // define has no type define->name, define->args, "", define->value, -1, StatementKind::skPreprocessor, StatementScope::Global, StatementAccessibility::None, StatementProperty::spHasDefinition); } } } bool CppParser::parsing() const { return mParsing; } void CppParser::resetParser() { while (true) { { QMutexLocker locker(&mMutex); if (!mParsing && mLockCount ==0) { mParsing = true; break; } } QThread::msleep(50); QCoreApplication* app = QApplication::instance(); app->processEvents(); } { auto action = finally([this]{ mParsing = false; }); emit onBusy(); mUniqId = 0; mParseLocalHeaders = true; mParseGlobalHeaders = true; mIsSystemHeader = false; mIsHeader = false; mIsProjectFile = false; mFilesScannedCount=0; mFilesToScanCount = 0; mCurrentScope.clear(); mMemberAccessibilities.clear(); mStatementList.clear(); mProjectFiles.clear(); mBlockBeginSkips.clear(); //list of for/catch block begin token index; mBlockEndSkips.clear(); //list of for/catch block end token index; mInlineNamespaceEndSkips.clear(); // list for inline namespace end token index; mFilesToScan.clear(); // list of base files to scan mNamespaces.clear(); // namespace and the statements in its scope mInlineNamespaces.clear(); mPreprocessor.clear(); mTokenizer.clear(); } } void CppParser::unFreeze() { QMutexLocker locker(&mMutex); mLockCount--; } QSet CppParser::scannedFiles() { return mPreprocessor.scannedFiles(); } bool CppParser::isFileParsed(const QString &filename) { return mPreprocessor.scannedFiles().contains(filename); } QString CppParser::getScopePrefix(const PStatement& statement) const{ switch (statement->accessibility) { case StatementAccessibility::Public: return "public"; case StatementAccessibility::Private: return "private"; case StatementAccessibility::Protected: return "protected"; default: return ""; } } QString CppParser::prettyPrintStatement(const PStatement& statement, const QString& filename, int line) { QString result; switch(statement->kind) { case StatementKind::skPreprocessor: if (statement->command == "__FILE__") result = '"'+filename+'"'; else if (statement->command == "__LINE__") result = QString("\"%1\"").arg(line); else if (statement->command == "__DATE__") result = QString("\"%1\"").arg(QDate::currentDate().toString(Qt::ISODate)); else if (statement->command == "__TIME__") result = QString("\"%1\"").arg(QTime::currentTime().toString(Qt::ISODate)); else { QString hintText = "#define"; if (statement->command != "") hintText += ' ' + statement->command; if (statement->args != "") hintText += ' ' + statement->args; if (statement->value != "") hintText += ' ' + statement->value; result = hintText; } break; case StatementKind::skEnumClassType: result = "enum class "+statement->command; break; case StatementKind::skEnumType: result = "enum "+statement->command; break; case StatementKind::skEnum: result = statement->type + "::" + statement->command; break; case StatementKind::skTypedef: result = "typedef "+statement->type+" "+statement->command; if (!statement->args.isEmpty()) result += " "+statement->args; break; case StatementKind::skAlias: result = "using "+statement->type; break; case StatementKind::skFunction: case StatementKind::skVariable: case StatementKind::skParameter: case StatementKind::skClass: if (statement->scope!= StatementScope::Local) result = getScopePrefix(statement)+ ' '; // public result += statement->type + ' '; // void result += statement->fullName; // A::B::C::Bar result += statement->args; // (int a) break; case StatementKind::skNamespace: result = statement->fullName; // Bar break; case StatementKind::skConstructor: result = getScopePrefix(statement); // public result += QObject::tr("constructor") + ' '; // constructor result += statement->type + ' '; // void result += statement->fullName; // A::B::C::Bar result += statement->args; // (int a) break; case StatementKind::skDestructor: result = getScopePrefix(statement); // public result += QObject::tr("destructor") + ' '; // constructor result += statement->type + ' '; // void result += statement->fullName; // A::B::C::Bar result += statement->args; // (int a) break; default: break; } return result; } QString CppParser::getTemplateParam(const PStatement& statement, const QString& filename, const QString& phrase, int index, const PStatement& currentScope) const { if (!statement) return ""; if (statement->kind != StatementKind::skTypedef) return ""; if (statement->type == phrase) // prevent infinite loop return ""; return doFindTemplateParamOf(filename,statement->type,index,currentScope); } int CppParser::getTemplateParamStart(const QString &s, int startAt, int index) const { int i = startAt+1; int count=0; while (count': level--; if (level==0) return i; } i++; } return startAt; } void CppParser::addProjectFile(const QString &fileName, bool needScan) { QMutexLocker locker(&mMutex); //value.replace('/','\\'); // only accept full file names // Update project listing mProjectFiles.insert(fileName); // Only parse given file if (needScan && !mPreprocessor.scannedFiles().contains(fileName)) { mFilesToScan.insert(fileName); } } PStatement CppParser::addInheritedStatement(const PStatement& derived, const PStatement& inherit, StatementAccessibility access) { PStatement statement = addStatement( derived, inherit->fileName, inherit->type, // "Type" is already in use inherit->command, inherit->args, inherit->noNameArgs, inherit->value, inherit->line, inherit->kind, inherit->scope, access, inherit->properties & StatementProperty::spInherited); return statement; } PStatement CppParser::addChildStatement(const PStatement& parent, const QString &fileName, const QString &aType, const QString &command, const QString &args, const QString& noNameArgs, const QString &value, int line, StatementKind kind, const StatementScope& scope, const StatementAccessibility& classScope, StatementProperties properties) { return addStatement( parent, fileName, aType, command, args, noNameArgs, value, line, kind, scope, classScope, properties); } PStatement CppParser::addStatement(const PStatement& parent, const QString &fileName, const QString &aType, const QString &command, const QString &args, const QString &noNameArgs, const QString &value, int line, StatementKind kind, const StatementScope& scope, const StatementAccessibility& accessibility, StatementProperties properties) { // Move '*', '&' to type rather than cmd (it's in the way for code-completion) QString newType = aType; QString newCommand = command; while (!newCommand.isEmpty() && (newCommand.front() == '*' || newCommand.front() == '&')) { newType += newCommand.front(); newCommand.remove(0,1); // remove first } // if (newCommand.startsWith("::") && parent && kind!=StatementKind::skBlock ) { // qDebug()<fullName; // } if (kind == StatementKind::skConstructor || kind == StatementKind::skFunction || kind == StatementKind::skDestructor || kind == StatementKind::skVariable ) { //find if (properties.testFlag(StatementProperty::spHasDefinition)) { PStatement oldStatement = findStatementInScope(newCommand,noNameArgs,kind,parent); if (oldStatement && !oldStatement->hasDefinition()) { oldStatement->setHasDefinition(true); if (oldStatement->fileName!=fileName) { PFileIncludes fileIncludes=mPreprocessor.includesList().value(fileName); if (fileIncludes) { fileIncludes->statements.insert(oldStatement->fullName, oldStatement); } } oldStatement->definitionLine = line; oldStatement->definitionFileName = fileName; return oldStatement; } } } PStatement result = std::make_shared(); result->parentScope = parent; result->type = newType; if (!newCommand.isEmpty()) result->command = newCommand; else { mUniqId++; result->command = QString("__STATEMENT__%1").arg(mUniqId); } result->args = args; result->noNameArgs = noNameArgs; result->value = value; result->kind = kind; result->scope = scope; result->accessibility = accessibility; result->properties = properties; result->line = line; result->definitionLine = line; result->fileName = fileName; result->definitionFileName = fileName; if (!fileName.isEmpty()) { result->setInProject(mIsProjectFile); result->setInSystemHeader(mIsSystemHeader); } else { result->setInProject(false); result->setInSystemHeader(true); } //result->children; //result->friends; if (scope == StatementScope::Local) result->fullName = newCommand; else result->fullName = getFullStatementName(newCommand, parent); result->usageCount = -1; mStatementList.add(result); if (result->kind == StatementKind::skNamespace) { PStatementList namespaceList = mNamespaces.value(result->fullName,PStatementList()); if (!namespaceList) { namespaceList=std::make_shared(); mNamespaces.insert(result->fullName,namespaceList); } namespaceList->append(result); } if (result->kind!= StatementKind::skBlock) { PFileIncludes fileIncludes = mPreprocessor.includesList().value(fileName); if (fileIncludes) { fileIncludes->statements.insert(result->fullName,result); } } return result; } PStatement CppParser::addStatement(const PStatement &parent, const QString &fileName, const QString &aType, const QString &command, int argStart, int argEnd, const QString &value, int line, StatementKind kind, const StatementScope &scope, const StatementAccessibility &classScope, StatementProperties properties) { Q_ASSERT(mTokenizer[argStart]->text=='('); QString args; QString noNameArgs; int start=argStart+1; bool typeGetted = false; int braceLevel=0; QString word; for (int i=start;itext[0]; if (this->isIdentChar(ch)) { QString spaces=(i>argStart)?" ":""; args+=spaces; word += mTokenizer[i]->text; if (!typeGetted) { noNameArgs+=spaces+word; if (mCppTypeKeywords.contains(word) || !isCppKeyword(word)) typeGetted = true; } else { if (isCppKeyword(word)) { noNameArgs+=spaces+word; } } word=""; } else if (this->isDigitChar(ch)) { args+=" "; } else { switch(ch.unicode()) { case ',': if (braceLevel==0) typeGetted=false; break; case '{': case '[': case '<': case '(': braceLevel++; break; case '}': case ']': case '>': case ')': braceLevel--; break; //todo: * and & processing case '*': case '&': if (braceLevel==0) word+=ch; break; } noNameArgs+= mTokenizer[i]->text; } args+=mTokenizer[i]->text; } args="("+args.trimmed()+")"; noNameArgs="("+noNameArgs.trimmed()+")"; return addStatement( parent, fileName, aType, command, args, noNameArgs, value, line, kind, scope, classScope, properties); } void CppParser::setInheritance(int index, const PStatement& classStatement, bool isStruct) { int tokenCount = mTokenizer.tokenCount(); // Clear it. Assume it is assigned StatementAccessibility lastInheritScopeType = StatementAccessibility::None; // Assemble a list of statements in text form we inherit from while (true) { StatementAccessibility inheritScopeType = getClassMemberAccessibility(mTokenizer[index]->text); QString currentText = mTokenizer[index]->text; if (currentText=='(') { //skip to matching ')' index=mTokenizer[index]->matchIndex; } else if (inheritScopeType == StatementAccessibility::None) { if (currentText !=',' && currentText!=':') { QString basename = currentText; //remove template staff if (basename.endsWith('>')) { int pBegin = basename.indexOf('<'); if (pBegin>=0) basename.truncate(pBegin); } // Find the corresponding PStatement PStatement statement = doFindStatementOf(mCurrentFile,basename, classStatement->parentScope.lock()); if (statement && statement->kind == StatementKind::skClass) { inheritClassStatement(classStatement,isStruct,statement,lastInheritScopeType); } } } index++; lastInheritScopeType = inheritScopeType; if (index >= tokenCount) break; if (mTokenizer[index]->text.front() == '{' || mTokenizer[index]->text.front() == ';') break; } } bool CppParser::isCurrentScope(const QString &command) const { PStatement statement = getCurrentScope(); if (!statement) return false; QString s = command; // remove template staff if (s.endsWith('>')) { int i= command.indexOf('<'); if (i>=0) { s.truncate(i); } } QString s2 = statement->command; if (s2.endsWith('>')) { int i= s2.indexOf('<'); if (i>=0) { s2.truncate(i); } } return (s2 == s); } void CppParser::addSoloScopeLevel(PStatement& statement, int line, bool shouldResetBlock) { // Add class list PStatement parentScope; if (shouldResetBlock && statement && (statement->kind == StatementKind::skBlock)) { parentScope = statement->parentScope.lock(); while (parentScope && (parentScope->kind == StatementKind::skBlock)) { parentScope = parentScope->parentScope.lock(); } if (!parentScope) statement.reset(); } if (mMemberAccessibilities.count()>0) { mMemberAccessibilities.back() = mCurrentMemberAccessibility; } mCurrentScope.append(statement); PFileIncludes fileIncludes = mPreprocessor.includesList().value(mCurrentFile); if (fileIncludes) { fileIncludes->scopes.addScope(line,statement); } // Set new scope if (!statement) mCurrentMemberAccessibility = StatementAccessibility::None; // {}, namespace or class that doesn't exist else if (statement->kind == StatementKind::skNamespace) mCurrentMemberAccessibility = StatementAccessibility::None; else if (statement->type == "class") mCurrentMemberAccessibility = StatementAccessibility::Private; // classes are private by default else mCurrentMemberAccessibility = StatementAccessibility::Public; // structs are public by default mMemberAccessibilities.push_back(mCurrentMemberAccessibility); #ifdef QT_DEBUG // if (mCurrentClassScope.count()<=2) // qDebug()<<"++add scope"<kind == StatementKind::skBlock) { if (currentScope->children.isEmpty()) { // remove no children block if (fileIncludes) { fileIncludes->scopes.removeLastScope(); } mStatementList.deleteStatement(currentScope); } else { fileIncludes->statements.insert(currentScope->fullName,currentScope); } } else if (currentScope->kind == StatementKind::skClass) { mIndex=indexOfNextSemicolon(mIndex); } } mCurrentScope.pop_back(); mMemberAccessibilities.pop_back(); // Set new scope currentScope = getCurrentScope(); // fileIncludes:=FindFileIncludes(fCurrentFile); if (fileIncludes && fileIncludes->scopes.lastScope()!=currentScope) { fileIncludes->scopes.addScope(line,currentScope); } if (!currentScope) { mCurrentMemberAccessibility = StatementAccessibility::None; } else { mCurrentMemberAccessibility = mMemberAccessibilities.back(); } } void CppParser::internalClear() { mCurrentScope.clear(); mMemberAccessibilities.clear(); mIndex = 0; mCurrentMemberAccessibility = StatementAccessibility::None; mBlockBeginSkips.clear(); mBlockEndSkips.clear(); mInlineNamespaceEndSkips.clear(); } QStringList CppParser::sortFilesByIncludeRelations(const QSet &files) { QStringList result; QSet saveScannedFiles; saveScannedFiles=mPreprocessor.scannedFiles(); //rebuild file include relations foreach(const QString& file, files) { if (mPreprocessor.scannedFiles().contains(file)) continue; //already removed in interalInvalidateFiles //mPreprocessor.removeScannedFile(file); //we only use local include relations mPreprocessor.setScanOptions(false, true); mPreprocessor.preprocess(file); mPreprocessor.clearTempResults(); } QSet fileSet=files; while (!fileSet.isEmpty()) { bool found=false; foreach (const QString& file,fileSet) { PFileIncludes fileIncludes = mPreprocessor.includesList().value(file); bool hasInclude=false; foreach(const QString& inc,fileIncludes->includeFiles.keys()) { if (fileSet.contains(inc)) { hasInclude=true; break; } } if (!hasInclude) { result.push_front(file); fileSet.remove(file); found=true; break; } } if (!found) { foreach (const QString& file,fileSet) { result.push_front(file); fileSet.remove(file); } } } QSet newScannedFiles = mPreprocessor.scannedFiles(); foreach(const QString& file, newScannedFiles) { if (!saveScannedFiles.contains(file)) mPreprocessor.removeScannedFile(file); } return result; } bool CppParser::checkForKeyword(KeywordType& keywordType) { keywordType = mCppKeywords.value(mTokenizer[mIndex]->text,KeywordType::NotKeyword); switch(keywordType) { case KeywordType::Catch: case KeywordType::For: case KeywordType::Public: case KeywordType::Private: case KeywordType::Struct: case KeywordType::Enum: case KeywordType::Inline: case KeywordType::Namespace: case KeywordType::Typedef: case KeywordType::Using: case KeywordType::Friend: case KeywordType::Protected: case KeywordType::None: case KeywordType::NotKeyword: case KeywordType::DeclType: case KeywordType::Operator: return false; default: return true; } } bool CppParser::checkForNamespace(KeywordType keywordType) { return (keywordType==KeywordType::Namespace &&(mIndex < mTokenizer.tokenCount()-1)) || ( keywordType==KeywordType::Inline && (mIndex+1 < mTokenizer.tokenCount()-1) &&mTokenizer[mIndex+1]->text == "namespace" ); } bool CppParser::checkForPreprocessor() { return (mTokenizer[mIndex]->text.startsWith('#')); } //bool CppParser::checkForLambda() //{ // return (mIndex+1text.startsWith('[') // && mTokenizer[mIndex+1]->text=='('); //} bool CppParser::checkForAccessibilitySpecifiers(KeywordType keywordType) { return (keywordType == KeywordType::Public || keywordType == KeywordType::Protected || keywordType == KeywordType::Private); } bool CppParser::checkForStructs(KeywordType keywordType) { int dis = 0; // int keyLen = calcKeyLenForStruct(word); // if (keyLen<0) // return false; // bool result = (word.length() == keyLen) || isSpaceChar(word[keyLen]) // || (word[keyLen] == '['); bool result; if (keywordType == KeywordType::Friend || keywordType == KeywordType::Public || keywordType == KeywordType::Private) { dis = 1; if (mIndex+dis>=mTokenizer.tokenCount()) { mIndex++; return false; } result = (mCppKeywords.value(mTokenizer[mIndex+dis]->text,KeywordType::None)==KeywordType::Struct); } else { result = (keywordType==KeywordType::Struct); } if (result) { int tokenCount = mTokenizer.tokenCount(); if (mIndex >= tokenCount - 2 - dis) return false; if (mTokenizer[mIndex + 2+dis]->text[0] != ';') { // not: class something; int i = mIndex+dis +1; // the check for ']' was added because of this example: // struct option long_options[] = { // {"debug", 1, 0, 'D'}, // {"info", 0, 0, 'i'}, // ... // }; while (i < tokenCount) { QChar ch = mTokenizer[i]->text.back(); if (ch=='{' || ch == ':') break; switch(ch.unicode()) { case ';': case '{': case '}': case ',': case '(': case ')': case '[': case ']': case '=': case '*': case '&': case '%': case '+': case '-': case '~': return false; } i++; } } } return result; } bool CppParser::checkForTypedefEnum() { //we assume that typedef is the current index, so we check the next //should call CheckForTypedef first!!! return (mIndex+1 < mTokenizer.tokenCount() ) && (mTokenizer[mIndex + 1]->text == "enum"); } bool CppParser::checkForTypedefStruct() { //we assume that typedef is the current index, so we check the next //should call CheckForTypedef first!!! if (mIndex+1 >= mTokenizer.tokenCount()) return false; return (mCppKeywords.value(mTokenizer[mIndex+1]->text,KeywordType::None)==KeywordType::Struct); } bool CppParser::checkForUsing(KeywordType keywordType) { return keywordType==KeywordType::Using && (mIndex < mTokenizer.tokenCount()-1); } void CppParser::checkAndHandleMethodOrVar(KeywordType keywordType) { int tokenCount = mTokenizer.tokenCount(); if (mIndex+2>=tokenCount) { mIndex+=2; // let's finish; return; } QString currentText=mTokenizer[mIndex]->text; if (keywordType==KeywordType::DeclType) { if (mTokenizer[mIndex+1]->text=='(') { currentText="auto"; mIndex=mTokenizer[mIndex+1]->matchIndex+1; } else { currentText=mTokenizer[mIndex+1]->text; mIndex+=2; } } else { if (keywordType == KeywordType::Operator) { handleOperatorOverloading("", mIndex, false); return; } mIndex++; } //next token must be */&/word/(/{ if (mTokenizer[mIndex]->text=='(') { int indexAfterParentheis=mTokenizer[mIndex]->matchIndex+1; if (indexAfterParentheis>=tokenCount) { //error mIndex=indexAfterParentheis; } else if (mTokenizer[indexAfterParentheis]->text=='(') { // operator overloading like (operator int) if (mTokenizer[mIndex+1]->text=="operator") { mIndex=indexAfterParentheis; handleMethod(StatementKind::skFunction,"", mergeArgs(mIndex+1,mTokenizer[mIndex]->matchIndex-1), indexAfterParentheis,false,false,true); } else { handleVar(currentText,false,false); } } else { if (currentText=="operator") { // operator overloading handleOperatorOverloading( "", mIndex, false); return; } //check for constructor like Foo::Foo() QString name; QString temp,temp2; QString parentName; if (splitLastMember(currentText,name,temp)) { //has '::' bool isDestructor=false; if (!splitLastMember(temp,parentName,temp2)) parentName=temp; if (name.startsWith('~')) name=name.mid(1); if (removeTemplateParams(name)==removeTemplateParams(parentName)) { handleMethod( (isDestructor?StatementKind::skDestructor:StatementKind::skConstructor), "", currentText, mIndex,false,false); return; } } // check for constructor like: // class Foo { // Foo(); // }; PStatement scope=getCurrentScope(); if (scope && scope->kind==StatementKind::skClass && removeTemplateParams(scope->command) == removeTemplateParams(currentText)) { handleMethod(StatementKind::skConstructor,"", currentText, mIndex,false,false); return; } // function call, skip it mIndex=moveToEndOfStatement(mIndex,true); } } else if (mTokenizer[mIndex]->text.startsWith('*') || mTokenizer[mIndex]->text.startsWith('&') || mTokenizer[mIndex]->text=="::" || tokenIsIdentifier(mTokenizer[mIndex]->text) ) { // it should be function/var bool isStatic = false; bool isFriend = false; bool isExtern = false; QString sType; // should contain type "int" QString sName; // should contain function name "foo::function" if (mTokenizer[mIndex]->text=="::") { mIndex--; } else { if (currentText=="::") { sName = currentText; } else { if (currentText == "static") isStatic = true; else if (currentText == "friend") isFriend = true; else if (currentText == "extern") isExtern = true; sType = currentText; } } // Gather data for the string parts while (mIndex+1 < tokenCount) { if (mTokenizer[mIndex]->text=="operator") { handleOperatorOverloading(sType, //sName, mIndex, isStatic); return; } else if (mTokenizer[mIndex + 1]->text == '(') { if (mIndex+2text == '*') { //foo(*blabla), it's a function pointer var handleVar(sType+" "+sName,isExtern,isStatic); return; } int indexAfter=mTokenizer[mIndex + 1]->matchIndex+1; if (indexAfter>=tokenCount) { //error mIndex=indexAfter; return; } //if it's like: foo(...)(...) if (mTokenizer[indexAfter]->text=='(') { //foo(...)(...), it's a function pointer var handleVar(sType+" "+sName,isExtern,isStatic); //Won't implement: ignore function decl like int (text)(int x) { }; return; } //it's not a function define if (mTokenizer[indexAfter]->text[0] == ',') { // var decl with init handleVar(sType+" "+sName,isExtern,isStatic); return; } if (mTokenizer[indexAfter]->text[0] == ';' && sType!="void") { //function can only be defined in global/namespaces/classes PStatement currentScope=getCurrentScope(); if (currentScope) { //in namespace, it might be function or object initilization if (currentScope->kind == StatementKind::skNamespace) { if (isNotFuncArgs(mIndex + 1)) { // var decl with init handleVar(sType+" "+sName,isExtern,isStatic); return; } } else if (currentScope->kind != StatementKind::skClass) { //not in class, it can't be a valid function definition // var decl with init handleVar(sType+" "+sName,isExtern,isStatic); return; } //variable can't be initialized in class definition, it must be a function } else if (isNotFuncArgs(mIndex + 1)){ // var decl with init handleVar(sType+" "+sName,isExtern,isStatic); return; } } if (!sName.isEmpty()) { if (sName.endsWith("::")) sName+=mTokenizer[mIndex]->text; else { sType += " "+sName; sName = mTokenizer[mIndex]->text; } } else sName = mTokenizer[mIndex]->text; mIndex++; handleMethod(StatementKind::skFunction,sType, sName,mIndex,isStatic,isFriend); return; } else if ( mTokenizer[mIndex + 1]->text == ',' ||mTokenizer[mIndex + 1]->text == ';' ||mTokenizer[mIndex + 1]->text == ':' ||mTokenizer[mIndex + 1]->text == '{' || mTokenizer[mIndex + 1]->text == '=') { handleVar(sType+" "+sName,isExtern,isStatic); return; } else if ( mTokenizer[mIndex + 1]->text == "::") { sName = sName + mTokenizer[mIndex]->text+ "::"; mIndex+=2; } else { QString s = mTokenizer[mIndex]->text; if (sName.endsWith("::")) { sName+=s; } else { if (!sName.isEmpty()) { sType = sType+" "+sName; sName = ""; } if (s == "static") isStatic = true; else if (s == "friend") isFriend = true; else if (s == "extern") isExtern = true; if (!s.isEmpty() && !(s=="extern")) sType = sType + ' '+ s; } mIndex++; } } } } PStatement CppParser::doFindTypeDefinitionOf(const QString &fileName, const QString &aType, const PStatement ¤tClass) const { if (aType.isEmpty()) return PStatement(); // Remove pointer stuff from type QString s = aType; // 'Type' is a keyword int position = s.length()-1; while ((position >= 0) && (s[position] == '*' || s[position] == ' ' || s[position] == '&')) position--; if (position != s.length()-1) s.truncate(position+1); // Strip template stuff position = s.indexOf('<'); if (position >= 0) { int endPos = getBracketEnd(s,position); s.remove(position,endPos-position+1); } // Use last word only (strip 'const', 'static', etc) position = s.lastIndexOf(' '); if (position >= 0) s = s.mid(position+1); PStatement scopeStatement = currentClass; PStatement statement = doFindStatementOf(fileName,s,currentClass); return getTypeDef(statement,fileName,aType); } QString CppParser::doFindFirstTemplateParamOf(const QString &fileName, const QString &phrase, const PStatement ¤tScope) const { return doFindTemplateParamOf(fileName,phrase,0,currentScope); } QString CppParser::doFindTemplateParamOf(const QString &fileName, const QString &phrase, int index, const PStatement ¤tScope) const { if (phrase.isEmpty()) return QString(); // Remove pointer stuff from type QString s = phrase; // 'Type' is a keyword int i = s.indexOf('<'); if (i>=0) { i=getTemplateParamStart(s,i,index); int t=getTemplateParamEnd(s,i); //qDebug()<= 0) && (s[position] == '*' || s[position] == ' ' || s[position] == '&')) position--; if (position != s.length()-1) s.truncate(position+1); PStatement scopeStatement = currentScope; PStatement statement = doFindStatementOf(fileName,s,currentScope); return getTemplateParam(statement,fileName, phrase,index, currentScope); } int CppParser::getCurrentBlockEndSkip() const { if (mBlockEndSkips.isEmpty()) return mTokenizer.tokenCount()+1; return mBlockEndSkips.back(); } int CppParser::getCurrentBlockBeginSkip() const { if (mBlockBeginSkips.isEmpty()) return mTokenizer.tokenCount()+1; return mBlockBeginSkips.back(); } int CppParser::getCurrentInlineNamespaceEndSkip() const { if (mInlineNamespaceEndSkips.isEmpty()) return mTokenizer.tokenCount()+1; return mInlineNamespaceEndSkips.back(); } PStatement CppParser::getCurrentScope() const { if (mCurrentScope.isEmpty()) { return PStatement(); } return mCurrentScope.back(); } void CppParser::getFullNamespace(const QString &phrase, QString &sNamespace, QString &member) const { sNamespace = ""; member = phrase; int strLen = phrase.length(); if (strLen==0) return; int lastI =-1; int i=0; while (i=strLen) { if (mNamespaces.contains(sNamespace)) { sNamespace = phrase; member = ""; return; } } if (lastI >= 0) { sNamespace = phrase.mid(0,lastI); member = phrase.mid(lastI+2); } else { sNamespace = ""; member = phrase; } } QString CppParser::getFullStatementName(const QString &command, const PStatement& parent) const { PStatement scopeStatement=parent; while (scopeStatement && !isNamedScope(scopeStatement->kind)) scopeStatement = scopeStatement->parentScope.lock(); if (scopeStatement) return scopeStatement->fullName + "::" + command; else return command; } PStatement CppParser::getIncompleteClass(const QString &command, const PStatement& parentScope) { QString s=command; //remove template parameter int p = s.indexOf('<'); if (p>=0) { s.truncate(p); } PStatement result = doFindStatementOf(mCurrentFile,s,parentScope); if (result && result->kind!=StatementKind::skClass) return PStatement(); return result; } StatementScope CppParser::getScope() { // Don't blindly trust levels. Namespaces and externs can have levels too PStatement currentScope = getCurrentScope(); // Invalid class or namespace/extern if (!currentScope || (currentScope->kind == StatementKind::skNamespace)) return StatementScope::Global; else if (currentScope->kind == StatementKind::skClass) return StatementScope::ClassLocal; else return StatementScope::Local; } QString CppParser::getStatementKey(const QString &sName, const QString &sType, const QString &sNoNameArgs) const { return sName + "--" + sType + "--" + sNoNameArgs; } PStatement CppParser::getTypeDef(const PStatement& statement, const QString& fileName, const QString& aType) const { if (!statement) { return PStatement(); } if (statement->kind == StatementKind::skClass || statement->kind == StatementKind::skEnumType || statement->kind == StatementKind::skEnumClassType) { return statement; } else if (statement->kind == StatementKind::skTypedef) { if (statement->type == aType) // prevent infinite loop return statement; PStatement result = doFindTypeDefinitionOf(fileName,statement->type, statement->parentScope.lock()); if (!result) // found end of typedef trail, return result return statement; return result; } else if (statement->kind == StatementKind::skAlias) { PStatement result = doFindAliasedStatement(statement); if (!result) // found end of typedef trail, return result return statement; return result; } else return PStatement(); } void CppParser::handleCatchBlock() { int tokenCount = mTokenizer.tokenCount(); int startLine= mTokenizer[mIndex]->line; mIndex++; // skip for/catch; if (!((mIndex < tokenCount) && (mTokenizer[mIndex]->text.startsWith('(')))) return; //skip params int i2=mTokenizer[mIndex]->matchIndex+1; if (i2>=tokenCount) return; if (mTokenizer[i2]->text.startsWith('{')) { mBlockBeginSkips.append(i2); int i = indexOfMatchingBrace(i2); mBlockEndSkips.append(i); } else { int i=indexOfNextSemicolon(i2); mBlockEndSkips.append(i); } // add a block PStatement block = addStatement( getCurrentScope(), mCurrentFile, "", "", "", "", "", startLine, StatementKind::skBlock, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); addSoloScopeLevel(block,startLine); scanMethodArgs(block,mIndex); mIndex=mTokenizer[mIndex]->matchIndex+1; } void CppParser::handleEnum(bool isTypedef) { int tokenCount = mTokenizer.tokenCount(); QString enumName = ""; bool isEnumClass = false; int startLine = mTokenizer[mIndex]->line; mIndex++; //skip 'enum' if (mIndex < tokenCount && mTokenizer[mIndex]->text == "class") { //enum class isEnumClass = true; mIndex++; //skip class } bool isAdhocVar=false; int endIndex=-1; if ((mIndex< tokenCount) && mTokenizer[mIndex]->text.startsWith('{')) { // enum {...} NAME // Skip to the closing brace int i = indexOfMatchingBrace(mIndex); // Have we found the name? if (i + 1 < tokenCount) { enumName = mTokenizer[i + 1]->text.trimmed(); if (!isIdentifierOrPointer(enumName)) { //not a valid enum, skip to j mIndex=indexOfNextSemicolon(i+1)+1; return; } if (!isTypedef) { //it's an ad-hoc enum var define; if (isEnumClass) { //Enum class can't add hoc, just skip to ; mIndex=indexOfNextSemicolon(i+1)+1; return; } enumName = "__enum__"+enumName+"__"; isAdhocVar=true; } } endIndex=i+1; } else if (mIndex+1< tokenCount && mTokenizer[mIndex+1]->text.startsWith('{')){ // enum NAME {...}; enumName = mTokenizer[mIndex]->text; } else { // enum NAME blahblah // it's an old c-style enum variable definition return; } // Add statement for enum name too PStatement enumStatement; if (isEnumClass) { enumStatement=addStatement( getCurrentScope(), mCurrentFile, "enum class", enumName, "", "", "", startLine, StatementKind::skEnumClassType, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } else { enumStatement=addStatement( getCurrentScope(), mCurrentFile, "enum", enumName, "", "", "", startLine, StatementKind::skEnumType, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } if (isAdhocVar) { //Ad-hoc var definition // Skip to the closing brace int i = indexOfMatchingBrace(mIndex)+1; QString typeSuffix=""; while (itext; if (isIdentifierOrPointer(name)) { QString suffix; QString args; parseCommandTypeAndArgs(name,suffix,args); if (!name.isEmpty()) { addStatement( getCurrentScope(), mCurrentFile, enumName+suffix, mTokenizer[i]->text, args, "", "", mTokenizer[i]->line, StatementKind::skVariable, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } } else if (name!=',') { break; } i++; } endIndex=indexOfNextSemicolon(i); } // Skip opening brace mIndex++; // Call every member "enum NAME ITEMNAME" QString lastType("enum"); if (!enumName.isEmpty()) lastType += ' ' + enumName; QString cmd; QString args; if (mTokenizer[mIndex]->text!='}') { while ((mIndex < tokenCount) && mTokenizer[mIndex]->text!='}') { if (mTokenizer[mIndex]->text=="=") { mIndex=indexOfNextPeriodOrSemicolon(mIndex); continue; } else if (tokenIsIdentifier(mTokenizer[mIndex]->text)) { cmd = mTokenizer[mIndex]->text; args = ""; if (isEnumClass) { if (enumStatement) { addStatement( enumStatement, mCurrentFile, lastType, cmd, args, "", "", mTokenizer[mIndex]->line, StatementKind::skEnum, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } } else { if (enumStatement) { addStatement( enumStatement, mCurrentFile, lastType, cmd, args, "", "", mTokenizer[mIndex]->line, StatementKind::skEnum, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } addStatement( getCurrentScope(), mCurrentFile, lastType, cmd, "", "", "", mTokenizer[mIndex]->line, StatementKind::skEnum, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } } mIndex ++ ; } } if (mIndexline; mIndex++; // skip for/catch; if (mIndex >= tokenCount) return; int i=indexOfNextSemicolon(mIndex); int i2 = i+1; //skip over ';' (tokenizer have change for(;;) to for(;) if (i2>=tokenCount) return; if (mTokenizer[i2]->text.startsWith('{')) { mBlockBeginSkips.append(i2); i=indexOfMatchingBrace(i2); // tokenizer will handle unbalanced braces, no need check here // if (i==i2) // mBlockEndSkips.append(mTokenizer.tokenCount()); // else mBlockEndSkips.append(i); } else { i=indexOfNextSemicolon(i2); mBlockEndSkips.append(i); } // add a block PStatement block = addStatement( getCurrentScope(), mCurrentFile, "", "", "", "", "", startLine, StatementKind::skBlock, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); addSoloScopeLevel(block,startLine); } void CppParser::handleKeyword(KeywordType skipType) { // Skip switch (skipType) { case KeywordType::SkipItself: // skip it; mIndex++; break; case KeywordType::SkipNextSemicolon: // Skip to ; and over it skipNextSemicolon(mIndex); break; case KeywordType::SkipNextColon: // Skip to : and over it mIndex = indexOfNextColon(mIndex)+1; break; case KeywordType::SkipNextParenthesis: // skip pass () skipParenthesis(mIndex); break; case KeywordType::MoveToLeftBrace: // Skip to { mIndex = indexOfNextLeftBrace(mIndex); break; case KeywordType::MoveToRightBrace: // Skip pass {} mIndex = indexPassBraces(mIndex); break; default: break; } } void CppParser::handleLambda(int index, int endIndex) { Q_ASSERT(mTokenizer[index]->text.startsWith('[')); int startLine=mTokenizer[index]->line; int argStart=index+1; if (mTokenizer[argStart]->text!='(') return; int argEnd= mTokenizer[argStart]->matchIndex; //TODO: parse captures int bodyStart=indexOfNextLeftBrace(argEnd+1); if (bodyStart>=endIndex) { return; } int bodyEnd = mTokenizer[bodyStart]->matchIndex; if (bodyEnd>=endIndex) { return; } PStatement lambdaBlock = addStatement( getCurrentScope(), mCurrentFile, "", "", "", "", "", startLine, StatementKind::skBlock, StatementScope::Local, StatementAccessibility::None, StatementProperty::spHasDefinition); scanMethodArgs(lambdaBlock,argStart); addSoloScopeLevel(lambdaBlock,mTokenizer[bodyStart]->line); int i=bodyStart+1; // start after '{'; while (i+2text) && !mTokenizer[i]->text.endsWith('.') && !mTokenizer[i]->text.endsWith("->") && (mTokenizer[i+1]->text.startsWith('*') || mTokenizer[i+1]->text.startsWith('&') || tokenIsTypeOrNonKeyword(mTokenizer[i+1]->text))) { QString sType; QString sName; while (i+1text==':' || mTokenizer[i+1]->text=='(' || mTokenizer[i+1]->text=='=' || mTokenizer[i+1]->text==';' || mTokenizer[i+1]->text==',' || mTokenizer[i+1]->text=='{' ) break; else { if (!sType.isEmpty()) sType+=' '; sType+=mTokenizer[i]->text; } i++; } QString tempType; while (itext.front() == ':') { while ( (i < bodyEnd) && !( mTokenizer[i]->text==',' || mTokenizer[i]->text==';' || mTokenizer[i]->text=='=' )) i++; } else if (mTokenizer[i]->text==';') { break; } else if (isIdentChar(mTokenizer[i]->text[0])) { QString cmd=mTokenizer[i]->text; if (cmd=="const") { tempType="const"; } else { QString suffix; QString args; parseCommandTypeAndArgs(cmd,suffix,args); if (!cmd.isEmpty()) { addChildStatement( lambdaBlock, mCurrentFile, (sType+' '+tempType+suffix).trimmed(), cmd, args, "", "", mTokenizer[mIndex]->line, StatementKind::skVariable, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); // TODO: not supported to pass list tempType=""; } } i++; } else if (mTokenizer[i]->text=='(') { i=mTokenizer[i]->matchIndex+1; } else if (mTokenizer[i]->text.endsWith('=')) { i = skipAssignment(i, bodyEnd); } else if (mTokenizer[i]->text=='{') { tempType=""; i=mTokenizer[i]->matchIndex+1; } else { tempType=""; i++; } } } i=moveToEndOfStatement(i, true, bodyEnd); } removeScopeLevel(mTokenizer[bodyEnd]->line); } void CppParser::handleOperatorOverloading(const QString &sType, //const QString &prefix, int operatorTokenIndex, bool isStatic) { //operatorTokenIndex is the token index of "operator" int tokenCount = mTokenizer.tokenCount(); int index=operatorTokenIndex+1; QString op=""; if (index>=tokenCount) { mIndex=index; return; } if (mTokenizer[index]->text=="(") { op="()"; index=mTokenizer[index]->matchIndex+1; } else if (mTokenizer[index]->text=="new" || mTokenizer[index]->text=="delete") { op=mTokenizer[index]->text; index++; if (indextext=="[]") { op+="[]"; index++; } } else { op=mTokenizer[index]->text; index++; while (indextext != "(") index++; } while (indextext == ")") index++; if (index>=tokenCount || mTokenizer[index]->text!="(") { mIndex=index; return; } Q_ASSERT(!op.isEmpty()); if (isIdentChar(op.front())) { handleMethod(StatementKind::skFunction, sType+" "+op, "operator("+op+")", index, isStatic, false, true); } else { handleMethod(StatementKind::skFunction, sType, "operator"+op, index, isStatic, false, true); } } void CppParser::handleMethod(StatementKind functionKind,const QString &sType, const QString &sName, int argStart, bool isStatic, bool isFriend,bool isOperatorOverload) { int tokenCount = mTokenizer.tokenCount(); bool isValid = true; bool isDeclaration = false; // assume it's not a prototype int startLine = mTokenizer[mIndex]->line; int argEnd = mTokenizer[argStart]->matchIndex; if (mIndex >= tokenCount) // not finished define, just skip it; return; PStatement scopeStatement = getCurrentScope(); //find start of the function body; bool foundColon=false; mIndex=argEnd+1; while ((mIndex < tokenCount) && !isblockChar(mTokenizer[mIndex]->text.front())) { if (mTokenizer[mIndex]->text=='(') { mIndex=mTokenizer[mIndex]->matchIndex+1; }else if (mTokenizer[mIndex]->text==':') { foundColon=true; break; } else mIndex++; } if (foundColon) { mIndex++; while ((mIndex < tokenCount) && !isblockChar(mTokenizer[mIndex]->text.front())) { if (isWordChar(mTokenizer[mIndex]->text[0]) && mIndex+1text=='{') { //skip parent {}intializer mIndex=mTokenizer[mIndex+1]->matchIndex+1; } else if (mTokenizer[mIndex]->text=='(') { mIndex=mTokenizer[mIndex]->matchIndex+1; } else mIndex++; } } if (mIndex>=tokenCount) return; // Check if this is a prototype if (mTokenizer[mIndex]->text.startsWith(';') || mTokenizer[mIndex]->text.startsWith('}')) {// prototype isDeclaration = true; } QString scopelessName; PStatement functionStatement; if (isFriend && isDeclaration && scopeStatement) { int delimPos = sName.indexOf("::"); if (delimPos >= 0) { scopelessName = sName.mid(delimPos+2); } else scopelessName = sName; //TODO : we should check namespace scopeStatement->friends.insert(scopelessName); } else if (isValid) { // Use the class the function belongs to as the parent ID if the function is declared outside of the class body QString scopelessName; QString parentClassName; if (!isOperatorOverload && splitLastMember(sName,scopelessName,parentClassName)) { // Provide Bar instead of Foo::Bar scopeStatement = getIncompleteClass(parentClassName,getCurrentScope()); //parent not found if (!parentClassName.isEmpty() && !scopeStatement) scopelessName=sName; } else scopelessName = sName; // qDebug()<fullName; // For function definitions, the parent class is given. Only use that as a parent if (!isDeclaration) { functionStatement=addStatement( scopeStatement, mCurrentFile, sType, scopelessName, argStart, argEnd, "", //mTokenizer[mIndex - 1]^.Line, startLine, functionKind, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition | (isStatic?StatementProperty::spStatic:StatementProperty::spNone) | (isOperatorOverload?StatementProperty::spOperatorOverloading:StatementProperty::spNone)); scanMethodArgs(functionStatement, argStart); // add variable this to the class function if (scopeStatement && scopeStatement->kind == StatementKind::skClass && !isStatic) { //add this to non-static class member function addStatement( functionStatement, mCurrentFile, scopeStatement->command+"*", "this", "", "", "", startLine, StatementKind::skVariable, StatementScope::Local, StatementAccessibility::None, StatementProperty::spHasDefinition | (isOperatorOverload?StatementProperty::spOperatorOverloading:StatementProperty::spNone)); } // add "__func__ variable" addStatement( functionStatement, mCurrentFile, "static const char ", "__func__", "[]", "", "\""+scopelessName+"\"", startLine+1, StatementKind::skVariable, StatementScope::Local, StatementAccessibility::None, StatementProperty::spHasDefinition | (isOperatorOverload?StatementProperty::spOperatorOverloading:StatementProperty::spNone)); } else { functionStatement = addStatement( scopeStatement, mCurrentFile, sType, scopelessName, argStart, argEnd, "", //mTokenizer[mIndex - 1]^.Line, startLine, functionKind, getScope(), mCurrentMemberAccessibility, (isStatic?StatementProperty::spStatic:StatementProperty::spNone) | (isOperatorOverload?StatementProperty::spOperatorOverloading:StatementProperty::spNone)); } } if ((mIndex < tokenCount) && mTokenizer[mIndex]->text.startsWith('{')) { addSoloScopeLevel(functionStatement,startLine); mIndex++; //skip '{' } else if ((mIndex < tokenCount) && mTokenizer[mIndex]->text.startsWith(';')) { addSoloScopeLevel(functionStatement,startLine); if (mTokenizer[mIndex]->line != startLine) removeScopeLevel(mTokenizer[mIndex]->line+1); else removeScopeLevel(startLine+1); mIndex++; } } void CppParser::handleNamespace(KeywordType skipType) { int tokenCount = mTokenizer.tokenCount(); bool isInline=false; int startLine = mTokenizer[mIndex]->line; if (skipType==KeywordType::Inline) { isInline = true; mIndex++; //skip 'inline' } mIndex++; //skip 'namespace' // if (!tokenIsIdentifier(mTokenizer[mIndex]->text)) // //wrong namespace define, stop handling // return; QString command = mTokenizer[mIndex]->text; QString fullName = getFullStatementName(command,getCurrentScope()); if (isInline) { mInlineNamespaces.insert(fullName); } else if (mInlineNamespaces.contains(fullName)) { isInline = true; } // if (command.startsWith("__")) // hack for inline namespaces // isInline = true; mIndex++; if (mIndex>=tokenCount) return; QString aliasName; if ((mIndex+2text == '=')) { aliasName=mTokenizer[mIndex+1]->text; //namespace alias addStatement( getCurrentScope(), mCurrentFile, aliasName, // name of the alias namespace command, // command "", // args "", // noname args "", // values //mTokenizer[mIndex]^.Line, startLine, StatementKind::skNamespaceAlias, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); mIndex+=2; //skip ; return; } else if (isInline) { //inline namespace , just skip it // Skip to '{' while ((mIndextext != '{')) mIndex++; int i =indexOfMatchingBrace(mIndex); //skip '}' if (i==mIndex) mInlineNamespaceEndSkips.append(tokenCount); else mInlineNamespaceEndSkips.append(i); if (mIndextext=='{') addSoloScopeLevel(namespaceStatement,startLine); //skip it mIndex++; } } void CppParser::handleOtherTypedefs() { int tokenCount = mTokenizer.tokenCount(); int startLine = mTokenizer[mIndex]->line; // Skip typedef word mIndex++; if (mIndex>=tokenCount) return; if (mTokenizer[mIndex]->text == '(' || mTokenizer[mIndex]->text == ',' || mTokenizer[mIndex]->text == ';') { // error typedef //skip over next ; mIndex=indexOfNextSemicolon(mIndex)+1; return; } if ((mIndex+1text == ';')) { //no old type, not valid mIndex+=2; //skip ; return; } QString oldType; // Walk up to first new word (before first comma or ;) while(true) { if (oldType.endsWith("::")) oldType += mTokenizer[mIndex]->text; else if (mTokenizer[mIndex]->text=="::") oldType += "::"; else oldType += ' ' + mTokenizer[mIndex]->text; mIndex++; if (mIndex+1>=tokenCount) { //not valid, just exit return; } if (mTokenizer[mIndex]->text=='(') { break; } if (mTokenizer[mIndex + 1]->text.front() == ',' || mTokenizer[mIndex + 1]->text == ';') break; //typedef function pointer } oldType = oldType.trimmed(); if (oldType.isEmpty()) { //skip over next ; mIndex=indexOfNextSemicolon(mIndex)+1; return; } QString newType; while(mIndex+1text == ',' ) { mIndex++; } else if (mTokenizer[mIndex]->text == ';' ) { break; } else if (mTokenizer[mIndex]->text == '(') { int paramStart=mTokenizer[mIndex]->matchIndex+1; if (paramStart>=tokenCount || mTokenizer[paramStart]->text!='(') { //not valid function pointer (no args) //skip over next ; mIndex=indexOfNextSemicolon(paramStart)+1; return; } QString newType = findFunctionPointerName(mIndex); if (!newType.isEmpty()) { addStatement( getCurrentScope(), mCurrentFile, oldType, newType, mergeArgs(paramStart,mTokenizer[paramStart]->matchIndex), "", "", startLine, StatementKind::skTypedef, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } mIndex = mTokenizer[paramStart]->matchIndex+1; } else if (mTokenizer[mIndex+1]->text.front() ==',' || mTokenizer[mIndex+1]->text.front() ==';') { newType += mTokenizer[mIndex]->text; QString suffix; QString args; parseCommandTypeAndArgs(newType,suffix,args); addStatement( getCurrentScope(), mCurrentFile, oldType+suffix, newType, args, "", "", startLine, StatementKind::skTypedef, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); newType = ""; mIndex++; } else { newType += mTokenizer[mIndex]->text; mIndex++; } } // Step over semicolon (saves one HandleStatement loop) mIndex++; } void CppParser::handlePreprocessor() { QString text = mTokenizer[mIndex]->text.mid(1).trimmed(); if (text.startsWith("include")) { // start of new file // format: #include fullfilename:line // Strip keyword QString s = text.mid(QString("include").length()); if (!s.startsWith(" ") && !s.startsWith("\t")) goto handlePreprocessorEnd; int delimPos = s.lastIndexOf(':'); if (delimPos>=0) { // qDebug()<line<line, StatementKind::skPreprocessor, StatementScope::Global, StatementAccessibility::None, StatementProperty::spHasDefinition); } // TODO: undef ( define has limited scope) handlePreprocessorEnd: mIndex++; } StatementAccessibility CppParser::getClassMemberAccessibility(const QString& text) const { KeywordType type = mCppKeywords.value(text,KeywordType::None); return getClassMemberAccessibility(type); } StatementAccessibility CppParser::getClassMemberAccessibility(KeywordType keywordType) const { switch(keywordType) { case KeywordType::Public: return StatementAccessibility::Public; case KeywordType::Private: return StatementAccessibility::Private; case KeywordType::Protected: return StatementAccessibility::Protected; default: return StatementAccessibility::None; } } void CppParser::handleAccessibilitySpecifiers(KeywordType keywordType) { mCurrentMemberAccessibility = getClassMemberAccessibility(keywordType); mIndex++; if (mIndex < mTokenizer.tokenCount() && mTokenizer[mIndex]->text == ':') mIndex++; // skip ':' } bool CppParser::handleStatement() { QString funcType,funcName; int idx=getCurrentBlockEndSkip(); int idx2=getCurrentBlockBeginSkip(); int idx3=getCurrentInlineNamespaceEndSkip(); KeywordType keywordType; #ifdef QT_DEBUG // qDebug()<=mLastIndex); mLastIndex=mIndex; #endif int tokenCount = mTokenizer.tokenCount(); if (mIndex >= idx2) { //skip (previous handled) block begin mBlockBeginSkips.pop_back(); if (mIndex == idx2) mIndex++; else if (mIndexline); } else if (mIndex >= idx) { //skip (previous handled) block end mBlockEndSkips.pop_back(); if (idx+1 < tokenCount) removeScopeLevel(mTokenizer[idx+1]->line); if (mIndex == idx) mIndex++; } else if (mIndex >= idx3) { //skip (previous handled) inline name space end mInlineNamespaceEndSkips.pop_back(); if (mIndex == idx3) mIndex++; } else if (mTokenizer[mIndex]->text.startsWith('{')) { PStatement block = addStatement( getCurrentScope(), mCurrentFile, "", "", "", "", "", //mTokenizer[mIndex]^.Line, mTokenizer[mIndex]->line, StatementKind::skBlock, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); addSoloScopeLevel(block,mTokenizer[mIndex]->line,true); mIndex++; } else if (mTokenizer[mIndex]->text[0] == '}') { removeScopeLevel(mTokenizer[mIndex]->line); mIndex++; } else if (checkForPreprocessor()) { handlePreprocessor(); // } else if (checkForLambda()) { // is lambda // handleLambda(); } else if (mTokenizer[mIndex]->text=='(') { if (mIndex+1text=="operator") { // things like (operator int) mIndex++; //just skip '(' } else skipParenthesis(mIndex); } else if (mTokenizer[mIndex]->text==')') { mIndex++; } else if (mTokenizer[mIndex]->text.startsWith('~')) { //it should be a destructor if (mIndex+1text=='(') { //dont further check to speed up handleMethod(StatementKind::skDestructor, "", '~'+mTokenizer[mIndex]->text, mIndex+1, false, false); } else { //error mIndex=moveToEndOfStatement(mIndex,false); } } else if (mTokenizer[mIndex]->text=="::") { checkAndHandleMethodOrVar(KeywordType::None); } else if (!isIdentChar(mTokenizer[mIndex]->text[0])) { mIndex=moveToEndOfStatement(mIndex,true); } else if (checkForKeyword(keywordType)) { // includes template now handleKeyword(keywordType); } else if (keywordType==KeywordType::For) { // (for/catch) handleForBlock(); } else if (keywordType==KeywordType::Catch) { // (for/catch) handleCatchBlock(); } else if (checkForAccessibilitySpecifiers(keywordType)) { // public /private/proteced handleAccessibilitySpecifiers(keywordType); } else if (keywordType==KeywordType::Enum) { handleEnum(false); } else if (keywordType==KeywordType::Typedef) { if (mIndex+1 < tokenCount) { if (checkForTypedefStruct()) { // typedef struct something mIndex++; // skip 'typedef' handleStructs(true); } else if (checkForTypedefEnum()) { // typedef enum something mIndex++; // skip 'typedef' handleEnum(true); } else handleOtherTypedefs(); // typedef Foo Bar } else mIndex++; } else if (checkForNamespace(keywordType)) { handleNamespace(keywordType); } else if (checkForUsing(keywordType)) { handleUsing(); } else if (checkForStructs(keywordType)) { handleStructs(false); } else if (keywordType == KeywordType::Inline) { mIndex++; }else { // it should be method/constructor/var checkAndHandleMethodOrVar(keywordType); } //Q_ASSERT(mIndex<999999); // while (mTokenizer.lambdasCount()>0 && mTokenizer.indexOfFirstLambda()text; if (prefix == "friend") { isFriend = true; mIndex++; } // Check if were dealing with a struct or union prefix = mTokenizer[mIndex]->text; bool isStruct = ("class" != prefix); //struct/union int startLine = mTokenizer[mIndex]->line; int tokenCount = mTokenizer.tokenCount(); mIndex++; //skip struct/class/union if (mIndex>=tokenCount) return; // Do not modifiy index int i=indexOfNextSemicolonOrLeftBrace(mIndex); if (i >= tokenCount) { //error mIndex=i; return; } // Forward class/struct decl *or* typedef, e.g. typedef struct some_struct synonym1, synonym2; if (mTokenizer[i]->text == ";") { // typdef struct Foo Bar if (isTypedef) { QString oldType = mTokenizer[mIndex]->text; while(true) { // Add definition statement for the synonym if ((mIndex + 1 < tokenCount) && (mTokenizer[mIndex + 1]->text=="," || mTokenizer[mIndex + 1]->text==";")) { QString newType = mTokenizer[mIndex]->text; addStatement( getCurrentScope(), mCurrentFile, oldType, newType, "", // args "", // noname args "", // values mTokenizer[mIndex]->line, StatementKind::skTypedef, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } mIndex++; if (mIndex >= tokenCount) break; if (mTokenizer[mIndex]->text.front() == ';') break; } } else { if (isFriend) { // friend class PStatement parentStatement = getCurrentScope(); if (parentStatement) { parentStatement->friends.insert(mTokenizer[mIndex]->text); } } else { // todo: Forward declaration, struct Foo. Don't mention in class browser } i++; // step over ; mIndex = i; } // normal class/struct decl } else { PStatement firstSynonym; // Add class/struct name BEFORE opening brace if (mTokenizer[mIndex]->text != "{") { while(mIndex < tokenCount) { if (mTokenizer[mIndex]->text == ":" || mTokenizer[mIndex]->text == "{" || mTokenizer[mIndex]->text == ";") { break; } else if ((mIndex + 1 < tokenCount) && (mTokenizer[mIndex + 1]->text == "," || mTokenizer[mIndex + 1]->text == ";" || mTokenizer[mIndex + 1]->text == "{" || mTokenizer[mIndex + 1]->text == ":")) { QString command = mTokenizer[mIndex]->text; PStatement scopeStatement=getCurrentScope(); QString scopelessName; QString parentName; if (splitLastMember(command,scopelessName,parentName)) { scopeStatement = getIncompleteClass(parentName,getCurrentScope()); } else { scopelessName=command; } if (!command.isEmpty()) { firstSynonym = addStatement( scopeStatement, mCurrentFile, prefix, // type scopelessName, // command "", // args "", // no name args, "", // values mTokenizer[mIndex]->line, //startLine, StatementKind::skClass, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); command = ""; } mIndex++; break; } else if ((mIndex + 2 < tokenCount) && (mTokenizer[mIndex + 1]->text == "final") && (mTokenizer[mIndex + 2]->text=="," || mTokenizer[mIndex + 2]->text==":" || isblockChar(mTokenizer[mIndex + 2]->text.front()))) { QString command = mTokenizer[mIndex]->text; if (!command.isEmpty()) { firstSynonym = addStatement( getCurrentScope(), mCurrentFile, prefix, // type command, // command "", // args "", // no name args "", // values mTokenizer[mIndex]->line, //startLine, StatementKind::skClass, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); command=""; } mIndex+=2; break; } else mIndex++; } } // Walk to opening brace if we encountered inheritance statements if ((mIndex < tokenCount) && (mTokenizer[mIndex]->text == ":")) { if (firstSynonym) setInheritance(mIndex, firstSynonym, isStruct); // set the _InheritanceList value mIndex=indexOfNextLeftBrace(mIndex); } // Check for struct synonyms after close brace if (isStruct) { // Walk to closing brace i = indexOfMatchingBrace(mIndex); // step onto closing brace if ((i + 1 < tokenCount) && !( mTokenizer[i + 1]->text.front() == ';' || mTokenizer[i + 1]->text.front() == '}')) { // When encountering names again after struct body scanning, skip it QString command = ""; QString args = ""; // Add synonym before opening brace while(true) { i++; if (mTokenizer[i]->text=='(' || mTokenizer[i]->text==')') { //skip } else if (!(mTokenizer[i]->text == '{' || mTokenizer[i]->text == ',' || mTokenizer[i]->text == ';')) { if (mTokenizer[i]->text.endsWith(']')) { // cut-off array brackets int pos = mTokenizer[i]->text.indexOf('['); command += mTokenizer[i]->text.mid(0,pos) + ' '; args = mTokenizer[i]->text.mid(pos); } else if (mTokenizer[i]->text.front() == '*' || mTokenizer[i]->text.front() == '&') { // do not add spaces after pointer operator command += mTokenizer[i]->text; } else { command += mTokenizer[i]->text + ' '; } } else { command = command.trimmed(); if (!command.isEmpty() && ( !firstSynonym || command!=firstSynonym->command )) { //not define the struct yet, we define a unamed struct if (!firstSynonym) { firstSynonym = addStatement( getCurrentScope(), mCurrentFile, prefix, "__"+command, "", "", "", mTokenizer[i]->line, //startLine, StatementKind::skClass, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } if (isTypedef) { //typedef addStatement( getCurrentScope(), mCurrentFile, firstSynonym->command, command, "", "", "", mTokenizer[mIndex]->line, StatementKind::skTypedef, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); // typedef } else { //variable define addStatement( getCurrentScope(), mCurrentFile, firstSynonym->command, command, args, "", "", mTokenizer[i]->line, StatementKind::skVariable, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); // TODO: not supported to pass list } } command = ""; } if (i >= tokenCount - 1) break; if (mTokenizer[i]->text=='{' || mTokenizer[i]->text== ';') break; } // Nothing worth mentioning after closing brace // Proceed to set first synonym as current class } } if (!firstSynonym) { PStatement scope = getCurrentScope(); if (scope && scope->kind == StatementKind::skClass && mIndextext=="{") { //C11 anonymous union/struct addSoloScopeLevel(scope, mTokenizer[mIndex]->line); //skip { mIndex++; return; } else { //anonymous union/struct/class, add as a block firstSynonym=addStatement( getCurrentScope(), mCurrentFile, "", "", "", "", "", mTokenizer[mIndex]->line, StatementKind::skBlock, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } } if (mIndex < tokenCount) addSoloScopeLevel(firstSynonym,mTokenizer[mIndex]->line); else addSoloScopeLevel(firstSynonym,startLine); // Step over { if ((mIndex < tokenCount) && (mTokenizer[mIndex]->text == "{")) mIndex++; } } void CppParser::handleUsing() { int startLine = mTokenizer[mIndex]->line; if (mCurrentFile.isEmpty()) { //skip pass next ; mIndex=indexOfNextSemicolon(mIndex)+1; return; } int tokenCount = mTokenizer.tokenCount(); mIndex++; //skip 'using' //handle things like 'using vec = std::vector; ' if (mIndex+1 < tokenCount && mTokenizer[mIndex+1]->text == "=") { QString fullName = mTokenizer[mIndex]->text; QString aliasName; mIndex+=2; while (mIndextext!=';') { aliasName += mTokenizer[mIndex]->text; mIndex++; } addStatement( getCurrentScope(), mCurrentFile, aliasName, // name of the alias (type) fullName, // command "", // args "", // noname args "", // values startLine, StatementKind::skTypedef, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); // skip ; mIndex++; return; } //handle things like 'using std::vector;' if ((mIndex+2>=tokenCount) || (mTokenizer[mIndex]->text != "namespace")) { QString fullName; QString usingName; while (mIndextext!=';') { fullName += mTokenizer[mIndex]->text; usingName = mTokenizer[mIndex]->text; mIndex++; } if (fullName!=usingName) { addStatement( getCurrentScope(), mCurrentFile, fullName, // name of the alias (type) usingName, // command "", // args "", // noname args "", // values startLine, StatementKind::skAlias, getScope(), mCurrentMemberAccessibility, StatementProperty::spHasDefinition); } //skip ; mIndex++; return; } mIndex++; // skip 'namespace' PStatement scopeStatement = getCurrentScope(); QString usingName; while (mIndextext!=';') { usingName += mTokenizer[mIndex]->text; mIndex++; } if (scopeStatement) { QString fullName = scopeStatement->fullName + "::" + usingName; if (!mNamespaces.contains(fullName)) { fullName = usingName; } if (mNamespaces.contains(fullName)) { scopeStatement->usingList.insert(fullName); } } else { PFileIncludes fileInfo = mPreprocessor.includesList().value(mCurrentFile); if (!fileInfo) return; if (mNamespaces.contains(usingName)) { fileInfo->usings.insert(usingName); } } //skip ; mIndex++; } void CppParser::handleVar(const QString& typePrefix,bool isExtern,bool isStatic) { QString lastType; if (typePrefix=="extern") { isExtern=true; } else if (typePrefix=="static") { isStatic=true; } else { lastType=typePrefix.trimmed(); } PStatement addedVar; QString tempType; int tokenCount = mTokenizer.tokenCount(); while(mIndextext[0].unicode()) { case ':': if (mTokenizer[mIndex]->text.length()>1) { //handle '::' tempType+=mTokenizer[mIndex]->text; mIndex++; } else { // Skip bit identifiers, // e.g.: // handle // unsigned short bAppReturnCode:8,reserved:6,fBusy:1,fAck:1 // as // unsigned short bAppReturnCode,reserved,fBusy,fAck if (mIndex+1text) && isIdentChar(mTokenizer[mIndex+1]->text.back()) && addedVar && !(addedVar->properties & StatementProperty::spFunctionPointer) && AutoTypes.contains(addedVar->type)) { //handle e.g.: for(auto x:vec) QStringList phraseExpression; phraseExpression.append(mTokenizer[mIndex+1]->text); int pos = 0; PEvalStatement aliasStatement = doEvalExpression(mCurrentFile, phraseExpression, pos, getCurrentScope(), PEvalStatement(), true); if(aliasStatement && aliasStatement->effectiveTypeStatement) { if (STLMaps.contains(aliasStatement->effectiveTypeStatement->fullName)) { addedVar->type = "std::pair"+aliasStatement->templateParams; } else if (STLContainers.contains(aliasStatement->effectiveTypeStatement->fullName)){ QString type=doFindFirstTemplateParamOf(mCurrentFile,aliasStatement->templateParams, getCurrentScope()); if (!type.isEmpty()) addedVar->type = type; } } } addedVar.reset(); bool should_exit=false; while (mIndex < tokenCount) { switch(mTokenizer[mIndex]->text[0].unicode()) { case ',': case ';': case '=': should_exit = true; break; case ')': mIndex++; return; case '(': mIndex=mTokenizer[mIndex]->matchIndex+1; break; default: mIndex++; } if (should_exit) break; } } break; case ';': mIndex++; return; case '=': if (mIndex+1text) && addedVar && !(addedVar->properties & StatementProperty::spFunctionPointer) && AutoTypes.contains(addedVar->type)) { //handle e.g.: auto x=blahblah; int pos = 0; int endIndex = skipAssignment(mIndex, tokenCount); QStringList phraseExpression; for (int i=mIndex+1;itext; if (cmd.length()>1 && cmd.endsWith(".")) { phraseExpression.append(cmd.left(cmd.length()-1)); phraseExpression.append("."); } else if (cmd.length()>2 && cmd.endsWith("->")) { phraseExpression.append(cmd.left(cmd.length()-2)); phraseExpression.append("->"); } else if (cmd.length()>2 && cmd.endsWith("::")) { phraseExpression.append(cmd.left(cmd.length()-2)); phraseExpression.append("::"); } else phraseExpression.append(cmd); } PEvalStatement aliasStatement = doEvalExpression(mCurrentFile, phraseExpression, pos, getCurrentScope(), PEvalStatement(), true); if(aliasStatement) { if (aliasStatement->effectiveTypeStatement) { addedVar->type = aliasStatement->effectiveTypeStatement->fullName; if (!addedVar->type.endsWith(">")) addedVar->type += aliasStatement->templateParams; if (aliasStatement->typeStatement && STLIterators.contains(aliasStatement->typeStatement->command) && !aliasStatement->templateParams.isEmpty()) { PStatement parentStatement = aliasStatement->typeStatement->parentScope.lock(); if (parentStatement && STLContainers.contains(parentStatement->fullName)) { addedVar->type = parentStatement->fullName+aliasStatement->templateParams+"::"+aliasStatement->typeStatement->command; } else if (parentStatement && STLMaps.contains(parentStatement->fullName)) { addedVar->type = parentStatement->fullName+aliasStatement->templateParams+"::"+aliasStatement->typeStatement->command; } } } else addedVar->type = aliasStatement->baseType + aliasStatement->templateParams; if (aliasStatement->pointerLevel>0) addedVar->type += QString(aliasStatement->pointerLevel,'*'); } mIndex = endIndex; } else mIndex = skipAssignment(mIndex, tokenCount); addedVar.reset(); break; case '*': case '&': tempType+=mTokenizer[mIndex]->text; mIndex++; break; case '(': if (mTokenizer[mIndex]->matchIndex+1matchIndex+1]->text=='(') { //function pointer QString cmd = findFunctionPointerName(mIndex); int argStart=mTokenizer[mIndex]->matchIndex+1; int argEnd=mTokenizer[argStart]->matchIndex; if (!cmd.isEmpty()) { QString type=lastType; if(type.endsWith("::")) type+=tempType.trimmed(); else type+=" "+tempType.trimmed(); addChildStatement( getCurrentScope(), mCurrentFile, type.trimmed(), cmd, mergeArgs(argStart,argEnd), "", "", mTokenizer[mIndex]->line, StatementKind::skVariable, getScope(), mCurrentMemberAccessibility, //True, (isExtern?StatementProperty::spNone:StatementProperty::spHasDefinition) | (isStatic?StatementProperty::spStatic:StatementProperty::spNone) | StatementProperty::spFunctionPointer); } addedVar.reset(); tempType=""; mIndex=indexOfNextPeriodOrSemicolon(argEnd+1); break; } //not function pointer, fall through case '{': tempType=""; if (mIndex+1text) && addedVar && !(addedVar->properties & StatementProperty::spFunctionPointer) && AutoTypes.contains(addedVar->type)) { int pos = 0; int endIndex = mTokenizer[mIndex]->matchIndex; QStringList phraseExpression; for (int i=mIndex+1;itext; if (cmd.length()>1 && cmd.endsWith(".")) { phraseExpression.append(cmd.left(cmd.length()-1)); phraseExpression.append("."); } else if (cmd.length()>2 && cmd.endsWith("->")) { phraseExpression.append(cmd.left(cmd.length()-2)); phraseExpression.append("->"); } else if (cmd.length()>2 && cmd.endsWith("::")) { phraseExpression.append(cmd.left(cmd.length()-2)); phraseExpression.append("::"); } else phraseExpression.append(cmd); } PEvalStatement aliasStatement = doEvalExpression(mCurrentFile, phraseExpression, pos, getCurrentScope(), PEvalStatement(), true); if(aliasStatement && aliasStatement->effectiveTypeStatement) { if (aliasStatement->effectiveTypeStatement) { addedVar->type = aliasStatement->effectiveTypeStatement->fullName; if (!addedVar->type.endsWith(">")) addedVar->type += aliasStatement->templateParams; if (aliasStatement->typeStatement && STLIterators.contains(aliasStatement->typeStatement->command)) { PStatement parentStatement = aliasStatement->typeStatement->parentScope.lock(); if (parentStatement && STLContainers.contains(parentStatement->fullName)) { addedVar->type = parentStatement->fullName+aliasStatement->templateParams+"::"+aliasStatement->typeStatement->command; } else if (parentStatement && STLMaps.contains(parentStatement->fullName)) { addedVar->type = parentStatement->fullName+aliasStatement->templateParams+"::"+aliasStatement->typeStatement->command; } } } else addedVar->type = aliasStatement->baseType + aliasStatement->templateParams; if (aliasStatement->pointerLevel>0) addedVar->type += QString(aliasStatement->pointerLevel,'*'); } } mIndex=mTokenizer[mIndex]->matchIndex+1; addedVar.reset(); break; default: if (isIdentChar(mTokenizer[mIndex]->text[0])) { QString cmd=mTokenizer[mIndex]->text; //normal var if (cmd=="const") { if (tempType.isEmpty()) { tempType = cmd; } else if (tempType.endsWith("*")) { tempType+=cmd; } else { tempType+=" "+cmd; } } else { QString suffix; QString args; cmd=mTokenizer[mIndex]->text; parseCommandTypeAndArgs(cmd,suffix,args); if (!cmd.isEmpty() && !isKeyword(cmd)) { addedVar = addChildStatement( getCurrentScope(), mCurrentFile, (lastType+' '+tempType+suffix).trimmed(), cmd, args, "", "", mTokenizer[mIndex]->line, StatementKind::skVariable, getScope(), mCurrentMemberAccessibility, //True, (isExtern?StatementProperty::spNone:StatementProperty::spHasDefinition) | (isStatic?StatementProperty::spStatic:StatementProperty::spNone)); tempType=""; } } mIndex++; } else { tempType=""; mIndex++; } } } // Skip ; mIndex++; } void CppParser::internalParse(const QString &fileName) { // Perform some validation before we start if (!mEnabled) return; // if (!isCfile(fileName) && !isHfile(fileName)) // support only known C/C++ files // return; //QElapsedTimer timer; // Preprocess the file... auto action = finally([this]{ mTokenizer.clear(); }); //timer.start(); // Let the preprocessor augment the include records mPreprocessor.setScanOptions(mParseGlobalHeaders, mParseLocalHeaders); mPreprocessor.preprocess(fileName); QStringList preprocessResult = mPreprocessor.result(); #ifdef QT_DEBUG // stringsToFile(mPreprocessor.result(),QString("r:\\preprocess-%1.txt").arg(extractFileName(fileName))); // mPreprocessor.dumpDefinesTo("r:\\defines.txt"); // mPreprocessor.dumpIncludesListTo("r:\\includes.txt"); #endif //qDebug()<<"preprocess"<children) { if (statement->accessibility == StatementAccessibility::Private || statement->kind == StatementKind::skConstructor || statement->kind == StatementKind::skDestructor) continue; StatementAccessibility m_acc; switch(access) { case StatementAccessibility::Public: m_acc = statement->accessibility; break; case StatementAccessibility::Protected: m_acc = StatementAccessibility::Protected; break; case StatementAccessibility::Private: m_acc = StatementAccessibility::Private; break; default: m_acc = StatementAccessibility::Private; } //inherit addInheritedStatement(derived,statement,m_acc); } } void CppParser::fillListOfFunctions(const QString& fileName, int line, const PStatement& statement, const PStatement& scopeStatement, QStringList &list) { StatementMap children = mStatementList.childrenStatements(scopeStatement); for (const PStatement& child:children) { if ((statement->command == child->command) #ifdef Q_OS_WIN || (statement->command +'A' == child->command) || (statement->command +'W' == child->command) #endif ) { if (line < child->line && (child->fileName == fileName)) continue; list.append(prettyPrintStatement(child,child->fileName,child->line)); } } } QList CppParser::getListOfFunctions(const QString &fileName, int line, const PStatement &statement, const PStatement &scopeStatement) const { QList result; StatementMap children = mStatementList.childrenStatements(scopeStatement); for (const PStatement& child:children) { if (( (statement->command == child->command) #ifdef Q_OS_WIN || (statement->command +'A' == child->command) || (statement->command +'W' == child->command) #endif ) ) { if (line < child->line && (child->fileName == fileName)) continue; result.append(child); } } return result; } PStatement CppParser::findMacro(const QString &phrase, const QString &fileName) const { const StatementMap& statementMap =mStatementList.childrenStatements(nullptr); if (statementMap.isEmpty()) return PStatement(); StatementList statements = statementMap.values(phrase); PFileIncludes includes = mPreprocessor.includesList().value(fileName,PFileIncludes()); foreach (const PStatement& s, statements) { if (s->kind == StatementKind::skPreprocessor) { if (includes && !includes->includeFiles.contains(s->fileName) && !includes->includeFiles.contains(s->definitionFileName)) continue; return s; } } return PStatement(); } PStatement CppParser::findMemberOfStatement(const QString &phrase, const PStatement& scopeStatement) const { const StatementMap& statementMap =mStatementList.childrenStatements(scopeStatement); if (statementMap.isEmpty()) return PStatement(); QString s = phrase; //remove [] int p = phrase.indexOf('['); if (p>=0) s.truncate(p); //remove () p = phrase.indexOf('('); if (p>=0) s.truncate(p); //remove <> p =s.indexOf('<'); if (p>=0) s.truncate(p); return statementMap.value(s,PStatement()); } QList CppParser::findMembersOfStatement(const QString &phrase, const PStatement &scopeStatement) const { const StatementMap& statementMap =mStatementList.childrenStatements(scopeStatement); if (statementMap.isEmpty()) return QList(); QString s = phrase; //remove [] int p = phrase.indexOf('['); if (p>=0) s.truncate(p); //remove () p = phrase.indexOf('('); if (p>=0) s.truncate(p); //remove <> p =s.indexOf('<'); if (p>=0) s.truncate(p); return statementMap.values(s); } PStatement CppParser::findStatementInScope(const QString &name, const QString &noNameArgs, StatementKind kind, const PStatement& scope) const { if (scope && scope->kind == StatementKind::skNamespace) { PStatementList namespaceStatementsList = doFindNamespace(scope->command); if (!namespaceStatementsList) return PStatement(); foreach (const PStatement& namespaceStatement, *namespaceStatementsList) { PStatement result=doFindStatementInScope(name,noNameArgs,kind,namespaceStatement); if (result) return result; } } else { return doFindStatementInScope(name,noNameArgs,kind,scope); } return PStatement(); } PStatement CppParser::findStatementInScope(const QString &name, const PStatement &scope) const { if (!scope) return findMemberOfStatement(name,scope); if (scope->kind == StatementKind::skNamespace) { return findStatementInNamespace(name, scope->fullName); } else { return findMemberOfStatement(name,scope); } } PStatement CppParser::findStatementInNamespace(const QString &name, const QString &namespaceName) const { PStatementList namespaceStatementsList=doFindNamespace(namespaceName); if (!namespaceStatementsList) return PStatement(); foreach (const PStatement& namespaceStatement,*namespaceStatementsList) { PStatement result = findMemberOfStatement(name,namespaceStatement); if (result) return result; } return PStatement(); } PEvalStatement CppParser::doEvalExpression(const QString& fileName, QStringList& phraseExpression, int &pos, const PStatement& scope, const PEvalStatement& previousResult, bool freeScoped) const { //dummy function to easy later upgrades return doEvalPointerArithmetic(fileName, phraseExpression, pos, scope, previousResult, freeScoped); } PEvalStatement CppParser::doEvalPointerArithmetic(const QString &fileName, QStringList &phraseExpression, int &pos, const PStatement &scope, const PEvalStatement &previousResult, bool freeScoped) const { if (pos>=phraseExpression.length()) return PEvalStatement(); //find the start scope statement PEvalStatement currentResult = doEvalPointerToMembers( fileName, phraseExpression, pos, scope, previousResult, freeScoped); while (pos < phraseExpression.length()) { if (!currentResult) break; if (currentResult && (phraseExpression[pos]=="+" || phraseExpression[pos]=="-")) { if (currentResult->kind == EvalStatementKind::Variable) { pos++; PEvalStatement op2=doEvalPointerToMembers( fileName, phraseExpression, pos, scope, currentResult, false); //todo operator+/- overload } else if (currentResult->kind == EvalStatementKind::Literal && currentResult->baseType == "int") { pos++; PEvalStatement op2=doEvalPointerToMembers( fileName, phraseExpression, pos, scope, currentResult, false); currentResult = op2; } else break; } else break; } // qDebug()<=phraseExpression.length()) return PEvalStatement(); //find the start scope statement PEvalStatement currentResult = doEvalCCast( fileName, phraseExpression, pos, scope, previousResult, freeScoped); while (pos < phraseExpression.length()) { if (!currentResult) break; if (currentResult && (currentResult->kind == EvalStatementKind::Variable) && (phraseExpression[pos]==".*" || phraseExpression[pos]=="->*")) { pos++; currentResult = doEvalCCast( fileName, phraseExpression, pos, scope, currentResult, false); if (currentResult) { currentResult->pointerLevel++; } } else break; } // qDebug()<=phraseExpression.length()) return PEvalStatement(); PEvalStatement result; if (phraseExpression[pos]=="*") { pos++; //skip "*" result = doEvalCCast( fileName, phraseExpression, pos, scope, previousResult, freeScoped); if (result) { //todo: STL container; if (result->pointerLevel==0) { //STL smart pointers if (result->typeStatement && STLIterators.contains(result->typeStatement->command) ) { PStatement parentScope = result->typeStatement->parentScope.lock(); if (STLContainers.contains(parentScope->fullName)) { QString typeName=doFindFirstTemplateParamOf(fileName,result->templateParams, parentScope); // qDebug()<<"typeName"<baseStatement->type<baseStatement->command; PStatement typeStatement=doFindTypeDefinitionOf(fileName, typeName,parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString=typeName; result->kind = EvalStatementKind::Variable; } else { result = PEvalStatement(); } } else if (STLMaps.contains(parentScope->fullName)) { QString typeName=doFindTemplateParamOf(fileName,result->templateParams,1,parentScope); // qDebug()<<"typeName"<baseStatement->type<baseStatement->command; PStatement typeStatement=doFindTypeDefinitionOf(fileName, typeName,parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString=typeName; result->kind = EvalStatementKind::Variable; } else { result = PEvalStatement(); } } } else { PStatement typeStatement = result->effectiveTypeStatement; if ((typeStatement) && STLPointers.contains(typeStatement->fullName) && result->kind == EvalStatementKind::Variable && result->baseStatement) { PStatement parentScope = result->baseStatement->parentScope.lock(); QString typeName; if (!previousResult || previousResult->definitionString.isEmpty()) typeName = doFindFirstTemplateParamOf(fileName,result->baseStatement->type, parentScope); else typeName = doFindFirstTemplateParamOf(fileName,previousResult->definitionString,parentScope); // qDebug()<<"typeName"<definitionString=typeName; result->kind = EvalStatementKind::Variable; } else { result = PEvalStatement(); } } } } else result->pointerLevel--; } } else if (phraseExpression[pos]=="&") { pos++; //skip "&" result = doEvalCCast( fileName, phraseExpression, pos, scope, previousResult, freeScoped); if (result) { result->pointerLevel++; } } else if (phraseExpression[pos]=="++" || phraseExpression[pos]=="--") { pos++; //skip "++" or "--" result = doEvalCCast( fileName, phraseExpression, pos, scope, previousResult, freeScoped); } else if (phraseExpression[pos]=="(") { //parse int startPos = pos; pos++; // qDebug()<<"parse type cast ()"; PEvalStatement evalType = doEvalExpression( fileName, phraseExpression, pos, scope, PEvalStatement(), true); // qDebug()<= phraseExpression.length() || phraseExpression[pos]!=")") { return PEvalStatement(); } else if (evalType && (evalType->kind == EvalStatementKind::Type)) { pos++; // skip ")" // qDebug()<<"parse type cast exp"; //it's a type cast result = doEvalCCast(fileName, phraseExpression, pos, scope, previousResult, freeScoped); if (result) { // qDebug()<<"type cast"; result->assignType(evalType); } } else //it's not a type cast result = doEvalMemberAccess( fileName, phraseExpression, startPos, //we must reparse it scope, previousResult, freeScoped); } else result = doEvalMemberAccess( fileName, phraseExpression, pos, scope, previousResult, freeScoped); // if (result) { // qDebug()<kind<baseType; // } else { // qDebug()<<"!!!!!!!!!!!not found"; // } return result; } PEvalStatement CppParser::doEvalMemberAccess(const QString &fileName, QStringList &phraseExpression, int &pos, const PStatement& scope, const PEvalStatement& previousResult, bool freeScoped) const { // qDebug()<<"eval member access "<=phraseExpression.length()) return result; PEvalStatement lastResult = previousResult; result = doEvalScopeResolution( fileName, phraseExpression, pos, scope, previousResult, freeScoped); if (!result) return PEvalStatement(); while (poskind == EvalStatementKind::Type) { pos++; // skip "(" PEvalStatement newResult = doEvalExpression( fileName, phraseExpression, pos, scope, PEvalStatement(), true); if (newResult) newResult->assignType(result); pos++; // skip ")" result = newResult; } else if (result->kind == EvalStatementKind::Function) { doSkipInExpression(phraseExpression,pos,"(",")"); // qDebug()<<"????"<<(result->baseStatement!=nullptr)<<(lastResult!=nullptr); if (result->baseStatement && lastResult) { PStatement parentScope = result->baseStatement->parentScope.lock(); if (parentScope && STLElementMethods.contains(result->baseStatement->command) ) { if (STLContainers.contains(parentScope->fullName)) { //stl container methods PStatement typeStatement = result->effectiveTypeStatement; QString typeName; if (!lastResult->definitionString.isEmpty()) typeName = doFindFirstTemplateParamOf(fileName,lastResult->definitionString,parentScope); else if (lastResult->baseStatement) typeName = doFindFirstTemplateParamOf(fileName,lastResult->baseStatement->type, parentScope); // qDebug()<<"typeName"<baseStatement->type<baseStatement->command; if (!typeName.isEmpty()) typeStatement=doFindTypeDefinitionOf(fileName, typeName,parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString = typeName; result->kind = EvalStatementKind::Variable; lastResult = result; } else { return PEvalStatement(); } } else if (STLMaps.contains(parentScope->fullName)) { //stl map methods PStatement typeStatement = result->effectiveTypeStatement; QString typeName; if (!lastResult->definitionString.isEmpty()) typeName = doFindTemplateParamOf(fileName,lastResult->definitionString,1,parentScope); else if (lastResult->baseStatement) typeName = doFindTemplateParamOf(fileName,lastResult->baseStatement->type,1, parentScope); // qDebug()<<"typeName"<baseStatement->type<baseStatement->command; if (!typeName.isEmpty()) typeStatement=doFindTypeDefinitionOf(fileName, typeName,parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString = typeName; result->kind = EvalStatementKind::Variable; lastResult = result; } else { return PEvalStatement(); } } } } result->kind = EvalStatementKind::Variable; } else result = PEvalStatement(); } else if (phraseExpression[pos] == "[") { //skip to "]" doSkipInExpression(phraseExpression,pos,"[","]"); if (result->pointerLevel>0) result->pointerLevel--; else { PStatement typeStatement = result->effectiveTypeStatement; if (typeStatement && result->kind == EvalStatementKind::Variable && result->baseStatement) { if (STLContainers.contains(typeStatement->fullName)) { PStatement parentScope = result->baseStatement->parentScope.lock(); QString typeName; if (!lastResult || lastResult->definitionString.isEmpty()) typeName = doFindFirstTemplateParamOf(fileName,result->baseStatement->type, parentScope); else typeName = doFindFirstTemplateParamOf(fileName,lastResult->definitionString,parentScope); typeStatement = doFindTypeDefinitionOf(fileName, typeName, parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString=typeName; result->kind = EvalStatementKind::Variable; lastResult = result; } else { return PEvalStatement(); } } else if (STLMaps.contains(typeStatement->fullName)) { PStatement parentScope = result->baseStatement->parentScope.lock(); QString typeName; if (!lastResult || lastResult->definitionString.isEmpty()) typeName = doFindTemplateParamOf(fileName,result->baseStatement->type, 1,parentScope); else typeName = doFindTemplateParamOf(fileName,lastResult->definitionString,1,parentScope); typeStatement = doFindTypeDefinitionOf(fileName, typeName, parentScope); if (typeStatement) { result = doCreateEvalType(fileName,typeStatement); result->definitionString=typeName; result->kind = EvalStatementKind::Variable; lastResult = result; } else { return PEvalStatement(); } } } else { return PEvalStatement(); } } } else if (phraseExpression[pos] == ".") { pos++; lastResult = result; result = doEvalScopeResolution( fileName, phraseExpression, pos, scope, result, false); } else if (phraseExpression[pos] == "->") { pos++; // qDebug()<<"pointer level"<pointerLevel; if (result->pointerLevel==0) { PStatement typeStatement = result->effectiveTypeStatement; if ((typeStatement) && STLPointers.contains(typeStatement->fullName) && result->kind == EvalStatementKind::Variable && result->baseStatement) { PStatement parentScope = result->baseStatement->parentScope.lock(); QString typeName; if (!previousResult || previousResult->definitionString.isEmpty()) typeName = doFindFirstTemplateParamOf(fileName,result->baseStatement->type, parentScope); else typeName = doFindFirstTemplateParamOf(fileName,previousResult->definitionString,parentScope); // qDebug()<<"typeName"<definitionString=typeName; result->kind = EvalStatementKind::Variable; } else { return PEvalStatement(); } } } else { result->pointerLevel--; } lastResult = result; result = doEvalScopeResolution( fileName, phraseExpression, pos, scope, result, false); } else break; } return result; } PEvalStatement CppParser::doEvalScopeResolution(const QString &fileName, QStringList &phraseExpression, int &pos, const PStatement& scope, const PEvalStatement& previousResult, bool freeScoped) const { // qDebug()<<"eval scope res "<=phraseExpression.length()) return result; result = doEvalTerm( fileName, phraseExpression, pos, scope, previousResult, freeScoped); while (poskind == EvalStatementKind::Type) { //class static member result = doEvalTerm(fileName, phraseExpression, pos, scope, result, false); } else if (result->kind == EvalStatementKind::Namespace) { //namespace result = doEvalTerm(fileName, phraseExpression, pos, scope, result, false); } if (!result) break; } else break; } // qDebug()<baseType<=phraseExpression.length()) return result; // qDebug()<<"eval term"<= phraseExpression.length() || phraseExpression[pos]!=")") return PEvalStatement(); else { pos++; // skip ")"; return result; } } else { int pointerLevel = 0; //skip "struct", "const", "static", etc while(pos < phraseExpression.length()) { QString token = phraseExpression[pos]; if (token=="*") // for expression like (const * char)? pointerLevel++; else if (mCppTypeKeywords.contains(token) || !mCppKeywords.contains(token)) break; pos++; } if (pos>=phraseExpression.length() || phraseExpression[pos]==")") return result; if (mCppKeywords.contains(phraseExpression[pos])) { result = doCreateEvalType(phraseExpression[pos]); pos++; } else if (isIdentifier(phraseExpression[pos])) { PStatement statement; if (freeScoped) { if (!previousResult) { statement = findStatementStartingFrom( fileName, phraseExpression[pos], scope); } else { statement = findStatementStartingFrom( fileName, phraseExpression[pos], previousResult->effectiveTypeStatement); } } else { if (!previousResult) { statement = findStatementInScope(phraseExpression[pos],PStatement()); } else { statement = findStatementInScope(phraseExpression[pos],previousResult->effectiveTypeStatement); } // if (!statement) { // statement = findMacro( // phraseExpression[pos], // fileName); // } } pos++; if (statement && statement->kind == StatementKind::skConstructor) { statement = statement->parentScope.lock(); } if (statement) { switch (statement->kind) { case StatementKind::skNamespace: result = doCreateEvalNamespace(statement); break; case StatementKind::skAlias: { statement = doFindAliasedStatement(statement); if (statement) result = doCreateEvalNamespace(statement); } break; case StatementKind::skVariable: case StatementKind::skParameter: result = doCreateEvalVariable(fileName,statement, previousResult?previousResult->templateParams:"",scope); break; case StatementKind::skEnumType: case StatementKind::skClass: case StatementKind::skEnumClassType: case StatementKind::skTypedef: result = doCreateEvalType(fileName,statement); break; case StatementKind::skFunction: result = doCreateEvalFunction(fileName,statement); break; case StatementKind::skPreprocessor: //qDebug()<<"before"<typeStatement && STLIterators.contains(result->typeStatement->command) && previousResult) { PStatement parentStatement = result->typeStatement->parentScope.lock(); if (parentStatement && STLContainers.contains(parentStatement->fullName)) { result->templateParams = previousResult->templateParams; } else if (parentStatement && STLMaps.contains(parentStatement->fullName)) { result->templateParams = previousResult->templateParams; } } } } else if (isIntegerLiteral(phraseExpression[pos])) { result = doCreateEvalLiteral("int"); pos++; } else if (isFloatLiteral(phraseExpression[pos])) { result = doCreateEvalLiteral("double"); pos++; } else if (isStringLiteral(phraseExpression[pos])) { result = doCreateEvalLiteral("char"); result->pointerLevel = 1; pos++; } else if (isCharLiteral(phraseExpression[pos])) { result = doCreateEvalLiteral("char"); pos++; } else return result; // if (result) { // qDebug()<<"term kind:"<<(int)result->kind; // } if (result && result->kind == EvalStatementKind::Type) { //skip "struct", "const", "static", etc while(pos < phraseExpression.length()) { QString token = phraseExpression[pos]; if (token=="*") // for expression like (const * char)? pointerLevel++; else if (mCppTypeKeywords.contains(token) || !mCppKeywords.contains(token)) break; pos++; } result->pointerLevel = pointerLevel; } } // qDebug()<args.isEmpty()) { s=macro->value; phraseExpression.removeAt(pos); } else { QString args=macro->args.mid(1,macro->args.length()-2).trimmed(); // remove '(' ')' if(args=="") return false; QStringList argList = args.split(','); QList argUsed; for (int i=0;i tokens = mPreprocessor.tokenizeValue(macro->value); QString formatStr = ""; DefineArgTokenType lastTokenType=DefineArgTokenType::Other; int index; foreach (const PDefineArgToken& token, tokens) { switch(token->type) { case DefineArgTokenType::Identifier: index = argList.indexOf(token->value); if (index>=0) { argUsed[index] = true; if (lastTokenType == DefineArgTokenType::Sharp) { formatStr+= "\"%"+QString("%1").arg(index+1)+"\""; break; } else { formatStr+= "%"+QString("%1").arg(index+1); break; } } formatStr += token->value; break; case DefineArgTokenType::DSharp: case DefineArgTokenType::Sharp: break; case DefineArgTokenType::Space: case DefineArgTokenType::Symbol: formatStr+=token->value; break; default: break; } lastTokenType = token->type; } QStringList values; int i; int level=1; QString current; for (i=pos+2;itokenType()) { case QSynedit::TokenType::Space: case QSynedit::TokenType::Comment: break; default: phraseExpression.insert(pos+i,token); i++; } syntaxer.next(); } return true; } PEvalStatement CppParser::doCreateEvalNamespace(const PStatement &namespaceStatement) const { if (!namespaceStatement) return PEvalStatement(); return std::make_shared( namespaceStatement->fullName, EvalStatementKind::Namespace, PStatement(), namespaceStatement, namespaceStatement); } PEvalStatement CppParser::doCreateEvalType(const QString& fileName,const PStatement &typeStatement) const { if (!typeStatement) return PEvalStatement(); if (typeStatement->kind == StatementKind::skTypedef) { QString baseType; int pointerLevel=0; QString templateParams; PStatement tempStatement; PStatement effetiveTypeStatement = doParseEvalTypeInfo( fileName, typeStatement->parentScope.lock(), typeStatement->type + typeStatement->args, baseType, tempStatement, pointerLevel, templateParams); return std::make_shared( baseType, EvalStatementKind::Type, PStatement(), typeStatement, effetiveTypeStatement, pointerLevel, templateParams ); } else { return std::make_shared( typeStatement->fullName, EvalStatementKind::Type, typeStatement, typeStatement, typeStatement); } } PEvalStatement CppParser::doCreateEvalType(const QString &primitiveType) const { return std::make_shared( primitiveType, EvalStatementKind::Type, PStatement(), PStatement(), PStatement()); } PEvalStatement CppParser::doCreateEvalVariable( const QString &fileName, const PStatement& varStatement, const QString& baseTemplateParams, const PStatement& scope) const { if (!varStatement) return PEvalStatement(); QString baseType; int pointerLevel=0; QString templateParams; PStatement typeStatement; PStatement effectiveTypeStatement; //todo: ugly implementation for std::pair if (varStatement->fullName == "std::pair::first") { effectiveTypeStatement = doParseEvalTypeInfo( fileName, scope, doFindFirstTemplateParamOf(fileName,baseTemplateParams,scope), baseType, typeStatement, pointerLevel, templateParams); } else if (varStatement->fullName == "std::pair::second") { effectiveTypeStatement = doParseEvalTypeInfo( fileName, scope, doFindTemplateParamOf(fileName,baseTemplateParams,1,scope), baseType, typeStatement, pointerLevel, templateParams); } else { effectiveTypeStatement = doParseEvalTypeInfo( fileName, varStatement->parentScope.lock(), varStatement->type+varStatement->args, baseType, typeStatement, pointerLevel, templateParams); } // qDebug()<<"parse ..."<( baseType, EvalStatementKind::Variable, varStatement, typeStatement, effectiveTypeStatement, pointerLevel, templateParams ); } PEvalStatement CppParser::doCreateEvalFunction( const QString &fileName, const PStatement& funcStatement) const { if (!funcStatement) return PEvalStatement(); QString baseType; int pointerLevel=0; QString templateParams; PStatement typeStatement; PStatement effetiveTypeStatement = doParseEvalTypeInfo( fileName, funcStatement->parentScope.lock(), funcStatement->type, baseType, typeStatement, pointerLevel, templateParams); return std::make_shared( baseType, EvalStatementKind::Function, funcStatement, typeStatement, effetiveTypeStatement, pointerLevel, templateParams ); } PEvalStatement CppParser::doCreateEvalLiteral(const QString &type) const { return std::make_shared( type, EvalStatementKind::Literal, PStatement(), PStatement(), PStatement()); } void CppParser::doSkipInExpression(const QStringList &expression, int &pos, const QString &startSymbol, const QString &endSymbol) const { int level = 0; while (postext=="("); int i=startIdx+1; int endIdx = mTokenizer[startIdx]->matchIndex; while (itext[0])) { return mTokenizer[i]->text; } i++; } return QString(); } PStatement CppParser::doParseEvalTypeInfo( const QString &fileName, const PStatement &scope, const QString &type, QString &baseType, PStatement& typeStatement, int &pointerLevel, QString& templateParams) const { // Remove pointer stuff from type QString s = type; // qDebug()<<"eval type info"<tokenType() == QSynedit::TokenType::Identifier) { baseType += token; } else if (token == "[") { pointerLevel++; bracketLevel++; } else if (token == "<") { templateLevel++; templateParams += token; } else if (token == "::") { baseType += token; } } else if (bracketLevel > 0) { if (token == "[") { bracketLevel++; } else if (token == "]") { bracketLevel--; } } else if (templateLevel > 0) { if (token == "<") { templateLevel++; } else if (token == ">") { templateLevel--; } templateParams += token; } syntaxer.next(); } while ((position >= 0) && (s[position] == '*' || s[position] == ' ' || s[position] == '&')) { if (s[position]=='*') { } position--; } typeStatement = doFindStatementOf(fileName,baseType,scope); return getTypeDef(typeStatement,fileName,baseType); } int CppParser::getBracketEnd(const QString &s, int startAt) const { int i = startAt; int level = 0; // assume we start on top of [ while (i < s.length()) { switch(s[i].unicode()) { case '<': level++; break; case '>': level--; if (level == 0) return i; } i++; } return startAt; } PStatement CppParser::doFindStatementInScope(const QString &name, const QString &noNameArgs, StatementKind kind, const PStatement& scope) const { const StatementMap& statementMap = mStatementList.childrenStatements(scope); foreach (const PStatement& statement, statementMap.values(name)) { if (statement->kind == kind && statement->noNameArgs == noNameArgs) { return statement; } } return PStatement(); } void CppParser::internalInvalidateFile(const QString &fileName) { if (fileName.isEmpty()) return; // remove its include files list PFileIncludes p = findFileIncludes(fileName, true); if (p) { //fPreprocessor.InvalidDefinesInFile(FileName); //we don't need this, since we reset defines after each parse //p->includeFiles.clear(); //p->usings.clear(); for (PStatement& statement:p->statements) { if (statement->fileName==fileName) { mStatementList.deleteStatement(statement); } else { statement->setHasDefinition(false); statement->definitionFileName = statement->fileName; statement->definitionLine = statement->line; } } p->statements.clear(); } //remove all statements from namespace cache const QList& keys=mNamespaces.keys(); for (const QString& key:keys) { PStatementList statements = mNamespaces.value(key); for (int i=statements->size()-1;i>=0;i--) { PStatement statement = statements->at(i); if (statement->fileName == fileName) { statements->removeAt(i); } } if (statements->isEmpty()) { mNamespaces.remove(key); } } // delete it from scannedfiles mPreprocessor.removeScannedFile(fileName); } void CppParser::internalInvalidateFiles(const QSet &files) { for (const QString& file:files) internalInvalidateFile(file); } QSet CppParser::calculateFilesToBeReparsed(const QString &fileName) { if (fileName.isEmpty()) return QSet(); QSet result; result.insert(fileName); foreach (const QString& file, mProjectFiles) { PFileIncludes fileIncludes = mPreprocessor.includesList().value(file,PFileIncludes()); if (fileIncludes && fileIncludes->includeFiles.contains(fileName)) { result.insert(file); } } return result; } //int CppParser::calcKeyLenForStruct(const QString &word) //{ // if (word.startsWith("struct")) // return 6; // else if (word.startsWith("class") // || word.startsWith("union")) // return 5; // return -1; //} void CppParser::scanMethodArgs(const PStatement& functionStatement, int argStart) { Q_ASSERT(mTokenizer[argStart]->text=='('); int argEnd=mTokenizer[argStart]->matchIndex; int paramStart = argStart+1; int i = paramStart ; // assume it starts with ( and ends with ) // Keep going and stop on top of the variable name QString varType = ""; while (i < argEnd) { if (mTokenizer[i]->text=='(' && mTokenizer[i]->matchIndex+1matchIndex+1]->text=='(') { //function pointer int argStart=mTokenizer[i]->matchIndex+1; int argEnd=mTokenizer[argStart]->matchIndex; QString cmd=findFunctionPointerName(i); QString args=mergeArgs(argStart,argEnd); if (!cmd.isEmpty()) { addStatement( functionStatement, mCurrentFile, varType, // 'int*' cmd, // a args, "", "", mTokenizer[i+1]->line, StatementKind::skParameter, StatementScope::Local, StatementAccessibility::None, StatementProperty::spHasDefinition); } i=argEnd+1; varType=""; } else if (mTokenizer[i]->text=='{') { i=mTokenizer[i]->matchIndex+1; } else if (mTokenizer[i]->text=='(') { i=mTokenizer[i]->matchIndex+1; } else if (mTokenizer[i]->text.endsWith('=')) { i=skipAssignment(i,argEnd); } else if (mTokenizer[i]->text=="::") { varType+=mTokenizer[i]->text; i++; } else if (isWordChar(mTokenizer[i]->text[0])) { QString cmd=mTokenizer[i]->text; if (i+1==argEnd || mTokenizer[i+1]->text==',' || mTokenizer[i+1]->text=='=') { QString args,suffix; parseCommandTypeAndArgs(cmd,suffix,args); if (!cmd.isEmpty()) { PStatement statement = doFindStatementOf(mCurrentFile,cmd,functionStatement); bool noCmd = (statement && isTypeStatement(statement->kind)); if (!noCmd) { addStatement( functionStatement, mCurrentFile, varType+suffix, // 'int*' cmd, // a args, "", "", mTokenizer[i]->line, StatementKind::skParameter, StatementScope::Local, StatementAccessibility::None, StatementProperty::spHasDefinition); } } } else { if (!varType.isEmpty()) varType+=' '; varType+=cmd; } i++; } else { i++; varType=""; } } } QString CppParser::splitPhrase(const QString &phrase, QString &sClazz, QString &sOperator, QString &sMember) const { sClazz=""; sMember=""; sOperator=""; QString result=""; int bracketLevel = 0; int phraseLength = phrase.length(); // Obtain stuff before first operator int firstOpStart = phraseLength + 1; int firstOpEnd = phraseLength + 1; for (int i = 0; i') && (bracketLevel==0)) { firstOpStart = i; firstOpEnd = i+2; sOperator = "->"; break; } else if ((i+1') && (bracketLevel=0)) { secondOp = i; break; } else if ((i+1=0) { return phrase.left(pos); } return phrase; } bool CppParser::splitLastMember(const QString &token, QString &lastMember, QString &remaining) { int pos = token.length()-1; int level=0; bool found=false; while (pos>=0 && !found) { switch(token[pos].unicode()) { case ']': case '>': case ')': level++; break; case '[': case '<': case '(': level--; break; case ':': if (level==0 && pos>0 && token[pos-1]==':') { found=true; break; } } pos--; } if (pos<0) return false; lastMember=token.mid(pos+2); remaining=token.left(pos); return true; } //static void appendArgWord(QString& args, const QString& word) { // QString s=word.trimmed(); // if (s.isEmpty()) // return; // if (args.isEmpty()) // args.append(s); // else if (isIdentChar(args.back()) && isIdentChar(word.front()) ) { // args+=" "; // args+=s; // } else { // args+=s; // } //} bool CppParser::isNotFuncArgs(int startIndex) { Q_ASSERT(mTokenizer[startIndex]->text=='('); int endIndex=mTokenizer[startIndex]->matchIndex; //no args, it must be a function if (endIndex-startIndex==1) return false; int i=startIndex+1; //skip '(' int endPos = endIndex; QString word = ""; while (itext[0]; switch(ch.unicode()) { // args contains a string/char, can't be a func define case '"': case '\'': case '+': case '-': case '/': case '|': case '!': case '{': return true; case '[': // function args like int f[10] i=mTokenizer[i]->matchIndex+1; if (itext=='(' || mTokenizer[i]->text=='{')) //lambda return true; continue; } if (isDigitChar(ch)) return true; if (isIdentChar(ch)) { QString currentText=mTokenizer[i]->text; // if (mTokenizer[i]->text.endsWith('.')) // return true; // if (mTokenizer[i]->text.endsWith("->")) // return true; if (!mCppTypeKeywords.contains(currentText)) { if (currentText=="true" || currentText=="false" || currentText=="nullptr" || currentText=="this") return true; if (currentText=="const") return false; if (isCppKeyword(currentText)) return false; PStatement statement = doFindStatementOf(mCurrentFile,word,getCurrentScope()); //template arguments if (!statement) return false; if (statement && isTypeStatement(statement->kind)) return false; } else { return false; } } i++; } //function with no args return false; } bool CppParser::isNamedScope(StatementKind kind) const { switch(kind) { case StatementKind::skClass: case StatementKind::skNamespace: case StatementKind::skFunction: return true; default: return false; } } bool CppParser::isTypeStatement(StatementKind kind) const { switch(kind) { case StatementKind::skClass: case StatementKind::skTypedef: case StatementKind::skEnumClassType: case StatementKind::skEnumType: return true; default: return false; } } void CppParser::updateSerialId() { mSerialId = QString("%1 %2").arg(mParserId).arg(mSerialCount); } int CppParser::indexOfNextSemicolon(int index, int endIndex) { if (endIndex<0) endIndex=mTokenizer.tokenCount(); while (indextext[0].unicode()) { case ';': return index; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } int CppParser::indexOfNextPeriodOrSemicolon(int index, int endIndex) { if (endIndex<0) endIndex=mTokenizer.tokenCount(); while (indextext[0].unicode()) { case ';': case ',': case '}': return index; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } int CppParser::indexOfNextSemicolonOrLeftBrace(int index) { int tokenCount = mTokenizer.tokenCount(); while (indextext[0].unicode()) { case ';': case '{': return index; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } int CppParser::indexOfNextColon(int index) { int tokenCount = mTokenizer.tokenCount(); while (indextext; switch(s[0].unicode()) { case ':': if (s.length()==1) return index; else index++; break; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } int CppParser::indexOfNextLeftBrace(int index) { int tokenCount = mTokenizer.tokenCount(); while (indextext[0].unicode()) { case '{': return index; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } int CppParser::indexPassParenthesis(int index) { int tokenCount = mTokenizer.tokenCount(); while (indextext=='(') { return mTokenizer[index]->matchIndex+1; } index++; } return index; } int CppParser::indexPassBraces(int index) { int tokenCount = mTokenizer.tokenCount(); while (tokenCount) { switch(mTokenizer[index]->text[0].unicode()) { case '{': return mTokenizer[index]->matchIndex+1; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } return index; } void CppParser::skipNextSemicolon(int index) { int tokenCount = mTokenizer.tokenCount(); mIndex=index; while (mIndextext[0].unicode()) { case ';': mIndex++; return; case '{': mIndex = mTokenizer[mIndex]->matchIndex+1; break; case '(': mIndex = mTokenizer[mIndex]->matchIndex+1; break; default: mIndex++; } } } int CppParser::moveToEndOfStatement(int index, bool checkLambda, int endIndex) { int startIndex=index; if (endIndex<0) endIndex=mTokenizer.tokenCount(); if (index>=endIndex) return index; index--; // compensate for the first loop bool skip=true; do { index++; bool stop = false; while (indextext[0].unicode()) { case ';': stop=true; break; case '=': stop=true; break; case '}': stop=true; skip=false; //don't skip the orphan '}' that we found break; case '{': //move to '}' index=mTokenizer[index]->matchIndex; stop=true; break; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } } while (indextext=='='); if (index0 && mTokenizer.indexOfFirstLambda()=startIndex) { handleLambda(i,index); } } } if (skip) index++; return index; } void CppParser::skipParenthesis(int index) { int tokenCount = mTokenizer.tokenCount(); mIndex=index; while (mIndextext=='(') { mIndex=mTokenizer[mIndex]->matchIndex+1; return; } mIndex++; } } int CppParser::skipAssignment(int index, int endIndex) { int startIndex=index; bool stop=false; while (indextext[0].unicode()) { case ';': case ',': case '{': stop=true; break; case '(': index = mTokenizer[index]->matchIndex+1; break; default: index++; } } if (stop) { while (mTokenizer.lambdasCount()>0 && mTokenizer.indexOfFirstLambda()=startIndex) { handleLambda(i,index); } } } return (indexstartIndex) result+=' '; result+=mTokenizer[i]->text; } return result; } void CppParser::parseCommandTypeAndArgs(QString &command, QString &typeSuffix, QString &args) const { int prefix=0; while (prefix0) { typeSuffix=command.left(prefix); command=command.mid(prefix); } else { typeSuffix=""; } int pos=command.indexOf('['); if (pos>=0) { args=command.mid(pos); command=command.left(pos); } else { args=""; } } const QSet &CppParser::projectFiles() const { return mProjectFiles; } QList CppParser::namespaces() { QMutexLocker locker(&mMutex); return mNamespaces.keys(); } ParserLanguage CppParser::language() const { return mLanguage; } void CppParser::setLanguage(ParserLanguage newLanguage) { mLanguage = newLanguage; } const StatementModel &CppParser::statementList() const { return mStatementList; } bool CppParser::parseGlobalHeaders() const { return mParseGlobalHeaders; } void CppParser::setParseGlobalHeaders(bool newParseGlobalHeaders) { mParseGlobalHeaders = newParseGlobalHeaders; } const QSet &CppParser::includePaths() { return mPreprocessor.includePaths(); } const QSet &CppParser::projectIncludePaths() { return mPreprocessor.projectIncludePaths(); } bool CppParser::parseLocalHeaders() const { return mParseLocalHeaders; } void CppParser::setParseLocalHeaders(bool newParseLocalHeaders) { mParseLocalHeaders = newParseLocalHeaders; } const QString &CppParser::serialId() const { return mSerialId; } int CppParser::parserId() const { return mParserId; } void CppParser::setOnGetFileStream(const GetFileStreamCallBack &newOnGetFileStream) { mPreprocessor.setOnGetFileStream(newOnGetFileStream); } const QSet &CppParser::filesToScan() const { return mFilesToScan; } void CppParser::setFilesToScan(const QSet &newFilesToScan) { mFilesToScan = newFilesToScan; } bool CppParser::enabled() const { return mEnabled; } void CppParser::setEnabled(bool newEnabled) { if (mEnabled!=newEnabled) { mEnabled = newEnabled; if (!mEnabled) { resetParser(); } } } CppFileParserThread::CppFileParserThread( PCppParser parser, QString fileName, bool inProject, bool onlyIfNotParsed, bool updateView, QObject *parent):QThread(parent), mParser(parser), mFileName(fileName), mInProject(inProject), mOnlyIfNotParsed(onlyIfNotParsed), mUpdateView(updateView) { connect(this,&QThread::finished, this,&QObject::deleteLater); } void CppFileParserThread::run() { if (mParser && !mParser->parsing()) { mParser->parseFile(mFileName,mInProject,mOnlyIfNotParsed,mUpdateView); } } CppFileListParserThread::CppFileListParserThread(PCppParser parser, bool updateView, QObject *parent): QThread(parent), mParser(parser), mUpdateView(updateView) { connect(this,&QThread::finished, this,&QObject::deleteLater); } void CppFileListParserThread::run() { if (mParser && !mParser->parsing()) { mParser->parseFileList(mUpdateView); } } void parseFile(PCppParser parser, const QString& fileName, bool inProject, bool onlyIfNotParsed, bool updateView) { if (!parser) return; if (!parser->enabled()) return; //delete when finished CppFileParserThread* thread = new CppFileParserThread(parser,fileName,inProject,onlyIfNotParsed,updateView); thread->connect(thread, &QThread::finished, thread, &QThread::deleteLater); thread->start(); } void parseFileList(PCppParser parser, bool updateView) { if (!parser) return; if (!parser->enabled()) return; //delete when finished CppFileListParserThread *thread = new CppFileListParserThread(parser,updateView); thread->connect(thread, &QThread::finished, thread, &QThread::deleteLater); thread->start(); }