aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib/corelib/language/probesresolver.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/lib/corelib/language/probesresolver.cpp')
-rw-r--r--src/lib/corelib/language/probesresolver.cpp303
1 files changed, 0 insertions, 303 deletions
diff --git a/src/lib/corelib/language/probesresolver.cpp b/src/lib/corelib/language/probesresolver.cpp
deleted file mode 100644
index 1462c706b..000000000
--- a/src/lib/corelib/language/probesresolver.cpp
+++ /dev/null
@@ -1,303 +0,0 @@
-/****************************************************************************
-**
-** Copyright (C) 2022 The Qt Company Ltd.
-** Copyright (C) 2022 Raphaƫl Cotty <raphael.cotty@gmail.com>
-** Contact: https://www.qt.io/licensing/
-**
-** This file is part of Qbs.
-**
-** $QT_BEGIN_LICENSE:LGPL$
-** 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 Lesser General Public License Usage
-** Alternatively, this file may be used under the terms of the GNU Lesser
-** General Public License version 3 as published by the Free Software
-** Foundation and appearing in the file LICENSE.LGPL3 included in the
-** packaging of this file. Please review the following information to
-** ensure the GNU Lesser General Public License version 3 requirements
-** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
-**
-** GNU General Public License Usage
-** Alternatively, this file may be used under the terms of the GNU
-** General Public License version 2.0 or (at your option) the GNU General
-** Public license version 3 or any later version approved by the KDE Free
-** Qt Foundation. The licenses are as published by the Free Software
-** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
-** 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-2.0.html and
-** https://www.gnu.org/licenses/gpl-3.0.html.
-**
-** $QT_END_LICENSE$
-**
-****************************************************************************/
-
-#include "probesresolver.h"
-
-#include "evaluator.h"
-#include "filecontext.h"
-#include "item.h"
-#include "itemreader.h"
-#include "language.h"
-#include "scriptengine.h"
-#include "value.h"
-
-#include <api/languageinfo.h>
-#include <language/language.h>
-#include <logging/categories.h>
-#include <logging/logger.h>
-#include <logging/translator.h>
-#include <tools/profiling.h>
-#include <tools/scripttools.h>
-#include <tools/setupprojectparameters.h>
-#include <tools/stringconstants.h>
-
-#include <quickjs.h>
-
-namespace qbs {
-namespace Internal {
-
-static QString probeGlobalId(Item *probe)
-{
- QString id;
-
- for (Item *obj = probe; obj; obj = obj->prototype()) {
- if (!obj->id().isEmpty()) {
- id = obj->id();
- break;
- }
- }
-
- if (id.isEmpty())
- return {};
-
- QBS_CHECK(probe->file());
- return id + QLatin1Char('_') + probe->file()->filePath();
-}
-
-ProbesResolver::ProbesResolver(const SetupProjectParameters &parameters, Evaluator &evaluator,
- Logger &logger)
- : m_parameters(parameters), m_evaluator(evaluator), m_logger(logger)
-{
-}
-
-void ProbesResolver::setOldProjectProbes(const std::vector<ProbeConstPtr> &oldProbes)
-{
- m_oldProjectProbes.clear();
- for (const ProbeConstPtr& probe : oldProbes)
- m_oldProjectProbes[probe->globalId()] << probe;
-}
-
-void ProbesResolver::setOldProductProbes(
- const QHash<QString, std::vector<ProbeConstPtr>> &oldProbes)
-{
- m_oldProductProbes = oldProbes;
-}
-
-std::vector<ProbeConstPtr> ProbesResolver::resolveProbes(const ProductContext &productContext, Item *item)
-{
- AccumulatingTimer probesTimer(m_parameters.logElapsedTime() ? &m_elapsedTimeProbes : nullptr);
- EvalContextSwitcher evalContextSwitcher(m_evaluator.engine(), EvalContext::ProbeExecution);
- std::vector<ProbeConstPtr> probes;
- for (Item * const child : item->children())
- if (child->type() == ItemType::Probe)
- probes.push_back(resolveProbe(productContext, item, child));
- return probes;
-}
-
-ProbeConstPtr ProbesResolver::resolveProbe(const ProductContext &productContext, Item *parent,
- Item *probe)
-{
- qCDebug(lcModuleLoader) << "Resolving Probe at " << probe->location().toString();
- ++m_probesEncountered;
- const QString &probeId = probeGlobalId(probe);
- if (Q_UNLIKELY(probeId.isEmpty()))
- throw ErrorInfo(Tr::tr("Probe.id must be set."), probe->location());
- const JSSourceValueConstPtr configureScript
- = probe->sourceProperty(StringConstants::configureProperty());
- QBS_CHECK(configureScript);
- if (Q_UNLIKELY(configureScript->sourceCode() == StringConstants::undefinedValue()))
- throw ErrorInfo(Tr::tr("Probe.configure must be set."), probe->location());
- using ProbeProperty = std::pair<QString, ScopedJsValue>;
- std::vector<ProbeProperty> probeBindings;
- ScriptEngine * const engine = m_evaluator.engine();
- JSContext * const ctx = engine->context();
- QVariantMap initialProperties;
- for (Item *obj = probe; obj; obj = obj->prototype()) {
- const Item::PropertyMap &props = obj->properties();
- for (auto it = props.cbegin(); it != props.cend(); ++it) {
- const QString &name = it.key();
- if (name == StringConstants::configureProperty())
- continue;
- const JSValue value = m_evaluator.value(probe, name);
- probeBindings.emplace_back(name, ScopedJsValue(ctx, value));
- if (name != StringConstants::conditionProperty())
- initialProperties.insert(name, getJsVariant(ctx, value));
- }
- }
- const bool condition = m_evaluator.boolValue(probe, StringConstants::conditionProperty());
- const QString &sourceCode = configureScript->sourceCode().toString();
- ProbeConstPtr resolvedProbe;
- if (parent->type() == ItemType::Project
- || productContext.name.startsWith(StringConstants::shadowProductPrefix())) {
- resolvedProbe = findOldProjectProbe(probeId, condition, initialProperties, sourceCode);
- } else {
- resolvedProbe = findOldProductProbe(productContext.uniqueName, condition,
- initialProperties, sourceCode);
- }
- if (!resolvedProbe) {
- resolvedProbe = findCurrentProbe(probe->location(), condition, initialProperties);
- if (resolvedProbe) {
- qCDebug(lcModuleLoader) << "probe results cached from current run";
- ++m_probesCachedCurrent;
- }
- } else {
- qCDebug(lcModuleLoader) << "probe results cached from earlier run";
- ++m_probesCachedOld;
- }
- ScopedJsValue configureScope(ctx, JS_UNDEFINED);
- std::vector<QString> importedFilesUsedInConfigure;
- if (!condition) {
- qCDebug(lcModuleLoader) << "Probe disabled; skipping";
- } else if (!resolvedProbe) {
- ++m_probesRun;
- qCDebug(lcModuleLoader) << "configure script needs to run";
- const Evaluator::FileContextScopes fileCtxScopes
- = m_evaluator.fileContextScopes(configureScript->file());
- configureScope.setValue(engine->newObject());
- for (const ProbeProperty &b : probeBindings)
- setJsProperty(ctx, configureScope, b.first, JS_DupValue(ctx, b.second));
- engine->clearRequestedProperties();
- ScopedJsValue sv(ctx, engine->evaluate(JsValueOwner::Caller,
- configureScript->sourceCodeForEvaluation(), {}, 1,
- {fileCtxScopes.fileScope, fileCtxScopes.importScope, configureScope}));
- if (JsException ex = engine->checkAndClearException(configureScript->location()))
- throw ex.toErrorInfo();
- importedFilesUsedInConfigure = engine->importedFilesUsedInScript();
- } else {
- importedFilesUsedInConfigure = resolvedProbe->importedFilesUsed();
- }
- QVariantMap properties;
- for (const ProbeProperty &b : probeBindings) {
- QVariant newValue;
- if (resolvedProbe) {
- newValue = resolvedProbe->properties().value(b.first);
- } else {
- if (condition) {
- JSValue v = getJsProperty(ctx, configureScope, b.first);
- const JSValue saved = v;
- ScopedJsValue valueMgr(ctx, saved);
- const PropertyDeclaration decl = probe->propertyDeclaration(b.first);
- m_evaluator.convertToPropertyType(decl, probe->location(), v);
-
- // If the value was converted from scalar to array as per our convenience
- // functionality, then the original value is now the only element of a
- // newly allocated array and thus gets deleted via that array.
- // The array itself is owned by the script engine, so we must stay out of
- // memory management here.
- if (v != saved)
- valueMgr.setValue(JS_UNDEFINED);
-
- if (JsException ex = engine->checkAndClearException({}))
- throw ex.toErrorInfo();
- newValue = getJsVariant(ctx, v);
- } else {
- newValue = initialProperties.value(b.first);
- }
- }
- if (newValue != getJsVariant(ctx, b.second))
- probe->setProperty(b.first, VariantValue::create(newValue));
- if (!resolvedProbe)
- properties.insert(b.first, newValue);
- }
- if (!resolvedProbe) {
- resolvedProbe = Probe::create(probeId, probe->location(), condition,
- sourceCode, properties, initialProperties,
- importedFilesUsedInConfigure);
- m_currentProbes[probe->location()] << resolvedProbe;
- }
- return resolvedProbe;
-}
-
-ProbeConstPtr ProbesResolver::findOldProjectProbe(
- const QString &globalId,
- bool condition,
- const QVariantMap &initialProperties,
- const QString &sourceCode) const
-{
- if (m_parameters.forceProbeExecution())
- return {};
-
- for (const ProbeConstPtr &oldProbe : m_oldProjectProbes.value(globalId)) {
- if (probeMatches(oldProbe, condition, initialProperties, sourceCode, CompareScript::Yes))
- return oldProbe;
- }
-
- return {};
-}
-
-ProbeConstPtr ProbesResolver::findOldProductProbe(
- const QString &productName,
- bool condition,
- const QVariantMap &initialProperties,
- const QString &sourceCode) const
-{
- if (m_parameters.forceProbeExecution())
- return {};
-
- for (const ProbeConstPtr &oldProbe : m_oldProductProbes.value(productName)) {
- if (probeMatches(oldProbe, condition, initialProperties, sourceCode, CompareScript::Yes))
- return oldProbe;
- }
-
- return {};
-}
-
-ProbeConstPtr ProbesResolver::findCurrentProbe(
- const CodeLocation &location,
- bool condition,
- const QVariantMap &initialProperties) const
-{
- const std::vector<ProbeConstPtr> &cachedProbes = m_currentProbes.value(location);
- for (const ProbeConstPtr &probe : cachedProbes) {
- if (probeMatches(probe, condition, initialProperties, QString(), CompareScript::No))
- return probe;
- }
- return {};
-}
-
-bool ProbesResolver::probeMatches(const ProbeConstPtr &probe, bool condition,
- const QVariantMap &initialProperties, const QString &configureScript,
- CompareScript compareScript) const
-{
- return probe->condition() == condition
- && probe->initialProperties() == initialProperties
- && (compareScript == CompareScript::No
- || (probe->configureScript() == configureScript
- && !probe->needsReconfigure(m_lastResolveTime)));
-}
-
-void ProbesResolver::printProfilingInfo(int indent)
-{
- if (!m_parameters.logElapsedTime())
- return;
- const QByteArray prefix(indent, ' ');
- m_logger.qbsLog(LoggerInfo, true)
- << prefix
- << Tr::tr("Running Probes took %1.").arg(elapsedTimeString(m_elapsedTimeProbes));
- m_logger.qbsLog(LoggerInfo, true)
- << prefix
- << Tr::tr("%1 probes encountered, %2 configure scripts executed, "
- "%3 re-used from current run, %4 re-used from earlier run.")
- .arg(m_probesEncountered).arg(m_probesRun).arg(m_probesCachedCurrent)
- .arg(m_probesCachedOld);
-}
-
-} // namespace Internal
-} // namespace qbs