Monado OpenXR Runtime
device.hpp
Go to the documentation of this file.
1// Copyright 2023, Shawn Wallace
2// SPDX-License-Identifier: BSL-1.0
3/*!
4 * @file
5 * @brief SteamVR driver device header - inherits xrt_device.
6 * @author Shawn Wallace <yungwallace@live.com>
7 * @ingroup drv_steamvr_lh
8 */
9
11
13
15#include "xrt/xrt_device.h"
16
17#include "vive/vive_common.h"
18
19#include "vp2/vp2_hid.h"
20
21#include "openvr_driver.h"
22
23#include <string>
24#include <vector>
25#include <memory>
26#include <unordered_map>
27#include <span>
28#include <array>
29#include <optional>
30
31#include <condition_variable>
32#include <mutex>
33
34class Context;
35struct InputClass;
36
38{
39 using ContextPtr = std::shared_ptr<Context>;
40 ContextPtr ctx;
41 vr::ITrackedDeviceServerDriver *driver;
42 const char *serial;
43 const std::string &steam_install;
44
45 DeviceBuilder(const ContextPtr &p_ctx,
46 vr::ITrackedDeviceServerDriver *p_driver,
47 const char *p_serial,
48 const std::string &p_stream_install)
49 : ctx{p_ctx}, driver{p_driver}, serial{p_serial}, steam_install{p_stream_install}
50 {}
51
52 // no copies!
53 DeviceBuilder(const DeviceBuilder &) = delete;
55 operator=(const DeviceBuilder &) = delete;
56};
57
59{
60public:
61 Property(vr::PropertyTypeTag_t tag, void *buffer, uint32_t bufferSize);
62
63 vr::PropertyTypeTag_t tag;
64 std::vector<uint8_t> buffer;
65};
66
67class Device : public xrt_device
68{
69
70public:
71 m_relation_history *relation_hist;
72
73 virtual ~Device();
74
75 xrt_input *
76 get_input_from_name(std::string_view name);
77
80
81 void
82 update_pose(const vr::DriverPose_t &newPose) const;
83
84 //! Helper to use the @ref m_relation_history member.
85 void
86 get_pose(uint64_t at_timestamp_ns, xrt_space_relation *out_relation);
87
88 vr::ETrackedPropertyError
89 handle_properties(const vr::PropertyWrite_t *batch, uint32_t count);
90
91 vr::ETrackedPropertyError
92 handle_read_properties(vr::PropertyRead_t *batch, uint32_t count);
93
94 //! Maps to @ref xrt_device::get_tracked_pose.
95 virtual xrt_result_t
96 get_tracked_pose(xrt_input_name name, uint64_t at_timestamp_ns, xrt_space_relation *out_relation) = 0;
97
99 get_battery_status(bool *out_present, bool *out_charging, float *out_charge);
100
101protected:
102 Device(const DeviceBuilder &builder);
103 std::shared_ptr<Context> ctx;
104 vr::PropertyContainerHandle_t container_handle{0};
105 std::unordered_map<vr::ETrackedDeviceProperty, Property> properties;
106 std::unordered_map<std::string_view, xrt_input *> inputs_map;
107 std::vector<xrt_input> inputs_vec;
108 inline static xrt_pose chaperone = XRT_POSE_IDENTITY;
109 const InputClass *input_class;
110 std::string manufacturer;
111 std::string model;
112 float vsync_to_photon_ns{0.f};
113 bool provides_battery_status{false};
114 bool charging{false};
115 float charge{1.0F};
116
117 vr::ETrackedPropertyError
118 handle_generic_property_write(const vr::PropertyWrite_t &prop);
119 vr::ETrackedPropertyError
120 handle_generic_property_read(vr::PropertyRead_t &prop);
121
122 virtual vr::ETrackedPropertyError
123 handle_property_write(const vr::PropertyWrite_t &prop);
124
125private:
126 vr::ITrackedDeviceServerDriver *driver;
127 uint64_t current_frame{0};
128
129 std::mutex frame_mutex;
130
131 void
132 init_chaperone(const std::string &steam_install);
133};
134
136{
137 vp2_hid *hid{nullptr};
138
140 {
141 if (hid != nullptr) {
142 vp2_hid_destroy(hid);
143 hid = nullptr;
144 }
145 }
146};
147
148class HmdDevice : public Device
149{
150public:
151 VIVE_VARIANT variant{VIVE_UNKNOWN};
152 xrt_pose eye[2] = {XRT_POSE_IDENTITY, XRT_POSE_IDENTITY};
153 float ipd{0.063}; // meters
154 struct Parts
155 {
156 xrt_hmd_parts base;
157 vr::IVRDisplayComponent *display;
158 };
159
161 {
162 float min{0.1f};
163 float max{1.0f};
164 };
165
167 {
168 };
169
170 HmdDevice(const DeviceBuilder &builder);
171
173 get_tracked_pose(xrt_input_name name, uint64_t at_timestamp_ns, xrt_space_relation *out_relation) override;
174
175 void
176 SetDisplayEyeToHead(uint32_t unWhichDevice,
177 const vr::HmdMatrix34_t &eyeToHeadLeft,
178 const vr::HmdMatrix34_t &eyeToHeadRight);
179
181 get_view_poses(const xrt_vec3 *default_eye_relation,
182 uint64_t at_timestamp_ns,
183 xrt_view_type view_type,
184 uint32_t view_count,
185 xrt_space_relation *out_head_relation,
186 xrt_fov *out_fovs,
187 xrt_pose *out_poses);
188
190 compute_distortion(uint32_t view, float u, float v, xrt_uv_triplet *out_result);
191
192 void
193 set_hmd_parts(std::unique_ptr<Parts> parts);
194
195 inline float
196 get_ipd() const
197 {
198 return ipd;
199 }
200
202 get_brightness(float *out_brightness);
204 set_brightness(float brightness, bool relative);
205
206 bool
207 init_vive_pro_2(struct xrt_prober *xp);
208
209private:
210 std::unique_ptr<Parts> hmd_parts{nullptr};
211
212 vr::ETrackedPropertyError
213 handle_property_write(const vr::PropertyWrite_t &prop) override;
214
215 void
216 set_nominal_frame_interval(uint64_t interval_ns);
217
218 void
219 set_scanout_type(enum xrt_scanout_direction direction, int64_t time_ns);
220
221 std::condition_variable hmd_parts_cv;
222 std::mutex hmd_parts_mut;
223 float brightness{1.0f};
224 AnalogGainRange analog_gain_range{};
225};
226
228{
229public:
230 ControllerDevice(vr::PropertyContainerHandle_t container_handle, const DeviceBuilder &builder);
231
234
235 void
236 set_haptic_handle(vr::VRInputComponentHandle_t handle);
237
239 get_tracked_pose(xrt_input_name name, uint64_t at_timestamp_ns, xrt_space_relation *out_relation) override;
240
243 int64_t desired_timestamp_ns,
244 struct xrt_hand_joint_set *out_value,
245 int64_t *out_timestamp_ns);
246
248 get_xrt_hand();
249
250 void
251 update_skeleton_transforms(std::span<const vr::VRBoneTransform_t> bones);
252
253 void
254 set_skeleton(std::span<const vr::VRBoneTransform_t> bones, xrt_hand hand, bool is_simulated, const char *path);
255
256 void
257 set_active_hand(xrt_hand hand);
258
259protected:
260 void
261 set_input_class(const InputClass *input_class);
262
263 void
264 generate_palm_pose_offset(std::span<const vr::VRBoneTransform_t> bones, xrt_hand hand);
265
266private:
267 vr::VRInputComponentHandle_t haptic_handle{0};
268 std::unique_ptr<xrt_output> output{nullptr};
269 bool has_hand_tracking{false};
270 xrt_hand skeleton_hand = XRT_HAND_LEFT;
271 std::array<std::optional<xrt_pose>, 2> palm_offsets;
272 std::array<xrt_input, 2> hand_tracking_inputs{};
273
274 struct JointsWithTimestamp
275 {
276 xrt_hand_joint_set joint_set;
277 int64_t timestamp{0};
278 };
280
281 vr::ETrackedPropertyError
282 handle_property_write(const vr::PropertyWrite_t &prop) override;
283};
Definition: context.hpp:42
Definition: device.hpp:228
Definition: device.hpp:68
void get_pose(uint64_t at_timestamp_ns, xrt_space_relation *out_relation)
Helper to use the m_relation_history member.
Definition: device.cpp:722
xrt_result_t update_inputs()
Update any attached inputs.
Definition: device.cpp:639
virtual xrt_result_t get_tracked_pose(xrt_input_name name, uint64_t at_timestamp_ns, xrt_space_relation *out_relation)=0
Maps to xrt_device::get_tracked_pose.
Definition: device.hpp:149
Definition: device.hpp:59
SteamVR driver context - implements xrt_tracking_origin and IVRDriverContext.
VIVE_VARIANT
Headset variant.
Definition: vive_common.h:42
xrt_hand
Enumeration for left and right hand.
Definition: xrt_defines.h:1400
xrt_input_name
Every internal input source known to monado with a baked in type.
Definition: xrt_defines.h:1344
enum xrt_result xrt_result_t
Result type used across Monado.
xrt_output_name
Name of a output with a baked in type.
Definition: xrt_defines.h:1974
Small utility for keeping track of the history of an xrt_space_relation, ie.
Definition: device.hpp:38
Definition: device.hpp:161
Definition: device.hpp:155
Definition: device.hpp:167
Definition: device.cpp:53
Definition: device.hpp:136
Definition: m_relation_history.cpp:49
Definition: vp2_hid.c:34
A single HMD or input device.
Definition: xrt_device.h:284
xrt_result_t(* set_output)(struct xrt_device *xdev, enum xrt_output_name name, const struct xrt_output_value *value)
Set a output value.
Definition: xrt_device.h:463
xrt_result_t(* compute_distortion)(struct xrt_device *xdev, uint32_t view, float u, float v, struct xrt_uv_triplet *out_result)
Compute the distortion at a single point.
Definition: xrt_device.h:591
xrt_result_t(* get_hand_tracking)(struct xrt_device *xdev, enum xrt_input_name name, int64_t desired_timestamp_ns, struct xrt_hand_joint_set *out_value, int64_t *out_timestamp_ns)
Get relationship of hand joints to the tracking origin space as the base space.
Definition: xrt_device.h:385
xrt_result_t(* set_brightness)(struct xrt_device *xdev, float brightness, bool relative)
Set the display brightness.
Definition: xrt_device.h:670
xrt_result_t(* get_battery_status)(struct xrt_device *xdev, bool *out_present, bool *out_charging, float *out_charge)
Get battery status information.
Definition: xrt_device.h:646
xrt_result_t(* get_brightness)(struct xrt_device *xdev, float *out_brightness)
Get the current display brightness.
Definition: xrt_device.h:658
xrt_result_t(* get_view_poses)(struct xrt_device *xdev, const struct xrt_vec3 *default_eye_relation, int64_t at_timestamp_ns, enum xrt_view_type view_type, uint32_t view_count, struct xrt_space_relation *out_head_relation, struct xrt_fov *out_fovs, struct xrt_pose *out_poses)
Get the per-view pose in relation to the view space.
Definition: xrt_device.h:568
enum xrt_device_name name
Enum identifier of the device.
Definition: xrt_device.h:286
xrt_result_t(* get_tracked_pose)(struct xrt_device *xdev, enum xrt_input_name name, int64_t at_timestamp_ns, struct xrt_space_relation *out_relation)
Get relationship of a tracked device to the tracking origin space as the base space.
Definition: xrt_device.h:357
Describes a projection matrix fov.
Definition: xrt_defines.h:499
Joint set type used for hand tracking.
Definition: xrt_defines.h:1443
All of the device components that deals with interfacing to a users head.
Definition: xrt_device.h:92
A single named input, that sits on a xrt_device.
Definition: xrt_device.h:165
A union of all output types.
Definition: xrt_defines.h:2092
A pose composed of a position and orientation.
Definition: xrt_defines.h:479
The main prober that probes and manages found but not opened HMD devices that are connected to the sy...
Definition: xrt_prober.h:132
A relation with two spaces, includes velocity and acceleration.
Definition: xrt_defines.h:670
Represents a uv triplet for distortion, basically just three xrt_vec2.
Definition: xrt_defines.h:279
A 3 element vector with single floats.
Definition: xrt_defines.h:289
Ringbuffer implementation for keeping track of the past state of things.
Common things like defines for Vive and Index.
Implementation of the Vive Pro 2 HID interface.
xrt_scanout_direction
Screen scanout direction.
Definition: xrt_defines.h:179
xrt_view_type
View type to be rendered to by the compositor.
Definition: xrt_defines.h:2123
Header defining an xrt display or controller device.