diff options
author | Nikolai Kosjar <nikolai.kosjar@digia.com> | 2012-10-29 13:54:33 +0100 |
---|---|---|
committer | Nikolai Kosjar <nikolai.kosjar@digia.com> | 2012-11-22 14:11:58 +0100 |
commit | d0f3d7cb89a234f88b06bc19a41e50c41b1eab0a (patch) | |
tree | 31c22e52a2f35be42d7298fab1cfc6a92e112550 /src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp | |
parent | 1a003ed29bd72009ed9c619d6af92804d099dbfc (diff) | |
download | qt-creator-d0f3d7cb89a234f88b06bc19a41e50c41b1eab0a.tar.gz |
C++: Clean up dev tools.
* Add -h and -help options describing the tools and their usage.
* Make the tools compile and run on Windows (MinGW, MSVC).
* Rename project dirs, executables and main source files to more
meaningful names:
- Use same base name for project dir, *.pro file, main source file
and (if applicable) script file.
- Use the prefix "cplusplus-".
- The names are now:
- gen-cpp-ast/generate-ast --> cplusplus-update-frontend
- mkvisitor --> cplusplus-mkvisitor
- cplusplus-dump/cplusplus0 --> cplusplus-ast2png
* Get rid of 'c++' shell scripts.
* Get rid of duplicates of 'conf.c++'. Rename to 'pp-configuration.inc'.
* Introduce src/tools/cplusplus-tools-utils containing common stuff
that is used at least in two tools. 'pp-configuration.inc' can also be
found here.
* cplusplus-update-frontend:
- Print file paths of written files to stdout.
- Convenience: Use default values referencing the appropriate dirs and
files.
* cplusplus-mkvisitor:
- Take only one argument, namely the path to AST.h.
- Convenience: Use default path to AST.h.
* cplusplus-ast2png:
- Make it run without LD_LIBRARY_PATH.
- As the name suggests, generate image files in png format (needs
'dot' from graphviz).
- Convenience: Read from stdin, which useful for small snippets.
Change-Id: I79c4061fce4a1571c0588dfedd50d4a70715d9df
Reviewed-by: Erik Verbruggen <erik.verbruggen@digia.com>
Diffstat (limited to 'src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp')
-rw-r--r-- | src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp | 140 |
1 files changed, 140 insertions, 0 deletions
diff --git a/src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp b/src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp new file mode 100644 index 0000000000..5ad4ce0906 --- /dev/null +++ b/src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp @@ -0,0 +1,140 @@ +/**************************************************************************** +** +** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies). +** Contact: http://www.qt-project.org/legal +** +** 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 Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +****************************************************************************/ + + +#include "cplusplus-tools-utils.h" +#include "environment.h" + +#include <QDebug> +#include <QDir> +#include <QFile> +#include <QProcess> + +namespace CplusplusToolsUtils { + +QString portableExecutableName(const QString &executable) +{ +#if defined(Q_OS_WIN) + return executable + QLatin1String(".exe"); +#else + return executable; +#endif +} + +void executeCommand(const QString &command, const QStringList &arguments, const QString &outputFile, + bool verbose) +{ + QTextStream out(stderr); + if (command.isEmpty()) { + out << "Error: " << Q_FUNC_INFO << "Got empty command to execute." << endl; + exit(EXIT_FAILURE); + } + + const QString fullCommand = command + QLatin1Char(' ') + arguments.join(QLatin1String(" ")); + if (verbose) + out << "Executing: " << fullCommand << endl; + + QProcess process; + if (!outputFile.isEmpty()) + process.setStandardOutputFile(outputFile, QIODevice::Truncate); + process.start(command, arguments); + if (!process.waitForStarted()) { + out << QString("Error: Process \"%1\" did not start within timeout: %2.") + .arg(fullCommand, process.errorString()) + << endl; + exit(EXIT_FAILURE); + } + if (!process.waitForFinished()) { + if (!verbose) + out << process.readAll() << endl; + out << QString("Error: Process \"%1\" did not finish within timeout.").arg(fullCommand) + << endl; + exit(EXIT_FAILURE); + } + const int exitCode = process.exitCode(); + if (exitCode != 0) { + out << process.readAllStandardError() << endl; + out << QString("Error: Process \"%1\" finished with non zero exit value %2") + .arg(fullCommand, exitCode) << endl; + exit(EXIT_FAILURE); + } +} + +SystemPreprocessor::SystemPreprocessor(bool verbose) + : m_verbose(verbose) +{ + m_knownCompilers[portableExecutableName("gcc")] + = QLatin1String("-DCPLUSPLUS_WITHOUT_QT -U__BLOCKS__ -xc++ -E -include"); + m_knownCompilers[portableExecutableName("cl")] + = QLatin1String("/DCPLUSPLUS_WITHOUT_QT /U__BLOCKS__ /TP /E /I . /FI"); + + QMapIterator<QString, QString> i(m_knownCompilers); + while (i.hasNext()) { + i.next(); + const QString executablePath + = Utils::Environment::systemEnvironment().searchInPath(i.key()); + if (!executablePath.isEmpty()) { + m_compiler = i.key(); + m_compilerArguments = i.value().split(QLatin1String(" "), QString::SkipEmptyParts); + m_compilerArguments + << QDir::toNativeSeparators(QLatin1String(PATH_PREPROCESSOR_CONFIG)); + break; + } + } +} + +void SystemPreprocessor::check() const +{ + QTextStream out(stderr); + if (!QFile::exists(PATH_PREPROCESSOR_CONFIG)) { + out << QString("Error: File \"%1\" does not exist.").arg(PATH_PREPROCESSOR_CONFIG) << endl; + exit(EXIT_FAILURE); + } + if (m_compiler.isEmpty()) { + const QString triedCompilers + = QStringList(m_knownCompilers.keys()).join(QLatin1String(", ")); + out << QString("Error: No compiler found. Tried %1.").arg(triedCompilers) << endl; + exit(EXIT_FAILURE); + } +} + +void SystemPreprocessor::preprocessFile(const QString &inputFile, const QString &outputFile) const +{ + check(); + if (!QFile::exists(inputFile)) { + QTextStream out(stderr); + out << QString("Error: File \"%1\" does not exist.").arg(inputFile) << endl; + exit(EXIT_FAILURE); + } + const QStringList arguments = QStringList(m_compilerArguments) + << QDir::toNativeSeparators(inputFile); + executeCommand(m_compiler, arguments, outputFile, m_verbose); +} + +} // namespace |