# ---------------------------------------------------------------------------- # pyglet # Copyright (c) 2006-2008 Alex Holkner # Copyright (c) 2008-2021 pyglet contributors # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # # * Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # * Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in # the documentation and/or other materials provided with the # distribution. # * Neither the name of pyglet nor the names of its # contributors may be used to endorse or promote products # derived from this software without specific prior written # permission. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE # COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. # ---------------------------------------------------------------------------- import os import select import threading from pyglet import app from pyglet.app.base import PlatformEventLoop from pyglet.util import asbytes class XlibSelectDevice: def fileno(self): """Get the file handle for ``select()`` for this device. :rtype: int """ raise NotImplementedError('abstract') def select(self): """Perform event processing on the device. Called when ``select()`` returns this device in its list of active files. """ raise NotImplementedError('abstract') def poll(self): """Check if the device has events ready to process. :rtype: bool :return: True if there are events to process, False otherwise. """ return False class NotificationDevice(XlibSelectDevice): def __init__(self): self._sync_file_read, self._sync_file_write = os.pipe() self._event = threading.Event() def fileno(self): return self._sync_file_read def set(self): self._event.set() os.write(self._sync_file_write, asbytes('1')) def select(self): self._event.clear() os.read(self._sync_file_read, 1) app.platform_event_loop.dispatch_posted_events() def poll(self): return self._event.isSet() class XlibEventLoop(PlatformEventLoop): def __init__(self): super(XlibEventLoop, self).__init__() self._notification_device = NotificationDevice() self._select_devices = set() self._select_devices.add(self._notification_device) def notify(self): self._notification_device.set() def step(self, timeout=None): # Timeout is from EventLoop.idle(). Return after that timeout or directly # after receiving a new event. None means: block for user input. # Poll devices to check for already pending events (select.select is not enough) pending_devices = [] for device in self._select_devices: if device.poll(): pending_devices.append(device) # If no devices were ready, wait until one gets ready if not pending_devices: pending_devices, _, _ = select.select(self._select_devices, (), (), timeout) if not pending_devices: # Notify caller that timeout expired without incoming events return False # Dispatch activity on matching devices for device in pending_devices: device.select() # Notify caller that events were handled before timeout expired return True