Monado OpenXR Runtime
ipc_server.h
Go to the documentation of this file.
1// Copyright 2020-2023, Collabora, Ltd.
2// SPDX-License-Identifier: BSL-1.0
3/*!
4 * @file
5 * @brief Common server side code.
6 * @author Pete Black <pblack@collabora.com>
7 * @author Jakob Bornecrantz <jakob@collabora.com>
8 * @author Rylie Pavlik <rylie.pavlik@collabora.com>
9 * @ingroup ipc_server
10 */
11
12#pragma once
13
14#include "xrt/xrt_compiler.h"
15#include "xrt/xrt_limits.h"
16#include "xrt/xrt_space.h"
17#include "xrt/xrt_system.h"
18
19#include "os/os_threading.h"
20
21#include "util/u_logging.h"
22
23#include "shared/ipc_protocol.h"
25
26#include <stdio.h>
27
28
29#ifdef __cplusplus
30extern "C" {
31#endif
32
33/*
34 *
35 * Logging
36 *
37 */
38
39#define IPC_TRACE(d, ...) U_LOG_IFL_T(d->log_level, __VA_ARGS__)
40#define IPC_DEBUG(d, ...) U_LOG_IFL_D(d->log_level, __VA_ARGS__)
41#define IPC_INFO(d, ...) U_LOG_IFL_I(d->log_level, __VA_ARGS__)
42#define IPC_WARN(d, ...) U_LOG_IFL_W(d->log_level, __VA_ARGS__)
43#define IPC_ERROR(d, ...) U_LOG_IFL_E(d->log_level, __VA_ARGS__)
44
45
46/*
47 *
48 * Structs
49 *
50 */
51
52#define IPC_MAX_CLIENT_SEMAPHORES 8
53#define IPC_MAX_CLIENT_SWAPCHAINS (XRT_MAX_LAYERS * 2)
54#define IPC_MAX_CLIENT_SPACES 128
55
56struct xrt_instance;
57struct xrt_compositor;
59
60
61/*!
62 * Information about a single swapchain.
63 *
64 * @ingroup ipc_server
65 */
67{
68 uint32_t width;
69 uint32_t height;
70 uint64_t format;
71 uint32_t image_count;
72
73 bool active;
74};
75
76/*!
77 * Holds the state for a single client.
78 *
79 * @ingroup ipc_server
80 */
82{
83 //! Link back to the main server.
85
86 //! Session for this client.
87 struct xrt_session *xs;
88
89 //! Compositor for this client.
91
92 //! Is the inputs and outputs active.
94
95 //! Number of swapchains in use by client
97
98 //! Ptrs to the swapchains
99 struct xrt_swapchain *xscs[IPC_MAX_CLIENT_SWAPCHAINS];
100
101 //! Data for the swapchains.
102 struct ipc_swapchain_data swapchain_data[IPC_MAX_CLIENT_SWAPCHAINS];
103
104 //! Number of compositor semaphores in use by client
106
107 //! Ptrs to the semaphores.
108 struct xrt_compositor_semaphore *xcsems[IPC_MAX_CLIENT_SEMAPHORES];
109
110 struct
111 {
112 uint32_t root;
113 uint32_t local;
114 uint32_t stage;
115 uint32_t unbounded;
116 } semantic_spaces;
117
118 //! Number of spaces.
119 uint32_t space_count;
120 //! Index of localspace in ipc client.
122 //! Index of localspace in space overseer.
124 //! Index of localfloorspace in ipc client.
126 //! Index of localfloorspace in space overseer.
128
129 //! Ptrs to the spaces.
130 struct xrt_space *xspcs[IPC_MAX_CLIENT_SPACES];
131
132 //! Which of the references spaces is the client using.
134
135 //! Which of the device features is the client using.
136 bool device_feature_used[XRT_DEVICE_FEATURE_MAX_ENUM];
137
138 //! Socket fd used for client comms
140
141 struct ipc_app_state client_state;
142
143 int server_thread_index;
144};
145
146enum ipc_thread_state
147{
148 IPC_THREAD_READY,
149 IPC_THREAD_STARTING,
150 IPC_THREAD_RUNNING,
151 IPC_THREAD_STOPPING,
152};
153
155{
156 struct os_thread thread;
157 volatile enum ipc_thread_state state;
158 volatile struct ipc_client_state ics;
159};
160
161
162/*!
163 *
164 */
166{
167 //! The actual device.
169
170 //! Is the IO suppressed for this device.
172};
173
174/*!
175 * Platform-specific mainloop object for the IPC server.
176 *
177 * Contents are essentially implementation details, but are listed in full here so they may be included by value in the
178 * main ipc_server struct.
179 *
180 * @see ipc_design
181 *
182 * @ingroup ipc_server
183 */
185{
186
187#if defined(XRT_OS_ANDROID) || defined(XRT_OS_LINUX) || defined(XRT_DOXYGEN)
188 //! For waiting on various events in the main thread.
190#endif
191
192#if defined(XRT_OS_ANDROID) || defined(XRT_DOXYGEN)
193 /*!
194 * @name Android Mainloop Members
195 * @{
196 */
197
198 //! File descriptor for the read end of our pipe for submitting new clients
200
201 /*!
202 * File descriptor for the write end of our pipe for submitting new clients
203 *
204 * Must hold client_push_mutex while writing.
205 */
207
208 /*!
209 * Mutex for being able to register oneself as a new client.
210 *
211 * Locked only by threads in `ipc_server_mainloop_add_fd()`.
212 *
213 * This must be locked first, and kept locked the entire time a client is attempting to register and wait for
214 * confirmation. It ensures no acknowledgements of acceptance are lost and moves the overhead of ensuring this
215 * to the client thread.
216 */
217 pthread_mutex_t client_push_mutex;
218
219
220 /*!
221 * The last client fd we accepted, to acknowledge client acceptance.
222 *
223 * Also used as a sentinel during shutdown.
224 *
225 * Must hold accept_mutex while writing.
226 */
228
229 /*!
230 * Condition variable for accepting clients.
231 *
232 * Signalled when @ref last_accepted_fd is updated.
233 *
234 * Associated with @ref accept_mutex
235 */
236 pthread_cond_t accept_cond;
237
238 /*!
239 * Mutex for accepting clients.
240 *
241 * Locked by both clients and server: that is, by threads in `ipc_server_mainloop_add_fd()` and in the
242 * server/compositor thread in an implementation function called from `ipc_server_mainloop_poll()`.
243 *
244 * Exists to operate in conjunction with @ref accept_cond - it exists to make sure that the client can be woken
245 * when the server accepts it.
246 */
247 pthread_mutex_t accept_mutex;
248
249
250 /*! @} */
251#define XRT_IPC_GOT_IMPL
252#endif
253
254#if (defined(XRT_OS_LINUX) && !defined(XRT_OS_ANDROID)) || defined(XRT_DOXYGEN)
255 /*!
256 * @name Desktop Linux Mainloop Members
257 * @{
258 */
259
260 //! Socket that we accept connections on.
262
263 //! Were we launched by socket activation, instead of explicitly?
265
266 //! The socket filename we bound to, if any.
268
269 /*! @} */
270
271#define XRT_IPC_GOT_IMPL
272#endif
273
274#if defined(XRT_OS_WINDOWS) || defined(XRT_DOXYGEN)
275 /*!
276 * @name Desktop Windows Mainloop Members
277 * @{
278 */
279
280 //! Named Pipe that we accept connections on.
282
283 //! Name of the Pipe that we accept connections on.
285
286 /*! @} */
287
288#define XRT_IPC_GOT_IMPL
289#endif
290
291#ifndef XRT_IPC_GOT_IMPL
292#error "Need port"
293#endif
294};
295
296/*!
297 * De-initialize the mainloop object.
298 * @public @memberof ipc_server_mainloop
299 */
300void
302
303/*!
304 * Initialize the mainloop object.
305 *
306 * @return <0 on error.
307 * @public @memberof ipc_server_mainloop
308 */
309int
311
312/*!
313 * @brief Poll the mainloop.
314 *
315 * Any errors are signalled by calling ipc_server_handle_failure()
316 * @public @memberof ipc_server_mainloop
317 */
318void
320
321/*!
322 * Main IPC object for the server.
323 *
324 * @ingroup ipc_server
325 */
327{
328 struct xrt_instance *xinst;
329
330 //! Handle for the current process, e.g. pidfile on linux
332
333 struct u_debug_gui *debug_gui;
334
335 //! The @ref xrt_iface level system.
337
338 //! System devices.
340
341 //! Space overseer.
343
344 //! System compositor.
346
349
350 struct ipc_shared_memory *ism;
351 xrt_shmem_handle_t ism_handle;
352
353 struct ipc_server_mainloop ml;
354
355 // Is the mainloop supposed to run.
356 volatile bool running;
357
358 // Should we exit when a client disconnects.
359 bool exit_on_disconnect;
360
361 enum u_logging_level log_level;
362
363 struct ipc_thread threads[IPC_MAX_CLIENTS];
364
365 volatile uint32_t current_slot_index;
366
367 //! Generator for IDs.
368 uint32_t id_generator;
369
370 struct
371 {
372 int active_client_index;
373 int last_active_client_index;
374
375 struct os_mutex lock;
376 } global_state;
377};
378
379
380/*!
381 * Get the current state of a client.
382 *
383 * @ingroup ipc_server
384 */
386ipc_server_get_client_app_state(struct ipc_server *s, uint32_t client_id, struct ipc_app_state *out_ias);
387
388/*!
389 * Set the new active client.
390 *
391 * @ingroup ipc_server
392 */
394ipc_server_set_active_client(struct ipc_server *s, uint32_t client_id);
395
396/*!
397 * Toggle the io for this client.
398 *
399 * @ingroup ipc_server
400 */
402ipc_server_toggle_io_client(struct ipc_server *s, uint32_t client_id);
403
404/*!
405 * Called by client threads to set a session to active.
406 *
407 * @ingroup ipc_server
408 */
409void
411
412/*!
413 * Called by client threads to set a session to deactivate.
414 *
415 * @ingroup ipc_server
416 */
417void
419
420/*!
421 * Called by client threads to recalculate active client.
422 *
423 * @ingroup ipc_server
424 */
425void
427
428/*!
429 * Thread function for the client side dispatching.
430 *
431 * @ingroup ipc_server
432 */
433void *
434ipc_server_client_thread(void *_ics);
435
436/*!
437 * This destroys the native compositor for this client and any extra objects
438 * created from it, like all of the swapchains.
439 */
440void
442
443/*!
444 * @defgroup ipc_server_internals Server Internals
445 * @brief These are only called by the platform-specific mainloop polling code.
446 * @ingroup ipc_server
447 * @{
448 */
449/*!
450 * Called when a client has connected, it takes the client's ipc handle.
451 * Handles all things needed to be done for a client connecting, like starting
452 * it's thread.
453 *
454 * @param vs The IPC server.
455 * @param ipc_handle Handle to communicate over.
456 * @memberof ipc_server
457 */
458void
460
461/*!
462 * Perform whatever needs to be done when the mainloop polling encounters a failure.
463 * @memberof ipc_server
464 */
465void
467
468/*!
469 * Perform whatever needs to be done when the mainloop polling identifies that the server should be shut down.
470 *
471 * Does something like setting a flag or otherwise signalling for shutdown: does not itself explicitly exit.
472 * @memberof ipc_server
473 */
474void
476
478ipc_server_get_system_properties(struct ipc_server *vs, struct xrt_system_properties *out_properties);
479//! @}
480
481/*
482 *
483 * Helpers
484 *
485 */
486
487/*!
488 * Get a xdev with the given device_id.
489 */
490static inline struct xrt_device *
491get_xdev(volatile struct ipc_client_state *ics, uint32_t device_id)
492{
493 return ics->server->idevs[device_id].xdev;
494}
495
496/*!
497 * Get a idev with the given device_id.
498 */
499static inline struct ipc_device *
500get_idev(volatile struct ipc_client_state *ics, uint32_t device_id)
501{
502 return &ics->server->idevs[device_id];
503}
504
505
506#ifdef __cplusplus
507}
508#endif
Generic typedef for platform-specific shared memory handle.
u_logging_level
Logging level enum.
Definition: u_logging.h:43
void ipc_server_handle_client_connected(struct ipc_server *vs, xrt_ipc_handle_t ipc_handle)
Called when a client has connected, it takes the client's ipc handle.
Definition: ipc_server_process.c:908
void ipc_server_handle_shutdown_signal(struct ipc_server *vs)
Perform whatever needs to be done when the mainloop polling identifies that the server should be shut...
Definition: ipc_server_process.c:902
void ipc_server_handle_failure(struct ipc_server *vs)
Perform whatever needs to be done when the mainloop polling encounters a failure.
Definition: ipc_server_process.c:895
xrt_result_t ipc_server_get_client_app_state(struct ipc_server *s, uint32_t client_id, struct ipc_app_state *out_ias)
Get the current state of a client.
Definition: ipc_server_process.c:805
void ipc_server_deactivate_session(volatile struct ipc_client_state *ics)
Called by client threads to set a session to deactivate.
Definition: ipc_server_process.c:869
void ipc_server_activate_session(volatile struct ipc_client_state *ics)
Called by client threads to set a session to active.
Definition: ipc_server_process.c:835
void * ipc_server_client_thread(void *_ics)
Thread function for the client side dispatching.
Definition: ipc_server_per_client_thread.c:351
void ipc_server_update_state(struct ipc_server *s)
Called by client threads to recalculate active client.
Definition: ipc_server_process.c:884
xrt_result_t ipc_server_set_active_client(struct ipc_server *s, uint32_t client_id)
Set the new active client.
Definition: ipc_server_process.c:815
xrt_result_t ipc_server_toggle_io_client(struct ipc_server *s, uint32_t client_id)
Toggle the io for this client.
Definition: ipc_server_process.c:825
#define XRT_SYSTEM_MAX_DEVICES
Maximum number of devices simultaneously usable by an implementation of xrt_system_devices.
Definition: xrt_system.h:141
#define XRT_SPACE_REFERENCE_TYPE_COUNT
The number of enumerations in xrt_reference_space_type.
Definition: xrt_defines.h:612
enum xrt_result xrt_result_t
Result type used across Monado.
IPC message channel functions.
Common protocol definition.
void ipc_server_client_destroy_session_and_compositor(volatile struct ipc_client_state *ics)
This destroys the native compositor for this client and any extra objects created from it,...
Definition: ipc_server_per_client_thread.c:320
static struct xrt_device * get_xdev(volatile struct ipc_client_state *ics, uint32_t device_id)
Get a xdev with the given device_id.
Definition: ipc_server.h:491
static struct ipc_device * get_idev(volatile struct ipc_client_state *ics, uint32_t device_id)
Get a idev with the given device_id.
Definition: ipc_server.h:500
Wrapper around OS threading native functions.
State for a connected application.
Definition: ipc_protocol.h:306
Holds the state for a single client.
Definition: ipc_server.h:82
uint32_t swapchain_count
Number of swapchains in use by client.
Definition: ipc_server.h:96
struct xrt_swapchain * xscs[(XRT_MAX_LAYERS *2)]
Ptrs to the swapchains.
Definition: ipc_server.h:99
struct xrt_session * xs
Session for this client.
Definition: ipc_server.h:87
struct ipc_message_channel imc
Socket fd used for client comms.
Definition: ipc_server.h:139
struct ipc_server * server
Link back to the main server.
Definition: ipc_server.h:84
uint32_t local_floor_space_index
Index of localfloorspace in ipc client.
Definition: ipc_server.h:125
struct ipc_swapchain_data swapchain_data[(XRT_MAX_LAYERS *2)]
Data for the swapchains.
Definition: ipc_server.h:102
uint32_t compositor_semaphore_count
Number of compositor semaphores in use by client.
Definition: ipc_server.h:105
uint32_t local_space_overseer_index
Index of localspace in space overseer.
Definition: ipc_server.h:123
struct xrt_space * xspcs[128]
Ptrs to the spaces.
Definition: ipc_server.h:130
uint32_t local_space_index
Index of localspace in ipc client.
Definition: ipc_server.h:121
bool ref_space_used[XRT_SPACE_REFERENCE_TYPE_COUNT]
Which of the references spaces is the client using.
Definition: ipc_server.h:133
struct xrt_compositor_semaphore * xcsems[8]
Ptrs to the semaphores.
Definition: ipc_server.h:108
uint32_t space_count
Number of spaces.
Definition: ipc_server.h:119
bool device_feature_used[XRT_DEVICE_FEATURE_MAX_ENUM]
Which of the device features is the client using.
Definition: ipc_server.h:136
bool io_active
Is the inputs and outputs active.
Definition: ipc_server.h:93
uint32_t local_floor_space_overseer_index
Index of localfloorspace in space overseer.
Definition: ipc_server.h:127
struct xrt_compositor * xc
Compositor for this client.
Definition: ipc_server.h:90
Definition: ipc_server.h:166
bool io_active
Is the IO suppressed for this device.
Definition: ipc_server.h:171
struct xrt_device * xdev
The actual device.
Definition: ipc_server.h:168
Wrapper for a socket and flags.
Definition: ipc_message_channel.h:30
Platform-specific mainloop object for the IPC server.
Definition: ipc_server.h:185
bool launched_by_socket
Were we launched by socket activation, instead of explicitly?
Definition: ipc_server.h:264
pthread_cond_t accept_cond
Condition variable for accepting clients.
Definition: ipc_server.h:236
char * pipe_name
Name of the Pipe that we accept connections on.
Definition: ipc_server.h:284
int last_accepted_fd
The last client fd we accepted, to acknowledge client acceptance.
Definition: ipc_server.h:227
int epoll_fd
For waiting on various events in the main thread.
Definition: ipc_server.h:189
void ipc_server_mainloop_deinit(struct ipc_server_mainloop *ml)
De-initialize the mainloop object.
Definition: ipc_server_mainloop_android.c:160
pthread_mutex_t accept_mutex
Mutex for accepting clients.
Definition: ipc_server.h:247
int pipe_read
File descriptor for the read end of our pipe for submitting new clients.
Definition: ipc_server.h:199
int listen_socket
Socket that we accept connections on.
Definition: ipc_server.h:261
int pipe_write
File descriptor for the write end of our pipe for submitting new clients.
Definition: ipc_server.h:206
int ipc_server_mainloop_init(struct ipc_server_mainloop *ml)
Initialize the mainloop object.
Definition: ipc_server_mainloop_android.c:142
char * socket_filename
The socket filename we bound to, if any.
Definition: ipc_server.h:267
pthread_mutex_t client_push_mutex
Mutex for being able to register oneself as a new client.
Definition: ipc_server.h:217
void ipc_server_mainloop_poll(struct ipc_server *vs, struct ipc_server_mainloop *ml)
Poll the mainloop.
Definition: ipc_server_mainloop_android.c:119
HANDLE pipe_handle
Named Pipe that we accept connections on.
Definition: ipc_server.h:281
Main IPC object for the server.
Definition: ipc_server.h:327
struct u_process * process
Handle for the current process, e.g. pidfile on linux.
Definition: ipc_server.h:331
struct xrt_system_compositor * xsysc
System compositor.
Definition: ipc_server.h:345
struct xrt_space_overseer * xso
Space overseer.
Definition: ipc_server.h:342
uint32_t id_generator
Generator for IDs.
Definition: ipc_server.h:368
struct xrt_system * xsys
The XRT interfaces level system.
Definition: ipc_server.h:336
struct xrt_system_devices * xsysd
System devices.
Definition: ipc_server.h:339
A big struct that contains all data that is shared to a client, no pointers allowed in this.
Definition: ipc_protocol.h:197
Information about a single swapchain.
Definition: ipc_server.h:67
Definition: ipc_server.h:155
A wrapper around a native mutex.
Definition: os_threading.h:55
A wrapper around a native thread.
Definition: os_threading.h:272
Definition: u_worker.c:37
Definition: u_process.c:43
Main compositor server interface.
Definition: xrt_compositor.h:2224
Compositor semaphore used for synchronization, needs to be as capable as a Vulkan pipeline semaphore.
Definition: xrt_compositor.h:788
Common compositor client interface/base.
Definition: xrt_compositor.h:988
A single HMD or input device.
Definition: xrt_device.h:241
This interface acts as a root object for Monado.
Definition: xrt_instance.h:114
The XRT representation of XrSession, this object does not have all of the functionality of a session,...
Definition: xrt_session.h:246
Object that oversees and manages spaces, one created for each XR system.
Definition: xrt_space.h:96
A space very similar to a OpenXR XrSpace but not a full one-to-one mapping, but used to power XrSpace...
Definition: xrt_space.h:31
Common swapchain interface/base.
Definition: xrt_compositor.h:536
The system compositor handles composition for a system.
Definition: xrt_compositor.h:2414
A collection of xrt_device, and an interface for identifying the roles they have been assigned.
Definition: xrt_system.h:219
Properties provided by the system.
Definition: xrt_system.h:44
A system is a collection of devices, policies and optionally a compositor that is organised into a ch...
Definition: xrt_system.h:62
A tracking system or device origin.
Definition: xrt_tracking.h:71
Basic logging functionality.
Header holding common defines.
int xrt_ipc_handle_t
The type for an IPC handle.
Definition: xrt_handles.h:74
Header for limits of the XRT interfaces.
Header defining xrt space and space overseer.
Header for system objects.