Skip to content

Commit

Permalink
Standardize facelift argument exchange
Browse files Browse the repository at this point in the history
Change-Id: Ifd09ce0e4b16a6a060d2a2634adc614f56b3a139
  • Loading branch information
😎 Mostafa Emami committed Sep 2, 2020
1 parent 6245cee commit 862cf2f
Show file tree
Hide file tree
Showing 52 changed files with 545 additions and 1,797 deletions.
10 changes: 10 additions & 0 deletions codegen/facelift/templates/IPCAdapter.template.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,12 @@
#include "InterfaceManager.h"

#ifdef DBUS_IPC_ENABLED
#include <QtDBus>
#include "{{module.fullyQualifiedPath}}/{{interfaceName}}IPCDBusAdapter.h"
{% for struct in module.structs %}
#include "{{struct.fullyQualifiedPath}}.h"
{% endfor %}

{% for property in interface.referencedInterfaceTypes %}
#include "{{property.fullyQualifiedPath}}{% if generateAsyncProxy %}Async{% endif %}IPCDBusAdapter.h"
{% endfor %}
Expand Down Expand Up @@ -80,6 +85,11 @@ struct {{interfaceName}}IPCAdapter::Impl {
{{interfaceName}}IPCAdapter::{{interfaceName}}IPCAdapter(QObject* parent) :
BaseType(facelift::InterfaceManager::instance(), parent)
{
#ifdef DBUS_IPC_ENABLED
{% for struct in module.structs %}
qDBusRegisterMetaType<{{struct.fullyQualifiedCppType}}>();
{% endfor %}
#endif
}

{{interfaceName}}IPCAdapter::~{{interfaceName}}IPCAdapter() {
Expand Down
13 changes: 0 additions & 13 deletions codegen/facelift/templates/IPCCommon.template.h
Original file line number Diff line number Diff line change
Expand Up @@ -47,25 +47,12 @@ class {{classExport}} {{interfaceName}}IPCCommon
{{operation.name}},
{% endfor %}
{% for property in interface.properties %}
{% if (not property.readonly) %}
set{{property.name}},
{% endif %}
{% if (property.type.is_model) %}
{{property.name}}, // model
{% endif %}
{% endfor %}
};

enum class SignalID {
invalid = static_cast<int>(facelift::CommonSignalID::firstSpecific),
{% for signal in interface.signals %}
{{signal.name}},
{% endfor %}
{% for property in interface.properties %}
{{property.name}},
{% endfor %}
};

};

{{module.namespaceCppClose}}
144 changes: 88 additions & 56 deletions codegen/facelift/templates/IPCProxyAdapter.template.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@
{% set className = interfaceName + proxyTypeNameSuffix %}

#include "{{className}}.h"
#include "FaceliftEnum.h"

{{module.namespaceCppOpen}}

Expand All @@ -55,76 +56,108 @@
{% endif %}
}

