2012-10-02 09:12:39 +02:00
/****************************************************************************
2010-09-27 15:51:49 +02:00
* *
2014-01-07 13:27:11 +01:00
* * Copyright ( C ) 2014 Digia Plc and / or its subsidiary ( - ies ) .
2012-10-02 09:12:39 +02:00
* * Contact : http : //www.qt-project.org/legal
2010-09-27 15:51:49 +02:00
* *
2012-10-02 09:12:39 +02:00
* * This file is part of Qt Creator .
2010-09-27 15:51:49 +02:00
* *
2012-10-02 09:12:39 +02:00
* * 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.
2010-09-27 15:51:49 +02:00
* *
* * GNU Lesser General Public License Usage
2012-10-02 09:12:39 +02: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.
* *
* * In addition , as a special exception , Digia gives you certain additional
* * rights . These rights are described in the Digia Qt LGPL Exception
2010-12-17 16:01:08 +01:00
* * version 1.1 , included in the file LGPL_EXCEPTION . txt in this package .
* *
2012-10-02 09:12:39 +02:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-09-27 15:51:49 +02:00
# include "buildablehelperlibrary.h"
2013-03-18 14:47:33 +01:00
# include "hostosinfo.h"
# include "synchronousprocess.h"
2010-09-27 15:51:49 +02:00
2012-02-15 10:42:41 +01:00
# include <QDir>
# include <QDateTime>
# include <QDebug>
2010-09-27 15:51:49 +02:00
namespace Utils {
2013-08-05 13:03:43 +02:00
bool BuildableHelperLibrary : : isQtChooser ( const QFileInfo & info )
{
return info . isSymLink ( ) & & info . symLinkTarget ( ) . endsWith ( QLatin1String ( " /qtchooser " ) ) ;
}
QString BuildableHelperLibrary : : qtChooserToQmakePath ( const QString & path )
{
2013-10-10 10:39:36 +02:00
const char toolDir [ ] = " QTTOOLDIR= \" " ;
2013-08-05 13:03:43 +02:00
QProcess proc ;
proc . start ( path , QStringList ( QLatin1String ( " -print-env " ) ) ) ;
if ( ! proc . waitForStarted ( 1000 ) )
return QString ( ) ;
if ( ! proc . waitForFinished ( 1000 ) )
return QString ( ) ;
QByteArray output = proc . readAllStandardOutput ( ) ;
2013-10-10 10:39:36 +02:00
int pos = output . indexOf ( toolDir ) ;
2013-08-05 13:03:43 +02:00
if ( pos = = - 1 )
return QString ( ) ;
2013-10-10 10:39:36 +02:00
pos + = int ( sizeof ( toolDir ) ) - 1 ;
2013-08-05 13:03:43 +02:00
int end = output . indexOf ( ' \" ' , pos ) ;
if ( end = = - 1 )
return QString ( ) ;
QString result = QString : : fromLocal8Bit ( output . mid ( pos , end - pos ) ) + QLatin1String ( " /qmake " ) ;
return result ;
}
2011-11-25 13:19:58 +01:00
Utils : : FileName BuildableHelperLibrary : : findSystemQt ( const Utils : : Environment & env )
2010-09-27 15:51:49 +02:00
{
QStringList paths = env . path ( ) ;
foreach ( const QString & path , paths ) {
2010-11-08 11:28:54 +01:00
QString prefix = path ;
if ( ! prefix . endsWith ( QLatin1Char ( ' / ' ) ) )
prefix . append ( QLatin1Char ( ' / ' ) ) ;
2010-09-27 15:51:49 +02:00
foreach ( const QString & possibleCommand , possibleQMakeCommands ( ) ) {
2013-08-05 13:03:43 +02:00
QFileInfo qmake ( prefix + possibleCommand ) ;
2010-09-27 15:51:49 +02:00
if ( qmake . exists ( ) ) {
2013-08-05 13:03:43 +02:00
if ( isQtChooser ( qmake ) )
qmake . setFile ( qtChooserToQmakePath ( qmake . symLinkTarget ( ) ) ) ;
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
if ( ! qtVersionForQMake ( qmake . absoluteFilePath ( ) ) . isNull ( ) )
2011-11-25 13:19:58 +01:00
return Utils : : FileName ( qmake ) ;
2010-09-27 15:51:49 +02:00
}
}
}
2011-11-25 13:19:58 +01:00
return Utils : : FileName ( ) ;
2010-09-27 15:51:49 +02:00
}
2011-11-25 13:19:58 +01:00
QString BuildableHelperLibrary : : qtInstallDataDir ( const Utils : : FileName & qmakePath )
2010-09-27 15:51:49 +02:00
{
QProcess proc ;
2011-11-25 13:19:58 +01:00
proc . start ( qmakePath . toString ( ) , QStringList ( ) < < QLatin1String ( " -query " ) < < QLatin1String ( " QT_INSTALL_DATA " ) ) ;
2010-09-27 15:51:49 +02:00
if ( proc . waitForFinished ( ) )
2012-01-04 17:34:08 +01:00
return QString : : fromLocal8Bit ( proc . readAll ( ) ) . trimmed ( ) ;
2010-09-27 15:51:49 +02:00
return QString ( ) ;
}
QString BuildableHelperLibrary : : qtVersionForQMake ( const QString & qmakePath )
{
2011-07-14 13:35:40 +02:00
bool qmakeIsExecutable ;
return BuildableHelperLibrary : : qtVersionForQMake ( qmakePath , & qmakeIsExecutable ) ;
}
QString BuildableHelperLibrary : : qtVersionForQMake ( const QString & qmakePath , bool * qmakeIsExecutable )
{
* qmakeIsExecutable = ! qmakePath . isEmpty ( ) ;
if ( ! * qmakeIsExecutable )
2010-09-27 15:51:49 +02:00
return QString ( ) ;
QProcess qmake ;
qmake . start ( qmakePath , QStringList ( QLatin1String ( " --version " ) ) ) ;
if ( ! qmake . waitForStarted ( ) ) {
2011-07-14 13:35:40 +02:00
* qmakeIsExecutable = false ;
2010-09-27 15:51:49 +02:00
qWarning ( " Cannot start '%s': %s " , qPrintable ( qmakePath ) , qPrintable ( qmake . errorString ( ) ) ) ;
return QString ( ) ;
}
if ( ! qmake . waitForFinished ( ) ) {
Utils : : SynchronousProcess : : stopProcess ( qmake ) ;
qWarning ( " Timeout running '%s'. " , qPrintable ( qmakePath ) ) ;
return QString ( ) ;
}
if ( qmake . exitStatus ( ) ! = QProcess : : NormalExit ) {
2011-07-14 13:35:40 +02:00
* qmakeIsExecutable = false ;
2010-09-27 15:51:49 +02:00
qWarning ( " '%s' crashed. " , qPrintable ( qmakePath ) ) ;
return QString ( ) ;
}
2011-07-14 13:35:40 +02:00
2010-09-27 15:51:49 +02:00
const QString output = QString : : fromLocal8Bit ( qmake . readAllStandardOutput ( ) ) ;
2010-11-08 09:41:40 +01:00
static QRegExp regexp ( QLatin1String ( " (QMake version|QMake version:)[ \\ s]*([ \\ d.]*) " ) ,
Qt : : CaseInsensitive ) ;
2010-09-27 15:51:49 +02:00
regexp . indexIn ( output ) ;
2012-09-25 15:07:36 +02:00
const QString qmakeVersion = regexp . cap ( 2 ) ;
if ( qmakeVersion . startsWith ( QLatin1String ( " 2. " ) )
| | qmakeVersion . startsWith ( QLatin1String ( " 3. " ) ) ) {
2010-11-08 09:41:40 +01:00
static QRegExp regexp2 ( QLatin1String ( " Using Qt version[ \\ s]*([ \\ d \\ .]*) " ) ,
Qt : : CaseInsensitive ) ;
2010-09-27 15:51:49 +02:00
regexp2 . indexIn ( output ) ;
const QString version = regexp2 . cap ( 1 ) ;
return version ;
}
return QString ( ) ;
}
QStringList BuildableHelperLibrary : : possibleQMakeCommands ( )
{
// On windows no one has renamed qmake, right?
2012-08-23 15:53:58 +02:00
if ( HostOsInfo : : isWindowsHost ( ) )
return QStringList ( QLatin1String ( " qmake.exe " ) ) ;
2010-09-27 15:51:49 +02:00
// On unix some distributions renamed qmake to avoid clashes
QStringList result ;
2013-04-04 13:35:10 +03:00
result < < QLatin1String ( " qmake " ) < < QLatin1String ( " qmake-qt4 " ) < < QLatin1String ( " qmake4 " )
< < QLatin1String ( " qmake-qt5 " ) < < QLatin1String ( " qmake5 " ) ;
2010-09-27 15:51:49 +02:00
return result ;
}
// Copy helper source files to a target directory, replacing older files.
bool BuildableHelperLibrary : : copyFiles ( const QString & sourcePath ,
const QStringList & files ,
const QString & targetDirectory ,
QString * errorMessage )
{
2011-07-20 13:47:04 +02:00
// try remove the directory
2012-09-05 15:52:53 +02:00
if ( ! FileUtils : : removeRecursively ( FileName : : fromString ( targetDirectory ) , errorMessage ) )
2011-07-20 13:47:04 +02:00
return false ;
2010-09-27 15:51:49 +02:00
if ( ! QDir ( ) . mkpath ( targetDirectory ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::DebuggingHelperLibrary " , " The target directory %1 could not be created. " ) . arg ( targetDirectory ) ;
return false ;
}
foreach ( const QString & file , files ) {
const QString source = sourcePath + file ;
const QString dest = targetDirectory + file ;
const QFileInfo destInfo ( dest ) ;
if ( destInfo . exists ( ) ) {
if ( destInfo . lastModified ( ) > = QFileInfo ( source ) . lastModified ( ) )
continue ;
if ( ! QFile : : remove ( dest ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::DebuggingHelperLibrary " , " The existing file %1 could not be removed. " ) . arg ( destInfo . absoluteFilePath ( ) ) ;
return false ;
}
}
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
if ( ! destInfo . dir ( ) . exists ( ) )
2010-12-10 13:09:28 +01:00
QDir ( ) . mkpath ( destInfo . dir ( ) . absolutePath ( ) ) ;
2010-09-27 15:51:49 +02:00
if ( ! QFile : : copy ( source , dest ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::DebuggingHelperLibrary " , " The file %1 could not be copied to %2. " ) . arg ( source , dest ) ;
return false ;
}
}
return true ;
}
2010-10-08 15:13:02 +02:00
// Helper: Run a build process with merged stdout/stderr
static inline bool runBuildProcessI ( QProcess & proc ,
const QString & binary ,
const QStringList & args ,
int timeoutMS ,
bool ignoreNonNullExitCode ,
QString * output , QString * errorMessage )
{
proc . start ( binary , args ) ;
if ( ! proc . waitForStarted ( ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" Cannot start process: %1 " ) .
arg ( proc . errorString ( ) ) ;
return false ;
}
// Read stdout/err and check for timeouts
QByteArray stdOut ;
QByteArray stdErr ;
if ( ! SynchronousProcess : : readDataFromProcess ( proc , timeoutMS , & stdOut , & stdErr , false ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" Timeout after %1s. " ) .
arg ( timeoutMS / 1000 ) ;
SynchronousProcess : : stopProcess ( proc ) ;
return false ;
}
if ( proc . exitStatus ( ) ! = QProcess : : NormalExit ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" The process crashed. " ) ;
return false ;
}
const QString stdOutS = QString : : fromLocal8Bit ( stdOut ) ;
if ( ! ignoreNonNullExitCode & & proc . exitCode ( ) ! = 0 ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" The process returned exit code %1: \n %2 " ) .
arg ( proc . exitCode ( ) ) . arg ( stdOutS ) ;
return false ;
}
output - > append ( stdOutS ) ;
return true ;
}
// Run a build process with merged stdout/stderr and qWarn about errors.
static bool runBuildProcess ( QProcess & proc ,
const QString & binary ,
const QStringList & args ,
int timeoutMS ,
bool ignoreNonNullExitCode ,
QString * output , QString * errorMessage )
{
const bool rc = runBuildProcessI ( proc , binary , args , timeoutMS , ignoreNonNullExitCode , output , errorMessage ) ;
if ( ! rc ) {
// Fail - reformat error.
QString cmd = binary ;
if ( ! args . isEmpty ( ) ) {
cmd + = QLatin1Char ( ' ' ) ;
cmd + = args . join ( QString ( QLatin1Char ( ' ' ) ) ) ;
}
* errorMessage =
QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" Error running '%1' in %2: %3 " ) .
arg ( cmd , proc . workingDirectory ( ) , * errorMessage ) ;
qWarning ( " %s " , qPrintable ( * errorMessage ) ) ;
}
return rc ;
}
2011-04-29 13:35:19 +02:00
bool BuildableHelperLibrary : : buildHelper ( const BuildHelperArguments & arguments ,
QString * log , QString * errorMessage )
2010-09-27 15:51:49 +02:00
{
const QChar newline = QLatin1Char ( ' \n ' ) ;
// Setup process
QProcess proc ;
2011-04-29 13:35:19 +02:00
proc . setEnvironment ( arguments . environment . toStringList ( ) ) ;
proc . setWorkingDirectory ( arguments . directory ) ;
2010-09-27 15:51:49 +02:00
proc . setProcessChannelMode ( QProcess : : MergedChannels ) ;
2011-04-29 13:35:19 +02:00
log - > append ( QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" Building helper '%1' in %2 \n " ) . arg ( arguments . helperName ,
arguments . directory ) ) ;
log - > append ( newline ) ;
2010-09-27 15:51:49 +02:00
2011-04-29 13:35:19 +02:00
const QString makeFullPath = arguments . environment . searchInPath ( arguments . makeCommand ) ;
if ( QFileInfo ( arguments . directory + QLatin1String ( " /Makefile " ) ) . exists ( ) ) {
2010-10-08 15:13:02 +02:00
if ( makeFullPath . isEmpty ( ) ) {
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::DebuggingHelperLibrary " ,
2011-04-29 13:35:19 +02:00
" %1 not found in PATH \n " ) . arg ( arguments . makeCommand ) ;
2010-10-08 15:13:02 +02:00
return false ;
2010-09-27 15:51:49 +02:00
}
2010-10-08 15:13:02 +02:00
const QString cleanTarget = QLatin1String ( " distclean " ) ;
2011-04-29 13:35:19 +02:00
log - > append ( QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
2010-10-08 15:13:02 +02:00
" Running %1 %2... \n " ) . arg ( makeFullPath , cleanTarget ) ) ;
2011-04-29 13:35:19 +02:00
if ( ! runBuildProcess ( proc , makeFullPath , QStringList ( cleanTarget ) , 30000 , true , log , errorMessage ) )
2010-10-08 15:13:02 +02:00
return false ;
2010-09-27 15:51:49 +02:00
}
2011-01-12 16:24:00 +01:00
QStringList qmakeArgs ;
2011-04-29 13:35:19 +02:00
if ( ! arguments . targetMode . isEmpty ( ) )
qmakeArgs < < arguments . targetMode ;
if ( ! arguments . mkspec . isEmpty ( ) )
2011-11-25 13:19:58 +01:00
qmakeArgs < < QLatin1String ( " -spec " ) < < arguments . mkspec . toUserOutput ( ) ;
2011-04-29 13:35:19 +02:00
qmakeArgs < < arguments . proFilename ;
qmakeArgs < < arguments . qmakeArguments ;
2011-02-10 15:24:49 +01:00
2011-04-29 13:35:19 +02:00
log - > append ( newline ) ;
log - > append ( QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
2011-11-25 13:19:58 +01:00
" Running %1 %2 ... \n " ) . arg ( arguments . qmakeCommand . toUserOutput ( ) ,
2012-01-04 17:34:08 +01:00
qmakeArgs . join ( QLatin1String ( " " ) ) ) ) ;
2011-02-10 15:24:49 +01:00
2011-11-25 13:19:58 +01:00
if ( ! runBuildProcess ( proc , arguments . qmakeCommand . toString ( ) , qmakeArgs , 30000 , false , log , errorMessage ) )
2010-10-08 15:13:02 +02:00
return false ;
2011-04-29 13:35:19 +02:00
log - > append ( newline ) ;
2010-10-08 15:13:02 +02:00
if ( makeFullPath . isEmpty ( ) ) {
2011-04-29 13:35:19 +02:00
* errorMessage = QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
" %1 not found in PATH \n " ) . arg ( arguments . makeCommand ) ;
2010-10-08 15:13:02 +02:00
return false ;
2010-09-27 15:51:49 +02:00
}
2011-04-29 13:35:19 +02:00
log - > append ( QCoreApplication : : translate ( " ProjectExplorer::BuildableHelperLibrary " ,
2012-01-04 17:34:08 +01:00
" Running %1 %2 ... \n " ) . arg ( makeFullPath , arguments . makeArguments . join ( QLatin1String ( " " ) ) ) ) ;
2011-04-29 14:07:14 +02:00
if ( ! runBuildProcess ( proc , makeFullPath , arguments . makeArguments , 120000 , false , log , errorMessage ) )
2010-10-08 15:13:02 +02:00
return false ;
return true ;
2010-09-27 15:51:49 +02:00
}
bool BuildableHelperLibrary : : getHelperFileInfoFor ( const QStringList & validBinaryFilenames ,
const QString & directory , QFileInfo * info )
{
if ( ! info )
return false ;
2012-11-28 20:44:03 +02:00
foreach ( const QString & binaryFilename , validBinaryFilenames ) {
2010-09-27 15:51:49 +02:00
info - > setFile ( directory + binaryFilename ) ;
if ( info - > exists ( ) )
return true ;
}
return false ;
}
2011-02-18 14:43:33 +01:00
QString BuildableHelperLibrary : : byInstallDataHelper ( const QString & sourcePath ,
const QStringList & sourceFileNames ,
2010-09-27 15:51:49 +02:00
const QStringList & installDirectories ,
2011-04-18 14:13:18 +02:00
const QStringList & validBinaryFilenames ,
bool acceptOutdatedHelper )
2010-09-27 15:51:49 +02:00
{
2011-02-18 14:43:33 +01:00
// find the latest change to the sources
QDateTime sourcesModified ;
2011-04-18 14:13:18 +02:00
if ( ! acceptOutdatedHelper ) {
foreach ( const QString & sourceFileName , sourceFileNames ) {
const QDateTime fileModified = QFileInfo ( sourcePath + sourceFileName ) . lastModified ( ) ;
if ( fileModified . isValid ( ) & & ( ! sourcesModified . isValid ( ) | | fileModified > sourcesModified ) )
sourcesModified = fileModified ;
}
2011-02-18 14:43:33 +01:00
}
2011-06-28 18:00:38 +02:00
// We pretend that the lastmodified of dumper.cpp is 5 minutes before what
// the file system says because afer a installation from the package the
// modified dates of dumper.cpp and the actual library are close to each
// other, but not deterministic in one direction.
2011-04-18 14:13:18 +02:00
if ( sourcesModified . isValid ( ) )
sourcesModified = sourcesModified . addSecs ( - 300 ) ;
2010-09-27 15:51:49 +02:00
// look for the newest helper library in the different locations
QString newestHelper ;
QDateTime newestHelperModified = sourcesModified ; // prevent using one that's older than the sources
QFileInfo fileInfo ;
2012-11-28 20:44:03 +02:00
foreach ( const QString & installDirectory , installDirectories ) {
2010-09-27 15:51:49 +02:00
if ( getHelperFileInfoFor ( validBinaryFilenames , installDirectory , & fileInfo ) ) {
2011-04-18 14:13:18 +02:00
if ( ! newestHelperModified . isValid ( )
| | ( fileInfo . lastModified ( ) > newestHelperModified ) ) {
2010-09-27 15:51:49 +02:00
newestHelper = fileInfo . filePath ( ) ;
newestHelperModified = fileInfo . lastModified ( ) ;
}
}
}
return newestHelper ;
}
} // namespace Utils