// Copyright 2021, Jan Schmidt // Copyright 2021, Philipp Zabel // Copyright 2021, Jakob Bornecrantz // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief WMR camera interface * @author Jan Schmidt * @author Philipp Zabel * @author Jakob Bornecrantz * @ingroup drv_wmr */ #include #include #include #include "os/os_threading.h" #include "util/u_var.h" #include "util/u_sink.h" #include "util/u_frame.h" #include "wmr_protocol.h" #include "wmr_camera.h" /* * * Defines and structs. * */ #define WMR_CAM_TRACE(c, ...) U_LOG_IFL_T((c)->log_level, __VA_ARGS__) #define WMR_CAM_DEBUG(c, ...) U_LOG_IFL_D((c)->log_level, __VA_ARGS__) #define WMR_CAM_INFO(c, ...) U_LOG_IFL_I((c)->log_level, __VA_ARGS__) #define WMR_CAM_WARN(c, ...) U_LOG_IFL_W((c)->log_level, __VA_ARGS__) #define WMR_CAM_ERROR(c, ...) U_LOG_IFL_E((c)->log_level, __VA_ARGS__) #define CAM_ENDPOINT 0x05 #define NUM_XFERS 2 #define WMR_CAMERA_CMD_GAIN 0x80 #define WMR_CAMERA_CMD_ON 0x81 #define WMR_CAMERA_CMD_OFF 0x82 #define DEFAULT_GAIN 0x60 struct wmr_camera_active_cmd { __le32 magic; __le32 len; __le32 cmd; } __attribute__((packed)); struct wmr_camera_gain_cmd { __le32 magic; __le32 len; __le16 cmd; __le16 camera_id; __le16 const_6000; __le16 gain; /* observed 82 to 255 */ __le16 camera_id2; /* same as camera_id */ } __attribute__((packed)); struct wmr_camera { libusb_context *ctx; libusb_device_handle *dev; bool running; struct os_thread_helper usb_thread; int usb_complete; const struct wmr_camera_config *configs; int config_count; size_t xfer_size; uint32_t frame_width, frame_height; uint8_t last_seq; struct libusb_transfer *xfers[NUM_XFERS]; struct u_sink_debug debug_sinks[2]; enum u_logging_level log_level; }; /* * * Helper functions. * */ /* Some WMR headsets use 616538 byte transfers. HP G2 needs 1233018 (4 cameras) * As a general formula, it seems we have: * 0x6000 byte packets. Each has a 32 byte header. * packet contains frame data for each camera in turn. * Each frame has an extra (first) line with metadata * Then, there's an extra 26 bytes on the end. * * F = camera frames X * (Y+1) + 26 * n_packets = F/(0x6000-32) * leftover = F - n_packets*(0x6000-32) * size = n_packets * 0x6000 + 32 + leftover, * * so for 2 x 640x480 cameras: * F = 2 * 640 * 481 + 26 = 615706 * n_packets = 615706 / 24544 = 25 * leftover = 615706 - 25 * 24544 = 2106 * size = 25 * 0x6000 + 32 + 2106 = 616538 * * For HP G2 = 4 x 640 * 480 cameras: * F = 4 * 640 * 481 + 26 = 1231386 * n_packets = 1231386 / 24544 = 50 * leftover = 1231386 - 50 * 24544 = 4186 * size = 50 * 0x6000 + 32 + 4186 = 1233018 * * It would be good to test these calculations on other headsets with * different camera setups. */ static bool compute_frame_size(struct wmr_camera *cam) { int i, cams_found = 0; int width, height; size_t F, n_packets, leftover; F = 26; for (i = 0; i < cam->config_count; i++) { const struct wmr_camera_config *config = &cam->configs[i]; if (config->purpose != WMR_CAMERA_PURPOSE_HEAD_TRACKING) { continue; } WMR_CAM_DEBUG(cam, "Found head tracking camera index %d width %d height %d", i, config->sensor_width, config->sensor_height); if (cams_found == 0) { width = config->sensor_width; height = config->sensor_height; } else if (height != config->sensor_height) { WMR_CAM_ERROR(cam, "Head tracking sensors have mismatched heights - %u != %u. Please report", height, config->sensor_height); return false; } else { width += config->sensor_width; } cams_found++; F += config->sensor_width * (config->sensor_height + 1); } if (cams_found == 0) { return false; } if (width < 1280 || height < 480) { return false; } n_packets = F / (0x6000 - 32); leftover = F - n_packets * (0x6000 - 32); cam->xfer_size = n_packets * 0x6000 + 32 + leftover; cam->frame_width = width; cam->frame_height = height; WMR_CAM_INFO(cam, "WMR camera framebuffer %u x %u - %zu transfer size", cam->frame_width, cam->frame_height, cam->xfer_size); return true; } static void * wmr_cam_usb_thread(void *ptr) { struct wmr_camera *cam = ptr; os_thread_helper_lock(&cam->usb_thread); while (os_thread_helper_is_running_locked(&cam->usb_thread) && !cam->usb_complete) { os_thread_helper_unlock(&cam->usb_thread); libusb_handle_events_completed(cam->ctx, &cam->usb_complete); os_thread_helper_lock(&cam->usb_thread); } os_thread_helper_wait_locked(&cam->usb_thread); os_thread_helper_unlock(&cam->usb_thread); return NULL; } static int send_buffer_to_device(struct wmr_camera *cam, uint8_t *buf, uint8_t len) { struct libusb_transfer *xfer; uint8_t *data; xfer = libusb_alloc_transfer(0); if (xfer == NULL) { return LIBUSB_ERROR_NO_MEM; } data = malloc(len); if (data == NULL) { libusb_free_transfer(xfer); return LIBUSB_ERROR_NO_MEM; } memcpy(data, buf, len); libusb_fill_bulk_transfer(xfer, cam->dev, CAM_ENDPOINT | LIBUSB_ENDPOINT_OUT, data, len, NULL, NULL, 0); xfer->flags |= LIBUSB_TRANSFER_FREE_BUFFER | LIBUSB_TRANSFER_FREE_TRANSFER; return libusb_submit_transfer(xfer); } static int set_active(struct wmr_camera *cam, bool active) { struct wmr_camera_active_cmd cmd = { .magic = __cpu_to_le32(WMR_MAGIC), .len = __cpu_to_le32(sizeof(struct wmr_camera_active_cmd)), .cmd = __cpu_to_le32(active ? WMR_CAMERA_CMD_ON : WMR_CAMERA_CMD_OFF), }; return send_buffer_to_device(cam, (uint8_t *)&cmd, sizeof(cmd)); } static void LIBUSB_CALL img_xfer_cb(struct libusb_transfer *xfer) { struct wmr_camera *cam = xfer->user_data; if (xfer->status != LIBUSB_TRANSFER_COMPLETED) { WMR_CAM_TRACE(cam, "Camera transfer completed with status %u", xfer->status); goto out; } if (xfer->actual_length < xfer->length) { WMR_CAM_DEBUG(cam, "Camera transfer only delivered %d bytes", xfer->actual_length); goto out; } WMR_CAM_TRACE(cam, "Camera transfer complete - %d bytes of %d", xfer->actual_length, xfer->length); /* Convert the output into frames and send them off to debug / tracking */ struct xrt_frame *xf = NULL; /* There's always one extra line of pixels with exposure info */ u_frame_create_one_off(XRT_FORMAT_L8, cam->frame_width, cam->frame_height + 1, &xf); const uint8_t *src = xfer->buffer; uint8_t *dst = xf->data; size_t dst_remain = xf->size; const size_t chunk_size = 0x6000 - 32; while (dst_remain > 0) { const size_t to_copy = dst_remain > chunk_size ? chunk_size : dst_remain; /* TODO: See if there is useful info in the 32 byte packet headers. * There seems to be a counter or timestamp there at least */ src += 0x20; memcpy(dst, src, to_copy); src += to_copy; dst += to_copy; dst_remain -= to_copy; } uint16_t exposure = xf->data[6] << 8 | xf->data[7]; uint8_t seq = xf->data[89]; WMR_CAM_TRACE(cam, "Camera frame seq %u (prev %u) - exposure %u", seq, cam->last_seq, exposure); /* Exposure of 0 is a dark frame for controller tracking */ int sink_index = (exposure == 0) ? 1 : 0; if (u_sink_debug_is_active(&cam->debug_sinks[sink_index])) { u_sink_debug_push_frame(&cam->debug_sinks[sink_index], xf); } /* TODO: Push frame for tracking */ xrt_frame_reference(&xf, NULL); cam->last_seq = seq; out: libusb_submit_transfer(xfer); } /* * * 'Exported' functions. * */ struct wmr_camera * wmr_camera_open(struct xrt_prober_device *dev_holo, enum u_logging_level ll) { struct wmr_camera *cam = calloc(1, sizeof(struct wmr_camera)); int res, i; cam->log_level = ll; if (os_thread_helper_init(&cam->usb_thread) != 0) { WMR_CAM_ERROR(cam, "Failed to initialise threading"); wmr_camera_free(cam); return NULL; } res = libusb_init(&cam->ctx); if (res < 0) { goto fail; } cam->dev = libusb_open_device_with_vid_pid(cam->ctx, dev_holo->vendor_id, dev_holo->product_id); if (cam->dev == NULL) { goto fail; } res = libusb_claim_interface(cam->dev, 3); if (res < 0) { goto fail; } cam->usb_complete = 0; if (os_thread_helper_start(&cam->usb_thread, wmr_cam_usb_thread, cam) != 0) { WMR_CAM_ERROR(cam, "Failed to start camera USB thread"); goto fail; } for (i = 0; i < NUM_XFERS; i++) { cam->xfers[i] = libusb_alloc_transfer(0); if (cam->xfers[i] == NULL) { res = LIBUSB_ERROR_NO_MEM; goto fail; } } u_sink_debug_init(&cam->debug_sinks[0]); u_sink_debug_init(&cam->debug_sinks[1]); u_var_add_root(cam, "WMR Camera", true); u_var_add_log_level(cam, &cam->log_level, "Log level"); u_var_add_sink_debug(cam, &cam->debug_sinks[0], "SLAM"); u_var_add_sink_debug(cam, &cam->debug_sinks[1], "Controllers"); return cam; fail: WMR_CAM_ERROR(cam, "Failed to open camera: %s", libusb_error_name(res)); wmr_camera_free(cam); return NULL; } void wmr_camera_free(struct wmr_camera *cam) { // Stop the camera. wmr_camera_stop(cam); // Tidy the variable tracking. u_var_remove_root(cam); u_sink_debug_destroy(&cam->debug_sinks[0]); u_sink_debug_destroy(&cam->debug_sinks[1]); if (cam->ctx != NULL) { int i; os_thread_helper_lock(&cam->usb_thread); cam->usb_complete = 1; os_thread_helper_unlock(&cam->usb_thread); if (cam->dev != NULL) { libusb_close(cam->dev); } os_thread_helper_destroy(&cam->usb_thread); for (i = 0; i < NUM_XFERS; i++) { if (cam->xfers[i] != NULL) libusb_free_transfer(cam->xfers[i]); } libusb_exit(cam->ctx); } free(cam); } bool wmr_camera_start(struct wmr_camera *cam, const struct wmr_camera_config *cam_configs, int config_count) { int res, i; cam->configs = cam_configs; cam->config_count = config_count; if (!compute_frame_size(cam)) { WMR_CAM_WARN(cam, "Invalid config or no head tracking cameras found"); goto fail; } res = set_active(cam, false); if (res < 0) { goto fail; } res = set_active(cam, true); if (res < 0) { goto fail; } for (i = 0; i < cam->config_count; i++) { const struct wmr_camera_config *config = &cam->configs[i]; if (config->purpose != WMR_CAMERA_PURPOSE_HEAD_TRACKING) { continue; } res = wmr_camera_set_gain(cam, config->location, DEFAULT_GAIN); if (res < 0) { WMR_CAM_ERROR(cam, "Failed to set initial gain for camera %d", i); goto fail; } } for (i = 0; i < NUM_XFERS; i++) { uint8_t *recv_buf = malloc(cam->xfer_size); libusb_fill_bulk_transfer(cam->xfers[i], cam->dev, 0x85, recv_buf, cam->xfer_size, img_xfer_cb, cam, 0); cam->xfers[i]->flags |= LIBUSB_TRANSFER_FREE_BUFFER; res = libusb_submit_transfer(cam->xfers[i]); if (res < 0) { goto fail; } } WMR_CAM_INFO(cam, "WMR camera started"); return true; fail: if (res < 0) WMR_CAM_ERROR(cam, "Error starting camera input: %s", libusb_error_name(res)); wmr_camera_stop(cam); return false; } bool wmr_camera_stop(struct wmr_camera *cam) { int res, i; if (!cam->running) { return true; } cam->running = false; for (i = 0; i < NUM_XFERS; i++) { if (cam->xfers[i] != NULL) { libusb_cancel_transfer(cam->xfers[i]); } } res = set_active(cam, false); if (res < 0) { goto fail; } WMR_CAM_INFO(cam, "WMR camera stopped"); return true; fail: if (res < 0) { WMR_CAM_ERROR(cam, "Error stopping camera input: %s", libusb_error_name(res)); } return false; } int wmr_camera_set_gain(struct wmr_camera *cam, uint8_t camera_id, uint8_t gain) { struct wmr_camera_gain_cmd cmd = { .magic = __cpu_to_le32(WMR_MAGIC), .len = __cpu_to_le32(sizeof(struct wmr_camera_gain_cmd)), .cmd = __cpu_to_le16(WMR_CAMERA_CMD_GAIN), .camera_id = __cpu_to_le16(camera_id), .const_6000 = __cpu_to_le16(6000), .gain = __cpu_to_le16(gain), .camera_id2 = __cpu_to_le16(camera_id), }; return send_buffer_to_device(cam, (uint8_t *)&cmd, sizeof(cmd)); }