monado/src/xrt/state_trackers/prober/p_prober.c
2019-06-18 14:14:01 -05:00

643 lines
13 KiB
C

// Copyright 2019, Collabora, Ltd.
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief Main prober code.
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup st_prober
*/
#include "util/u_debug.h"
#include "util/u_misc.h"
#include "p_prober.h"
#include <stdio.h>
#include <string.h>
/*
*
* Pre-declare functions.
*
*/
DEBUG_GET_ONCE_BOOL_OPTION(prober_spew, "PROBER_PRINT_SPEW", false)
DEBUG_GET_ONCE_BOOL_OPTION(prober_debug, "PROBER_PRINT_DEBUG", false)
static void
add_device(struct prober* p, struct prober_device** out_dev);
static int
initialize(struct prober* p, struct xrt_prober_entry_lists* lists);
static void
teardown_devices(struct prober* p);
static void
teardown(struct prober* p);
static int
probe(struct xrt_prober* xp);
static int
dump(struct xrt_prober* xp);
static int
select_device(struct xrt_prober* xp,
struct xrt_device** xdevs,
size_t num_xdevs);
static int
open_hid_interface(struct xrt_prober* xp,
struct xrt_prober_device* xpdev,
int interface,
struct os_hid_device** out_hid_dev);
static void
destroy(struct xrt_prober** xp);
/*
*
* "Exported" functions.
*
*/
int
xrt_prober_create_with_lists(struct xrt_prober** out_xp,
struct xrt_prober_entry_lists* lists)
{
struct prober* p = U_TYPED_CALLOC(struct prober);
int ret = initialize(p, lists);
if (ret != 0) {
free(p);
return ret;
}
*out_xp = &p->base;
return 0;
}
int
p_dev_get_usb_dev(struct prober* p,
uint16_t bus,
uint16_t addr,
uint16_t vendor_id,
uint16_t product_id,
struct prober_device** out_pdev)
{
struct prober_device* pdev;
for (size_t i = 0; i < p->num_devices; i++) {
struct prober_device* pdev = &p->devices[i];
if (pdev->base.bus != XRT_BUS_TYPE_USB ||
pdev->usb.bus != bus || pdev->usb.addr != addr) {
continue;
}
if (pdev->base.vendor_id != vendor_id ||
pdev->base.product_id != product_id) {
P_ERROR(p,
"USB device with same address but different "
"vendor and product found!\n"
"\tvendor: %04x %04x\n"
"\tproduct: %04x %04x",
pdev->base.vendor_id, vendor_id,
pdev->base.product_id, product_id);
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
p_dev_get_bluetooth_dev(struct prober* p,
uint64_t id,
uint16_t vendor_id,
uint16_t product_id,
struct prober_device** out_pdev)
{
struct prober_device* pdev;
for (size_t i = 0; i < p->num_devices; i++) {
struct prober_device* pdev = &p->devices[i];
if (pdev->base.bus != XRT_BUS_TYPE_BLUETOOTH ||
pdev->bluetooth.id != id) {
continue;
}
if (pdev->base.vendor_id != vendor_id ||
pdev->base.product_id != product_id) {
P_ERROR(p,
"Bluetooth device with same address but "
"different vendor and product found!\n"
"\tvendor: %04x %04x\n"
"\tproduct: %04x %04x",
pdev->base.vendor_id, vendor_id,
pdev->base.product_id, product_id);
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
add_device(struct prober* p, struct prober_device** out_dev)
{
size_t new_size = (p->num_devices + 1) * sizeof(struct prober_device);
p->devices = realloc(p->devices, new_size);
struct prober_device* dev = &p->devices[p->num_devices++];
U_ZERO(dev);
*out_dev = dev;
}
static void
add_usb_entry(struct prober* p, struct xrt_prober_entry* entry)
{
size_t new_size =
(p->num_entries + 1) * sizeof(struct xrt_prober_entry_usb*);
p->entries = realloc(p->entries, new_size);
p->entries[p->num_entries++] = entry;
}
static int
collect_entries(struct prober* p)
{
struct xrt_prober_entry_lists* lists = p->lists;
while (lists) {
for (size_t j = 0; lists->entries != NULL && lists->entries[j];
j++) {
struct xrt_prober_entry* entry = lists->entries[j];
for (size_t k = 0; entry[k].found != NULL; k++) {
add_usb_entry(p, &entry[k]);
}
}
lists = lists->next;
}
return 0;
}
static int
initialize(struct prober* p, struct xrt_prober_entry_lists* lists)
{
p->base.probe = probe;
p->base.dump = dump;
p->base.select = select_device;
p->base.open_hid_interface = open_hid_interface;
p->base.destroy = destroy;
p->lists = lists;
p->print_spew = debug_get_bool_option_prober_spew();
p->print_debug = debug_get_bool_option_prober_debug();
int ret;
ret = collect_entries(p);
if (ret != 0) {
teardown(p);
return -1;
}
ret = libusb_init(&p->usb.ctx);
if (ret != 0) {
teardown(p);
return -1;
}
#ifdef XRT_HAVE_LIBUVC
ret = uvc_init(&p->uvc.ctx, p->usb.ctx);
if (ret != 0) {
teardown(p);
return -1;
}
#endif
for (int i = 0; i < MAX_AUTO_PROBERS && lists->auto_probers[i]; i++) {
p->auto_probers[i] = lists->auto_probers[i]();
}
return 0;
}
static void
teardown_devices(struct prober* p)
{
// for (size_t i; i)
// Need to free all devices.
for (size_t i = 0; i < p->num_devices; i++) {
struct prober_device* pdev = &p->devices[i];
for (size_t j = 0; j < pdev->num_hidraws; j++) {
struct prober_hidraw* hidraw = &pdev->hidraws[j];
free((char*)hidraw->path);
hidraw->path = NULL;
}
if (pdev->hidraws != NULL) {
free(pdev->hidraws);
pdev->hidraws = NULL;
pdev->num_hidraws = 0;
}
}
if (p->devices != NULL) {
free(p->devices);
p->devices = NULL;
p->num_devices = 0;
}
}
static void
teardown(struct prober* p)
{
// Clean up all auto_probers.
for (int i = 0; i < MAX_AUTO_PROBERS && p->auto_probers[i]; i++) {
p->auto_probers[i]->destroy(p->auto_probers[i]);
p->auto_probers[i] = NULL;
}
// 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
// Free all libuvc resources.
if (p->uvc.list != NULL) {
uvc_free_device_list(p->uvc.list, 1);
p->uvc.list = NULL;
}
if (p->uvc.ctx != NULL) {
uvc_exit(p->uvc.ctx);
p->uvc.ctx = NULL;
}
#endif
// Free all libusb resources.
if (p->usb.list != NULL) {
libusb_free_device_list(p->usb.list, 1);
p->usb.list = NULL;
}
if (p->usb.ctx != NULL) {
libusb_exit(p->usb.ctx);
p->usb.ctx = NULL;
}
}
static int
p_libusb_probe(struct prober* p)
{
int ret;
// Free old list first.
if (p->usb.list != NULL) {
libusb_free_device_list(p->usb.list, 1);
p->usb.list = NULL;
}
// Probe for USB devices.
p->usb.count = libusb_get_device_list(p->usb.ctx, &p->usb.list);
if (p->usb.count < 0) {
P_ERROR(p, "\tFailed to enumerate usb devices\n");
return -1;
}
for (ssize_t i = 0; i < p->usb.count; i++) {
libusb_device* device = p->usb.list[i];
struct libusb_device_descriptor desc;
struct prober_device* pdev = NULL;
libusb_get_device_descriptor(device, &desc);
uint8_t bus = libusb_get_bus_number(device);
uint8_t addr = libusb_get_device_address(device);
uint16_t vendor = desc.idVendor;
uint16_t product = desc.idProduct;
ret = p_dev_get_usb_dev(p, bus, addr, vendor, product, &pdev);
P_SPEW(p,
"libusb\n"
"\t\tptr: %p (%i)\n"
"\t\tvendor_id: %04x\n"
"\t\tproduct_id: %04x\n"
"\t\tbus: %i\n"
"\t\taddr: %i",
(void*)pdev, ret, vendor, product, bus, addr);
if (ret != 0) {
P_ERROR(p, "p_dev_get_usb_device failed!");
continue;
}
// Attach the libusb device to it.
pdev->usb.dev = device;
}
return 0;
}
static int
p_libuvc_probe(struct prober* p)
{
#ifdef XRT_HAVE_LIBUVC
int ret;
// Free old list first.
if (p->uvc.list != NULL) {
uvc_free_device_list(p->uvc.list, 1);
p->uvc.list = NULL;
}
ret = uvc_get_device_list(p->uvc.ctx, &p->uvc.list);
if (ret < 0) {
P_ERROR(p, "\tFailed to enumerate uvc devices\n");
return -1;
}
// Count the number of UVC devices.
while (p->uvc.list != NULL && p->uvc.list[p->uvc.count] != NULL) {
p->uvc.count++;
}
for (ssize_t k = 0; k < p->uvc.count; k++) {
uvc_device_t* device = p->uvc.list[k];
struct uvc_device_descriptor* desc;
struct prober_device* pdev = NULL;
uvc_get_device_descriptor(device, &desc);
uint8_t bus = uvc_get_bus_number(device);
uint8_t addr = uvc_get_device_address(device);
uint16_t vendor = desc->idVendor;
uint16_t product = desc->idProduct;
uvc_free_device_descriptor(desc);
ret = p_dev_get_usb_dev(p, bus, addr, vendor, product, &pdev);
P_SPEW(p,
"libuvc\n"
"\t\tptr: %p (%i)\n"
"\t\tvendor_id: %04x\n"
"\t\tproduct_id: %04x\n"
"\t\tbus: %i\n"
"\t\taddr: %i",
(void*)pdev, ret, vendor, product, bus, addr);
if (ret != 0) {
P_ERROR(p, "p_dev_get_usb_device failed!");
continue;
}
// Attach the libuvc device to it.
pdev->uvc.dev = p->uvc.list[k];
}
#endif
return 0;
}
/*
*
* Member functions.
*
*/
static int
probe(struct xrt_prober* xp)
{
struct prober* p = (struct prober*)xp;
XRT_MAYBE_UNUSED int ret = 0;
// Free old list first.
teardown_devices(p);
ret = p_libusb_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate libusb devices\n");
return -1;
}
ret = p_libuvc_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate libuvc devices\n");
return -1;
}
ret = p_udev_probe(p);
if (ret != 0) {
P_ERROR(p, "Failed to enumerate udev devices\n");
return -1;
}
return 0;
}
static int
dump(struct xrt_prober* xp)
{
struct prober* p = (struct prober*)xp;
XRT_MAYBE_UNUSED ssize_t k = 0;
XRT_MAYBE_UNUSED size_t j = 0;
for (size_t i = 0; i < p->num_devices; i++) {
struct prober_device* pdev = &p->devices[i];
p_dump_device(p, pdev, (int)i);
}
return 0;
}
static void
handle_found_device(struct prober* p,
struct xrt_device** xdevs,
size_t num_xdevs,
struct xrt_device* xdev)
{
P_DEBUG(p, "Found '%s' %p", xdev->name, (void*)xdev);
// For controllers we put them after the first found HMD.
if (xdev->hmd == NULL) {
for (size_t i = 1; i < num_xdevs; i++) {
if (xdevs[i] == NULL) {
xdevs[i] = xdev;
return;
}
}
P_ERROR(p, "Too many controller devices closing '%s'",
xdev->name);
xdev->destroy(xdev);
return;
}
// Not found a HMD before, add it first in the list.
if (xdevs[0] == NULL) {
xdevs[0] = xdev;
return;
}
P_ERROR(p, "Found more then one, HMD closing '%s'", xdev->name);
xdev->destroy(xdev);
return;
}
static int
select_device(struct xrt_prober* xp,
struct xrt_device** xdevs,
size_t num_xdevs)
{
struct prober* p = (struct prober*)xp;
// Build a list of all current probed devices.
struct xrt_prober_device** dev_list =
U_TYPED_ARRAY_CALLOC(struct xrt_prober_device*, p->num_devices);
for (size_t i = 0; i < p->num_devices; i++) {
dev_list[i] = &p->devices[i].base;
}
// Loop over all devices and entries that might match them.
for (size_t i = 0; i < p->num_devices; i++) {
struct prober_device* pdev = &p->devices[i];
for (size_t k = 0; k < p->num_entries; k++) {
struct xrt_prober_entry* entry = p->entries[k];
if (pdev->base.vendor_id != entry->vendor_id ||
pdev->base.product_id != entry->product_id) {
continue;
}
struct xrt_device* xdev = NULL;
entry->found(xp, dev_list, i, &xdev);
if (xdev == NULL) {
continue;
}
handle_found_device(p, xdevs, num_xdevs, xdev);
}
}
// Free the temporary list.
free(dev_list);
for (int i = 0; i < MAX_AUTO_PROBERS && p->auto_probers[i]; i++) {
struct xrt_device* xdev =
p->auto_probers[i]->lelo_dallas_autoprobe(
p->auto_probers[i]);
if (xdev == NULL) {
continue;
}
handle_found_device(p, xdevs, num_xdevs, xdev);
}
if (xdevs[0] != NULL) {
P_DEBUG(p, "Found HMD! '%s'", xdevs[0]->name);
return 0;
}
P_DEBUG(p, "Didn't find any HMD devices");
// Destroy all other found devices.
for (size_t i = 1; i < num_xdevs; i++) {
if (xdevs[i] == NULL) {
continue;
}
P_DEBUG(p, "Destroying '%s'", xdevs[i]->name);
xdevs[i]->destroy(xdevs[i]);
xdevs[i] = NULL;
}
return 0;
}
static int
open_hid_interface(struct xrt_prober* xp,
struct xrt_prober_device* xpdev,
int interface,
struct os_hid_device** out_hid_dev)
{
struct prober_device* pdev = (struct prober_device*)xpdev;
int ret;
for (size_t j = 0; j < pdev->num_hidraws; j++) {
struct prober_hidraw* hidraw = &pdev->hidraws[j];
if (hidraw->interface != interface) {
continue;
}
ret = os_hid_open_hidraw(hidraw->path, out_hid_dev);
if (ret != 0) {
P_ERROR(p, "Failed to open device!");
return ret;
}
return 0;
}
P_ERROR(p,
"Could not find the requested "
"hid interface (%i) on the device!",
interface);
return -1;
}
static void
destroy(struct xrt_prober** xp)
{
struct prober* p = (struct prober*)*xp;
if (p == NULL) {
return;
}
teardown(p);
free(p);
*xp = NULL;
}