diff options
Diffstat (limited to 'vcl/unx')
-rw-r--r-- | vcl/unx/kde4/KDE4FilePicker.cxx | 8 | ||||
-rw-r--r-- | vcl/unx/kde4/KDEData.cxx | 2 | ||||
-rw-r--r-- | vcl/unx/kde4/KDESalInstance.cxx | 10 | ||||
-rw-r--r-- | vcl/unx/kde4/KDEXLib.cxx | 30 | ||||
-rw-r--r-- | vcl/unx/kde4/KDEXLib.hxx | 2 | ||||
-rw-r--r-- | vcl/unx/kde4/tst_exclude_socket_notifiers.hxx | 126 |
6 files changed, 165 insertions, 13 deletions
diff --git a/vcl/unx/kde4/KDE4FilePicker.cxx b/vcl/unx/kde4/KDE4FilePicker.cxx index 467e8d871093..4f819fb4c0a1 100644 --- a/vcl/unx/kde4/KDE4FilePicker.cxx +++ b/vcl/unx/kde4/KDE4FilePicker.cxx @@ -261,8 +261,12 @@ sal_Int16 SAL_CALL KDE4FilePicker::execute() _dialog->filterWidget()->setEditable(false); // We're entering a nested loop. - // Release the yield mutex to prevent deadlocks. - int result = _dialog->exec(); + int result; + { + // Release the yield mutex to prevent deadlocks. + SalYieldMutexReleaser aReleaser; + result = _dialog->exec(); + } // HACK: KFileDialog uses KConfig("kdeglobals") for saving some settings // (such as the auto-extension flag), but that doesn't update KGlobal::config() diff --git a/vcl/unx/kde4/KDEData.cxx b/vcl/unx/kde4/KDEData.cxx index ccbbd9998320..0e87c6209cf6 100644 --- a/vcl/unx/kde4/KDEData.cxx +++ b/vcl/unx/kde4/KDEData.cxx @@ -27,6 +27,7 @@ #include "KDEData.hxx" #include "KDEXLib.hxx" +#include "KDESalDisplay.hxx" KDEData::~KDEData() @@ -37,6 +38,7 @@ void KDEData::Init() { pXLib_ = new KDEXLib(); pXLib_->Init(); + SetDisplay( SalKDEDisplay::self() ); } void KDEData::initNWF() diff --git a/vcl/unx/kde4/KDESalInstance.cxx b/vcl/unx/kde4/KDESalInstance.cxx index 9670172e216e..023d79051741 100644 --- a/vcl/unx/kde4/KDESalInstance.cxx +++ b/vcl/unx/kde4/KDESalInstance.cxx @@ -32,10 +32,14 @@ SalFrame* KDESalInstance::CreateFrame( SalFrame *pParent, sal_uLong nState ) } uno::Reference< ui::dialogs::XFilePicker2 > KDESalInstance::createFilePicker( - const uno::Reference< uno::XComponentContext >& xMSF ) + const uno::Reference< uno::XComponentContext >& xMSF ) { - return uno::Reference< ui::dialogs::XFilePicker2 >( - static_cast<KDEXLib*>( mpXLib )->createFilePicker(xMSF) ); + KDEXLib* kdeXLib = static_cast<KDEXLib*>( mpXLib ); + if (kdeXLib->haveQt4SocketExcludeFix()) + return uno::Reference< ui::dialogs::XFilePicker2 >( + kdeXLib->createFilePicker(xMSF) ); + else + return X11SalInstance::createFilePicker( xMSF ); } int KDESalInstance::getFrameWidth() diff --git a/vcl/unx/kde4/KDEXLib.cxx b/vcl/unx/kde4/KDEXLib.cxx index 5c4cd10f01e3..820d39aa661e 100644 --- a/vcl/unx/kde4/KDEXLib.cxx +++ b/vcl/unx/kde4/KDEXLib.cxx @@ -19,7 +19,6 @@ #include "VCLKDEApplication.hxx" -#include "KDE4FilePicker.hxx" #include "KDESalInstance.hxx" #include <kapplication.h> @@ -45,15 +44,17 @@ #include <config_kde4.h> +#if KDE_HAVE_GLIB +#include "KDE4FilePicker.hxx" +#include "tst_exclude_socket_notifiers.moc" +#endif + KDEXLib::KDEXLib() : SalXLib(), m_bStartupDone(false), m_pApplication(0), m_pFreeCmdLineArgs(0), m_pAppCmdLineArgs(0), m_nFakeCmdLineArgs( 0 ), - m_frameWidth( -1 ), m_isGlibEventLoopType(false) + m_frameWidth( -1 ), m_isGlibEventLoopType(false), + m_haveQt4SocketExcludeFix(false) { -#if KDE_HAVE_GLIB - m_isGlibEventLoopType = QAbstractEventDispatcher::instance()->inherits( "QEventDispatcherGlib" ); -#endif - // the timers created here means they belong to the main thread. // As the timeoutTimer runs the LO event queue, which may block on a dialog, // the timer has to use a Qt::QueuedConnection, otherwise the nested event @@ -168,6 +169,14 @@ void KDEXLib::Init() m_pApplication = new VCLKDEApplication(); kapp->disableSessionManagement(); KApplication::setQuitOnLastWindowClosed(false); + +#if KDE_HAVE_GLIB + m_isGlibEventLoopType = QAbstractEventDispatcher::instance()->inherits( "QEventDispatcherGlib" ); + if (m_isGlibEventLoopType && (0 == tst_processEventsExcludeSocket())) + // See http://bugreports.qt.nokia.com/browse/QTBUG-37380 + m_haveQt4SocketExcludeFix = true; +#endif + setupEventLoop(); Display* pDisp = QX11Info::display(); @@ -188,9 +197,8 @@ void KDEXLib::Init() #include <glib.h> static GPollFunc old_gpoll = NULL; -static gint gpoll_wrapper( GPollFD*, guint, gint ); -gint gpoll_wrapper( GPollFD* ufds, guint nfds, gint timeout ) +static gint gpoll_wrapper( GPollFD* ufds, guint nfds, gint timeout ) { SalYieldMutexReleaser release; // release YieldMutex (and re-acquire at block end) return old_gpoll( ufds, nfds, timeout ); @@ -215,6 +223,8 @@ void KDEXLib::setupEventLoop() { old_gpoll = g_main_context_get_poll_func( NULL ); g_main_context_set_poll_func( NULL, gpoll_wrapper ); + if( m_haveQt4SocketExcludeFix ) + m_pApplication->clipboard()->setProperty( "useEventLoopWhenWaiting", true ); return; } #endif @@ -367,11 +377,15 @@ using namespace com::sun::star; uno::Reference< ui::dialogs::XFilePicker2 > KDEXLib::createFilePicker( const uno::Reference< uno::XComponentContext >& xMSF ) { +#if KDE_HAVE_GLIB if( qApp->thread() != QThread::currentThread()) { SalYieldMutexReleaser aReleaser; return Q_EMIT createFilePickerSignal( xMSF ); } return uno::Reference< ui::dialogs::XFilePicker2 >( new KDE4FilePicker( xMSF ) ); +#else + return NULL; +#endif } #define Region QtXRegion diff --git a/vcl/unx/kde4/KDEXLib.hxx b/vcl/unx/kde4/KDEXLib.hxx index f26948d48e14..a88258c5985e 100644 --- a/vcl/unx/kde4/KDEXLib.hxx +++ b/vcl/unx/kde4/KDEXLib.hxx @@ -53,6 +53,7 @@ class KDEXLib : public QObject, public SalXLib QTimer userEventTimer; int m_frameWidth; bool m_isGlibEventLoopType; + bool m_haveQt4SocketExcludeFix; private: void setupEventLoop(); @@ -87,6 +88,7 @@ class KDEXLib : public QObject, public SalXLib virtual void PostUserEvent(); void doStartup(); + bool haveQt4SocketExcludeFix() { return m_haveQt4SocketExcludeFix; } public Q_SLOTS: com::sun::star::uno::Reference< com::sun::star::ui::dialogs::XFilePicker2 > diff --git a/vcl/unx/kde4/tst_exclude_socket_notifiers.hxx b/vcl/unx/kde4/tst_exclude_socket_notifiers.hxx new file mode 100644 index 000000000000..0c874fd7147d --- /dev/null +++ b/vcl/unx/kde4/tst_exclude_socket_notifiers.hxx @@ -0,0 +1,126 @@ +/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ +/* + * This file is part of the LibreOffice project. + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * This file incorporates work covered by the following license notice: + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed + * with this work for additional information regarding copyright + * ownership. The ASF licenses this file to you under the Apache + * License, Version 2.0 (the "License"); you may not use this file + * except in compliance with the License. You may obtain a copy of + * the License at http://www.apache.org/licenses/LICENSE-2.0 . + * + * This code is based on the SocketEventsTester from the Qt4 test suite. + */ + +#pragma once + +#include <qcoreapplication.h> +#include <qeventloop.h> +#include <qthread.h> +#include <qtimer.h> +#include <QtNetwork/qtcpserver.h> +#include <QtNetwork/qtcpsocket.h> + +class SocketEventsTester: public QObject +{ + Q_OBJECT +public: + SocketEventsTester() + { + socket = 0; + server = 0; + dataSent = false; + testResult = false; + dataArrived = false; + } + ~SocketEventsTester() + { + delete socket; + delete server; + } + bool init() + { + bool ret = false; + server = new QTcpServer(); + socket = new QTcpSocket(); + connect(server, SIGNAL(newConnection()), this, SLOT(sendHello())); + connect(socket, SIGNAL(readyRead()), this, SLOT(sendAck()), Qt::DirectConnection); + if((ret = server->listen(QHostAddress::LocalHost, 0))) { + socket->connectToHost(server->serverAddress(), server->serverPort()); + socket->waitForConnected(); + } + return ret; + } + + QTcpSocket *socket; + QTcpServer *server; + bool dataSent; + bool testResult; + bool dataArrived; +public slots: + void sendAck() + { + dataArrived = true; + } + void sendHello() + { + char data[10] ="HELLO"; + qint64 size = sizeof(data); + + QTcpSocket *serverSocket = server->nextPendingConnection(); + serverSocket->write(data, size); + dataSent = serverSocket->waitForBytesWritten(-1); + QEventLoop loop; + //allow the TCP/IP stack time to loopback the data, so our socket is ready to read + QTimer::singleShot(200, &loop, SLOT(quit())); + loop.exec(QEventLoop::ExcludeSocketNotifiers); + testResult = dataArrived; + //check the deferred event is processed + QTimer::singleShot(200, &loop, SLOT(quit())); + loop.exec(); + serverSocket->close(); + QThread::currentThread()->exit(0); + } +}; + +class SocketTestThread : public QThread +{ + Q_OBJECT +public: + SocketTestThread():QThread(0),testResult(false){}; + void run() + { + SocketEventsTester *tester = new SocketEventsTester(); + if (tester->init()) + exec(); + dataSent = tester->dataSent; + testResult = tester->testResult; + dataArrived = tester->dataArrived; + delete tester; + } + bool dataSent; + bool testResult; + bool dataArrived; +}; + +#define QVERIFY(a) \ + if (!a) return 1; + +static int tst_processEventsExcludeSocket() +{ + SocketTestThread thread; + thread.start(); + QVERIFY(thread.wait()); + QVERIFY(thread.dataSent); + QVERIFY(!thread.testResult); + QVERIFY(thread.dataArrived); + return 0; +} + |