mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-25 16:11:45 +00:00
204 lines
5.2 KiB
C
204 lines
5.2 KiB
C
|
// Copyright 2019, Collabora, Ltd.
|
||
|
// SPDX-License-Identifier: BSL-1.0
|
||
|
/*!
|
||
|
* @file
|
||
|
* @brief Implementation
|
||
|
* @author Ryan Pavlik <ryan.pavlik@collabora.com>
|
||
|
*/
|
||
|
|
||
|
#include "util/u_debug.h"
|
||
|
|
||
|
#include "oxr_objects.h"
|
||
|
#include "oxr_logger.h"
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
|
||
|
DEBUG_GET_ONCE_BOOL_OPTION(lifecycle_verbose, "OXR_LIFECYCLE_VERBOSE", false)
|
||
|
|
||
|
#define HANDLE_LIFECYCLE_LOG(log, ...) \
|
||
|
if (debug_get_bool_option_lifecycle_verbose()) { \
|
||
|
oxr_log(log, " Handle Lifecycle: " __VA_ARGS__); \
|
||
|
}
|
||
|
|
||
|
|
||
|
const char *
|
||
|
oxr_handle_state_to_string(enum oxr_handle_state state)
|
||
|
{
|
||
|
switch (state) {
|
||
|
case OXR_HANDLE_STATE_UNINITIALIZED: return "UNINITIALIZED";
|
||
|
case OXR_HANDLE_STATE_LIVE: return "LIVE";
|
||
|
case OXR_HANDLE_STATE_DESTROYED: return "DESTROYED";
|
||
|
default: return "<UNKNOWN>";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
XrResult
|
||
|
oxr_handle_init(struct oxr_logger *log,
|
||
|
struct oxr_handle_base *hb,
|
||
|
uint64_t debug,
|
||
|
oxr_handle_destroyer destroy,
|
||
|
struct oxr_handle_base *parent)
|
||
|
{
|
||
|
assert(log != NULL);
|
||
|
assert(hb != NULL);
|
||
|
assert(destroy != NULL);
|
||
|
assert(debug != 0);
|
||
|
|
||
|
HANDLE_LIFECYCLE_LOG(
|
||
|
log, "[init %p] Initializing handle, parent handle = %p",
|
||
|
(void *)hb, (void *)parent);
|
||
|
|
||
|
|
||
|
hb->state = OXR_HANDLE_STATE_UNINITIALIZED;
|
||
|
|
||
|
if (parent != NULL) {
|
||
|
if (parent->state != OXR_HANDLE_STATE_LIVE) {
|
||
|
return oxr_error(
|
||
|
log, XR_ERROR_RUNTIME_FAILURE,
|
||
|
" Handle %p given parent %p in invalid state: %s",
|
||
|
(void *)parent, (void *)hb,
|
||
|
oxr_handle_state_to_string(parent->state));
|
||
|
}
|
||
|
|
||
|
bool placed = false;
|
||
|
for (int i = 0; i < XRT_MAX_HANDLE_CHILDREN; ++i) {
|
||
|
if (parent->children[i] == NULL) {
|
||
|
HANDLE_LIFECYCLE_LOG(log,
|
||
|
"[init %p] Assigned to "
|
||
|
"child slot %d in parent",
|
||
|
(void *)hb, i);
|
||
|
parent->children[i] = hb;
|
||
|
placed = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!placed) {
|
||
|
return oxr_error(log, XR_ERROR_LIMIT_REACHED,
|
||
|
" parent handle has no more room for "
|
||
|
"child handles");
|
||
|
}
|
||
|
}
|
||
|
memset(hb, 0, sizeof(*hb));
|
||
|
hb->debug = debug;
|
||
|
hb->parent = parent;
|
||
|
hb->state = OXR_HANDLE_STATE_LIVE;
|
||
|
hb->destroy = destroy;
|
||
|
return XR_SUCCESS;
|
||
|
}
|
||
|
|
||
|
XrResult
|
||
|
oxr_handle_allocate_and_init(struct oxr_logger *log,
|
||
|
size_t size,
|
||
|
uint64_t debug,
|
||
|
oxr_handle_destroyer destroy,
|
||
|
struct oxr_handle_base *parent,
|
||
|
void **out)
|
||
|
{
|
||
|
/*
|
||
|
* This bare calloc call, taking a size, not a type, is why this
|
||
|
* function isn't recommended for direct use.
|
||
|
*/
|
||
|
struct oxr_handle_base *hb = (struct oxr_handle_base *)calloc(1, size);
|
||
|
XrResult result = oxr_handle_init(log, hb, debug, destroy, parent);
|
||
|
if (result != XR_SUCCESS) {
|
||
|
free(hb);
|
||
|
return result;
|
||
|
}
|
||
|
*out = (void *)hb;
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
/*!
|
||
|
* This is the actual recursive call that destroys handles.
|
||
|
*
|
||
|
* oxr_handle_destroy wraps this to provide some extra output and start `level`
|
||
|
* at 0. `level`, which is reported in debug output, is the current depth of
|
||
|
* recursion.
|
||
|
*/
|
||
|
static XrResult
|
||
|
oxr_handle_do_destroy(struct oxr_logger *log,
|
||
|
struct oxr_handle_base *hb,
|
||
|
int level)
|
||
|
{
|
||
|
|
||
|
HANDLE_LIFECYCLE_LOG(log,
|
||
|
"[%d: destroying %p] Destroying handle and all "
|
||
|
"contained handles (recursively)",
|
||
|
level, (void *)hb);
|
||
|
|
||
|
/* Remove from parent, if any. */
|
||
|
if (hb->parent != NULL) {
|
||
|
bool found = false;
|
||
|
struct oxr_handle_base *parent = hb->parent;
|
||
|
|
||
|
for (int i = 0; i < XRT_MAX_HANDLE_CHILDREN; ++i) {
|
||
|
if (parent->children[i] == hb) {
|
||
|
HANDLE_LIFECYCLE_LOG(
|
||
|
log,
|
||
|
"[%d: destroying %p] Removing handle from "
|
||
|
"child slot %d in parent %p",
|
||
|
level, (void *)hb, i, (void *)hb->parent);
|
||
|
|
||
|
parent->children[i] = NULL;
|
||
|
found = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!found) {
|
||
|
return oxr_error(
|
||
|
log, XR_ERROR_RUNTIME_FAILURE,
|
||
|
" parent handle does not refer to this handle");
|
||
|
}
|
||
|
|
||
|
/* clear parent pointer */
|
||
|
hb->parent = NULL;
|
||
|
}
|
||
|
|
||
|
/* Destroy child handles */
|
||
|
for (size_t i = 0; i < XRT_MAX_HANDLE_CHILDREN; ++i) {
|
||
|
struct oxr_handle_base *child = hb->children[i];
|
||
|
|
||
|
if (child != NULL) {
|
||
|
XrResult result =
|
||
|
oxr_handle_do_destroy(log, child, level + 1);
|
||
|
if (result != XR_SUCCESS) {
|
||
|
return result;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Destroy self */
|
||
|
HANDLE_LIFECYCLE_LOG(
|
||
|
log, "[%d: destroying %p] Calling handle object destructor", level,
|
||
|
(void *)hb);
|
||
|
hb->state = OXR_HANDLE_STATE_DESTROYED;
|
||
|
XrResult result = hb->destroy(log, hb);
|
||
|
if (result != XR_SUCCESS) {
|
||
|
return result;
|
||
|
}
|
||
|
HANDLE_LIFECYCLE_LOG(log, "[%d: destroying %p] Done", level,
|
||
|
(void *)hb);
|
||
|
return XR_SUCCESS;
|
||
|
}
|
||
|
|
||
|
XrResult
|
||
|
oxr_handle_destroy(struct oxr_logger *log, struct oxr_handle_base *hb)
|
||
|
{
|
||
|
assert(log != NULL);
|
||
|
assert(hb != NULL);
|
||
|
|
||
|
HANDLE_LIFECYCLE_LOG(
|
||
|
log, "[~: destroying %p] oxr_handle_destroy starting", (void *)hb);
|
||
|
|
||
|
XrResult result = oxr_handle_do_destroy(log, hb, 0);
|
||
|
|
||
|
HANDLE_LIFECYCLE_LOG(
|
||
|
log, "[~: destroying %p] oxr_handle_destroy finished", (void *)hb);
|
||
|
|
||
|
return result;
|
||
|
}
|