monado/src/xrt/state_trackers/prober/p_prober.c

1370 lines
32 KiB
C
Raw Normal View History

2022-05-11 12:07:28 +00:00
// Copyright 2019-2022, Collabora, Ltd.
2019-04-30 13:33:34 +00:00
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Main prober code.
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup st_prober
*/
2020-10-15 17:11:25 +00:00
#include "xrt/xrt_config_drivers.h"
#include "xrt/xrt_settings.h"
2020-10-15 17:11:25 +00:00
2019-08-31 11:49:32 +00:00
#include "util/u_var.h"
2019-04-30 13:33:34 +00:00
#include "util/u_misc.h"
#include "util/u_config_json.h"
2019-08-31 11:49:32 +00:00
#include "util/u_debug.h"
#include "util/u_pretty_print.h"
2021-09-29 21:22:24 +00:00
#include "util/u_trace_marker.h"
2020-03-03 22:39:03 +00:00
#include "os/os_hid.h"
2019-04-30 13:33:34 +00:00
#include "p_prober.h"
2019-11-10 01:36:34 +00:00
#ifdef XRT_HAVE_V4L2
#include "v4l2/v4l2_interface.h"
#endif
#ifdef XRT_BUILD_DRIVER_VF
2020-11-26 19:16:02 +00:00
#include "vf/vf_interface.h"
#endif
#ifdef XRT_BUILD_DRIVER_EUROC
#include "euroc/euroc_interface.h"
#endif
2021-12-06 20:53:00 +00:00
#ifdef XRT_BUILD_DRIVER_REALSENSE
#include "realsense/rs_interface.h"
#endif
2019-04-30 13:33:34 +00:00
#include <stdio.h>
#include <string.h>
#include <assert.h>
2019-04-30 13:33:34 +00:00
#include "multi_wrapper/multi.h"
2019-04-30 13:33:34 +00:00
2021-09-29 21:28:23 +00:00
2019-04-30 13:33:34 +00:00
/*
*
2021-09-29 21:28:23 +00:00
* Env variable options.
2019-04-30 13:33:34 +00:00
*
*/
DEBUG_GET_ONCE_LOG_OPTION(prober_log, "PROBER_LOG", U_LOGGING_INFO)
DEBUG_GET_ONCE_BOOL_OPTION(qwerty_enable, "QWERTY_ENABLE", false)
DEBUG_GET_ONCE_BOOL_OPTION(qwerty_combine, "QWERTY_COMBINE", false)
2021-09-29 21:28:23 +00:00
DEBUG_GET_ONCE_OPTION(vf_path, "VF_PATH", NULL)
DEBUG_GET_ONCE_OPTION(euroc_path, "EUROC_PATH", NULL)
DEBUG_GET_ONCE_NUM_OPTION(rs_source_index, "RS_SOURCE_INDEX", -1)
2021-09-29 21:28:23 +00:00
/*
*
* Pre-declare functions.
*
*/
2019-04-30 13:33:34 +00:00
static void
add_device(struct prober *p, struct prober_device **out_dev);
2019-04-30 13:33:34 +00:00
static int
initialize(struct prober *p, struct xrt_prober_entry_lists *lists);
2019-04-30 13:33:34 +00:00
static void
teardown_devices(struct prober *p);
2019-04-30 13:33:34 +00:00
static void
teardown(struct prober *p);
2019-04-30 13:33:34 +00:00
2022-05-18 15:36:49 +00:00
static xrt_result_t
2021-09-29 21:28:23 +00:00
p_probe(struct xrt_prober *xp);
2019-04-30 13:33:34 +00:00
static xrt_result_t
p_lock_list(struct xrt_prober *xp, struct xrt_prober_device ***out_devices, size_t *out_device_count);
static xrt_result_t
p_unlock_list(struct xrt_prober *xp, struct xrt_prober_device ***devices);
2019-04-30 13:33:34 +00:00
static int
2021-09-29 21:28:23 +00:00
p_dump(struct xrt_prober *xp);
2019-04-30 13:33:34 +00:00
static xrt_result_t
p_create_system(struct xrt_prober *xp, struct xrt_system_devices **out_xsysd);
2019-04-30 13:33:34 +00:00
static int
p_select_device(struct xrt_prober *xp, struct xrt_device **xdevs, size_t xdev_count);
2019-04-30 13:33:34 +00:00
static int
2021-09-29 21:28:23 +00:00
p_open_hid_interface(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
int interface,
struct os_hid_device **out_hid_dev);
2019-04-30 13:33:34 +00:00
static int
2021-09-29 21:28:23 +00:00
p_open_video_device(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
struct xrt_frame_context *xfctx,
struct xrt_fs **out_xfs);
static int
2022-05-11 12:07:28 +00:00
p_list_video_devices(struct xrt_prober *xp, xrt_prober_list_video_func_t cb, void *ptr);
static int
2021-09-29 21:28:23 +00:00
p_get_entries(struct xrt_prober *xp,
size_t *out_num_entries,
struct xrt_prober_entry ***out_entries,
struct xrt_auto_prober ***out_auto_probers);
static int
2021-09-29 21:28:23 +00:00
p_get_string_descriptor(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
enum xrt_prober_string which_string,
unsigned char *buffer,
size_t length);
static bool
2021-09-29 21:28:23 +00:00
p_can_open(struct xrt_prober *xp, struct xrt_prober_device *xpdev);
2019-04-30 13:33:34 +00:00
static void
2021-09-29 21:28:23 +00:00
p_destroy(struct xrt_prober **xp);
2019-04-30 13:33:34 +00:00
/*
*
* "Exported" functions.
*
*/
int
2021-01-14 14:13:48 +00:00
xrt_prober_create_with_lists(struct xrt_prober **out_xp, struct xrt_prober_entry_lists *lists)
2019-04-30 13:33:34 +00:00
{
struct prober *p = U_TYPED_CALLOC(struct prober);
2019-04-30 13:33:34 +00:00
int ret = initialize(p, lists);
if (ret != 0) {
free(p);
return ret;
}
*out_xp = &p->base;
2019-04-30 13:33:34 +00:00
return 0;
}
int
p_dev_get_usb_dev(struct prober *p,
2019-04-30 13:33:34 +00:00
uint16_t bus,
uint16_t addr,
uint16_t vendor_id,
uint16_t product_id,
struct prober_device **out_pdev)
2019-04-30 13:33:34 +00:00
{
struct prober_device *pdev;
2019-04-30 13:33:34 +00:00
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
struct prober_device *pdev = &p->devices[i];
2019-04-30 13:33:34 +00:00
2021-01-14 14:13:48 +00:00
if (pdev->base.bus != XRT_BUS_TYPE_USB || pdev->usb.bus != bus || pdev->usb.addr != addr) {
2019-04-30 13:33:34 +00:00
continue;
}
2021-01-14 14:13:48 +00:00
if (pdev->base.vendor_id != vendor_id || pdev->base.product_id != product_id) {
2019-04-30 13:33:34 +00:00
P_ERROR(p,
"USB device with same address but different "
"vendor and product found!\n"
"\tvendor: %04x %04x\n"
"\tproduct: %04x %04x",
2021-01-14 14:13:48 +00:00
pdev->base.vendor_id, vendor_id, pdev->base.product_id, product_id);
2019-04-30 13:33:34 +00:00
continue;
}
*out_pdev = pdev;
return 0;
}
add_device(p, &pdev);
pdev->base.vendor_id = vendor_id;
pdev->base.product_id = product_id;
pdev->base.bus = XRT_BUS_TYPE_USB;
pdev->usb.bus = bus;
pdev->usb.addr = addr;
*out_pdev = pdev;
return 0;
}
int
2021-01-14 14:13:48 +00:00
p_dev_get_bluetooth_dev(
struct prober *p, uint64_t id, uint16_t vendor_id, uint16_t product_id, struct prober_device **out_pdev)
2019-04-30 13:33:34 +00:00
{
struct prober_device *pdev;
2019-04-30 13:33:34 +00:00
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
struct prober_device *pdev = &p->devices[i];
2019-04-30 13:33:34 +00:00
2021-01-14 14:13:48 +00:00
if (pdev->base.bus != XRT_BUS_TYPE_BLUETOOTH || pdev->bluetooth.id != id) {
2019-04-30 13:33:34 +00:00
continue;
}
2021-01-14 14:13:48 +00:00
if (pdev->base.vendor_id != vendor_id || pdev->base.product_id != product_id) {
2019-04-30 13:33:34 +00:00
P_ERROR(p,
"Bluetooth device with same address but "
"different vendor and product found!\n"
"\tvendor: %04x %04x\n"
"\tproduct: %04x %04x",
2021-01-14 14:13:48 +00:00
pdev->base.vendor_id, vendor_id, pdev->base.product_id, product_id);
2019-04-30 13:33:34 +00:00
continue;
}
*out_pdev = pdev;
return 0;
}
add_device(p, &pdev);
pdev->base.vendor_id = vendor_id;
pdev->base.product_id = product_id;
pdev->base.bus = XRT_BUS_TYPE_BLUETOOTH;
pdev->bluetooth.id = id;
*out_pdev = pdev;
return 0;
}
/*
*
* Internal functions.
*
*/
static void
fill_out_product(struct prober *p, struct prober_device *pdev)
{
2021-01-14 14:13:48 +00:00
const char *bus = pdev->base.bus == XRT_BUS_TYPE_BLUETOOTH ? "bluetooth" : "usb";
char *str = NULL;
int ret = 0;
do {
if (strlen(pdev->base.product_name)) {
ret = snprintf(str, ret, "%s device: %s", bus, pdev->base.product_name);
} else {
ret = snprintf(str, ret, "Unknown %s device: %04x:%04x", bus, pdev->base.vendor_id,
pdev->base.product_id);
}
if (ret <= 0) {
return;
}
if (str == NULL) {
str = U_CALLOC_WITH_CAST(char, ret + 1);
} else {
pdev->usb.product = str;
return;
}
} while (true);
}
2019-04-30 13:33:34 +00:00
static void
add_device(struct prober *p, struct prober_device **out_dev)
2019-04-30 13:33:34 +00:00
{
2021-11-08 22:54:01 +00:00
U_ARRAY_REALLOC_OR_FREE(p->devices, struct prober_device, (p->device_count + 1));
2019-04-30 13:33:34 +00:00
2021-11-08 22:54:01 +00:00
struct prober_device *dev = &p->devices[p->device_count++];
2019-06-18 19:03:53 +00:00
U_ZERO(dev);
2019-04-30 13:33:34 +00:00
*out_dev = dev;
}
static void
add_usb_entry(struct prober *p, struct xrt_prober_entry *entry)
2019-04-30 13:33:34 +00:00
{
2021-01-14 14:13:48 +00:00
U_ARRAY_REALLOC_OR_FREE(p->entries, struct xrt_prober_entry *, (p->num_entries + 1));
2019-04-30 13:33:34 +00:00
p->entries[p->num_entries++] = entry;
}
static void
add_builder(struct prober *p, struct xrt_builder *xb)
{
U_ARRAY_REALLOC_OR_FREE(p->builders, struct xrt_builder *, (p->builder_count + 1));
p->builders[p->builder_count++] = xb;
P_TRACE(p, "%s: %s", xb->identifier, xb->name);
}
2019-04-30 13:33:34 +00:00
static int
collect_entries(struct prober *p)
2019-04-30 13:33:34 +00:00
{
struct xrt_prober_entry_lists *lists = p->lists;
2019-04-30 13:33:34 +00:00
while (lists) {
for (size_t i = 0; lists->builders[i] != NULL; i++) {
struct xrt_builder *xb = lists->builders[i]();
if (xb == NULL) {
continue;
}
add_builder(p, xb);
}
2021-01-14 14:13:48 +00:00
for (size_t j = 0; lists->entries != NULL && lists->entries[j]; j++) {
struct xrt_prober_entry *entry = lists->entries[j];
2019-04-30 13:33:34 +00:00
for (size_t k = 0; entry[k].found != NULL; k++) {
add_usb_entry(p, &entry[k]);
}
}
lists = lists->next;
}
return 0;
}
#define num_driver_conflicts 1
char *driver_conflicts[num_driver_conflicts][2] = {{"survive", "vive"}};
static void
disable_drivers_from_conflicts(struct prober *p)
{
if (debug_get_bool_option_qwerty_enable() && !debug_get_bool_option_qwerty_combine()) {
for (size_t entry = 0; entry < p->num_entries; entry++) {
if (strcmp(p->entries[entry]->driver_name, "Qwerty") != 0) {
P_INFO(p, "Disabling %s because we have %s", p->entries[entry]->driver_name, "Qwerty");
size_t index = p->num_disabled_drivers++;
U_ARRAY_REALLOC_OR_FREE(p->disabled_drivers, char *, p->num_disabled_drivers);
p->disabled_drivers[index] = (char *)p->entries[entry]->driver_name;
}
}
for (size_t ap = 0; ap < XRT_MAX_AUTO_PROBERS; ap++) {
if (p->auto_probers[ap] == NULL) {
continue;
}
if (strcmp(p->auto_probers[ap]->name, "Qwerty") != 0) {
P_INFO(p, "Disabling %s because we have %s", p->auto_probers[ap]->name, "Qwerty");
size_t index = p->num_disabled_drivers++;
U_ARRAY_REALLOC_OR_FREE(p->disabled_drivers, char *, p->num_disabled_drivers);
p->disabled_drivers[index] = (char *)p->auto_probers[ap]->name;
}
}
return;
}
for (size_t i = 0; i < num_driver_conflicts; i++) {
bool have_first = false;
bool have_second = false;
char *first = driver_conflicts[i][0];
char *second = driver_conflicts[i][1];
// disable second driver if we have first driver
for (size_t entry = 0; entry < p->num_entries; entry++) {
if (strcmp(p->entries[entry]->driver_name, first) == 0) {
have_first = true;
}
if (strcmp(p->entries[entry]->driver_name, second) == 0) {
have_second = true;
}
}
for (size_t ap = 0; ap < XRT_MAX_AUTO_PROBERS; ap++) {
if (p->auto_probers[ap] == NULL) {
continue;
}
if (strcmp(p->auto_probers[ap]->name, first) == 0) {
have_first = true;
}
if (strcmp(p->auto_probers[ap]->name, second) == 0) {
have_second = true;
}
}
if (have_first && have_second) {
// except don't disable second driver, if first driver is already disabled'
bool first_already_disabled = false;
;
for (size_t disabled = 0; disabled < p->num_disabled_drivers; disabled++) {
if (strcmp(p->disabled_drivers[disabled], first) == 0) {
first_already_disabled = true;
break;
}
}
if (first_already_disabled) {
P_INFO(p, "Not disabling %s because %s is disabled", second, first);
continue;
}
P_INFO(p, "Disabling %s because we have %s", second, first);
size_t index = p->num_disabled_drivers++;
U_ARRAY_REALLOC_OR_FREE(p->disabled_drivers, char *, p->num_disabled_drivers);
p->disabled_drivers[index] = second;
}
}
}
static void
parse_disabled_drivers(struct prober *p)
{
cJSON *disabled_drivers = cJSON_GetObjectItemCaseSensitive(p->json.root, "disabled");
if (!disabled_drivers) {
return;
}
cJSON *disabled_driver = NULL;
cJSON_ArrayForEach(disabled_driver, disabled_drivers)
{
if (!cJSON_IsString(disabled_driver)) {
continue;
}
size_t index = p->num_disabled_drivers++;
U_ARRAY_REALLOC_OR_FREE(p->disabled_drivers, char *, p->num_disabled_drivers);
p->disabled_drivers[index] = disabled_driver->valuestring;
}
}
2019-04-30 13:33:34 +00:00
static int
initialize(struct prober *p, struct xrt_prober_entry_lists *lists)
2019-04-30 13:33:34 +00:00
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
2021-09-29 21:28:23 +00:00
p->base.probe = p_probe;
p->base.lock_list = p_lock_list;
p->base.unlock_list = p_unlock_list;
2021-09-29 21:28:23 +00:00
p->base.dump = p_dump;
p->base.create_system = p_create_system;
2021-09-29 21:28:23 +00:00
p->base.select = p_select_device;
p->base.open_hid_interface = p_open_hid_interface;
p->base.open_video_device = p_open_video_device;
p->base.list_video_devices = p_list_video_devices;
p->base.get_entries = p_get_entries;
p->base.get_string_descriptor = p_get_string_descriptor;
p->base.can_open = p_can_open;
p->base.destroy = p_destroy;
2019-04-30 13:33:34 +00:00
p->lists = lists;
p->log_level = debug_get_log_option_prober_log();
2019-04-30 13:33:34 +00:00
p->json.file_loaded = false;
p->json.root = NULL;
u_var_add_root((void *)p, "Prober", true);
u_var_add_log_level(p, &p->log_level, "Log level");
2019-08-31 11:49:32 +00:00
2019-04-30 13:33:34 +00:00
int ret;
u_config_json_open_or_create_main_file(&p->json);
2019-04-30 13:33:34 +00:00
ret = collect_entries(p);
if (ret != 0) {
teardown(p);
return -1;
}
#ifdef XRT_HAVE_LIBUSB
ret = p_libusb_init(p);
2019-04-30 13:33:34 +00:00
if (ret != 0) {
teardown(p);
return -1;
}
#endif
2019-04-30 13:33:34 +00:00
#ifdef XRT_HAVE_LIBUVC
ret = p_libuvc_init(p);
2019-04-30 13:33:34 +00:00
if (ret != 0) {
teardown(p);
return -1;
}
#endif
ret = p_tracking_init(p);
if (ret != 0) {
teardown(p);
return -1;
}
for (int i = 0; i < XRT_MAX_AUTO_PROBERS && lists->auto_probers[i]; i++) {
2019-04-30 13:33:34 +00:00
p->auto_probers[i] = lists->auto_probers[i]();
}
p->num_disabled_drivers = 0;
parse_disabled_drivers(p);
disable_drivers_from_conflicts(p);
2019-04-30 13:33:34 +00:00
return 0;
}
static void
teardown_devices(struct prober *p)
2019-04-30 13:33:34 +00:00
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
2019-04-30 13:33:34 +00:00
// Need to free all devices.
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
struct prober_device *pdev = &p->devices[i];
2019-04-30 13:33:34 +00:00
if (pdev->usb.product != NULL) {
free((char *)pdev->usb.product);
pdev->usb.product = NULL;
}
if (pdev->usb.manufacturer != NULL) {
free((char *)pdev->usb.manufacturer);
pdev->usb.manufacturer = NULL;
}
if (pdev->usb.serial != NULL) {
free((char *)pdev->usb.serial);
pdev->usb.serial = NULL;
}
if (pdev->usb.path != NULL) {
free((char *)pdev->usb.path);
pdev->usb.path = NULL;
}
#ifdef XRT_HAVE_LIBUSB
if (pdev->usb.dev != NULL) {
//! @todo Free somewhere else
}
#endif
#ifdef XRT_HAVE_LIBUVC
if (pdev->uvc.dev != NULL) {
//! @todo Free somewhere else
}
#endif
#ifdef XRT_HAVE_V4L2
2019-06-20 17:27:21 +00:00
for (size_t j = 0; j < pdev->num_v4ls; j++) {
struct prober_v4l *v4l = &pdev->v4ls[j];
free((char *)v4l->path);
2019-06-20 17:27:21 +00:00
v4l->path = NULL;
}
if (pdev->v4ls != NULL) {
free(pdev->v4ls);
pdev->v4ls = NULL;
pdev->num_v4ls = 0;
}
#endif
2019-06-20 17:27:21 +00:00
#ifdef XRT_OS_LINUX
2019-04-30 13:33:34 +00:00
for (size_t j = 0; j < pdev->num_hidraws; j++) {
struct prober_hidraw *hidraw = &pdev->hidraws[j];
free((char *)hidraw->path);
2019-04-30 13:33:34 +00:00
hidraw->path = NULL;
}
if (pdev->hidraws != NULL) {
free(pdev->hidraws);
pdev->hidraws = NULL;
pdev->num_hidraws = 0;
}
#endif
2019-04-30 13:33:34 +00:00
}
if (p->devices != NULL) {
free(p->devices);
p->devices = NULL;
2021-11-08 22:54:01 +00:00
p->device_count = 0;
2019-04-30 13:33:34 +00:00
}
}
static void
teardown(struct prober *p)
2019-04-30 13:33:34 +00:00
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
2019-08-31 11:49:32 +00:00
// First remove the variable tracking.
u_var_remove_root((void *)p);
2019-08-31 11:49:32 +00:00
// Clean up all setter uppers.
for (size_t i = 0; i < p->builder_count; i++) {
xrt_builder_destroy(&p->builders[i]);
}
p->builder_count = 0;
2022-06-17 14:48:38 +00:00
free(p->builders);
p->builders = NULL;
2019-04-30 13:33:34 +00:00
// Clean up all auto_probers.
for (int i = 0; i < XRT_MAX_AUTO_PROBERS && p->auto_probers[i]; i++) {
2019-04-30 13:33:34 +00:00
p->auto_probers[i]->destroy(p->auto_probers[i]);
p->auto_probers[i] = NULL;
}
// Need to turn off tracking early.
p_tracking_teardown(p);
2019-04-30 13:33:34 +00:00
// Need to free all entries.
if (p->entries != NULL) {
free(p->entries);
p->entries = NULL;
p->num_entries = 0;
}
teardown_devices(p);
#ifdef XRT_HAVE_LIBUVC
p_libuvc_teardown(p);
2019-04-30 13:33:34 +00:00
#endif
#ifdef XRT_HAVE_LIBUSB
p_libusb_teardown(p);
#endif
u_config_json_close(&p->json);
free(p->disabled_drivers);
2019-04-30 13:33:34 +00:00
}
2019-05-31 16:55:19 +00:00
static void
2021-01-14 14:13:48 +00:00
handle_found_device(
struct prober *p, struct xrt_device **xdevs, size_t xdev_count, bool *have_hmd, struct xrt_device *xdev)
2019-05-31 16:55:19 +00:00
{
P_DEBUG(p, "Found '%s' %p", xdev->str, (void *)xdev);
2019-05-31 16:55:19 +00:00
size_t i = 0;
for (; i < xdev_count; i++) {
if (xdevs[i] == NULL) {
break;
}
}
2019-05-31 16:55:19 +00:00
if (i + 1 > xdev_count) {
P_ERROR(p, "Too many devices, closing '%s'", xdev->str);
2019-05-31 16:55:19 +00:00
xdev->destroy(xdev);
return;
}
// we can have only one HMD
if (xdev->device_type == XRT_DEVICE_TYPE_HMD) {
if (*have_hmd) {
P_ERROR(p, "Too many HMDs, closing '%s'", xdev->str);
xdev->destroy(xdev);
return;
}
*have_hmd = true;
2019-05-31 16:55:19 +00:00
}
xdevs[i] = xdev;
2019-05-31 16:55:19 +00:00
}
static void
add_from_devices(struct prober *p, struct xrt_device **xdevs, size_t xdev_count, bool *have_hmd)
2019-04-30 13:33:34 +00:00
{
struct xrt_prober_device **dev_list = NULL;
size_t dev_count = 0;
xrt_result_t xret;
xret = xrt_prober_lock_list(&p->base, &dev_list, &dev_count);
if (xret != XRT_SUCCESS) {
P_ERROR(p, "Failed to lock list!");
return;
}
2019-04-30 13:33:34 +00:00
// Loop over all devices and entries that might match them.
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
struct prober_device *pdev = &p->devices[i];
2019-04-30 13:33:34 +00:00
for (size_t k = 0; k < p->num_entries; k++) {
struct xrt_prober_entry *entry = p->entries[k];
2021-01-14 14:13:48 +00:00
if (pdev->base.vendor_id != entry->vendor_id || pdev->base.product_id != entry->product_id) {
2019-04-30 13:33:34 +00:00
continue;
}
bool skip = false;
for (size_t disabled = 0; disabled < p->num_disabled_drivers; disabled++) {
if (strcmp(entry->driver_name, p->disabled_drivers[disabled]) == 0) {
P_INFO(p, "Skipping disabled driver %s", entry->driver_name);
skip = true;
break;
}
}
if (skip) {
continue;
}
2021-01-14 14:13:48 +00:00
struct xrt_device *new_xdevs[XRT_MAX_DEVICES_PER_PROBE] = {NULL};
2021-11-08 22:54:01 +00:00
int num_found = entry->found(&p->base, dev_list, p->device_count, i, NULL, &(new_xdevs[0]));
2019-04-30 13:33:34 +00:00
if (num_found <= 0) {
2019-05-31 16:55:19 +00:00
continue;
2019-04-30 13:33:34 +00:00
}
2021-01-14 14:13:48 +00:00
for (int created_idx = 0; created_idx < num_found; ++created_idx) {
if (new_xdevs[created_idx] == NULL) {
2021-01-14 14:13:48 +00:00
P_DEBUG(p,
"Leaving device creation loop "
"early: found function reported %i "
"created, but only %i non-null",
num_found, created_idx);
continue;
}
handle_found_device(p, xdevs, xdev_count, have_hmd, new_xdevs[created_idx]);
}
2019-04-30 13:33:34 +00:00
}
}
xret = xrt_prober_unlock_list(&p->base, &dev_list);
if (xret != XRT_SUCCESS) {
P_ERROR(p, "Failed to unlock list!");
}
}
2019-04-30 13:33:34 +00:00
static void
add_from_auto_probers(struct prober *p, struct xrt_device **xdevs, size_t xdev_count, bool *have_hmd)
{
for (int i = 0; i < XRT_MAX_AUTO_PROBERS && p->auto_probers[i]; i++) {
bool skip = false;
for (size_t disabled = 0; disabled < p->num_disabled_drivers; disabled++) {
if (strcmp(p->auto_probers[i]->name, p->disabled_drivers[disabled]) == 0) {
P_INFO(p, "Skipping disabled driver %s", p->auto_probers[i]->name);
skip = true;
break;
}
}
if (skip) {
continue;
}
/*
* If we have found a HMD, tell the auto probers not to open
* any more HMDs. This is mostly to stop OpenHMD and Monado
* fighting over devices.
*/
bool no_hmds = *have_hmd;
struct xrt_device *new_xdevs[XRT_MAX_DEVICES_PER_PROBE] = {NULL};
int num_found =
p->auto_probers[i]->lelo_dallas_autoprobe(p->auto_probers[i], NULL, no_hmds, &p->base, new_xdevs);
if (num_found <= 0) {
2019-05-31 16:55:19 +00:00
continue;
2019-04-30 13:33:34 +00:00
}
2019-05-31 16:55:19 +00:00
for (int created_idx = 0; created_idx < num_found; ++created_idx) {
if (new_xdevs[created_idx] == NULL) {
P_DEBUG(p,
"Leaving device creation loop early: %s autoprobe function reported %i "
"created, but only %i non-null",
p->auto_probers[i]->name, num_found, created_idx);
continue;
}
handle_found_device(p, xdevs, xdev_count, have_hmd, new_xdevs[created_idx]);
}
2019-04-30 13:33:34 +00:00
}
}
static void
apply_tracking_override(struct prober *p, struct xrt_device **xdevs, size_t xdev_count, struct xrt_tracking_override *o)
{
struct xrt_device *target_xdev = NULL;
size_t target_idx = 0;
struct xrt_device *tracker_xdev = NULL;
for (size_t i = 0; i < xdev_count; i++) {
struct xrt_device *xdev = xdevs[i];
if (xdev == NULL) {
continue;
}
if (strncmp(xdev->serial, o->target_device_serial, XRT_DEVICE_NAME_LEN) == 0) {
target_xdev = xdev;
target_idx = i;
}
if (strncmp(xdev->serial, o->tracker_device_serial, XRT_DEVICE_NAME_LEN) == 0) {
tracker_xdev = xdev;
}
}
if (target_xdev == NULL) {
P_WARN(p, "Tracking override target xdev %s not found", o->target_device_serial);
}
if (tracker_xdev == NULL) {
P_WARN(p, "Tracking override tracker xdev %s not found", o->tracker_device_serial);
}
if (target_xdev != NULL && tracker_xdev != NULL) {
struct xrt_device *multi = multi_create_tracking_override(o->override_type, target_xdev, tracker_xdev,
o->input_name, &o->offset);
if (multi) {
P_INFO(p, "Applying Tracking override %s <- %s", o->target_device_serial,
o->tracker_device_serial);
// drops the target device from the list, but keeps the tracker
// a tracker could be attached to multiple targets with different names
xdevs[target_idx] = multi;
} else {
P_ERROR(p, "Failed to create tracking override multi device");
}
}
}
struct xrt_builder *
find_builder_by_identifier(struct prober *p, const char *ident)
{
for (size_t i = 0; i < p->builder_count; i++) {
if (strcmp(p->builders[i]->identifier, ident) != 0) {
continue;
}
// This is what we want.
return p->builders[i];
}
struct u_pp_sink_stack_only sink;
u_pp_delegate_t dg = u_pp_sink_stack_only_init(&sink);
u_pp(dg, "Could not find builder with identifier '%s' among %u supported builders:", ident,
(uint32_t)p->builder_count);
for (size_t i = 0; i < p->builder_count; i++) {
struct xrt_builder *xb = p->builders[i];
u_pp(dg, "\n\t%s: %s", xb->identifier, xb->name);
}
P_WARN(p, "%s", sink.buffer);
return NULL;
}
2021-09-29 21:28:23 +00:00
/*
*
* Member functions.
*
*/
static xrt_result_t
2021-09-29 21:28:23 +00:00
p_probe(struct xrt_prober *xp)
{
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
XRT_MAYBE_UNUSED int ret = 0;
if (p->list_locked) {
return XRT_ERROR_PROBER_LIST_LOCKED;
}
2021-09-29 21:28:23 +00:00
// Free old list first.
teardown_devices(p);
#ifdef XRT_HAVE_LIBUDEV
ret = p_udev_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate udev devices\n");
return XRT_ERROR_PROBING_FAILED;
2021-09-29 21:28:23 +00:00
}
#endif
#ifdef XRT_HAVE_LIBUSB
ret = p_libusb_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate libusb devices\n");
return XRT_ERROR_PROBING_FAILED;
2021-09-29 21:28:23 +00:00
}
#endif
#ifdef XRT_HAVE_LIBUVC
ret = p_libuvc_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate libuvc devices\n");
return XRT_ERROR_PROBING_FAILED;
2021-09-29 21:28:23 +00:00
}
#endif
return XRT_SUCCESS;
2021-09-29 21:28:23 +00:00
}
static xrt_result_t
p_lock_list(struct xrt_prober *xp, struct xrt_prober_device ***out_devices, size_t *out_device_count)
{
struct prober *p = (struct prober *)xp;
if (p->list_locked) {
return XRT_ERROR_PROBER_LIST_LOCKED;
}
assert(out_devices != NULL);
assert(*out_devices == NULL);
// Build a list of all current probed devices.
struct xrt_prober_device **dev_list = U_TYPED_ARRAY_CALLOC(struct xrt_prober_device *, p->device_count);
for (size_t i = 0; i < p->device_count; i++) {
dev_list[i] = &p->devices[i].base;
}
p->list_locked = true;
*out_devices = dev_list;
*out_device_count = p->device_count;
return XRT_SUCCESS;
}
static xrt_result_t
p_unlock_list(struct xrt_prober *xp, struct xrt_prober_device ***devices)
{
struct prober *p = (struct prober *)xp;
if (!p->list_locked) {
return XRT_ERROR_PROBER_LIST_NOT_LOCKED;
}
assert(devices != NULL);
p->list_locked = false;
free(*devices);
*devices = NULL;
return XRT_SUCCESS;
}
2021-09-29 21:28:23 +00:00
static int
p_dump(struct xrt_prober *xp)
{
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
XRT_MAYBE_UNUSED ssize_t k = 0;
XRT_MAYBE_UNUSED size_t j = 0;
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
2021-09-29 21:28:23 +00:00
struct prober_device *pdev = &p->devices[i];
p_dump_device(p, pdev, (int)i);
}
return 0;
}
static xrt_result_t
p_create_system(struct xrt_prober *xp, struct xrt_system_devices **out_xsysd)
{
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
struct xrt_builder *select = NULL;
enum u_config_json_active_config active;
xrt_result_t xret = XRT_SUCCESS;
struct u_pp_sink_stack_only sink; // Not inited, very large.
u_pp_delegate_t dg = u_pp_sink_stack_only_init(&sink);
/*
* Logging.
*/
u_pp(dg, "Creating system:");
u_pp(dg, "\n\tBuilders:");
for (size_t i = 0; i < p->builder_count; i++) {
u_pp(dg, "\n\t\t%s: %s", p->builders[i]->identifier, p->builders[i]->name);
}
/*
* Config.
*/
u_config_json_get_active(&p->json, &active);
switch (active) {
case U_ACTIVE_CONFIG_NONE: break;
case U_ACTIVE_CONFIG_REMOTE: select = find_builder_by_identifier(p, "remote"); break;
case U_ACTIVE_CONFIG_TRACKING: select = find_builder_by_identifier(p, "rgb_tracking"); break;
default: assert(false);
}
if (select != NULL) {
u_pp(dg, "\n\tConfig selected %s", select->identifier);
} else {
u_pp(dg, "\n\tNo builder selected in config (or wasn't compiled in)");
}
/*
* Estimate.
*/
//! @todo Improve estimation selection logic.
if (select == NULL) {
for (size_t i = 0; i < p->builder_count; i++) {
struct xrt_builder *xb = p->builders[i];
if (xb->exclude_from_automatic_discovery) {
continue;
}
struct xrt_builder_estimate estimate = {0};
xrt_builder_estimate_system(xb, p->json.root, xp, &estimate);
if (estimate.certain.head) {
select = xb;
break;
}
}
if (select != NULL) {
u_pp(dg, "\n\tSelected %s because it was certain it could create a head", select->identifier);
} else {
u_pp(dg, "\n\tNo builder was certain that it could create a head device");
}
}
if (select == NULL) {
for (size_t i = 0; i < p->builder_count; i++) {
struct xrt_builder *xb = p->builders[i];
if (xb->exclude_from_automatic_discovery) {
continue;
}
struct xrt_builder_estimate estimate = {0};
xrt_builder_estimate_system(xb, p->json.root, xp, &estimate);
if (estimate.maybe.head) {
select = xb;
break;
}
}
if (select != NULL) {
u_pp(dg, "\n\tSelected %s because it maybe could create a head", select->identifier);
} else {
u_pp(dg, "\n\tNo builder could maybe create a head device");
}
}
if (select != NULL) {
u_pp(dg, "\n\tUsing builder %s: %s", select->identifier, select->name);
xret = xrt_builder_open_system(select, p->json.root, xp, out_xsysd);
u_pp(dg, "\n\tResult: ");
u_pp_xrt_result(dg, xret);
}
P_INFO(p, "%s", sink.buffer);
return xret;
}
static int
p_select_device(struct xrt_prober *xp, struct xrt_device **xdevs, size_t xdev_count)
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
enum u_config_json_active_config active;
bool have_hmd = false;
u_config_json_get_active(&p->json, &active);
switch (active) {
case U_ACTIVE_CONFIG_NONE:
case U_ACTIVE_CONFIG_TRACKING:
add_from_devices(p, xdevs, xdev_count, &have_hmd);
add_from_auto_probers(p, xdevs, xdev_count, &have_hmd);
break;
case U_ACTIVE_CONFIG_REMOTE: assert(false); // Should never get here.
default: assert(false);
}
2019-04-30 13:33:34 +00:00
// It's easier if we just put the first hmd first,
// but keep other internal ordering of devices.
for (size_t i = 1; i < xdev_count; i++) {
if (xdevs[i] == NULL) {
continue;
}
if (xdevs[i]->hmd == NULL) {
continue;
}
// This is a HMD, but it's not in the first slot.
struct xrt_device *hmd = xdevs[i];
for (size_t k = i; k > 0; k--) {
xdevs[k] = xdevs[k - 1];
}
xdevs[0] = hmd;
break;
}
2019-05-31 16:55:19 +00:00
struct xrt_tracking_override overrides[XRT_MAX_TRACKING_OVERRIDES];
size_t num_overrides = 0;
if (u_config_json_get_tracking_overrides(&p->json, overrides, &num_overrides)) {
for (size_t i = 0; i < num_overrides; i++) {
struct xrt_tracking_override *o = &overrides[i];
apply_tracking_override(p, xdevs, xdev_count, o);
}
}
if (have_hmd) {
P_DEBUG(p, "Found HMD! '%s'", xdevs[0]->str);
2019-05-31 16:55:19 +00:00
return 0;
}
P_DEBUG(p, "Didn't find any HMD devices");
// Even if we've found some controllers, we don't use them without an
// HMD. So, destroy all other found devices.
for (size_t i = 1; i < xdev_count; i++) {
2019-05-31 16:55:19 +00:00
if (xdevs[i] == NULL) {
continue;
}
P_DEBUG(p, "Destroying '%s'", xdevs[i]->str);
xrt_device_destroy(&xdevs[i]);
2019-05-31 16:55:19 +00:00
}
return 0;
2019-04-30 13:33:34 +00:00
}
static int
2021-09-29 21:28:23 +00:00
p_open_hid_interface(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
int interface,
struct os_hid_device **out_hid_dev)
2019-04-30 13:33:34 +00:00
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
struct prober_device *pdev = (struct prober_device *)xpdev;
2022-04-30 19:16:43 +00:00
int ret;
2019-04-30 13:33:34 +00:00
2022-04-30 19:16:43 +00:00
#if defined(XRT_OS_LINUX)
2019-04-30 13:33:34 +00:00
for (size_t j = 0; j < pdev->num_hidraws; j++) {
struct prober_hidraw *hidraw = &pdev->hidraws[j];
2019-04-30 13:33:34 +00:00
if (hidraw->interface != interface) {
continue;
}
ret = os_hid_open_hidraw(hidraw->path, out_hid_dev);
if (ret != 0) {
2021-01-14 14:13:48 +00:00
U_LOG_E("Failed to open device '%s' got '%i'", hidraw->path, ret);
2019-04-30 13:33:34 +00:00
return ret;
}
return 0;
}
2022-04-30 19:17:29 +00:00
U_LOG_E("Could not find the requested hid interface (%i) on the device!", interface);
2019-04-30 13:33:34 +00:00
return -1;
2022-04-30 19:16:43 +00:00
#elif defined(XRT_OS_WINDOWS)
(void)ret;
U_LOG_E("HID devices not yet supported on Windows, can not open interface (%i)", interface);
2022-04-30 19:16:43 +00:00
return -1;
#else
#error "no port of hid code"
#endif
2019-04-30 13:33:34 +00:00
}
static int
2021-09-29 21:28:23 +00:00
p_open_video_device(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
struct xrt_frame_context *xfctx,
struct xrt_fs **out_xfs)
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
2021-01-14 14:13:48 +00:00
XRT_MAYBE_UNUSED struct prober_device *pdev = (struct prober_device *)xpdev;
#if defined(XRT_BUILD_DRIVER_VF)
2020-11-26 19:16:02 +00:00
const char *path = debug_get_option_vf_path();
if (path != NULL) {
struct xrt_fs *xfs = vf_fs_open_file(xfctx, path);
2020-11-26 19:16:02 +00:00
if (xfs) {
*out_xfs = xfs;
return 0;
}
}
#endif
#if defined(XRT_BUILD_DRIVER_EUROC)
const char *euroc_path = debug_get_option_euroc_path();
if (euroc_path != NULL) {
*out_xfs = euroc_player_create(xfctx, euroc_path, NULL); // Euroc will exit if it can't be created
return 0;
}
#endif
2021-12-06 20:53:00 +00:00
#if defined(XRT_BUILD_DRIVER_REALSENSE)
int rs_source_index = debug_get_num_option_rs_source_index();
if (rs_source_index != -1) {
*out_xfs = rs_source_create(xfctx, rs_source_index);
return 0;
}
#endif
2019-11-10 01:36:34 +00:00
#if defined(XRT_HAVE_V4L2)
if (pdev->num_v4ls == 0) {
return -1;
}
2020-09-06 12:31:10 +00:00
struct xrt_fs *xfs =
v4l2_fs_create(xfctx, pdev->v4ls[0].path, pdev->usb.product, pdev->usb.manufacturer, pdev->usb.serial);
if (xfs == NULL) {
return -1;
}
*out_xfs = xfs;
return 0;
#else
return -1;
#endif
}
static int
2022-05-11 12:07:28 +00:00
p_list_video_devices(struct xrt_prober *xp, xrt_prober_list_video_func_t cb, void *ptr)
{
struct prober *p = (struct prober *)xp;
// Video sources from drivers (at most one will be listed)
const char *vf_path = debug_get_option_vf_path();
const char *euroc_path = debug_get_option_euroc_path();
int rs_source_index = debug_get_num_option_rs_source_index();
if (vf_path != NULL) {
cb(xp, NULL, "Video File", "Collabora", vf_path, ptr);
} else if (euroc_path != NULL) {
cb(xp, NULL, "Euroc Dataset", "Collabora", euroc_path, ptr);
} else if (rs_source_index != -1) {
cb(xp, NULL, "RealSense Source", "Collabora", "", ptr);
}
// Video sources from video devices
2021-11-08 22:54:01 +00:00
for (size_t i = 0; i < p->device_count; i++) {
struct prober_device *pdev = &p->devices[i];
bool has = false;
#ifdef XRT_HAVE_LIBUVC
has |= pdev->uvc.dev != NULL;
#endif
#ifdef XRT_HAVE_V4L2
has |= pdev->num_v4ls > 0;
#endif
if (!has) {
continue;
}
if (pdev->usb.product == NULL) {
fill_out_product(p, pdev);
}
2021-01-14 14:13:48 +00:00
cb(xp, &pdev->base, pdev->usb.product, pdev->usb.manufacturer, pdev->usb.serial, ptr);
}
return 0;
}
static int
2021-09-29 21:28:23 +00:00
p_get_entries(struct xrt_prober *xp,
size_t *out_num_entries,
struct xrt_prober_entry ***out_entries,
struct xrt_auto_prober ***out_auto_probers)
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
*out_num_entries = p->num_entries;
*out_entries = p->entries;
*out_auto_probers = p->auto_probers;
2021-09-29 21:22:24 +00:00
return 0;
}
static int
2021-09-29 21:28:23 +00:00
p_get_string_descriptor(struct xrt_prober *xp,
struct xrt_prober_device *xpdev,
enum xrt_prober_string which_string,
unsigned char *buffer,
size_t max_length)
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
XRT_MAYBE_UNUSED struct prober *p = (struct prober *)xp;
2021-11-10 15:42:12 +00:00
struct prober_device *pdev = (struct prober_device *)xpdev;
int ret = 0;
#ifdef XRT_HAVE_LIBUSB
if (pdev->base.bus == XRT_BUS_TYPE_USB && pdev->usb.dev != NULL) {
2022-04-30 19:16:43 +00:00
assert(max_length < INT_MAX);
ret = p_libusb_get_string_descriptor(p, pdev, which_string, buffer, (int)max_length);
if (ret >= 0) {
return ret;
}
}
#else
if (pdev->base.bus == XRT_BUS_TYPE_USB) {
P_WARN(p, "Can not get usb descriptors (libusb-dev not installed)!");
return ret;
}
#endif
if (pdev && pdev->base.bus == XRT_BUS_TYPE_BLUETOOTH) {
switch (which_string) {
case XRT_PROBER_STRING_SERIAL_NUMBER: {
union {
uint8_t arr[8];
uint64_t v;
} u;
u.v = pdev->bluetooth.id;
ret = snprintf((char *)buffer, max_length, "%02X:%02X:%02X:%02X:%02X:%02X", u.arr[5], u.arr[4],
u.arr[3], u.arr[2], u.arr[1], u.arr[0]);
}; break;
case XRT_PROBER_STRING_PRODUCT:
ret = snprintf((char *)buffer, max_length, "%s", pdev->base.product_name);
break;
default: ret = 0; break;
}
}
//! @todo add more backends
//! @todo make this unicode (utf-16)? utf-8 would be better...
return ret;
}
static bool
2021-09-29 21:28:23 +00:00
p_can_open(struct xrt_prober *xp, struct xrt_prober_device *xpdev)
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)xp;
struct prober_device *pdev = (struct prober_device *)xpdev;
bool has_been_queried = false;
#ifdef XRT_HAVE_LIBUSB
has_been_queried = true;
if (pdev->usb.dev != NULL) {
return p_libusb_can_open(p, pdev);
}
#endif
// No backend compiled in to judge the ability to open the device.
if (!has_been_queried) {
P_WARN(p, "Can not tell if '%s' can be opened, assuming yes!", pdev->usb.product);
return true;
}
//! @todo add more backends
return false;
}
2019-04-30 13:33:34 +00:00
static void
2021-09-29 21:28:23 +00:00
p_destroy(struct xrt_prober **xp)
2019-04-30 13:33:34 +00:00
{
2021-09-29 21:22:24 +00:00
XRT_TRACE_MARKER();
struct prober *p = (struct prober *)*xp;
2019-04-30 13:33:34 +00:00
if (p == NULL) {
return;
}
teardown(p);
free(p);
*xp = NULL;
}