2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** Contact: Qt Software Information (qt-info@nokia.com)
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Nokia.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
|
|
|
** contact the sales department at qt-sales@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "cmakeprojectmanager.h"
|
|
|
|
#include "cmakeprojectconstants.h"
|
|
|
|
#include "cmakeproject.h"
|
|
|
|
#include "cmakeprojectconstants.h"
|
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <coreplugin/icore.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <coreplugin/uniqueidmanager.h>
|
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
2009-01-21 16:25:21 +01:00
|
|
|
#include <projectexplorer/environment.h>
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <qtconcurrent/QtConcurrentTools>
|
|
|
|
#include <QtCore/QtConcurrentRun>
|
2009-01-21 16:25:21 +01:00
|
|
|
#include <QtCore/QSettings>
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <QtGui/QFormLayout>
|
2009-03-25 17:30:48 +01:00
|
|
|
#include <QtGui/QDesktopServices>
|
|
|
|
#include <QtGui/QApplication>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
using namespace CMakeProjectManager::Internal;
|
|
|
|
|
2009-01-21 16:25:21 +01:00
|
|
|
CMakeManager::CMakeManager(CMakeSettingsPage *cmakeSettingsPage)
|
|
|
|
: m_settingsPage(cmakeSettingsPage)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-01-21 15:52:34 +01:00
|
|
|
Core::UniqueIDManager *uidm = Core::UniqueIDManager::instance();
|
|
|
|
m_projectContext = uidm->uniqueIdentifier(CMakeProjectManager::Constants::PROJECTCONTEXT);
|
|
|
|
m_projectLanguage = uidm->uniqueIdentifier(ProjectExplorer::Constants::LANG_CXX);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2009-01-21 16:25:21 +01:00
|
|
|
CMakeSettingsPage::~CMakeSettingsPage()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
int CMakeManager::projectContext() const
|
|
|
|
{
|
|
|
|
return m_projectContext;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CMakeManager::projectLanguage() const
|
|
|
|
{
|
|
|
|
return m_projectLanguage;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::Project *CMakeManager::openProject(const QString &fileName)
|
|
|
|
{
|
|
|
|
// TODO check wheter this project is already opened
|
2009-01-21 16:25:21 +01:00
|
|
|
// Check that we have a cmake executable first
|
|
|
|
// Look at the settings first
|
|
|
|
QString cmakeExecutable = m_settingsPage->cmakeExecutable();
|
|
|
|
if (cmakeExecutable.isNull())
|
|
|
|
m_settingsPage->askUserForCMakeExecutable();
|
|
|
|
cmakeExecutable = m_settingsPage->cmakeExecutable();
|
|
|
|
if (cmakeExecutable.isNull())
|
|
|
|
return 0;
|
2008-12-02 12:01:29 +01:00
|
|
|
return new CMakeProject(this, fileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeManager::mimeType() const
|
|
|
|
{
|
|
|
|
return Constants::CMAKEMIMETYPE;
|
|
|
|
}
|
2009-01-21 16:25:21 +01:00
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
QString CMakeManager::cmakeExecutable() const
|
|
|
|
{
|
|
|
|
return m_settingsPage->cmakeExecutable();
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO need to refactor this out
|
|
|
|
// we probably want the process instead of this function
|
|
|
|
// cmakeproject then could even run the cmake process in the background, adding the files afterwards
|
|
|
|
// sounds like a plan
|
2009-03-09 18:13:19 +01:00
|
|
|
QProcess *CMakeManager::createXmlFile(const QStringList &arguments, const QString &sourceDirectory, const QDir &buildDirectory)
|
2009-01-23 16:57:38 +01:00
|
|
|
{
|
|
|
|
// We create a cbp file, only if we didn't find a cbp file in the base directory
|
|
|
|
// Yet that can still override cbp files in subdirectories
|
|
|
|
// And we are creating tons of files in the source directories
|
|
|
|
// All of that is not really nice.
|
|
|
|
// The mid term plan is to move away from the CodeBlocks Generator and use our own
|
|
|
|
// QtCreator generator, which actually can be very similar to the CodeBlock Generator
|
|
|
|
|
|
|
|
|
|
|
|
// TODO we need to pass on the same paremeters as the cmakestep
|
|
|
|
QString buildDirectoryPath = buildDirectory.absolutePath();
|
|
|
|
qDebug()<<"Creating cbp file in"<<buildDirectoryPath;
|
|
|
|
buildDirectory.mkpath(buildDirectoryPath);
|
2009-04-20 12:18:32 +02:00
|
|
|
QProcess *cmake = new QProcess;
|
2009-03-09 18:13:19 +01:00
|
|
|
cmake->setWorkingDirectory(buildDirectoryPath);
|
2009-04-20 12:18:32 +02:00
|
|
|
cmake->setProcessChannelMode(QProcess::MergedChannels);
|
2009-03-03 17:56:03 +01:00
|
|
|
|
2009-03-10 11:48:37 +01:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QString generator = "-GCodeBlocks - MinGW Makefiles";
|
|
|
|
#else // Q_OS_WIN
|
2009-03-27 20:45:53 +01:00
|
|
|
QString generator = "-GCodeBlocks - Unix Makefiles";
|
2009-03-10 11:48:37 +01:00
|
|
|
#endif // Q_OS_WIN
|
2009-03-27 20:45:53 +01:00
|
|
|
QString srcdir = buildDirectory.exists("CMakeCache.txt") ? QString(".") : sourceDirectory;
|
|
|
|
qDebug()<<cmakeExecutable()<<srcdir<<arguments<<generator;
|
|
|
|
cmake->start(cmakeExecutable(), QStringList() << srcdir << arguments << generator);
|
2009-03-09 18:13:19 +01:00
|
|
|
return cmake;
|
2009-01-23 16:57:38 +01:00
|
|
|
}
|
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
QString CMakeManager::findCbpFile(const QDir &directory)
|
|
|
|
{
|
|
|
|
// Find the cbp file
|
|
|
|
// TODO the cbp file is named like the project() command in the CMakeList.txt file
|
|
|
|
// so this method below could find the wrong cbp file, if the user changes the project()
|
|
|
|
// 2name
|
|
|
|
foreach (const QString &cbpFile , directory.entryList()) {
|
|
|
|
if (cbpFile.endsWith(".cbp"))
|
|
|
|
return directory.path() + "/" + cbpFile;
|
|
|
|
}
|
|
|
|
return QString::null;
|
|
|
|
}
|
|
|
|
|
2009-03-25 17:30:48 +01:00
|
|
|
// This code is duplicated from qtversionmanager
|
|
|
|
QString CMakeManager::qtVersionForQMake(const QString &qmakePath)
|
|
|
|
{
|
|
|
|
QProcess qmake;
|
|
|
|
qmake.start(qmakePath, QStringList()<<"--version");
|
|
|
|
if (!qmake.waitForFinished())
|
|
|
|
return false;
|
|
|
|
QString output = qmake.readAllStandardOutput();
|
|
|
|
QRegExp regexp("(QMake version|Qmake version:)[\\s]*([\\d.]*)");
|
|
|
|
regexp.indexIn(output);
|
|
|
|
if (regexp.cap(2).startsWith("2.")) {
|
|
|
|
QRegExp regexp2("Using Qt version[\\s]*([\\d\\.]*)");
|
|
|
|
regexp2.indexIn(output);
|
|
|
|
return regexp2.cap(1);
|
|
|
|
}
|
|
|
|
return QString();
|
|
|
|
}
|
2009-03-09 18:13:19 +01:00
|
|
|
|
2009-04-17 12:27:04 +02:00
|
|
|
// this is mostly a copy from qt4versionmanager
|
|
|
|
// TODO refactor this code
|
|
|
|
// returns QPair< QTDIR, QT_INSTALL_DATA >
|
|
|
|
QPair<QString, QString> CMakeManager::findQtDir(const ProjectExplorer::Environment &env)
|
2009-03-25 17:30:48 +01:00
|
|
|
{
|
|
|
|
QStringList possibleCommands;
|
|
|
|
// On windows noone has renamed qmake, right?
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
possibleCommands << "qmake.exe";
|
|
|
|
#endif
|
|
|
|
// On unix some distributions renamed qmake to avoid clashes
|
|
|
|
possibleCommands << "qmake-qt4" << "qmake4" << "qmake";
|
|
|
|
|
|
|
|
QStringList paths = env.path();
|
|
|
|
foreach (const QString &path, paths) {
|
|
|
|
foreach (const QString &possibleCommand, possibleCommands) {
|
|
|
|
QFileInfo qmake(path + "/" + possibleCommand);
|
|
|
|
if (qmake.exists()) {
|
|
|
|
if (!qtVersionForQMake(qmake.absoluteFilePath()).isNull()) {
|
2009-04-17 12:27:04 +02:00
|
|
|
QDir qtDir = qmake.absoluteDir();
|
|
|
|
qtDir.cdUp();
|
2009-04-08 15:16:02 +02:00
|
|
|
QProcess proc;
|
|
|
|
proc.start(qmake.absoluteFilePath(), QStringList() << "-query" << "QT_INSTALL_DATA");
|
|
|
|
if (proc.waitForFinished()) {
|
2009-04-17 12:27:04 +02:00
|
|
|
return qMakePair(qtDir.absolutePath(), QString(proc.readAll().trimmed()));
|
2009-04-08 15:16:02 +02:00
|
|
|
} else {
|
|
|
|
proc.kill();
|
|
|
|
QDir dir(qmake.absoluteDir());
|
|
|
|
dir.cdUp();
|
2009-04-17 12:27:04 +02:00
|
|
|
return qMakePair(qtDir.absolutePath(), dir.absolutePath());
|
2009-04-08 15:16:02 +02:00
|
|
|
}
|
2009-03-25 17:30:48 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-04-17 12:27:04 +02:00
|
|
|
return qMakePair(QString(), QString());
|
2009-03-25 17:30:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// This code is more or less duplicated in qtversionmanager
|
|
|
|
QString CMakeManager::findDumperLibrary(const ProjectExplorer::Environment &env)
|
|
|
|
{
|
|
|
|
static ProjectExplorer::Environment lastenv;
|
|
|
|
static QString lastpath;
|
|
|
|
if (lastenv == env)
|
|
|
|
return lastpath;
|
2009-04-17 12:27:04 +02:00
|
|
|
|
|
|
|
QPair<QString, QString> pair = findQtDir(env);
|
|
|
|
QString qtInstallDataDir = pair.second;
|
|
|
|
if (qtInstallDataDir.isEmpty())
|
2009-03-25 17:30:48 +01:00
|
|
|
return QString();
|
|
|
|
|
2009-04-17 12:27:04 +02:00
|
|
|
uint hash = qHash(pair.first);
|
2009-03-25 17:30:48 +01:00
|
|
|
QStringList directories;
|
|
|
|
directories
|
2009-04-17 12:27:04 +02:00
|
|
|
<< (qtInstallDataDir + "/qtc-debugging-helper/")
|
2009-03-25 17:30:48 +01:00
|
|
|
<< (QApplication::applicationDirPath() + "/../qtc-debugging-helper/" + QString::number(hash)) + "/"
|
|
|
|
<< (QDesktopServices::storageLocation(QDesktopServices::DataLocation) + "/qtc-debugging-helper/" + QString::number(hash)) + "/";
|
|
|
|
foreach(const QString &directory, directories) {
|
|
|
|
#if defined(Q_OS_WIN)
|
|
|
|
QFileInfo fi(directory + "debug/gdbmacros.dll");
|
|
|
|
#elif defined(Q_OS_MAC)
|
|
|
|
QFileInfo fi(directory + "libgdbmacros.dylib");
|
|
|
|
#else // generic UNIX
|
|
|
|
QFileInfo fi(directory + "libgdbmacros.so");
|
|
|
|
#endif
|
|
|
|
if (fi.exists()) {
|
|
|
|
lastpath = fi.filePath();
|
|
|
|
return lastpath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
lastpath = QString();
|
|
|
|
return lastpath;
|
|
|
|
}
|
2009-01-23 16:57:38 +01:00
|
|
|
|
2009-01-21 16:25:21 +01:00
|
|
|
/////
|
2009-01-23 16:57:38 +01:00
|
|
|
// CMakeRunner
|
2009-01-21 16:25:21 +01:00
|
|
|
////
|
2009-01-23 16:57:38 +01:00
|
|
|
// TODO give a better name, what this class is to update cached information
|
|
|
|
// about a cmake executable, with qtconcurrent
|
|
|
|
// The nifty feature of this class is that it does so in a seperate thread,
|
|
|
|
// not blocking the main thread
|
2009-01-21 16:25:21 +01:00
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
CMakeRunner::CMakeRunner()
|
|
|
|
: m_cacheUpToDate(false)
|
2009-01-21 16:25:21 +01:00
|
|
|
{
|
2009-01-23 16:57:38 +01:00
|
|
|
|
2009-01-21 17:21:59 +01:00
|
|
|
}
|
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
void CMakeRunner::run(QFutureInterface<void> &fi)
|
2009-01-21 17:21:59 +01:00
|
|
|
{
|
2009-01-23 16:57:38 +01:00
|
|
|
m_mutex.lock();
|
|
|
|
QString executable = m_executable;
|
|
|
|
m_mutex.unlock();
|
2009-01-21 17:21:59 +01:00
|
|
|
QProcess cmake;
|
2009-01-23 16:57:38 +01:00
|
|
|
cmake.start(executable, QStringList()<<"--help");
|
2009-01-21 17:21:59 +01:00
|
|
|
cmake.waitForFinished();
|
|
|
|
QString response = cmake.readAll();
|
|
|
|
QRegExp versionRegexp("^cmake version ([*\\d\\.]*)-(|patch (\\d*))(|\\r)\\n");
|
|
|
|
versionRegexp.indexIn(response);
|
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
m_mutex.lock();
|
2009-01-21 17:21:59 +01:00
|
|
|
m_supportsQtCreator = response.contains("QtCreator");
|
|
|
|
m_version = versionRegexp.cap(1);
|
|
|
|
if (!versionRegexp.capturedTexts().size()>3)
|
|
|
|
m_version += "." + versionRegexp.cap(3);
|
2009-01-23 16:57:38 +01:00
|
|
|
m_cacheUpToDate = true;
|
|
|
|
m_mutex.unlock();
|
|
|
|
fi.reportFinished();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeRunner::setExecutable(const QString &executable)
|
|
|
|
{
|
|
|
|
waitForUpToDate();
|
|
|
|
m_mutex.lock();
|
|
|
|
m_executable = executable;
|
|
|
|
m_cacheUpToDate = false;
|
|
|
|
m_mutex.unlock();
|
|
|
|
m_future = QtConcurrent::run(&CMakeRunner::run, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeRunner::executable() const
|
|
|
|
{
|
|
|
|
waitForUpToDate();
|
|
|
|
m_mutex.lock();
|
|
|
|
QString result = m_executable;
|
|
|
|
m_mutex.unlock();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeRunner::version() const
|
|
|
|
{
|
|
|
|
waitForUpToDate();
|
|
|
|
m_mutex.lock();
|
|
|
|
QString result = m_version;
|
|
|
|
m_mutex.unlock();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMakeRunner::supportsQtCreator() const
|
|
|
|
{
|
|
|
|
waitForUpToDate();
|
|
|
|
m_mutex.lock();
|
|
|
|
bool result = m_supportsQtCreator;
|
|
|
|
m_mutex.unlock();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeRunner::waitForUpToDate() const
|
|
|
|
{
|
|
|
|
m_future.waitForFinished();
|
|
|
|
}
|
|
|
|
|
|
|
|
/////
|
|
|
|
// CMakeSettingsPage
|
|
|
|
////
|
|
|
|
|
|
|
|
|
|
|
|
CMakeSettingsPage::CMakeSettingsPage()
|
|
|
|
{
|
|
|
|
Core::ICore *core = Core::ICore::instance();
|
|
|
|
QSettings * settings = core->settings();
|
|
|
|
settings->beginGroup("CMakeSettings");
|
|
|
|
m_cmakeRunner.setExecutable(settings->value("cmakeExecutable").toString());
|
|
|
|
settings->endGroup();
|
2009-01-21 16:25:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeSettingsPage::findCmakeExecutable() const
|
|
|
|
{
|
|
|
|
ProjectExplorer::Environment env = ProjectExplorer::Environment::systemEnvironment();
|
|
|
|
return env.searchInPath("cmake");
|
|
|
|
}
|
|
|
|
|
2009-03-18 16:43:01 +01:00
|
|
|
QString CMakeSettingsPage::id() const
|
|
|
|
{
|
|
|
|
return QLatin1String("CMake");
|
|
|
|
}
|
2009-01-21 16:25:21 +01:00
|
|
|
|
2009-03-18 16:43:01 +01:00
|
|
|
QString CMakeSettingsPage::trName() const
|
2009-01-21 16:25:21 +01:00
|
|
|
{
|
2009-03-18 16:43:01 +01:00
|
|
|
return tr("CMake");
|
2009-01-21 16:25:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeSettingsPage::category() const
|
|
|
|
{
|
2009-03-18 16:43:01 +01:00
|
|
|
return QLatin1String("CMake");
|
2009-01-21 16:25:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeSettingsPage::trCategory() const
|
|
|
|
{
|
|
|
|
return tr("CMake");
|
|
|
|
}
|
|
|
|
|
|
|
|
QWidget *CMakeSettingsPage::createPage(QWidget *parent)
|
|
|
|
{
|
|
|
|
QWidget *w = new QWidget(parent);
|
|
|
|
QFormLayout *fl = new QFormLayout(w);
|
|
|
|
m_pathchooser = new Core::Utils::PathChooser(w);
|
|
|
|
m_pathchooser->setExpectedKind(Core::Utils::PathChooser::Command);
|
|
|
|
fl->addRow("CMake executable", m_pathchooser);
|
|
|
|
m_pathchooser->setPath(cmakeExecutable());
|
|
|
|
return w;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeSettingsPage::saveSettings() const
|
|
|
|
{
|
|
|
|
QSettings *settings = Core::ICore::instance()->settings();
|
|
|
|
settings->beginGroup("CMakeSettings");
|
2009-01-23 16:57:38 +01:00
|
|
|
settings->setValue("cmakeExecutable", m_cmakeRunner.executable());
|
2009-01-21 16:25:21 +01:00
|
|
|
settings->endGroup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeSettingsPage::apply()
|
|
|
|
{
|
2009-01-23 16:57:38 +01:00
|
|
|
m_cmakeRunner.setExecutable(m_pathchooser->path());
|
2009-01-21 16:25:21 +01:00
|
|
|
saveSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeSettingsPage::finish()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeSettingsPage::cmakeExecutable() const
|
|
|
|
{
|
2009-01-23 16:57:38 +01:00
|
|
|
if (m_cmakeRunner.executable().isEmpty()) {
|
|
|
|
QString cmakeExecutable = findCmakeExecutable();
|
|
|
|
if (!cmakeExecutable.isEmpty()) {
|
|
|
|
m_cmakeRunner.setExecutable(cmakeExecutable);
|
2009-01-21 16:25:21 +01:00
|
|
|
saveSettings();
|
|
|
|
}
|
|
|
|
}
|
2009-01-23 16:57:38 +01:00
|
|
|
return m_cmakeRunner.executable();
|
2009-01-21 16:25:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeSettingsPage::askUserForCMakeExecutable()
|
|
|
|
{
|
|
|
|
// TODO implement
|
|
|
|
// That is ideally add a label to the settings page, which says something
|
|
|
|
// to the effect: please configure the cmake executable
|
|
|
|
// and show the settings page
|
|
|
|
// ensure that we rehide the label in the finish() function
|
|
|
|
// But to test that i need an environment without cmake, e.g. windows
|
|
|
|
}
|