mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-19 13:18:32 +00:00
132 lines
7 KiB
Markdown
132 lines
7 KiB
Markdown
|
# IPC Design {#ipc-design}
|
||
|
|
||
|
- Last updated: 8-February-2021
|
||
|
|
||
|
When the service starts, an `xrt_instance` is created and selected, a native
|
||
|
system compositor is initialized, a shared memory segment for device data is
|
||
|
initialized, and other internal state is set up. (See `ipc_server_process.c`.)
|
||
|
|
||
|
There are three main communication needs:
|
||
|
|
||
|
- The client shared library needs to be able to **locate** a running service, if
|
||
|
any, to start communication. (Auto-starting, where available, is handled by
|
||
|
platform-specific mechanisms: the client currently has no code to explicitly
|
||
|
start up the service.)
|
||
|
- The client and service must share a dedicated channel for IPC calls (aka
|
||
|
**RPC** - remote procedure call), typically a socket.
|
||
|
- The service must share access to device data updating at various rates, shared
|
||
|
by all clients. This is typically done with a form of **shared memory**.
|
||
|
|
||
|
Each platform's implementation has a way of meeting each of these needs. The
|
||
|
specific way each need is met is highlighted below.
|
||
|
|
||
|
## Linux Platform Details
|
||
|
|
||
|
In an typical Linux environment, the Monado service can be launched one of two
|
||
|
ways: manually, or by socket activation (e.g. from systemd). In either case,
|
||
|
there is a Unix domain socket with a well-known name (known at compile time, and
|
||
|
built-in to both the service executable and the client shared library) used by
|
||
|
clients to connect to the service: this provides the **locating** function.
|
||
|
This socket is polled in the service mainloop, using epoll, to detect any new
|
||
|
client connections.
|
||
|
|
||
|
Upon a client connection to this "locating" socket, the service will [accept][]
|
||
|
the connection, returning an FD, which is passed to
|
||
|
`start_client_listener_thread()` to start a thread specific to that client. The
|
||
|
FD produced this way is now also used for the IPC calls - the **RPC** function -
|
||
|
since it is specific to that client-server communication channel. One of the
|
||
|
first calls made transports a duplicate of the **shared memory** segment file
|
||
|
descriptor to the client, so it has (read) access to this data.
|
||
|
|
||
|
[accept]: https://man7.org/linux/man-pages/man2/accept.2.html
|
||
|
|
||
|
## Android Platform Details
|
||
|
|
||
|
On Android, in order to pass platform objects, allow for service activation, and
|
||
|
fit better within the idioms of the platform, Monado provides a Binder/AIDL
|
||
|
service instead of a named socket. (The named sockets we typically use are not
|
||
|
permitted by the platform, and "abstract" named sockets are currently available,
|
||
|
but are not idiomatic for the platform and lack other useful capabilities.)
|
||
|
Specifically, we provide a [foreground and started][foreground] (to be able to
|
||
|
display), [bound][bound_service] [service][android_service] with an interface
|
||
|
defined using [AIDL][]. (See also
|
||
|
[this third-party guide about such AIDL services][AidlServices]) This is not
|
||
|
like the system services which provide hardware data or system framework data
|
||
|
from native code. this has a Java (JVM/Dalvik/ART) component provided by code in
|
||
|
an APK, exposed by properties in the package manifest.
|
||
|
|
||
|
[NdkBinder][] is not used because it is mainly suitable for the system type of
|
||
|
binder services. An APK-based service would still require some JVM code to
|
||
|
expose it, and since the AIDL service is used for so little, mixing languages
|
||
|
did not make sense.
|
||
|
|
||
|
The service we expose provides an implementation of our AIDL-described
|
||
|
interface, `org.freedesktop.monado.ipc.IMonado`. This can be modified freely, as
|
||
|
both the client and server are built at the same time and packaged in the same
|
||
|
APK, even though they get loaded in different processes.
|
||
|
|
||
|
[foreground]: https://developer.android.com/guide/components/foreground-services
|
||
|
[bound_service]: https://developer.android.com/guide/components/bound-services
|
||
|
[android_service]: https://developer.android.com/guide/components/services
|
||
|
[aidl]: https://developer.android.com/guide/components/aidl
|
||
|
[AidlServices]: https://devarea.com/android-services-and-aidl/
|
||
|
[NdkBinder]: https://developer.android.com/ndk/reference/group/ndk-binder
|
||
|
|
||
|
The first main purpose of this service is for automatic startup and the
|
||
|
**locating** function: helping establish communication between the client and
|
||
|
the service. The Android framework takes care of launching the service process
|
||
|
when the client requests to start and bind our service by name and package. The
|
||
|
framework also provides us with method calls when we're started/bound. In this
|
||
|
way, the "entry point" of the Monado service on Android is the
|
||
|
`org.freedesktop.monado.ipc.MonadoService` class, which exposes the
|
||
|
implementation of our AIDL interface, `org.freedesktop.monado.ipc.MonadoImpl`.
|
||
|
|
||
|
At startup, just as on Linux, the shared memory segment is created. The
|
||
|
[ashmem][] API is used to create/destroy an anonymous **shared memory** segment
|
||
|
on Android, instead of standard POSIX shared memory, but is otherwise treated
|
||
|
and used exactly the same as on standard Linux: file descriptors are duplicated
|
||
|
and passed through IPC calls, etc.
|
||
|
|
||
|
When the client side starts up, it creates an __anonymous socket pair__ to use
|
||
|
for IPC calls (the **RPC** function) later. It then passes one of the two file
|
||
|
descriptors into the AIDL method we defined named "connect". This transports the
|
||
|
FD to the service process, which uses it as the unique communication channel for
|
||
|
that client in its own thread. This replaces the socket pair produced by
|
||
|
connecting/accepting the named socket as used in standard Linux.
|
||
|
|
||
|
[ashmem]: https://developer.android.com/ndk/reference/group/memory
|
||
|
|
||
|
The AIDL interface is also used for transporting some platform objects. At this
|
||
|
time, the only one transported in this way is the [Surface][] injected into the
|
||
|
client activity which is used for displaying rendered output.
|
||
|
|
||
|
[Surface]: https://developer.android.com/reference/android/view/Surface
|
||
|
|
||
|
### Synchronization
|
||
|
|
||
|
Synchronization of new client connections is a special challenge on the Android
|
||
|
platform, since new clients arrive via calls into JVM code while the mainloop is
|
||
|
native code. Unlike Linux, we cannot simply use epoll to check if there are new
|
||
|
connections to our locating socket.
|
||
|
|
||
|
We have the following design goals/constraints:
|
||
|
|
||
|
- All we need to communicate is an integer (file descriptor) within a process.
|
||
|
- Make it fast in the server mainloop in the most common case that there are no
|
||
|
new clients.
|
||
|
- This suggests that we should be able to check if there may be a waiting
|
||
|
client in purely native code, without JNI.
|
||
|
- Make it relatively fast in the server mainloop even when there is a client,
|
||
|
since it's the compositor thread.
|
||
|
- This might mean we want to do it all without JNI on the main thread.
|
||
|
- The client should know (and be unblocked) when the server has accepted its
|
||
|
connection.
|
||
|
- This suggests that the method called in `MonadoImpl` should block until the
|
||
|
server consumes/accepts the connection.
|
||
|
- Not 100% sure this is required, but maybe.
|
||
|
- Resources (file descriptors, etc) should not be leaked.
|
||
|
- Each should have a well-known owner at each point in time.
|
||
|
- It is OK if only one new client is accepted per mainloop.
|
||
|
- The mainloop is high rate (compositor rate) and new client connections are
|
||
|
relatively infrequent.
|