void {{className}}::deserializePropertyValues(InputIPCMessage &msg, bool isCompleteSnapshot)
void {{className}}::unmarshalPropertyValues(InputIPCMessage &msg)
{
{% for property in interface.properties %}
{% if property.type.is_interface %}
bool emit_{{property.name}}ChangeSignal = false;
QString {{property.name}}_objectPath;
if (deserializeOptionalValue(msg, {{property.name}}_objectPath, isCompleteSnapshot))
{
m_{{property.name}}Proxy.update({{property.name}}_objectPath);
m_{{property.name}} = m_{{property.name}}Proxy.getValue();
emit_{{property.name}}ChangeSignal = true;
QListIterator<QVariant> argumentsIterator(msg.arguments());
if (argumentsIterator.hasNext()) {
QMap<QString, QDBusVariant> values = qdbus_cast<QMap<QString, QDBusVariant>>(argumentsIterator.next());
for (const QString &propertyName: values.keys()) {
{% for property in interface.properties %}
{% if property.type.is_interface %}
if (propertyName == QStringLiteral("{{property.name}}")) {
bool emit_{{property.name}}ChangeSignal = false;
QString {{property.name}}_objectPath = qvariant_cast<QString>(values[propertyName].variant());
m_{{property.name}}Proxy.update({{property.name}}_objectPath);
m_{{property.name}} = m_{{property.name}}Proxy.getValue();
emit_{{property.name}}ChangeSignal = true;
if (emit_{{property.name}}ChangeSignal)
emit {{property.name}}Changed();
}
{% elif property.type.is_model %}
if (propertyName == QStringLiteral("{{property.name}}")) {
bool emit_{{property.name}}ChangeSignal = false;
int {{property.name}}Size = qvariant_cast<int>(values[propertyName].variant());
m_{{property.name}}.beginResetModel();
m_{{property.name}}.reset({{property.name}}Size, std::bind(&ThisType::{{property.name}}Data, this, std::placeholders::_1));
m_{{property.name}}.endResetModel();
emit_{{property.name}}ChangeSignal = true;
if (emit_{{property.name}}ChangeSignal)
emit {{property.name}}Changed();
}
{% else %}
if (propertyName == QStringLiteral("{{property.name}}")) {
const auto previous_{{property.name}}_Value = m_{{property.name}};
m_{{property.name}} = qvariant_cast<{{property.interfaceCppType}}>(values[propertyName].variant());
bool emit_{{property.name}}ChangeSignal = ((previous_{{property.name}}_Value != m_{{property.name}}));
if (emit_{{property.name}}ChangeSignal)
emit {{property.name}}Changed();
}
{% endif %}
{% endfor %}
if (propertyName == QStringLiteral("ready")) {
bool previousIsReady = this->ready();
m_serviceReady = qvariant_cast<bool>(values[propertyName].variant());
bool emit_ReadyChangeSignal = (previousIsReady != m_serviceReady);
if (emit_ReadyChangeSignal)
emit readyChanged();
}
}
}
{% elif property.type.is_model %}
bool emit_{{property.name}}ChangeSignal = false;
if (isCompleteSnapshot) {
int {{property.name}}Size;
deserializeValue(msg, {{property.name}}Size);
m_{{property.name}}.beginResetModel();
m_{{property.name}}.reset({{property.name}}Size, std::bind(&ThisType::{{property.name}}Data, this, std::placeholders::_1));
m_{{property.name}}.endResetModel();
emit_{{property.name}}ChangeSignal = true;
}
{% else %}
const auto previous_{{property.name}}_Value = m_{{property.name}};
deserializeOptionalValue(msg, m_{{property.name}}, isCompleteSnapshot);
bool emit_{{property.name}}ChangeSignal = isCompleteSnapshot && ((previous_{{property.name}}_Value != m_{{property.name}}));
{% endif %}
{% endfor %}

bool emit_ReadyChangeSignal = deserializeReadyValue(msg, isCompleteSnapshot) && isCompleteSnapshot;
}

{% for property in interface.properties %}
if (emit_{{property.name}}ChangeSignal)
emit {{property.name}}Changed();
QMap<const char*, const char*> {{className}}::getSignals()
{
static QMap<const char*, const char*> rawSignals{
{%- set comma = joiner(", ") -%}
{% for event in interface.signals %}
{{comma()}}{"{{event}}", SIGNAL({{event}}Raw(const InputIPCMessage& msg))}
{% endfor %}

if (emit_ReadyChangeSignal)
emit readyChanged();

};
return rawSignals;
}

void {{className}}::deserializeSignal(InputIPCMessage &msg)
void {{className}}::connectSignals()
{
SignalID member;
deserializeValue(msg, member);

{% for event in interface.signals %}
if (member == SignalID::{{event}})
{
connect(this, &{{className}}::{{event}}Raw, this, [this](const InputIPCMessage& msg) {
QListIterator<QVariant> argumentsIterator(msg.arguments());
{% for parameter in event.parameters %}
{{parameter.interfaceCppType}} param_{{parameter.name}};
deserializeValue(msg, param_{{parameter.name}});
param_{{parameter.name}} = (argumentsIterator.hasNext() ? qdbus_cast<{{parameter.interfaceCppType}}>(argumentsIterator.next()): {{parameter.interfaceCppType}}());
{% endfor %}
emit {{event}}(
{%- set comma = joiner(", ") -%}
{%- for parameter in event.parameters -%}
{{ comma() }}param_{{parameter.name}}
{%- endfor -%} );
} else
});
{% endfor %}
{% for property in interface.properties %}
if (member == SignalID::{{property.name}}) {
{% if property.type.is_model %}
m_{{property.name}}.handleSignal(msg);
{% else %}
emit {{property.name}}Changed();
{% endif %}
} else
}

void {{className}}::disconnectSignals()
{
{% for event in interface.signals %}
disconnect(this, &{{className}}::{{event}}Raw, this, nullptr);
{% endfor %}
BaseType::deserializeCommonSignal(static_cast<facelift::CommonSignalID>(member), this);
}

void {{className}}::unmarshalPropertiesChanged(InputIPCMessage &msg)
{
QListIterator<QVariant> argumentsIterator(msg.arguments());
QString interfaceName = (argumentsIterator.hasNext() ? qdbus_cast<QString>(argumentsIterator.next()): QString());
QVariantMap changedProperties = (argumentsIterator.hasNext() ? qdbus_cast<QVariantMap>(argumentsIterator.next()): QVariantMap());
for (const QString &propertyName: changedProperties.keys()) {
{% for property in interface.properties %}
{% if property.type.is_interface %}
{% elif property.type.is_model %}
{% else %}
if (propertyName == QStringLiteral("{{property.name}}")) {
m_{{property.name}} = qdbus_cast<{{property.cppType}}>(changedProperties[propertyName]);
emit {{property.name}}Changed(); // trust the propertiesChanged signal and emit without checking
}
{% endif %}
{% endfor %}
}
}

{% for property in interface.properties %}
Expand All @@ -134,7 +167,7 @@ void {{className}}::deserializeSignal(InputIPCMessage &msg)
void {{className}}::set{{property}}({{property.cppMethodArgumentType}} newValue)
{
{% if (not property.type.is_interface) %}
ipc()->sendSetterCall(memberID(MethodID::set{{property.name}}, "set{{property.name}}"), newValue);
ipc()->sendSetterCall("{{property.name}}", newValue);
{% else %}
Q_ASSERT(false); // Writable interface properties are unsupported
{% endif %}
Expand Down Expand Up @@ -168,12 +201,11 @@ void {{className}}::{{operation.name}}(
{%- endfor -%} ){% if operation.is_const %} const{% endif %}
{
{% if (operation.hasReturnValue) %}
{{operation.interfaceCppType}} returnValue;
ipc()->sendMethodCallWithReturn(memberID(MethodID::{{operation.name}}, "{{operation.name}}"), returnValue
QList<QVariant> args = ipc()->sendMethodCallWithReturn(memberID(MethodID::{{operation.name}}, "{{operation.name}}")
{%- for parameter in operation.parameters -%}
, {{parameter.name}}
{%- endfor -%} );
return returnValue;
return (!args.isEmpty() ? qdbus_cast<{{operation.interfaceCppType}}>(args[0]): {{operation.interfaceCppType}}());
{% else %}
ipc()->sendMethodCall(memberID(MethodID::{{operation.name}}, "{{operation.name}}")
{%- for parameter in operation.parameters -%}
Expand Down
14 changes: 10 additions & 4 deletions codegen/facelift/templates/IPCProxyAdapter.template.h
Original file line number Diff line number Diff line change
Expand Up @@ -60,15 +60,14 @@ class {{classExport}} {{className}} : public {{baseClass}}

using ThisType = {{className}};
using BaseType = {{baseClass}};
using SignalID = {{interface}}IPCCommon::SignalID;
using MethodID = {{interface}}IPCCommon::MethodID;

// override the default QMLAdapter type to add the IPC related properties
using QMLAdapterType = {{className}}QMLAdapterType;

{{className}}(QObject *parent = nullptr);

void deserializePropertyValues(InputIPCMessage &msg, bool isCompleteSnapshot) override;
void unmarshalPropertyValues(InputIPCMessage &msg) override;

{% if interface.hasModelProperty %}
void setServiceRegistered(bool isRegistered) override
Expand All @@ -85,7 +84,11 @@ class {{classExport}} {{className}} : public {{baseClass}}

{% endif %}

void deserializeSignal(InputIPCMessage &msg) override;
QMap<const char*, const char*> getSignals() override;
void connectSignals() override;
void disconnectSignals() override;

void unmarshalPropertiesChanged(InputIPCMessage &msg) override;

{% for operation in interface.operations %}

Expand Down Expand Up @@ -122,7 +125,6 @@ class {{classExport}} {{className}} : public {{baseClass}}
}

{% elif property.type.is_list %}

const {{property.interfaceCppType}}& {{property}}() const override
{
return m_{{property.name}};
Expand Down Expand Up @@ -159,6 +161,10 @@ class {{classExport}} {{className}} : public {{baseClass}}
facelift::IPCProxyModelProperty<ThisType, {{property.nestedType.interfaceCppType}}> m_{{property.name}};
{% endif %}
{% endfor %}

{% for event in interface.signals %}
Q_SIGNAL void {{event}}Raw(const InputIPCMessage& msg);
{% endfor %}
};


Expand Down
Loading

0 comments on commit 862cf2f

Please sign in to comment.