2050 lines
62 KiB
C++
2050 lines
62 KiB
C++
/*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "cpppreprocessor.h"
|
|
|
|
#include <QFile>
|
|
#include <QTextCodec>
|
|
#include <QDebug>
|
|
#include <QMessageBox>
|
|
#include "../utils.h"
|
|
|
|
CppPreprocessor::CppPreprocessor()
|
|
{
|
|
}
|
|
|
|
void CppPreprocessor::clear()
|
|
{
|
|
//don't use reset(), it will reset(add) defines.
|
|
clearTempResults();
|
|
|
|
//Result across processings.
|
|
//used by parser even preprocess finished
|
|
mIncludesList.clear();
|
|
mFileDefines.clear(); //dictionary to save defines for each headerfile;
|
|
mFileUndefines.clear(); //dictionary to save undefines for each headerfile;
|
|
mScannedFiles.clear();
|
|
|
|
//option data for the parser
|
|
//{ List of current project's include path }
|
|
mDefines.clear();
|
|
//mHardDefines.clear(); // set by "cpp -dM -E -xc NUL"
|
|
mProjectIncludePaths.clear();
|
|
//we also need include paths in order (for #include_next)
|
|
mIncludePathList.clear();
|
|
mProjectIncludePathList.clear();
|
|
//{ List of current compiler set's include path}
|
|
mIncludePaths.clear();
|
|
}
|
|
|
|
void CppPreprocessor::clearTempResults()
|
|
{
|
|
//temporary data when preprocessing single file
|
|
mFileName="";
|
|
mBuffer.clear();
|
|
mResult.clear();
|
|
mCurrentIncludes=nullptr;
|
|
mIncludes.clear(); // stack of files we've stepped into. last one is current file, first one is source file
|
|
mBranchResults.clear();// stack of branch results (boolean). last one is current branch, first one is outermost branch
|
|
//mDefines.clear(); // working set, editable
|
|
mProcessed.clear(); // dictionary to save filename already processed
|
|
}
|
|
|
|
void CppPreprocessor::addDefineByParts(const QString &name, const QString &args, const QString &value, bool hardCoded)
|
|
{
|
|
// Check for duplicates
|
|
PDefine define = std::make_shared<Define>();
|
|
define->name = name;
|
|
define->args = args;
|
|
define->value = value;
|
|
define->filename = mFileName;
|
|
//define->argList;
|
|
define->formatValue = value;
|
|
define->hardCoded = hardCoded;
|
|
define->varArgIndex = -1;
|
|
if (!args.isEmpty())
|
|
parseArgs(define);
|
|
if (hardCoded) {
|
|
mHardDefines.insert(name,define);
|
|
mDefines.insert(name,define);
|
|
} else {
|
|
PDefineMap defineMap = mFileDefines.value(mFileName,PDefineMap());
|
|
if (!defineMap) {
|
|
defineMap = std::make_shared<DefineMap>();
|
|
mFileDefines.insert(mFileName,defineMap);
|
|
}
|
|
defineMap->insert(define->name,define);
|
|
mDefines.insert(name,define);
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::getDefineParts(const QString &input, QString &name, QString &args, QString &value)
|
|
{
|
|
QString s = input.trimmed();
|
|
name = "";
|
|
args = "";
|
|
value = "";
|
|
|
|
// Rules:
|
|
// When the character before the first opening brace is nonblank, a function is defined.
|
|
// After that point, switch from name to args
|
|
// The value starts after the first blank character outside of the outermost () pair
|
|
|
|
int i = 0;
|
|
int level = 0;
|
|
bool isFunction = false;
|
|
int argStart = 0;
|
|
while (i < s.length()) {
|
|
// When we find the first opening brace, check if this is a function define
|
|
if (s[i] == '(') {
|
|
level++;
|
|
if ((level == 1) && (!isFunction)) { // found a function define!
|
|
name = s.mid(0,i);
|
|
argStart = i;
|
|
isFunction = true;
|
|
}
|
|
} else if (s[i]==')') {
|
|
level--;
|
|
} else if (isSpaceChar(s[i]) && (level == 0)) {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
if (isFunction) {
|
|
// Name has already been found
|
|
args = s.mid(argStart,i-argStart);
|
|
//todo: expand macro (if already have)
|
|
} else {
|
|
name = s.mid(0,i);
|
|
args = "";
|
|
}
|
|
value = removeGCCAttributes(s.mid(i+1).trimmed());
|
|
name.squeeze();
|
|
value.squeeze();
|
|
args.squeeze();
|
|
}
|
|
|
|
void CppPreprocessor::addHardDefineByLine(const QString &line)
|
|
{
|
|
addDefineByLine(line,true);
|
|
}
|
|
|
|
void CppPreprocessor::addDefineByLine(const QString &line, bool hardCoded)
|
|
{
|
|
// Remove define
|
|
constexpr int DEFINE_LEN=6;
|
|
QString s = line.mid(DEFINE_LEN).trimmed();
|
|
|
|
QString name, args, value;
|
|
// Get parts from generalized function
|
|
getDefineParts(s, name, args, value);
|
|
|
|
// Add to the list
|
|
addDefineByParts(name, args, value, hardCoded);
|
|
}
|
|
|
|
void CppPreprocessor::setScanOptions(bool parseSystem, bool parseLocal)
|
|
{
|
|
mParseSystem = parseSystem;
|
|
mParseLocal=parseLocal;
|
|
}
|
|
|
|
void CppPreprocessor::preprocess(const QString &fileName)
|
|
{
|
|
clearTempResults();
|
|
mFileName = fileName;
|
|
//mDefines = mHardDefines;
|
|
openInclude(fileName);
|
|
// StringsToFile(mBuffer,"f:\\buffer.txt");
|
|
preprocessBuffer();
|
|
// StringsToFile(mBuffer,"f:\\buffer.txt");
|
|
// StringsToFile(mResult,"f:\\log.txt");
|
|
}
|
|
|
|
void CppPreprocessor::invalidDefinesInFile(const QString &fileName)
|
|
{
|
|
//remove all defines defined in this file
|
|
PDefineMap defineMap = mFileDefines.value(fileName,PDefineMap());
|
|
if (defineMap) {
|
|
foreach (const PDefine& define, *defineMap) {
|
|
const PDefine& p = mDefines.value(define->name);
|
|
if (p == define) {
|
|
mDefines.remove(define->name);
|
|
PDefine p2 = mHardDefines.value(define->name);
|
|
if (p2)
|
|
mDefines.insert(define->name, p2);
|
|
}
|
|
}
|
|
mFileDefines.remove(fileName);
|
|
}
|
|
//restore all defines undefined in this file
|
|
PDefineMap undefineMap = mFileUndefines.value(fileName,PDefineMap());
|
|
if (undefineMap) {
|
|
foreach (const PDefine& define, *undefineMap) {
|
|
mDefines.insert(define->name, define);
|
|
}
|
|
mFileUndefines.remove(fileName);
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::dumpDefinesTo(const QString &fileName) const
|
|
{
|
|
QFile file(fileName);
|
|
if (file.open(QIODevice::WriteOnly|QIODevice::Truncate)) {
|
|
QTextStream stream(&file);
|
|
for (const PDefine& define:mDefines) {
|
|
stream<<QString("%1 %2 %3 %4 %5\n")
|
|
.arg(define->name,define->args,define->value)
|
|
.arg(define->hardCoded).arg(define->formatValue)
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::dumpIncludesListTo(const QString &fileName) const
|
|
{
|
|
QFile file(fileName);
|
|
if (file.open(QIODevice::WriteOnly|QIODevice::Truncate)) {
|
|
QTextStream stream(&file);
|
|
for (const PFileIncludes& fileIncludes:mIncludesList) {
|
|
stream<<fileIncludes->baseFile<<" : "
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
stream<<"\t**includes:**"
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
foreach (const QString& s,fileIncludes->includeFiles.keys()) {
|
|
stream<<"\t--"+s
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
}
|
|
stream<<"\t**depends on:**"
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
stream<<"\t**depended by:**"
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
<<Qt::endl;
|
|
#else
|
|
<<endl;
|
|
#endif
|
|
// stream<<"\t**using:**"
|
|
// #if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
// <<Qt::endl;
|
|
// #else
|
|
// <<endl;
|
|
// #endif
|
|
// foreach (const QString& s,fileIncludes->usings) {
|
|
// stream<<"\t++"+s
|
|
// #if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
// <<Qt::endl;
|
|
// #else
|
|
// <<endl;
|
|
// #endif
|
|
// }
|
|
// stream<<"\t**statements:**"
|
|
// #if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
// <<Qt::endl;
|
|
// #else
|
|
// <<endl;
|
|
// #endif
|
|
// foreach (const PStatement& statement,fileIncludes->statements) {
|
|
// if (statement) {
|
|
// stream<<QString("\t**%1 , %2")
|
|
// .arg(statement->command,statement->fullName)
|
|
// #if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
// <<Qt::endl;
|
|
// #else
|
|
// <<endl;
|
|
// #endif
|
|
// }
|
|
// }
|
|
}
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::addIncludePath(const QString &fileName)
|
|
{
|
|
if (!mIncludePaths.contains(fileName)) {
|
|
mIncludePaths.insert(fileName);
|
|
mIncludePathList.append(fileName);
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::addProjectIncludePath(const QString &fileName)
|
|
{
|
|
if (!mProjectIncludePaths.contains(fileName)) {
|
|
mProjectIncludePaths.insert(fileName);
|
|
mProjectIncludePathList.append(fileName);
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::clearIncludePaths()
|
|
{
|
|
mIncludePaths.clear();
|
|
mIncludePathList.clear();
|
|
}
|
|
|
|
void CppPreprocessor::clearProjectIncludePaths()
|
|
{
|
|
mProjectIncludePaths.clear();
|
|
mProjectIncludePathList.clear();
|
|
}
|
|
|
|
void CppPreprocessor::removeScannedFile(const QString &filename)
|
|
{
|
|
invalidDefinesInFile(filename);
|
|
mScannedFiles.remove(filename);
|
|
mIncludesList.remove(filename);
|
|
mFileDefines.remove(filename);
|
|
mFileUndefines.remove(filename);
|
|
}
|
|
|
|
QString CppPreprocessor::getNextPreprocessor()
|
|
{
|
|
skipToPreprocessor(); // skip until # at start of line
|
|
int preProcFrom = mIndex;
|
|
if (preProcFrom >= mBuffer.count())
|
|
return ""; // we've gone past the final #preprocessor line. Yay
|
|
skipToEndOfPreprocessor();
|
|
int preProcTo = mIndex;
|
|
|
|
// Calculate index to insert defines in in result file
|
|
mPreProcIndex = (mResult.count() - 1) + 1; // offset by one for #include rootfile
|
|
|
|
// Assemble whole line, convert newlines to space
|
|
QString result;
|
|
for (int i=preProcFrom;i<=preProcTo;i++) {
|
|
if (mBuffer[i].endsWith('\\')) {
|
|
result+=mBuffer[i].left(mBuffer[i].size()-1)+' ';
|
|
} else {
|
|
result+=mBuffer[i]+' ';
|
|
}
|
|
mResult.append("");// defines resolve into empty files, except #define and #include
|
|
}
|
|
// Step over
|
|
mIndex++;
|
|
return result;
|
|
}
|
|
|
|
void CppPreprocessor::handleBranch(const QString &line)
|
|
{
|
|
if (line.startsWith("ifdef")) {
|
|
// // if a branch that is not at our level is false, current branch is false too;
|
|
// for (int i=0;i<=mBranchResults.count()-2;i++) {
|
|
// if (!mBranchResults[i]) {
|
|
// setCurrentBranch(false);
|
|
// return;
|
|
// }
|
|
// }
|
|
if (getCurrentBranch()!=BranchResult::isTrue) {
|
|
setCurrentBranch(BranchResult::parentIsFalse);
|
|
} else {
|
|
constexpr int IFDEF_LEN = 5; //length of ifdef;
|
|
QString name = line.mid(IFDEF_LEN).trimmed();
|
|
setCurrentBranch( getDefine(name)!=nullptr?(BranchResult::isTrue):(BranchResult::isFalse) );
|
|
|
|
}
|
|
} else if (line.startsWith("ifndef")) {
|
|
// // if a branch that is not at our level is false, current branch is false too;
|
|
// for (int i=0;i<=mBranchResults.count()-2;i++) {
|
|
// if (!mBranchResults[i]) {
|
|
// setCurrentBranch(false);
|
|
// return;
|
|
// }
|
|
// }
|
|
if (getCurrentBranch()!=BranchResult::isTrue) {
|
|
setCurrentBranch(BranchResult::parentIsFalse);
|
|
} else {
|
|
constexpr int IFNDEF_LEN = 6; //length of ifndef;
|
|
QString name = line.mid(IFNDEF_LEN).trimmed();
|
|
setCurrentBranch( getDefine(name)==nullptr?(BranchResult::isTrue):(BranchResult::isFalse) );
|
|
}
|
|
} else if (line.startsWith("if")) {
|
|
// // if a branch that is not at our level is false, current branch is false too;
|
|
// for (int i=0;i<=mBranchResults.count()-2;i++) {
|
|
// if (!mBranchResults[i]) {
|
|
// setCurrentBranch(false);
|
|
// return;
|
|
// }
|
|
// }
|
|
if (getCurrentBranch()!=BranchResult::isTrue) {// we are already inside an if that is NOT being taken
|
|
setCurrentBranch(BranchResult::parentIsFalse);// so don't take this one either
|
|
} else {
|
|
constexpr int IF_LEN = 2; //length of if;
|
|
QString ifLine = line.mid(IF_LEN).trimmed();
|
|
|
|
bool testResult = evaluateIf(ifLine);
|
|
setCurrentBranch(testResult?(BranchResult::isTrue):(BranchResult::isFalse));
|
|
}
|
|
} else if (line.startsWith("else")) {
|
|
BranchResult oldResult = getCurrentBranch(); // take either if or else
|
|
removeCurrentBranch();
|
|
setCurrentBranch(calcElseBranchResult(oldResult));
|
|
} else if (line.startsWith("elif")) {
|
|
BranchResult oldResult = getCurrentBranch(); // take either if or else
|
|
removeCurrentBranch();
|
|
BranchResult elseResult = calcElseBranchResult(oldResult);
|
|
if (elseResult == BranchResult::isTrue) { // don't take this one, if previous has been taken
|
|
constexpr int ELIF_LEN = 4; //length of if;
|
|
QString ifLine = line.mid(ELIF_LEN).trimmed();
|
|
bool testResult = evaluateIf(ifLine);
|
|
setCurrentBranch(testResult?(BranchResult::isTrue):(BranchResult::isFalse));
|
|
} else {
|
|
setCurrentBranch(elseResult);
|
|
}
|
|
} else if (line.startsWith("endif")) {
|
|
removeCurrentBranch();
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::handleDefine(const QString &line)
|
|
{
|
|
if (getCurrentBranch() == BranchResult::isTrue) {
|
|
addDefineByLine(line, false);
|
|
mResult[mPreProcIndex] = '#' + line; // add define to result file so the parser can handle it
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::handleInclude(const QString &line, bool fromNext)
|
|
{
|
|
if (getCurrentBranch()!=BranchResult::isTrue) // we're skipping due to a branch failure
|
|
return;
|
|
|
|
PParsedFile file = mIncludes.back();
|
|
QString fileName;
|
|
// Get full header file name
|
|
QString currentDir = includeTrailingPathDelimiter(extractFileDir(file->fileName));
|
|
QStringList includes;
|
|
QStringList projectIncludes;
|
|
bool found=false;
|
|
if (fromNext && mIncludePaths.contains(currentDir)) {
|
|
foreach(const QString& s, mIncludePathList) {
|
|
if (found) {
|
|
includes.append(s);
|
|
continue;
|
|
} else if (s == currentDir)
|
|
found = true;
|
|
}
|
|
projectIncludes = mProjectIncludePathList;
|
|
} else if (fromNext && mProjectIncludePaths.contains(currentDir)) {
|
|
includes = mIncludePathList;
|
|
foreach(const QString& s, mProjectIncludePathList) {
|
|
if (found) {
|
|
includes.append(s);
|
|
continue;
|
|
} else if (s == currentDir)
|
|
found = true;
|
|
}
|
|
} else {
|
|
includes = mIncludePathList;
|
|
projectIncludes = mProjectIncludePathList;
|
|
}
|
|
fileName = getHeaderFilename(
|
|
file->fileName,
|
|
line,
|
|
includes,
|
|
projectIncludes);
|
|
|
|
if (fileName.isEmpty())
|
|
return;
|
|
|
|
PFileIncludes oldCurrentIncludes = mCurrentIncludes;
|
|
openInclude(fileName);
|
|
}
|
|
|
|
void CppPreprocessor::handlePreprocessor(const QString &value)
|
|
{
|
|
switch(value[0].unicode()) {
|
|
case 'd':
|
|
if (value.startsWith("define"))
|
|
handleDefine(value);
|
|
break;
|
|
case 'e':
|
|
if (value.startsWith("else") || value.startsWith("elif")
|
|
|| value.startsWith("endif"))
|
|
handleBranch(value);
|
|
break;
|
|
case 'i':
|
|
if (value.startsWith("if"))
|
|
handleBranch(value);
|
|
else if (value.startsWith("include"))
|
|
handleInclude(value, value.startsWith("include_next"));
|
|
break;
|
|
case 'u':
|
|
if (value.startsWith("undef"))
|
|
handleUndefine(value);
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::handleUndefine(const QString &line)
|
|
{
|
|
// Remove undef
|
|
constexpr int UNDEF_LEN = 5;
|
|
QString name = line.mid(UNDEF_LEN).trimmed();
|
|
|
|
// //may be defined many times
|
|
// while (true) {
|
|
PDefine define = getDefine(name);
|
|
if (define) {
|
|
//remove the define from defines set
|
|
mDefines.remove(name);
|
|
if (define->filename == mFileName) {
|
|
//remove the define form the file where it defines
|
|
PDefineMap defineMap = mFileDefines.value(mFileName);
|
|
if (defineMap) {
|
|
defineMap->remove(name);
|
|
}
|
|
} else {
|
|
// add it to undefine map
|
|
PDefineMap undefineMap = mFileUndefines.value(mFileName);
|
|
if (!undefineMap) {
|
|
undefineMap = std::make_shared<DefineMap>();
|
|
mFileUndefines.insert(mFileName,undefineMap);
|
|
}
|
|
if (!undefineMap->contains(name))
|
|
undefineMap->insert(name, define);
|
|
}
|
|
}
|
|
}
|
|
|
|
QString CppPreprocessor::expandMacros(const QString &text, QSet<QString> usedMacros) const
|
|
{
|
|
QString word;
|
|
QString newLine;
|
|
int lenLine = text.length();
|
|
int i=0;
|
|
while (i< lenLine) {
|
|
QChar ch=text[i];
|
|
if (isWordChar(ch)) {
|
|
word += ch;
|
|
} else {
|
|
if (!word.isEmpty()) {
|
|
expandMacro(text,newLine,word,i,usedMacros);
|
|
}
|
|
word = "";
|
|
if (i< lenLine) {
|
|
newLine += text[i];
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
if (!word.isEmpty()) {
|
|
expandMacro(text,newLine,word,i, usedMacros);
|
|
}
|
|
return newLine;
|
|
}
|
|
|
|
QString CppPreprocessor::expandMacros()
|
|
{
|
|
//prevent infinit recursion
|
|
QString word;
|
|
QString newLine;
|
|
QString line = mBuffer[mIndex];
|
|
QSet<QString> usedMacros;
|
|
int i=0;
|
|
while (mIndex < mBuffer.size() && i<line.length()) {
|
|
QChar ch=line[i];
|
|
if (isWordChar(ch)) {
|
|
word += ch;
|
|
} else {
|
|
if (!word.isEmpty()) {
|
|
expandMacro(newLine,word,i,usedMacros);
|
|
if (mIndex>=mBuffer.length())
|
|
return newLine;
|
|
line = mBuffer[mIndex];
|
|
}
|
|
word = "";
|
|
if (i< line.length()) {
|
|
newLine += line[i];
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
if (!word.isEmpty()) {
|
|
expandMacro(newLine,word,i,usedMacros);
|
|
}
|
|
return newLine;
|
|
}
|
|
|
|
void CppPreprocessor::expandMacro(const QString &text, QString &newText, const QString &word, int &i, QSet<QString> usedMacros) const
|
|
{
|
|
if (usedMacros.contains(word))
|
|
return;
|
|
int lenLine = text.length();
|
|
PDefine define = getDefine(word);
|
|
if (define && define->args=="" ) {
|
|
usedMacros.insert(word);
|
|
if (define->value != word ) {
|
|
newText += expandMacros(define->value,usedMacros);
|
|
} else
|
|
newText += word;
|
|
} else if (define && (define->args!="")) {
|
|
while ((i<lenLine) && (text[i] == ' ' || text[i]=='\t'))
|
|
i++;
|
|
int argStart=-1;
|
|
int argEnd=-1;
|
|
if ((i<lenLine) && (text[i]=='(')) {
|
|
argStart =i+1;
|
|
int level=0;
|
|
bool inString=false;
|
|
while (i<lenLine) {
|
|
switch(text[i].unicode()) {
|
|
case '\\':
|
|
if (inString)
|
|
i++;
|
|
break;
|
|
case '"':
|
|
inString = !inString;
|
|
break;
|
|
case '(':
|
|
if (!inString)
|
|
level++;
|
|
break;
|
|
case ')':
|
|
if (!inString)
|
|
level--;
|
|
}
|
|
i++;
|
|
if (level==0)
|
|
break;
|
|
}
|
|
if (level==0) {
|
|
argEnd = i-2;
|
|
QString args = text.mid(argStart,argEnd-argStart+1).trimmed();
|
|
QString formattedValue = expandFunction(define,args);
|
|
usedMacros.insert(word);
|
|
newText += expandMacros(formattedValue,usedMacros);
|
|
}
|
|
}
|
|
} else {
|
|
newText += word;
|
|
}
|
|
// }
|
|
}
|
|
|
|
void CppPreprocessor::expandMacro(QString &newLine, const QString &word, int &i, QSet<QString> usedMacros)
|
|
{
|
|
if (usedMacros.contains(word))
|
|
return;
|
|
QString line = mBuffer[mIndex];
|
|
PDefine define = getDefine(word);
|
|
if (define && define->args=="" ) {
|
|
usedMacros.insert(word);
|
|
if (define->value != word )
|
|
newLine += expandMacros(define->value,usedMacros);
|
|
else
|
|
newLine += word;
|
|
} else if (define && (define->args!="")) {
|
|
int origI=i;
|
|
int origIndex=mIndex;
|
|
while(true) {
|
|
while ((i<line.length()) && (line[i] == ' ' || line[i]=='\t'))
|
|
i++;
|
|
if (i<line.length())
|
|
break;
|
|
mIndex++;
|
|
if (mIndex>=mBuffer.length())
|
|
return;
|
|
line = mBuffer[mIndex];
|
|
i=0;
|
|
}
|
|
int argStart=-1;
|
|
int argEnd=-1;
|
|
int argLineStart=mIndex;
|
|
int argLineEnd=mIndex;
|
|
if ((i<line.length()) && (line[i]=='(')) {
|
|
argStart =i+1;
|
|
int level=0;
|
|
bool inString=false;
|
|
while (true) {
|
|
while (i<line.length()) {
|
|
switch(line[i].unicode()) {
|
|
case '\\':
|
|
if (inString)
|
|
i++;
|
|
break;
|
|
case '"':
|
|
inString = !inString;
|
|
break;
|
|
case '(':
|
|
if (!inString)
|
|
level++;
|
|
break;
|
|
case ')':
|
|
if (!inString)
|
|
level--;
|
|
break;
|
|
}
|
|
i++;
|
|
if (level==0)
|
|
break;
|
|
}
|
|
if (level==0)
|
|
break;
|
|
mIndex++;
|
|
i=0;
|
|
if (mIndex>=mBuffer.length())
|
|
break;
|
|
line = mBuffer[mIndex];
|
|
if (!inString && line.startsWith('#')) {
|
|
break;
|
|
}
|
|
} ;
|
|
if (level==0) {
|
|
argEnd = i-1;
|
|
argLineEnd = mIndex;
|
|
QString args;
|
|
if (argLineStart==argLineEnd) {
|
|
args = line.mid(argStart,argEnd-argStart).trimmed();
|
|
//qDebug()<<"--"<<args;
|
|
} else {
|
|
args = mBuffer[argLineStart].mid(argStart);
|
|
for (int i=argLineStart+1;i<argLineEnd;i++) {
|
|
args += mBuffer[i];
|
|
}
|
|
args += mBuffer[argLineEnd].left(argEnd);
|
|
}
|
|
QString formattedValue = expandFunction(define,args);
|
|
usedMacros.insert(word);
|
|
newLine += expandMacros(formattedValue,usedMacros);
|
|
}
|
|
} else {
|
|
newLine+=word;
|
|
i=origI;
|
|
mIndex=origIndex;
|
|
}
|
|
} else {
|
|
newLine += word;
|
|
}
|
|
}
|
|
|
|
QString CppPreprocessor::removeGCCAttributes(const QString &line)
|
|
{
|
|
QString newLine = "";
|
|
QString word = "";
|
|
int lenLine = line.length();
|
|
int i=0;
|
|
while(i< lenLine) {
|
|
if (isWordChar(line[i])) {
|
|
word += line[i];
|
|
} else {
|
|
if (!word.isEmpty()) {
|
|
removeGCCAttribute(line,newLine,i,word);
|
|
}
|
|
word = "";
|
|
if (i<lenLine) {
|
|
newLine = newLine+line[i];
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
if (!word.isEmpty())
|
|
removeGCCAttribute(line,newLine,i,word);
|
|
return newLine;
|
|
}
|
|
|
|
void CppPreprocessor::removeGCCAttribute(const QString &line, QString &newLine, int &i, const QString &word)
|
|
{
|
|
int lenLine = line.length();
|
|
int level = 0;
|
|
if (word=="__attribute__") {
|
|
while ( (i<lenLine) && isSpaceChar(line[i]))
|
|
i++;
|
|
if ((i<lenLine) && (line[i]=='(')) {
|
|
level=0;
|
|
while (i<lenLine) {
|
|
switch(line[i].unicode()) {
|
|
case '(': level++;
|
|
break;
|
|
case ')': level--;
|
|
break;
|
|
}
|
|
i++;
|
|
if (level==0)
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
newLine += word;
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::openInclude(QString fileName)
|
|
{
|
|
fileName.squeeze();
|
|
|
|
PFileIncludes fileIncludes = getFileIncludesEntry(fileName);
|
|
if (fileIncludes) {
|
|
fileName = fileIncludes->baseFile;
|
|
} else {
|
|
fileName.squeeze();
|
|
}
|
|
|
|
if (mIncludes.size()>0) {
|
|
PParsedFile topFile = mIncludes.front();
|
|
if (topFile->fileIncludes->includeFiles.contains(fileName)) {
|
|
return; //already included
|
|
}
|
|
for (PParsedFile& parsedFile:mIncludes) {
|
|
parsedFile->fileIncludes->includeFiles.insert(fileName,false);
|
|
}
|
|
// Backup old position if we're entering a new file
|
|
PParsedFile innerMostFile = mIncludes.back();
|
|
innerMostFile->index = mIndex;
|
|
innerMostFile->branches = mBranchResults.count();
|
|
|
|
innerMostFile->fileIncludes->includeFiles.insert(fileName,true);
|
|
innerMostFile->fileIncludes->directIncludes.append(fileName);
|
|
}
|
|
|
|
// // Add the new file to the includes of the current file
|
|
// // Only add items to the include list of the given file if the file hasn't been scanned yet
|
|
// // The above is fixed by checking for duplicates.
|
|
// // The proper way would be to do backtracking of files we have FINISHED scanned.
|
|
// // These are not the same files as the ones in fScannedFiles. We have STARTED scanning these.
|
|
// {
|
|
// if Assigned(fCurrentIncludes) then
|
|
// with fCurrentIncludes^ do
|
|
// if not ContainsText(IncludeFiles, FileName) then
|
|
// IncludeFiles := IncludeFiles + AnsiQuotedStr(FileName, '"') + ',';
|
|
// }
|
|
|
|
// Create and add new buffer/position
|
|
PParsedFile parsedFile = std::make_shared<ParsedFile>();
|
|
parsedFile->index = 0;
|
|
parsedFile->fileName = fileName;
|
|
parsedFile->branches = 0;
|
|
// parsedFile->buffer; it's auto initialized
|
|
|
|
|
|
// Keep track of files we include here
|
|
// Only create new items for files we have NOT scanned yet
|
|
mCurrentIncludes = getFileIncludesEntry(fileName);
|
|
if (!mCurrentIncludes) {
|
|
// do NOT create a new item for a file that's already in the list
|
|
mCurrentIncludes = std::make_shared<FileIncludes>();
|
|
mCurrentIncludes->baseFile = fileName;
|
|
mIncludesList.insert(fileName,mCurrentIncludes);
|
|
}
|
|
parsedFile->fileIncludes = mCurrentIncludes;
|
|
|
|
// Don't parse stuff we have already parsed
|
|
if (!mScannedFiles.contains(fileName)) {
|
|
// Parse ONCE
|
|
//if not Assigned(Stream) then
|
|
mScannedFiles.insert(fileName);
|
|
|
|
// Only load up the file if we are allowed to parse it
|
|
bool isSystemFile = isSystemHeaderFile(fileName, mIncludePaths) || isSystemHeaderFile(fileName, mProjectIncludePaths);
|
|
if ((mParseSystem && isSystemFile) || (mParseLocal && !isSystemFile)) {
|
|
QStringList bufferedText;
|
|
if (mOnGetFileStream && mOnGetFileStream(fileName,bufferedText)) {
|
|
parsedFile->buffer = bufferedText;
|
|
} else {
|
|
parsedFile->buffer = readFileToLines(fileName);
|
|
}
|
|
}
|
|
} else {
|
|
//add defines of already parsed including headers;
|
|
addDefinesInFile(fileName);
|
|
PFileIncludes fileIncludes = getFileIncludesEntry(fileName);
|
|
if (fileIncludes) {
|
|
for (PParsedFile& file:mIncludes) {
|
|
foreach (const QString& incFile,fileIncludes->includeFiles.keys()) {
|
|
file->fileIncludes->includeFiles.insert(incFile,false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
mIncludes.append(parsedFile);
|
|
|
|
// Process it
|
|
mIndex = parsedFile->index;
|
|
mFileName = parsedFile->fileName;
|
|
parsedFile->buffer = removeComments(parsedFile->buffer);
|
|
mBuffer = parsedFile->buffer;
|
|
|
|
// for (int i=0;i<mBuffer.count();i++) {
|
|
// mBuffer[i] = mBuffer[i].trimmed();
|
|
// }
|
|
|
|
// Update result file
|
|
QString includeLine = "#include " + fileName + ":1";
|
|
if (mIncludes.count()>1) { // include from within a file
|
|
mResult[mPreProcIndex] = includeLine;
|
|
} else {
|
|
mResult.append(includeLine);
|
|
}
|
|
}
|
|
|
|
|
|
void CppPreprocessor::closeInclude()
|
|
{
|
|
if (mIncludes.isEmpty())
|
|
return;
|
|
mIncludes.pop_back();
|
|
|
|
if (mIncludes.isEmpty())
|
|
return;
|
|
PParsedFile parsedFile = mIncludes.back();
|
|
|
|
// Continue where we left off
|
|
mIndex = parsedFile->index;
|
|
mFileName = parsedFile->fileName;
|
|
// Point to previous buffer and start past the include we walked into
|
|
mBuffer = parsedFile->buffer;
|
|
while (mBranchResults.count()>parsedFile->branches) {
|
|
mBranchResults.pop_back();
|
|
}
|
|
|
|
|
|
// Start augmenting previous include list again
|
|
//fCurrentIncludes := GetFileIncludesEntry(fFileName);
|
|
mCurrentIncludes = parsedFile->fileIncludes;
|
|
|
|
// Update result file (we've left the previous file)
|
|
mResult.append(
|
|
QString("#include %1:%2").arg(parsedFile->fileName)
|
|
.arg(parsedFile->index+1));
|
|
}
|
|
|
|
CppPreprocessor::BranchResult CppPreprocessor::calcElseBranchResult(BranchResult oldResult)
|
|
{
|
|
switch(oldResult) {
|
|
case BranchResult::isTrue: return BranchResult::isFalse_but_trued;
|
|
case BranchResult::isFalse: return BranchResult::isTrue;
|
|
case BranchResult::isFalse_but_trued: return BranchResult::isFalse_but_trued;
|
|
case BranchResult::parentIsFalse: return BranchResult::parentIsFalse;
|
|
}
|
|
Q_ASSERT( false ); //We should fail here.
|
|
return BranchResult::isFalse;
|
|
}
|
|
|
|
void CppPreprocessor::addDefinesInFile(const QString &fileName)
|
|
{
|
|
if (mProcessed.contains(fileName))
|
|
return;
|
|
mProcessed.insert(fileName);
|
|
|
|
// then add the defines defined in it
|
|
PDefineMap defineList = mFileDefines.value(fileName, PDefineMap());
|
|
if (defineList) {
|
|
foreach (const PDefine& define, defineList->values()) {
|
|
mDefines.insert(define->name,define);
|
|
}
|
|
}
|
|
|
|
PFileIncludes fileIncludes = getFileIncludesEntry(fileName);
|
|
if (fileIncludes) {
|
|
foreach (const QString& file, fileIncludes->includeFiles.keys()) {
|
|
addDefinesInFile(file);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::parseArgs(PDefine define)
|
|
{
|
|
QString args=define->args.mid(1,define->args.length()-2).trimmed(); // remove '(' ')'
|
|
|
|
if(args=="")
|
|
return ;
|
|
QStringList argList = args.split(',');
|
|
for (int i=0;i<argList.size();i++) {
|
|
argList[i]=argList[i].trimmed();
|
|
define->argUsed.append(false);
|
|
}
|
|
QList<PDefineArgToken> tokens = tokenizeValue(define->value);
|
|
|
|
QString formatStr = "";
|
|
DefineArgTokenType lastTokenType=DefineArgTokenType::Other;
|
|
int index;
|
|
foreach (const PDefineArgToken& token, tokens) {
|
|
switch(token->type) {
|
|
case DefineArgTokenType::Identifier:
|
|
if (token->value == "__VA_ARGS__") {
|
|
index = argList.indexOf("...");
|
|
define->varArgIndex = index;
|
|
} else {
|
|
index = argList.indexOf(token->value);
|
|
}
|
|
if (index>=0) {
|
|
define->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;
|
|
}
|
|
define->formatValue = formatStr;
|
|
define->formatValue.squeeze();
|
|
}
|
|
|
|
QList<PDefineArgToken> CppPreprocessor::tokenizeValue(const QString &value)
|
|
{
|
|
int i=0;
|
|
PDefineArgToken token = std::make_shared<DefineArgToken>();
|
|
token->type = DefineArgTokenType::Other;
|
|
QList<PDefineArgToken> tokens;
|
|
bool skipSpaces=false;
|
|
while (i<value.length()) {
|
|
QChar ch = value[i];
|
|
if (isSpaceChar(ch)) {
|
|
if (token->type==DefineArgTokenType::Other) {
|
|
token->value = " ";
|
|
token->type = DefineArgTokenType::Space;
|
|
} else if (token->type!=DefineArgTokenType::Space) {
|
|
tokens.append(token);
|
|
token = std::make_shared<DefineArgToken>();
|
|
token->value = " ";
|
|
token->type = DefineArgTokenType::Space;
|
|
}
|
|
i++;
|
|
} else if (ch=='#') {
|
|
if (token->type!=DefineArgTokenType::Other
|
|
&& token->type!=DefineArgTokenType::Space) {
|
|
tokens.append(token);
|
|
token = std::make_shared<DefineArgToken>();
|
|
}
|
|
if ((i+1<value.length()) && (value[i+1]=='#')) {
|
|
i+=2;
|
|
token->value = "##";
|
|
token->type = DefineArgTokenType::DSharp;
|
|
} else {
|
|
i++;
|
|
token->value = "#";
|
|
token->type = DefineArgTokenType::Sharp;
|
|
}
|
|
skipSpaces=true;
|
|
tokens.append(token);
|
|
token = std::make_shared<DefineArgToken>();
|
|
token->value = "";
|
|
token->type = DefineArgTokenType::Other;
|
|
} else if (isWordChar(ch)) {
|
|
if (token->type==DefineArgTokenType::Other) {
|
|
token->value = ch ;
|
|
token->type = DefineArgTokenType::Identifier;
|
|
} else if (token->type==DefineArgTokenType::Identifier) {
|
|
token->value+=ch;
|
|
} else if (skipSpaces && token->type==DefineArgTokenType::Space) {
|
|
//dont use space;
|
|
token->value = ch ;
|
|
token->type = DefineArgTokenType::Identifier;
|
|
} else {
|
|
tokens.append(token);
|
|
token = std::make_shared<DefineArgToken>();
|
|
token->value = ch ;
|
|
token->type = DefineArgTokenType::Identifier;
|
|
}
|
|
skipSpaces=false;
|
|
i++;
|
|
} else {
|
|
if (skipSpaces && token->type==DefineArgTokenType::Space) {
|
|
//dont use space;
|
|
} else if (token->type!=DefineArgTokenType::Other) {
|
|
tokens.append(token);
|
|
token = std::make_shared<DefineArgToken>();
|
|
}
|
|
skipSpaces=false;
|
|
token->value = ch ;
|
|
token->type = DefineArgTokenType::Symbol;
|
|
i++;
|
|
}
|
|
}
|
|
if(token->type!=DefineArgTokenType::Other)
|
|
tokens.append(token);
|
|
return tokens;
|
|
}
|
|
|
|
QStringList CppPreprocessor::removeComments(const QStringList &text)
|
|
{
|
|
QStringList result;
|
|
ContentType currentType = ContentType::Other;
|
|
QString delimiter;
|
|
|
|
for (const QString& line:text) {
|
|
QString s;
|
|
int pos = 0;
|
|
bool stopProcess=false;
|
|
int lineLen=line.length();
|
|
s.reserve(line.length());
|
|
while (pos<lineLen) {
|
|
QChar ch =line[pos];
|
|
if (currentType == ContentType::AnsiCComment) {
|
|
if (ch=='*' && (pos+1<lineLen) && line[pos+1]=='/') {
|
|
pos+=2;
|
|
currentType = ContentType::Other;
|
|
} else {
|
|
pos+=1;
|
|
}
|
|
continue;
|
|
}
|
|
switch (ch.unicode()) {
|
|
case '"':
|
|
switch (currentType) {
|
|
case ContentType::String:
|
|
currentType=ContentType::Other;
|
|
break;
|
|
case ContentType::RawString:
|
|
if (line.midRef(0,pos).endsWith(')'+delimiter))
|
|
currentType = ContentType::Other;
|
|
break;
|
|
case ContentType::Other:
|
|
currentType=ContentType::String;
|
|
break;
|
|
case ContentType::RawStringPrefix:
|
|
delimiter+=ch;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
s+=ch;
|
|
break;
|
|
case '\'':
|
|
switch (currentType) {
|
|
case ContentType::Character:
|
|
currentType=ContentType::Other;
|
|
break;
|
|
case ContentType::Other:
|
|
currentType=ContentType::Character;
|
|
break;
|
|
case ContentType::RawStringPrefix:
|
|
delimiter+=ch;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
s+=ch;
|
|
break;
|
|
case 'R':
|
|
if (currentType == ContentType::Other && pos+1<lineLen && line[pos+1]=='"') {
|
|
s+=ch;
|
|
pos++;
|
|
ch = line[pos];
|
|
currentType=ContentType::RawStringPrefix;
|
|
delimiter = "";
|
|
}
|
|
if (currentType == ContentType::RawStringPrefix ) {
|
|
delimiter += ch;
|
|
}
|
|
s+=ch;
|
|
break;
|
|
case '(':
|
|
switch(currentType) {
|
|
case ContentType::RawStringPrefix:
|
|
currentType = ContentType::RawString;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
s+=ch;
|
|
break;
|
|
case '/':
|
|
if (currentType == ContentType::Other) {
|
|
if (pos+1<lineLen && line[pos+1]=='/') {
|
|
// line comment , skip all remainings of the current line
|
|
stopProcess = true;
|
|
break;
|
|
} else if (pos+1<lineLen && line[pos+1]=='*') {
|
|
/* ansi c comment */
|
|
pos++;
|
|
currentType = ContentType::AnsiCComment;
|
|
break;
|
|
}
|
|
}
|
|
s+=ch;
|
|
break;
|
|
case '\\':
|
|
switch (currentType) {
|
|
case ContentType::String:
|
|
case ContentType::Character:
|
|
pos++;
|
|
s+=ch;
|
|
if (pos<lineLen) {
|
|
ch = line[pos];
|
|
s+=ch;
|
|
}
|
|
break;
|
|
default:
|
|
s+=ch;
|
|
}
|
|
break;
|
|
default:
|
|
s+=ch;
|
|
}
|
|
if (stopProcess)
|
|
break;
|
|
pos++;
|
|
}
|
|
result.append(s.trimmed());
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void CppPreprocessor::preprocessBuffer()
|
|
{
|
|
while (mIncludes.count() > 0) {
|
|
QString s;
|
|
do {
|
|
s = getNextPreprocessor();
|
|
if (s.startsWith('#')) {
|
|
s = s.mid(1).trimmed(); // remove #
|
|
if (!s.isEmpty()) {
|
|
handlePreprocessor(s);
|
|
}
|
|
}
|
|
} while (!s.isEmpty());
|
|
closeInclude();
|
|
}
|
|
}
|
|
|
|
void CppPreprocessor::skipToEndOfPreprocessor()
|
|
{
|
|
int indexLimit = mBuffer.count()-1;
|
|
// Skip until last char of line is NOT \ anymore
|
|
while ((mIndex < indexLimit) && mBuffer[mIndex].endsWith('\\'))
|
|
mIndex++;
|
|
}
|
|
|
|
void CppPreprocessor::skipToPreprocessor()
|
|
{
|
|
int bufferCount = mBuffer.count();
|
|
// Increment until a line begins with a #
|
|
while ((mIndex < bufferCount) && !mBuffer[mIndex].startsWith('#')) {
|
|
if (getCurrentBranch()==BranchResult::isTrue) { // if not skipping, expand current macros
|
|
int startIndex = mIndex;
|
|
QString expanded = expandMacros();
|
|
mResult.append(expanded);
|
|
for (int i=startIndex;i<mIndex;i++) {
|
|
mResult.append("");
|
|
}
|
|
//mResult.append(expandMacros(mBuffer[mIndex],1));
|
|
} else // If skipping due to a failed branch, clear line
|
|
mResult.append("");
|
|
mIndex++;
|
|
}
|
|
}
|
|
|
|
bool CppPreprocessor::isWordChar(const QChar &ch)
|
|
{
|
|
if (ch=='_'
|
|
// || (ch>='a' && ch<='z') || (ch>='A' && ch<='Z')
|
|
|| ch.isLetter()
|
|
|| (ch>='0' && ch<='9')) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CppPreprocessor::isIdentChar(const QChar &ch)
|
|
{
|
|
if (ch=='_' || ch == '*' || ch == '&' || ch == '~' ||
|
|
ch.isLetter()
|
|
//(ch>='a' && ch<='z') || (ch>='A' && ch<='Z')
|
|
|| (ch>='0' && ch<='9')) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CppPreprocessor::isLineChar(const QChar &ch)
|
|
{
|
|
return ch=='\r' || ch == '\n';
|
|
}
|
|
|
|
bool CppPreprocessor::isSpaceChar(const QChar &ch)
|
|
{
|
|
return ch == ' ' || ch == '\t';
|
|
}
|
|
|
|
//bool CppPreprocessor::isOperatorChar(const QChar &ch)
|
|
//{
|
|
|
|
// switch(ch.unicode()) {
|
|
// case '+':
|
|
// case '-':
|
|
// case '*':
|
|
// case '/':
|
|
// case '!':
|
|
// case '=':
|
|
// case '<':
|
|
// case '>':
|
|
// case '&':
|
|
// case '|':
|
|
// case '^':
|
|
// return true;
|
|
// default:
|
|
// return false;
|
|
// }
|
|
//}
|
|
|
|
bool CppPreprocessor::isMacroIdentChar(const QChar &ch)
|
|
{
|
|
//return (ch>='A' && ch<='Z') || (ch>='a' && ch<='z')
|
|
return ch.isLetter()
|
|
|| ch == '_';
|
|
}
|
|
|
|
bool CppPreprocessor::isDigit(const QChar &ch)
|
|
{
|
|
return (ch>='0' && ch<='9');
|
|
}
|
|
|
|
bool CppPreprocessor::isNumberChar(const QChar &ch)
|
|
{
|
|
if (ch>='0' && ch<='9')
|
|
return true;
|
|
if (ch>='a' && ch<='f')
|
|
return true;
|
|
if (ch>='A' && ch<='F')
|
|
return true;
|
|
switch(ch.unicode()) {
|
|
case 'x':
|
|
case 'X':
|
|
case 'u':
|
|
case 'U':
|
|
case 'l':
|
|
case 'L':
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
QString CppPreprocessor::lineBreak()
|
|
{
|
|
return "\n";
|
|
}
|
|
|
|
bool CppPreprocessor::evaluateIf(const QString &line)
|
|
{
|
|
QString newLine = expandDefines(line); // replace FOO by numerical value of FOO
|
|
return evaluateExpression(newLine);
|
|
}
|
|
|
|
QString CppPreprocessor::expandDefines(QString line)
|
|
{
|
|
int searchPos = 0;
|
|
while (searchPos < line.length()) {
|
|
// We have found an identifier. It is not a number suffix. Try to expand it
|
|
if (isMacroIdentChar(line[searchPos]) && (
|
|
(searchPos == 0) || !isDigit(line[searchPos - 1]))) {
|
|
int head = searchPos;
|
|
int tail = searchPos;
|
|
|
|
// Get identifier name (numbers are allowed, but not at the start
|
|
while ((tail < line.length()) && (isMacroIdentChar(line[tail]) || isDigit(line[head])))
|
|
tail++;
|
|
// qDebug()<<"1 "<<head<<tail<<line;
|
|
QString name = line.mid(head,tail-head);
|
|
int nameStart = head;
|
|
int nameEnd = tail;
|
|
|
|
if (name == "defined") {
|
|
//expand define
|
|
//tail = searchPos + name.length();
|
|
while ((tail < line.length()) && isSpaceChar(line[tail]))
|
|
tail++; // skip spaces
|
|
int defineStart;
|
|
|
|
// Skip over its arguments
|
|
if ((tail < line.length()) && (line[tail]=='(')) {
|
|
//braced argument (next word)
|
|
defineStart = tail+1;
|
|
if (!skipParenthesis(line, tail)) {
|
|
line = ""; // broken line
|
|
break;
|
|
}
|
|
} else {
|
|
//none braced argument (next word)
|
|
defineStart = tail;
|
|
if ((tail>=line.length()) || !isMacroIdentChar(line[defineStart])) {
|
|
line = ""; // broken line
|
|
break;
|
|
}
|
|
while ((tail < line.length()) && (isMacroIdentChar(line[tail]) || isDigit(line[tail])))
|
|
tail++;
|
|
}
|
|
// qDebug()<<"2 "<<defineStart<<tail<<line;
|
|
name = line.mid(defineStart, tail - defineStart);
|
|
PDefine define = getDefine(name);
|
|
QString insertValue;
|
|
if (!define) {
|
|
insertValue = "0";
|
|
} else {
|
|
insertValue = "1";
|
|
}
|
|
// Insert found value at place
|
|
line.remove(searchPos, tail-searchPos+1);
|
|
line.insert(searchPos,insertValue);
|
|
} else if ((name == "and") || (name == "or")) {
|
|
searchPos = tail; // Skip logical operators
|
|
} else {
|
|
// We have found a regular define. Replace it by its value
|
|
// Does it exist in the database?
|
|
PDefine define = getDefine(name);
|
|
QString insertValue;
|
|
if (!define) {
|
|
insertValue = "0";
|
|
} else {
|
|
while ((tail < line.length()) && isSpaceChar(line[tail]))
|
|
tail++;// skip spaces
|
|
// It is a function. Expand arguments
|
|
if ((tail < line.length()) && (line[tail] == '(')) {
|
|
head=tail;
|
|
if (skipParenthesis(line, tail)) {
|
|
if (name == "__has_builtin") {
|
|
insertValue = "0";
|
|
} else {
|
|
QString args = line.mid(head+1,tail-head-1);
|
|
insertValue = expandFunction(define,args);
|
|
}
|
|
nameEnd = tail+1;
|
|
} else {
|
|
line = "";// broken line
|
|
break;
|
|
}
|
|
// Replace regular define
|
|
} else {
|
|
if (!define->value.isEmpty())
|
|
insertValue = define->value;
|
|
else
|
|
insertValue = "0";
|
|
}
|
|
}
|
|
// Insert found value at place
|
|
line.remove(nameStart, nameEnd - nameStart);
|
|
line.insert(searchPos,insertValue);
|
|
}
|
|
} else {
|
|
searchPos ++ ;
|
|
}
|
|
}
|
|
return line;
|
|
}
|
|
|
|
bool CppPreprocessor::skipParenthesis(const QString &line, int &index, int step)
|
|
{
|
|
int level = 0;
|
|
while ((index >= 0) && (index < line.length())) { // Find the corresponding opening brace
|
|
if (line[index] == '(') {
|
|
level++;
|
|
} else if (line[index] == ')') {
|
|
level--;
|
|
}
|
|
if (level == 0)
|
|
return true;
|
|
index+=step;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString CppPreprocessor::expandFunction(PDefine define, const QString &args)
|
|
{
|
|
// Replace function by this string
|
|
QString result = define->formatValue;
|
|
// if (args.startsWith('(') && args.endsWith(')')) {
|
|
// qDebug()<<define->name<<args;
|
|
// args = args.mid(1,args.length()-2);
|
|
// }
|
|
|
|
if (define->argUsed.length()==0) {
|
|
// do nothing
|
|
} else if (define->argUsed.length()==1) {
|
|
if (define->argUsed[0])
|
|
result=result.arg(args);
|
|
} else {
|
|
QStringList argValues;
|
|
int i=0;
|
|
bool inString = false;
|
|
bool inChar = false;
|
|
int lastSplit=0;
|
|
int level=0;
|
|
while (i<args.length()) {
|
|
switch(args[i].unicode()) {
|
|
case '\\':
|
|
if (inString || inChar)
|
|
i++;
|
|
break;
|
|
case '(':
|
|
case '{':
|
|
if (!inString && !inChar)
|
|
level++;
|
|
break;
|
|
case ')':
|
|
case '}':
|
|
if (!inString && !inChar)
|
|
level--;
|
|
break;
|
|
case '"':
|
|
if (!inChar)
|
|
inString = !inString;
|
|
break;
|
|
case '\'':
|
|
if (!inString)
|
|
inChar = !inChar;
|
|
break;
|
|
case ',':
|
|
if (!inString && !inChar && level == 0) {
|
|
argValues.append(args.mid(lastSplit,i-lastSplit));
|
|
lastSplit=i+1;
|
|
}
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
argValues.append(args.mid(lastSplit,i-lastSplit));
|
|
#ifdef QT_DEBUG
|
|
if (
|
|
(define->varArgIndex==-1 && argValues.length() != define->argUsed.length())
|
|
|| (define->varArgIndex!=-1 && argValues.length() < define->argUsed.length()-1)
|
|
) {
|
|
qDebug()<<"*** Expand Macro error ***";
|
|
qDebug()<<"Macro: "<<define->name<<define->args;
|
|
qDebug()<<"Actual param: "<<args;
|
|
qDebug()<<"Params splitted: "<<argValues;
|
|
qDebug()<<"**********";
|
|
}
|
|
#endif
|
|
if (argValues.length() >= define->argUsed.length()
|
|
&& argValues.length()>0) {
|
|
QStringList varArgs;
|
|
for (int i=0;i<argValues.length();i++) {
|
|
if (define->varArgIndex != -1
|
|
&& i >= define->varArgIndex ) {
|
|
varArgs.append(argValues[i].trimmed());
|
|
} else if (i<define->argUsed.length()
|
|
&& define->argUsed[i]) {
|
|
QString argValue = argValues[i];
|
|
result=result.arg(argValue.trimmed());
|
|
}
|
|
}
|
|
if (!varArgs.isEmpty() && define->varArgIndex != -1) {
|
|
result=result.arg(varArgs.join(","));
|
|
}
|
|
}
|
|
}
|
|
result.replace("%%","%");
|
|
|
|
return result;
|
|
}
|
|
|
|
bool CppPreprocessor::skipSpaces(const QString &expr, int &pos)
|
|
{
|
|
while (pos<expr.length() && isSpaceChar(expr[pos]))
|
|
pos++;
|
|
return pos<expr.length();
|
|
}
|
|
|
|
bool CppPreprocessor::evalNumber(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!skipSpaces(expr,pos))
|
|
return false;
|
|
QString s;
|
|
while (pos<expr.length() && isNumberChar(expr[pos])) {
|
|
s+=expr[pos];
|
|
pos++;
|
|
}
|
|
bool ok;
|
|
|
|
if (s.endsWith("LL",Qt::CaseInsensitive)) {
|
|
s.remove(s.length()-2,2);
|
|
result = s.toLongLong(&ok);
|
|
} else if (s.endsWith("L",Qt::CaseInsensitive)) {
|
|
s.remove(s.length()-1,1);
|
|
result = s.toLong(&ok);
|
|
} else if (s.endsWith("ULL",Qt::CaseInsensitive)) {
|
|
s.remove(s.length()-3,3);
|
|
result = s.toULongLong(&ok);
|
|
} else if (s.endsWith("UL",Qt::CaseInsensitive)) {
|
|
s.remove(s.length()-2,2);
|
|
result = s.toULong(&ok);
|
|
} else if (s.endsWith("U",Qt::CaseInsensitive)) {
|
|
s.remove(s.length()-1,1);
|
|
result = s.toUInt(&ok);
|
|
} else {
|
|
result = s.toInt(&ok);
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
bool CppPreprocessor::evalTerm(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!skipSpaces(expr,pos))
|
|
return false;
|
|
if (expr[pos]=='(') {
|
|
pos++;
|
|
if (!evalExpr(expr,result,pos))
|
|
return false;
|
|
if (!skipSpaces(expr,pos))
|
|
return false;
|
|
if (expr[pos]!=')')
|
|
return false;
|
|
pos++;
|
|
return true;
|
|
} else {
|
|
return evalNumber(expr,result,pos);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* unary_expr = term
|
|
| '+' term
|
|
| '-' term
|
|
| '!' term
|
|
| '~' term
|
|
*/
|
|
bool CppPreprocessor::evalUnaryExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!skipSpaces(expr,pos))
|
|
return false;
|
|
if (expr[pos]=='+') {
|
|
pos++;
|
|
if (!evalTerm(expr,result,pos))
|
|
return false;
|
|
} else if (expr[pos]=='-') {
|
|
pos++;
|
|
if (!evalTerm(expr,result,pos))
|
|
return false;
|
|
result = -result;
|
|
} else if (expr[pos]=='~') {
|
|
pos++;
|
|
if (!evalTerm(expr,result,pos))
|
|
return false;
|
|
result = ~result;
|
|
} else if (expr[pos]=='!') {
|
|
pos++;
|
|
if (!evalTerm(expr,result,pos))
|
|
return false;
|
|
result = !result;
|
|
} else {
|
|
return evalTerm(expr,result,pos);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* mul_expr = unary_expr
|
|
| mul_expr '*' unary_expr
|
|
| mul_expr '/' unary_expr
|
|
| mul_expr '%' unary_expr
|
|
*/
|
|
bool CppPreprocessor::evalMulExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalUnaryExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
int rightResult;
|
|
if (expr[pos]=='*') {
|
|
pos++;
|
|
if (!evalUnaryExpr(expr,rightResult,pos))
|
|
return false;
|
|
result *= rightResult;
|
|
} else if (expr[pos]=='/') {
|
|
pos++;
|
|
if (!evalUnaryExpr(expr,rightResult,pos))
|
|
return false;
|
|
if (rightResult != 0)
|
|
result /= rightResult;
|
|
else
|
|
result = 0;
|
|
} else if (expr[pos]=='%') {
|
|
pos++;
|
|
if (!evalUnaryExpr(expr,rightResult,pos))
|
|
return false;
|
|
if (rightResult != 0)
|
|
result %= rightResult;
|
|
else
|
|
result = 0;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* add_expr = mul_expr
|
|
| add_expr '+' mul_expr
|
|
| add_expr '-' mul_expr
|
|
*/
|
|
bool CppPreprocessor::evalAddExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalMulExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
int rightResult;
|
|
if (expr[pos]=='+') {
|
|
pos++;
|
|
if (!evalMulExpr(expr,rightResult,pos))
|
|
return false;
|
|
result += rightResult;
|
|
} else if (expr[pos]=='-') {
|
|
pos++;
|
|
if (!evalMulExpr(expr,rightResult,pos))
|
|
return false;
|
|
result -= rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* shift_expr = add_expr
|
|
| shift_expr "<<" add_expr
|
|
| shift_expr ">>" add_expr
|
|
*/
|
|
bool CppPreprocessor::evalShiftExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalAddExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
int rightResult;
|
|
if (pos+1<expr.length() && expr[pos] == '<' && expr[pos+1]=='<') {
|
|
pos += 2;
|
|
if (!evalAddExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result << rightResult);
|
|
} else if (pos+1<expr.length() && expr[pos] == '>' && expr[pos+1]=='>') {
|
|
pos += 2;
|
|
if (!evalAddExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result >> rightResult);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* relation_expr = shift_expr
|
|
| relation_expr ">=" shift_expr
|
|
| relation_expr ">" shift_expr
|
|
| relation_expr "<=" shift_expr
|
|
| relation_expr "<" shift_expr
|
|
*/
|
|
bool CppPreprocessor::evalRelationExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalShiftExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
int rightResult;
|
|
if (expr[pos]=='<') {
|
|
if (pos+1<expr.length() && expr[pos+1]=='=') {
|
|
pos+=2;
|
|
if (!evalShiftExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result <= rightResult);
|
|
} else {
|
|
pos++;
|
|
if (!evalShiftExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result < rightResult);
|
|
}
|
|
} else if (expr[pos]=='>') {
|
|
if (pos+1<expr.length() && expr[pos+1]=='=') {
|
|
pos+=2;
|
|
if (!evalShiftExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result >= rightResult);
|
|
} else {
|
|
pos++;
|
|
if (!evalShiftExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result > rightResult);
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* equal_expr = relation_expr
|
|
| equal_expr "==" relation_expr
|
|
| equal_expr "!=" relation_expr
|
|
*/
|
|
bool CppPreprocessor::evalEqualExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalRelationExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (pos+1<expr.length() && expr[pos]=='!' && expr[pos+1]=='=') {
|
|
pos+=2;
|
|
int rightResult;
|
|
if (!evalRelationExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result != rightResult);
|
|
} else if (pos+1<expr.length() && expr[pos]=='=' && expr[pos+1]=='=') {
|
|
pos+=2;
|
|
int rightResult;
|
|
if (!evalRelationExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = (result == rightResult);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* bit_and_expr = equal_expr
|
|
| bit_and_expr "&" equal_expr
|
|
*/
|
|
bool CppPreprocessor::evalBitAndExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalEqualExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (expr[pos]=='&'
|
|
&& (pos == expr.length()
|
|
|| expr[pos+1]!='&')) {
|
|
pos++;
|
|
int rightResult;
|
|
if (!evalEqualExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = result & rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* bit_xor_expr = bit_and_expr
|
|
| bit_xor_expr "^" bit_and_expr
|
|
*/
|
|
bool CppPreprocessor::evalBitXorExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalBitAndExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (expr[pos]=='^') {
|
|
pos++;
|
|
int rightResult;
|
|
if (!evalBitAndExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = result ^ rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* bit_or_expr = bit_xor_expr
|
|
| bit_or_expr "|" bit_xor_expr
|
|
*/
|
|
bool CppPreprocessor::evalBitOrExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalBitXorExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (expr[pos] == '|'
|
|
&& (pos == expr.length()
|
|
|| expr[pos+1]!='|')) {
|
|
pos++;
|
|
int rightResult;
|
|
if (!evalBitXorExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = result | rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* logic_and_expr = bit_or_expr
|
|
| logic_and_expr "&&" bit_or_expr
|
|
*/
|
|
bool CppPreprocessor::evalLogicAndExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalBitOrExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (pos+1<expr.length() && expr[pos]=='&' && expr[pos+1] =='&') {
|
|
pos+=2;
|
|
int rightResult;
|
|
if (!evalBitOrExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = result && rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* logic_or_expr = logic_and_expr
|
|
| logic_or_expr "||" logic_and_expr
|
|
*/
|
|
bool CppPreprocessor::evalLogicOrExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
if (!evalLogicAndExpr(expr,result,pos))
|
|
return false;
|
|
while (true) {
|
|
if (!skipSpaces(expr,pos))
|
|
break;
|
|
if (pos+1<expr.length() && expr[pos]=='|' && expr[pos+1] =='|') {
|
|
pos+=2;
|
|
int rightResult;
|
|
if (!evalLogicAndExpr(expr,rightResult,pos))
|
|
return false;
|
|
result = result || rightResult;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CppPreprocessor::evalExpr(const QString &expr, int &result, int &pos)
|
|
{
|
|
return evalLogicOrExpr(expr,result,pos);
|
|
}
|
|
|
|
/* BNF for C constant expression evaluation
|
|
* term = number
|
|
| '(' expression ')'
|
|
unary_expr = term
|
|
| '+' term
|
|
| '-' term
|
|
| '!' term
|
|
| '~' term
|
|
mul_expr = term
|
|
| mul_expr '*' term
|
|
| mul_expr '/' term
|
|
| mul_expr '%' term
|
|
add_expr = mul_expr
|
|
| add_expr '+' mul_expr
|
|
| add_expr '-' mul_expr
|
|
shift_expr = add_expr
|
|
| shift_expr "<<" add_expr
|
|
| shift_expr ">>" add_expr
|
|
relation_expr = shift_expr
|
|
| relation_expr ">=" shift_expr
|
|
| relation_expr ">" shift_expr
|
|
| relation_expr "<=" shift_expr
|
|
| relation_expr "<" shift_expr
|
|
equal_expr = relation_expr
|
|
| equal_expr "==" relation_expr
|
|
| equal_expr "!=" relation_expr
|
|
bit_and_expr = equal_expr
|
|
| bit_and_expr "&" equal_expr
|
|
bit_xor_expr = bit_and_expr
|
|
| bit_xor_expr "^" bit_and_expr
|
|
bit_or_expr = bit_xor_expr
|
|
| bit_or_expr "|" bit_xor_expr
|
|
logic_and_expr = bit_or_expr
|
|
| logic_and_expr "&&" bit_or_expr
|
|
logic_or_expr = logic_and_expr
|
|
| logic_or_expr "||" logic_and_expr
|
|
*/
|
|
|
|
int CppPreprocessor::evaluateExpression(QString line)
|
|
{
|
|
int pos = 0;
|
|
int result;
|
|
bool ok = evalExpr(line,result,pos);
|
|
if (!ok)
|
|
return -1;
|
|
//expr not finished
|
|
if (skipSpaces(line,pos))
|
|
return -1;
|
|
return result;
|
|
}
|
|
|
|
void CppPreprocessor::setOnGetFileStream(const GetFileStreamCallBack &newOnGetFileStream)
|
|
{
|
|
mOnGetFileStream = newOnGetFileStream;
|
|
}
|
|
|
|
const QList<QString> &CppPreprocessor::projectIncludePathList() const
|
|
{
|
|
return mProjectIncludePathList;
|
|
}
|
|
|
|
const QList<QString> &CppPreprocessor::includePathList() const
|
|
{
|
|
return mIncludePathList;
|
|
}
|
|
|
|
const DefineMap &CppPreprocessor::hardDefines() const
|
|
{
|
|
return mHardDefines;
|
|
}
|