1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
use std::io;
use std::os::unix::io::RawFd;
use std::sync::Arc;
use protocol::wl_display::WlDisplay;
use wayland_sys::client::*;
use {ConnectError, Proxy};
use super::EventQueueInner;
pub(crate) struct DisplayInner {
proxy: Proxy<WlDisplay>,
display: *mut wl_display,
}
unsafe impl Send for DisplayInner {}
unsafe impl Sync for DisplayInner {}
unsafe fn make_display(ptr: *mut wl_display) -> Result<(Arc<DisplayInner>, EventQueueInner), ConnectError> {
if ptr.is_null() {
return Err(ConnectError::NoCompositorListening);
}
let display = Arc::new(DisplayInner {
proxy: Proxy::from_c_ptr(ptr as *mut _),
display: ptr,
});
let evq = EventQueueInner::new(display.clone(), None);
Ok((display, evq))
}
impl DisplayInner {
pub unsafe fn from_fd(fd: RawFd) -> Result<(Arc<DisplayInner>, EventQueueInner), ConnectError> {
if !::wayland_sys::client::is_lib_available() {
return Err(ConnectError::NoWaylandLib);
}
let display_ptr = ffi_dispatch!(WAYLAND_CLIENT_HANDLE, wl_display_connect_to_fd, fd);
make_display(display_ptr)
}
pub(crate) fn ptr(&self) -> *mut wl_display {
self.display
}
pub(crate) fn flush(&self) -> io::Result<()> {
let ret = unsafe { ffi_dispatch!(WAYLAND_CLIENT_HANDLE, wl_display_flush, self.ptr()) };
if ret >= 0 {
Ok(())
} else {
Err(io::Error::last_os_error())
}
}
pub(crate) fn create_event_queue(me: &Arc<DisplayInner>) -> EventQueueInner {
unsafe {
let ptr = ffi_dispatch!(WAYLAND_CLIENT_HANDLE, wl_display_create_queue, me.ptr());
EventQueueInner::new(me.clone(), Some(ptr))
}
}
pub(crate) fn get_proxy(&self) -> &Proxy<WlDisplay> {
&self.proxy
}
pub(crate) unsafe fn from_external(display_ptr: *mut wl_display) -> (Arc<DisplayInner>, EventQueueInner) {
let evq_ptr = ffi_dispatch!(WAYLAND_CLIENT_HANDLE, wl_display_create_queue, display_ptr);
let wrapper_ptr = ffi_dispatch!(
WAYLAND_CLIENT_HANDLE,
wl_proxy_create_wrapper,
display_ptr as *mut _
);
ffi_dispatch!(WAYLAND_CLIENT_HANDLE, wl_proxy_set_queue, wrapper_ptr, evq_ptr);
let display = Arc::new(DisplayInner {
proxy: Proxy::from_c_display_wrapper(wrapper_ptr),
display: display_ptr,
});
let evq = EventQueueInner::new(display.clone(), Some(evq_ptr));
(display, evq)
}
}
impl Drop for DisplayInner {
fn drop(&mut self) {
if self.proxy.c_ptr() == (self.display as *mut _) {
unsafe {
ffi_dispatch!(
WAYLAND_CLIENT_HANDLE,
wl_display_disconnect,
self.proxy.c_ptr() as *mut wl_display
);
}
}
}
}