/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "FastPreprocessor.h" #include #include #include using namespace CPlusPlus; FastPreprocessor::FastPreprocessor(const Snapshot &snapshot) : _snapshot(snapshot) , _preproc(this, &_env) , _addIncludesToCurrentDoc(false) { } QByteArray FastPreprocessor::run(Document::Ptr newDoc, const QByteArray &source, bool mergeDefinedMacrosOfDocument) { std::swap(newDoc, _currentDoc); _addIncludesToCurrentDoc = _currentDoc->resolvedIncludes().isEmpty() && _currentDoc->unresolvedIncludes().isEmpty(); const QString fileName = _currentDoc->fileName(); _preproc.setExpandFunctionlikeMacros(false); _preproc.setKeepComments(true); if (Document::Ptr doc = _snapshot.document(fileName)) { _merged.insert(fileName); for (Snapshot::const_iterator i = _snapshot.begin(), ei = _snapshot.end(); i != ei; ++i) { if (isInjectedFile(i.key().toString())) mergeEnvironment(i.key().toString()); } foreach (const Document::Include &i, doc->resolvedIncludes()) mergeEnvironment(i.resolvedFileName()); if (mergeDefinedMacrosOfDocument) _env.addMacros(_currentDoc->definedMacros()); } const QByteArray preprocessed = _preproc.run(fileName, source); // qDebug("FastPreprocessor::run for %s produced [[%s]]", fileName.toUtf8().constData(), preprocessed.constData()); std::swap(newDoc, _currentDoc); return preprocessed; } void FastPreprocessor::sourceNeeded(unsigned line, const QString &fileName, IncludeType mode, const QStringList &initialIncludes) { Q_UNUSED(initialIncludes) Q_ASSERT(_currentDoc); if (_addIncludesToCurrentDoc) { // CHECKME: Is that cleanName needed? const QString cleanName = QDir::cleanPath(fileName); _currentDoc->addIncludeFile(Document::Include(fileName, cleanName, line, mode)); } mergeEnvironment(fileName); } void FastPreprocessor::mergeEnvironment(const QString &fileName) { if (! _merged.contains(fileName)) { _merged.insert(fileName); if (Document::Ptr doc = _snapshot.document(fileName)) { foreach (const Document::Include &i, doc->resolvedIncludes()) mergeEnvironment(i.resolvedFileName()); _env.addMacros(doc->definedMacros()); } } } void FastPreprocessor::macroAdded(const Macro ¯o) { Q_ASSERT(_currentDoc); _currentDoc->appendMacro(macro); } static const Macro revision(const Snapshot &s, const Macro &m) { if (Document::Ptr d = s.document(m.fileName())) { Macro newMacro(m); newMacro.setFileRevision(d->revision()); return newMacro; } return m; } void FastPreprocessor::passedMacroDefinitionCheck(unsigned bytesOffset, unsigned utf16charsOffset, unsigned line, const Macro ¯o) { Q_ASSERT(_currentDoc); _currentDoc->addMacroUse(revision(_snapshot, macro), bytesOffset, macro.name().size(), utf16charsOffset, macro.nameToQString().size(), line, QVector()); } void FastPreprocessor::failedMacroDefinitionCheck(unsigned bytesOffset, unsigned utf16charsOffset, const ByteArrayRef &name) { Q_ASSERT(_currentDoc); _currentDoc->addUndefinedMacroUse(QByteArray(name.start(), name.size()), bytesOffset, utf16charsOffset); } void FastPreprocessor::notifyMacroReference(unsigned bytesOffset, unsigned utf16charsOffset, unsigned line, const Macro ¯o) { Q_ASSERT(_currentDoc); _currentDoc->addMacroUse(revision(_snapshot, macro), bytesOffset, macro.name().size(), utf16charsOffset, macro.nameToQString().size(), line, QVector()); } void FastPreprocessor::startExpandingMacro(unsigned bytesOffset, unsigned utf16charsOffset, unsigned line, const Macro ¯o, const QVector &actuals) { Q_ASSERT(_currentDoc); _currentDoc->addMacroUse(revision(_snapshot, macro), bytesOffset, macro.name().size(), utf16charsOffset, macro.nameToQString().size(), line, actuals); } void FastPreprocessor::markAsIncludeGuard(const QByteArray ¯oName) { if (!_currentDoc) return; _currentDoc->setIncludeGuardMacroName(macroName); }