Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qqmlscriptblob.cpp
Go to the documentation of this file.
1// Copyright (C) 2019 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant
4
5#include <private/qqmlengine_p.h>
6#include <private/qqmlirbuilder_p.h>
7#include <private/qqmlscriptblob_p.h>
8#include <private/qqmlsourcecoordinate_p.h>
9#include <private/qqmlcontextdata_p.h>
10#include <private/qv4runtimecodegen_p.h>
11#include <private/qv4script_p.h>
12
13#include <QtCore/qloggingcategory.h>
14
16
17Q_LOGGING_CATEGORY(DBG_DISK_CACHE, "qt.qml.diskcache")
18
19QQmlScriptBlob::QQmlScriptBlob(const QUrl &url, QQmlTypeLoader *loader, IsESModule isESModule)
20 : QQmlNotifyingBlob(url, JavaScriptFile, loader)
21 , m_isModule(isESModule == IsESModule::Yes)
22{
23}
24
25QQmlScriptBlob::~QQmlScriptBlob()
26{
27}
28
29void QQmlScriptBlob::dataReceived(const SourceCodeData &data)
30{
31 assertTypeLoaderThread();
32
33 if (data.isCacheable()) {
34 if (m_typeLoader->readCacheFile()) {
35 auto unit = QQml::makeRefPointer<QV4::CompiledData::CompilationUnit>();
36 QString error;
37 if (unit->loadFromDisk(url(), data.sourceTimeStamp(), &error)) {
38 initializeFromCompilationUnit(std::move(unit));
39 return;
40 } else {
41 qCDebug(DBG_DISK_CACHE()) << "Error loading" << urlString()
42 << "from disk cache:" << error;
43 }
44 }
45 }
46
47 if (!data.exists()) {
48 if (m_cachedUnitStatus == QQmlMetaType::CachedUnitLookupError::VersionMismatch)
49 setError(QQmlTypeLoader::tr("File was compiled ahead of time with an incompatible version of Qt and the original file cannot be found. Please recompile"));
50 else
51 setError(QQmlTypeLoader::tr("No such file or directory"));
52 return;
53 }
54
55 QString error;
56 QString source = data.readAll(&error);
57 if (!error.isEmpty()) {
58 setError(error);
59 return;
60 }
61
62 QQmlRefPointer<QV4::CompiledData::CompilationUnit> unit;
63
64 if (m_isModule) {
65 QList<QQmlJS::DiagnosticMessage> diagnostics;
66 unit = QV4::Compiler::Codegen::compileModule(
67 m_typeLoader->isDebugging(), urlString(), source, data.sourceTimeStamp(),
68 &diagnostics);
69 QList<QQmlError> errors = QQmlEnginePrivate::qmlErrorFromDiagnostics(urlString(), diagnostics);
70 if (!errors.isEmpty()) {
71 setError(errors);
72 return;
73 }
74 } else {
75 QmlIR::Document irUnit(urlString(), finalUrlString(), m_typeLoader->isDebugging());
76
77 irUnit.jsModule.sourceTimeStamp = data.sourceTimeStamp();
78
79 QmlIR::ScriptDirectivesCollector collector(&irUnit);
80 irUnit.jsParserEngine.setDirectives(&collector);
81
82 QList<QQmlError> errors;
83 const QString fragment = finalUrl().fragment();
84 irUnit.javaScriptCompilationUnit = QV4::Script::precompile(
85 &irUnit.jsModule, &irUnit.jsParserEngine, &irUnit.jsGenerator, urlString(),
86 source, &errors, fragment == QLatin1String("global")
87 ? QV4::Compiler::ContextType::Global
88 : QV4::Compiler::ContextType::ScriptImportedByQML,
89 fragment == QLatin1String("include")
90 ? QV4::Script::InheritContext::Yes
91 : QV4::Script::InheritContext::No);
92
93 source.clear();
94 if (!errors.isEmpty()) {
95 setError(errors);
96 return;
97 }
98
99 QmlIR::QmlUnitGenerator qmlGenerator;
100 qmlGenerator.generate(irUnit);
101 unit = std::move(irUnit.javaScriptCompilationUnit);
102 }
103
104 if (m_typeLoader->writeCacheFile()) {
105 QString errorString;
106 if (unit->saveToDisk(url(), &errorString)) {
107 QString error;
108 if (!unit->loadFromDisk(url(), data.sourceTimeStamp(), &error)) {
109 // ignore error, keep using the in-memory compilation unit.
110 }
111 } else {
112 qCDebug(DBG_DISK_CACHE()) << "Error saving cached version of"
113 << unit->fileName() << "to disk:" << errorString;
114 }
115 }
116
117 initializeFromCompilationUnit(std::move(unit));
118}
119
120void QQmlScriptBlob::initializeFromCachedUnit(const QQmlPrivate::CachedQmlUnit *cachedUnit)
121{
122 assertTypeLoaderThread();
123 initializeFromCompilationUnit(QQml::makeRefPointer<QV4::CompiledData::CompilationUnit>(
124 cachedUnit->qmlData, cachedUnit->aotCompiledFunctions,
125 cachedUnit->validateLookupSignatures, urlString(), finalUrlString()));
126}
127
128void QQmlScriptBlob::done()
129{
130 assertTypeLoaderThread();
131
132 if (isError())
133 return;
134
135 // Check all script dependencies for errors
136 for (int ii = 0; ii < m_scripts.size(); ++ii) {
137 const ScriptReference &script = m_scripts.at(ii);
138 // We would like to assert on isCompleteOrError() here, but since ECMAScript dependencies
139 // can be cyclic, we need to live with certain scripts formally not being complete.
140 // However, since we only omit the dependency if the compilation unit already exists when
141 // loading, we will still catch all errors here.
142 if (script.script->isError()) {
143 QList<QQmlError> errors = script.script->errors();
144 QQmlError error;
145 error.setUrl(url());
146 error.setLine(qmlConvertSourceCoordinate<quint32, int>(script.location.line()));
147 error.setColumn(qmlConvertSourceCoordinate<quint32, int>(script.location.column()));
148 error.setDescription(QQmlTypeLoader::tr("Script %1 unavailable").arg(script.script->urlString()));
149 errors.prepend(error);
150 setError(errors);
151 return;
152 }
153 }
154
155 if (!m_isModule) {
156 m_scriptData->typeNameCache.adopt(new QQmlTypeNameCache(m_importCache));
157
158 QSet<QString> ns;
159
160 for (int scriptIndex = 0; scriptIndex < m_scripts.size(); ++scriptIndex) {
161 const ScriptReference &script = m_scripts.at(scriptIndex);
162
163 m_scriptData->scripts.append(script.script->scriptData());
164
165 if (!script.nameSpace.isNull()) {
166 if (!ns.contains(script.nameSpace)) {
167 ns.insert(script.nameSpace);
168 m_scriptData->typeNameCache->add(script.nameSpace);
169 }
170 }
171 m_scriptData->typeNameCache->add(script.qualifier, scriptIndex, script.nameSpace);
172 }
173
174 m_importCache->populateCache(m_scriptData->typeNameCache.data());
175 }
176
177 if (auto cu = m_scriptData->compilationUnit()) {
178 cu->qmlType = QQmlMetaType::findCompositeType(url(), cu, QQmlMetaType::JavaScript);
179 for (const auto &script : std::as_const(m_scripts))
180 cu->dependentScripts.append(script.script->scriptData());
181 QQmlMetaType::registerInternalCompositeType(cu);
182 }
183
184 m_scripts.clear();
185}
186
187QString QQmlScriptBlob::stringAt(int index) const
188{
189 return m_scriptData->m_precompiledScript->stringAt(index);
190}
191
192void QQmlScriptBlob::scriptImported(const QQmlRefPointer<QQmlScriptBlob> &blob, const QV4::CompiledData::Location &location, const QString &qualifier, const QString &nameSpace)
193{
194 assertTypeLoaderThread();
195
196 ScriptReference ref;
197 ref.script = blob;
198 ref.location = location;
199 ref.qualifier = qualifier;
200 ref.nameSpace = nameSpace;
201
202 m_scripts << ref;
203}
204
205void QQmlScriptBlob::initializeFromCompilationUnit(
206 QQmlRefPointer<QV4::CompiledData::CompilationUnit> &&unit)
207{
208 assertTypeLoaderThread();
209 Q_ASSERT(!m_scriptData);
210 Q_ASSERT(unit);
211
212 m_scriptData.adopt(new QQmlScriptData());
213 m_scriptData->url = finalUrl();
214 m_scriptData->urlString = finalUrlString();
215 m_scriptData->m_precompiledScript = unit;
216
217 m_importCache->setBaseUrl(finalUrl(), finalUrlString());
218
219 if (!m_isModule) {
220 QList<QQmlError> errors;
221 for (quint32 i = 0, count = unit->importCount(); i < count; ++i) {
222 const QV4::CompiledData::Import *import = unit->importAt(i);
223 if (!addImport(import, {}, &errors)) {
224 Q_ASSERT(errors.size());
225 QQmlError error(errors.takeFirst());
226 error.setUrl(m_importCache->baseUrl());
227 error.setLine(import->location.line());
228 error.setColumn(import->location.column());
229 errors.prepend(error); // put it back on the list after filling out information.
230 setError(errors);
231 return;
232 }
233 }
234 }
235
236 const QStringList moduleRequests = unit->moduleRequests();
237 for (const QString &request: moduleRequests) {
238 const QUrl relativeRequest(request);
239 QUrl absoluteRequest = unit->finalUrl().resolved(relativeRequest);
240 if (!request.endsWith(QLatin1String(".mjs")))
241 absoluteRequest.setFragment(QLatin1String("module"));
242 QQmlRefPointer<QQmlScriptBlob> absoluteBlob
243 = typeLoader()->getScript(absoluteRequest, relativeRequest);
244 if (!absoluteBlob->m_scriptData || !absoluteBlob->m_scriptData->m_precompiledScript)
245 addDependency(absoluteBlob.data());
246 scriptImported(
247 absoluteBlob, /* ### */QV4::CompiledData::Location(), /*qualifier*/QString(),
248 /*namespace*/QString());
249 }
250}
251
252QT_END_NAMESPACE
Q_LOGGING_CATEGORY(lcEventDispatcher, "qt.eventdispatcher")