Commit cc40bead authored by Ronan Abhamon's avatar Ronan Abhamon

fix(app): avoid double free on shared singleton (qml/c++)

parent 9e8b6b59
...@@ -56,7 +56,7 @@ if(NOT WIN32) ...@@ -56,7 +56,7 @@ if(NOT WIN32)
endif() endif()
endif() endif()
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CUSTOM_FLAGS}") set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${CUSTOM_FLAGS}")
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG}") set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DQT_QML_DEBUG -DQT_DECLARATIVE_DEBUG")
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
# Define packages, libs, sources, headers, resources and languages. # Define packages, libs, sources, headers, resources and languages.
......
...@@ -99,9 +99,6 @@ void App::initContentApp () { ...@@ -99,9 +99,6 @@ void App::initContentApp () {
CoreManager::init(this, m_parser.value("config")); CoreManager::init(this, m_parser.value("config"));
qInfo() << "Activated selectors:" << QQmlFileSelector::get(&m_engine)->selector()->allSelectors(); qInfo() << "Activated selectors:" << QQmlFileSelector::get(&m_engine)->selector()->allSelectors();
// Avoid double free.
m_engine.setObjectOwnership(this, QQmlEngine::CppOwnership);
// Provide `+custom` folders for custom components. // Provide `+custom` folders for custom components.
{ {
QQmlFileSelector *file_selector = new QQmlFileSelector(&m_engine); QQmlFileSelector *file_selector = new QQmlFileSelector(&m_engine);
...@@ -243,8 +240,9 @@ QQuickWindow *App::getCallsWindow () { ...@@ -243,8 +240,9 @@ QQuickWindow *App::getCallsWindow () {
} }
QQuickWindow *App::getMainWindow () const { QQuickWindow *App::getMainWindow () const {
QQmlApplicationEngine &engine = const_cast<QQmlApplicationEngine &>(m_engine); return qobject_cast<QQuickWindow *>(
return qobject_cast<QQuickWindow *>(engine.rootObjects().at(0)); const_cast<QQmlApplicationEngine *>(&m_engine)->rootObjects().at(0)
);
} }
QQuickWindow *App::getSettingsWindow () { QQuickWindow *App::getSettingsWindow () {
...@@ -273,9 +271,36 @@ bool App::hasFocus () const { ...@@ -273,9 +271,36 @@ bool App::hasFocus () const {
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
#define REGISTER_EXISTING_SINGLETON(TYPE, NAME, METHOD) qmlRegisterSingletonType<TYPE>( \
"Linphone", 1, 0, NAME, \
[](QQmlEngine *, QJSEngine *) -> QObject *{ \
QObject *object = METHOD(); \
QQmlEngine::setObjectOwnership(object, QQmlEngine::CppOwnership); \
return object; \
} \
)
template<class T>
void registerSingletonType (const char *name) {
qmlRegisterSingletonType<T>(
"Linphone", 1, 0, name,
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new T();
}
);
}
void App::registerTypes () { void App::registerTypes () {
qInfo() << "Registering types..."; qInfo() << "Registering types...";
qmlRegisterType<Camera>("Linphone", 1, 0, "Camera");
qmlRegisterType<ContactsListProxyModel>("Linphone", 1, 0, "ContactsListProxyModel");
qmlRegisterType<ChatModel>("Linphone", 1, 0, "ChatModel");
qmlRegisterType<ChatProxyModel>("Linphone", 1, 0, "ChatProxyModel");
qmlRegisterType<SmartSearchBarModel>("Linphone", 1, 0, "SmartSearchBarModel");
qRegisterMetaType<ChatModel::EntryType>("ChatModel::EntryType");
qmlRegisterUncreatableType<CallModel>( qmlRegisterUncreatableType<CallModel>(
"Linphone", 1, 0, "CallModel", "CallModel is uncreatable." "Linphone", 1, 0, "CallModel", "CallModel is uncreatable."
); );
...@@ -289,85 +314,21 @@ void App::registerTypes () { ...@@ -289,85 +314,21 @@ void App::registerTypes () {
"Linphone", 1, 0, "VcardModel", "VcardModel is uncreatable." "Linphone", 1, 0, "VcardModel", "VcardModel is uncreatable."
); );
qmlRegisterSingletonType<App>( registerSingletonType<AccountSettingsModel>("AccountSettingsModel");
"Linphone", 1, 0, "App", registerSingletonType<Presence>("Presence");
[](QQmlEngine *, QJSEngine *) -> QObject *{ registerSingletonType<PresenceStatusModel>("PresenceStatusModel");
return App::getInstance(); registerSingletonType<TimelineModel>("TimelineModel");
}
); REGISTER_EXISTING_SINGLETON(App, "App", App::getInstance);
REGISTER_EXISTING_SINGLETON(CoreManager, "CoreManager", CoreManager::getInstance);
qmlRegisterSingletonType<CoreManager>( REGISTER_EXISTING_SINGLETON(SettingsModel, "SettingsModel", CoreManager::getInstance()->getSettingsModel);
"Linphone", 1, 0, "CoreManager", REGISTER_EXISTING_SINGLETON(SipAddressesModel, "SipAddressesModel", CoreManager::getInstance()->getSipAddressesModel);
[](QQmlEngine *, QJSEngine *) -> QObject *{ REGISTER_EXISTING_SINGLETON(CallsListModel, "CallsListModel", CoreManager::getInstance()->getCallsListModel);
return CoreManager::getInstance(); REGISTER_EXISTING_SINGLETON(ContactsListModel, "ContactsListModel", CoreManager::getInstance()->getContactsListModel);
}
);
qmlRegisterSingletonType<AccountSettingsModel>(
"Linphone", 1, 0, "AccountSettingsModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new AccountSettingsModel();
}
);
qmlRegisterSingletonType<CallsListModel>(
"Linphone", 1, 0, "CallsListModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return CoreManager::getInstance()->getCallsListModel();
}
);
qmlRegisterSingletonType<ContactsListModel>(
"Linphone", 1, 0, "ContactsListModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return CoreManager::getInstance()->getContactsListModel();
}
);
qmlRegisterSingletonType<Presence>(
"Linphone", 1, 0, "Presence",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new Presence();
}
);
qmlRegisterSingletonType<PresenceStatusModel>(
"Linphone", 1, 0, "PresenceStatusModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new PresenceStatusModel();
}
);
qmlRegisterSingletonType<SettingsModel>(
"Linphone", 1, 0, "SettingsModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return CoreManager::getInstance()->getSettingsModel();
}
);
qmlRegisterSingletonType<SipAddressesModel>(
"Linphone", 1, 0, "SipAddressesModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return CoreManager::getInstance()->getSipAddressesModel();
}
);
qmlRegisterSingletonType<TimelineModel>(
"Linphone", 1, 0, "TimelineModel",
[](QQmlEngine *, QJSEngine *) -> QObject *{
return new TimelineModel();
}
);
qmlRegisterType<Camera>("Linphone", 1, 0, "Camera");
qmlRegisterType<ContactsListProxyModel>("Linphone", 1, 0, "ContactsListProxyModel");
qmlRegisterType<ChatModel>("Linphone", 1, 0, "ChatModel");
qmlRegisterType<ChatProxyModel>("Linphone", 1, 0, "ChatProxyModel");
qmlRegisterType<SmartSearchBarModel>("Linphone", 1, 0, "SmartSearchBarModel");
qRegisterMetaType<ChatModel::EntryType>("ChatModel::EntryType");
} }
#undef REGISTER_EXISTING_SINGLETON
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
void App::setTrayIcon () { void App::setTrayIcon () {
......
...@@ -89,15 +89,15 @@ private: ...@@ -89,15 +89,15 @@ private:
} }
QCommandLineParser m_parser; QCommandLineParser m_parser;
QVariantList m_available_locales;
QString m_locale;
QQmlApplicationEngine m_engine; QQmlApplicationEngine m_engine;
DefaultTranslator *m_translator = nullptr; DefaultTranslator *m_translator = nullptr;
Notifier *m_notifier = nullptr; Notifier *m_notifier = nullptr;
QVariantList m_available_locales;
QString m_locale;
QQuickWindow *m_calls_window = nullptr; QQuickWindow *m_calls_window = nullptr;
QQuickWindow *m_settings_window = nullptr; QQuickWindow *m_settings_window = nullptr;
}; };
......
...@@ -35,11 +35,11 @@ int main (int argc, char *argv[]) { ...@@ -35,11 +35,11 @@ int main (int argc, char *argv[]) {
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// Options to get a nice video render. // Options to get a nice video render.
#ifdef _WIN32 #ifdef _WIN32
QCoreApplication::setAttribute(Qt::AA_UseOpenGLES, true); QCoreApplication::setAttribute(Qt::AA_UseOpenGLES, true);
#else #else
QCoreApplication::setAttribute(Qt::AA_UseDesktopOpenGL, true); QCoreApplication::setAttribute(Qt::AA_UseDesktopOpenGL, true);
#endif #endif // ifdef _WIN32
QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts, true); QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts, true);
{ {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment