23bce6b169
A bug in the Windows C Runtime causes text mode pipes to drop newlines
sometimes. This bug was hidden because of another bug in rcc which
caused newlines to be redundantly duplicated. When the latter bug was
fixed (commit 53d5811b
) the former bug was exposed, causing invalid
vcxproj files to be generated. The Windows bug is described here:
https://connect.microsoft.com/VisualStudio/feedback/details/1902345
The workaround is to avoid text mode, and do the conversion of "\r\n"
to "\n" ourselves (which we were already doing anyway).
Change-Id: I792599a4cd7822f109fa921f02207fb1b144b1d1
Reviewed-by: Oswald Buddenhagen <oswald.buddenhagen@theqtcompany.com>
3373 lines
140 KiB
C++
3373 lines
140 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
** Contact: http://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the qmake application of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL21$
|
|
** 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 http://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at http://www.qt.io/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 or version 3 as published by the Free
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
** following information to ensure the GNU Lesser General Public License
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "makefile.h"
|
|
#include "option.h"
|
|
#include "cachekeys.h"
|
|
#include "meta.h"
|
|
|
|
#include <ioutils.h>
|
|
|
|
#include <qdir.h>
|
|
#include <qfile.h>
|
|
#include <qtextstream.h>
|
|
#include <qregexp.h>
|
|
#include <qhash.h>
|
|
#include <qdebug.h>
|
|
#include <qbuffer.h>
|
|
#include <qdatetime.h>
|
|
|
|
#if defined(Q_OS_UNIX)
|
|
#include <unistd.h>
|
|
#else
|
|
#include <io.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include <fcntl.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
using namespace QMakeInternal;
|
|
|
|
bool MakefileGenerator::canExecute(const QStringList &cmdline, int *a) const
|
|
{
|
|
int argv0 = -1;
|
|
for(int i = 0; i < cmdline.count(); ++i) {
|
|
if(!cmdline.at(i).contains('=')) {
|
|
argv0 = i;
|
|
break;
|
|
}
|
|
}
|
|
if(a)
|
|
*a = argv0;
|
|
if(argv0 != -1) {
|
|
const QString c = Option::normalizePath(cmdline.at(argv0));
|
|
if(exists(c))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString MakefileGenerator::mkdir_p_asstring(const QString &dir, bool escape) const
|
|
{
|
|
QString edir = escape ? escapeFilePath(Option::fixPathToTargetOS(dir, false, false)) : dir;
|
|
return "@" + makedir.arg(edir);
|
|
}
|
|
|
|
bool MakefileGenerator::mkdir(const QString &in_path) const
|
|
{
|
|
QString path = Option::normalizePath(in_path);
|
|
if(QFile::exists(path))
|
|
return true;
|
|
|
|
return QDir().mkpath(path);
|
|
}
|
|
|
|
// ** base makefile generator
|
|
MakefileGenerator::MakefileGenerator() :
|
|
no_io(false), project(0)
|
|
{
|
|
}
|
|
|
|
|
|
void
|
|
MakefileGenerator::verifyCompilers()
|
|
{
|
|
ProValueMap &v = project->variables();
|
|
ProStringList &quc = v["QMAKE_EXTRA_COMPILERS"];
|
|
for(int i = 0; i < quc.size(); ) {
|
|
bool error = false;
|
|
const ProString &comp = quc.at(i);
|
|
const ProKey okey(comp + ".output");
|
|
if (v[okey].isEmpty()) {
|
|
const ProKey ofkey(comp + ".output_function");
|
|
if (!v[ofkey].isEmpty()) {
|
|
v[okey].append("${QMAKE_FUNC_FILE_IN_" + v[ofkey].first() + "}");
|
|
} else {
|
|
error = true;
|
|
warn_msg(WarnLogic, "Compiler: %s: No output file specified", comp.toLatin1().constData());
|
|
}
|
|
} else if (v[ProKey(comp + ".input")].isEmpty()) {
|
|
error = true;
|
|
warn_msg(WarnLogic, "Compiler: %s: No input variable specified", comp.toLatin1().constData());
|
|
}
|
|
if(error)
|
|
quc.removeAt(i);
|
|
else
|
|
++i;
|
|
}
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::initOutPaths()
|
|
{
|
|
ProValueMap &v = project->variables();
|
|
//for shadow builds
|
|
if(!v.contains("QMAKE_ABSOLUTE_SOURCE_PATH")) {
|
|
if (Option::globals->do_cache && !project->cacheFile().isEmpty() &&
|
|
v.contains("QMAKE_ABSOLUTE_SOURCE_ROOT")) {
|
|
QString root = v["QMAKE_ABSOLUTE_SOURCE_ROOT"].first().toQString();
|
|
root = QDir::fromNativeSeparators(root);
|
|
if(!root.isEmpty()) {
|
|
QFileInfo fi = fileInfo(project->cacheFile());
|
|
if(!fi.makeAbsolute()) {
|
|
QString cache_r = fi.path(), pwd = Option::output_dir;
|
|
if(pwd.startsWith(cache_r) && !pwd.startsWith(root)) {
|
|
pwd = root + pwd.mid(cache_r.length());
|
|
if(exists(pwd))
|
|
v.insert("QMAKE_ABSOLUTE_SOURCE_PATH", ProStringList(pwd));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if(!v["QMAKE_ABSOLUTE_SOURCE_PATH"].isEmpty()) {
|
|
ProString &asp = v["QMAKE_ABSOLUTE_SOURCE_PATH"].first();
|
|
asp = QDir::fromNativeSeparators(asp.toQString());
|
|
if(asp.isEmpty() || asp == Option::output_dir) //if they're the same, why bother?
|
|
v["QMAKE_ABSOLUTE_SOURCE_PATH"].clear();
|
|
}
|
|
|
|
//some builtin directories
|
|
if(project->isEmpty("PRECOMPILED_DIR") && !project->isEmpty("OBJECTS_DIR"))
|
|
v["PRECOMPILED_DIR"] = v["OBJECTS_DIR"];
|
|
static const char * const dirs[] = { "OBJECTS_DIR", "DESTDIR",
|
|
"SUBLIBS_DIR", "DLLDESTDIR",
|
|
"PRECOMPILED_DIR", 0 };
|
|
for (int x = 0; dirs[x]; x++) {
|
|
const ProKey dkey(dirs[x]);
|
|
if (v[dkey].isEmpty())
|
|
continue;
|
|
const ProString orig_path = v[dkey].first();
|
|
|
|
ProString &pathRef = v[dkey].first();
|
|
pathRef = fileFixify(pathRef.toQString(), FileFixifyFromOutdir);
|
|
|
|
#ifdef Q_OS_WIN
|
|
// We don't want to add a separator for DLLDESTDIR on Windows (###why?)
|
|
if(!(dirs[x] == "DLLDESTDIR"))
|
|
#endif
|
|
{
|
|
if(!pathRef.endsWith(Option::dir_sep))
|
|
pathRef += Option::dir_sep;
|
|
}
|
|
|
|
if (noIO() || (project->first("TEMPLATE") == "subdirs"))
|
|
continue;
|
|
|
|
QString path = project->first(dkey).toQString(); //not to be changed any further
|
|
path = fileFixify(path, FileFixifyBackwards);
|
|
debug_msg(3, "Fixed output_dir %s (%s) into %s", dirs[x],
|
|
orig_path.toLatin1().constData(), path.toLatin1().constData());
|
|
if(!mkdir(path))
|
|
warn_msg(WarnLogic, "%s: Cannot access directory '%s'", dirs[x],
|
|
path.toLatin1().constData());
|
|
}
|
|
|
|
//out paths from the extra compilers
|
|
const ProStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
|
|
for (ProStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
|
|
QString tmp_out = project->first(ProKey(*it + ".output")).toQString();
|
|
if(tmp_out.isEmpty())
|
|
continue;
|
|
const ProStringList &tmp = project->values(ProKey(*it + ".input"));
|
|
for (ProStringList::ConstIterator it2 = tmp.begin(); it2 != tmp.end(); ++it2) {
|
|
ProStringList &inputs = project->values((*it2).toKey());
|
|
for (ProStringList::Iterator input = inputs.begin(); input != inputs.end(); ++input) {
|
|
QString finp = fileFixify((*input).toQString(), FileFixifyFromOutdir);
|
|
*input = ProString(finp);
|
|
QString path = replaceExtraCompilerVariables(tmp_out, finp, QString(), NoShell);
|
|
path = Option::normalizePath(path);
|
|
int slash = path.lastIndexOf('/');
|
|
if(slash != -1) {
|
|
path = path.left(slash);
|
|
// Make out path only if it does not contain makefile variables
|
|
if(!path.contains("${"))
|
|
if(path != "." &&
|
|
!mkdir(fileFixify(path, FileFixifyBackwards)))
|
|
warn_msg(WarnLogic, "%s: Cannot access directory '%s'",
|
|
(*it).toLatin1().constData(), path.toLatin1().constData());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!v["DESTDIR"].isEmpty()) {
|
|
QDir d(v["DESTDIR"].first().toQString());
|
|
if (Option::normalizePath(d.absolutePath()) == Option::normalizePath(Option::output_dir))
|
|
v.remove("DESTDIR");
|
|
}
|
|
}
|
|
|
|
QMakeProject
|
|
*MakefileGenerator::projectFile() const
|
|
{
|
|
return project;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::setProjectFile(QMakeProject *p)
|
|
{
|
|
if(project)
|
|
return;
|
|
project = p;
|
|
if (project->isActiveConfig("win32"))
|
|
target_mode = TARG_WIN_MODE;
|
|
else if (project->isActiveConfig("mac"))
|
|
target_mode = TARG_MAC_MODE;
|
|
else
|
|
target_mode = TARG_UNIX_MODE;
|
|
init();
|
|
bool linkPrl = (Option::qmake_mode == Option::QMAKE_GENERATE_MAKEFILE)
|
|
&& project->isActiveConfig("link_prl");
|
|
bool mergeLflags = !project->isActiveConfig("no_smart_library_merge")
|
|
&& !project->isActiveConfig("no_lflags_merge");
|
|
findLibraries(linkPrl, mergeLflags);
|
|
}
|
|
|
|
ProStringList
|
|
MakefileGenerator::findFilesInVPATH(ProStringList l, uchar flags, const QString &vpath_var)
|
|
{
|
|
ProStringList vpath;
|
|
const ProValueMap &v = project->variables();
|
|
for(int val_it = 0; val_it < l.count(); ) {
|
|
bool remove_file = false;
|
|
ProString &val = l[val_it];
|
|
if(!val.isEmpty()) {
|
|
QString qval = val.toQString();
|
|
QString file = fixEnvVariables(qval);
|
|
if (file.isEmpty()) {
|
|
++val_it;
|
|
continue;
|
|
}
|
|
if(!(flags & VPATH_NoFixify))
|
|
file = fileFixify(file, FileFixifyBackwards);
|
|
|
|
if(exists(file)) {
|
|
++val_it;
|
|
continue;
|
|
}
|
|
bool found = false;
|
|
if (QDir::isRelativePath(qval)) {
|
|
if(vpath.isEmpty()) {
|
|
if(!vpath_var.isEmpty())
|
|
vpath = v[ProKey(vpath_var)];
|
|
vpath += v["VPATH"] + v["QMAKE_ABSOLUTE_SOURCE_PATH"];
|
|
if(Option::output_dir != qmake_getpwd())
|
|
vpath << Option::output_dir;
|
|
}
|
|
for (ProStringList::Iterator vpath_it = vpath.begin();
|
|
vpath_it != vpath.end(); ++vpath_it) {
|
|
QString real_dir = Option::normalizePath((*vpath_it).toQString());
|
|
if (exists(real_dir + '/' + val)) {
|
|
ProString dir = (*vpath_it);
|
|
if(!dir.endsWith(Option::dir_sep))
|
|
dir += Option::dir_sep;
|
|
val = dir + val;
|
|
if(!(flags & VPATH_NoFixify))
|
|
val = fileFixify(val.toQString());
|
|
found = true;
|
|
debug_msg(1, "Found file through vpath %s -> %s",
|
|
file.toLatin1().constData(), val.toLatin1().constData());
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(!found) {
|
|
QString dir, regex = val.toQString(), real_dir;
|
|
if(regex.lastIndexOf(Option::dir_sep) != -1) {
|
|
dir = regex.left(regex.lastIndexOf(Option::dir_sep) + 1);
|
|
real_dir = dir;
|
|
if(!(flags & VPATH_NoFixify))
|
|
real_dir = fileFixify(real_dir, FileFixifyBackwards) + '/';
|
|
regex.remove(0, dir.length());
|
|
}
|
|
if(real_dir.isEmpty() || exists(real_dir)) {
|
|
QStringList files = QDir(real_dir).entryList(QStringList(regex),
|
|
QDir::NoDotAndDotDot | QDir::AllEntries);
|
|
if(files.isEmpty()) {
|
|
debug_msg(1, "%s:%d Failure to find %s in vpath (%s)",
|
|
__FILE__, __LINE__,
|
|
val.toLatin1().constData(), vpath.join("::").toLatin1().constData());
|
|
if(flags & VPATH_RemoveMissingFiles)
|
|
remove_file = true;
|
|
else if(flags & VPATH_WarnMissingFiles)
|
|
warn_msg(WarnLogic, "Failure to find: %s", val.toLatin1().constData());
|
|
} else {
|
|
l.removeAt(val_it);
|
|
QString a;
|
|
for(int i = (int)files.count()-1; i >= 0; i--) {
|
|
a = real_dir + files[i];
|
|
if(!(flags & VPATH_NoFixify))
|
|
a = fileFixify(a);
|
|
l.insert(val_it, a);
|
|
}
|
|
}
|
|
} else {
|
|
debug_msg(1, "%s:%d Cannot match %s%s, as %s does not exist.",
|
|
__FILE__, __LINE__, real_dir.toLatin1().constData(),
|
|
regex.toLatin1().constData(), real_dir.toLatin1().constData());
|
|
if(flags & VPATH_RemoveMissingFiles)
|
|
remove_file = true;
|
|
else if(flags & VPATH_WarnMissingFiles)
|
|
warn_msg(WarnLogic, "Failure to find: %s", val.toLatin1().constData());
|
|
}
|
|
}
|
|
}
|
|
if(remove_file)
|
|
l.removeAt(val_it);
|
|
else
|
|
++val_it;
|
|
}
|
|
return l;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::initCompiler(const MakefileGenerator::Compiler &comp)
|
|
{
|
|
ProValueMap &v = project->variables();
|
|
ProStringList &l = v[ProKey(comp.variable_in)];
|
|
// find all the relevant file inputs
|
|
if(!init_compiler_already.contains(comp.variable_in)) {
|
|
init_compiler_already.insert(comp.variable_in, true);
|
|
if(!noIO())
|
|
l = findFilesInVPATH(l, (comp.flags & Compiler::CompilerRemoveNoExist) ?
|
|
VPATH_RemoveMissingFiles : VPATH_WarnMissingFiles, "VPATH_" + comp.variable_in);
|
|
}
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::init()
|
|
{
|
|
verifyCompilers();
|
|
initOutPaths();
|
|
|
|
ProValueMap &v = project->variables();
|
|
|
|
v["QMAKE_BUILTIN_COMPILERS"] = ProStringList() << "C" << "CXX";
|
|
|
|
v["QMAKE_LANGUAGE_C"] = ProString("c");
|
|
v["QMAKE_LANGUAGE_CXX"] = ProString("c++");
|
|
v["QMAKE_LANGUAGE_OBJC"] = ProString("objective-c");
|
|
v["QMAKE_LANGUAGE_OBJCXX"] = ProString("objective-c++");
|
|
|
|
if (v["TARGET"].isEmpty())
|
|
warn_msg(WarnLogic, "TARGET is empty");
|
|
|
|
makedir = v["QMAKE_MKDIR_CMD"].join(' ');
|
|
chkexists = v["QMAKE_CHK_EXISTS"].join(' ');
|
|
if (makedir.isEmpty()) { // Backwards compat with Qt < 5.0.2 specs
|
|
if (isWindowsShell()) {
|
|
makedir = "if not exist %1 mkdir %1 & if not exist %1 exit 1";
|
|
chkexists = "if not exist %1";
|
|
} else {
|
|
makedir = "test -d %1 || mkdir -p %1";
|
|
chkexists = "test -e %1 ||";
|
|
}
|
|
}
|
|
|
|
if (v["QMAKE_CC_O_FLAG"].isEmpty())
|
|
v["QMAKE_CC_O_FLAG"].append("-o ");
|
|
|
|
if (v["QMAKE_LINK_O_FLAG"].isEmpty())
|
|
v["QMAKE_LINK_O_FLAG"].append("-o ");
|
|
|
|
setSystemIncludes(v["QMAKE_DEFAULT_INCDIRS"]);
|
|
|
|
ProStringList &incs = project->values("INCLUDEPATH");
|
|
if (!project->isActiveConfig("no_include_pwd")) {
|
|
if (Option::output_dir != qmake_getpwd()) {
|
|
// Pretend that the build dir is the source dir for #include purposes,
|
|
// consistently with the "transparent shadow builds" strategy. This is
|
|
// also consistent with #include "foo.h" falling back to #include <foo.h>
|
|
// behavior if it doesn't find the file in the source dir.
|
|
incs.prepend(Option::output_dir);
|
|
}
|
|
// This makes #include <foo.h> work if the header lives in the source dir.
|
|
// The benefit of that is questionable, as generally the user should use the
|
|
// correct include style, and extra compilers that put stuff in the source dir
|
|
// should add the dir themselves.
|
|
// More importantly, it makes #include "foo.h" work with MSVC when shadow-building,
|
|
// as this compiler looks files up relative to %CD%, not the source file's parent.
|
|
incs.prepend(qmake_getpwd());
|
|
}
|
|
incs.append(project->specDir());
|
|
|
|
const char * const cacheKeys[] = { "_QMAKE_STASH_", "_QMAKE_SUPER_CACHE_", 0 };
|
|
for (int i = 0; cacheKeys[i]; ++i) {
|
|
if (v[cacheKeys[i]].isEmpty())
|
|
continue;
|
|
const ProString &file = v[cacheKeys[i]].first();
|
|
if (file.isEmpty())
|
|
continue;
|
|
|
|
QFileInfo fi(fileInfo(file.toQString()));
|
|
|
|
// If the file lives in the output dir we treat it as 'owned' by
|
|
// the current project, so it should be distcleaned by it as well.
|
|
if (fi.path() == Option::output_dir)
|
|
v["QMAKE_DISTCLEAN"].append(fi.fileName());
|
|
}
|
|
|
|
ProStringList &quc = v["QMAKE_EXTRA_COMPILERS"];
|
|
|
|
//make sure the COMPILERS are in the correct input/output chain order
|
|
for(int comp_out = 0, jump_count = 0; comp_out < quc.size(); ++comp_out) {
|
|
continue_compiler_chain:
|
|
if(jump_count > quc.size()) //just to avoid an infinite loop here
|
|
break;
|
|
const ProKey vokey(quc.at(comp_out) + ".variable_out");
|
|
if (v.contains(vokey)) {
|
|
const ProStringList &outputs = v.value(vokey);
|
|
for(int out = 0; out < outputs.size(); ++out) {
|
|
for(int comp_in = 0; comp_in < quc.size(); ++comp_in) {
|
|
if(comp_in == comp_out)
|
|
continue;
|
|
const ProKey ikey(quc.at(comp_in) + ".input");
|
|
if (v.contains(ikey)) {
|
|
const ProStringList &inputs = v.value(ikey);
|
|
for(int in = 0; in < inputs.size(); ++in) {
|
|
if(inputs.at(in) == outputs.at(out) && comp_out > comp_in) {
|
|
++jump_count;
|
|
//move comp_out to comp_in and continue the compiler chain
|
|
// quc.move(comp_out, comp_in);
|
|
quc.insert(comp_in, quc.value(comp_out));
|
|
// comp_out > comp_in, so the insertion did move everything up
|
|
quc.remove(comp_out + 1);
|
|
comp_out = comp_in;
|
|
goto continue_compiler_chain;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!project->isEmpty("QMAKE_SUBSTITUTES")) {
|
|
const ProStringList &subs = v["QMAKE_SUBSTITUTES"];
|
|
for(int i = 0; i < subs.size(); ++i) {
|
|
QString sub = subs.at(i).toQString();
|
|
QString inn = sub + ".input", outn = sub + ".output";
|
|
const ProKey innkey(inn), outnkey(outn);
|
|
if (v.contains(innkey) || v.contains(outnkey)) {
|
|
if (!v.contains(innkey) || !v.contains(outnkey)) {
|
|
warn_msg(WarnLogic, "Substitute '%s' has only one of .input and .output",
|
|
sub.toLatin1().constData());
|
|
continue;
|
|
}
|
|
const ProStringList &tinn = v[innkey], &toutn = v[outnkey];
|
|
if (tinn.length() != 1) {
|
|
warn_msg(WarnLogic, "Substitute '%s.input' does not have exactly one value",
|
|
sub.toLatin1().constData());
|
|
continue;
|
|
}
|
|
if (toutn.length() != 1) {
|
|
warn_msg(WarnLogic, "Substitute '%s.output' does not have exactly one value",
|
|
sub.toLatin1().constData());
|
|
continue;
|
|
}
|
|
inn = fileFixify(tinn.first().toQString(), FileFixifyToIndir);
|
|
outn = fileFixify(toutn.first().toQString(), FileFixifyBackwards);
|
|
} else {
|
|
inn = fileFixify(sub, FileFixifyToIndir);
|
|
if (!QFile::exists(inn)) {
|
|
// random insanity for backwards compat: .in file specified with absolute out dir
|
|
inn = fileFixify(sub);
|
|
}
|
|
if(!inn.endsWith(".in")) {
|
|
warn_msg(WarnLogic, "Substitute '%s' does not end with '.in'",
|
|
inn.toLatin1().constData());
|
|
continue;
|
|
}
|
|
outn = fileFixify(inn.left(inn.length() - 3), FileFixifyBackwards);
|
|
}
|
|
|
|
const ProKey confign(sub + ".CONFIG");
|
|
bool verbatim = false;
|
|
if (v.contains(confign))
|
|
verbatim = v[confign].contains(QLatin1String("verbatim"));
|
|
|
|
QFile in(inn);
|
|
if (in.open(QFile::ReadOnly)) {
|
|
QByteArray contentBytes;
|
|
if (verbatim) {
|
|
contentBytes = in.readAll();
|
|
} else {
|
|
QString contents;
|
|
QStack<int> state;
|
|
enum { IN_CONDITION, MET_CONDITION, PENDING_CONDITION };
|
|
for (int count = 1; !in.atEnd(); ++count) {
|
|
QString line = QString::fromUtf8(in.readLine());
|
|
if (line.startsWith("!!IF ")) {
|
|
if (state.isEmpty() || state.top() == IN_CONDITION) {
|
|
QString test = line.mid(5, line.length()-(5+1));
|
|
if (project->test(test, inn, count))
|
|
state.push(IN_CONDITION);
|
|
else
|
|
state.push(PENDING_CONDITION);
|
|
} else {
|
|
state.push(MET_CONDITION);
|
|
}
|
|
} else if (line.startsWith("!!ELIF ")) {
|
|
if (state.isEmpty()) {
|
|
warn_msg(WarnLogic, "(%s:%d): Unexpected else condition",
|
|
in.fileName().toLatin1().constData(), count);
|
|
} else if (state.top() == PENDING_CONDITION) {
|
|
QString test = line.mid(7, line.length()-(7+1));
|
|
if (project->test(test, inn, count)) {
|
|
state.pop();
|
|
state.push(IN_CONDITION);
|
|
}
|
|
} else if (state.top() == IN_CONDITION) {
|
|
state.pop();
|
|
state.push(MET_CONDITION);
|
|
}
|
|
} else if (line.startsWith("!!ELSE")) {
|
|
if (state.isEmpty()) {
|
|
warn_msg(WarnLogic, "(%s:%d): Unexpected else condition",
|
|
in.fileName().toLatin1().constData(), count);
|
|
} else if (state.top() == PENDING_CONDITION) {
|
|
state.pop();
|
|
state.push(IN_CONDITION);
|
|
} else if (state.top() == IN_CONDITION) {
|
|
state.pop();
|
|
state.push(MET_CONDITION);
|
|
}
|
|
} else if (line.startsWith("!!ENDIF")) {
|
|
if (state.isEmpty())
|
|
warn_msg(WarnLogic, "(%s:%d): Unexpected endif",
|
|
in.fileName().toLatin1().constData(), count);
|
|
else
|
|
state.pop();
|
|
} else if (state.isEmpty() || state.top() == IN_CONDITION) {
|
|
contents += project->expand(line, in.fileName(), count);
|
|
}
|
|
}
|
|
contentBytes = contents.toUtf8();
|
|
}
|
|
QFile out(outn);
|
|
QFileInfo outfi(out);
|
|
if (out.exists() && out.open(QFile::ReadOnly)) {
|
|
QByteArray old = out.readAll();
|
|
if (contentBytes == old) {
|
|
v["QMAKE_INTERNAL_INCLUDED_FILES"].append(in.fileName());
|
|
v["QMAKE_DISTCLEAN"].append(outfi.absoluteFilePath());
|
|
continue;
|
|
}
|
|
out.close();
|
|
if(!out.remove()) {
|
|
warn_msg(WarnLogic, "Cannot clear substitute '%s'",
|
|
out.fileName().toLatin1().constData());
|
|
continue;
|
|
}
|
|
}
|
|
mkdir(outfi.absolutePath());
|
|
if(out.open(QFile::WriteOnly)) {
|
|
v["QMAKE_INTERNAL_INCLUDED_FILES"].append(in.fileName());
|
|
v["QMAKE_DISTCLEAN"].append(outfi.absoluteFilePath());
|
|
out.write(contentBytes);
|
|
} else {
|
|
warn_msg(WarnLogic, "Cannot open substitute for output '%s'",
|
|
out.fileName().toLatin1().constData());
|
|
}
|
|
} else {
|
|
warn_msg(WarnLogic, "Cannot open substitute for input '%s'",
|
|
in.fileName().toLatin1().constData());
|
|
}
|
|
}
|
|
}
|
|
|
|
int x;
|
|
|
|
//build up a list of compilers
|
|
QList<Compiler> compilers;
|
|
{
|
|
const char *builtins[] = { "OBJECTS", "SOURCES", "PRECOMPILED_HEADER", 0 };
|
|
for(x = 0; builtins[x]; ++x) {
|
|
Compiler compiler;
|
|
compiler.variable_in = builtins[x];
|
|
compiler.flags = Compiler::CompilerBuiltin;
|
|
compiler.type = QMakeSourceFileInfo::TYPE_C;
|
|
if(!strcmp(builtins[x], "OBJECTS"))
|
|
compiler.flags |= Compiler::CompilerNoCheckDeps;
|
|
compilers.append(compiler);
|
|
}
|
|
for (ProStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
|
|
const ProStringList &inputs = v[ProKey(*it + ".input")];
|
|
for(x = 0; x < inputs.size(); ++x) {
|
|
Compiler compiler;
|
|
compiler.variable_in = inputs.at(x).toQString();
|
|
compiler.flags = Compiler::CompilerNoFlags;
|
|
const ProStringList &config = v[ProKey(*it + ".CONFIG")];
|
|
if (config.indexOf("ignore_no_exist") != -1)
|
|
compiler.flags |= Compiler::CompilerRemoveNoExist;
|
|
if (config.indexOf("no_dependencies") != -1)
|
|
compiler.flags |= Compiler::CompilerNoCheckDeps;
|
|
if (config.indexOf("add_inputs_as_makefile_deps") != -1)
|
|
compiler.flags |= Compiler::CompilerAddInputsAsMakefileDeps;
|
|
|
|
const ProKey dkey(*it + ".dependency_type");
|
|
ProString dep_type;
|
|
if (!project->isEmpty(dkey))
|
|
dep_type = project->first(dkey);
|
|
if (dep_type.isEmpty())
|
|
compiler.type = QMakeSourceFileInfo::TYPE_UNKNOWN;
|
|
else if(dep_type == "TYPE_UI")
|
|
compiler.type = QMakeSourceFileInfo::TYPE_UI;
|
|
else
|
|
compiler.type = QMakeSourceFileInfo::TYPE_C;
|
|
compilers.append(compiler);
|
|
}
|
|
}
|
|
}
|
|
{ //do the path fixifying
|
|
ProStringList paths;
|
|
for(x = 0; x < compilers.count(); ++x) {
|
|
if(!paths.contains(compilers.at(x).variable_in))
|
|
paths << compilers.at(x).variable_in;
|
|
}
|
|
paths << "INCLUDEPATH" << "QMAKE_INTERNAL_INCLUDED_FILES" << "PRECOMPILED_HEADER";
|
|
for(int y = 0; y < paths.count(); y++) {
|
|
ProStringList &l = v[paths[y].toKey()];
|
|
for (ProStringList::Iterator it = l.begin(); it != l.end(); ++it) {
|
|
if((*it).isEmpty())
|
|
continue;
|
|
QString fn = (*it).toQString();
|
|
if (exists(fn))
|
|
(*it) = fileFixify(fn);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(noIO() || !doDepends() || project->isActiveConfig("GNUmake"))
|
|
QMakeSourceFileInfo::setDependencyMode(QMakeSourceFileInfo::NonRecursive);
|
|
for(x = 0; x < compilers.count(); ++x)
|
|
initCompiler(compilers.at(x));
|
|
|
|
//merge actual compiler outputs into their variable_out. This is done last so that
|
|
//files are already properly fixified.
|
|
for (ProStringList::Iterator it = quc.begin(); it != quc.end(); ++it) {
|
|
const ProKey ikey(*it + ".input");
|
|
const ProKey vokey(*it + ".variable_out");
|
|
const ProStringList &config = project->values(ProKey(*it + ".CONFIG"));
|
|
const ProString &tmp_out = project->first(ProKey(*it + ".output"));
|
|
if(tmp_out.isEmpty())
|
|
continue;
|
|
if (config.indexOf("combine") != -1) {
|
|
const ProStringList &compilerInputs = project->values(ikey);
|
|
// Don't generate compiler output if it doesn't have input.
|
|
if (compilerInputs.isEmpty() || project->values(compilerInputs.first().toKey()).isEmpty())
|
|
continue;
|
|
if(tmp_out.indexOf("$") == -1) {
|
|
if(!verifyExtraCompiler((*it), QString())) //verify
|
|
continue;
|
|
QString out = fileFixify(tmp_out.toQString(), FileFixifyFromOutdir);
|
|
bool pre_dep = (config.indexOf("target_predeps") != -1);
|
|
if (v.contains(vokey)) {
|
|
const ProStringList &var_out = v.value(vokey);
|
|
for(int i = 0; i < var_out.size(); ++i) {
|
|
ProKey v = var_out.at(i).toKey();
|
|
if(v == QLatin1String("SOURCES"))
|
|
v = "GENERATED_SOURCES";
|
|
else if(v == QLatin1String("OBJECTS"))
|
|
pre_dep = false;
|
|
ProStringList &list = project->values(v);
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
} else if (config.indexOf("no_link") == -1) {
|
|
ProStringList &list = project->values("OBJECTS");
|
|
pre_dep = false;
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
} else {
|
|
ProStringList &list = project->values("UNUSED_SOURCES");
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
if(pre_dep) {
|
|
ProStringList &list = project->values("PRE_TARGETDEPS");
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
}
|
|
} else {
|
|
const ProStringList &tmp = project->values(ikey);
|
|
for (ProStringList::ConstIterator it2 = tmp.begin(); it2 != tmp.end(); ++it2) {
|
|
const ProStringList &inputs = project->values((*it2).toKey());
|
|
for (ProStringList::ConstIterator input = inputs.constBegin(); input != inputs.constEnd(); ++input) {
|
|
if((*input).isEmpty())
|
|
continue;
|
|
QString inpf = (*input).toQString();
|
|
if (!verifyExtraCompiler((*it).toQString(), inpf)) //verify
|
|
continue;
|
|
QString out = replaceExtraCompilerVariables(tmp_out.toQString(), inpf, QString(), NoShell);
|
|
out = fileFixify(out, FileFixifyFromOutdir);
|
|
bool pre_dep = (config.indexOf("target_predeps") != -1);
|
|
if (v.contains(vokey)) {
|
|
const ProStringList &var_out = project->values(vokey);
|
|
for(int i = 0; i < var_out.size(); ++i) {
|
|
ProKey v = var_out.at(i).toKey();
|
|
if(v == QLatin1String("SOURCES"))
|
|
v = "GENERATED_SOURCES";
|
|
else if(v == QLatin1String("OBJECTS"))
|
|
pre_dep = false;
|
|
ProStringList &list = project->values(v);
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
} else if (config.indexOf("no_link") == -1) {
|
|
pre_dep = false;
|
|
ProStringList &list = project->values("OBJECTS");
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
} else {
|
|
ProStringList &list = project->values("UNUSED_SOURCES");
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
if(pre_dep) {
|
|
ProStringList &list = project->values("PRE_TARGETDEPS");
|
|
if(!list.contains(out))
|
|
list.append(out);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//handle dependencies
|
|
depHeuristicsCache.clear();
|
|
if(!noIO()) {
|
|
// dependency paths
|
|
ProStringList incDirs = v["DEPENDPATH"] + v["QMAKE_ABSOLUTE_SOURCE_PATH"];
|
|
if(project->isActiveConfig("depend_includepath"))
|
|
incDirs += v["INCLUDEPATH"];
|
|
QList<QMakeLocalFileName> deplist;
|
|
for (ProStringList::Iterator it = incDirs.begin(); it != incDirs.end(); ++it)
|
|
deplist.append(QMakeLocalFileName((*it).toQString()));
|
|
QMakeSourceFileInfo::setDependencyPaths(deplist);
|
|
debug_msg(1, "Dependency Directories: %s", incDirs.join(" :: ").toLatin1().constData());
|
|
//cache info
|
|
if(project->isActiveConfig("qmake_cache")) {
|
|
QString cache_file;
|
|
if(!project->isEmpty("QMAKE_INTERNAL_CACHE_FILE")) {
|
|
cache_file = QDir::fromNativeSeparators(project->first("QMAKE_INTERNAL_CACHE_FILE").toQString());
|
|
} else {
|
|
cache_file = ".qmake.internal.cache";
|
|
if(project->isActiveConfig("build_pass"))
|
|
cache_file += ".BUILD." + project->first("BUILD_PASS");
|
|
}
|
|
if(cache_file.indexOf('/') == -1)
|
|
cache_file.prepend(Option::output_dir + '/');
|
|
QMakeSourceFileInfo::setCacheFile(cache_file);
|
|
}
|
|
|
|
//add to dependency engine
|
|
for(x = 0; x < compilers.count(); ++x) {
|
|
const MakefileGenerator::Compiler &comp = compilers.at(x);
|
|
if(!(comp.flags & Compiler::CompilerNoCheckDeps)) {
|
|
const ProKey ikey(comp.variable_in);
|
|
addSourceFiles(v[ikey], QMakeSourceFileInfo::SEEK_DEPS,
|
|
(QMakeSourceFileInfo::SourceFileType)comp.type);
|
|
|
|
if (comp.flags & Compiler::CompilerAddInputsAsMakefileDeps) {
|
|
ProStringList &l = v[ikey];
|
|
for (int i=0; i < l.size(); ++i) {
|
|
if(v["QMAKE_INTERNAL_INCLUDED_FILES"].indexOf(l.at(i)) == -1)
|
|
v["QMAKE_INTERNAL_INCLUDED_FILES"].append(l.at(i));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
processSources(); //remove anything in SOURCES which is included (thus it need not be linked in)
|
|
|
|
//all sources and generated sources must be turned into objects at some point (the one builtin compiler)
|
|
v["OBJECTS"] += createObjectList(v["SOURCES"]) + createObjectList(v["GENERATED_SOURCES"]);
|
|
|
|
//Translation files
|
|
if(!project->isEmpty("TRANSLATIONS")) {
|
|
ProStringList &trf = project->values("TRANSLATIONS");
|
|
for (ProStringList::Iterator it = trf.begin(); it != trf.end(); ++it)
|
|
(*it) = Option::fixPathToTargetOS((*it).toQString());
|
|
}
|
|
|
|
//fix up the target deps
|
|
static const char * const fixpaths[] = { "PRE_TARGETDEPS", "POST_TARGETDEPS", 0 };
|
|
for (int path = 0; fixpaths[path]; path++) {
|
|
ProStringList &l = v[fixpaths[path]];
|
|
for (ProStringList::Iterator val_it = l.begin(); val_it != l.end(); ++val_it) {
|
|
if(!(*val_it).isEmpty())
|
|
(*val_it) = Option::fixPathToTargetOS((*val_it).toQString(), false, false);
|
|
}
|
|
}
|
|
|
|
//extra depends
|
|
if(!project->isEmpty("DEPENDS")) {
|
|
ProStringList &l = v["DEPENDS"];
|
|
for (ProStringList::Iterator it = l.begin(); it != l.end(); ++it) {
|
|
const ProStringList &files = v[ProKey(*it + ".file")] + v[ProKey(*it + ".files")]; //why do I support such evil things?
|
|
for (ProStringList::ConstIterator file_it = files.begin(); file_it != files.end(); ++file_it) {
|
|
QStringList &out_deps = findDependencies((*file_it).toQString());
|
|
const ProStringList &in_deps = v[ProKey(*it + ".depends")]; //even more evilness..
|
|
for (ProStringList::ConstIterator dep_it = in_deps.begin(); dep_it != in_deps.end(); ++dep_it) {
|
|
QString dep = (*dep_it).toQString();
|
|
if (exists(dep)) {
|
|
out_deps.append(dep);
|
|
} else {
|
|
QString dir, regex = Option::normalizePath(dep);
|
|
if (regex.lastIndexOf('/') != -1) {
|
|
dir = regex.left(regex.lastIndexOf('/') + 1);
|
|
regex.remove(0, dir.length());
|
|
}
|
|
QStringList files = QDir(dir).entryList(QStringList(regex));
|
|
if(files.isEmpty()) {
|
|
warn_msg(WarnLogic, "Dependency for [%s]: Not found %s", (*file_it).toLatin1().constData(),
|
|
dep.toLatin1().constData());
|
|
} else {
|
|
for(int i = 0; i < files.count(); i++)
|
|
out_deps.append(dir + files[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// escape qmake command
|
|
project->values("QMAKE_QMAKE") =
|
|
ProStringList(escapeFilePath(Option::fixPathToTargetOS(Option::globals->qmake_abslocation, false)));
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::processPrlFile(QString &file)
|
|
{
|
|
bool try_replace_file = false;
|
|
QString f = fileFixify(file, FileFixifyBackwards);
|
|
QString meta_file = QMakeMetaInfo::findLib(f);
|
|
if (!meta_file.isEmpty()) {
|
|
try_replace_file = true;
|
|
} else {
|
|
QString tmp = f;
|
|
int ext = tmp.lastIndexOf('.');
|
|
if(ext != -1)
|
|
tmp = tmp.left(ext);
|
|
meta_file = QMakeMetaInfo::findLib(tmp);
|
|
}
|
|
if (meta_file.isEmpty())
|
|
return false;
|
|
QMakeMetaInfo libinfo(project);
|
|
debug_msg(1, "Processing PRL file: %s", meta_file.toLatin1().constData());
|
|
if (!libinfo.readLib(meta_file)) {
|
|
fprintf(stderr, "Error processing meta file %s\n", meta_file.toLatin1().constData());
|
|
return false;
|
|
}
|
|
if (project->isActiveConfig("no_read_prl_" + libinfo.type().toLower())) {
|
|
debug_msg(2, "Ignored meta file %s [%s]",
|
|
meta_file.toLatin1().constData(), libinfo.type().toLatin1().constData());
|
|
return false;
|
|
}
|
|
project->values("QMAKE_CURRENT_PRL_LIBS") = libinfo.values("QMAKE_PRL_LIBS");
|
|
ProStringList &defs = project->values("DEFINES");
|
|
const ProStringList &prl_defs = project->values("PRL_EXPORT_DEFINES");
|
|
foreach (const ProString &def, libinfo.values("QMAKE_PRL_DEFINES"))
|
|
if (!defs.contains(def) && prl_defs.contains(def))
|
|
defs.append(def);
|
|
if (try_replace_file) {
|
|
ProString tgt = libinfo.first("QMAKE_PRL_TARGET");
|
|
if (tgt.isEmpty()) {
|
|
fprintf(stderr, "Error: %s does not define QMAKE_PRL_TARGET\n",
|
|
meta_file.toLatin1().constData());
|
|
} else if (!tgt.contains('.')
|
|
&& !libinfo.values("QMAKE_PRL_CONFIG").contains("lib_bundle")) {
|
|
fprintf(stderr, "Error: %s defines QMAKE_PRL_TARGET without extension\n",
|
|
meta_file.toLatin1().constData());
|
|
} else {
|
|
int off = qMax(file.lastIndexOf('/'), file.lastIndexOf('\\')) + 1;
|
|
debug_msg(1, " Replacing library reference %s with %s",
|
|
file.mid(off).toLatin1().constData(),
|
|
tgt.toQString().toLatin1().constData());
|
|
file.replace(off, 1000, tgt.toQString());
|
|
}
|
|
}
|
|
QString mf = fileFixify(meta_file);
|
|
if (!project->values("QMAKE_PRL_INTERNAL_FILES").contains(mf))
|
|
project->values("QMAKE_PRL_INTERNAL_FILES").append(mf);
|
|
if (!project->values("QMAKE_INTERNAL_INCLUDED_FILES").contains(mf))
|
|
project->values("QMAKE_INTERNAL_INCLUDED_FILES").append(mf);
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::filterIncludedFiles(const char *var)
|
|
{
|
|
ProStringList &inputs = project->values(var);
|
|
for (ProStringList::Iterator input = inputs.begin(); input != inputs.end(); ) {
|
|
if (QMakeSourceFileInfo::included((*input).toQString()) > 0)
|
|
input = inputs.erase(input);
|
|
else
|
|
++input;
|
|
}
|
|
}
|
|
|
|
static QString
|
|
qv(const ProString &val)
|
|
{
|
|
return ' ' + QMakeEvaluator::quoteValue(val);
|
|
}
|
|
|
|
static QString
|
|
qv(const ProStringList &val)
|
|
{
|
|
QString ret;
|
|
foreach (const ProString &v, val)
|
|
ret += qv(v);
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writePrlFile(QTextStream &t)
|
|
{
|
|
QString bdir = Option::output_dir;
|
|
if(bdir.isEmpty())
|
|
bdir = qmake_getpwd();
|
|
t << "QMAKE_PRL_BUILD_DIR =" << qv(bdir) << endl;
|
|
|
|
t << "QMAKE_PRO_INPUT =" << qv(project->projectFile().section('/', -1)) << endl;
|
|
|
|
if(!project->isEmpty("QMAKE_ABSOLUTE_SOURCE_PATH"))
|
|
t << "QMAKE_PRL_SOURCE_DIR =" << qv(project->first("QMAKE_ABSOLUTE_SOURCE_PATH")) << endl;
|
|
t << "QMAKE_PRL_TARGET =" << qv(project->first("LIB_TARGET")) << endl;
|
|
if(!project->isEmpty("PRL_EXPORT_DEFINES"))
|
|
t << "QMAKE_PRL_DEFINES =" << qv(project->values("PRL_EXPORT_DEFINES")) << endl;
|
|
if(!project->isEmpty("PRL_EXPORT_CFLAGS"))
|
|
t << "QMAKE_PRL_CFLAGS =" << qv(project->values("PRL_EXPORT_CFLAGS")) << endl;
|
|
if(!project->isEmpty("PRL_EXPORT_CXXFLAGS"))
|
|
t << "QMAKE_PRL_CXXFLAGS =" << qv(project->values("PRL_EXPORT_CXXFLAGS")) << endl;
|
|
if(!project->isEmpty("CONFIG"))
|
|
t << "QMAKE_PRL_CONFIG =" << qv(project->values("CONFIG")) << endl;
|
|
if(!project->isEmpty("TARGET_VERSION_EXT"))
|
|
t << "QMAKE_PRL_VERSION = " << project->first("TARGET_VERSION_EXT") << endl;
|
|
else if(!project->isEmpty("VERSION"))
|
|
t << "QMAKE_PRL_VERSION = " << project->first("VERSION") << endl;
|
|
if(project->isActiveConfig("staticlib") || project->isActiveConfig("explicitlib")) {
|
|
ProStringList libs;
|
|
if(!project->isEmpty("QMAKE_INTERNAL_PRL_LIBS"))
|
|
libs = project->values("QMAKE_INTERNAL_PRL_LIBS");
|
|
else
|
|
libs << "QMAKE_LIBS"; //obvious one
|
|
if(project->isActiveConfig("staticlib"))
|
|
libs << "QMAKE_LIBS_PRIVATE";
|
|
t << "QMAKE_PRL_LIBS =";
|
|
for (ProStringList::Iterator it = libs.begin(); it != libs.end(); ++it)
|
|
t << qv(project->values((*it).toKey()));
|
|
t << endl;
|
|
}
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::writeProjectMakefile()
|
|
{
|
|
QTextStream t(&Option::output);
|
|
|
|
//header
|
|
writeHeader(t);
|
|
|
|
QList<SubTarget*> targets;
|
|
{
|
|
ProStringList builds = project->values("BUILDS");
|
|
for (ProStringList::Iterator it = builds.begin(); it != builds.end(); ++it) {
|
|
SubTarget *st = new SubTarget;
|
|
targets.append(st);
|
|
st->makefile = "$(MAKEFILE)." + (*it);
|
|
st->name = (*it).toQString();
|
|
const ProKey tkey(*it + ".target");
|
|
st->target = (project->isEmpty(tkey) ? (*it) : project->first(tkey)).toQString();
|
|
}
|
|
}
|
|
if(project->isActiveConfig("build_all")) {
|
|
t << "first: all\n";
|
|
QList<SubTarget*>::Iterator it;
|
|
|
|
//install
|
|
t << "install: ";
|
|
for(it = targets.begin(); it != targets.end(); ++it)
|
|
t << (*it)->target << "-install ";
|
|
t << endl;
|
|
|
|
//uninstall
|
|
t << "uninstall: ";
|
|
for(it = targets.begin(); it != targets.end(); ++it)
|
|
t << (*it)->target << "-uninstall ";
|
|
t << endl;
|
|
} else {
|
|
t << "first: " << targets.first()->target << endl
|
|
<< "install: " << targets.first()->target << "-install\n"
|
|
<< "uninstall: " << targets.first()->target << "-uninstall\n";
|
|
}
|
|
|
|
writeSubTargets(t, targets, SubTargetsNoFlags);
|
|
if(!project->isActiveConfig("no_autoqmake")) {
|
|
QString mkf = escapeDependencyPath(fileFixify(Option::output.fileName()));
|
|
for(QList<SubTarget*>::Iterator it = targets.begin(); it != targets.end(); ++it)
|
|
t << escapeDependencyPath((*it)->makefile) << ": " << mkf << endl;
|
|
}
|
|
qDeleteAll(targets);
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::write()
|
|
{
|
|
if(!project)
|
|
return false;
|
|
writePrlFile();
|
|
if(Option::qmake_mode == Option::QMAKE_GENERATE_MAKEFILE || //write makefile
|
|
Option::qmake_mode == Option::QMAKE_GENERATE_PROJECT) {
|
|
QTextStream t(&Option::output);
|
|
if(!writeMakefile(t)) {
|
|
#if 1
|
|
warn_msg(WarnLogic, "Unable to generate output for: %s [TEMPLATE %s]",
|
|
Option::output.fileName().toLatin1().constData(),
|
|
project->first("TEMPLATE").toLatin1().constData());
|
|
if(Option::output.exists())
|
|
Option::output.remove();
|
|
#endif
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::prlFileName(bool fixify)
|
|
{
|
|
QString ret = project->first("TARGET_PRL").toQString();
|
|
if(ret.isEmpty())
|
|
ret = project->first("TARGET").toQString();
|
|
int slsh = ret.lastIndexOf(Option::dir_sep);
|
|
if(slsh != -1)
|
|
ret.remove(0, slsh);
|
|
if(!ret.endsWith(Option::prl_ext)) {
|
|
int dot = ret.indexOf('.');
|
|
if(dot != -1)
|
|
ret.truncate(dot);
|
|
ret += Option::prl_ext;
|
|
}
|
|
if(!project->isEmpty("QMAKE_BUNDLE"))
|
|
ret.prepend(project->first("QMAKE_BUNDLE") + Option::dir_sep);
|
|
if(fixify) {
|
|
if(!project->isEmpty("DESTDIR"))
|
|
ret.prepend(project->first("DESTDIR").toQString());
|
|
ret = fileFixify(ret, FileFixifyBackwards);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writePrlFile()
|
|
{
|
|
if((Option::qmake_mode == Option::QMAKE_GENERATE_MAKEFILE ||
|
|
Option::qmake_mode == Option::QMAKE_GENERATE_PRL)
|
|
&& project->values("QMAKE_FAILED_REQUIREMENTS").isEmpty()
|
|
&& project->isActiveConfig("create_prl")
|
|
&& (project->first("TEMPLATE") == "lib"
|
|
|| project->first("TEMPLATE") == "vclib")
|
|
&& (!project->isActiveConfig("plugin") || project->isActiveConfig("static"))) { //write prl file
|
|
QString local_prl = prlFileName();
|
|
QString prl = fileFixify(local_prl);
|
|
mkdir(fileInfo(local_prl).path());
|
|
QFile ft(local_prl);
|
|
if(ft.open(QIODevice::WriteOnly)) {
|
|
project->values("ALL_DEPS").append(prl);
|
|
project->values("QMAKE_INTERNAL_PRL_FILE").append(prl);
|
|
project->values("QMAKE_DISTCLEAN").append(prl);
|
|
QTextStream t(&ft);
|
|
writePrlFile(t);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeObj(QTextStream &t, const char *src)
|
|
{
|
|
const ProStringList &srcl = project->values(src);
|
|
const ProStringList objl = createObjectList(srcl);
|
|
|
|
ProStringList::ConstIterator oit = objl.begin();
|
|
ProStringList::ConstIterator sit = srcl.begin();
|
|
QLatin1String stringSrc("$src");
|
|
QLatin1String stringObj("$obj");
|
|
for(;sit != srcl.end() && oit != objl.end(); ++oit, ++sit) {
|
|
if((*sit).isEmpty())
|
|
continue;
|
|
|
|
QString srcf = (*sit).toQString();
|
|
QString dstf = (*oit).toQString();
|
|
t << escapeDependencyPath(dstf) << ": " << escapeDependencyPath(srcf)
|
|
<< " " << escapeDependencyPaths(findDependencies(srcf)).join(" \\\n\t\t");
|
|
|
|
ProKey comp;
|
|
foreach (const ProString &compiler, project->values("QMAKE_BUILTIN_COMPILERS")) {
|
|
// Unfortunately we were not consistent about the C++ naming
|
|
ProString extensionSuffix = compiler;
|
|
if (extensionSuffix == "CXX")
|
|
extensionSuffix = ProString("CPP");
|
|
|
|
// Nor the C naming
|
|
ProString compilerSuffix = compiler;
|
|
if (compilerSuffix == "C")
|
|
compilerSuffix = ProString("CC");
|
|
|
|
foreach (const ProString &extension, project->values(ProKey("QMAKE_EXT_" + extensionSuffix))) {
|
|
if ((*sit).endsWith(extension)) {
|
|
comp = ProKey("QMAKE_RUN_" + compilerSuffix);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!comp.isNull())
|
|
break;
|
|
}
|
|
|
|
if (comp.isEmpty())
|
|
comp = "QMAKE_RUN_CC";
|
|
if (!project->isEmpty(comp)) {
|
|
QString p = var(comp);
|
|
p.replace(stringSrc, escapeFilePath(srcf));
|
|
p.replace(stringObj, escapeFilePath(dstf));
|
|
t << "\n\t" << p;
|
|
}
|
|
t << endl << endl;
|
|
}
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::filePrefixRoot(const QString &root, const QString &path)
|
|
{
|
|
QString ret(path);
|
|
if(path.length() > 2 && path[1] == ':') //c:\foo
|
|
ret.insert(2, root);
|
|
else
|
|
ret.prepend(root);
|
|
while (ret.endsWith('\\'))
|
|
ret.chop(1);
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeInstalls(QTextStream &t, bool noBuild)
|
|
{
|
|
QString rm_dir_contents("-$(DEL_FILE)");
|
|
if (!isWindowsShell()) //ick
|
|
rm_dir_contents = "-$(DEL_FILE) -r";
|
|
|
|
QString all_installs, all_uninstalls;
|
|
QSet<QString> made_dirs, removed_dirs;
|
|
const ProStringList &l = project->values("INSTALLS");
|
|
for (ProStringList::ConstIterator it = l.begin(); it != l.end(); ++it) {
|
|
const ProKey pvar(*it + ".path");
|
|
const ProStringList &installConfigValues = project->values(ProKey(*it + ".CONFIG"));
|
|
if (installConfigValues.indexOf("no_path") == -1 &&
|
|
installConfigValues.indexOf("dummy_install") == -1 &&
|
|
project->values(pvar).isEmpty()) {
|
|
warn_msg(WarnLogic, "%s is not defined: install target not created\n", pvar.toLatin1().constData());
|
|
continue;
|
|
}
|
|
|
|
bool do_default = true;
|
|
const QString root = installRoot();
|
|
QString dst;
|
|
if (installConfigValues.indexOf("no_path") == -1 &&
|
|
installConfigValues.indexOf("dummy_install") == -1) {
|
|
dst = fileFixify(project->first(pvar).toQString(), FileFixifyAbsolute, false);
|
|
if(!dst.endsWith(Option::dir_sep))
|
|
dst += Option::dir_sep;
|
|
}
|
|
|
|
QStringList tmp, inst, uninst;
|
|
//other
|
|
ProStringList tmp2 = project->values(ProKey(*it + ".extra"));
|
|
if (tmp2.isEmpty())
|
|
tmp2 = project->values(ProKey(*it + ".commands")); //to allow compatible name
|
|
if (!tmp2.isEmpty()) {
|
|
do_default = false;
|
|
inst << tmp2.join(' ');
|
|
}
|
|
//masks
|
|
tmp2 = findFilesInVPATH(project->values(ProKey(*it + ".files")), VPATH_NoFixify);
|
|
tmp = fileFixify(tmp2.toQStringList(), FileFixifyAbsolute);
|
|
if(!tmp.isEmpty()) {
|
|
do_default = false;
|
|
QString base_path = project->first(ProKey(*it + ".base")).toQString();
|
|
if (!base_path.isEmpty()) {
|
|
base_path = Option::fixPathToTargetOS(base_path, false, true);
|
|
if (!base_path.endsWith(Option::dir_sep))
|
|
base_path += Option::dir_sep;
|
|
}
|
|
for(QStringList::Iterator wild_it = tmp.begin(); wild_it != tmp.end(); ++wild_it) {
|
|
QString wild = Option::fixPathToTargetOS((*wild_it), false, false);
|
|
QString dirstr = qmake_getpwd(), filestr = wild;
|
|
int slsh = filestr.lastIndexOf(Option::dir_sep);
|
|
if(slsh != -1) {
|
|
dirstr = filestr.left(slsh+1);
|
|
filestr.remove(0, slsh+1);
|
|
}
|
|
if(!dirstr.endsWith(Option::dir_sep))
|
|
dirstr += Option::dir_sep;
|
|
QString dst_dir = dst;
|
|
if (!base_path.isEmpty()) {
|
|
if (!dirstr.startsWith(base_path)) {
|
|
warn_msg(WarnLogic, "File %s in install rule %s does not start with base %s",
|
|
qPrintable(wild), qPrintable((*it).toQString()),
|
|
qPrintable(base_path));
|
|
} else {
|
|
QString dir_sfx = dirstr.mid(base_path.length());
|
|
dst_dir += dir_sfx;
|
|
if (!dir_sfx.isEmpty() && !made_dirs.contains(dir_sfx)) {
|
|
made_dirs.insert(dir_sfx);
|
|
QString tmp_dst = fileFixify(dst_dir, FileFixifyAbsolute, false);
|
|
tmp_dst.chop(1);
|
|
inst << mkdir_p_asstring(filePrefixRoot(root, tmp_dst));
|
|
for (int i = dst.length(); i < dst_dir.length(); i++) {
|
|
if (dst_dir.at(i) == Option::dir_sep) {
|
|
QString subd = dst_dir.left(i);
|
|
if (!removed_dirs.contains(subd)) {
|
|
removed_dirs.insert(subd);
|
|
tmp_dst = fileFixify(subd, FileFixifyAbsolute, false);
|
|
uninst << "-$(DEL_DIR) "
|
|
+ escapeFilePath(filePrefixRoot(root, tmp_dst));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
bool is_target = (wild == fileFixify(var("TARGET"), FileFixifyAbsolute));
|
|
if(is_target || exists(wild)) { //real file or target
|
|
QFileInfo fi(fileInfo(wild));
|
|
QString dst_file = filePrefixRoot(root, dst_dir);
|
|
if(fi.isDir() && project->isActiveConfig("copy_dir_files")) {
|
|
if(!dst_file.endsWith(Option::dir_sep))
|
|
dst_file += Option::dir_sep;
|
|
dst_file += fi.fileName();
|
|
}
|
|
QString cmd;
|
|
if (fi.isDir())
|
|
cmd = "-$(INSTALL_DIR)";
|
|
else if (is_target || fi.isExecutable())
|
|
cmd = "-$(INSTALL_PROGRAM)";
|
|
else
|
|
cmd = "-$(INSTALL_FILE)";
|
|
cmd += " " + escapeFilePath(wild) + " " + escapeFilePath(dst_file);
|
|
inst << cmd;
|
|
if (!project->isActiveConfig("debug_info") && !project->isActiveConfig("nostrip") &&
|
|
!fi.isDir() && fi.isExecutable() && !project->isEmpty("QMAKE_STRIP"))
|
|
inst << QString("-") + var("QMAKE_STRIP") + " " +
|
|
escapeFilePath(filePrefixRoot(root, fileFixify(dst_dir + filestr, FileFixifyAbsolute, false)));
|
|
uninst.append(rm_dir_contents + " " + escapeFilePath(filePrefixRoot(root, fileFixify(dst_dir + filestr, FileFixifyAbsolute, false))));
|
|
continue;
|
|
}
|
|
QString local_dirstr = Option::normalizePath(dirstr);
|
|
QStringList files = QDir(local_dirstr).entryList(QStringList(filestr),
|
|
QDir::NoDotAndDotDot | QDir::AllEntries);
|
|
if (installConfigValues.contains("no_check_exist") && files.isEmpty()) {
|
|
QString dst_file = filePrefixRoot(root, dst_dir);
|
|
QString cmd;
|
|
if (installConfigValues.contains("directory")) {
|
|
cmd = QLatin1String("-$(INSTALL_DIR)");
|
|
if (project->isActiveConfig("copy_dir_files")) {
|
|
if (!dst_file.endsWith(Option::dir_sep))
|
|
dst_file += Option::dir_sep;
|
|
dst_file += filestr;
|
|
}
|
|
} else if (installConfigValues.contains("executable")) {
|
|
cmd = QLatin1String("-$(INSTALL_PROGRAM)");
|
|
} else {
|
|
cmd = QLatin1String("-$(INSTALL_FILE)");
|
|
}
|
|
cmd += " " + escapeFilePath(wild) + " " + escapeFilePath(dst_file);
|
|
inst << cmd;
|
|
uninst.append(rm_dir_contents + " " + escapeFilePath(filePrefixRoot(root, fileFixify(dst_dir + filestr, FileFixifyAbsolute, false))));
|
|
}
|
|
for(int x = 0; x < files.count(); x++) {
|
|
QString file = files[x];
|
|
uninst.append(rm_dir_contents + " " + escapeFilePath(filePrefixRoot(root, fileFixify(dst_dir + file, FileFixifyAbsolute, false))));
|
|
QFileInfo fi(fileInfo(dirstr + file));
|
|
QString dst_file = filePrefixRoot(root, fileFixify(dst_dir, FileFixifyAbsolute, false));
|
|
if(fi.isDir() && project->isActiveConfig("copy_dir_files")) {
|
|
if(!dst_file.endsWith(Option::dir_sep))
|
|
dst_file += Option::dir_sep;
|
|
dst_file += fi.fileName();
|
|
}
|
|
QString cmd = QString(fi.isDir() ? "-$(INSTALL_DIR)" : "-$(INSTALL_FILE)") + " " +
|
|
escapeFilePath(dirstr + file) + " " + escapeFilePath(dst_file);
|
|
inst << cmd;
|
|
if (!project->isActiveConfig("debug_info") && !project->isActiveConfig("nostrip") &&
|
|
!fi.isDir() && fi.isExecutable() && !project->isEmpty("QMAKE_STRIP"))
|
|
inst << QString("-") + var("QMAKE_STRIP") + " " +
|
|
escapeFilePath(filePrefixRoot(root, fileFixify(dst_dir + file, FileFixifyAbsolute, false)));
|
|
}
|
|
}
|
|
}
|
|
QString target;
|
|
//default?
|
|
if (do_default)
|
|
target = defaultInstall((*it).toQString());
|
|
else
|
|
target = inst.join("\n\t");
|
|
QString puninst = project->values(ProKey(*it + ".uninstall")).join(' ');
|
|
if (!puninst.isEmpty())
|
|
uninst << puninst;
|
|
|
|
if (!target.isEmpty() || installConfigValues.indexOf("dummy_install") != -1) {
|
|
if (noBuild || installConfigValues.indexOf("no_build") != -1)
|
|
t << "install_" << (*it) << ":";
|
|
else if(project->isActiveConfig("build_all"))
|
|
t << "install_" << (*it) << ": all";
|
|
else
|
|
t << "install_" << (*it) << ": first";
|
|
const ProStringList &deps = project->values(ProKey(*it + ".depends"));
|
|
if(!deps.isEmpty()) {
|
|
for (ProStringList::ConstIterator dep_it = deps.begin(); dep_it != deps.end(); ++dep_it) {
|
|
QString targ = var(ProKey(*dep_it + ".target"));
|
|
if(targ.isEmpty())
|
|
targ = (*dep_it).toQString();
|
|
t << " " << escapeDependencyPath(targ);
|
|
}
|
|
}
|
|
t << " FORCE\n\t";
|
|
const ProStringList &dirs = project->values(pvar);
|
|
for (ProStringList::ConstIterator pit = dirs.begin(); pit != dirs.end(); ++pit) {
|
|
QString tmp_dst = fileFixify((*pit).toQString(), FileFixifyAbsolute, false);
|
|
t << mkdir_p_asstring(filePrefixRoot(root, tmp_dst)) << "\n\t";
|
|
}
|
|
t << target << endl << endl;
|
|
if(!uninst.isEmpty()) {
|
|
t << "uninstall_" << (*it) << ": FORCE";
|
|
for (int i = uninst.size(); --i >= 0; )
|
|
t << "\n\t" << uninst.at(i);
|
|
t << "\n\t-$(DEL_DIR) " << escapeFilePath(filePrefixRoot(root, dst)) << " \n\n";
|
|
}
|
|
t << endl;
|
|
|
|
if (installConfigValues.indexOf("no_default_install") == -1) {
|
|
all_installs += QString("install_") + (*it) + " ";
|
|
if(!uninst.isEmpty())
|
|
all_uninstalls += "uninstall_" + (*it) + " ";
|
|
}
|
|
} else {
|
|
debug_msg(1, "no definition for install %s: install target not created",(*it).toLatin1().constData());
|
|
}
|
|
}
|
|
t << "install:" << depVar("INSTALLDEPS") << ' ' << all_installs
|
|
<< " FORCE\n\nuninstall: " << all_uninstalls << depVar("UNINSTALLDEPS")
|
|
<< " FORCE\n\n";
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::var(const ProKey &var) const
|
|
{
|
|
return val(project->values(var));
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::fileVar(const ProKey &var) const
|
|
{
|
|
return val(escapeFilePaths(project->values(var)));
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::fileVarList(const ProKey &var) const
|
|
{
|
|
return valList(escapeFilePaths(project->values(var)));
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::depVar(const ProKey &var) const
|
|
{
|
|
return val(escapeDependencyPaths(project->values(var)));
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::val(const ProStringList &varList) const
|
|
{
|
|
return valGlue(varList, "", " ", "");
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::val(const QStringList &varList) const
|
|
{
|
|
return valGlue(varList, "", " ", "");
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::varGlue(const ProKey &var, const QString &before, const QString &glue, const QString &after) const
|
|
{
|
|
return valGlue(project->values(var), before, glue, after);
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::fileVarGlue(const ProKey &var, const QString &before, const QString &glue, const QString &after) const
|
|
{
|
|
return valGlue(escapeFilePaths(project->values(var)), before, glue, after);
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::fixFileVarGlue(const ProKey &var, const QString &before, const QString &glue, const QString &after) const
|
|
{
|
|
ProStringList varList;
|
|
foreach (const ProString &val, project->values(var))
|
|
varList << escapeFilePath(Option::fixPathToTargetOS(val.toQString()));
|
|
return valGlue(varList, before, glue, after);
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::valGlue(const ProStringList &varList, const QString &before, const QString &glue, const QString &after) const
|
|
{
|
|
QString ret;
|
|
for (ProStringList::ConstIterator it = varList.begin(); it != varList.end(); ++it) {
|
|
if (!(*it).isEmpty()) {
|
|
if (!ret.isEmpty())
|
|
ret += glue;
|
|
ret += (*it).toQString();
|
|
}
|
|
}
|
|
return ret.isEmpty() ? QString("") : before + ret + after;
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::valGlue(const QStringList &varList, const QString &before, const QString &glue, const QString &after) const
|
|
{
|
|
QString ret;
|
|
for(QStringList::ConstIterator it = varList.begin(); it != varList.end(); ++it) {
|
|
if(!(*it).isEmpty()) {
|
|
if(!ret.isEmpty())
|
|
ret += glue;
|
|
ret += (*it);
|
|
}
|
|
}
|
|
return ret.isEmpty() ? QString("") : before + ret + after;
|
|
}
|
|
|
|
|
|
QString
|
|
MakefileGenerator::varList(const ProKey &var) const
|
|
{
|
|
return valList(project->values(var));
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::valList(const ProStringList &varList) const
|
|
{
|
|
return valGlue(varList, "", " \\\n\t\t", "");
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::valList(const QStringList &varList) const
|
|
{
|
|
return valGlue(varList, "", " \\\n\t\t", "");
|
|
}
|
|
|
|
ProStringList
|
|
MakefileGenerator::createObjectList(const ProStringList &sources)
|
|
{
|
|
ProStringList ret;
|
|
QString objdir;
|
|
if(!project->values("OBJECTS_DIR").isEmpty())
|
|
objdir = project->first("OBJECTS_DIR").toQString();
|
|
for (ProStringList::ConstIterator it = sources.begin(); it != sources.end(); ++it) {
|
|
QString sfn = (*it).toQString();
|
|
QFileInfo fi(fileInfo(Option::normalizePath(sfn)));
|
|
QString dir;
|
|
if (project->isActiveConfig("object_parallel_to_source")) {
|
|
// The source paths are relative to the output dir, but we need source-relative paths
|
|
QString sourceRelativePath = fileFixify(sfn, FileFixifyBackwards);
|
|
|
|
if (sourceRelativePath.startsWith(".." + Option::dir_sep))
|
|
sourceRelativePath = fileFixify(sourceRelativePath, FileFixifyAbsolute);
|
|
|
|
if (QDir::isAbsolutePath(sourceRelativePath))
|
|
sourceRelativePath.remove(0, sourceRelativePath.indexOf(Option::dir_sep) + 1);
|
|
|
|
dir = objdir; // We still respect OBJECTS_DIR
|
|
|
|
int lastDirSepPosition = sourceRelativePath.lastIndexOf(Option::dir_sep);
|
|
if (lastDirSepPosition != -1)
|
|
dir += sourceRelativePath.leftRef(lastDirSepPosition + 1);
|
|
|
|
if (!noIO()) {
|
|
// Ensure that the final output directory of each object exists
|
|
QString outRelativePath = fileFixify(dir, FileFixifyBackwards);
|
|
if (!mkdir(outRelativePath))
|
|
warn_msg(WarnLogic, "Cannot create directory '%s'", outRelativePath.toLatin1().constData());
|
|
}
|
|
} else {
|
|
dir = objdir;
|
|
}
|
|
ret.append(dir + fi.completeBaseName() + Option::obj_ext);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
ReplaceExtraCompilerCacheKey::ReplaceExtraCompilerCacheKey(
|
|
const QString &v, const QStringList &i, const QStringList &o, MakefileGenerator::ReplaceFor s)
|
|
{
|
|
static QString doubleColon = QLatin1String("::");
|
|
|
|
hash = 0;
|
|
pwd = qmake_getpwd();
|
|
var = v;
|
|
{
|
|
QStringList il = i;
|
|
il.sort();
|
|
in = il.join(doubleColon);
|
|
}
|
|
{
|
|
QStringList ol = o;
|
|
ol.sort();
|
|
out = ol.join(doubleColon);
|
|
}
|
|
forShell = s;
|
|
}
|
|
|
|
bool ReplaceExtraCompilerCacheKey::operator==(const ReplaceExtraCompilerCacheKey &f) const
|
|
{
|
|
return (hashCode() == f.hashCode() &&
|
|
f.forShell == forShell &&
|
|
f.in == in &&
|
|
f.out == out &&
|
|
f.var == var &&
|
|
f.pwd == pwd);
|
|
}
|
|
|
|
|
|
QString
|
|
MakefileGenerator::replaceExtraCompilerVariables(
|
|
const QString &orig_var, const QStringList &in, const QStringList &out, ReplaceFor forShell)
|
|
{
|
|
//lazy cache
|
|
ReplaceExtraCompilerCacheKey cacheKey(orig_var, in, out, forShell);
|
|
QString cacheVal = extraCompilerVariablesCache.value(cacheKey);
|
|
if(!cacheVal.isNull())
|
|
return cacheVal;
|
|
|
|
//do the work
|
|
QString ret = orig_var;
|
|
QRegExp reg_var("\\$\\{.*\\}");
|
|
reg_var.setMinimal(true);
|
|
for(int rep = 0; (rep = reg_var.indexIn(ret, rep)) != -1; ) {
|
|
QStringList val;
|
|
const ProString var(ret.mid(rep + 2, reg_var.matchedLength() - 3));
|
|
bool filePath = false;
|
|
if(val.isEmpty() && var.startsWith(QLatin1String("QMAKE_VAR_"))) {
|
|
const ProKey varname = var.mid(10).toKey();
|
|
val += project->values(varname).toQStringList();
|
|
}
|
|
if(val.isEmpty() && var.startsWith(QLatin1String("QMAKE_VAR_FIRST_"))) {
|
|
const ProKey varname = var.mid(16).toKey();
|
|
val += project->first(varname).toQString();
|
|
}
|
|
|
|
if(val.isEmpty() && !in.isEmpty()) {
|
|
if(var.startsWith(QLatin1String("QMAKE_FUNC_FILE_IN_"))) {
|
|
filePath = true;
|
|
const ProKey funcname = var.mid(19).toKey();
|
|
val += project->expand(funcname, QList<ProStringList>() << ProStringList(in));
|
|
} else if(var == QLatin1String("QMAKE_FILE_BASE") || var == QLatin1String("QMAKE_FILE_IN_BASE")) {
|
|
filePath = true;
|
|
for(int i = 0; i < in.size(); ++i) {
|
|
QFileInfo fi(fileInfo(Option::normalizePath(in.at(i))));
|
|
QString base = fi.completeBaseName();
|
|
if(base.isNull())
|
|
base = fi.fileName();
|
|
val += base;
|
|
}
|
|
} else if (var == QLatin1String("QMAKE_FILE_EXT") || var == QLatin1String("QMAKE_FILE_IN_EXT")) {
|
|
filePath = true;
|
|
for(int i = 0; i < in.size(); ++i) {
|
|
QFileInfo fi(fileInfo(Option::normalizePath(in.at(i))));
|
|
QString ext;
|
|
// Ensure complementarity with QMAKE_FILE_BASE
|
|
int baseLen = fi.completeBaseName().length();
|
|
if(baseLen == 0)
|
|
ext = fi.fileName();
|
|
else
|
|
ext = fi.fileName().remove(0, baseLen);
|
|
val += ext;
|
|
}
|
|
} else if (var == QLatin1String("QMAKE_FILE_IN_NAME")) {
|
|
filePath = true;
|
|
for (int i = 0; i < in.size(); ++i)
|
|
val += fileInfo(Option::normalizePath(in.at(i))).fileName();
|
|
} else if(var == QLatin1String("QMAKE_FILE_PATH") || var == QLatin1String("QMAKE_FILE_IN_PATH")) {
|
|
filePath = true;
|
|
for(int i = 0; i < in.size(); ++i)
|
|
val += fileInfo(Option::normalizePath(in.at(i))).path();
|
|
} else if(var == QLatin1String("QMAKE_FILE_NAME") || var == QLatin1String("QMAKE_FILE_IN")) {
|
|
filePath = true;
|
|
for(int i = 0; i < in.size(); ++i)
|
|
val += fileInfo(Option::normalizePath(in.at(i))).filePath();
|
|
|
|
}
|
|
}
|
|
if(val.isEmpty() && !out.isEmpty()) {
|
|
if(var.startsWith(QLatin1String("QMAKE_FUNC_FILE_OUT_"))) {
|
|
filePath = true;
|
|
const ProKey funcname = var.mid(20).toKey();
|
|
val += project->expand(funcname, QList<ProStringList>() << ProStringList(out));
|
|
} else if (var == QLatin1String("QMAKE_FILE_OUT_PATH")) {
|
|
filePath = true;
|
|
for (int i = 0; i < out.size(); ++i)
|
|
val += fileInfo(Option::normalizePath(out.at(i))).path();
|
|
} else if(var == QLatin1String("QMAKE_FILE_OUT")) {
|
|
filePath = true;
|
|
for(int i = 0; i < out.size(); ++i)
|
|
val += fileInfo(Option::normalizePath(out.at(i))).filePath();
|
|
} else if(var == QLatin1String("QMAKE_FILE_OUT_BASE")) {
|
|
filePath = true;
|
|
for(int i = 0; i < out.size(); ++i) {
|
|
QFileInfo fi(fileInfo(Option::normalizePath(out.at(i))));
|
|
QString base = fi.completeBaseName();
|
|
if(base.isNull())
|
|
base = fi.fileName();
|
|
val += base;
|
|
}
|
|
}
|
|
}
|
|
if(val.isEmpty() && var.startsWith(QLatin1String("QMAKE_FUNC_"))) {
|
|
const ProKey funcname = var.mid(11).toKey();
|
|
val += project->expand(funcname, QList<ProStringList>() << ProStringList(in) << ProStringList(out));
|
|
}
|
|
|
|
if(!val.isEmpty()) {
|
|
QString fullVal;
|
|
if (filePath && forShell != NoShell) {
|
|
for(int i = 0; i < val.size(); ++i) {
|
|
if(!fullVal.isEmpty())
|
|
fullVal += " ";
|
|
if (forShell == LocalShell)
|
|
fullVal += IoUtils::shellQuote(Option::fixPathToLocalOS(val.at(i), false));
|
|
else
|
|
fullVal += escapeFilePath(Option::fixPathToTargetOS(val.at(i), false));
|
|
}
|
|
} else {
|
|
fullVal = val.join(' ');
|
|
}
|
|
ret.replace(rep, reg_var.matchedLength(), fullVal);
|
|
rep += fullVal.length();
|
|
} else {
|
|
rep += reg_var.matchedLength();
|
|
}
|
|
}
|
|
|
|
//cache the value
|
|
extraCompilerVariablesCache.insert(cacheKey, ret);
|
|
return ret;
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::verifyExtraCompiler(const ProString &comp, const QString &file_unfixed)
|
|
{
|
|
if(noIO())
|
|
return false;
|
|
const QString file = Option::normalizePath(file_unfixed);
|
|
|
|
const ProStringList &config = project->values(ProKey(comp + ".CONFIG"));
|
|
if (config.indexOf("moc_verify") != -1) {
|
|
if(!file.isNull()) {
|
|
QMakeSourceFileInfo::addSourceFile(file, QMakeSourceFileInfo::SEEK_MOCS);
|
|
if(!mocable(file)) {
|
|
return false;
|
|
} else {
|
|
project->values("MOCABLES").append(file);
|
|
}
|
|
}
|
|
} else if (config.indexOf("function_verify") != -1) {
|
|
ProString tmp_out = project->first(ProKey(comp + ".output"));
|
|
if(tmp_out.isEmpty())
|
|
return false;
|
|
ProStringList verify_function = project->values(ProKey(comp + ".verify_function"));
|
|
if(verify_function.isEmpty())
|
|
return false;
|
|
|
|
for(int i = 0; i < verify_function.size(); ++i) {
|
|
bool invert = false;
|
|
ProString verify = verify_function.at(i);
|
|
if(verify.at(0) == QLatin1Char('!')) {
|
|
invert = true;
|
|
verify = verify.mid(1);
|
|
}
|
|
|
|
if (config.indexOf("combine") != -1) {
|
|
bool pass = project->test(verify.toKey(), QList<ProStringList>() << ProStringList(tmp_out) << ProStringList(file));
|
|
if(invert)
|
|
pass = !pass;
|
|
if(!pass)
|
|
return false;
|
|
} else {
|
|
const ProStringList &tmp = project->values(ProKey(comp + ".input"));
|
|
for (ProStringList::ConstIterator it = tmp.begin(); it != tmp.end(); ++it) {
|
|
const ProStringList &inputs = project->values((*it).toKey());
|
|
for (ProStringList::ConstIterator input = inputs.begin(); input != inputs.end(); ++input) {
|
|
if((*input).isEmpty())
|
|
continue;
|
|
QString inpf = (*input).toQString();
|
|
QString in = fileFixify(inpf);
|
|
if(in == file) {
|
|
bool pass = project->test(verify.toKey(),
|
|
QList<ProStringList>() << ProStringList(replaceExtraCompilerVariables(tmp_out.toQString(), inpf, QString(), NoShell)) <<
|
|
ProStringList(file));
|
|
if(invert)
|
|
pass = !pass;
|
|
if(!pass)
|
|
return false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if (config.indexOf("verify") != -1) {
|
|
QString tmp_out = project->first(ProKey(comp + ".output")).toQString();
|
|
if(tmp_out.isEmpty())
|
|
return false;
|
|
const QString tmp_cmd = project->values(ProKey(comp + ".commands")).join(' ');
|
|
if (config.indexOf("combine") != -1) {
|
|
QString cmd = replaceExtraCompilerVariables(tmp_cmd, QString(), tmp_out, LocalShell);
|
|
if(system(cmd.toLatin1().constData()))
|
|
return false;
|
|
} else {
|
|
const ProStringList &tmp = project->values(ProKey(comp + ".input"));
|
|
for (ProStringList::ConstIterator it = tmp.begin(); it != tmp.end(); ++it) {
|
|
const ProStringList &inputs = project->values((*it).toKey());
|
|
for (ProStringList::ConstIterator input = inputs.begin(); input != inputs.end(); ++input) {
|
|
if((*input).isEmpty())
|
|
continue;
|
|
QString inpf = (*input).toQString();
|
|
QString in = fileFixify(inpf);
|
|
if(in == file) {
|
|
QString out = replaceExtraCompilerVariables(tmp_out, inpf, QString(), NoShell);
|
|
QString cmd = replaceExtraCompilerVariables(tmp_cmd, in, out, LocalShell);
|
|
if(system(cmd.toLatin1().constData()))
|
|
return false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeExtraTargets(QTextStream &t)
|
|
{
|
|
const ProStringList &qut = project->values("QMAKE_EXTRA_TARGETS");
|
|
for (ProStringList::ConstIterator it = qut.begin(); it != qut.end(); ++it) {
|
|
QString targ = var(ProKey(*it + ".target")),
|
|
cmd = var(ProKey(*it + ".commands")), deps;
|
|
if(targ.isEmpty())
|
|
targ = (*it).toQString();
|
|
const ProStringList &deplist = project->values(ProKey(*it + ".depends"));
|
|
for (ProStringList::ConstIterator dep_it = deplist.begin(); dep_it != deplist.end(); ++dep_it) {
|
|
QString dep = var(ProKey(*dep_it + ".target"));
|
|
if(dep.isEmpty())
|
|
dep = (*dep_it).toQString();
|
|
deps += " " + escapeDependencyPath(dep);
|
|
}
|
|
const ProStringList &config = project->values(ProKey(*it + ".CONFIG"));
|
|
if (config.indexOf("fix_target") != -1)
|
|
targ = fileFixify(targ, FileFixifyFromOutdir);
|
|
if (config.indexOf("phony") != -1)
|
|
deps += QLatin1String(" FORCE");
|
|
t << escapeDependencyPath(targ) << ":" << deps;
|
|
if(!cmd.isEmpty())
|
|
t << "\n\t" << cmd;
|
|
t << endl << endl;
|
|
}
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeExtraCompilerTargets(QTextStream &t)
|
|
{
|
|
QString clean_targets;
|
|
const ProStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
|
|
for (ProStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
|
|
QString tmp_out = fileFixify(project->first(ProKey(*it + ".output")).toQString(),
|
|
FileFixifyFromOutdir);
|
|
const QString tmp_cmd = project->values(ProKey(*it + ".commands")).join(' ');
|
|
const QString tmp_dep_cmd = project->values(ProKey(*it + ".depend_command")).join(' ');
|
|
QString dep_cd_cmd;
|
|
if (!tmp_dep_cmd.isEmpty()) {
|
|
dep_cd_cmd = QLatin1String("cd ")
|
|
+ escapeFilePath(Option::fixPathToLocalOS(Option::output_dir, false))
|
|
+ QLatin1String(" && ");
|
|
}
|
|
const ProStringList &vars = project->values(ProKey(*it + ".variables"));
|
|
if(tmp_out.isEmpty() || tmp_cmd.isEmpty())
|
|
continue;
|
|
ProStringList tmp_inputs;
|
|
{
|
|
const ProStringList &comp_inputs = project->values(ProKey(*it + ".input"));
|
|
for (ProStringList::ConstIterator it2 = comp_inputs.begin(); it2 != comp_inputs.end(); ++it2) {
|
|
const ProStringList &tmp = project->values((*it2).toKey());
|
|
for (ProStringList::ConstIterator input = tmp.begin(); input != tmp.end(); ++input) {
|
|
if (verifyExtraCompiler((*it), (*input).toQString()))
|
|
tmp_inputs.append((*input));
|
|
}
|
|
}
|
|
}
|
|
|
|
t << "compiler_" << (*it) << "_make_all:";
|
|
const ProStringList &config = project->values(ProKey(*it + ".CONFIG"));
|
|
if (config.indexOf("combine") != -1) {
|
|
// compilers with a combined input only have one output
|
|
QString input = project->first(ProKey(*it + ".output")).toQString();
|
|
t << ' ' << escapeDependencyPath(Option::fixPathToTargetOS(
|
|
replaceExtraCompilerVariables(tmp_out, input, QString(), NoShell)));
|
|
} else {
|
|
for (ProStringList::ConstIterator input = tmp_inputs.begin(); input != tmp_inputs.end(); ++input) {
|
|
t << ' ' << escapeDependencyPath(Option::fixPathToTargetOS(
|
|
replaceExtraCompilerVariables(tmp_out, (*input).toQString(), QString(), NoShell)));
|
|
}
|
|
}
|
|
t << endl;
|
|
|
|
if (config.indexOf("no_clean") == -1) {
|
|
QStringList raw_clean = project->values(ProKey(*it + ".clean")).toQStringList();
|
|
if (raw_clean.isEmpty())
|
|
raw_clean << tmp_out;
|
|
QString tmp_clean;
|
|
foreach (const QString &rc, raw_clean)
|
|
tmp_clean += ' ' + escapeFilePath(Option::fixPathToTargetOS(rc));
|
|
QString tmp_clean_cmds = project->values(ProKey(*it + ".clean_commands")).join(' ');
|
|
if(!tmp_inputs.isEmpty())
|
|
clean_targets += QString("compiler_" + (*it) + "_clean ");
|
|
t << "compiler_" << (*it) << "_clean:";
|
|
bool wrote_clean_cmds = false, wrote_clean = false;
|
|
if(tmp_clean_cmds.isEmpty()) {
|
|
wrote_clean_cmds = true;
|
|
} else if(tmp_clean_cmds.indexOf("${QMAKE_") == -1) {
|
|
t << "\n\t" << tmp_clean_cmds;
|
|
wrote_clean_cmds = true;
|
|
}
|
|
if(tmp_clean.indexOf("${QMAKE_") == -1) {
|
|
t << "\n\t-$(DEL_FILE)" << tmp_clean;
|
|
wrote_clean = true;
|
|
}
|
|
if(!wrote_clean_cmds || !wrote_clean) {
|
|
QStringList cleans;
|
|
const QString del_statement("-$(DEL_FILE)");
|
|
if(!wrote_clean) {
|
|
QStringList dels;
|
|
for (ProStringList::ConstIterator input = tmp_inputs.begin(); input != tmp_inputs.end(); ++input) {
|
|
QString tinp = (*input).toQString();
|
|
QString out = replaceExtraCompilerVariables(tmp_out, tinp, QString(), NoShell);
|
|
foreach (const QString &rc, raw_clean) {
|
|
dels << ' ' + escapeFilePath(Option::fixPathToTargetOS(
|
|
replaceExtraCompilerVariables(rc, tinp, out, NoShell), false));
|
|
}
|
|
}
|
|
if(project->isActiveConfig("no_delete_multiple_files")) {
|
|
cleans = dels;
|
|
} else {
|
|
QString files;
|
|
const int commandlineLimit = 2047; // NT limit, expanded
|
|
foreach (const QString &file, dels) {
|
|
if(del_statement.length() + files.length() +
|
|
qMax(fixEnvVariables(file).length(), file.length()) > commandlineLimit) {
|
|
cleans.append(files);
|
|
files.clear();
|
|
}
|
|
files += file;
|
|
}
|
|
if(!files.isEmpty())
|
|
cleans.append(files);
|
|
}
|
|
}
|
|
if(!cleans.isEmpty())
|
|
t << valGlue(cleans, "\n\t" + del_statement, "\n\t" + del_statement, "");
|
|
if(!wrote_clean_cmds) {
|
|
for (ProStringList::ConstIterator input = tmp_inputs.begin(); input != tmp_inputs.end(); ++input) {
|
|
QString tinp = (*input).toQString();
|
|
t << "\n\t" << replaceExtraCompilerVariables(tmp_clean_cmds, tinp,
|
|
replaceExtraCompilerVariables(tmp_out, tinp, QString(), NoShell), TargetShell);
|
|
}
|
|
}
|
|
}
|
|
t << endl;
|
|
}
|
|
QStringList tmp_dep = project->values(ProKey(*it + ".depends")).toQStringList();
|
|
if (config.indexOf("combine") != -1) {
|
|
if (tmp_out.contains(QRegExp("(^|[^$])\\$\\{QMAKE_(?!VAR_)"))) {
|
|
warn_msg(WarnLogic, "QMAKE_EXTRA_COMPILERS(%s) with combine has variable output.",
|
|
(*it).toLatin1().constData());
|
|
continue;
|
|
}
|
|
QStringList deps, inputs;
|
|
if(!tmp_dep.isEmpty())
|
|
deps += fileFixify(tmp_dep, FileFixifyFromOutdir);
|
|
for (ProStringList::ConstIterator input = tmp_inputs.begin(); input != tmp_inputs.end(); ++input) {
|
|
QString inpf = (*input).toQString();
|
|
deps += findDependencies(inpf);
|
|
inputs += Option::fixPathToTargetOS(inpf, false);
|
|
if(!tmp_dep_cmd.isEmpty() && doDepends()) {
|
|
char buff[256];
|
|
QString dep_cmd = replaceExtraCompilerVariables(tmp_dep_cmd, inpf, tmp_out, LocalShell);
|
|
dep_cmd = dep_cd_cmd + fixEnvVariables(dep_cmd);
|
|
if (FILE *proc = QT_POPEN(dep_cmd.toLatin1().constData(), QT_POPEN_READ)) {
|
|
QString indeps;
|
|
while(!feof(proc)) {
|
|
int read_in = (int)fread(buff, 1, 255, proc);
|
|
if(!read_in)
|
|
break;
|
|
indeps += QByteArray(buff, read_in);
|
|
}
|
|
QT_PCLOSE(proc);
|
|
if(!indeps.isEmpty()) {
|
|
// ### This is basically fubar. Add 'lines' flag to CONFIG?
|
|
QStringList dep_cmd_deps = indeps.replace('\n', ' ').simplified().split(' ');
|
|
for(int i = 0; i < dep_cmd_deps.count(); ++i) {
|
|
QString &file = dep_cmd_deps[i];
|
|
QString absFile = QDir(Option::output_dir).absoluteFilePath(file);
|
|
if (exists(absFile)) {
|
|
file = absFile;
|
|
} else {
|
|
QString localFile;
|
|
QList<QMakeLocalFileName> depdirs = QMakeSourceFileInfo::dependencyPaths();
|
|
for (QList<QMakeLocalFileName>::Iterator dit = depdirs.begin();
|
|
dit != depdirs.end(); ++dit) {
|
|
if (exists((*dit).local() + '/' + file)) {
|
|
localFile = (*dit).local() + '/' + file;
|
|
break;
|
|
}
|
|
}
|
|
if (localFile.isEmpty()) {
|
|
if (exists(file))
|
|
warn_msg(WarnDeprecated, ".depend_command for extra compiler %s"
|
|
" prints paths relative to source directory",
|
|
(*it).toLatin1().constData());
|
|
else
|
|
file.clear();
|
|
} else {
|
|
file = localFile;
|
|
}
|
|
}
|
|
if(!file.isEmpty())
|
|
file = fileFixify(file);
|
|
}
|
|
deps += dep_cmd_deps;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for(int i = 0; i < inputs.size(); ) {
|
|
if(tmp_out == inputs.at(i))
|
|
inputs.removeAt(i);
|
|
else
|
|
++i;
|
|
}
|
|
for(int i = 0; i < deps.size(); ) {
|
|
if(tmp_out == deps.at(i))
|
|
deps.removeAt(i);
|
|
else
|
|
++i;
|
|
}
|
|
if (inputs.isEmpty())
|
|
continue;
|
|
|
|
QString out = replaceExtraCompilerVariables(tmp_out, QString(), QString(), NoShell);
|
|
QString cmd = replaceExtraCompilerVariables(tmp_cmd, inputs, QStringList() << out, TargetShell);
|
|
t << escapeDependencyPath(Option::fixPathToTargetOS(out)) << ":";
|
|
// compiler.CONFIG+=explicit_dependencies means that ONLY compiler.depends gets to cause Makefile dependencies
|
|
if (config.indexOf("explicit_dependencies") != -1) {
|
|
t << " " << valList(escapeDependencyPaths(fileFixify(tmp_dep, FileFixifyFromOutdir)));
|
|
} else {
|
|
t << " " << valList(escapeDependencyPaths(inputs)) << " " << valList(escapeDependencyPaths(deps));
|
|
}
|
|
t << "\n\t" << cmd << endl << endl;
|
|
continue;
|
|
}
|
|
for (ProStringList::ConstIterator input = tmp_inputs.begin(); input != tmp_inputs.end(); ++input) {
|
|
QString inpf = (*input).toQString();
|
|
QString in = Option::fixPathToTargetOS(inpf, false);
|
|
QStringList deps = findDependencies(inpf);
|
|
deps << in;
|
|
QString out = Option::fixPathToTargetOS(replaceExtraCompilerVariables(tmp_out, inpf, QString(), NoShell));
|
|
if(!tmp_dep.isEmpty()) {
|
|
QStringList pre_deps = fileFixify(tmp_dep, FileFixifyFromOutdir);
|
|
for(int i = 0; i < pre_deps.size(); ++i)
|
|
deps << replaceExtraCompilerVariables(pre_deps.at(i), inpf, out, NoShell);
|
|
}
|
|
QString cmd = replaceExtraCompilerVariables(tmp_cmd, inpf, out, TargetShell);
|
|
// NOTE: The var -> QMAKE_COMP_var replace feature is unsupported, do not use!
|
|
for (ProStringList::ConstIterator it3 = vars.constBegin(); it3 != vars.constEnd(); ++it3)
|
|
cmd.replace("$(" + (*it3) + ")", "$(QMAKE_COMP_" + (*it3)+")");
|
|
if(!tmp_dep_cmd.isEmpty() && doDepends()) {
|
|
char buff[256];
|
|
QString dep_cmd = replaceExtraCompilerVariables(tmp_dep_cmd, inpf, out, LocalShell);
|
|
dep_cmd = dep_cd_cmd + fixEnvVariables(dep_cmd);
|
|
if (FILE *proc = QT_POPEN(dep_cmd.toLatin1().constData(), QT_POPEN_READ)) {
|
|
QString indeps;
|
|
while(!feof(proc)) {
|
|
int read_in = (int)fread(buff, 1, 255, proc);
|
|
if(!read_in)
|
|
break;
|
|
indeps += QByteArray(buff, read_in);
|
|
}
|
|
QT_PCLOSE(proc);
|
|
if(!indeps.isEmpty()) {
|
|
// ### This is basically fubar. Add 'lines' flag to CONFIG?
|
|
QStringList dep_cmd_deps = indeps.replace('\n', ' ').simplified().split(' ');
|
|
for(int i = 0; i < dep_cmd_deps.count(); ++i) {
|
|
QString &file = dep_cmd_deps[i];
|
|
QString absFile = QDir(Option::output_dir).absoluteFilePath(file);
|
|
if (exists(absFile)) {
|
|
file = absFile;
|
|
} else {
|
|
QString localFile;
|
|
QList<QMakeLocalFileName> depdirs = QMakeSourceFileInfo::dependencyPaths();
|
|
for (QList<QMakeLocalFileName>::Iterator dit = depdirs.begin();
|
|
dit != depdirs.end(); ++dit) {
|
|
if (exists((*dit).local() + '/' + file)) {
|
|
localFile = (*dit).local() + '/' + file;
|
|
break;
|
|
}
|
|
}
|
|
if (localFile.isEmpty()) {
|
|
if (exists(file))
|
|
warn_msg(WarnDeprecated, ".depend_command for extra compiler %s"
|
|
" prints paths relative to source directory",
|
|
(*it).toLatin1().constData());
|
|
else
|
|
file.clear();
|
|
} else {
|
|
file = localFile;
|
|
}
|
|
}
|
|
if(!file.isEmpty())
|
|
file = fileFixify(file);
|
|
}
|
|
deps += dep_cmd_deps;
|
|
}
|
|
}
|
|
//use the depend system to find includes of these included files
|
|
QStringList inc_deps;
|
|
for(int i = 0; i < deps.size(); ++i) {
|
|
const QString dep = deps.at(i);
|
|
if(QFile::exists(dep)) {
|
|
SourceFileType type = TYPE_UNKNOWN;
|
|
if(type == TYPE_UNKNOWN) {
|
|
for(QStringList::Iterator cit = Option::c_ext.begin();
|
|
cit != Option::c_ext.end(); ++cit) {
|
|
if(dep.endsWith((*cit))) {
|
|
type = TYPE_C;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(type == TYPE_UNKNOWN) {
|
|
for(QStringList::Iterator cppit = Option::cpp_ext.begin();
|
|
cppit != Option::cpp_ext.end(); ++cppit) {
|
|
if(dep.endsWith((*cppit))) {
|
|
type = TYPE_C;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(type == TYPE_UNKNOWN) {
|
|
for(QStringList::Iterator hit = Option::h_ext.begin();
|
|
type == TYPE_UNKNOWN && hit != Option::h_ext.end(); ++hit) {
|
|
if(dep.endsWith((*hit))) {
|
|
type = TYPE_C;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(type != TYPE_UNKNOWN) {
|
|
if(!QMakeSourceFileInfo::containsSourceFile(dep, type))
|
|
QMakeSourceFileInfo::addSourceFile(dep, type);
|
|
inc_deps += QMakeSourceFileInfo::dependencies(dep);
|
|
}
|
|
}
|
|
}
|
|
deps += inc_deps;
|
|
}
|
|
for(int i = 0; i < deps.size(); ) {
|
|
QString &dep = deps[i];
|
|
dep = Option::fixPathToTargetOS(dep, false);
|
|
if(out == dep)
|
|
deps.removeAt(i);
|
|
else
|
|
++i;
|
|
}
|
|
t << escapeDependencyPath(out) << ": " << valList(escapeDependencyPaths(deps)) << "\n\t"
|
|
<< cmd << endl << endl;
|
|
}
|
|
}
|
|
t << "compiler_clean: " << clean_targets << endl << endl;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeExtraCompilerVariables(QTextStream &t)
|
|
{
|
|
bool first = true;
|
|
const ProStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
|
|
for (ProStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
|
|
const ProStringList &vars = project->values(ProKey(*it + ".variables"));
|
|
for (ProStringList::ConstIterator varit = vars.begin(); varit != vars.end(); ++varit) {
|
|
if(first) {
|
|
t << "\n####### Custom Compiler Variables\n";
|
|
first = false;
|
|
}
|
|
t << "QMAKE_COMP_" << (*varit) << " = "
|
|
<< valList(project->values((*varit).toKey())) << endl;
|
|
}
|
|
}
|
|
if(!first)
|
|
t << endl;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeExtraVariables(QTextStream &t)
|
|
{
|
|
t << endl;
|
|
|
|
ProStringList outlist;
|
|
const ProValueMap &vars = project->variables();
|
|
const ProStringList &exports = project->values("QMAKE_EXTRA_VARIABLES");
|
|
for (ProValueMap::ConstIterator it = vars.begin(); it != vars.end(); ++it) {
|
|
for (ProStringList::ConstIterator exp_it = exports.begin(); exp_it != exports.end(); ++exp_it) {
|
|
QRegExp rx((*exp_it).toQString(), Qt::CaseInsensitive, QRegExp::Wildcard);
|
|
if (rx.exactMatch(it.key().toQString()))
|
|
outlist << ("EXPORT_" + it.key() + " = " + it.value().join(' '));
|
|
}
|
|
}
|
|
if (!outlist.isEmpty()) {
|
|
t << "####### Custom Variables\n";
|
|
t << outlist.join("\n") << endl << endl;
|
|
}
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::writeDummyMakefile(QTextStream &t)
|
|
{
|
|
if (project->values("QMAKE_FAILED_REQUIREMENTS").isEmpty())
|
|
return false;
|
|
t << "QMAKE = " << var("QMAKE_QMAKE") << endl;
|
|
const ProStringList &qut = project->values("QMAKE_EXTRA_TARGETS");
|
|
for (ProStringList::ConstIterator it = qut.begin(); it != qut.end(); ++it)
|
|
t << *it << " ";
|
|
t << "first all clean install distclean uninstall qmake_all:\n\t"
|
|
<< "@echo \"Some of the required modules ("
|
|
<< var("QMAKE_FAILED_REQUIREMENTS") << ") are not available.\"\n\t"
|
|
<< "@echo \"Skipped.\"\n\n";
|
|
writeMakeQmake(t);
|
|
t << "FORCE:\n\n";
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::writeStubMakefile(QTextStream &t)
|
|
{
|
|
t << "QMAKE = " << var("QMAKE_QMAKE") << endl;
|
|
const ProStringList &qut = project->values("QMAKE_EXTRA_TARGETS");
|
|
for (ProStringList::ConstIterator it = qut.begin(); it != qut.end(); ++it)
|
|
t << *it << " ";
|
|
//const QString ofile = Option::fixPathToTargetOS(fileFixify(Option::output.fileName()));
|
|
t << "first all clean install distclean uninstall: qmake\n"
|
|
<< "qmake_all:\n";
|
|
writeMakeQmake(t);
|
|
t << "FORCE:\n\n";
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::writeMakefile(QTextStream &t)
|
|
{
|
|
t << "####### Compile\n\n";
|
|
writeObj(t, "SOURCES");
|
|
writeObj(t, "GENERATED_SOURCES");
|
|
|
|
t << "####### Install\n\n";
|
|
writeInstalls(t);
|
|
|
|
t << "FORCE:\n\n";
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeDefaultVariables(QTextStream &t)
|
|
{
|
|
t << "QMAKE = " << var("QMAKE_QMAKE") << endl;
|
|
t << "DEL_FILE = " << var("QMAKE_DEL_FILE") << endl;
|
|
t << "CHK_DIR_EXISTS= " << var("QMAKE_CHK_DIR_EXISTS") << endl;
|
|
t << "MKDIR = " << var("QMAKE_MKDIR") << endl;
|
|
t << "COPY = " << var("QMAKE_COPY") << endl;
|
|
t << "COPY_FILE = " << var("QMAKE_COPY_FILE") << endl;
|
|
t << "COPY_DIR = " << var("QMAKE_COPY_DIR") << endl;
|
|
t << "INSTALL_FILE = " << var("QMAKE_INSTALL_FILE") << endl;
|
|
t << "INSTALL_PROGRAM = " << var("QMAKE_INSTALL_PROGRAM") << endl;
|
|
t << "INSTALL_DIR = " << var("QMAKE_INSTALL_DIR") << endl;
|
|
t << "DEL_FILE = " << var("QMAKE_DEL_FILE") << endl;
|
|
t << "SYMLINK = " << var("QMAKE_SYMBOLIC_LINK") << endl;
|
|
t << "DEL_DIR = " << var("QMAKE_DEL_DIR") << endl;
|
|
t << "MOVE = " << var("QMAKE_MOVE") << endl;
|
|
}
|
|
|
|
QString MakefileGenerator::buildArgs()
|
|
{
|
|
QString ret;
|
|
|
|
foreach (const QString &arg, Option::globals->qmake_args)
|
|
ret += " " + shellQuote(arg);
|
|
return ret;
|
|
}
|
|
|
|
//could get stored argv, but then it would have more options than are
|
|
//probably necesary this will try to guess the bare minimum..
|
|
QString MakefileGenerator::build_args()
|
|
{
|
|
QString ret = "$(QMAKE)";
|
|
|
|
// general options and arguments
|
|
ret += buildArgs();
|
|
|
|
//output
|
|
QString ofile = fileFixify(Option::output.fileName());
|
|
if(!ofile.isEmpty() && ofile != project->first("QMAKE_MAKEFILE"))
|
|
ret += " -o " + escapeFilePath(ofile);
|
|
|
|
//inputs
|
|
ret += " " + escapeFilePath(fileFixify(project->projectFile()));
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeHeader(QTextStream &t)
|
|
{
|
|
t << "#############################################################################\n";
|
|
t << "# Makefile for building: " << escapeFilePath(var("TARGET")) << endl;
|
|
t << "# Generated by qmake (" QMAKE_VERSION_STR ") (Qt " QT_VERSION_STR ")\n";
|
|
t << "# Project: " << fileFixify(project->projectFile()) << endl;
|
|
t << "# Template: " << var("TEMPLATE") << endl;
|
|
if(!project->isActiveConfig("build_pass"))
|
|
t << "# Command: " << build_args().replace(QLatin1String("$(QMAKE)"), var("QMAKE_QMAKE")) << endl;
|
|
t << "#############################################################################\n";
|
|
t << endl;
|
|
QString ofile = Option::fixPathToTargetOS(Option::output.fileName());
|
|
if (ofile.lastIndexOf(Option::dir_sep) != -1)
|
|
ofile.remove(0, ofile.lastIndexOf(Option::dir_sep) +1);
|
|
t << "MAKEFILE = " << escapeFilePath(ofile) << endl << endl;
|
|
}
|
|
|
|
QList<MakefileGenerator::SubTarget*>
|
|
MakefileGenerator::findSubDirsSubTargets() const
|
|
{
|
|
QList<SubTarget*> targets;
|
|
{
|
|
const ProStringList &subdirs = project->values("SUBDIRS");
|
|
for(int subdir = 0; subdir < subdirs.size(); ++subdir) {
|
|
ProString ofile = subdirs[subdir];
|
|
QString oname = ofile.toQString();
|
|
QString fixedSubdir = oname;
|
|
fixedSubdir.replace(QRegExp("[^a-zA-Z0-9_]"),"-");
|
|
|
|
SubTarget *st = new SubTarget;
|
|
st->name = oname;
|
|
targets.append(st);
|
|
|
|
bool fromFile = false;
|
|
const ProKey fkey(fixedSubdir + ".file");
|
|
const ProKey skey(fixedSubdir + ".subdir");
|
|
if (!project->isEmpty(fkey)) {
|
|
if (!project->isEmpty(skey))
|
|
warn_msg(WarnLogic, "Cannot assign both file and subdir for subdir %s",
|
|
subdirs[subdir].toLatin1().constData());
|
|
ofile = project->first(fkey);
|
|
fromFile = true;
|
|
} else if (!project->isEmpty(skey)) {
|
|
ofile = project->first(skey);
|
|
fromFile = false;
|
|
} else {
|
|
fromFile = ofile.endsWith(Option::pro_ext);
|
|
}
|
|
QString file = Option::fixPathToTargetOS(ofile.toQString());
|
|
|
|
if(fromFile) {
|
|
int slsh = file.lastIndexOf(Option::dir_sep);
|
|
if(slsh != -1) {
|
|
st->in_directory = file.left(slsh+1);
|
|
st->profile = file.mid(slsh+1);
|
|
} else {
|
|
st->profile = file;
|
|
}
|
|
} else {
|
|
if(!file.isEmpty() && !project->isActiveConfig("subdir_first_pro"))
|
|
st->profile = file.section(Option::dir_sep, -1) + Option::pro_ext;
|
|
st->in_directory = file;
|
|
}
|
|
while(st->in_directory.endsWith(Option::dir_sep))
|
|
st->in_directory.chop(1);
|
|
if(fileInfo(st->in_directory).isRelative())
|
|
st->out_directory = st->in_directory;
|
|
else
|
|
st->out_directory = fileFixify(st->in_directory, FileFixifyBackwards);
|
|
const ProKey mkey(fixedSubdir + ".makefile");
|
|
if (!project->isEmpty(mkey)) {
|
|
st->makefile = project->first(mkey).toQString();
|
|
} else {
|
|
st->makefile = "Makefile";
|
|
if(!st->profile.isEmpty()) {
|
|
QString basename = st->in_directory;
|
|
int new_slsh = basename.lastIndexOf(Option::dir_sep);
|
|
if(new_slsh != -1)
|
|
basename = basename.mid(new_slsh+1);
|
|
if(st->profile != basename + Option::pro_ext)
|
|
st->makefile += "." + st->profile.left(st->profile.length() - Option::pro_ext.length());
|
|
}
|
|
}
|
|
const ProKey dkey(fixedSubdir + ".depends");
|
|
if (!project->isEmpty(dkey)) {
|
|
const ProStringList &depends = project->values(dkey);
|
|
for(int depend = 0; depend < depends.size(); ++depend) {
|
|
bool found = false;
|
|
for(int subDep = 0; subDep < subdirs.size(); ++subDep) {
|
|
if(subdirs[subDep] == depends.at(depend)) {
|
|
QString subName = subdirs[subDep].toQString();
|
|
QString fixedSubDep = subName;
|
|
fixedSubDep.replace(QRegExp("[^a-zA-Z0-9_]"),"-");
|
|
const ProKey dtkey(fixedSubDep + ".target");
|
|
if (!project->isEmpty(dtkey)) {
|
|
st->depends += project->first(dtkey);
|
|
} else {
|
|
QString d = Option::fixPathToTargetOS(subName);
|
|
const ProKey dfkey(fixedSubDep + ".file");
|
|
if (!project->isEmpty(dfkey)) {
|
|
d = project->first(dfkey).toQString();
|
|
} else {
|
|
const ProKey dskey(fixedSubDep + ".subdir");
|
|
if (!project->isEmpty(dskey))
|
|
d = project->first(dskey).toQString();
|
|
}
|
|
st->depends += "sub-" + d.replace(QRegExp("[^a-zA-Z0-9_]"),"-");
|
|
}
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if(!found) {
|
|
QString depend_str = depends.at(depend).toQString();
|
|
st->depends += depend_str.replace(QRegExp("[^a-zA-Z0-9_]"),"-");
|
|
}
|
|
}
|
|
}
|
|
const ProKey tkey(fixedSubdir + ".target");
|
|
if (!project->isEmpty(tkey)) {
|
|
st->target = project->first(tkey).toQString();
|
|
} else {
|
|
st->target = "sub-" + file;
|
|
st->target.replace(QRegExp("[^a-zA-Z0-9_]"), "-");
|
|
}
|
|
}
|
|
}
|
|
return targets;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeSubDirs(QTextStream &t)
|
|
{
|
|
QList<SubTarget*> targets = findSubDirsSubTargets();
|
|
t << "first: make_first\n";
|
|
int flags = SubTargetInstalls;
|
|
if(project->isActiveConfig("ordered"))
|
|
flags |= SubTargetOrdered;
|
|
writeSubTargets(t, targets, flags);
|
|
qDeleteAll(targets);
|
|
}
|
|
|
|
void MakefileGenerator::writeSubMakeCall(QTextStream &t, const QString &callPrefix,
|
|
const QString &makeArguments)
|
|
{
|
|
t << callPrefix << "$(MAKE)" << makeArguments << endl;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeSubTargetCall(QTextStream &t,
|
|
const QString &in_directory, const QString &in, const QString &out_directory, const QString &out,
|
|
const QString &out_directory_cdin, const QString &makefilein)
|
|
{
|
|
QString pfx;
|
|
if (!in.isEmpty()) {
|
|
if (!in_directory.isEmpty())
|
|
t << "\n\t" << mkdir_p_asstring(out_directory);
|
|
pfx = "( " + chkexists.arg(out) +
|
|
+ " $(QMAKE) " + in + buildArgs() + " -o " + out
|
|
+ " ) && ";
|
|
}
|
|
writeSubMakeCall(t, out_directory_cdin + pfx, makefilein);
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeSubTargets(QTextStream &t, QList<MakefileGenerator::SubTarget*> targets, int flags)
|
|
{
|
|
// blasted includes
|
|
const ProStringList &qeui = project->values("QMAKE_EXTRA_INCLUDES");
|
|
for (ProStringList::ConstIterator qeui_it = qeui.begin(); qeui_it != qeui.end(); ++qeui_it)
|
|
t << "include " << (*qeui_it) << endl;
|
|
|
|
if (!(flags & SubTargetSkipDefaultVariables)) {
|
|
writeDefaultVariables(t);
|
|
t << "SUBTARGETS = "; // subtargets are sub-directory
|
|
for(int target = 0; target < targets.size(); ++target)
|
|
t << " \\\n\t\t" << targets.at(target)->target;
|
|
t << endl << endl;
|
|
}
|
|
writeExtraVariables(t);
|
|
|
|
QStringList targetSuffixes;
|
|
const QString abs_source_path = project->first("QMAKE_ABSOLUTE_SOURCE_PATH").toQString();
|
|
if (!(flags & SubTargetSkipDefaultTargets)) {
|
|
targetSuffixes << "make_first" << "all" << "clean" << "distclean"
|
|
<< QString((flags & SubTargetInstalls) ? "install_subtargets" : "install")
|
|
<< QString((flags & SubTargetInstalls) ? "uninstall_subtargets" : "uninstall");
|
|
}
|
|
|
|
bool dont_recurse = project->isActiveConfig("dont_recurse");
|
|
|
|
// generate target rules
|
|
for(int target = 0; target < targets.size(); ++target) {
|
|
SubTarget *subtarget = targets.at(target);
|
|
QString in_directory = subtarget->in_directory;
|
|
if(!in_directory.isEmpty() && !in_directory.endsWith(Option::dir_sep))
|
|
in_directory += Option::dir_sep;
|
|
QString out_directory = subtarget->out_directory;
|
|
if(!out_directory.isEmpty() && !out_directory.endsWith(Option::dir_sep))
|
|
out_directory += Option::dir_sep;
|
|
if(!abs_source_path.isEmpty() && out_directory.startsWith(abs_source_path))
|
|
out_directory = Option::output_dir + out_directory.mid(abs_source_path.length());
|
|
|
|
QString out_directory_cdin = out_directory.isEmpty() ? "\n\t"
|
|
: "\n\tcd " + escapeFilePath(out_directory) + " && ";
|
|
QString makefilein = " -f " + escapeFilePath(subtarget->makefile);
|
|
|
|
//qmake it
|
|
QString out;
|
|
QString in;
|
|
if(!subtarget->profile.isEmpty()) {
|
|
out = subtarget->makefile;
|
|
in = escapeFilePath(fileFixify(in_directory + subtarget->profile, FileFixifyAbsolute));
|
|
if(out.startsWith(in_directory))
|
|
out = out.mid(in_directory.length());
|
|
out = escapeFilePath(out);
|
|
t << subtarget->target << "-qmake_all: ";
|
|
if (flags & SubTargetOrdered) {
|
|
if (target)
|
|
t << targets.at(target - 1)->target << "-qmake_all";
|
|
} else {
|
|
if (!subtarget->depends.isEmpty())
|
|
t << valGlue(subtarget->depends, QString(), "-qmake_all ", "-qmake_all");
|
|
}
|
|
t << " FORCE\n\t";
|
|
if(!in_directory.isEmpty()) {
|
|
t << mkdir_p_asstring(out_directory)
|
|
<< out_directory_cdin;
|
|
}
|
|
t << "$(QMAKE) " << in << buildArgs() << " -o " << out;
|
|
if (!dont_recurse)
|
|
writeSubMakeCall(t, out_directory_cdin, makefilein + " qmake_all");
|
|
else
|
|
t << endl;
|
|
}
|
|
|
|
{ //actually compile
|
|
t << subtarget->target << ":";
|
|
if(!subtarget->depends.isEmpty())
|
|
t << " " << valList(subtarget->depends);
|
|
t << " FORCE";
|
|
writeSubTargetCall(t, in_directory, in, out_directory, out,
|
|
out_directory_cdin, makefilein);
|
|
}
|
|
|
|
for(int suffix = 0; suffix < targetSuffixes.size(); ++suffix) {
|
|
QString s = targetSuffixes.at(suffix);
|
|
if(s == "install_subtargets")
|
|
s = "install";
|
|
else if(s == "uninstall_subtargets")
|
|
s = "uninstall";
|
|
else if(s == "make_first")
|
|
s = QString();
|
|
|
|
if(flags & SubTargetOrdered) {
|
|
t << subtarget->target << "-" << targetSuffixes.at(suffix) << "-ordered:";
|
|
if(target)
|
|
t << " " << targets.at(target-1)->target << "-" << targetSuffixes.at(suffix) << "-ordered ";
|
|
t << " FORCE";
|
|
writeSubTargetCall(t, in_directory, in, out_directory, out,
|
|
out_directory_cdin, makefilein + " " + s);
|
|
}
|
|
t << subtarget->target << "-" << targetSuffixes.at(suffix) << ":";
|
|
if(!subtarget->depends.isEmpty())
|
|
t << " " << valGlue(subtarget->depends, QString(), "-" + targetSuffixes.at(suffix) + " ",
|
|
"-"+targetSuffixes.at(suffix));
|
|
t << " FORCE";
|
|
writeSubTargetCall(t, in_directory, in, out_directory, out,
|
|
out_directory_cdin, makefilein + " " + s);
|
|
}
|
|
}
|
|
t << endl;
|
|
|
|
if (!(flags & SubTargetSkipDefaultTargets)) {
|
|
writeMakeQmake(t, true);
|
|
|
|
t << "qmake_all:";
|
|
if(!targets.isEmpty()) {
|
|
for(QList<SubTarget*>::Iterator it = targets.begin(); it != targets.end(); ++it) {
|
|
if(!(*it)->profile.isEmpty())
|
|
t << " " << (*it)->target << "-qmake_all";
|
|
}
|
|
}
|
|
t << " FORCE\n\n";
|
|
}
|
|
|
|
for(int s = 0; s < targetSuffixes.size(); ++s) {
|
|
QString suffix = targetSuffixes.at(s);
|
|
if(!(flags & SubTargetInstalls) && suffix.endsWith("install"))
|
|
continue;
|
|
|
|
t << suffix << ":";
|
|
for(int target = 0; target < targets.size(); ++target) {
|
|
SubTarget *subTarget = targets.at(target);
|
|
const ProStringList &config = project->values(ProKey(subTarget->name + ".CONFIG"));
|
|
if (suffix == "make_first"
|
|
&& config.indexOf("no_default_target") != -1) {
|
|
continue;
|
|
}
|
|
if((suffix == "install_subtargets" || suffix == "uninstall_subtargets")
|
|
&& config.indexOf("no_default_install") != -1) {
|
|
continue;
|
|
}
|
|
QString targetRule = subTarget->target + "-" + suffix;
|
|
if(flags & SubTargetOrdered)
|
|
targetRule += "-ordered";
|
|
t << " " << targetRule;
|
|
}
|
|
if(suffix == "all" || suffix == "make_first")
|
|
t << ' ' << depVar("ALL_DEPS");
|
|
if(suffix == "clean")
|
|
t << ' ' << depVar("CLEAN_DEPS");
|
|
else if (suffix == "distclean")
|
|
t << ' ' << depVar("DISTCLEAN_DEPS");
|
|
t << " FORCE\n";
|
|
if(suffix == "clean") {
|
|
t << fixFileVarGlue("QMAKE_CLEAN", "\t-$(DEL_FILE) ", "\n\t-$(DEL_FILE) ", "\n");
|
|
} else if(suffix == "distclean") {
|
|
QString ofile = fileFixify(Option::output.fileName());
|
|
if(!ofile.isEmpty())
|
|
t << "\t-$(DEL_FILE) " << escapeFilePath(ofile) << endl;
|
|
t << fixFileVarGlue("QMAKE_DISTCLEAN", "\t-$(DEL_FILE) ", " ", "\n");
|
|
}
|
|
}
|
|
|
|
// user defined targets
|
|
const ProStringList &qut = project->values("QMAKE_EXTRA_TARGETS");
|
|
for (ProStringList::ConstIterator qut_it = qut.begin(); qut_it != qut.end(); ++qut_it) {
|
|
const ProStringList &config = project->values(ProKey(*qut_it + ".CONFIG"));
|
|
QString targ = var(ProKey(*qut_it + ".target")),
|
|
cmd = var(ProKey(*qut_it + ".commands")), deps;
|
|
if(targ.isEmpty())
|
|
targ = (*qut_it).toQString();
|
|
t << endl;
|
|
|
|
const ProStringList &deplist = project->values(ProKey(*qut_it + ".depends"));
|
|
for (ProStringList::ConstIterator dep_it = deplist.begin(); dep_it != deplist.end(); ++dep_it) {
|
|
QString dep = var(ProKey(*dep_it + ".target"));
|
|
if(dep.isEmpty())
|
|
dep = Option::fixPathToTargetOS((*dep_it).toQString(), false);
|
|
deps += ' ' + escapeDependencyPath(dep);
|
|
}
|
|
if (config.indexOf("recursive") != -1) {
|
|
QSet<QString> recurse;
|
|
const ProKey rkey(*qut_it + ".recurse");
|
|
if (project->isSet(rkey)) {
|
|
recurse = project->values(rkey).toQStringList().toSet();
|
|
} else {
|
|
for(int target = 0; target < targets.size(); ++target)
|
|
recurse.insert(targets.at(target)->name);
|
|
}
|
|
for(int target = 0; target < targets.size(); ++target) {
|
|
SubTarget *subtarget = targets.at(target);
|
|
QString in_directory = subtarget->in_directory;
|
|
if(!in_directory.isEmpty() && !in_directory.endsWith(Option::dir_sep))
|
|
in_directory += Option::dir_sep;
|
|
QString out_directory = subtarget->out_directory;
|
|
if(!out_directory.isEmpty() && !out_directory.endsWith(Option::dir_sep))
|
|
out_directory += Option::dir_sep;
|
|
if(!abs_source_path.isEmpty() && out_directory.startsWith(abs_source_path))
|
|
out_directory = Option::output_dir + out_directory.mid(abs_source_path.length());
|
|
|
|
if(!recurse.contains(subtarget->name))
|
|
continue;
|
|
|
|
QString out_directory_cdin = out_directory.isEmpty() ? "\n\t"
|
|
: "\n\tcd " + escapeFilePath(out_directory) + " && ";
|
|
QString makefilein = " -f " + escapeFilePath(subtarget->makefile);
|
|
|
|
QString out;
|
|
QString in;
|
|
if (!subtarget->profile.isEmpty()) {
|
|
out = subtarget->makefile;
|
|
in = escapeFilePath(fileFixify(in_directory + subtarget->profile, FileFixifyAbsolute));
|
|
if (out.startsWith(in_directory))
|
|
out = out.mid(in_directory.length());
|
|
out = escapeFilePath(out);
|
|
}
|
|
|
|
//write the rule/depends
|
|
if(flags & SubTargetOrdered) {
|
|
const QString dep = subtarget->target + "-" + (*qut_it) + "_ordered";
|
|
t << dep << ":";
|
|
if(target)
|
|
t << " " << targets.at(target-1)->target << "-" << (*qut_it) << "_ordered ";
|
|
deps += " " + dep;
|
|
} else {
|
|
const QString dep = subtarget->target + "-" + (*qut_it);
|
|
t << dep << ":";
|
|
if(!subtarget->depends.isEmpty())
|
|
t << " " << valGlue(subtarget->depends, QString(), "-" + (*qut_it) + " ", "-" + (*qut_it));
|
|
deps += " " + dep;
|
|
}
|
|
|
|
QString sub_targ = targ;
|
|
const ProKey rtkey(*qut_it + ".recurse_target");
|
|
if (project->isSet(rtkey))
|
|
sub_targ = project->first(rtkey).toQString();
|
|
|
|
//write the commands
|
|
writeSubTargetCall(t, in_directory, in, out_directory, out,
|
|
out_directory_cdin, makefilein + " " + sub_targ);
|
|
}
|
|
}
|
|
if (config.indexOf("phony") != -1)
|
|
deps += " FORCE";
|
|
t << targ << ":" << deps << "\n";
|
|
if(!cmd.isEmpty())
|
|
t << "\t" << cmd << endl;
|
|
}
|
|
|
|
if(flags & SubTargetInstalls) {
|
|
project->values("INSTALLDEPS") += "install_subtargets";
|
|
project->values("UNINSTALLDEPS") += "uninstall_subtargets";
|
|
writeInstalls(t, true);
|
|
}
|
|
t << "FORCE:\n\n";
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writeMakeQmake(QTextStream &t, bool noDummyQmakeAll)
|
|
{
|
|
QString ofile = fileFixify(Option::output.fileName());
|
|
if(project->isEmpty("QMAKE_FAILED_REQUIREMENTS") && !project->isEmpty("QMAKE_INTERNAL_PRL_FILE")) {
|
|
QStringList files = escapeFilePaths(fileFixify(Option::mkfile::project_files));
|
|
t << escapeDependencyPath(project->first("QMAKE_INTERNAL_PRL_FILE").toQString()) << ": \n\t"
|
|
<< "@$(QMAKE) -prl " << buildArgs() << " " << files.join(' ') << endl;
|
|
}
|
|
|
|
QString qmake = build_args();
|
|
if(!ofile.isEmpty() && !project->isActiveConfig("no_autoqmake")) {
|
|
t << escapeDependencyPath(ofile) << ": "
|
|
<< escapeDependencyPath(fileFixify(project->projectFile())) << " ";
|
|
if (Option::globals->do_cache) {
|
|
if (!project->confFile().isEmpty())
|
|
t << escapeDependencyPath(fileFixify(project->confFile())) << " ";
|
|
if (!project->cacheFile().isEmpty())
|
|
t << escapeDependencyPath(fileFixify(project->cacheFile())) << " ";
|
|
}
|
|
if(!specdir().isEmpty()) {
|
|
if (exists(Option::normalizePath(specdir() + "/qmake.conf")))
|
|
t << escapeDependencyPath(specdir() + Option::dir_sep + "qmake.conf") << " ";
|
|
}
|
|
const ProStringList &included = escapeDependencyPaths(project->values("QMAKE_INTERNAL_INCLUDED_FILES"));
|
|
t << included.join(" \\\n\t\t") << "\n\t"
|
|
<< qmake << endl;
|
|
for(int include = 0; include < included.size(); ++include) {
|
|
const ProString &i = included.at(include);
|
|
if(!i.isEmpty())
|
|
t << i << ":\n";
|
|
}
|
|
}
|
|
if(project->first("QMAKE_ORIG_TARGET") != "qmake") {
|
|
t << "qmake: FORCE\n\t@" << qmake << endl << endl;
|
|
if (!noDummyQmakeAll)
|
|
t << "qmake_all: FORCE\n\n";
|
|
}
|
|
}
|
|
|
|
QFileInfo
|
|
MakefileGenerator::fileInfo(QString file) const
|
|
{
|
|
static QHash<FileInfoCacheKey, QFileInfo> *cache = 0;
|
|
static QFileInfo noInfo = QFileInfo();
|
|
if(!cache) {
|
|
cache = new QHash<FileInfoCacheKey, QFileInfo>;
|
|
qmakeAddCacheClear(qmakeDeleteCacheClear<QHash<FileInfoCacheKey, QFileInfo> >, (void**)&cache);
|
|
}
|
|
FileInfoCacheKey cacheKey(file);
|
|
QFileInfo value = cache->value(cacheKey, noInfo);
|
|
if (value != noInfo)
|
|
return value;
|
|
|
|
QFileInfo fi(file);
|
|
if (fi.exists())
|
|
cache->insert(cacheKey, fi);
|
|
return fi;
|
|
}
|
|
|
|
MakefileGenerator::LibFlagType
|
|
MakefileGenerator::parseLibFlag(const ProString &flag, ProString *arg)
|
|
{
|
|
if (flag.startsWith("-L")) {
|
|
*arg = flag.mid(2);
|
|
return LibFlagPath;
|
|
}
|
|
if (flag.startsWith("-l")) {
|
|
*arg = flag.mid(2);
|
|
return LibFlagLib;
|
|
}
|
|
if (flag.startsWith('-'))
|
|
return LibFlagOther;
|
|
return LibFlagFile;
|
|
}
|
|
|
|
ProStringList
|
|
MakefileGenerator::fixLibFlags(const ProKey &var)
|
|
{
|
|
ProStringList in = project->values(var);
|
|
ProStringList ret;
|
|
|
|
ret.reserve(in.length());
|
|
foreach (const ProString &v, in)
|
|
ret << fixLibFlag(v);
|
|
return ret;
|
|
}
|
|
|
|
ProString MakefileGenerator::fixLibFlag(const ProString &)
|
|
{
|
|
qFatal("MakefileGenerator::fixLibFlag() called");
|
|
return ProString();
|
|
}
|
|
|
|
ProString
|
|
MakefileGenerator::escapeFilePath(const ProString &path) const
|
|
{
|
|
return ProString(escapeFilePath(path.toQString()));
|
|
}
|
|
|
|
QStringList
|
|
MakefileGenerator::escapeFilePaths(const QStringList &paths) const
|
|
{
|
|
QStringList ret;
|
|
for(int i = 0; i < paths.size(); ++i)
|
|
ret.append(escapeFilePath(paths.at(i)));
|
|
return ret;
|
|
}
|
|
|
|
ProStringList
|
|
MakefileGenerator::escapeFilePaths(const ProStringList &paths) const
|
|
{
|
|
ProStringList ret;
|
|
for (int i = 0; i < paths.size(); ++i)
|
|
ret.append(escapeFilePath(paths.at(i)));
|
|
return ret;
|
|
}
|
|
|
|
ProString
|
|
MakefileGenerator::escapeDependencyPath(const ProString &path) const
|
|
{
|
|
return ProString(escapeDependencyPath(path.toQString()));
|
|
}
|
|
|
|
QStringList
|
|
MakefileGenerator::escapeDependencyPaths(const QStringList &paths) const
|
|
{
|
|
QStringList ret;
|
|
for(int i = 0; i < paths.size(); ++i)
|
|
ret.append(escapeDependencyPath(paths.at(i)));
|
|
return ret;
|
|
}
|
|
|
|
ProStringList
|
|
MakefileGenerator::escapeDependencyPaths(const ProStringList &paths) const
|
|
{
|
|
ProStringList ret;
|
|
for (int i = 0; i < paths.size(); ++i)
|
|
ret.append(escapeDependencyPath(paths.at(i).toQString()));
|
|
return ret;
|
|
}
|
|
|
|
QStringList
|
|
MakefileGenerator::fileFixify(const QStringList &files, FileFixifyTypes fix, bool canon) const
|
|
{
|
|
if(files.isEmpty())
|
|
return files;
|
|
QStringList ret;
|
|
for(QStringList::ConstIterator it = files.begin(); it != files.end(); ++it) {
|
|
if(!(*it).isEmpty())
|
|
ret << fileFixify((*it), fix, canon);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::fileFixify(const QString &file, FileFixifyTypes fix, bool canon) const
|
|
{
|
|
if(file.isEmpty())
|
|
return file;
|
|
QString ret = file;
|
|
|
|
//do the fixin'
|
|
QString orig_file = ret;
|
|
if(ret.startsWith(QLatin1Char('~'))) {
|
|
if(ret.startsWith(QLatin1String("~/")))
|
|
ret = QDir::homePath() + ret.mid(1);
|
|
else
|
|
warn_msg(WarnLogic, "Unable to expand ~ in %s", ret.toLatin1().constData());
|
|
}
|
|
if ((fix & FileFixifyAbsolute)
|
|
|| (!(fix & FileFixifyRelative) && project->isActiveConfig("no_fixpath"))) {
|
|
if ((fix & FileFixifyAbsolute) && QDir::isRelativePath(ret)) {
|
|
QString pwd = !(fix & FileFixifyFromOutdir) ? project->projectDir() : Option::output_dir;
|
|
{
|
|
QFileInfo in_fi(fileInfo(pwd));
|
|
if (in_fi.exists())
|
|
pwd = in_fi.canonicalFilePath();
|
|
}
|
|
if (!pwd.endsWith(QLatin1Char('/')))
|
|
pwd += QLatin1Char('/');
|
|
ret.prepend(pwd);
|
|
}
|
|
ret = Option::fixPathToTargetOS(ret, false, canon);
|
|
} else { //fix it..
|
|
QString out_dir = (fix & FileFixifyToIndir) ? project->projectDir() : Option::output_dir;
|
|
QString in_dir = !(fix & FileFixifyFromOutdir) ? project->projectDir() : Option::output_dir;
|
|
{
|
|
QFileInfo in_fi(fileInfo(in_dir));
|
|
if(in_fi.exists())
|
|
in_dir = in_fi.canonicalFilePath();
|
|
QFileInfo out_fi(fileInfo(out_dir));
|
|
if(out_fi.exists())
|
|
out_dir = out_fi.canonicalFilePath();
|
|
}
|
|
|
|
QString qfile(Option::normalizePath(ret));
|
|
QFileInfo qfileinfo(fileInfo(qfile));
|
|
if(out_dir != in_dir || !qfileinfo.isRelative()) {
|
|
if(qfileinfo.isRelative()) {
|
|
ret = in_dir + "/" + qfile;
|
|
qfileinfo.setFile(ret);
|
|
}
|
|
ret = Option::fixPathToTargetOS(ret, false, canon);
|
|
QString match_dir = Option::fixPathToTargetOS(out_dir, false, canon);
|
|
if(ret == match_dir) {
|
|
ret = "";
|
|
} else if(ret.startsWith(match_dir + Option::dir_sep)) {
|
|
ret = ret.mid(match_dir.length() + Option::dir_sep.length());
|
|
} else {
|
|
//figure out the depth
|
|
int depth = 4;
|
|
if(Option::qmake_mode == Option::QMAKE_GENERATE_MAKEFILE ||
|
|
Option::qmake_mode == Option::QMAKE_GENERATE_PRL) {
|
|
if(project && !project->isEmpty("QMAKE_PROJECT_DEPTH"))
|
|
depth = project->first("QMAKE_PROJECT_DEPTH").toInt();
|
|
else if(Option::mkfile::cachefile_depth != -1)
|
|
depth = Option::mkfile::cachefile_depth;
|
|
}
|
|
//calculate how much can be removed
|
|
QString dot_prefix;
|
|
for(int i = 1; i <= depth; i++) {
|
|
int sl = match_dir.lastIndexOf(Option::dir_sep);
|
|
if(sl == -1)
|
|
break;
|
|
match_dir = match_dir.left(sl);
|
|
if(match_dir.isEmpty())
|
|
break;
|
|
if(ret.startsWith(match_dir + Option::dir_sep)) {
|
|
//concat
|
|
int remlen = ret.length() - (match_dir.length() + 1);
|
|
if(remlen < 0)
|
|
remlen = 0;
|
|
ret = ret.right(remlen);
|
|
//prepend
|
|
for(int o = 0; o < i; o++)
|
|
dot_prefix += ".." + Option::dir_sep;
|
|
break;
|
|
}
|
|
}
|
|
ret.prepend(dot_prefix);
|
|
}
|
|
} else {
|
|
ret = Option::fixPathToTargetOS(ret, false, canon);
|
|
}
|
|
}
|
|
if(ret.isEmpty())
|
|
ret = ".";
|
|
debug_msg(3, "Fixed[%d,%d] %s :: to :: %s [%s::%s]",
|
|
int(fix), canon, orig_file.toLatin1().constData(), ret.toLatin1().constData(),
|
|
qmake_getpwd().toLatin1().constData(), Option::output_dir.toLatin1().constData());
|
|
return ret;
|
|
}
|
|
|
|
QMakeLocalFileName
|
|
MakefileGenerator::fixPathForFile(const QMakeLocalFileName &file, bool forOpen)
|
|
{
|
|
if(forOpen)
|
|
return QMakeLocalFileName(fileFixify(file.real(), FileFixifyBackwards));
|
|
return QMakeLocalFileName(fileFixify(file.real()));
|
|
}
|
|
|
|
QFileInfo
|
|
MakefileGenerator::findFileInfo(const QMakeLocalFileName &file)
|
|
{
|
|
return fileInfo(file.local());
|
|
}
|
|
|
|
QMakeLocalFileName
|
|
MakefileGenerator::findFileForDep(const QMakeLocalFileName &dep, const QMakeLocalFileName &file)
|
|
{
|
|
QMakeLocalFileName ret;
|
|
if(!project->isEmpty("SKIP_DEPENDS")) {
|
|
bool found = false;
|
|
const ProStringList &nodeplist = project->values("SKIP_DEPENDS");
|
|
for (ProStringList::ConstIterator it = nodeplist.begin();
|
|
it != nodeplist.end(); ++it) {
|
|
QRegExp regx((*it).toQString());
|
|
if(regx.indexIn(dep.local()) != -1) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if(found)
|
|
return ret;
|
|
}
|
|
|
|
ret = QMakeSourceFileInfo::findFileForDep(dep, file);
|
|
if(!ret.isNull())
|
|
return ret;
|
|
|
|
//these are some "hacky" heuristics it will try to do on an include
|
|
//however these can be turned off at runtime, I'm not sure how
|
|
//reliable these will be, most likely when problems arise turn it off
|
|
//and see if they go away..
|
|
if(Option::mkfile::do_dep_heuristics) {
|
|
if(depHeuristicsCache.contains(dep.real()))
|
|
return depHeuristicsCache[dep.real()];
|
|
|
|
if(Option::output_dir != qmake_getpwd()
|
|
&& QDir::isRelativePath(dep.real())) { //is it from the shadow tree
|
|
QList<QMakeLocalFileName> depdirs = QMakeSourceFileInfo::dependencyPaths();
|
|
depdirs.prepend(fileInfo(file.real()).absoluteDir().path());
|
|
QString pwd = qmake_getpwd();
|
|
if(pwd.at(pwd.length()-1) != '/')
|
|
pwd += '/';
|
|
for(int i = 0; i < depdirs.count(); i++) {
|
|
QString dir = depdirs.at(i).real();
|
|
if(!QDir::isRelativePath(dir) && dir.startsWith(pwd))
|
|
dir = dir.mid(pwd.length());
|
|
if(QDir::isRelativePath(dir)) {
|
|
if(!dir.endsWith(Option::dir_sep))
|
|
dir += Option::dir_sep;
|
|
QString shadow = fileFixify(dir + dep.local(), FileFixifyBackwards);
|
|
if(exists(shadow)) {
|
|
ret = QMakeLocalFileName(shadow);
|
|
goto found_dep_from_heuristic;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
{ //is it from an EXTRA_TARGET
|
|
const QString dep_basename = dep.local().section('/', -1);
|
|
const ProStringList &qut = project->values("QMAKE_EXTRA_TARGETS");
|
|
for (ProStringList::ConstIterator it = qut.begin(); it != qut.end(); ++it) {
|
|
QString targ = var(ProKey(*it + ".target"));
|
|
if(targ.isEmpty())
|
|
targ = (*it).toQString();
|
|
QString out = Option::fixPathToTargetOS(targ);
|
|
if(out == dep.real() || out.section(Option::dir_sep, -1) == dep_basename) {
|
|
ret = QMakeLocalFileName(out);
|
|
goto found_dep_from_heuristic;
|
|
}
|
|
}
|
|
}
|
|
{ //is it from an EXTRA_COMPILER
|
|
const QString dep_basename = dep.local().section('/', -1);
|
|
const ProStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
|
|
for (ProStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
|
|
const ProString &tmp_out = project->first(ProKey(*it + ".output"));
|
|
if(tmp_out.isEmpty())
|
|
continue;
|
|
const ProStringList &tmp = project->values(ProKey(*it + ".input"));
|
|
for (ProStringList::ConstIterator it2 = tmp.begin(); it2 != tmp.end(); ++it2) {
|
|
const ProStringList &inputs = project->values((*it2).toKey());
|
|
for (ProStringList::ConstIterator input = inputs.begin(); input != inputs.end(); ++input) {
|
|
QString out = Option::fixPathToTargetOS(
|
|
replaceExtraCompilerVariables(tmp_out.toQString(), (*input).toQString(), QString(), NoShell));
|
|
if (out == dep.real() || out.section(Option::dir_sep, -1) == dep_basename) {
|
|
ret = QMakeLocalFileName(fileFixify(out, FileFixifyBackwards));
|
|
goto found_dep_from_heuristic;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
found_dep_from_heuristic:
|
|
depHeuristicsCache.insert(dep.real(), ret);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
QStringList
|
|
&MakefileGenerator::findDependencies(const QString &file)
|
|
{
|
|
const QString fixedFile = fileFixify(file);
|
|
if(!dependsCache.contains(fixedFile)) {
|
|
#if 1
|
|
QStringList deps = QMakeSourceFileInfo::dependencies(file);
|
|
if(file != fixedFile)
|
|
deps += QMakeSourceFileInfo::dependencies(fixedFile);
|
|
#else
|
|
QStringList deps = QMakeSourceFileInfo::dependencies(fixedFile);
|
|
#endif
|
|
dependsCache.insert(fixedFile, deps);
|
|
}
|
|
return dependsCache[fixedFile];
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::specdir()
|
|
{
|
|
if (spec.isEmpty())
|
|
spec = fileFixify(project->specDir());
|
|
return spec;
|
|
}
|
|
|
|
bool
|
|
MakefileGenerator::openOutput(QFile &file, const QString &build) const
|
|
{
|
|
{
|
|
QString outdir;
|
|
if(!file.fileName().isEmpty()) {
|
|
if(QDir::isRelativePath(file.fileName()))
|
|
file.setFileName(Option::output_dir + "/" + file.fileName()); //pwd when qmake was run
|
|
QFileInfo fi(fileInfo(file.fileName()));
|
|
if(fi.isDir())
|
|
outdir = file.fileName() + '/';
|
|
}
|
|
if(!outdir.isEmpty() || file.fileName().isEmpty()) {
|
|
QString fname = "Makefile";
|
|
if(!project->isEmpty("MAKEFILE"))
|
|
fname = project->first("MAKEFILE").toQString();
|
|
file.setFileName(outdir + fname);
|
|
}
|
|
}
|
|
if(QDir::isRelativePath(file.fileName())) {
|
|
QString fname = Option::output_dir; //pwd when qmake was run
|
|
if(!fname.endsWith("/"))
|
|
fname += "/";
|
|
fname += file.fileName();
|
|
file.setFileName(fname);
|
|
}
|
|
if(!build.isEmpty())
|
|
file.setFileName(file.fileName() + "." + build);
|
|
if(project->isEmpty("QMAKE_MAKEFILE"))
|
|
project->values("QMAKE_MAKEFILE").append(file.fileName());
|
|
int slsh = file.fileName().lastIndexOf('/');
|
|
if(slsh != -1)
|
|
mkdir(file.fileName().left(slsh));
|
|
if(file.open(QIODevice::WriteOnly | QIODevice::Text | QIODevice::Truncate)) {
|
|
QFileInfo fi(fileInfo(Option::output.fileName()));
|
|
QString od;
|
|
if(fi.isSymLink())
|
|
od = fileInfo(fi.readLink()).absolutePath();
|
|
else
|
|
od = fi.path();
|
|
od = QDir::fromNativeSeparators(od);
|
|
if(QDir::isRelativePath(od)) {
|
|
QString dir = Option::output_dir;
|
|
if (!dir.endsWith('/') && !od.isEmpty())
|
|
dir += '/';
|
|
od.prepend(dir);
|
|
}
|
|
Option::output_dir = od;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::pkgConfigFileName(bool fixify)
|
|
{
|
|
QString ret = project->first("QMAKE_PKGCONFIG_FILE").toQString();
|
|
if (ret.isEmpty()) {
|
|
ret = project->first("TARGET").toQString();
|
|
int slsh = ret.lastIndexOf(Option::dir_sep);
|
|
if (slsh != -1)
|
|
ret = ret.right(ret.length() - slsh - 1);
|
|
if (ret.startsWith("lib"))
|
|
ret = ret.mid(3);
|
|
int dot = ret.indexOf('.');
|
|
if (dot != -1)
|
|
ret = ret.left(dot);
|
|
}
|
|
ret += Option::pkgcfg_ext;
|
|
QString subdir = project->first("QMAKE_PKGCONFIG_DESTDIR").toQString();
|
|
if(!subdir.isEmpty()) {
|
|
// initOutPaths() appends dir_sep, but just to be safe..
|
|
if (!subdir.endsWith(Option::dir_sep))
|
|
ret.prepend(Option::dir_sep);
|
|
ret.prepend(subdir);
|
|
}
|
|
if(fixify) {
|
|
if(QDir::isRelativePath(ret) && !project->isEmpty("DESTDIR"))
|
|
ret.prepend(project->first("DESTDIR").toQString());
|
|
ret = fileFixify(ret, FileFixifyBackwards);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::pkgConfigPrefix() const
|
|
{
|
|
if(!project->isEmpty("QMAKE_PKGCONFIG_PREFIX"))
|
|
return project->first("QMAKE_PKGCONFIG_PREFIX").toQString();
|
|
return project->propertyValue(ProKey("QT_INSTALL_PREFIX")).toQString();
|
|
}
|
|
|
|
QString
|
|
MakefileGenerator::pkgConfigFixPath(QString path) const
|
|
{
|
|
QString prefix = pkgConfigPrefix();
|
|
if(path.startsWith(prefix))
|
|
path.replace(prefix, QLatin1String("${prefix}"));
|
|
return path;
|
|
}
|
|
|
|
void
|
|
MakefileGenerator::writePkgConfigFile()
|
|
{
|
|
QString fname = pkgConfigFileName();
|
|
mkdir(fileInfo(fname).path());
|
|
QFile ft(fname);
|
|
if(!ft.open(QIODevice::WriteOnly))
|
|
return;
|
|
QString ffname(fileFixify(fname));
|
|
project->values("ALL_DEPS").append(ffname);
|
|
project->values("QMAKE_DISTCLEAN").append(ffname);
|
|
QTextStream t(&ft);
|
|
|
|
QString prefix = pkgConfigPrefix();
|
|
QString libDir = project->first("QMAKE_PKGCONFIG_LIBDIR").toQString();
|
|
if(libDir.isEmpty())
|
|
libDir = prefix + "/lib";
|
|
QString includeDir = project->first("QMAKE_PKGCONFIG_INCDIR").toQString();
|
|
if(includeDir.isEmpty())
|
|
includeDir = prefix + "/include";
|
|
|
|
t << "prefix=" << prefix << endl;
|
|
t << "exec_prefix=${prefix}\n"
|
|
<< "libdir=" << pkgConfigFixPath(libDir) << "\n"
|
|
<< "includedir=" << pkgConfigFixPath(includeDir) << endl;
|
|
t << endl;
|
|
|
|
//extra PKGCONFIG variables
|
|
const ProStringList &pkgconfig_vars = project->values("QMAKE_PKGCONFIG_VARIABLES");
|
|
for(int i = 0; i < pkgconfig_vars.size(); ++i) {
|
|
const ProString &var = project->first(ProKey(pkgconfig_vars.at(i) + ".name"));
|
|
QString val = project->values(ProKey(pkgconfig_vars.at(i) + ".value")).join(' ');
|
|
if(var.isEmpty())
|
|
continue;
|
|
if(val.isEmpty()) {
|
|
const ProStringList &var_vars = project->values(ProKey(pkgconfig_vars.at(i) + ".variable"));
|
|
for(int v = 0; v < var_vars.size(); ++v) {
|
|
const ProStringList &vars = project->values(var_vars.at(v).toKey());
|
|
for(int var = 0; var < vars.size(); ++var) {
|
|
if(!val.isEmpty())
|
|
val += " ";
|
|
val += pkgConfigFixPath(vars.at(var).toQString());
|
|
}
|
|
}
|
|
}
|
|
if (!val.isEmpty())
|
|
t << var << "=" << val << endl;
|
|
}
|
|
|
|
t << endl;
|
|
|
|
QString name = project->first("QMAKE_PKGCONFIG_NAME").toQString();
|
|
if(name.isEmpty()) {
|
|
name = project->first("QMAKE_ORIG_TARGET").toQString().toLower();
|
|
name.replace(0, 1, name[0].toUpper());
|
|
}
|
|
t << "Name: " << name << endl;
|
|
QString desc = project->values("QMAKE_PKGCONFIG_DESCRIPTION").join(' ');
|
|
if(desc.isEmpty()) {
|
|
if(name.isEmpty()) {
|
|
desc = project->first("QMAKE_ORIG_TARGET").toQString().toLower();
|
|
desc.replace(0, 1, desc[0].toUpper());
|
|
} else {
|
|
desc = name;
|
|
}
|
|
if(project->first("TEMPLATE") == "lib") {
|
|
if(project->isActiveConfig("plugin"))
|
|
desc += " Plugin";
|
|
else
|
|
desc += " Library";
|
|
} else if(project->first("TEMPLATE") == "app") {
|
|
desc += " Application";
|
|
}
|
|
}
|
|
t << "Description: " << desc << endl;
|
|
ProString version = project->first("QMAKE_PKGCONFIG_VERSION");
|
|
if (version.isEmpty())
|
|
version = project->first("VERSION");
|
|
if (!version.isEmpty())
|
|
t << "Version: " << version << endl;
|
|
|
|
// libs
|
|
t << "Libs: ";
|
|
QString pkgConfiglibName;
|
|
if (target_mode == TARG_MAC_MODE && project->isActiveConfig("lib_bundle")) {
|
|
if (libDir != QLatin1String("/System/Library/Frameworks")
|
|
&& libDir != QLatin1String("/Library/Frameworks")) {
|
|
t << "-F${libdir} ";
|
|
}
|
|
ProString bundle;
|
|
if (!project->isEmpty("QMAKE_FRAMEWORK_BUNDLE_NAME"))
|
|
bundle = project->first("QMAKE_FRAMEWORK_BUNDLE_NAME");
|
|
else
|
|
bundle = project->first("TARGET");
|
|
int suffix = bundle.lastIndexOf(".framework");
|
|
if (suffix != -1)
|
|
bundle = bundle.left(suffix);
|
|
t << "-framework ";
|
|
pkgConfiglibName = bundle.toQString();
|
|
} else {
|
|
if (!project->values("QMAKE_DEFAULT_LIBDIRS").contains(libDir))
|
|
t << "-L${libdir} ";
|
|
pkgConfiglibName = "-l" + project->first("QMAKE_ORIG_TARGET");
|
|
if (project->isActiveConfig("shared"))
|
|
pkgConfiglibName += project->first("TARGET_VERSION_EXT").toQString();
|
|
}
|
|
t << shellQuote(pkgConfiglibName) << " \n";
|
|
|
|
ProStringList libs;
|
|
if(!project->isEmpty("QMAKE_INTERNAL_PRL_LIBS")) {
|
|
libs = project->values("QMAKE_INTERNAL_PRL_LIBS");
|
|
} else {
|
|
libs << "QMAKE_LIBS"; //obvious one
|
|
}
|
|
libs << "QMAKE_LIBS_PRIVATE";
|
|
libs << "QMAKE_LFLAGS_THREAD"; //not sure about this one, but what about things like -pthread?
|
|
t << "Libs.private: ";
|
|
for (ProStringList::ConstIterator it = libs.begin(); it != libs.end(); ++it) {
|
|
t << fixLibFlags((*it).toKey()).join(' ') << ' ';
|
|
}
|
|
t << endl;
|
|
|
|
// flags
|
|
// ### too many
|
|
t << "Cflags: "
|
|
// << var("QMAKE_CXXFLAGS") << " "
|
|
<< varGlue("PRL_EXPORT_DEFINES","-D"," -D"," ")
|
|
<< varGlue("PRL_EXPORT_CXXFLAGS", "", " ", " ")
|
|
<< varGlue("QMAKE_PKGCONFIG_CFLAGS", "", " ", " ")
|
|
// << varGlue("DEFINES","-D"," -D"," ")
|
|
;
|
|
if (!project->values("QMAKE_DEFAULT_INCDIRS").contains(includeDir))
|
|
t << "-I${includedir}";
|
|
t << endl;
|
|
|
|
// requires
|
|
const QString requires = project->values("QMAKE_PKGCONFIG_REQUIRES").join(' ');
|
|
if (!requires.isEmpty()) {
|
|
t << "Requires: " << requires << endl;
|
|
}
|
|
|
|
t << endl;
|
|
}
|
|
|
|
static QString windowsifyPath(const QString &str)
|
|
{
|
|
// The paths are escaped in prl files, so every slash needs to turn into two backslashes.
|
|
// Then each backslash needs to be escaped for sed. And another level for C quoting here.
|
|
return QString(str).replace('/', QLatin1String("\\\\\\\\"));
|
|
}
|
|
|
|
QString MakefileGenerator::installMetaFile(const ProKey &replace_rule, const QString &src, const QString &dst)
|
|
{
|
|
QString ret;
|
|
if (project->isEmpty(replace_rule)
|
|
|| project->isActiveConfig("no_sed_meta_install")) {
|
|
ret += "-$(INSTALL_FILE) " + escapeFilePath(src) + ' ' + escapeFilePath(dst);
|
|
} else {
|
|
ret += "-$(SED)";
|
|
const ProStringList &replace_rules = project->values(replace_rule);
|
|
for (int r = 0; r < replace_rules.size(); ++r) {
|
|
const ProString match = project->first(ProKey(replace_rules.at(r) + ".match")),
|
|
replace = project->first(ProKey(replace_rules.at(r) + ".replace"));
|
|
if (!match.isEmpty() /*&& match != replace*/) {
|
|
ret += " -e " + shellQuote("s," + match + "," + replace + ",g");
|
|
if (isWindowsShell() && project->first(ProKey(replace_rules.at(r) + ".CONFIG")).contains("path"))
|
|
ret += " -e " + shellQuote("s," + windowsifyPath(match.toQString())
|
|
+ "," + windowsifyPath(replace.toQString()) + ",gi");
|
|
}
|
|
}
|
|
ret += ' ' + escapeFilePath(src) + " > " + escapeFilePath(dst);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
QString MakefileGenerator::shellQuote(const QString &str)
|
|
{
|
|
return isWindowsShell() ? IoUtils::shellQuoteWin(str) : IoUtils::shellQuoteUnix(str);
|
|
}
|
|
|
|
QT_END_NAMESPACE
|