📄 qeventdispatcher_glib.cpp
字号:
/******************************************************************************** Copyright (C) 1992-2007 Trolltech ASA. All rights reserved.**** This file is part of the QtCore module of the Qt Toolkit.**** This file may be used under the terms of the GNU General Public** License version 2.0 as published by the Free Software Foundation** and appearing in the file LICENSE.GPL included in the packaging of** this file. Please review the following information to ensure GNU** General Public Licensing requirements will be met:** http://trolltech.com/products/qt/licenses/licensing/opensource/**** If you are unsure which license is appropriate for your use, please** review the following information:** http://trolltech.com/products/qt/licenses/licensing/licensingoverview** or contact the sales department at sales@trolltech.com.**** In addition, as a special exception, Trolltech gives you certain** additional rights. These rights are described in the Trolltech GPL** Exception version 1.0, which can be found at** http://www.trolltech.com/products/qt/gplexception/ and in the file** GPL_EXCEPTION.txt in this package.**** In addition, as a special exception, Trolltech, as the sole copyright** holder for Qt Designer, grants users of the Qt/Eclipse Integration** plug-in the right for the Qt/Eclipse Integration to link to** functionality provided by Qt Designer and its related libraries.**** Trolltech reserves all rights not expressly granted herein.**** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.******************************************************************************/#include "qeventdispatcher_glib_p.h"#include "qeventdispatcher_unix_p.h"#include <private/qthread_p.h>#include "qcoreapplication.h"#include "qsocketnotifier.h"#include <QtCore/qhash.h>#include <QtCore/qlist.h>#include <QtCore/qpair.h>#include <glib.h>struct GPollFDWithQSocketNotifier{ GPollFD pollfd; QSocketNotifier *socketNotifier;};struct GSocketNotifierSource{ GSource source; QList<GPollFDWithQSocketNotifier *> pollfds;};static gboolean socketNotifierSourcePrepare(GSource *, gint *timeout){ if (timeout) *timeout = -1; return false;}static gboolean socketNotifierSourceCheck(GSource *source){ GSocketNotifierSource *src = reinterpret_cast<GSocketNotifierSource *>(source); bool pending = false; for (int i = 0; !pending && i < src->pollfds.count(); ++i) { GPollFDWithQSocketNotifier *p = src->pollfds.at(i); if (p->pollfd.revents & G_IO_NVAL) { // disable the invalid socket notifier static const char *t[] = { "Read", "Write", "Exception" }; qWarning("QSocketNotifier: Invalid socket %d and type '%s', disabling...", p->pollfd.fd, t[int(p->socketNotifier->type())]); // ### note, modifies src->pollfds! p->socketNotifier->setEnabled(false); } pending = ((p->pollfd.revents & p->pollfd.events) != 0); } return pending;}static gboolean socketNotifierSourceDispatch(GSource *source, GSourceFunc, gpointer){ QEvent event(QEvent::SockAct); GSocketNotifierSource *src = reinterpret_cast<GSocketNotifierSource *>(source); for (int i = 0; i < src->pollfds.count(); ++i) { GPollFDWithQSocketNotifier *p = src->pollfds.at(i); if ((p->pollfd.revents & p->pollfd.events) != 0) QCoreApplication::sendEvent(p->socketNotifier, &event); } return true; // ??? don't remove, right?}static GSourceFuncs socketNotifierSourceFuncs = { socketNotifierSourcePrepare, socketNotifierSourceCheck, socketNotifierSourceDispatch, NULL, NULL, NULL};struct GTimerSource{ GSource source; QTimerInfoList timerList;};static gboolean timerSourcePrepare(GSource *source, gint *timeout){ gint dummy; if (!timeout) timeout = &dummy; GTimerSource *src = reinterpret_cast<GTimerSource *>(source); timeval tv = { 0l, 0l }; if (src->timerList.timerWait(tv)) *timeout = (tv.tv_sec * 1000) + (tv.tv_usec / 1000); else *timeout = -1; return false;}static gboolean timerSourceCheck(GSource *source){ GTimerSource *src = reinterpret_cast<GTimerSource *>(source); if (src->timerList.isEmpty()) return false; if (src->timerList.updateCurrentTime() < src->timerList.first()->timeout) return false; return true;}static gboolean timerSourceDispatch(GSource *source, GSourceFunc, gpointer){ (void) reinterpret_cast<GTimerSource *>(source)->timerList.activateTimers(); return true; // ??? don't remove, right again?}static GSourceFuncs timerSourceFuncs = { timerSourcePrepare, timerSourceCheck, timerSourceDispatch, NULL, NULL, NULL};struct GPostEventSource{ GSource source; GPollFD pollfd; int wakeUpPipe[2]; QEventLoop::ProcessEventsFlags flags, previousFlags;};static gboolean postEventSourcePrepare(GSource *s, gint *timeout){ QThreadData *data = QThreadData::current(); if (!data) return false; gint dummy; if (!timeout) timeout = &dummy; *timeout = data->canWait ? -1 : 0; GPostEventSource *source = reinterpret_cast<GPostEventSource *>(s); return (!data->canWait || (source->flags != source->previousFlags));}static gboolean postEventSourceCheck(GSource *source){ return (postEventSourcePrepare(source, 0) || reinterpret_cast<GPostEventSource *>(source)->pollfd.revents != 0);}static gboolean postEventSourceDispatch(GSource *s, GSourceFunc, gpointer){ GPostEventSource *source = reinterpret_cast<GPostEventSource *>(s); char c[16]; while (::read(source->wakeUpPipe[0], c, sizeof(c)) > 0) ; source->previousFlags = source->flags; QCoreApplication::sendPostedEvents(0, (source->flags & QEventLoop::DeferredDeletion) ? -1 : 0); return true; // i dunno, george...}static GSourceFuncs postEventSourceFuncs = { postEventSourcePrepare, postEventSourceCheck, postEventSourceDispatch, NULL, NULL, NULL};QEventDispatcherGlibPrivate::QEventDispatcherGlibPrivate(GMainContext *context) : mainContext(context){ if (qgetenv("QT_NO_THREADED_GLIB").isEmpty()) { if (!g_thread_supported()) g_thread_init(NULL); } if (mainContext) { g_main_context_ref(mainContext); } else { QCoreApplication *app = QCoreApplication::instance(); if (app && QThread::currentThread() == app->thread()) { mainContext = g_main_context_default(); g_main_context_ref(mainContext); } else { mainContext = g_main_context_new(); } } postEventSource = reinterpret_cast<GPostEventSource *>(g_source_new(&postEventSourceFuncs, sizeof(GPostEventSource))); g_source_set_can_recurse(&postEventSource->source, true); pipe(postEventSource->wakeUpPipe); fcntl(postEventSource->wakeUpPipe[0], F_SETFD, FD_CLOEXEC); fcntl(postEventSource->wakeUpPipe[1], F_SETFD, FD_CLOEXEC); fcntl(postEventSource->wakeUpPipe[0], F_SETFL, fcntl(postEventSource->wakeUpPipe[0], F_GETFL) | O_NONBLOCK); fcntl(postEventSource->wakeUpPipe[1], F_SETFL, fcntl(postEventSource->wakeUpPipe[1], F_GETFL) | O_NONBLOCK); postEventSource->pollfd.fd = postEventSource->wakeUpPipe[0]; postEventSource->pollfd.events = G_IO_IN | G_IO_HUP | G_IO_ERR; postEventSource->flags = postEventSource->previousFlags = QEventLoop::AllEvents; g_source_add_poll(&postEventSource->source, &postEventSource->pollfd);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -