c925400bde
The code in question is obviously checking for support of QProcess, not for general bootstrap mode. You can manually disable QProcess, in which case it is still not available after bootstrapping is done. Change-Id: Ia99810b8900621911a31912034358a01af4f18a0 Reviewed-by: Oswald Buddenhagen <oswald.buddenhagen@qt.io>
2031 lines
78 KiB
C++
2031 lines
78 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the qmake application of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
|
** 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 https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "qmakeevaluator.h"
|
|
|
|
#include "qmakeevaluator_p.h"
|
|
#include "qmakeglobals.h"
|
|
#include "qmakeparser.h"
|
|
#include "qmakevfs.h"
|
|
#include "ioutils.h"
|
|
|
|
#include <qbytearray.h>
|
|
#include <qdir.h>
|
|
#include <qfile.h>
|
|
#include <qfileinfo.h>
|
|
#include <qlist.h>
|
|
#include <qregexp.h>
|
|
#include <qset.h>
|
|
#include <qstringlist.h>
|
|
#include <qtextstream.h>
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)
|
|
# include <qjsondocument.h>
|
|
# include <qjsonobject.h>
|
|
# include <qjsonarray.h>
|
|
#endif
|
|
#ifdef PROEVALUATOR_THREAD_SAFE
|
|
# include <qthreadpool.h>
|
|
#endif
|
|
|
|
#include <algorithm>
|
|
|
|
#ifdef Q_OS_UNIX
|
|
#include <time.h>
|
|
#include <utime.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/utsname.h>
|
|
#else
|
|
#include <windows.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#ifdef Q_OS_WIN32
|
|
#define QT_POPEN _popen
|
|
#define QT_POPEN_READ "rb"
|
|
#define QT_PCLOSE _pclose
|
|
#else
|
|
#define QT_POPEN popen
|
|
#define QT_POPEN_READ "r"
|
|
#define QT_PCLOSE pclose
|
|
#endif
|
|
|
|
using namespace QMakeInternal;
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
#define fL1S(s) QString::fromLatin1(s)
|
|
|
|
enum ExpandFunc {
|
|
E_INVALID = 0, E_MEMBER, E_STR_MEMBER, E_FIRST, E_TAKE_FIRST, E_LAST, E_TAKE_LAST,
|
|
E_SIZE, E_STR_SIZE, E_CAT, E_FROMFILE, E_EVAL, E_LIST, E_SPRINTF, E_FORMAT_NUMBER,
|
|
E_NUM_ADD, E_JOIN, E_SPLIT, E_BASENAME, E_DIRNAME, E_SECTION,
|
|
E_FIND, E_SYSTEM, E_UNIQUE, E_SORTED, E_REVERSE, E_QUOTE, E_ESCAPE_EXPAND,
|
|
E_UPPER, E_LOWER, E_TITLE, E_FILES, E_PROMPT, E_RE_ESCAPE, E_VAL_ESCAPE,
|
|
E_REPLACE, E_SORT_DEPENDS, E_RESOLVE_DEPENDS, E_ENUMERATE_VARS,
|
|
E_SHADOWED, E_ABSOLUTE_PATH, E_RELATIVE_PATH, E_CLEAN_PATH,
|
|
E_SYSTEM_PATH, E_SHELL_PATH, E_SYSTEM_QUOTE, E_SHELL_QUOTE, E_GETENV
|
|
};
|
|
|
|
enum TestFunc {
|
|
T_INVALID = 0, T_REQUIRES, T_GREATERTHAN, T_LESSTHAN, T_EQUALS,
|
|
T_EXISTS, T_EXPORT, T_CLEAR, T_UNSET, T_EVAL, T_CONFIG, T_SYSTEM,
|
|
T_DEFINED, T_DISCARD_FROM, T_CONTAINS, T_INFILE,
|
|
T_COUNT, T_ISEMPTY, T_PARSE_JSON, T_INCLUDE, T_LOAD, T_DEBUG, T_LOG, T_MESSAGE, T_WARNING, T_ERROR, T_IF,
|
|
T_MKPATH, T_WRITE_FILE, T_TOUCH, T_CACHE, T_RELOAD_PROPERTIES
|
|
};
|
|
|
|
void QMakeEvaluator::initFunctionStatics()
|
|
{
|
|
static const struct {
|
|
const char * const name;
|
|
const ExpandFunc func;
|
|
} expandInits[] = {
|
|
{ "member", E_MEMBER },
|
|
{ "str_member", E_STR_MEMBER },
|
|
{ "first", E_FIRST },
|
|
{ "take_first", E_TAKE_FIRST },
|
|
{ "last", E_LAST },
|
|
{ "take_last", E_TAKE_LAST },
|
|
{ "size", E_SIZE },
|
|
{ "str_size", E_STR_SIZE },
|
|
{ "cat", E_CAT },
|
|
{ "fromfile", E_FROMFILE },
|
|
{ "eval", E_EVAL },
|
|
{ "list", E_LIST },
|
|
{ "sprintf", E_SPRINTF },
|
|
{ "format_number", E_FORMAT_NUMBER },
|
|
{ "num_add", E_NUM_ADD },
|
|
{ "join", E_JOIN },
|
|
{ "split", E_SPLIT },
|
|
{ "basename", E_BASENAME },
|
|
{ "dirname", E_DIRNAME },
|
|
{ "section", E_SECTION },
|
|
{ "find", E_FIND },
|
|
{ "system", E_SYSTEM },
|
|
{ "unique", E_UNIQUE },
|
|
{ "sorted", E_SORTED },
|
|
{ "reverse", E_REVERSE },
|
|
{ "quote", E_QUOTE },
|
|
{ "escape_expand", E_ESCAPE_EXPAND },
|
|
{ "upper", E_UPPER },
|
|
{ "lower", E_LOWER },
|
|
{ "title", E_TITLE },
|
|
{ "re_escape", E_RE_ESCAPE },
|
|
{ "val_escape", E_VAL_ESCAPE },
|
|
{ "files", E_FILES },
|
|
{ "prompt", E_PROMPT },
|
|
{ "replace", E_REPLACE },
|
|
{ "sort_depends", E_SORT_DEPENDS },
|
|
{ "resolve_depends", E_RESOLVE_DEPENDS },
|
|
{ "enumerate_vars", E_ENUMERATE_VARS },
|
|
{ "shadowed", E_SHADOWED },
|
|
{ "absolute_path", E_ABSOLUTE_PATH },
|
|
{ "relative_path", E_RELATIVE_PATH },
|
|
{ "clean_path", E_CLEAN_PATH },
|
|
{ "system_path", E_SYSTEM_PATH },
|
|
{ "shell_path", E_SHELL_PATH },
|
|
{ "system_quote", E_SYSTEM_QUOTE },
|
|
{ "shell_quote", E_SHELL_QUOTE },
|
|
{ "getenv", E_GETENV },
|
|
};
|
|
statics.expands.reserve((int)(sizeof(expandInits)/sizeof(expandInits[0])));
|
|
for (unsigned i = 0; i < sizeof(expandInits)/sizeof(expandInits[0]); ++i)
|
|
statics.expands.insert(ProKey(expandInits[i].name), expandInits[i].func);
|
|
|
|
static const struct {
|
|
const char * const name;
|
|
const TestFunc func;
|
|
} testInits[] = {
|
|
{ "requires", T_REQUIRES },
|
|
{ "greaterThan", T_GREATERTHAN },
|
|
{ "lessThan", T_LESSTHAN },
|
|
{ "equals", T_EQUALS },
|
|
{ "isEqual", T_EQUALS },
|
|
{ "exists", T_EXISTS },
|
|
{ "export", T_EXPORT },
|
|
{ "clear", T_CLEAR },
|
|
{ "unset", T_UNSET },
|
|
{ "eval", T_EVAL },
|
|
{ "CONFIG", T_CONFIG },
|
|
{ "if", T_IF },
|
|
{ "isActiveConfig", T_CONFIG },
|
|
{ "system", T_SYSTEM },
|
|
{ "discard_from", T_DISCARD_FROM },
|
|
{ "defined", T_DEFINED },
|
|
{ "contains", T_CONTAINS },
|
|
{ "infile", T_INFILE },
|
|
{ "count", T_COUNT },
|
|
{ "isEmpty", T_ISEMPTY },
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)
|
|
{ "parseJson", T_PARSE_JSON },
|
|
#endif
|
|
{ "load", T_LOAD },
|
|
{ "include", T_INCLUDE },
|
|
{ "debug", T_DEBUG },
|
|
{ "log", T_LOG },
|
|
{ "message", T_MESSAGE },
|
|
{ "warning", T_WARNING },
|
|
{ "error", T_ERROR },
|
|
{ "mkpath", T_MKPATH },
|
|
{ "write_file", T_WRITE_FILE },
|
|
{ "touch", T_TOUCH },
|
|
{ "cache", T_CACHE },
|
|
{ "reload_properties", T_RELOAD_PROPERTIES },
|
|
};
|
|
statics.functions.reserve((int)(sizeof(testInits)/sizeof(testInits[0])));
|
|
for (unsigned i = 0; i < sizeof(testInits)/sizeof(testInits[0]); ++i)
|
|
statics.functions.insert(ProKey(testInits[i].name), testInits[i].func);
|
|
}
|
|
|
|
static bool isTrue(const ProString &str)
|
|
{
|
|
return !str.compare(statics.strtrue, Qt::CaseInsensitive) || str.toInt();
|
|
}
|
|
|
|
bool
|
|
QMakeEvaluator::getMemberArgs(const ProKey &func, int srclen, const ProStringList &args,
|
|
int *start, int *end)
|
|
{
|
|
*start = 0, *end = 0;
|
|
if (args.count() >= 2) {
|
|
bool ok = true;
|
|
const ProString &start_str = args.at(1);
|
|
*start = start_str.toInt(&ok);
|
|
if (!ok) {
|
|
if (args.count() == 2) {
|
|
int dotdot = start_str.indexOf(statics.strDotDot);
|
|
if (dotdot != -1) {
|
|
*start = start_str.left(dotdot).toInt(&ok);
|
|
if (ok)
|
|
*end = start_str.mid(dotdot+2).toInt(&ok);
|
|
}
|
|
}
|
|
if (!ok) {
|
|
evalError(fL1S("%1() argument 2 (start) '%2' invalid.")
|
|
.arg(func.toQString(m_tmp1), start_str.toQString(m_tmp2)));
|
|
return false;
|
|
}
|
|
} else {
|
|
*end = *start;
|
|
if (args.count() == 3)
|
|
*end = args.at(2).toInt(&ok);
|
|
if (!ok) {
|
|
evalError(fL1S("%1() argument 3 (end) '%2' invalid.")
|
|
.arg(func.toQString(m_tmp1), args.at(2).toQString(m_tmp2)));
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
if (*start < 0)
|
|
*start += srclen;
|
|
if (*end < 0)
|
|
*end += srclen;
|
|
if (*start < 0 || *start >= srclen || *end < 0 || *end >= srclen)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
#if defined(Q_OS_WIN) && defined(PROEVALUATOR_FULL)
|
|
static QString windowsErrorCode()
|
|
{
|
|
wchar_t *string = 0;
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
GetLastError(),
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
(LPWSTR)&string,
|
|
0,
|
|
NULL);
|
|
QString ret = QString::fromWCharArray(string);
|
|
LocalFree((HLOCAL)string);
|
|
return ret.trimmed();
|
|
}
|
|
#endif
|
|
|
|
QString
|
|
QMakeEvaluator::quoteValue(const ProString &val)
|
|
{
|
|
QString ret;
|
|
ret.reserve(val.size());
|
|
const QChar *chars = val.constData();
|
|
bool quote = val.isEmpty();
|
|
bool escaping = false;
|
|
for (int i = 0, l = val.size(); i < l; i++) {
|
|
QChar c = chars[i];
|
|
ushort uc = c.unicode();
|
|
if (uc < 32) {
|
|
if (!escaping) {
|
|
escaping = true;
|
|
ret += QLatin1String("$$escape_expand(");
|
|
}
|
|
switch (uc) {
|
|
case '\r':
|
|
ret += QLatin1String("\\\\r");
|
|
break;
|
|
case '\n':
|
|
ret += QLatin1String("\\\\n");
|
|
break;
|
|
case '\t':
|
|
ret += QLatin1String("\\\\t");
|
|
break;
|
|
default:
|
|
ret += QString::fromLatin1("\\\\x%1").arg(uc, 2, 16, QLatin1Char('0'));
|
|
break;
|
|
}
|
|
} else {
|
|
if (escaping) {
|
|
escaping = false;
|
|
ret += QLatin1Char(')');
|
|
}
|
|
switch (uc) {
|
|
case '\\':
|
|
ret += QLatin1String("\\\\");
|
|
break;
|
|
case '"':
|
|
ret += QLatin1String("\\\"");
|
|
break;
|
|
case '\'':
|
|
ret += QLatin1String("\\'");
|
|
break;
|
|
case '$':
|
|
ret += QLatin1String("\\$");
|
|
break;
|
|
case '#':
|
|
ret += QLatin1String("$${LITERAL_HASH}");
|
|
break;
|
|
case 32:
|
|
quote = true;
|
|
Q_FALLTHROUGH();
|
|
default:
|
|
ret += c;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (escaping)
|
|
ret += QLatin1Char(')');
|
|
if (quote) {
|
|
ret.prepend(QLatin1Char('"'));
|
|
ret.append(QLatin1Char('"'));
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)
|
|
static void addJsonValue(const QJsonValue &value, const QString &keyPrefix, ProValueMap *map);
|
|
|
|
static void insertJsonKeyValue(const QString &key, const QStringList &values, ProValueMap *map)
|
|
{
|
|
map->insert(ProKey(key), ProStringList(values));
|
|
}
|
|
|
|
static void addJsonArray(const QJsonArray &array, const QString &keyPrefix, ProValueMap *map)
|
|
{
|
|
QStringList keys;
|
|
const int size = array.count();
|
|
keys.reserve(size);
|
|
for (int i = 0; i < size; ++i) {
|
|
const QString number = QString::number(i);
|
|
keys.append(number);
|
|
addJsonValue(array.at(i), keyPrefix + number, map);
|
|
}
|
|
insertJsonKeyValue(keyPrefix + QLatin1String("_KEYS_"), keys, map);
|
|
}
|
|
|
|
static void addJsonObject(const QJsonObject &object, const QString &keyPrefix, ProValueMap *map)
|
|
{
|
|
QStringList keys;
|
|
keys.reserve(object.size());
|
|
for (auto it = object.begin(), end = object.end(); it != end; ++it) {
|
|
const QString key = it.key();
|
|
keys.append(key);
|
|
addJsonValue(it.value(), keyPrefix + key, map);
|
|
}
|
|
insertJsonKeyValue(keyPrefix + QLatin1String("_KEYS_"), keys, map);
|
|
}
|
|
|
|
static void addJsonValue(const QJsonValue &value, const QString &keyPrefix, ProValueMap *map)
|
|
{
|
|
switch (value.type()) {
|
|
case QJsonValue::Bool:
|
|
insertJsonKeyValue(keyPrefix, QStringList() << (value.toBool() ? QLatin1String("true") : QLatin1String("false")), map);
|
|
break;
|
|
case QJsonValue::Double:
|
|
insertJsonKeyValue(keyPrefix, QStringList() << QString::number(value.toDouble()), map);
|
|
break;
|
|
case QJsonValue::String:
|
|
insertJsonKeyValue(keyPrefix, QStringList() << value.toString(), map);
|
|
break;
|
|
case QJsonValue::Array:
|
|
addJsonArray(value.toArray(), keyPrefix + QLatin1Char('.'), map);
|
|
break;
|
|
case QJsonValue::Object:
|
|
addJsonObject(value.toObject(), keyPrefix + QLatin1Char('.'), map);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
struct ErrorPosition {
|
|
int line;
|
|
int column;
|
|
};
|
|
|
|
static ErrorPosition calculateErrorPosition(const QByteArray &json, int offset)
|
|
{
|
|
ErrorPosition pos = { 0, 0 };
|
|
offset--; // offset is 1-based, switching to 0-based
|
|
for (int i = 0; i < offset; ++i) {
|
|
switch (json.at(i)) {
|
|
case '\n':
|
|
pos.line++;
|
|
pos.column = 0;
|
|
break;
|
|
case '\r':
|
|
break;
|
|
case '\t':
|
|
pos.column = (pos.column + 8) & ~7;
|
|
break;
|
|
default:
|
|
pos.column++;
|
|
break;
|
|
}
|
|
}
|
|
// Lines and columns in text editors are 1-based:
|
|
pos.line++;
|
|
pos.column++;
|
|
return pos;
|
|
}
|
|
|
|
QMakeEvaluator::VisitReturn QMakeEvaluator::parseJsonInto(const QByteArray &json, const QString &into, ProValueMap *value)
|
|
{
|
|
QJsonParseError error;
|
|
QJsonDocument document = QJsonDocument::fromJson(json, &error);
|
|
if (document.isNull()) {
|
|
if (error.error != QJsonParseError::NoError) {
|
|
ErrorPosition errorPos = calculateErrorPosition(json, error.offset);
|
|
evalError(fL1S("Error parsing JSON at %1:%2: %3")
|
|
.arg(errorPos.line).arg(errorPos.column).arg(error.errorString()));
|
|
}
|
|
return QMakeEvaluator::ReturnFalse;
|
|
}
|
|
|
|
QString currentKey = into + QLatin1Char('.');
|
|
|
|
// top-level item is either an array or object
|
|
if (document.isArray())
|
|
addJsonArray(document.array(), currentKey, value);
|
|
else if (document.isObject())
|
|
addJsonObject(document.object(), currentKey, value);
|
|
else
|
|
return QMakeEvaluator::ReturnFalse;
|
|
|
|
return QMakeEvaluator::ReturnTrue;
|
|
}
|
|
#endif
|
|
|
|
QMakeEvaluator::VisitReturn
|
|
QMakeEvaluator::writeFile(const QString &ctx, const QString &fn, QIODevice::OpenMode mode,
|
|
bool exe, const QString &contents)
|
|
{
|
|
QString errStr;
|
|
if (!m_vfs->writeFile(fn, mode, exe, contents, &errStr)) {
|
|
evalError(fL1S("Cannot write %1file %2: %3")
|
|
.arg(ctx, QDir::toNativeSeparators(fn), errStr));
|
|
return ReturnFalse;
|
|
}
|
|
m_parser->discardFileFromCache(fn);
|
|
return ReturnTrue;
|
|
}
|
|
|
|
#if QT_CONFIG(process)
|
|
void QMakeEvaluator::runProcess(QProcess *proc, const QString &command) const
|
|
{
|
|
proc->setWorkingDirectory(currentDirectory());
|
|
# ifdef PROEVALUATOR_SETENV
|
|
if (!m_option->environment.isEmpty())
|
|
proc->setProcessEnvironment(m_option->environment);
|
|
# endif
|
|
# ifdef Q_OS_WIN
|
|
proc->setNativeArguments(QLatin1String("/v:off /s /c \"") + command + QLatin1Char('"'));
|
|
proc->start(m_option->getEnv(QLatin1String("COMSPEC")), QStringList());
|
|
# else
|
|
proc->start(QLatin1String("/bin/sh"), QStringList() << QLatin1String("-c") << command);
|
|
# endif
|
|
proc->waitForFinished(-1);
|
|
}
|
|
#endif
|
|
|
|
QByteArray QMakeEvaluator::getCommandOutput(const QString &args, int *exitCode) const
|
|
{
|
|
QByteArray out;
|
|
#if QT_CONFIG(process)
|
|
QProcess proc;
|
|
runProcess(&proc, args);
|
|
*exitCode = (proc.exitStatus() == QProcess::NormalExit) ? proc.exitCode() : -1;
|
|
QByteArray errout = proc.readAllStandardError();
|
|
# ifdef PROEVALUATOR_FULL
|
|
// FIXME: Qt really should have the option to set forwarding per channel
|
|
fputs(errout.constData(), stderr);
|
|
# else
|
|
if (!errout.isEmpty()) {
|
|
if (errout.endsWith('\n'))
|
|
errout.chop(1);
|
|
m_handler->message(
|
|
QMakeHandler::EvalError | (m_cumulative ? QMakeHandler::CumulativeEvalMessage : 0),
|
|
QString::fromLocal8Bit(errout));
|
|
}
|
|
# endif
|
|
out = proc.readAllStandardOutput();
|
|
# ifdef Q_OS_WIN
|
|
// FIXME: Qt's line end conversion on sequential files should really be fixed
|
|
out.replace("\r\n", "\n");
|
|
# endif
|
|
#else
|
|
if (FILE *proc = QT_POPEN(QString(QLatin1String("cd ")
|
|
+ IoUtils::shellQuote(QDir::toNativeSeparators(currentDirectory()))
|
|
+ QLatin1String(" && ") + args).toLocal8Bit().constData(), QT_POPEN_READ)) {
|
|
while (!feof(proc)) {
|
|
char buff[10 * 1024];
|
|
int read_in = int(fread(buff, 1, sizeof(buff), proc));
|
|
if (!read_in)
|
|
break;
|
|
out += QByteArray(buff, read_in);
|
|
}
|
|
int ec = QT_PCLOSE(proc);
|
|
# ifdef Q_OS_WIN
|
|
*exitCode = ec >= 0 ? ec : -1;
|
|
# else
|
|
*exitCode = WIFEXITED(ec) ? WEXITSTATUS(ec) : -1;
|
|
# endif
|
|
}
|
|
# ifdef Q_OS_WIN
|
|
out.replace("\r\n", "\n");
|
|
# endif
|
|
#endif
|
|
return out;
|
|
}
|
|
|
|
void QMakeEvaluator::populateDeps(
|
|
const ProStringList &deps, const ProString &prefix, const ProStringList &suffixes,
|
|
const ProString &priosfx,
|
|
QHash<ProKey, QSet<ProKey> > &dependencies, ProValueMap &dependees,
|
|
QMultiMap<int, ProString> &rootSet) const
|
|
{
|
|
for (const ProString &item : deps)
|
|
if (!dependencies.contains(item.toKey())) {
|
|
QSet<ProKey> &dset = dependencies[item.toKey()]; // Always create entry
|
|
ProStringList depends;
|
|
for (const ProString &suffix : suffixes)
|
|
depends += values(ProKey(prefix + item + suffix));
|
|
if (depends.isEmpty()) {
|
|
rootSet.insert(first(ProKey(prefix + item + priosfx)).toInt(), item);
|
|
} else {
|
|
for (const ProString &dep : qAsConst(depends)) {
|
|
dset.insert(dep.toKey());
|
|
dependees[dep.toKey()] << item;
|
|
}
|
|
populateDeps(depends, prefix, suffixes, priosfx, dependencies, dependees, rootSet);
|
|
}
|
|
}
|
|
}
|
|
|
|
QMakeEvaluator::VisitReturn QMakeEvaluator::evaluateBuiltinExpand(
|
|
int func_t, const ProKey &func, const ProStringList &args, ProStringList &ret)
|
|
{
|
|
traceMsg("calling built-in $$%s(%s)", dbgKey(func), dbgSepStrList(args));
|
|
|
|
switch (func_t) {
|
|
case E_BASENAME:
|
|
case E_DIRNAME:
|
|
case E_SECTION: {
|
|
bool regexp = false;
|
|
QString sep;
|
|
ProString var;
|
|
int beg = 0;
|
|
int end = -1;
|
|
if (func_t == E_SECTION) {
|
|
if (args.count() != 3 && args.count() != 4) {
|
|
evalError(fL1S("section(var, sep, begin, end) requires three or four arguments."));
|
|
} else {
|
|
var = args[0];
|
|
sep = args.at(1).toQString();
|
|
beg = args.at(2).toInt();
|
|
if (args.count() == 4)
|
|
end = args.at(3).toInt();
|
|
}
|
|
} else {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(var) requires one argument.").arg(func.toQString(m_tmp1)));
|
|
} else {
|
|
var = args[0];
|
|
regexp = true;
|
|
sep = QLatin1String("[\\\\/]");
|
|
if (func_t == E_DIRNAME)
|
|
end = -2;
|
|
else
|
|
beg = -1;
|
|
}
|
|
}
|
|
if (!var.isEmpty()) {
|
|
const auto strings = values(map(var));
|
|
if (regexp) {
|
|
QRegExp sepRx(sep);
|
|
for (const ProString &str : strings) {
|
|
const QString &rstr = str.toQString(m_tmp1).section(sepRx, beg, end);
|
|
ret << (rstr.isSharedWith(m_tmp1) ? str : ProString(rstr).setSource(str));
|
|
}
|
|
} else {
|
|
for (const ProString &str : strings) {
|
|
const QString &rstr = str.toQString(m_tmp1).section(sep, beg, end);
|
|
ret << (rstr.isSharedWith(m_tmp1) ? str : ProString(rstr).setSource(str));
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case E_SPRINTF:
|
|
if (args.count() < 1) {
|
|
evalError(fL1S("sprintf(format, ...) requires at least one argument."));
|
|
} else {
|
|
QString tmp = args.at(0).toQString(m_tmp1);
|
|
for (int i = 1; i < args.count(); ++i)
|
|
tmp = tmp.arg(args.at(i).toQString(m_tmp2));
|
|
ret << (tmp.isSharedWith(m_tmp1) ? args.at(0) : ProString(tmp).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_FORMAT_NUMBER:
|
|
if (args.count() > 2) {
|
|
evalError(fL1S("format_number(number[, options...]) requires one or two arguments."));
|
|
} else {
|
|
int ibase = 10;
|
|
int obase = 10;
|
|
int width = 0;
|
|
bool zeropad = false;
|
|
bool leftalign = false;
|
|
enum { DefaultSign, PadSign, AlwaysSign } sign = DefaultSign;
|
|
if (args.count() >= 2) {
|
|
const auto opts = split_value_list(args.at(1).toQStringRef());
|
|
for (const ProString &opt : opts) {
|
|
if (opt.startsWith(QLatin1String("ibase="))) {
|
|
ibase = opt.mid(6).toInt();
|
|
} else if (opt.startsWith(QLatin1String("obase="))) {
|
|
obase = opt.mid(6).toInt();
|
|
} else if (opt.startsWith(QLatin1String("width="))) {
|
|
width = opt.mid(6).toInt();
|
|
} else if (opt == QLatin1String("zeropad")) {
|
|
zeropad = true;
|
|
} else if (opt == QLatin1String("padsign")) {
|
|
sign = PadSign;
|
|
} else if (opt == QLatin1String("alwayssign")) {
|
|
sign = AlwaysSign;
|
|
} else if (opt == QLatin1String("leftalign")) {
|
|
leftalign = true;
|
|
} else {
|
|
evalError(fL1S("format_number(): invalid format option %1.")
|
|
.arg(opt.toQString(m_tmp3)));
|
|
goto formfail;
|
|
}
|
|
}
|
|
}
|
|
if (args.at(0).contains(QLatin1Char('.'))) {
|
|
evalError(fL1S("format_number(): floats are currently not supported."));
|
|
break;
|
|
}
|
|
bool ok;
|
|
qlonglong num = args.at(0).toLongLong(&ok, ibase);
|
|
if (!ok) {
|
|
evalError(fL1S("format_number(): malformed number %2 for base %1.")
|
|
.arg(ibase).arg(args.at(0).toQString(m_tmp3)));
|
|
break;
|
|
}
|
|
QString outstr;
|
|
if (num < 0) {
|
|
num = -num;
|
|
outstr = QLatin1Char('-');
|
|
} else if (sign == AlwaysSign) {
|
|
outstr = QLatin1Char('+');
|
|
} else if (sign == PadSign) {
|
|
outstr = QLatin1Char(' ');
|
|
}
|
|
QString numstr = QString::number(num, obase);
|
|
int space = width - outstr.length() - numstr.length();
|
|
if (space <= 0) {
|
|
outstr += numstr;
|
|
} else if (leftalign) {
|
|
outstr += numstr + QString(space, QLatin1Char(' '));
|
|
} else if (zeropad) {
|
|
outstr += QString(space, QLatin1Char('0')) + numstr;
|
|
} else {
|
|
outstr.prepend(QString(space, QLatin1Char(' ')));
|
|
outstr += numstr;
|
|
}
|
|
ret += ProString(outstr);
|
|
}
|
|
formfail:
|
|
break;
|
|
case E_NUM_ADD:
|
|
if (args.count() < 1 || args.at(0).isEmpty()) {
|
|
evalError(fL1S("num_add(num, ...) requires at least one argument."));
|
|
} else {
|
|
qlonglong sum = 0;
|
|
for (const ProString &arg : qAsConst(args)) {
|
|
if (arg.contains(QLatin1Char('.'))) {
|
|
evalError(fL1S("num_add(): floats are currently not supported."));
|
|
goto nafail;
|
|
}
|
|
bool ok;
|
|
qlonglong num = arg.toLongLong(&ok);
|
|
if (!ok) {
|
|
evalError(fL1S("num_add(): malformed number %1.")
|
|
.arg(arg.toQString(m_tmp3)));
|
|
goto nafail;
|
|
}
|
|
sum += num;
|
|
}
|
|
ret += ProString(QString::number(sum));
|
|
}
|
|
nafail:
|
|
break;
|
|
case E_JOIN: {
|
|
if (args.count() < 1 || args.count() > 4) {
|
|
evalError(fL1S("join(var, glue, before, after) requires one to four arguments."));
|
|
} else {
|
|
ProString glue, before, after;
|
|
if (args.count() >= 2)
|
|
glue = args.at(1);
|
|
if (args.count() >= 3)
|
|
before = args[2];
|
|
if (args.count() == 4)
|
|
after = args[3];
|
|
const ProStringList &var = values(map(args.at(0)));
|
|
if (!var.isEmpty()) {
|
|
const ProFile *src = currentProFile();
|
|
for (const ProString &v : var)
|
|
if (const ProFile *s = v.sourceFile()) {
|
|
src = s;
|
|
break;
|
|
}
|
|
ret << ProString(before + var.join(glue) + after).setSource(src);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case E_SPLIT:
|
|
if (args.count() < 1 || args.count() > 2) {
|
|
evalError(fL1S("split(var, sep) requires one or two arguments."));
|
|
} else {
|
|
const QString &sep = (args.count() == 2) ? args.at(1).toQString(m_tmp1) : statics.field_sep;
|
|
const auto vars = values(map(args.at(0)));
|
|
for (const ProString &var : vars) {
|
|
const auto splits = var.toQString(m_tmp2).split(sep);
|
|
for (const QString &splt : splits)
|
|
ret << (splt.isSharedWith(m_tmp2) ? var : ProString(splt).setSource(var));
|
|
}
|
|
}
|
|
break;
|
|
case E_MEMBER:
|
|
if (args.count() < 1 || args.count() > 3) {
|
|
evalError(fL1S("member(var, start, end) requires one to three arguments."));
|
|
} else {
|
|
const ProStringList &src = values(map(args.at(0)));
|
|
int start, end;
|
|
if (getMemberArgs(func, src.size(), args, &start, &end)) {
|
|
ret.reserve(qAbs(end - start) + 1);
|
|
if (start < end) {
|
|
for (int i = start; i <= end && src.size() >= i; i++)
|
|
ret += src.at(i);
|
|
} else {
|
|
for (int i = start; i >= end && src.size() >= i && i >= 0; i--)
|
|
ret += src.at(i);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case E_STR_MEMBER:
|
|
if (args.count() < 1 || args.count() > 3) {
|
|
evalError(fL1S("str_member(str, start, end) requires one to three arguments."));
|
|
} else {
|
|
const ProString &src = args.at(0);
|
|
int start, end;
|
|
if (getMemberArgs(func, src.size(), args, &start, &end)) {
|
|
QString res;
|
|
res.reserve(qAbs(end - start) + 1);
|
|
if (start < end) {
|
|
for (int i = start; i <= end && src.size() >= i; i++)
|
|
res += src.at(i);
|
|
} else {
|
|
for (int i = start; i >= end && src.size() >= i && i >= 0; i--)
|
|
res += src.at(i);
|
|
}
|
|
ret += ProString(res);
|
|
}
|
|
}
|
|
break;
|
|
case E_FIRST:
|
|
case E_LAST:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(var) requires one argument.").arg(func.toQString(m_tmp1)));
|
|
} else {
|
|
const ProStringList &var = values(map(args.at(0)));
|
|
if (!var.isEmpty()) {
|
|
if (func_t == E_FIRST)
|
|
ret.append(var[0]);
|
|
else
|
|
ret.append(var.last());
|
|
}
|
|
}
|
|
break;
|
|
case E_TAKE_FIRST:
|
|
case E_TAKE_LAST:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(var) requires one argument.").arg(func.toQString(m_tmp1)));
|
|
} else {
|
|
ProStringList &var = valuesRef(map(args.at(0)));
|
|
if (!var.isEmpty()) {
|
|
if (func_t == E_TAKE_FIRST)
|
|
ret.append(var.takeFirst());
|
|
else
|
|
ret.append(var.takeLast());
|
|
}
|
|
}
|
|
break;
|
|
case E_SIZE:
|
|
if (args.count() != 1)
|
|
evalError(fL1S("size(var) requires one argument."));
|
|
else
|
|
ret.append(ProString(QString::number(values(map(args.at(0))).size())));
|
|
break;
|
|
case E_STR_SIZE:
|
|
if (args.count() != 1)
|
|
evalError(fL1S("str_size(str) requires one argument."));
|
|
else
|
|
ret.append(ProString(QString::number(args.at(0).size())));
|
|
break;
|
|
case E_CAT:
|
|
if (args.count() < 1 || args.count() > 2) {
|
|
evalError(fL1S("cat(file, singleline=true) requires one or two arguments."));
|
|
} else {
|
|
const QString &file = args.at(0).toQString(m_tmp1);
|
|
|
|
bool blob = false;
|
|
bool lines = false;
|
|
bool singleLine = true;
|
|
if (args.count() > 1) {
|
|
if (!args.at(1).compare(QLatin1String("false"), Qt::CaseInsensitive))
|
|
singleLine = false;
|
|
else if (!args.at(1).compare(QLatin1String("blob"), Qt::CaseInsensitive))
|
|
blob = true;
|
|
else if (!args.at(1).compare(QLatin1String("lines"), Qt::CaseInsensitive))
|
|
lines = true;
|
|
}
|
|
|
|
QFile qfile(resolvePath(m_option->expandEnvVars(file)));
|
|
if (qfile.open(QIODevice::ReadOnly)) {
|
|
QTextStream stream(&qfile);
|
|
if (blob) {
|
|
ret += ProString(stream.readAll());
|
|
} else {
|
|
while (!stream.atEnd()) {
|
|
if (lines) {
|
|
ret += ProString(stream.readLine());
|
|
} else {
|
|
const QString &line = stream.readLine();
|
|
ret += split_value_list(QStringRef(&line).trimmed());
|
|
if (!singleLine)
|
|
ret += ProString("\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case E_FROMFILE:
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("fromfile(file, variable) requires two arguments."));
|
|
} else {
|
|
ProValueMap vars;
|
|
QString fn = resolvePath(m_option->expandEnvVars(args.at(0).toQString(m_tmp1)));
|
|
fn.detach();
|
|
if (evaluateFileInto(fn, &vars, LoadProOnly) == ReturnTrue)
|
|
ret = vars.value(map(args.at(1)));
|
|
}
|
|
break;
|
|
case E_EVAL:
|
|
if (args.count() != 1)
|
|
evalError(fL1S("eval(variable) requires one argument."));
|
|
else
|
|
ret += values(map(args.at(0)));
|
|
break;
|
|
case E_LIST: {
|
|
QString tmp;
|
|
tmp.sprintf(".QMAKE_INTERNAL_TMP_variableName_%d", m_listCount++);
|
|
ret = ProStringList(ProString(tmp));
|
|
ProStringList lst;
|
|
for (const ProString &arg : args)
|
|
lst += split_value_list(arg.toQStringRef(), arg.sourceFile()); // Relies on deep copy
|
|
m_valuemapStack.top()[ret.at(0).toKey()] = lst;
|
|
break; }
|
|
case E_FIND:
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("find(var, str) requires two arguments."));
|
|
} else {
|
|
QRegExp regx(args.at(1).toQString());
|
|
int t = 0;
|
|
const auto vals = values(map(args.at(0)));
|
|
for (const ProString &val : vals) {
|
|
if (regx.indexIn(val.toQString(m_tmp[t])) != -1)
|
|
ret += val;
|
|
t ^= 1;
|
|
}
|
|
}
|
|
break;
|
|
case E_SYSTEM:
|
|
if (!m_skipLevel) {
|
|
if (args.count() < 1 || args.count() > 3) {
|
|
evalError(fL1S("system(command, [mode], [stsvar]) requires one to three arguments."));
|
|
} else {
|
|
bool blob = false;
|
|
bool lines = false;
|
|
bool singleLine = true;
|
|
if (args.count() > 1) {
|
|
if (!args.at(1).compare(QLatin1String("false"), Qt::CaseInsensitive))
|
|
singleLine = false;
|
|
else if (!args.at(1).compare(QLatin1String("blob"), Qt::CaseInsensitive))
|
|
blob = true;
|
|
else if (!args.at(1).compare(QLatin1String("lines"), Qt::CaseInsensitive))
|
|
lines = true;
|
|
}
|
|
int exitCode;
|
|
QByteArray bytes = getCommandOutput(args.at(0).toQString(m_tmp2), &exitCode);
|
|
if (args.count() > 2 && !args.at(2).isEmpty()) {
|
|
m_valuemapStack.top()[args.at(2).toKey()] =
|
|
ProStringList(ProString(QString::number(exitCode)));
|
|
}
|
|
if (lines) {
|
|
QTextStream stream(bytes);
|
|
while (!stream.atEnd())
|
|
ret += ProString(stream.readLine());
|
|
} else {
|
|
QString output = QString::fromLocal8Bit(bytes);
|
|
if (blob) {
|
|
ret += ProString(output);
|
|
} else {
|
|
output.replace(QLatin1Char('\t'), QLatin1Char(' '));
|
|
if (singleLine)
|
|
output.replace(QLatin1Char('\n'), QLatin1Char(' '));
|
|
ret += split_value_list(QStringRef(&output));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case E_UNIQUE:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("unique(var) requires one argument."));
|
|
} else {
|
|
ret = values(map(args.at(0)));
|
|
ret.removeDuplicates();
|
|
}
|
|
break;
|
|
case E_SORTED:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("sorted(var) requires one argument."));
|
|
} else {
|
|
ret = values(map(args.at(0)));
|
|
std::sort(ret.begin(), ret.end());
|
|
}
|
|
break;
|
|
case E_REVERSE:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("reverse(var) requires one argument."));
|
|
} else {
|
|
ProStringList var = values(args.at(0).toKey());
|
|
for (int i = 0; i < var.size() / 2; i++)
|
|
qSwap(var[i], var[var.size() - i - 1]);
|
|
ret += var;
|
|
}
|
|
break;
|
|
case E_QUOTE:
|
|
ret += args;
|
|
break;
|
|
case E_ESCAPE_EXPAND:
|
|
for (int i = 0; i < args.size(); ++i) {
|
|
QString str = args.at(i).toQString();
|
|
QChar *i_data = str.data();
|
|
int i_len = str.length();
|
|
for (int x = 0; x < i_len; ++x) {
|
|
if (*(i_data+x) == QLatin1Char('\\') && x < i_len-1) {
|
|
if (*(i_data+x+1) == QLatin1Char('\\')) {
|
|
++x;
|
|
} else {
|
|
struct {
|
|
char in, out;
|
|
} mapped_quotes[] = {
|
|
{ 'n', '\n' },
|
|
{ 't', '\t' },
|
|
{ 'r', '\r' },
|
|
{ 0, 0 }
|
|
};
|
|
for (int i = 0; mapped_quotes[i].in; ++i) {
|
|
if (*(i_data+x+1) == QLatin1Char(mapped_quotes[i].in)) {
|
|
*(i_data+x) = QLatin1Char(mapped_quotes[i].out);
|
|
if (x < i_len-2)
|
|
memmove(i_data+x+1, i_data+x+2, (i_len-x-2)*sizeof(QChar));
|
|
--i_len;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
ret.append(ProString(QString(i_data, i_len)).setSource(args.at(i)));
|
|
}
|
|
break;
|
|
case E_RE_ESCAPE:
|
|
for (int i = 0; i < args.size(); ++i) {
|
|
const QString &rstr = QRegExp::escape(args.at(i).toQString(m_tmp1));
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(i) : ProString(rstr).setSource(args.at(i)));
|
|
}
|
|
break;
|
|
case E_VAL_ESCAPE:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("val_escape(var) requires one argument."));
|
|
} else {
|
|
const ProStringList &vals = values(args.at(0).toKey());
|
|
ret.reserve(vals.size());
|
|
for (const ProString &str : vals)
|
|
ret += ProString(quoteValue(str));
|
|
}
|
|
break;
|
|
case E_UPPER:
|
|
case E_LOWER:
|
|
case E_TITLE:
|
|
for (int i = 0; i < args.count(); ++i) {
|
|
QString rstr = args.at(i).toQString(m_tmp1);
|
|
if (func_t == E_UPPER) {
|
|
rstr = rstr.toUpper();
|
|
} else {
|
|
rstr = rstr.toLower();
|
|
if (func_t == E_TITLE && rstr.length() > 0)
|
|
rstr[0] = rstr.at(0).toTitleCase();
|
|
}
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(i) : ProString(rstr).setSource(args.at(i)));
|
|
}
|
|
break;
|
|
case E_FILES:
|
|
if (args.count() != 1 && args.count() != 2) {
|
|
evalError(fL1S("files(pattern, recursive=false) requires one or two arguments."));
|
|
} else {
|
|
bool recursive = false;
|
|
if (args.count() == 2)
|
|
recursive = isTrue(args.at(1));
|
|
QStringList dirs;
|
|
QString r = m_option->expandEnvVars(args.at(0).toQString(m_tmp1))
|
|
.replace(QLatin1Char('\\'), QLatin1Char('/'));
|
|
QString pfx;
|
|
if (IoUtils::isRelativePath(r)) {
|
|
pfx = currentDirectory();
|
|
if (!pfx.endsWith(QLatin1Char('/')))
|
|
pfx += QLatin1Char('/');
|
|
}
|
|
int slash = r.lastIndexOf(QLatin1Char('/'));
|
|
if (slash != -1) {
|
|
dirs.append(r.left(slash+1));
|
|
r = r.mid(slash+1);
|
|
} else {
|
|
dirs.append(QString());
|
|
}
|
|
|
|
r.detach(); // Keep m_tmp out of QRegExp's cache
|
|
QRegExp regex(r, Qt::CaseSensitive, QRegExp::Wildcard);
|
|
for (int d = 0; d < dirs.count(); d++) {
|
|
QString dir = dirs[d];
|
|
QDir qdir(pfx + dir);
|
|
for (int i = 0; i < (int)qdir.count(); ++i) {
|
|
if (qdir[i] == statics.strDot || qdir[i] == statics.strDotDot)
|
|
continue;
|
|
QString fname = dir + qdir[i];
|
|
if (IoUtils::fileType(pfx + fname) == IoUtils::FileIsDir) {
|
|
if (recursive)
|
|
dirs.append(fname + QLatin1Char('/'));
|
|
}
|
|
if (regex.exactMatch(qdir[i]))
|
|
ret += ProString(fname).setSource(currentProFile());
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
#ifdef PROEVALUATOR_FULL
|
|
case E_PROMPT: {
|
|
if (args.count() != 1 && args.count() != 2) {
|
|
evalError(fL1S("prompt(question, [decorate=true]) requires one or two arguments."));
|
|
// } else if (currentFileName() == QLatin1String("-")) {
|
|
// evalError(fL1S("prompt(question) cannot be used when '-o -' is used"));
|
|
} else {
|
|
QString msg = m_option->expandEnvVars(args.at(0).toQString(m_tmp1));
|
|
bool decorate = true;
|
|
if (args.count() == 2)
|
|
decorate = isTrue(args.at(1));
|
|
if (decorate) {
|
|
if (!msg.endsWith(QLatin1Char('?')))
|
|
msg += QLatin1Char('?');
|
|
fprintf(stderr, "Project PROMPT: %s ", qPrintable(msg));
|
|
} else {
|
|
fputs(qPrintable(msg), stderr);
|
|
}
|
|
QFile qfile;
|
|
if (qfile.open(stdin, QIODevice::ReadOnly)) {
|
|
QTextStream t(&qfile);
|
|
const QString &line = t.readLine();
|
|
if (t.atEnd()) {
|
|
fputs("\n", stderr);
|
|
evalError(fL1S("Unexpected EOF."));
|
|
return ReturnError;
|
|
}
|
|
ret = split_value_list(QStringRef(&line));
|
|
}
|
|
}
|
|
break; }
|
|
#endif
|
|
case E_REPLACE:
|
|
if (args.count() != 3 ) {
|
|
evalError(fL1S("replace(var, before, after) requires three arguments."));
|
|
} else {
|
|
const QRegExp before(args.at(1).toQString());
|
|
const QString &after(args.at(2).toQString(m_tmp2));
|
|
const auto vals = values(map(args.at(0)));
|
|
for (const ProString &val : vals) {
|
|
QString rstr = val.toQString(m_tmp1);
|
|
QString copy = rstr; // Force a detach on modify
|
|
rstr.replace(before, after);
|
|
ret << (rstr.isSharedWith(m_tmp1) ? val : ProString(rstr).setSource(val));
|
|
}
|
|
}
|
|
break;
|
|
case E_SORT_DEPENDS:
|
|
case E_RESOLVE_DEPENDS:
|
|
if (args.count() < 1 || args.count() > 4) {
|
|
evalError(fL1S("%1(var, [prefix, [suffixes, [prio-suffix]]]) requires one to four arguments.")
|
|
.arg(func.toQString(m_tmp1)));
|
|
} else {
|
|
QHash<ProKey, QSet<ProKey> > dependencies;
|
|
ProValueMap dependees;
|
|
QMultiMap<int, ProString> rootSet;
|
|
ProStringList orgList = values(args.at(0).toKey());
|
|
ProString prefix = args.count() < 2 ? ProString() : args.at(1);
|
|
ProString priosfx = args.count() < 4 ? ProString(".priority") : args.at(3);
|
|
populateDeps(orgList, prefix,
|
|
args.count() < 3 ? ProStringList(ProString(".depends"))
|
|
: split_value_list(args.at(2).toQStringRef()),
|
|
priosfx, dependencies, dependees, rootSet);
|
|
while (!rootSet.isEmpty()) {
|
|
QMultiMap<int, ProString>::iterator it = rootSet.begin();
|
|
const ProString item = *it;
|
|
rootSet.erase(it);
|
|
if ((func_t == E_RESOLVE_DEPENDS) || orgList.contains(item))
|
|
ret.prepend(item);
|
|
for (const ProString &dep : qAsConst(dependees[item.toKey()])) {
|
|
QSet<ProKey> &dset = dependencies[dep.toKey()];
|
|
dset.remove(item.toKey());
|
|
if (dset.isEmpty())
|
|
rootSet.insert(first(ProKey(prefix + dep + priosfx)).toInt(), dep);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case E_ENUMERATE_VARS: {
|
|
QSet<ProString> keys;
|
|
for (const ProValueMap &vmap : qAsConst(m_valuemapStack))
|
|
for (ProValueMap::ConstIterator it = vmap.constBegin(); it != vmap.constEnd(); ++it)
|
|
keys.insert(it.key());
|
|
ret.reserve(keys.size());
|
|
for (const ProString &key : qAsConst(keys))
|
|
ret << key;
|
|
break; }
|
|
case E_SHADOWED:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("shadowed(path) requires one argument."));
|
|
} else {
|
|
QString rstr = m_option->shadowedPath(resolvePath(args.at(0).toQString(m_tmp1)));
|
|
if (rstr.isEmpty())
|
|
break;
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_ABSOLUTE_PATH:
|
|
if (args.count() > 2) {
|
|
evalError(fL1S("absolute_path(path[, base]) requires one or two arguments."));
|
|
} else {
|
|
QString rstr = QDir::cleanPath(
|
|
QDir(args.count() > 1 ? args.at(1).toQString(m_tmp2) : currentDirectory())
|
|
.absoluteFilePath(args.at(0).toQString(m_tmp1)));
|
|
ret << (rstr.isSharedWith(m_tmp1)
|
|
? args.at(0)
|
|
: args.count() > 1 && rstr.isSharedWith(m_tmp2)
|
|
? args.at(1)
|
|
: ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_RELATIVE_PATH:
|
|
if (args.count() > 2) {
|
|
evalError(fL1S("relative_path(path[, base]) requires one or two arguments."));
|
|
} else {
|
|
QDir baseDir(args.count() > 1 ? args.at(1).toQString(m_tmp2) : currentDirectory());
|
|
QString rstr = baseDir.relativeFilePath(baseDir.absoluteFilePath(
|
|
args.at(0).toQString(m_tmp1)));
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_CLEAN_PATH:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("clean_path(path) requires one argument."));
|
|
} else {
|
|
QString rstr = QDir::cleanPath(args.at(0).toQString(m_tmp1));
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_SYSTEM_PATH:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("system_path(path) requires one argument."));
|
|
} else {
|
|
QString rstr = args.at(0).toQString(m_tmp1);
|
|
#ifdef Q_OS_WIN
|
|
rstr.replace(QLatin1Char('/'), QLatin1Char('\\'));
|
|
#else
|
|
rstr.replace(QLatin1Char('\\'), QLatin1Char('/'));
|
|
#endif
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_SHELL_PATH:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("shell_path(path) requires one argument."));
|
|
} else {
|
|
QString rstr = args.at(0).toQString(m_tmp1);
|
|
if (m_dirSep.startsWith(QLatin1Char('\\'))) {
|
|
rstr.replace(QLatin1Char('/'), QLatin1Char('\\'));
|
|
} else {
|
|
rstr.replace(QLatin1Char('\\'), QLatin1Char('/'));
|
|
#ifdef Q_OS_WIN
|
|
// Convert d:/foo/bar to msys-style /d/foo/bar.
|
|
if (rstr.length() > 2 && rstr.at(1) == QLatin1Char(':') && rstr.at(2) == QLatin1Char('/')) {
|
|
rstr[1] = rstr.at(0);
|
|
rstr[0] = QLatin1Char('/');
|
|
}
|
|
#endif
|
|
}
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_SYSTEM_QUOTE:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("system_quote(arg) requires one argument."));
|
|
} else {
|
|
QString rstr = IoUtils::shellQuote(args.at(0).toQString(m_tmp1));
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_SHELL_QUOTE:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("shell_quote(arg) requires one argument."));
|
|
} else {
|
|
QString rstr = args.at(0).toQString(m_tmp1);
|
|
if (m_dirSep.startsWith(QLatin1Char('\\')))
|
|
rstr = IoUtils::shellQuoteWin(rstr);
|
|
else
|
|
rstr = IoUtils::shellQuoteUnix(rstr);
|
|
ret << (rstr.isSharedWith(m_tmp1) ? args.at(0) : ProString(rstr).setSource(args.at(0)));
|
|
}
|
|
break;
|
|
case E_GETENV:
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("getenv(arg) requires one argument."));
|
|
} else {
|
|
const ProString &var = args.at(0);
|
|
const ProString &val = ProString(m_option->getEnv(var.toQString(m_tmp1)));
|
|
ret << val;
|
|
}
|
|
break;
|
|
default:
|
|
evalError(fL1S("Function '%1' is not implemented.").arg(func.toQString(m_tmp1)));
|
|
break;
|
|
}
|
|
|
|
return ReturnTrue;
|
|
}
|
|
|
|
QMakeEvaluator::VisitReturn QMakeEvaluator::evaluateBuiltinConditional(
|
|
int func_t, const ProKey &function, const ProStringList &args)
|
|
{
|
|
traceMsg("calling built-in %s(%s)", dbgKey(function), dbgSepStrList(args));
|
|
|
|
switch (func_t) {
|
|
case T_DEFINED: {
|
|
if (args.count() < 1 || args.count() > 2) {
|
|
evalError(fL1S("defined(function, [\"test\"|\"replace\"|\"var\"])"
|
|
" requires one or two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
const ProKey &var = args.at(0).toKey();
|
|
if (args.count() > 1) {
|
|
if (args[1] == QLatin1String("test")) {
|
|
return returnBool(m_functionDefs.testFunctions.contains(var));
|
|
} else if (args[1] == QLatin1String("replace")) {
|
|
return returnBool(m_functionDefs.replaceFunctions.contains(var));
|
|
} else if (args[1] == QLatin1String("var")) {
|
|
ProValueMap::Iterator it;
|
|
return returnBool(findValues(var, &it));
|
|
}
|
|
evalError(fL1S("defined(function, type): unexpected type [%1].")
|
|
.arg(args.at(1).toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
return returnBool(m_functionDefs.replaceFunctions.contains(var)
|
|
|| m_functionDefs.testFunctions.contains(var));
|
|
}
|
|
case T_EXPORT: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("export(variable) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
const ProKey &var = map(args.at(0));
|
|
for (ProValueMapStack::Iterator vmi = m_valuemapStack.end();
|
|
--vmi != m_valuemapStack.begin(); ) {
|
|
ProValueMap::Iterator it = (*vmi).find(var);
|
|
if (it != (*vmi).end()) {
|
|
if (it->constBegin() == statics.fakeValue.constBegin()) {
|
|
// This is stupid, but qmake doesn't propagate deletions
|
|
m_valuemapStack.first()[var] = ProStringList();
|
|
} else {
|
|
m_valuemapStack.first()[var] = *it;
|
|
}
|
|
(*vmi).erase(it);
|
|
while (--vmi != m_valuemapStack.begin())
|
|
(*vmi).remove(var);
|
|
break;
|
|
}
|
|
}
|
|
return ReturnTrue;
|
|
}
|
|
case T_DISCARD_FROM: {
|
|
if (args.count() != 1 || args.at(0).isEmpty()) {
|
|
evalError(fL1S("discard_from(file) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
if (m_valuemapStack.count() != 1) {
|
|
evalError(fL1S("discard_from() cannot be called from functions."));
|
|
return ReturnFalse;
|
|
}
|
|
QString fn = resolvePath(args.at(0).toQString(m_tmp1));
|
|
ProFile *pro = m_parser->parsedProFile(fn, QMakeParser::ParseOnlyCached);
|
|
if (!pro)
|
|
return ReturnFalse;
|
|
ProValueMap &vmap = m_valuemapStack.first();
|
|
for (auto vit = vmap.begin(); vit != vmap.end(); ) {
|
|
if (!vit->isEmpty()) {
|
|
auto isFrom = [pro](const ProString &s) {
|
|
return s.sourceFile() == pro;
|
|
};
|
|
vit->erase(std::remove_if(vit->begin(), vit->end(), isFrom), vit->end());
|
|
if (vit->isEmpty()) {
|
|
// When an initially non-empty variable becomes entirely empty,
|
|
// undefine it altogether.
|
|
vit = vmap.erase(vit);
|
|
continue;
|
|
}
|
|
}
|
|
++vit;
|
|
}
|
|
for (auto fit = m_functionDefs.testFunctions.begin(); fit != m_functionDefs.testFunctions.end(); ) {
|
|
if (fit->pro() == pro)
|
|
fit = m_functionDefs.testFunctions.erase(fit);
|
|
else
|
|
++fit;
|
|
}
|
|
for (auto fit = m_functionDefs.replaceFunctions.begin(); fit != m_functionDefs.replaceFunctions.end(); ) {
|
|
if (fit->pro() == pro)
|
|
fit = m_functionDefs.replaceFunctions.erase(fit);
|
|
else
|
|
++fit;
|
|
}
|
|
pro->deref();
|
|
ProStringList &iif = m_valuemapStack.first()[ProKey("QMAKE_INTERNAL_INCLUDED_FILES")];
|
|
int idx = iif.indexOf(ProString(fn));
|
|
if (idx >= 0)
|
|
iif.removeAt(idx);
|
|
return ReturnTrue;
|
|
}
|
|
case T_INFILE:
|
|
if (args.count() < 2 || args.count() > 3) {
|
|
evalError(fL1S("infile(file, var, [values]) requires two or three arguments."));
|
|
} else {
|
|
ProValueMap vars;
|
|
QString fn = resolvePath(m_option->expandEnvVars(args.at(0).toQString(m_tmp1)));
|
|
fn.detach();
|
|
VisitReturn ok = evaluateFileInto(fn, &vars, LoadProOnly);
|
|
if (ok != ReturnTrue)
|
|
return ok;
|
|
if (args.count() == 2)
|
|
return returnBool(vars.contains(map(args.at(1))));
|
|
QRegExp regx;
|
|
const QString &qry = args.at(2).toQString(m_tmp1);
|
|
if (qry != QRegExp::escape(qry)) {
|
|
QString copy = qry;
|
|
copy.detach();
|
|
regx.setPattern(copy);
|
|
}
|
|
int t = 0;
|
|
const auto strings = vars.value(map(args.at(1)));
|
|
for (const ProString &s : strings) {
|
|
if ((!regx.isEmpty() && regx.exactMatch(s.toQString(m_tmp[t]))) || s == qry)
|
|
return ReturnTrue;
|
|
t ^= 1;
|
|
}
|
|
}
|
|
return ReturnFalse;
|
|
case T_REQUIRES:
|
|
#ifdef PROEVALUATOR_FULL
|
|
if (checkRequirements(args) == ReturnError)
|
|
return ReturnError;
|
|
#endif
|
|
return ReturnFalse; // Another qmake breakage
|
|
case T_EVAL: {
|
|
VisitReturn ret = ReturnFalse;
|
|
QString contents = args.join(statics.field_sep);
|
|
ProFile *pro = m_parser->parsedProBlock(QStringRef(&contents),
|
|
m_current.pro->fileName(), m_current.line);
|
|
if (m_cumulative || pro->isOk()) {
|
|
m_locationStack.push(m_current);
|
|
visitProBlock(pro, pro->tokPtr());
|
|
ret = ReturnTrue; // This return value is not too useful, but that's qmake
|
|
m_current = m_locationStack.pop();
|
|
}
|
|
pro->deref();
|
|
return ret;
|
|
}
|
|
case T_IF: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("if(condition) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
return evaluateConditional(args.at(0).toQStringRef(),
|
|
m_current.pro->fileName(), m_current.line);
|
|
}
|
|
case T_CONFIG: {
|
|
if (args.count() < 1 || args.count() > 2) {
|
|
evalError(fL1S("CONFIG(config) requires one or two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
if (args.count() == 1)
|
|
return returnBool(isActiveConfig(args.at(0).toQStringRef()));
|
|
const QStringList &mutuals = args.at(1).toQString(m_tmp2).split(QLatin1Char('|'));
|
|
const ProStringList &configs = values(statics.strCONFIG);
|
|
|
|
for (int i = configs.size() - 1; i >= 0; i--) {
|
|
for (int mut = 0; mut < mutuals.count(); mut++) {
|
|
if (configs[i] == mutuals[mut].trimmed())
|
|
return returnBool(configs[i] == args[0]);
|
|
}
|
|
}
|
|
return ReturnFalse;
|
|
}
|
|
case T_CONTAINS: {
|
|
if (args.count() < 2 || args.count() > 3) {
|
|
evalError(fL1S("contains(var, val) requires two or three arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
|
|
const QString &qry = args.at(1).toQString(m_tmp1);
|
|
QRegExp regx;
|
|
if (qry != QRegExp::escape(qry)) {
|
|
QString copy = qry;
|
|
copy.detach();
|
|
regx.setPattern(copy);
|
|
}
|
|
const ProStringList &l = values(map(args.at(0)));
|
|
if (args.count() == 2) {
|
|
int t = 0;
|
|
for (int i = 0; i < l.size(); ++i) {
|
|
const ProString &val = l[i];
|
|
if ((!regx.isEmpty() && regx.exactMatch(val.toQString(m_tmp[t]))) || val == qry)
|
|
return ReturnTrue;
|
|
t ^= 1;
|
|
}
|
|
} else {
|
|
const QStringList &mutuals = args.at(2).toQString(m_tmp3).split(QLatin1Char('|'));
|
|
for (int i = l.size() - 1; i >= 0; i--) {
|
|
const ProString val = l[i];
|
|
for (int mut = 0; mut < mutuals.count(); mut++) {
|
|
if (val == mutuals[mut].trimmed()) {
|
|
return returnBool((!regx.isEmpty()
|
|
&& regx.exactMatch(val.toQString(m_tmp2)))
|
|
|| val == qry);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ReturnFalse;
|
|
}
|
|
case T_COUNT: {
|
|
if (args.count() != 2 && args.count() != 3) {
|
|
evalError(fL1S("count(var, count, op=\"equals\") requires two or three arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
int cnt = values(map(args.at(0))).count();
|
|
int val = args.at(1).toInt();
|
|
if (args.count() == 3) {
|
|
const ProString &comp = args.at(2);
|
|
if (comp == QLatin1String(">") || comp == QLatin1String("greaterThan")) {
|
|
return returnBool(cnt > val);
|
|
} else if (comp == QLatin1String(">=")) {
|
|
return returnBool(cnt >= val);
|
|
} else if (comp == QLatin1String("<") || comp == QLatin1String("lessThan")) {
|
|
return returnBool(cnt < val);
|
|
} else if (comp == QLatin1String("<=")) {
|
|
return returnBool(cnt <= val);
|
|
} else if (comp == QLatin1String("equals") || comp == QLatin1String("isEqual")
|
|
|| comp == QLatin1String("=") || comp == QLatin1String("==")) {
|
|
// fallthrough
|
|
} else {
|
|
evalError(fL1S("Unexpected modifier to count(%2).").arg(comp.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
}
|
|
return returnBool(cnt == val);
|
|
}
|
|
case T_GREATERTHAN:
|
|
case T_LESSTHAN: {
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("%1(variable, value) requires two arguments.")
|
|
.arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
const ProString &rhs = args.at(1);
|
|
const QString &lhs = values(map(args.at(0))).join(statics.field_sep);
|
|
bool ok;
|
|
int rhs_int = rhs.toInt(&ok);
|
|
if (ok) { // do integer compare
|
|
int lhs_int = lhs.toInt(&ok);
|
|
if (ok) {
|
|
if (func_t == T_GREATERTHAN)
|
|
return returnBool(lhs_int > rhs_int);
|
|
return returnBool(lhs_int < rhs_int);
|
|
}
|
|
}
|
|
if (func_t == T_GREATERTHAN)
|
|
return returnBool(lhs > rhs.toQStringRef());
|
|
return returnBool(lhs < rhs.toQStringRef());
|
|
}
|
|
case T_EQUALS:
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("%1(variable, value) requires two arguments.")
|
|
.arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
return returnBool(values(map(args.at(0))).join(statics.field_sep)
|
|
== args.at(1).toQString(m_tmp1));
|
|
case T_CLEAR: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(variable) requires one argument.")
|
|
.arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
ProValueMap *hsh;
|
|
ProValueMap::Iterator it;
|
|
const ProKey &var = map(args.at(0));
|
|
if (!(hsh = findValues(var, &it)))
|
|
return ReturnFalse;
|
|
if (hsh == &m_valuemapStack.top())
|
|
it->clear();
|
|
else
|
|
m_valuemapStack.top()[var].clear();
|
|
return ReturnTrue;
|
|
}
|
|
case T_UNSET: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(variable) requires one argument.")
|
|
.arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
ProValueMap *hsh;
|
|
ProValueMap::Iterator it;
|
|
const ProKey &var = map(args.at(0));
|
|
if (!(hsh = findValues(var, &it)))
|
|
return ReturnFalse;
|
|
if (m_valuemapStack.size() == 1)
|
|
hsh->erase(it);
|
|
else if (hsh == &m_valuemapStack.top())
|
|
*it = statics.fakeValue;
|
|
else
|
|
m_valuemapStack.top()[var] = statics.fakeValue;
|
|
return ReturnTrue;
|
|
}
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 0, 0)
|
|
case T_PARSE_JSON: {
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("parseJson(variable, into) requires two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
|
|
QByteArray json = values(args.at(0).toKey()).join(QLatin1Char(' ')).toUtf8();
|
|
QString parseInto = args.at(1).toQString(m_tmp2);
|
|
return parseJsonInto(json, parseInto, &m_valuemapStack.top());
|
|
}
|
|
#endif
|
|
case T_INCLUDE: {
|
|
if (args.count() < 1 || args.count() > 3) {
|
|
evalError(fL1S("include(file, [into, [silent]]) requires one, two or three arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
QString parseInto;
|
|
LoadFlags flags = 0;
|
|
if (m_cumulative)
|
|
flags = LoadSilent;
|
|
if (args.count() >= 2) {
|
|
if (!args.at(1).isEmpty())
|
|
parseInto = args.at(1) + QLatin1Char('.');
|
|
if (args.count() >= 3 && isTrue(args.at(2)))
|
|
flags = LoadSilent;
|
|
}
|
|
QString fn = resolvePath(m_option->expandEnvVars(args.at(0).toQString(m_tmp1)));
|
|
fn.detach();
|
|
VisitReturn ok;
|
|
if (parseInto.isEmpty()) {
|
|
ok = evaluateFileChecked(fn, QMakeHandler::EvalIncludeFile, LoadProOnly | flags);
|
|
} else {
|
|
ProValueMap symbols;
|
|
if ((ok = evaluateFileInto(fn, &symbols, LoadAll | flags)) == ReturnTrue) {
|
|
ProValueMap newMap;
|
|
for (ProValueMap::ConstIterator
|
|
it = m_valuemapStack.top().constBegin(),
|
|
end = m_valuemapStack.top().constEnd();
|
|
it != end; ++it) {
|
|
const ProString &ky = it.key();
|
|
if (!ky.startsWith(parseInto))
|
|
newMap[it.key()] = it.value();
|
|
}
|
|
for (ProValueMap::ConstIterator it = symbols.constBegin();
|
|
it != symbols.constEnd(); ++it) {
|
|
const QString &ky = it.key().toQString(m_tmp1);
|
|
if (!ky.startsWith(QLatin1Char('.')))
|
|
newMap.insert(ProKey(parseInto + ky), it.value());
|
|
}
|
|
m_valuemapStack.top() = newMap;
|
|
}
|
|
}
|
|
if (ok == ReturnFalse && (flags & LoadSilent))
|
|
ok = ReturnTrue;
|
|
return ok;
|
|
}
|
|
case T_LOAD: {
|
|
bool ignore_error = false;
|
|
if (args.count() == 2) {
|
|
ignore_error = isTrue(args.at(1));
|
|
} else if (args.count() != 1) {
|
|
evalError(fL1S("load(feature) requires one or two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
VisitReturn ok = evaluateFeatureFile(m_option->expandEnvVars(args.at(0).toQString()),
|
|
ignore_error);
|
|
if (ok == ReturnFalse && ignore_error)
|
|
ok = ReturnTrue;
|
|
return ok;
|
|
}
|
|
case T_DEBUG: {
|
|
#ifdef PROEVALUATOR_DEBUG
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("debug(level, message) requires two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
int level = args.at(0).toInt();
|
|
if (level <= m_debugLevel) {
|
|
const QString &msg = m_option->expandEnvVars(args.at(1).toQString(m_tmp2));
|
|
debugMsg(level, "Project DEBUG: %s", qPrintable(msg));
|
|
}
|
|
#endif
|
|
return ReturnTrue;
|
|
}
|
|
case T_LOG:
|
|
case T_ERROR:
|
|
case T_WARNING:
|
|
case T_MESSAGE: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("%1(message) requires one argument.")
|
|
.arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
const QString &msg = m_option->expandEnvVars(args.at(0).toQString(m_tmp2));
|
|
if (!m_skipLevel) {
|
|
if (func_t == T_LOG) {
|
|
#ifdef PROEVALUATOR_FULL
|
|
fputs(msg.toLatin1().constData(), stderr);
|
|
#endif
|
|
} else if (!msg.isEmpty() || func_t != T_ERROR) {
|
|
m_handler->fileMessage(
|
|
(func_t == T_ERROR ? QMakeHandler::ErrorMessage :
|
|
func_t == T_WARNING ? QMakeHandler::WarningMessage :
|
|
QMakeHandler::InfoMessage)
|
|
| (m_cumulative ? QMakeHandler::CumulativeEvalMessage : 0),
|
|
fL1S("Project %1: %2").arg(function.toQString(m_tmp1).toUpper(), msg));
|
|
}
|
|
}
|
|
return (func_t == T_ERROR && !m_cumulative) ? ReturnError : ReturnTrue;
|
|
}
|
|
case T_SYSTEM: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("system(exec) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
#ifdef PROEVALUATOR_FULL
|
|
if (m_cumulative) // Anything else would be insanity
|
|
return ReturnFalse;
|
|
#if QT_CONFIG(process)
|
|
QProcess proc;
|
|
proc.setProcessChannelMode(QProcess::ForwardedChannels);
|
|
runProcess(&proc, args.at(0).toQString(m_tmp2));
|
|
return returnBool(proc.exitStatus() == QProcess::NormalExit && proc.exitCode() == 0);
|
|
#else
|
|
int ec = system((QLatin1String("cd ")
|
|
+ IoUtils::shellQuote(QDir::toNativeSeparators(currentDirectory()))
|
|
+ QLatin1String(" && ") + args.at(0)).toLocal8Bit().constData());
|
|
# ifdef Q_OS_UNIX
|
|
if (ec != -1 && WIFSIGNALED(ec) && (WTERMSIG(ec) == SIGQUIT || WTERMSIG(ec) == SIGINT))
|
|
raise(WTERMSIG(ec));
|
|
# endif
|
|
return returnBool(ec == 0);
|
|
#endif
|
|
#else
|
|
return ReturnTrue;
|
|
#endif
|
|
}
|
|
case T_ISEMPTY: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("isEmpty(var) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
return returnBool(values(map(args.at(0))).isEmpty());
|
|
}
|
|
case T_EXISTS: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("exists(file) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
const QString &file = resolvePath(m_option->expandEnvVars(args.at(0).toQString(m_tmp1)));
|
|
|
|
// Don't use VFS here:
|
|
// - it supports neither listing nor even directories
|
|
// - it's unlikely that somebody would test for files they created themselves
|
|
if (IoUtils::exists(file))
|
|
return ReturnTrue;
|
|
int slsh = file.lastIndexOf(QLatin1Char('/'));
|
|
QString fn = file.mid(slsh+1);
|
|
if (fn.contains(QLatin1Char('*')) || fn.contains(QLatin1Char('?'))) {
|
|
QString dirstr = file.left(slsh+1);
|
|
if (!QDir(dirstr).entryList(QStringList(fn)).isEmpty())
|
|
return ReturnTrue;
|
|
}
|
|
|
|
return ReturnFalse;
|
|
}
|
|
case T_MKPATH: {
|
|
if (args.count() != 1) {
|
|
evalError(fL1S("mkpath(file) requires one argument."));
|
|
return ReturnFalse;
|
|
}
|
|
#ifdef PROEVALUATOR_FULL
|
|
const QString &fn = resolvePath(args.at(0).toQString(m_tmp1));
|
|
if (!QDir::current().mkpath(fn)) {
|
|
evalError(fL1S("Cannot create directory %1.").arg(QDir::toNativeSeparators(fn)));
|
|
return ReturnFalse;
|
|
}
|
|
#endif
|
|
return ReturnTrue;
|
|
}
|
|
case T_WRITE_FILE: {
|
|
if (args.count() > 3) {
|
|
evalError(fL1S("write_file(name, [content var, [append] [exe]]) requires one to three arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
QIODevice::OpenMode mode = QIODevice::Truncate;
|
|
bool exe = false;
|
|
QString contents;
|
|
if (args.count() >= 2) {
|
|
const ProStringList &vals = values(args.at(1).toKey());
|
|
if (!vals.isEmpty())
|
|
contents = vals.join(QLatin1Char('\n')) + QLatin1Char('\n');
|
|
if (args.count() >= 3) {
|
|
const auto opts = split_value_list(args.at(2).toQStringRef());
|
|
for (const ProString &opt : opts) {
|
|
opt.toQString(m_tmp3);
|
|
if (m_tmp3 == QLatin1String("append")) {
|
|
mode = QIODevice::Append;
|
|
} else if (m_tmp3 == QLatin1String("exe")) {
|
|
exe = true;
|
|
} else {
|
|
evalError(fL1S("write_file(): invalid flag %1.").arg(m_tmp3));
|
|
return ReturnFalse;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
QString path = resolvePath(args.at(0).toQString(m_tmp1));
|
|
path.detach(); // make sure to not leak m_tmp1 into the map of written files.
|
|
return writeFile(QString(), path, mode, exe, contents);
|
|
}
|
|
case T_TOUCH: {
|
|
if (args.count() != 2) {
|
|
evalError(fL1S("touch(file, reffile) requires two arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
#ifdef PROEVALUATOR_FULL
|
|
const QString &tfn = resolvePath(args.at(0).toQString(m_tmp1));
|
|
const QString &rfn = resolvePath(args.at(1).toQString(m_tmp2));
|
|
#ifdef Q_OS_UNIX
|
|
struct stat st;
|
|
if (stat(rfn.toLocal8Bit().constData(), &st)) {
|
|
evalError(fL1S("Cannot stat() reference file %1: %2.").arg(rfn, fL1S(strerror(errno))));
|
|
return ReturnFalse;
|
|
}
|
|
struct utimbuf utb;
|
|
utb.actime = time(0);
|
|
utb.modtime = st.st_mtime;
|
|
if (utime(tfn.toLocal8Bit().constData(), &utb)) {
|
|
evalError(fL1S("Cannot touch %1: %2.").arg(tfn, fL1S(strerror(errno))));
|
|
return ReturnFalse;
|
|
}
|
|
#else
|
|
HANDLE rHand = CreateFile((wchar_t*)rfn.utf16(),
|
|
GENERIC_READ, FILE_SHARE_READ,
|
|
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
if (rHand == INVALID_HANDLE_VALUE) {
|
|
evalError(fL1S("Cannot open reference file %1: %2").arg(rfn, windowsErrorCode()));
|
|
return ReturnFalse;
|
|
}
|
|
FILETIME ft;
|
|
GetFileTime(rHand, 0, 0, &ft);
|
|
CloseHandle(rHand);
|
|
HANDLE wHand = CreateFile((wchar_t*)tfn.utf16(),
|
|
GENERIC_WRITE, FILE_SHARE_READ,
|
|
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
if (wHand == INVALID_HANDLE_VALUE) {
|
|
evalError(fL1S("Cannot open %1: %2").arg(tfn, windowsErrorCode()));
|
|
return ReturnFalse;
|
|
}
|
|
SetFileTime(wHand, 0, 0, &ft);
|
|
CloseHandle(wHand);
|
|
#endif
|
|
#endif
|
|
return ReturnTrue;
|
|
}
|
|
case T_CACHE: {
|
|
if (args.count() > 3) {
|
|
evalError(fL1S("cache(var, [set|add|sub] [transient] [super|stash], [srcvar]) requires one to three arguments."));
|
|
return ReturnFalse;
|
|
}
|
|
bool persist = true;
|
|
enum { TargetStash, TargetCache, TargetSuper } target = TargetCache;
|
|
enum { CacheSet, CacheAdd, CacheSub } mode = CacheSet;
|
|
ProKey srcvar;
|
|
if (args.count() >= 2) {
|
|
const auto opts = split_value_list(args.at(1).toQStringRef());
|
|
for (const ProString &opt : opts) {
|
|
opt.toQString(m_tmp3);
|
|
if (m_tmp3 == QLatin1String("transient")) {
|
|
persist = false;
|
|
} else if (m_tmp3 == QLatin1String("super")) {
|
|
target = TargetSuper;
|
|
} else if (m_tmp3 == QLatin1String("stash")) {
|
|
target = TargetStash;
|
|
} else if (m_tmp3 == QLatin1String("set")) {
|
|
mode = CacheSet;
|
|
} else if (m_tmp3 == QLatin1String("add")) {
|
|
mode = CacheAdd;
|
|
} else if (m_tmp3 == QLatin1String("sub")) {
|
|
mode = CacheSub;
|
|
} else {
|
|
evalError(fL1S("cache(): invalid flag %1.").arg(m_tmp3));
|
|
return ReturnFalse;
|
|
}
|
|
}
|
|
if (args.count() >= 3) {
|
|
srcvar = args.at(2).toKey();
|
|
} else if (mode != CacheSet) {
|
|
evalError(fL1S("cache(): modes other than 'set' require a source variable."));
|
|
return ReturnFalse;
|
|
}
|
|
}
|
|
QString varstr;
|
|
ProKey dstvar = args.at(0).toKey();
|
|
if (!dstvar.isEmpty()) {
|
|
if (srcvar.isEmpty())
|
|
srcvar = dstvar;
|
|
ProValueMap::Iterator srcvarIt;
|
|
if (!findValues(srcvar, &srcvarIt)) {
|
|
evalError(fL1S("Variable %1 is not defined.").arg(srcvar.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
// The caches for the host and target may differ (e.g., when we are manipulating
|
|
// CONFIG), so we cannot compute a common new value for both.
|
|
const ProStringList &diffval = *srcvarIt;
|
|
ProStringList newval;
|
|
bool changed = false;
|
|
for (bool hostBuild = false; ; hostBuild = true) {
|
|
#ifdef PROEVALUATOR_THREAD_SAFE
|
|
m_option->mutex.lock();
|
|
#endif
|
|
QMakeBaseEnv *baseEnv =
|
|
m_option->baseEnvs.value(QMakeBaseKey(m_buildRoot, m_stashfile, hostBuild));
|
|
#ifdef PROEVALUATOR_THREAD_SAFE
|
|
// It's ok to unlock this before locking baseEnv,
|
|
// as we have no intention to initialize the env.
|
|
m_option->mutex.unlock();
|
|
#endif
|
|
do {
|
|
if (!baseEnv)
|
|
break;
|
|
#ifdef PROEVALUATOR_THREAD_SAFE
|
|
QMutexLocker locker(&baseEnv->mutex);
|
|
if (baseEnv->inProgress && baseEnv->evaluator != this) {
|
|
// The env is still in the works, but it may be already past the cache
|
|
// loading. So we need to wait for completion and amend it as usual.
|
|
QThreadPool::globalInstance()->releaseThread();
|
|
baseEnv->cond.wait(&baseEnv->mutex);
|
|
QThreadPool::globalInstance()->reserveThread();
|
|
}
|
|
if (!baseEnv->isOk)
|
|
break;
|
|
#endif
|
|
QMakeEvaluator *baseEval = baseEnv->evaluator;
|
|
const ProStringList &oldval = baseEval->values(dstvar);
|
|
if (mode == CacheSet) {
|
|
newval = diffval;
|
|
} else {
|
|
newval = oldval;
|
|
if (mode == CacheAdd)
|
|
newval += diffval;
|
|
else
|
|
newval.removeEach(diffval);
|
|
}
|
|
if (oldval != newval) {
|
|
if (target != TargetStash || !m_stashfile.isEmpty()) {
|
|
baseEval->valuesRef(dstvar) = newval;
|
|
if (target == TargetSuper) {
|
|
do {
|
|
if (dstvar == QLatin1String("QMAKEPATH")) {
|
|
baseEval->m_qmakepath = newval.toQStringList();
|
|
baseEval->updateMkspecPaths();
|
|
} else if (dstvar == QLatin1String("QMAKEFEATURES")) {
|
|
baseEval->m_qmakefeatures = newval.toQStringList();
|
|
} else {
|
|
break;
|
|
}
|
|
baseEval->updateFeaturePaths();
|
|
if (hostBuild == m_hostBuild)
|
|
m_featureRoots = baseEval->m_featureRoots;
|
|
} while (false);
|
|
}
|
|
}
|
|
changed = true;
|
|
}
|
|
} while (false);
|
|
if (hostBuild)
|
|
break;
|
|
}
|
|
// We assume that whatever got the cached value to be what it is now will do so
|
|
// the next time as well, so we just skip the persisting if nothing changed.
|
|
if (!persist || !changed)
|
|
return ReturnTrue;
|
|
varstr = dstvar.toQString();
|
|
if (mode == CacheAdd)
|
|
varstr += QLatin1String(" +=");
|
|
else if (mode == CacheSub)
|
|
varstr += QLatin1String(" -=");
|
|
else
|
|
varstr += QLatin1String(" =");
|
|
if (diffval.count() == 1) {
|
|
varstr += QLatin1Char(' ');
|
|
varstr += quoteValue(diffval.at(0));
|
|
} else if (!diffval.isEmpty()) {
|
|
for (const ProString &vval : diffval) {
|
|
varstr += QLatin1String(" \\\n ");
|
|
varstr += quoteValue(vval);
|
|
}
|
|
}
|
|
varstr += QLatin1Char('\n');
|
|
}
|
|
QString fn;
|
|
if (target == TargetSuper) {
|
|
if (m_superfile.isEmpty()) {
|
|
m_superfile = QDir::cleanPath(m_outputDir + QLatin1String("/.qmake.super"));
|
|
printf("Info: creating super cache file %s\n", qPrintable(QDir::toNativeSeparators(m_superfile)));
|
|
valuesRef(ProKey("_QMAKE_SUPER_CACHE_")) << ProString(m_superfile);
|
|
}
|
|
fn = m_superfile;
|
|
} else if (target == TargetCache) {
|
|
if (m_cachefile.isEmpty()) {
|
|
m_cachefile = QDir::cleanPath(m_outputDir + QLatin1String("/.qmake.cache"));
|
|
printf("Info: creating cache file %s\n", qPrintable(QDir::toNativeSeparators(m_cachefile)));
|
|
valuesRef(ProKey("_QMAKE_CACHE_")) << ProString(m_cachefile);
|
|
// We could update m_{source,build}Root and m_featureRoots here, or even
|
|
// "re-home" our rootEnv, but this doesn't sound too useful - if somebody
|
|
// wanted qmake to find something in the build directory, he could have
|
|
// done so "from the outside".
|
|
// The sub-projects will find the new cache all by themselves.
|
|
}
|
|
fn = m_cachefile;
|
|
} else {
|
|
fn = m_stashfile;
|
|
if (fn.isEmpty())
|
|
fn = QDir::cleanPath(m_outputDir + QLatin1String("/.qmake.stash"));
|
|
if (!m_vfs->exists(fn)) {
|
|
printf("Info: creating stash file %s\n", qPrintable(QDir::toNativeSeparators(fn)));
|
|
valuesRef(ProKey("_QMAKE_STASH_")) << ProString(fn);
|
|
}
|
|
}
|
|
return writeFile(fL1S("cache "), fn, QIODevice::Append, false, varstr);
|
|
}
|
|
case T_RELOAD_PROPERTIES:
|
|
#ifdef QT_BUILD_QMAKE
|
|
m_option->reloadProperties();
|
|
#endif
|
|
return ReturnTrue;
|
|
default:
|
|
evalError(fL1S("Function '%1' is not implemented.").arg(function.toQString(m_tmp1)));
|
|
return ReturnFalse;
|
|
}
|
|
}
|
|
|
|
QT_END_NAMESPACE
|