3#ifndef QQMLIRBUILDER_P_H
4#define QQMLIRBUILDER_P_H
17#include <private/qqmljsast_p.h>
18#include <private/qqmljsengine_p.h>
19#include <private/qv4compiler_p.h>
20#include <private/qv4compileddata_p.h>
21#include <private/qqmljsmemorypool_p.h>
22#include <private/qqmljsfixedpoolarray_p.h>
23#include <private/qv4codegen_p.h>
24#include <private/qv4compiler_p.h>
26#include <QCoreApplication>
30class QQmlPropertyCache;
32class QQmlTypeNameCache;
69 template <
typename Sortable,
typename Base, Sortable Base::*sortMember>
72 T *insertPos =
nullptr;
74 for (T *it =
first; it; it = it->next) {
75 if (!(it->*sortMember <= item->*sortMember))
84 if (!insertionPoint) {
86 }
else if (insertionPoint ==
last) {
89 item->next = insertionPoint->next;
90 insertionPoint->next = item;
96 T *
const newNext = item->next;
99 before->next = newNext;
117 while (index > 0 && result) {
118 result = result->next;
206 template<
typename IdGenerator>
208 QV4::CompiledData::ParameterType *type,
const IdGenerator &idGenerator,
209 const QQmlJS::AST::Type *annotation)
211 using Flag = QV4::CompiledData::ParameterType::Flag;
214 return initType(type, QString(), idGenerator(QString()), Flag::NoFlag);
216 const QString typeId = annotation->typeId->toString();
217 const QString typeArgument =
218 annotation->typeArgument ? annotation->typeArgument->toString() : QString();
220 if (typeArgument.isEmpty())
221 return initType(type, typeId, idGenerator(typeId), Flag::NoFlag);
223 if (typeId == QLatin1String(
"list"))
224 return initType(type, typeArgument, idGenerator(typeArgument), Flag::List);
226 const QString annotationString = annotation->toString();
227 return initType(type, annotationString, idGenerator(annotationString), Flag::NoFlag);
233 static bool initType(
234 QV4::CompiledData::ParameterType *paramType,
const QString &typeName,
235 int typeNameIndex, QV4::CompiledData::ParameterType::Flag listFlag);
312 Q_DECLARE_TR_FUNCTIONS(Object)
314 quint32 inheritedTypeNameIndex;
363 void init(QQmlJS::MemoryPool *pool,
int typeNameIndex,
int idIndex,
const QV4::CompiledData::Location &location);
367 QString appendProperty(Property *prop,
const QString &propertyName,
bool isDefaultProperty,
const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation);
368 QString appendAlias(Alias *prop,
const QString &aliasName,
bool isDefaultProperty,
const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation);
386 bool hasFlag(QV4::CompiledData::Object::Flag flag)
const {
return flags & flag; }
391 friend struct ::QQmlIRLoader;
393 PoolList<Property> *properties;
394 PoolList<Alias> *aliases;
395 PoolList<Enum> *qmlEnums;
396 PoolList<Signal> *qmlSignals;
397 PoolList<Binding> *bindings;
398 PoolList<Function> *functions;
399 PoolList<InlineComponent> *inlineComponents;
400 PoolList<RequiredPropertyExtraData> *requiredPropertyExtraDatas;
409 ListPropertyAssignBehavior,
411 FunctionSignatureBehavior,
412 NativeMethodBehavior,
417 enum ListPropertyAssignBehaviorValue
424 enum ComponentBehaviorValue
430 enum FunctionSignatureBehaviorValue
436 enum NativeMethodBehaviorValue
442 enum ValueTypeBehaviorValue
448 Q_DECLARE_FLAGS(ValueTypeBehaviorValues, ValueTypeBehaviorValue);
453 ListPropertyAssignBehaviorValue listPropertyAssignBehavior;
454 ComponentBehaviorValue componentBehavior;
455 FunctionSignatureBehaviorValue functionSignatureBehavior;
456 NativeMethodBehaviorValue nativeMethodBehavior;
457 ValueTypeBehaviorValues::Int valueTypeBehavior;
458 uint translationContextIndex;
461 QV4::CompiledData::Location location;
466 Document(
const QString &fileName,
const QString &finalUrl,
bool debugMode);
468 QQmlJS::Engine jsParserEngine;
469 QV4::Compiler::Module jsModule;
470 QList<
const QV4::CompiledData::Import *> imports;
471 QList<Pragma*> pragmas;
472 QQmlJS::AST::UiProgram *program;
473 QVector<Object*> objects;
474 QV4::Compiler::JSUnitGenerator jsGenerator;
476 QQmlRefPointer<QV4::CompiledData::CompilationUnit> javaScriptCompilationUnit;
478 bool isSingleton()
const {
479 return std::any_of(pragmas.constBegin(), pragmas.constEnd(), [](
const Pragma *pragma) {
480 return pragma->type == Pragma::Singleton;
484 int registerString(
const QString &str) {
return jsGenerator.registerString(str); }
485 QString stringAt(
int index)
const {
return jsGenerator.stringForIndex(index); }
487 int objectCount()
const {
return objects.size();}
488 Object* objectAt(
int i)
const {
return objects.at(i);}
493 QmlIR::Document *document;
494 QQmlJS::Engine *engine;
495 QV4::Compiler::JSUnitGenerator *jsGenerator;
498 ScriptDirectivesCollector(QmlIR::Document *doc);
500 void pragmaLibrary() override;
501 void importFile(
const QString &jsfile,
const QString &module,
int lineNumber,
int column) override;
502 void importModule(
const QString &uri,
const QString &version,
const QString &module,
int lineNumber,
int column) override;
505struct Q_QML_COMPILER_EXPORT
IRBuilder :
public QQmlJS::AST::Visitor
507 Q_DECLARE_TR_FUNCTIONS(QQmlCodeGenerator)
510 bool generateFromQml(
const QString &code,
const QString &url, Document *output);
512 using QQmlJS::AST::Visitor::visit;
513 using QQmlJS::AST::Visitor::endVisit;
515 bool visit(QQmlJS::AST::UiArrayMemberList *ast) override;
516 bool visit(QQmlJS::AST::UiImport *ast) override;
517 bool visit(QQmlJS::AST::UiPragma *ast) override;
518 bool visit(QQmlJS::AST::UiHeaderItemList *ast) override;
519 bool visit(QQmlJS::AST::UiObjectInitializer *ast) override;
520 bool visit(QQmlJS::AST::UiObjectMemberList *ast) override;
521 bool visit(QQmlJS::AST::UiParameterList *ast) override;
522 bool visit(QQmlJS::AST::UiProgram *) override;
523 bool visit(QQmlJS::AST::UiQualifiedId *ast) override;
524 bool visit(QQmlJS::AST::UiArrayBinding *ast) override;
525 bool visit(QQmlJS::AST::UiObjectBinding *ast) override;
526 bool visit(QQmlJS::AST::UiObjectDefinition *ast) override;
527 bool visit(QQmlJS::AST::UiInlineComponent *ast) override;
528 bool visit(QQmlJS::AST::UiEnumDeclaration *ast) override;
529 bool visit(QQmlJS::AST::UiPublicMember *ast) override;
530 bool visit(QQmlJS::AST::UiScriptBinding *ast) override;
531 bool visit(QQmlJS::AST::UiSourceElement *ast) override;
532 bool visit(QQmlJS::AST::UiRequired *ast) override;
534 void throwRecursionDepthError() override
536 recordError(QQmlJS::SourceLocation(),
537 QStringLiteral(
"Maximum statement or expression depth exceeded"));
540 void accept(QQmlJS::AST::Node *node);
543 bool defineQMLObject(
544 int *objectIndex, QQmlJS::AST::UiQualifiedId *qualifiedTypeNameId,
545 const QV4::CompiledData::Location &location,
546 QQmlJS::AST::UiObjectInitializer *initializer, Object *declarationsOverride =
nullptr);
548 bool defineQMLObject(
549 int *objectIndex, QQmlJS::AST::UiObjectDefinition *node,
550 Object *declarationsOverride =
nullptr)
552 const QQmlJS::SourceLocation location = node->qualifiedTypeNameId->firstSourceLocation();
553 return defineQMLObject(
554 objectIndex, node->qualifiedTypeNameId,
555 { location.startLine, location.startColumn }, node->initializer,
556 declarationsOverride);
559 static QString asString(QQmlJS::AST::UiQualifiedId *node);
560 QStringView asStringRef(QQmlJS::AST::Node *node);
561 static QTypeRevision extractVersion(QStringView string);
562 QStringView textRefAt(
const QQmlJS::SourceLocation &loc)
const
563 {
return QStringView(sourceCode).mid(loc.offset, loc.length); }
564 QStringView textRefAt(
const QQmlJS::SourceLocation &first,
565 const QQmlJS::SourceLocation &last)
const;
567 void setBindingValue(QV4::CompiledData::Binding *binding, QQmlJS::AST::Statement *statement,
568 QQmlJS::AST::Node *parentNode);
569 void tryGeneratingTranslationBinding(QStringView base, QQmlJS::AST::ArgumentList *args, QV4::CompiledData::Binding *binding);
571 void appendBinding(QQmlJS::AST::UiQualifiedId *name, QQmlJS::AST::Statement *value,
572 QQmlJS::AST::Node *parentNode);
573 void appendBinding(QQmlJS::AST::UiQualifiedId *name,
int objectIndex,
bool isOnAssignment =
false);
574 void appendBinding(
const QQmlJS::SourceLocation &qualifiedNameLocation,
575 const QQmlJS::SourceLocation &nameLocation, quint32 propertyNameIndex,
576 QQmlJS::AST::Statement *value, QQmlJS::AST::Node *parentNode);
577 void appendBinding(
const QQmlJS::SourceLocation &qualifiedNameLocation,
578 const QQmlJS::SourceLocation &nameLocation, quint32 propertyNameIndex,
579 int objectIndex,
bool isListItem =
false,
bool isOnAssignment =
false);
581 bool appendAlias(QQmlJS::AST::UiPublicMember *node);
583 Object *bindingsTarget()
const;
585 bool setId(
const QQmlJS::SourceLocation &idLocation, QQmlJS::AST::Statement *value);
589 bool resolveQualifiedId(QQmlJS::AST::UiQualifiedId **nameToResolve, Object **object,
bool onAssignment =
false);
591 void recordError(
const QQmlJS::SourceLocation &location,
const QString &description);
593 quint32 registerString(
const QString &str)
const {
return jsGenerator->registerString(str); }
594 template <
typename _Tp> _Tp *New() {
return pool->New<_Tp>(); }
596 QString stringAt(
int index)
const {
return jsGenerator->stringForIndex(index); }
598 static bool isStatementNodeScript(QQmlJS::AST::Statement *statement);
599 static bool isRedundantNullInitializerForPropertyDeclaration(Property *property, QQmlJS::AST::Statement *statement);
601 QString sanityCheckFunctionNames(Object *obj, QQmlJS::SourceLocation *errorLocation);
603 QList<QQmlJS::DiagnosticMessage> errors;
605 QSet<QString> inlineComponentsNames;
607 QList<
const QV4::CompiledData::Import *> _imports;
608 QList<Pragma*> _pragmas;
609 QVector<Object*> _objects;
611 QV4::CompiledData::TypeReferenceMap _typeReferences;
614 Property *_propertyDeclaration;
616 QQmlJS::MemoryPool *pool;
618 QV4::Compiler::JSUnitGenerator *jsGenerator;
620 bool insideInlineComponent =
false;
625 void generate(Document &output,
const QV4::CompiledData::DependentTypesHasher &dependencyHasher = QV4::CompiledData::DependentTypesHasher());
628 typedef bool (Binding::*BindingFilter)()
const;
629 char *writeBindings(
char *bindingPtr,
const Object *o, BindingFilter filter)
const;
632struct Q_QML_COMPILER_EXPORT
JSCodeGen :
public QV4::Compiler::Codegen
634 JSCodeGen(Document *document,
635 QV4::Compiler::CodegenWarningInterface *iface =
636 QV4::Compiler::defaultCodegenWarningInterface(),
637 bool storeSourceLocations =
false);
641 generateJSCodeForFunctionsAndBindings(
const QList<CompiledFunctionOrExpression> &functions);
643 bool generateRuntimeFunctions(QmlIR::Object *object);
652
653
654
655
656
657
658
659
660
661
663 typename RegisterMainString,
664 typename RegisterCommentString,
665 typename RegisterContextString,
666 typename FinalizeTranslationData>
668 RegisterMainString registerMainString,
669 RegisterCommentString registerCommentString,
670 RegisterContextString registerContextString,
671 FinalizeTranslationData finalizeTranslationData
674 if (base == QLatin1String(
"qsTr")) {
675 QV4::CompiledData::TranslationData translationData;
676 translationData.number = -1;
679 translationData.commentIndex = 0;
682 translationData.contextIndex = QV4::CompiledData::TranslationData::NoContextIndex;
684 if (!args || !args->expression)
687 QStringView translation;
688 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
689 translation = arg1->value;
694 translationData.stringIndex = registerMainString(translation);
699 QQmlJS::AST::StringLiteral *arg2 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
702 translationData.commentIndex = registerCommentString(arg2->value);
706 if (QQmlJS::AST::NumericLiteral *arg3 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
707 translationData.number =
int(arg3->value);
718 finalizeTranslationData(QV4::CompiledData::Binding::Type_Translation, translationData);
719 }
else if (base == QLatin1String(
"qsTrId")) {
720 QV4::CompiledData::TranslationData translationData;
721 translationData.number = -1;
724 translationData.commentIndex = 0;
727 translationData.contextIndex = QV4::CompiledData::TranslationData::NoContextIndex;
729 if (!args || !args->expression)
733 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
738 translationData.stringIndex = registerMainString(id);
743 if (QQmlJS::AST::NumericLiteral *arg3 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
744 translationData.number =
int(arg3->value);
754 finalizeTranslationData(QV4::CompiledData::Binding::Type_TranslationById, translationData);
755 }
else if (base == QLatin1String(
"QT_TR_NOOP") || base == QLatin1String(
"QT_TRID_NOOP")) {
756 if (!args || !args->expression)
760 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
768 if (args && base == QLatin1String(
"QT_TR_NOOP")) {
770 if (QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression))
779 QV4::CompiledData::TranslationData translationData;
780 translationData.number = registerMainString(str);
781 finalizeTranslationData(QV4::CompiledData::Binding::Type_String, translationData);
782 }
else if (base == QLatin1String(
"QT_TRANSLATE_NOOP")) {
783 if (!args || !args->expression)
787 if (!args || !args->expression)
791 if (QQmlJS::AST::StringLiteral *arg2 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
800 if (QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression))
809 QV4::CompiledData::TranslationData fakeTranslationData;
810 fakeTranslationData.number = registerMainString(str);
811 finalizeTranslationData(QV4::CompiledData::Binding::Type_String, fakeTranslationData);
812 }
else if (base == QLatin1String(
"qsTranslate")) {
813 QV4::CompiledData::TranslationData translationData;
814 translationData.number = -1;
815 translationData.commentIndex = 0;
817 if (!args || !args->next)
820 QStringView translation;
821 if (QQmlJS::AST::StringLiteral *arg1
822 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
823 translation = arg1->value;
828 translationData.contextIndex = registerContextString(translation);
833 QQmlJS::AST::StringLiteral *arg2
834 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
837 translationData.stringIndex = registerMainString(arg2->value);
841 QQmlJS::AST::StringLiteral *arg3
842 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
845 translationData.commentIndex = registerCommentString(arg3->value);
849 if (QQmlJS::AST::NumericLiteral *arg4
850 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
851 translationData.number =
int(arg4->value);
862 finalizeTranslationData(QV4::CompiledData::Binding::Type_Translation, translationData);
void tryGeneratingTranslationBindingBase(QStringView base, QQmlJS::AST::ArgumentList *args, RegisterMainString registerMainString, RegisterCommentString registerCommentString, RegisterContextString registerContextString, FinalizeTranslationData finalizeTranslationData)
static const quint32 emptyStringIndex
#define COMPILE_EXCEPTION(location, desc)
static QStringList astNodeToStringList(QQmlJS::AST::Node *node)
static bool run(IRBuilder *builder, QQmlJS::AST::UiPragma *node, Pragma *pragma)
CompiledFunctionOrExpression * next
QV4::CompiledData::Location location
PoolList< EnumValue >::Iterator enumValuesEnd() const
PoolList< EnumValue > * enumValues
int enumValueCount() const
PoolList< EnumValue >::Iterator enumValuesBegin() const
QV4::CompiledData::Location location
const Parameter * formalsBegin() const
QV4::CompiledData::ParameterType returnType
QQmlJS::FixedPoolArray< Parameter > formals
const Parameter * formalsEnd() const
bool hasFlag(QV4::CompiledData::Object::Flag flag) const
int indexOfDefaultPropertyOrAlias
PoolList< Property >::Iterator propertiesEnd() const
PoolList< Property >::Iterator propertiesBegin() const
int requiredPropertyExtraDataCount() const
QString appendEnum(Enum *enumeration)
PoolList< Signal >::Iterator signalsEnd() const
int inlineComponentCount() const
const quint32 * namedObjectsInComponentTable() const
void appendRequiredPropertyExtraData(RequiredPropertyExtraData *extraData)
PoolList< RequiredPropertyExtraData >::Iterator requiredPropertyExtraDataEnd() const
void insertSorted(Binding *b)
PoolList< Binding >::Iterator bindingsEnd() const
PoolList< Function >::Iterator functionsBegin() const
void simplifyRequiredProperties()
QString appendBinding(Binding *b, bool isListBinding)
QV4::CompiledData::Location location
void appendInlineComponent(InlineComponent *ic)
bool defaultPropertyIsAlias
Alias * firstAlias() const
PoolList< Function >::Iterator functionsEnd() const
PoolList< Alias >::Iterator aliasesBegin() const
PoolList< RequiredPropertyExtraData >::Iterator requiredPropertyExtraDataBegin() const
PoolList< Enum >::Iterator enumsEnd() const
QString appendSignal(Signal *signal)
Binding * firstBinding() const
const Property * firstProperty() const
QString bindingAsString(Document *doc, int scriptIndex) const
const Enum * firstEnum() const
void init(QQmlJS::MemoryPool *pool, int typeNameIndex, int idIndex, const QV4::CompiledData::Location &location)
Object * declarationsOverride
QString appendAlias(Alias *prop, const QString &aliasName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation)
int functionCount() const
QString appendProperty(Property *prop, const QString &propertyName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation)
PoolList< Enum >::Iterator enumsBegin() const
PoolList< CompiledFunctionOrExpression > * functionsAndExpressions
Binding * unlinkBinding(Binding *before, Binding *binding)
const RequiredPropertyExtraData * requiredPropertyExtraData() const
bool hasAliasAsDefaultProperty() const
PoolList< InlineComponent >::Iterator inlineComponentsEnd() const
PoolList< Signal >::Iterator signalsBegin() const
void appendFunction(QmlIR::Function *f)
const Function * firstFunction() const
PoolList< Binding >::Iterator bindingsBegin() const
PoolList< InlineComponent >::Iterator inlineComponentsBegin() const
const InlineComponent * inlineComponent() const
QQmlJS::FixedPoolArray< int > runtimeFunctionIndices
const Signal * firstSignal() const
QV4::CompiledData::Location locationOfIdProperty
QQmlJS::FixedPoolArray< quint32 > namedObjectsInComponent
Binding * findBinding(quint32 nameIndex) const
int propertyCount() const
int namedObjectsInComponentCount() const
PoolList< Alias >::Iterator aliasesEnd() const
static bool initType(QV4::CompiledData::ParameterType *type, const IdGenerator &idGenerator, const QQmlJS::AST::Type *annotation)
static QV4::CompiledData::CommonType stringToBuiltinType(const QString &typeName)
operator const T *() const
bool operator!=(const Iterator &rhs) const
const T * operator->() const
bool operator==(const Iterator &rhs) const
const T & operator*() const
T * findSortedInsertionPoint(T *item) const
T * unlink(T *before, T *item)
T * slowAt(int index) const
void insertAfter(T *insertionPoint, T *item)
int parameterCount() const
PoolList< Parameter > * parameters
PoolList< Parameter >::Iterator parametersEnd() const
QStringList parameterStringList(const QV4::Compiler::StringTableGenerator *stringPool) const
PoolList< Parameter >::Iterator parametersBegin() const
QV4::CompiledData::Location location