wayland: add some comments

This commit is contained in:
Victor Berger 2017-03-10 23:56:31 +01:00
parent 3ff9eb08e8
commit d3356763dc
4 changed files with 32 additions and 7 deletions

View file

@ -47,7 +47,7 @@ pub enum Window2 {
#[doc(hidden)] #[doc(hidden)]
X(x11::Window2), X(x11::Window2),
#[doc(hidden)] #[doc(hidden)]
Wayland(wayland::Window2) Wayland(wayland::Window)
} }
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
@ -130,7 +130,7 @@ impl Window2 {
match *UNIX_BACKEND { match *UNIX_BACKEND {
UnixBackend::Wayland(ref ctxt) => { UnixBackend::Wayland(ref ctxt) => {
if let EventsLoop::Wayland(ref evlp) = *events_loop { if let EventsLoop::Wayland(ref evlp) = *events_loop {
wayland::Window2::new(evlp, ctxt.clone(), window).map(Window2::Wayland) wayland::Window::new(evlp, ctxt.clone(), window).map(Window2::Wayland)
} else { } else {
// It is not possible to instanciate an EventsLoop not matching its backend // It is not possible to instanciate an EventsLoop not matching its backend
unreachable!() unreachable!()

View file

@ -14,6 +14,15 @@ use super::wayland_window::DecoratedSurface;
use super::wayland_kbd::MappedKeyboard; use super::wayland_kbd::MappedKeyboard;
use super::keyboard::KbdHandler; use super::keyboard::KbdHandler;
/// This struct is used as a holder for the callback
/// during the dispatching of events.
///
/// The proper ay to use it is:
/// - set a callback in it (and retrieve the noop one it contains)
/// - dispatch the EventQueue
/// - put back the noop callback in it
///
/// Failure to do so is unsafe™
pub struct EventsLoopSink { pub struct EventsLoopSink {
callback: Box<FnMut(::Event)> callback: Box<FnMut(::Event)>
} }
@ -50,11 +59,17 @@ impl EventsLoopSink {
} }
pub struct EventsLoop { pub struct EventsLoop {
// the global wayland context
ctxt: Arc<WaylandContext>, ctxt: Arc<WaylandContext>,
// our EventQueue
evq: Arc<Mutex<EventQueue>>, evq: Arc<Mutex<EventQueue>>,
// ids of the DecoratedHandlers of the surfaces we know
decorated_ids: Mutex<Vec<(usize, Arc<wl_surface::WlSurface>)>>, decorated_ids: Mutex<Vec<(usize, Arc<wl_surface::WlSurface>)>>,
// our sink, receiver of callbacks, shared with some handlers
sink: Arc<Mutex<EventsLoopSink>>, sink: Arc<Mutex<EventsLoopSink>>,
// trigger interruption of the run
interrupted: AtomicBool, interrupted: AtomicBool,
// trigger cleanup of the dead surfaces
cleanup_needed: Arc<AtomicBool>, cleanup_needed: Arc<AtomicBool>,
hid: usize hid: usize
} }
@ -75,6 +90,7 @@ impl EventsLoop {
} }
} }
// some internals that Window needs access to
pub fn get_window_init(&self) -> (Arc<Mutex<EventQueue>>, Arc<AtomicBool>) { pub fn get_window_init(&self) -> (Arc<Mutex<EventQueue>>, Arc<AtomicBool>) {
(self.evq.clone(), self.cleanup_needed.clone()) (self.evq.clone(), self.cleanup_needed.clone())
} }
@ -183,15 +199,15 @@ impl EventsLoop {
|cb| Self::process_resize(&mut evq_guard, &ids_guard, cb) |cb| Self::process_resize(&mut evq_guard, &ids_guard, cb)
); );
self.ctxt.flush(); self.ctxt.flush();
}
// replace the old noop callback
unsafe { self.sink.lock().unwrap().set_callback(old_cb) };
if self.cleanup_needed.swap(false, ::std::sync::atomic::Ordering::Relaxed) { if self.cleanup_needed.swap(false, ::std::sync::atomic::Ordering::Relaxed) {
self.prune_dead_windows() self.prune_dead_windows()
} }
} }
// replace the old noop callback
unsafe { self.sink.lock().unwrap().set_callback(old_cb) };
}
} }
enum KbdType { enum KbdType {

View file

@ -1,6 +1,6 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd"))] #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "openbsd"))]
pub use self::window::{Window as Window2, WindowId}; pub use self::window::{Window, WindowId};
pub use self::event_loop::EventsLoop; pub use self::event_loop::EventsLoop;
pub use self::context::{WaylandContext, MonitorId, get_available_monitors, pub use self::context::{WaylandContext, MonitorId, get_available_monitors,
get_primary_monitor}; get_primary_monitor};

View file

@ -12,11 +12,17 @@ use super::wayland_window;
use super::wayland_window::DecoratedSurface; use super::wayland_window::DecoratedSurface;
pub struct Window { pub struct Window {
// the global wayland context
ctxt: Arc<WaylandContext>, ctxt: Arc<WaylandContext>,
// the EventQueue of our EventsLoop
evq: Arc<Mutex<EventQueue>>, evq: Arc<Mutex<EventQueue>>,
// signal to advertize the EventsLoop when we are destroyed
cleanup_signal: Arc<AtomicBool>, cleanup_signal: Arc<AtomicBool>,
// our wayland surface
surface: Arc<wl_surface::WlSurface>, surface: Arc<wl_surface::WlSurface>,
// our current inner dimensions
size: Mutex<(u32, u32)>, size: Mutex<(u32, u32)>,
// the id of our DecoratedHandler in the EventQueue
decorated_id: usize decorated_id: usize
} }
@ -41,10 +47,12 @@ impl Window {
let mut evq_guard = evq.lock().unwrap(); let mut evq_guard = evq.lock().unwrap();
let decorated_id = evq_guard.add_handler_with_init(decorated); let decorated_id = evq_guard.add_handler_with_init(decorated);
{ {
// initialize the DecoratedHandler
let mut state = evq_guard.state(); let mut state = evq_guard.state();
let decorated = state.get_mut_handler::<DecoratedSurface<DecoratedHandler>>(decorated_id); let decorated = state.get_mut_handler::<DecoratedSurface<DecoratedHandler>>(decorated_id);
*(decorated.handler()) = Some(DecoratedHandler::new()); *(decorated.handler()) = Some(DecoratedHandler::new());
// set fullscreen if necessary
if let Some(PlatformMonitorId::Wayland(ref monitor_id)) = attributes.monitor { if let Some(PlatformMonitorId::Wayland(ref monitor_id)) = attributes.monitor {
ctxt.with_output(monitor_id.clone(), |output| { ctxt.with_output(monitor_id.clone(), |output| {
decorated.set_fullscreen( decorated.set_fullscreen(
@ -70,6 +78,7 @@ impl Window {
decorated_id: decorated_id decorated_id: decorated_id
}; };
// register ourselves to the EventsLoop
evlp.register_window(me.decorated_id, me.surface.clone()); evlp.register_window(me.decorated_id, me.surface.clone());
Ok(me) Ok(me)