Skip to main content

wayland_client/
conn.rs

1use std::{
2    env, fmt,
3    io::ErrorKind,
4    os::unix::io::{AsFd, BorrowedFd, FromRawFd, OwnedFd},
5    os::unix::net::UnixStream,
6    path::PathBuf,
7    sync::{
8        Arc,
9        atomic::{AtomicBool, Ordering},
10    },
11};
12
13use wayland_backend::{
14    client::{Backend, InvalidId, ObjectData, ObjectId, ReadEventsGuard, WaylandError},
15    protocol::{ObjectInfo, ProtocolError},
16};
17
18use crate::{EventQueue, Proxy, protocol::wl_display::WlDisplay};
19
20/// The Wayland connection
21///
22/// This is the main type representing your connection to the Wayland server, though most of the interaction
23/// with the protocol are actually done using other types. The two main uses a simple app has for the
24/// [`Connection`] are:
25///
26/// - Obtaining the initial [`WlDisplay`] through the [`display()`][Self::display()] method.
27/// - Creating new [`EventQueue`]s with the [`new_event_queue()`][Self::new_event_queue()] method.
28///
29/// It can be created through the [`connect_to_env()`][Self::connect_to_env()] method to follow the
30/// configuration from the environment (which is what you'll do most of the time), or using the
31/// [`from_socket()`][Self::from_socket()] method if you retrieved your connected Wayland socket through
32/// other means.
33///
34/// In case you need to plug yourself into an external Wayland connection that you don't control, you'll
35/// likely get access to it as a [`Backend`], in which case you can create a [`Connection`] from it using
36/// the [`from_backend()`][Self::from_backend()] method.
37#[derive(Debug, Clone, PartialEq, Eq)]
38pub struct Connection {
39    pub(crate) backend: Backend,
40}
41
42impl Connection {
43    /// Try to connect to the Wayland server following the environment
44    ///
45    /// This is the standard way to initialize a Wayland connection.
46    pub fn connect_to_env() -> Result<Self, ConnectError> {
47        let stream = if let Ok(txt) = env::var("WAYLAND_SOCKET") {
48            // We should connect to the provided WAYLAND_SOCKET
49            let fd = txt.parse::<i32>().map_err(|_| ConnectError::InvalidFd)?;
50            let fd = unsafe { OwnedFd::from_raw_fd(fd) };
51            // remove the variable so any child processes don't see it
52            // TODO: Audit that the environment access only happens in single-threaded code.
53            unsafe { env::remove_var("WAYLAND_SOCKET") };
54            // set the CLOEXEC flag on this FD
55            let flags = rustix::io::fcntl_getfd(&fd);
56            let result = flags
57                .map(|f| f | rustix::io::FdFlags::CLOEXEC)
58                .and_then(|f| rustix::io::fcntl_setfd(&fd, f));
59            match result {
60                Ok(_) => {
61                    // setting the O_CLOEXEC worked
62                    UnixStream::from(fd)
63                }
64                Err(_) => {
65                    // something went wrong in F_GETFD or F_SETFD
66                    return Err(ConnectError::InvalidFd);
67                }
68            }
69        } else {
70            let socket_name = env::var_os("WAYLAND_DISPLAY")
71                .map(Into::<PathBuf>::into)
72                .ok_or(ConnectError::NoCompositor)?;
73
74            let socket_path = if socket_name.is_absolute() {
75                socket_name
76            } else {
77                let mut socket_path = env::var_os("XDG_RUNTIME_DIR")
78                    .map(Into::<PathBuf>::into)
79                    .ok_or(ConnectError::NoCompositor)?;
80                if !socket_path.is_absolute() {
81                    return Err(ConnectError::NoCompositor);
82                }
83                socket_path.push(socket_name);
84                socket_path
85            };
86
87            UnixStream::connect(socket_path).map_err(|_| ConnectError::NoCompositor)?
88        };
89
90        let backend = Backend::connect(stream).map_err(|_| ConnectError::NoWaylandLib)?;
91        Ok(Self { backend })
92    }
93
94    /// Initialize a Wayland connection from an already existing Unix stream
95    pub fn from_socket(stream: UnixStream) -> Result<Self, ConnectError> {
96        let backend = Backend::connect(stream).map_err(|_| ConnectError::NoWaylandLib)?;
97        Ok(Self { backend })
98    }
99
100    /// Get the `WlDisplay` associated with this connection
101    pub fn display(&self) -> WlDisplay {
102        let display_id = self.backend.display_id();
103        Proxy::from_id(self, display_id).unwrap()
104    }
105
106    /// Create a new event queue
107    pub fn new_event_queue<State>(&self) -> EventQueue<State> {
108        EventQueue::new(self.clone())
109    }
110
111    /// Wrap an existing [`Backend`] into a [`Connection`]
112    pub fn from_backend(backend: Backend) -> Self {
113        Self { backend }
114    }
115
116    /// Get the [`Backend`] underlying this [`Connection`]
117    pub fn backend(&self) -> Backend {
118        self.backend.clone()
119    }
120
121    /// Flush pending outgoing events to the server
122    ///
123    /// This needs to be done regularly to ensure the server receives all your requests, though several
124    /// dispatching methods do it implicitly (this is stated in their documentation when they do).
125    pub fn flush(&self) -> Result<(), WaylandError> {
126        self.backend.flush()
127    }
128
129    /// Start a synchronized read from the socket
130    ///
131    /// This is needed if you plan to wait on readiness of the Wayland socket using an event loop. See
132    /// [`ReadEventsGuard`] for details. Once the events are received, you'll then need to dispatch them from
133    /// their event queues using [`EventQueue::dispatch_pending()`].
134    ///
135    /// If you don't need to manage multiple event sources, see
136    /// [`EventQueue::blocking_dispatch()`] for a simpler mechanism.
137    #[must_use]
138    pub fn prepare_read(&self) -> Option<ReadEventsGuard> {
139        self.backend.prepare_read()
140    }
141
142    /// Do a roundtrip to the server
143    ///
144    /// This method will block until the Wayland server has processed and answered all your
145    /// preceding requests. This is notably useful during the initial setup of an app, to wait for
146    /// the initial state from the server.
147    ///
148    /// See [`EventQueue::roundtrip()`] for a version that includes the dispatching of the event queue.
149    pub fn roundtrip(&self) -> Result<usize, WaylandError> {
150        let done = Arc::new(SyncData::default());
151        let display = self.display();
152        self.send_request(
153            &display,
154            crate::protocol::wl_display::Request::Sync {},
155            Some(done.clone()),
156        )
157        .map_err(|_| WaylandError::Io(rustix::io::Errno::PIPE.into()))?;
158
159        let mut dispatched = 0;
160
161        loop {
162            self.backend.flush()?;
163
164            if let Some(guard) = self.backend.prepare_read() {
165                dispatched += blocking_read(guard)?;
166            } else {
167                dispatched += self.backend.dispatch_inner_queue()?;
168            }
169
170            // see if the successful read included our callback
171            if done.done.load(Ordering::Relaxed) {
172                break;
173            }
174        }
175
176        Ok(dispatched)
177    }
178
179    /// Retrieve the protocol error that occured on the connection if any
180    ///
181    /// If this method returns [`Some`], it means your Wayland connection is already dead.
182    pub fn protocol_error(&self) -> Option<ProtocolError> {
183        match self.backend.last_error()? {
184            WaylandError::Protocol(err) => Some(err),
185            WaylandError::Io(_) => None,
186        }
187    }
188
189    /// Send a request associated with the provided object
190    ///
191    /// This is a low-level interface used by the code generated by `wayland-scanner`, you will likely
192    /// instead use the methods of the types representing each interface, or the [`Proxy::send_request()`] and
193    /// [`Proxy::send_constructor()`].
194    pub fn send_request<I: Proxy>(
195        &self,
196        proxy: &I,
197        request: I::Request<'_>,
198        data: Option<Arc<dyn ObjectData>>,
199    ) -> Result<ObjectId, InvalidId> {
200        let (msg, child_spec) = proxy.write_request(self, request)?;
201        self.backend.send_request(msg, data, child_spec)
202    }
203
204    /// Get the protocol information related to given object ID
205    pub fn object_info(&self, id: ObjectId) -> Result<ObjectInfo, InvalidId> {
206        self.backend.info(id)
207    }
208
209    /// Get the object data for a given object ID
210    ///
211    /// This is a low-level interface used by the code generated by `wayland-scanner`, a higher-level
212    /// interface for manipulating the user-data assocated to [`Dispatch`][crate::Dispatch] implementations
213    /// is given as [`Proxy::data()`]. Also see [`Proxy::object_data()`].
214    pub fn get_object_data(&self, id: ObjectId) -> Result<Arc<dyn ObjectData>, InvalidId> {
215        self.backend.get_data(id)
216    }
217
218    /// Set maximum buffer size.
219    #[cfg(feature = "libwayland_1_23")]
220    pub fn set_max_buffer_size(&self, max_buffer_size: Option<usize>) {
221        self.backend.set_max_buffer_size(max_buffer_size);
222    }
223}
224
225pub(crate) fn blocking_read(guard: ReadEventsGuard) -> Result<usize, WaylandError> {
226    let fd = guard.connection_fd();
227    let mut fds = [rustix::event::PollFd::new(
228        &fd,
229        rustix::event::PollFlags::IN | rustix::event::PollFlags::ERR,
230    )];
231
232    loop {
233        match rustix::event::poll(&mut fds, None) {
234            Ok(_) => break,
235            Err(rustix::io::Errno::INTR) => continue,
236            Err(e) => return Err(WaylandError::Io(e.into())),
237        }
238    }
239
240    // at this point the fd is ready
241    match guard.read() {
242        Ok(n) => Ok(n),
243        // if we are still "wouldblock", just return 0; the caller will retry.
244        Err(WaylandError::Io(e)) if e.kind() == ErrorKind::WouldBlock => Ok(0),
245        Err(e) => Err(e),
246    }
247}
248
249/// An error when trying to establish a Wayland connection.
250#[derive(Debug)]
251pub enum ConnectError {
252    /// The wayland library could not be loaded.
253    NoWaylandLib,
254
255    /// Could not find wayland compositor
256    NoCompositor,
257
258    /// `WAYLAND_SOCKET` was set but contained garbage
259    InvalidFd,
260}
261
262impl std::error::Error for ConnectError {}
263
264impl fmt::Display for ConnectError {
265    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
266        match self {
267            ConnectError::NoWaylandLib => {
268                write!(f, "The wayland library could not be loaded")
269            }
270            ConnectError::NoCompositor => {
271                write!(f, "Could not find wayland compositor")
272            }
273            ConnectError::InvalidFd => {
274                write!(f, "WAYLAND_SOCKET was set but contained garbage")
275            }
276        }
277    }
278}
279
280impl AsFd for Connection {
281    /// Provides fd from [`Backend::poll_fd()`] for polling.
282    fn as_fd(&self) -> BorrowedFd<'_> {
283        self.backend.poll_fd()
284    }
285}
286
287/*
288    wl_callback object data for wl_display.sync
289*/
290
291#[derive(Default)]
292pub(crate) struct SyncData {
293    pub(crate) done: AtomicBool,
294}
295
296impl ObjectData for SyncData {
297    fn event(
298        self: Arc<Self>,
299        _handle: &Backend,
300        _msg: wayland_backend::protocol::Message<ObjectId, OwnedFd>,
301    ) -> Option<Arc<dyn ObjectData>> {
302        self.done.store(true, Ordering::Relaxed);
303        None
304    }
305
306    fn destroyed(&self, _: ObjectId) {}
307}