mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-01 04:36:07 +00:00
doc: Add IPC design/details doc, with only the existing Android details
This commit is contained in:
parent
ab5af29d05
commit
d829fac08c
131
doc/ipc.md
Normal file
131
doc/ipc.md
Normal file
|
@ -0,0 +1,131 @@
|
|||
# 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.
|
|
@ -17,6 +17,7 @@ release.
|
|||
* @ref md_targets
|
||||
* @ref vulkan-extensions
|
||||
* @ref writing-driver (**not complete**)
|
||||
* @ref ipc-design
|
||||
|
||||
## Source layout
|
||||
|
||||
|
|
Loading…
Reference in a new issue