2020-04-11 00:28:35 +00:00
|
|
|
// Copyright 2020, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Common server side code.
|
|
|
|
* @author Pete Black <pblack@collabora.com>
|
|
|
|
* @ingroup ipc_server
|
|
|
|
*/
|
|
|
|
|
2020-07-15 22:18:34 +00:00
|
|
|
#include "xrt/xrt_gfx_native.h"
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
#include "util/u_misc.h"
|
|
|
|
|
|
|
|
#include "ipc_server.h"
|
|
|
|
#include "ipc_server_generated.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/epoll.h>
|
|
|
|
|
|
|
|
|
2020-07-16 15:52:55 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Helper functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static xrt_result_t
|
|
|
|
validate_swapchain_state(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t *out_index)
|
|
|
|
{
|
|
|
|
// Our handle is just the index for now.
|
|
|
|
uint32_t index = 0;
|
|
|
|
for (; index < IPC_MAX_CLIENT_SWAPCHAINS; index++) {
|
|
|
|
if (!ics->swapchain_data[index].active) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index >= IPC_MAX_CLIENT_SWAPCHAINS) {
|
|
|
|
fprintf(stderr, "ERROR: Too many swapchains!\n");
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_index = index;
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_swapchain_info(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t index,
|
2020-08-07 16:00:01 +00:00
|
|
|
const struct xrt_swapchain_create_info *info,
|
2020-07-16 15:52:55 +00:00
|
|
|
struct xrt_swapchain *xsc)
|
|
|
|
{
|
|
|
|
ics->xscs[index] = xsc;
|
|
|
|
ics->swapchain_data[index].active = true;
|
|
|
|
ics->swapchain_data[index].width = info->width;
|
|
|
|
ics->swapchain_data[index].height = info->height;
|
|
|
|
ics->swapchain_data[index].format = info->format;
|
|
|
|
ics->swapchain_data[index].num_images = xsc->num_images;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-04-11 00:28:35 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Handle functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-07-07 22:08:33 +00:00
|
|
|
ipc_handle_instance_get_shm_fd(volatile struct ipc_client_state *ics,
|
2020-07-07 23:01:46 +00:00
|
|
|
uint32_t max_num_handles,
|
2020-07-07 20:56:29 +00:00
|
|
|
xrt_shmem_handle_t *out_handles,
|
2020-07-07 23:01:46 +00:00
|
|
|
uint32_t *out_num_handles)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-07-07 20:56:29 +00:00
|
|
|
assert(max_num_handles >= 1);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-07-16 16:22:40 +00:00
|
|
|
out_handles[0] = ics->server->ism_handle;
|
2020-07-07 20:56:29 +00:00
|
|
|
*out_num_handles = 1;
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
2020-06-17 06:36:38 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_session_create(volatile struct ipc_client_state *ics,
|
2020-08-26 11:43:33 +00:00
|
|
|
const struct xrt_session_prepare_info *xspi)
|
2020-06-17 06:36:38 +00:00
|
|
|
{
|
|
|
|
ics->client_state.session_active = false;
|
|
|
|
ics->client_state.session_overlay = false;
|
|
|
|
ics->client_state.session_visible = false;
|
|
|
|
|
|
|
|
if (xspi->is_overlay) {
|
|
|
|
ics->client_state.session_overlay = true;
|
|
|
|
ics->client_state.z_order = xspi->z_order;
|
|
|
|
}
|
|
|
|
|
|
|
|
update_server_state(ics->server);
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_session_begin(volatile struct ipc_client_state *ics)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
// ics->client_state.session_active = true;
|
|
|
|
// update_server_state(ics->server);
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_session_end(volatile struct ipc_client_state *ics)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->client_state.session_active = false;
|
|
|
|
update_server_state(ics->server);
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-08-03 21:38:26 +00:00
|
|
|
ipc_handle_compositor_get_info(volatile struct ipc_client_state *ics,
|
|
|
|
struct xrt_compositor_info *out_info)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-08-03 21:38:26 +00:00
|
|
|
*out_info = ics->xc->info;
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_compositor_wait_frame(volatile struct ipc_client_state *ics,
|
2020-06-23 20:30:18 +00:00
|
|
|
int64_t *out_frame_id,
|
|
|
|
uint64_t *predicted_display_time,
|
|
|
|
uint64_t *wake_up_time,
|
|
|
|
uint64_t *predicted_display_period,
|
|
|
|
uint64_t *min_display_period)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
u_rt_helper_predict((struct u_rt_helper *)&ics->urth, out_frame_id,
|
2020-06-23 20:30:18 +00:00
|
|
|
predicted_display_time, wake_up_time,
|
|
|
|
predicted_display_period, min_display_period);
|
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
ics->client_state.session_active = true;
|
|
|
|
update_server_state(ics->server);
|
|
|
|
|
2020-06-23 20:30:18 +00:00
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_compositor_wait_woke(volatile struct ipc_client_state *ics,
|
2020-06-23 20:30:18 +00:00
|
|
|
int64_t frame_id)
|
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
u_rt_helper_mark_wait_woke((struct u_rt_helper *)&ics->urth, frame_id);
|
|
|
|
|
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
2020-06-23 20:30:18 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_compositor_begin_frame(volatile struct ipc_client_state *ics,
|
2020-06-23 20:30:18 +00:00
|
|
|
int64_t frame_id)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
u_rt_helper_mark_begin((struct u_rt_helper *)&ics->urth, frame_id);
|
|
|
|
|
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
2020-06-23 20:30:18 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_compositor_discard_frame(volatile struct ipc_client_state *ics,
|
2020-06-23 20:30:18 +00:00
|
|
|
int64_t frame_id)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
u_rt_helper_mark_discarded((struct u_rt_helper *)&ics->urth, frame_id);
|
|
|
|
|
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
2020-06-23 20:30:18 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_compositor_layer_sync(volatile struct ipc_client_state *ics,
|
2020-06-23 20:30:18 +00:00
|
|
|
int64_t frame_id,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t slot_id,
|
|
|
|
uint32_t *out_free_slot_id)
|
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
struct ipc_shared_memory *ism = ics->server->ism;
|
2020-04-11 00:28:35 +00:00
|
|
|
struct ipc_layer_slot *slot = &ism->slots[slot_id];
|
2020-05-08 19:24:21 +00:00
|
|
|
|
2020-06-09 22:13:06 +00:00
|
|
|
// Copy current slot data to our state.
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->render_state = *slot;
|
|
|
|
ics->rendering_state = true;
|
|
|
|
|
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
*out_free_slot_id =
|
|
|
|
(ics->server->current_slot_index + 1) % IPC_MAX_SLOTS;
|
|
|
|
ics->server->current_slot_index = *out_free_slot_id;
|
|
|
|
|
|
|
|
// Also protected by the global lock.
|
|
|
|
u_rt_helper_mark_delivered((struct u_rt_helper *)&ics->urth, frame_id);
|
|
|
|
|
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
2020-07-07 22:08:33 +00:00
|
|
|
ipc_handle_compositor_poll_events(volatile struct ipc_client_state *ics,
|
2020-06-17 06:36:38 +00:00
|
|
|
union xrt_compositor_event *out_xce)
|
|
|
|
{
|
|
|
|
uint64_t l_timestamp = UINT64_MAX;
|
|
|
|
volatile struct ipc_queued_event *event_to_send = NULL;
|
|
|
|
for (uint32_t i = 0; i < IPC_EVENT_QUEUE_SIZE; i++) {
|
2020-07-07 22:08:33 +00:00
|
|
|
volatile struct ipc_queued_event *e = &ics->queued_events[i];
|
2020-06-17 06:36:38 +00:00
|
|
|
if (e->pending == true && e->timestamp < l_timestamp) {
|
|
|
|
event_to_send = e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We always return an event in response to this call -
|
|
|
|
// We signal no events with a special event type.
|
|
|
|
out_xce->type = XRT_COMPOSITOR_EVENT_NONE;
|
|
|
|
|
|
|
|
if (event_to_send) {
|
|
|
|
*out_xce = event_to_send->event;
|
|
|
|
event_to_send->pending = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_get_client_info(volatile struct ipc_client_state *_ics,
|
|
|
|
uint32_t id,
|
|
|
|
struct ipc_app_state *out_client_desc)
|
|
|
|
{
|
|
|
|
if (id >= IPC_MAX_CLIENTS) {
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
volatile struct ipc_client_state *ics = &_ics->server->threads[id].ics;
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-07-07 21:06:41 +00:00
|
|
|
if (ics->imc.socket_fd <= 0) {
|
2020-06-17 06:36:38 +00:00
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_client_desc = ics->client_state;
|
2020-08-18 16:05:43 +00:00
|
|
|
out_client_desc->io_active = ics->io_active;
|
2020-06-17 06:36:38 +00:00
|
|
|
|
|
|
|
//@todo: track this data in the ipc_client_state struct
|
|
|
|
out_client_desc->primary_application = false;
|
|
|
|
if (ics->server->active_client_index == (int)id) {
|
|
|
|
out_client_desc->primary_application = true;
|
|
|
|
}
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
2020-06-23 20:30:18 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_set_client_info(volatile struct ipc_client_state *ics,
|
|
|
|
struct ipc_app_state *client_desc)
|
|
|
|
{
|
|
|
|
ics->client_state.info = client_desc->info;
|
|
|
|
ics->client_state.pid = client_desc->pid;
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_system_get_clients(volatile struct ipc_client_state *_ics,
|
|
|
|
struct ipc_client_list *list)
|
|
|
|
{
|
|
|
|
for (uint32_t i = 0; i < IPC_MAX_CLIENTS; i++) {
|
|
|
|
list->ids[i] = _ics->server->threads[i].ics.server_thread_index;
|
|
|
|
}
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_set_primary_client(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t client_id)
|
|
|
|
{
|
|
|
|
|
|
|
|
ics->server->active_client_index = client_id;
|
|
|
|
printf("system setting active client to %d\n", client_id);
|
|
|
|
update_server_state(ics->server);
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_set_focused_client(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t client_id)
|
|
|
|
{
|
|
|
|
printf("UNIMPLEMENTED: system setting focused client to %d\n",
|
|
|
|
client_id);
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-08-18 16:05:43 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_toggle_io_client(volatile struct ipc_client_state *_ics,
|
|
|
|
uint32_t client_id)
|
|
|
|
{
|
|
|
|
volatile struct ipc_client_state *ics = NULL;
|
|
|
|
|
|
|
|
if (client_id >= IPC_MAX_CLIENTS) {
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ics = &_ics->server->threads[client_id].ics;
|
|
|
|
|
|
|
|
if (ics->imc.socket_fd <= 0) {
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ics->io_active = !ics->io_active;
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_system_toggle_io_device(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t device_id)
|
|
|
|
{
|
|
|
|
if (device_id >= IPC_MAX_DEVICES) {
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ipc_device *idev = &ics->server->idevs[device_id];
|
|
|
|
|
|
|
|
idev->io_active = !idev->io_active;
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_swapchain_create(volatile struct ipc_client_state *ics,
|
2020-08-07 16:00:01 +00:00
|
|
|
const struct xrt_swapchain_create_info *info,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t *out_id,
|
|
|
|
uint32_t *out_num_images,
|
|
|
|
uint64_t *out_size,
|
2020-07-07 23:01:46 +00:00
|
|
|
uint32_t max_num_handles,
|
2020-07-07 20:56:29 +00:00
|
|
|
xrt_graphics_buffer_handle_t *out_handles,
|
2020-07-07 23:01:46 +00:00
|
|
|
uint32_t *out_num_handles)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-07-16 14:01:06 +00:00
|
|
|
xrt_result_t xret = XRT_SUCCESS;
|
2020-05-30 15:34:40 +00:00
|
|
|
uint32_t index = 0;
|
|
|
|
|
2020-07-16 15:52:55 +00:00
|
|
|
xret = validate_swapchain_state(ics, &index);
|
|
|
|
if (xret != XRT_SUCCESS) {
|
|
|
|
return xret;
|
2020-05-30 15:34:40 +00:00
|
|
|
}
|
|
|
|
|
2020-08-18 18:07:43 +00:00
|
|
|
// Create the swapchain
|
2020-07-16 14:01:06 +00:00
|
|
|
struct xrt_swapchain *xsc = NULL;
|
|
|
|
xret = xrt_comp_create_swapchain(ics->xc, info, &xsc);
|
|
|
|
if (xret != XRT_SUCCESS) {
|
2020-08-18 18:07:43 +00:00
|
|
|
fprintf(stderr, "ERROR: xrt_comp_create_swapchain failed!\n");
|
2020-07-16 14:01:06 +00:00
|
|
|
return xret;
|
|
|
|
}
|
|
|
|
|
2020-05-30 15:34:40 +00:00
|
|
|
// It's now safe to increment the number of swapchains.
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->num_swapchains++;
|
2020-05-30 15:34:40 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
IPC_SPEW(ics->server, "IPC: Created swapchain %d\n", index);
|
2020-06-19 12:41:34 +00:00
|
|
|
|
2020-07-16 15:52:55 +00:00
|
|
|
set_swapchain_info(ics, index, info, xsc);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// return our result to the caller.
|
2020-07-14 22:13:07 +00:00
|
|
|
struct xrt_swapchain_native *xscn = (struct xrt_swapchain_native *)xsc;
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Sanity checking.
|
2020-07-16 20:20:53 +00:00
|
|
|
assert(xsc->num_images <= IPC_MAX_SWAPCHAIN_HANDLES);
|
2020-07-16 15:52:55 +00:00
|
|
|
assert(xsc->num_images <= max_num_handles);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-05-30 15:34:40 +00:00
|
|
|
*out_id = index;
|
2020-07-14 22:13:07 +00:00
|
|
|
*out_size = xscn->images[0].size;
|
2020-07-16 15:52:55 +00:00
|
|
|
*out_num_images = xsc->num_images;
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Setup the fds.
|
2020-07-16 15:52:55 +00:00
|
|
|
*out_num_handles = xsc->num_images;
|
|
|
|
for (size_t i = 0; i < xsc->num_images; i++) {
|
2020-07-16 20:20:53 +00:00
|
|
|
out_handles[i] = xscn->images[i].handle;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 15:52:55 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_swapchain_import(volatile struct ipc_client_state *ics,
|
2020-08-07 16:00:01 +00:00
|
|
|
const struct xrt_swapchain_create_info *info,
|
2020-07-16 15:52:55 +00:00
|
|
|
struct ipc_arg_swapchain_from_native *args,
|
|
|
|
uint32_t *out_id,
|
2020-08-06 14:43:09 +00:00
|
|
|
const xrt_graphics_buffer_handle_t *handles,
|
2020-07-16 15:52:55 +00:00
|
|
|
uint32_t num_handles)
|
|
|
|
{
|
|
|
|
xrt_result_t xret = XRT_SUCCESS;
|
|
|
|
uint32_t index = 0;
|
|
|
|
|
|
|
|
xret = validate_swapchain_state(ics, &index);
|
|
|
|
if (xret != XRT_SUCCESS) {
|
|
|
|
return xret;
|
|
|
|
}
|
|
|
|
|
2020-07-16 20:20:53 +00:00
|
|
|
struct xrt_image_native xins[IPC_MAX_SWAPCHAIN_HANDLES] = {0};
|
2020-07-16 15:52:55 +00:00
|
|
|
for (uint32_t i = 0; i < num_handles; i++) {
|
2020-07-16 20:20:53 +00:00
|
|
|
xins[i].handle = handles[i];
|
2020-07-16 15:52:55 +00:00
|
|
|
xins[i].size = args->sizes[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
// create the swapchain
|
|
|
|
struct xrt_swapchain *xsc = NULL;
|
|
|
|
xret =
|
|
|
|
xrt_comp_import_swapchain(ics->xc, info, xins, num_handles, &xsc);
|
|
|
|
if (xret != XRT_SUCCESS) {
|
|
|
|
return xret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// It's now safe to increment the number of swapchains.
|
|
|
|
ics->num_swapchains++;
|
|
|
|
|
|
|
|
IPC_SPEW(ics->server, "IPC: Created swapchain %d\n", index);
|
|
|
|
|
|
|
|
set_swapchain_info(ics, index, info, xsc);
|
|
|
|
*out_id = index;
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_swapchain_wait_image(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
uint64_t timeout,
|
|
|
|
uint32_t index)
|
|
|
|
{
|
|
|
|
//! @todo Look up the index.
|
|
|
|
uint32_t sc_index = id;
|
2020-06-17 06:36:38 +00:00
|
|
|
struct xrt_swapchain *xsc = ics->xscs[sc_index];
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
xrt_swapchain_wait_image(xsc, timeout, index);
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_swapchain_acquire_image(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
uint32_t *out_index)
|
|
|
|
{
|
|
|
|
//! @todo Look up the index.
|
|
|
|
uint32_t sc_index = id;
|
2020-06-17 06:36:38 +00:00
|
|
|
struct xrt_swapchain *xsc = ics->xscs[sc_index];
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
xrt_swapchain_acquire_image(xsc, out_index);
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_swapchain_release_image(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
uint32_t index)
|
|
|
|
{
|
|
|
|
//! @todo Look up the index.
|
|
|
|
uint32_t sc_index = id;
|
2020-06-17 06:36:38 +00:00
|
|
|
struct xrt_swapchain *xsc = ics->xscs[sc_index];
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
xrt_swapchain_release_image(xsc, index);
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_swapchain_destroy(volatile struct ipc_client_state *ics, uint32_t id)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
|
|
|
//! @todo Implement destroy swapchain.
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->num_swapchains--;
|
2020-05-30 15:34:40 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
xrt_swapchain_destroy((struct xrt_swapchain **)&ics->xscs[id]);
|
|
|
|
ics->swapchain_data[id].active = false;
|
2020-05-30 15:34:40 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_device_update_input(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
// To make the code a bit more readable.
|
|
|
|
uint32_t device_id = id;
|
2020-06-17 06:36:38 +00:00
|
|
|
struct ipc_shared_memory *ism = ics->server->ism;
|
2020-08-18 16:05:43 +00:00
|
|
|
struct ipc_device *idev = get_idev(ics, device_id);
|
|
|
|
struct xrt_device *xdev = idev->xdev;
|
|
|
|
struct ipc_shared_device *isdev = &ism->isdevs[device_id];
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Update inputs.
|
|
|
|
xrt_device_update_inputs(xdev);
|
|
|
|
|
|
|
|
// Copy data into the shared memory.
|
|
|
|
struct xrt_input *src = xdev->inputs;
|
2020-08-18 16:05:43 +00:00
|
|
|
struct xrt_input *dst = &ism->inputs[isdev->first_input_index];
|
|
|
|
size_t size = sizeof(struct xrt_input) * isdev->num_inputs;
|
|
|
|
|
|
|
|
bool io_active = ics->io_active && idev->io_active;
|
|
|
|
if (io_active) {
|
|
|
|
memcpy(dst, src, size);
|
|
|
|
} else {
|
|
|
|
memset(dst, 0, size);
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < isdev->num_inputs; i++) {
|
|
|
|
dst[i].name = src[i].name;
|
|
|
|
|
|
|
|
// Special case the rotation of the head.
|
|
|
|
if (dst[i].name == XRT_INPUT_GENERIC_HEAD_POSE) {
|
|
|
|
dst[i].active = src[i].active;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Reply.
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-08-18 16:05:43 +00:00
|
|
|
static struct xrt_input *
|
|
|
|
find_input(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t device_id,
|
|
|
|
enum xrt_input_name name)
|
|
|
|
{
|
|
|
|
struct ipc_shared_memory *ism = ics->server->ism;
|
|
|
|
struct ipc_shared_device *isdev = &ism->isdevs[device_id];
|
|
|
|
struct xrt_input *io = &ism->inputs[isdev->first_input_index];
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < isdev->num_inputs; i++) {
|
|
|
|
if (io[i].name == name) {
|
|
|
|
return &io[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_device_get_tracked_pose(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
enum xrt_input_name name,
|
|
|
|
uint64_t at_timestamp,
|
|
|
|
struct xrt_space_relation *out_relation)
|
|
|
|
{
|
|
|
|
|
|
|
|
// To make the code a bit more readable.
|
|
|
|
uint32_t device_id = id;
|
2020-08-18 16:05:43 +00:00
|
|
|
struct ipc_device *isdev = &ics->server->idevs[device_id];
|
|
|
|
struct xrt_device *xdev = isdev->xdev;
|
|
|
|
|
|
|
|
// Find the input
|
|
|
|
struct xrt_input *input = find_input(ics, device_id, name);
|
|
|
|
if (input == NULL) {
|
|
|
|
return XRT_ERROR_IPC_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Special case the headpose.
|
|
|
|
bool disabled = (!isdev->io_active || !ics->io_active) &&
|
|
|
|
name != XRT_INPUT_GENERIC_HEAD_POSE;
|
|
|
|
bool active_on_client = input->active;
|
|
|
|
|
|
|
|
// We have been disabled but the client hasn't called update.
|
|
|
|
if (disabled && active_on_client) {
|
|
|
|
U_ZERO(out_relation);
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (disabled || !active_on_client) {
|
|
|
|
return XRT_ERROR_POSE_NOT_ACTIVE;
|
|
|
|
}
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Get the pose.
|
2020-09-04 13:37:50 +00:00
|
|
|
xrt_device_get_tracked_pose(xdev, name, at_timestamp, out_relation);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-10-12 00:09:19 +00:00
|
|
|
xrt_result_t
|
|
|
|
ipc_handle_device_get_hand_tracking(volatile struct ipc_client_state *ics,
|
|
|
|
uint32_t id,
|
|
|
|
enum xrt_input_name name,
|
|
|
|
uint64_t at_timestamp,
|
|
|
|
union xrt_hand_joint_set *out_value)
|
|
|
|
{
|
|
|
|
|
|
|
|
// To make the code a bit more readable.
|
|
|
|
uint32_t device_id = id;
|
|
|
|
struct xrt_device *xdev = get_xdev(ics, device_id);
|
|
|
|
|
|
|
|
// Get the pose.
|
|
|
|
xrt_device_get_hand_tracking(xdev, name, at_timestamp, out_value);
|
|
|
|
|
|
|
|
return XRT_SUCCESS;
|
|
|
|
}
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_device_get_view_pose(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
struct xrt_vec3 *eye_relation,
|
|
|
|
uint32_t view_index,
|
|
|
|
struct xrt_pose *out_pose)
|
|
|
|
{
|
|
|
|
// To make the code a bit more readable.
|
|
|
|
uint32_t device_id = id;
|
2020-08-18 16:05:43 +00:00
|
|
|
struct xrt_device *xdev = get_xdev(ics, device_id);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Get the pose.
|
|
|
|
xrt_device_get_view_pose(xdev, eye_relation, view_index, out_pose);
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
xrt_result_t
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_handle_device_set_output(volatile struct ipc_client_state *ics,
|
2020-04-11 00:28:35 +00:00
|
|
|
uint32_t id,
|
|
|
|
enum xrt_output_name name,
|
|
|
|
union xrt_output_value *value)
|
|
|
|
{
|
|
|
|
// To make the code a bit more readable.
|
|
|
|
uint32_t device_id = id;
|
2020-08-18 16:05:43 +00:00
|
|
|
struct xrt_device *xdev = get_xdev(ics, device_id);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
// Set the output.
|
|
|
|
xrt_device_set_output(xdev, name, value);
|
|
|
|
|
2020-06-04 11:48:49 +00:00
|
|
|
return XRT_SUCCESS;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Helper functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
setup_epoll(int listen_socket)
|
|
|
|
{
|
|
|
|
int ret = epoll_create1(EPOLL_CLOEXEC);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int epoll_fd = ret;
|
|
|
|
|
|
|
|
struct epoll_event ev = {0};
|
|
|
|
|
|
|
|
ev.events = EPOLLIN;
|
|
|
|
ev.data.fd = listen_socket;
|
|
|
|
ret = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, listen_socket, &ev);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "ERROR: epoll_ctl(listen_socket) failed '%i'\n",
|
|
|
|
ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return epoll_fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Client loop.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
2020-06-17 06:36:38 +00:00
|
|
|
client_loop(volatile struct ipc_client_state *ics)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "SERVER: Client connected\n");
|
|
|
|
|
2020-07-01 12:22:03 +00:00
|
|
|
// Make sure it's ready for the client.
|
|
|
|
u_rt_helper_client_clear((struct u_rt_helper *)&ics->urth);
|
2020-06-17 06:36:38 +00:00
|
|
|
|
2020-04-11 00:28:35 +00:00
|
|
|
// Claim the client fd.
|
2020-07-07 21:06:41 +00:00
|
|
|
int epoll_fd = setup_epoll(ics->imc.socket_fd);
|
2020-04-11 00:28:35 +00:00
|
|
|
if (epoll_fd < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t buf[IPC_BUF_SIZE];
|
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
while (ics->server->running) {
|
2020-04-11 00:28:35 +00:00
|
|
|
const int half_a_second_ms = 500;
|
|
|
|
struct epoll_event event = {0};
|
|
|
|
|
|
|
|
// We use epoll here to be able to timeout.
|
|
|
|
int ret = epoll_wait(epoll_fd, &event, 1, half_a_second_ms);
|
|
|
|
if (ret < 0) {
|
2020-04-30 19:37:24 +00:00
|
|
|
fprintf(stderr,
|
2020-07-07 19:33:31 +00:00
|
|
|
"ERROR: Failed epoll_wait '%i', disconnecting "
|
2020-04-30 19:37:24 +00:00
|
|
|
"client.\n",
|
|
|
|
ret);
|
2020-04-11 00:28:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Timed out, loop again.
|
|
|
|
if (ret == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Detect clients disconnecting gracefully.
|
|
|
|
if (ret > 0 && (event.events & EPOLLHUP) != 0) {
|
2020-04-30 19:37:24 +00:00
|
|
|
fprintf(stderr, "SERVER: Client disconnected\n");
|
2020-04-11 00:28:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally get the data that is waiting for us.
|
2020-07-07 21:06:41 +00:00
|
|
|
//! @todo replace this call
|
|
|
|
ssize_t len = recv(ics->imc.socket_fd, &buf, IPC_BUF_SIZE, 0);
|
2020-04-11 00:28:35 +00:00
|
|
|
if (len < 4) {
|
2020-04-30 19:37:24 +00:00
|
|
|
fprintf(stderr,
|
2020-06-17 06:36:38 +00:00
|
|
|
"ERROR: Invalid packet received, "
|
|
|
|
"disconnecting "
|
2020-04-30 19:37:24 +00:00
|
|
|
"client.\n");
|
2020-04-11 00:28:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the first 4 bytes of the message and dispatch.
|
|
|
|
ipc_command_t *ipc_command = (uint32_t *)buf;
|
2020-07-07 19:33:22 +00:00
|
|
|
xrt_result_t result = ipc_dispatch(ics, ipc_command);
|
|
|
|
if (result != XRT_SUCCESS) {
|
2020-04-30 19:37:24 +00:00
|
|
|
fprintf(stderr,
|
2020-06-17 06:36:38 +00:00
|
|
|
"ERROR: During packet handling, "
|
|
|
|
"disconnecting "
|
2020-04-30 19:37:24 +00:00
|
|
|
"client.\n");
|
2020-04-11 00:28:35 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
close(epoll_fd);
|
|
|
|
epoll_fd = -1;
|
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
// Multiple threads might be looking at these fields.
|
|
|
|
os_mutex_lock(&ics->server->global_state_lock);
|
|
|
|
|
2020-07-07 21:06:41 +00:00
|
|
|
ipc_message_channel_close((struct ipc_message_channel *)&ics->imc);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-07-01 12:22:03 +00:00
|
|
|
// Reset the urth for the next client.
|
|
|
|
u_rt_helper_client_clear((struct u_rt_helper *)&ics->urth);
|
2020-06-23 20:30:18 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->num_swapchains = 0;
|
|
|
|
|
|
|
|
ics->server->threads[ics->server_thread_index].state =
|
|
|
|
IPC_THREAD_STOPPING;
|
|
|
|
ics->server_thread_index = -1;
|
|
|
|
memset((void *)&ics->client_state, 0, sizeof(struct ipc_app_state));
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-05-24 15:58:41 +00:00
|
|
|
// Make sure to reset the renderstate fully.
|
2020-06-17 06:36:38 +00:00
|
|
|
ics->rendering_state = false;
|
|
|
|
ics->render_state.num_layers = 0;
|
|
|
|
for (uint32_t i = 0; i < ARRAY_SIZE(ics->render_state.layers); ++i) {
|
2020-06-09 22:04:06 +00:00
|
|
|
volatile struct ipc_layer_entry *rl =
|
2020-06-17 06:36:38 +00:00
|
|
|
&ics->render_state.layers[i];
|
2020-05-08 19:24:21 +00:00
|
|
|
|
2020-06-09 21:42:46 +00:00
|
|
|
rl->swapchain_ids[0] = 0;
|
|
|
|
rl->swapchain_ids[1] = 0;
|
|
|
|
rl->data.flip_y = false;
|
|
|
|
/*!
|
|
|
|
* @todo this is redundant, we're setting both elements of a
|
2020-06-09 22:04:06 +00:00
|
|
|
* union. Why? Can we just zero the whole render_state?
|
2020-06-09 21:42:46 +00:00
|
|
|
*/
|
|
|
|
rl->data.stereo.l.sub.image_index = 0;
|
|
|
|
rl->data.stereo.r.sub.image_index = 0;
|
|
|
|
rl->data.quad.sub.image_index = 0;
|
2020-08-13 13:12:21 +00:00
|
|
|
rl->data.cube.sub.image_index = 0;
|
|
|
|
rl->data.cylinder.sub.image_index = 0;
|
|
|
|
rl->data.equirect.sub.image_index = 0;
|
2020-06-09 21:42:46 +00:00
|
|
|
|
|
|
|
//! @todo set rects or array index?
|
2020-05-08 19:24:21 +00:00
|
|
|
}
|
2020-05-24 15:58:41 +00:00
|
|
|
|
2020-05-08 19:24:21 +00:00
|
|
|
// Destroy all swapchains now.
|
2020-04-11 00:28:35 +00:00
|
|
|
for (uint32_t j = 0; j < IPC_MAX_CLIENT_SWAPCHAINS; j++) {
|
2020-06-17 06:36:38 +00:00
|
|
|
xrt_swapchain_destroy((struct xrt_swapchain **)&ics->xscs[j]);
|
|
|
|
ics->swapchain_data[j].active = false;
|
|
|
|
IPC_SPEW(ics->server, "IPC: Destroyed swapchain %d\n", j);
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
os_mutex_unlock(&ics->server->global_state_lock);
|
|
|
|
|
2020-04-11 00:28:35 +00:00
|
|
|
// Should we stop the server when a client disconnects?
|
2020-06-17 06:36:38 +00:00
|
|
|
if (ics->server->exit_on_disconnect) {
|
|
|
|
ics->server->running = false;
|
2020-04-11 00:28:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Entry point.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
void *
|
2020-06-17 06:36:38 +00:00
|
|
|
ipc_server_client_thread(void *_ics)
|
2020-04-11 00:28:35 +00:00
|
|
|
{
|
2020-06-17 06:36:38 +00:00
|
|
|
volatile struct ipc_client_state *ics = _ics;
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
client_loop(ics);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
2020-06-17 06:36:38 +00:00
|
|
|
update_server_state(ics->server);
|
2020-04-11 00:28:35 +00:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|