source: vendor/synergy/1.3.1/lib/platform/CXWindowsEventQueueBuffer.cpp@ 2749

Last change on this file since 2749 was 2749, checked in by bird, 19 years ago

synergy v1.3.1 sources (zip).

File size: 4.5 KB
Line 
1/*
2 * synergy -- mouse and keyboard sharing utility
3 * Copyright (C) 2004 Chris Schoeneman
4 *
5 * This package is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License
7 * found in the file COPYING that should have accompanied this file.
8 *
9 * This package is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 */
14
15#include "CXWindowsEventQueueBuffer.h"
16#include "CLock.h"
17#include "CThread.h"
18#include "CEvent.h"
19#include "IEventQueue.h"
20#if HAVE_POLL
21# include <poll.h>
22#else
23# if HAVE_SYS_SELECT_H
24# include <sys/select.h>
25# endif
26# if HAVE_SYS_TIME_H
27# include <sys/time.h>
28# endif
29# if HAVE_SYS_TYPES_H
30# include <sys/types.h>
31# endif
32# if HAVE_UNISTD_H
33# include <unistd.h>
34# endif
35#endif
36
37//
38// CEventQueueTimer
39//
40
41class CEventQueueTimer { };
42
43
44//
45// CXWindowsEventQueueBuffer
46//
47
48CXWindowsEventQueueBuffer::CXWindowsEventQueueBuffer(
49 Display* display, Window window) :
50 m_display(display),
51 m_window(window),
52 m_waiting(false)
53{
54 assert(m_display != NULL);
55 assert(m_window != None);
56
57 m_userEvent = XInternAtom(m_display, "SYNERGY_USER_EVENT", False);
58}
59
60CXWindowsEventQueueBuffer::~CXWindowsEventQueueBuffer()
61{
62 // do nothing
63}
64
65void
66CXWindowsEventQueueBuffer::waitForEvent(double dtimeout)
67{
68 CThread::testCancel();
69
70 {
71 CLock lock(&m_mutex);
72 // we're now waiting for events
73 m_waiting = true;
74
75 // push out pending events
76 flush();
77 }
78
79 // use poll() to wait for a message from the X server or for timeout.
80 // this is a good deal more efficient than polling and sleeping.
81#if HAVE_POLL
82 struct pollfd pfds[1];
83 pfds[0].fd = ConnectionNumber(m_display);
84 pfds[0].events = POLLIN;
85 int timeout = (dtimeout < 0.0) ? -1 :
86 static_cast<int>(1000.0 * dtimeout);
87#else
88 struct timeval timeout;
89 struct timeval* timeoutPtr;
90 if (dtimeout < 0.0) {
91 timeoutPtr = NULL;
92 }
93 else {
94 timeout.tv_sec = static_cast<int>(dtimeout);
95 timeout.tv_usec = static_cast<int>(1.0e+6 *
96 (dtimeout - timeout.tv_sec));
97 timeoutPtr = &timeout;
98 }
99
100 // initialize file descriptor sets
101 fd_set rfds;
102 FD_ZERO(&rfds);
103 FD_SET(ConnectionNumber(m_display), &rfds);
104#endif
105
106 // wait for message from X server or for timeout. also check
107 // if the thread has been cancelled. poll() should return -1
108 // with EINTR when the thread is cancelled.
109#if HAVE_POLL
110 poll(pfds, 1, timeout);
111#else
112 select(ConnectionNumber(m_display) + 1,
113 SELECT_TYPE_ARG234 &rfds,
114 SELECT_TYPE_ARG234 NULL,
115 SELECT_TYPE_ARG234 NULL,
116 SELECT_TYPE_ARG5 timeoutPtr);
117#endif
118
119 {
120 // we're no longer waiting for events
121 CLock lock(&m_mutex);
122 m_waiting = false;
123 }
124
125 CThread::testCancel();
126}
127
128IEventQueueBuffer::Type
129CXWindowsEventQueueBuffer::getEvent(CEvent& event, UInt32& dataID)
130{
131 CLock lock(&m_mutex);
132
133 // push out pending events
134 flush();
135
136 // get next event
137 XNextEvent(m_display, &m_event);
138
139 // process event
140 if (m_event.xany.type == ClientMessage &&
141 m_event.xclient.message_type == m_userEvent) {
142 dataID = static_cast<UInt32>(m_event.xclient.data.l[0]);
143 return kUser;
144 }
145 else {
146 event = CEvent(CEvent::kSystem,
147 IEventQueue::getSystemTarget(), &m_event);
148 return kSystem;
149 }
150}
151
152bool
153CXWindowsEventQueueBuffer::addEvent(UInt32 dataID)
154{
155 // prepare a message
156 XEvent xevent;
157 xevent.xclient.type = ClientMessage;
158 xevent.xclient.window = m_window;
159 xevent.xclient.message_type = m_userEvent;
160 xevent.xclient.format = 32;
161 xevent.xclient.data.l[0] = static_cast<long>(dataID);
162
163 // save the message
164 CLock lock(&m_mutex);
165 m_postedEvents.push_back(xevent);
166
167 // if we're currently waiting for an event then send saved events to
168 // the X server now. if we're not waiting then some other thread
169 // might be using the display connection so we can't safely use it
170 // too.
171 if (m_waiting) {
172 flush();
173 }
174
175 return true;
176}
177
178bool
179CXWindowsEventQueueBuffer::isEmpty() const
180{
181 CLock lock(&m_mutex);
182 return (XPending(m_display) == 0);
183}
184
185CEventQueueTimer*
186CXWindowsEventQueueBuffer::newTimer(double, bool) const
187{
188 return new CEventQueueTimer;
189}
190
191void
192CXWindowsEventQueueBuffer::deleteTimer(CEventQueueTimer* timer) const
193{
194 delete timer;
195}
196
197void
198CXWindowsEventQueueBuffer::flush()
199{
200 // note -- m_mutex must be locked on entry
201
202 // flush the posted event list to the X server
203 for (size_t i = 0; i < m_postedEvents.size(); ++i) {
204 XSendEvent(m_display, m_window, False, 0, &m_postedEvents[i]);
205 }
206 XFlush(m_display);
207 m_postedEvents.clear();
208}
Note: See TracBrowser for help on using the repository browser.