summaryrefslogtreecommitdiff
path: root/src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp
diff options
context:
space:
mode:
authorNikolai Kosjar <nikolai.kosjar@digia.com>2012-10-29 13:54:33 +0100
committerNikolai Kosjar <nikolai.kosjar@digia.com>2012-11-22 14:11:58 +0100
commitd0f3d7cb89a234f88b06bc19a41e50c41b1eab0a (patch)
tree31c22e52a2f35be42d7298fab1cfc6a92e112550 /src/tools/cplusplus-tools-utils/cplusplus-tools-utils.cpp
parent1a003ed29bd72009ed9c619d6af92804d099dbfc (diff)
downloadqt-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.cpp140
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