From c7087aca1905cb48b5993da84f1f9effc2882b31 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:19:27 -0400 Subject: [PATCH 01/50] Update hl2ss.py --- viewer/hl2ss.py | 1 + 1 file changed, 1 insertion(+) diff --git a/viewer/hl2ss.py b/viewer/hl2ss.py index 26d2e9da7..164e30372 100644 --- a/viewer/hl2ss.py +++ b/viewer/hl2ss.py @@ -45,6 +45,7 @@ class ChunkSize: PERSONAL_VIDEO = 4096 MICROPHONE = 512 SPATIAL_INPUT = 1024 + EXTENDED_EYE_TRACKER = 256 SINGLE_TRANSFER = 4096 From de41f3c01c315ed2d5800bd8f82f5a70468b9cb2 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:19:51 -0400 Subject: [PATCH 02/50] Update hl2ss_mp.py --- viewer/hl2ss_mp.py | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/viewer/hl2ss_mp.py b/viewer/hl2ss_mp.py index d746ea60b..c86165872 100644 --- a/viewer/hl2ss_mp.py +++ b/viewer/hl2ss_mp.py @@ -381,6 +381,9 @@ def configure_microphone(self, decoded, host, port, chunk, profile): def configure_si(self, host, port, chunk): self.configure(port, hl2ss.rx_si(host, port, chunk)) + def configure_eet(self, host, port, chunk, fps): + self.configure(port, hl2ss.rx_eet(host, port, chunk, fps)) + def initialize(self, port, buffer_size): self._producer[port] = _module(self._rx[port], buffer_size) @@ -453,6 +456,10 @@ def get_sync_period_si(): return 1 +def get_sync_period_eet(): + return 1 + + def get_sync_frame_stamp(frame_stamp, sync_period): return frame_stamp + ((sync_period - (frame_stamp % sync_period)) % sync_period) From 4f91d070176e358db254b6c547a9d9083cd73116 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:20:11 -0400 Subject: [PATCH 03/50] Update hl2ss_utilities.py --- viewer/hl2ss_utilities.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/viewer/hl2ss_utilities.py b/viewer/hl2ss_utilities.py index 34513f5a8..0a8796900 100644 --- a/viewer/hl2ss_utilities.py +++ b/viewer/hl2ss_utilities.py @@ -29,7 +29,7 @@ def microphone_packed_to_planar(array): # SI #------------------------------------------------------------------------------ -class SI_Hand: +class _SI_Hand: def __init__(self, poses, orientations, positions, radii, accuracies): self.poses = poses self.orientations = orientations @@ -44,7 +44,7 @@ def si_unpack_hand(hand): positions = np.array([pose.position for pose in poses]) radii = np.array([pose.radius for pose in poses]) accuracies = np.array([pose.accuracy for pose in poses]) - return SI_Hand(poses, orientations, positions, radii, accuracies) + return _SI_Hand(poses, orientations, positions, radii, accuracies) def si_head_pose_rotation_matrix(head_pose): @@ -86,6 +86,8 @@ def get_av_framerate(reader): return hl2ss.Parameters_RM_VLC.FPS if (reader.header.port == hl2ss.StreamPort.RM_DEPTH_AHAT): return hl2ss.Parameters_RM_DEPTH_AHAT.FPS + if (reader.header.port == hl2ss.StreamPort.RM_DEPTH_LONGTHROW): + return hl2ss.Parameters_RM_DEPTH_LONGTHROW.FPS if (reader.header.port == hl2ss.StreamPort.PERSONAL_VIDEO): return reader.header.framerate if (reader.header.port == hl2ss.StreamPort.MICROPHONE): From be6faefe989b279f94423fba82ccdcda221030b7 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:25:18 -0400 Subject: [PATCH 04/50] Update hl2ss_io.py --- viewer/hl2ss_io.py | 445 ++++++++++++++++++++++++--------------------- 1 file changed, 237 insertions(+), 208 deletions(-) diff --git a/viewer/hl2ss_io.py b/viewer/hl2ss_io.py index 4bdc5ace7..95bc5c665 100644 --- a/viewer/hl2ss_io.py +++ b/viewer/hl2ss_io.py @@ -30,12 +30,8 @@ def close(self): # Header Pack #------------------------------------------------------------------------------ -def _create_header(port): - return struct.pack(f'<{len(_MAGIC)}sH', _MAGIC.encode(), port) - - -def _create_user(user): - return struct.pack(' Date: Fri, 28 Apr 2023 22:30:51 -0400 Subject: [PATCH 05/50] Update hl2ss_utilities.py --- viewer/hl2ss_utilities.py | 8 -------- 1 file changed, 8 deletions(-) diff --git a/viewer/hl2ss_utilities.py b/viewer/hl2ss_utilities.py index 0a8796900..0ef1babff 100644 --- a/viewer/hl2ss_utilities.py +++ b/viewer/hl2ss_utilities.py @@ -130,14 +130,6 @@ def unpack_to_mp4(input_filenames, output_filename): [reader.close() for reader in readers] -#------------------------------------------------------------------------------ -# Math -#------------------------------------------------------------------------------ - -def clamp(v, min, max): - return min if (v < min) else max if (v > max) else v - - #------------------------------------------------------------------------------ # Timing #------------------------------------------------------------------------------ From 16efcfd51cb2fa00fad41d974b0cd34e5b9532e7 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:30:54 -0400 Subject: [PATCH 06/50] Update unity_demo_stickers.py --- viewer/unity_demo_stickers.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/viewer/unity_demo_stickers.py b/viewer/unity_demo_stickers.py index 87bebaf87..208e0681b 100644 --- a/viewer/unity_demo_stickers.py +++ b/viewer/unity_demo_stickers.py @@ -13,7 +13,6 @@ import hl2ss_imshow import hl2ss import hl2ss_3dcv -import hl2ss_utilities import rus import numpy as np @@ -40,6 +39,9 @@ enable = True trigger = False +def clamp(v, min, max): + return min if (v < min) else max if (v > max) else v + def on_press(key): global enable global trigger @@ -137,7 +139,7 @@ def on_release(key): canonical_normal = np.array([0, 0, -1]).reshape((1, 3)) # Normal that looks at the camera when the camera transform is the identity axis = np.cross(canonical_normal, normal) axis = axis / np.linalg.norm(axis) - angle = np.arccos(hl2ss_utilities.clamp(np.dot(canonical_normal, normal), -1, 1)) + angle = np.arccos(clamp(np.dot(canonical_normal, normal), -1, 1)) # Convert axis-angle rotation to quaternion cos = np.cos(angle / 2) From 6b9db0f487504894ce0993a675770358e6a94435 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Fri, 28 Apr 2023 22:40:22 -0400 Subject: [PATCH 07/50] Update hl2ss_utilities.py --- viewer/hl2ss_utilities.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/viewer/hl2ss_utilities.py b/viewer/hl2ss_utilities.py index 0ef1babff..d4e0b30ca 100644 --- a/viewer/hl2ss_utilities.py +++ b/viewer/hl2ss_utilities.py @@ -25,6 +25,16 @@ def microphone_packed_to_planar(array): return data +class microphone_resampler: + def open(self, target_format=None, target_layout=None, target_rate=None): + self._resampler = av.AudioResampler(format=target_format, layout=target_layout, rate=target_rate) + + def resample(self, data, format, layout): + in_frame = av.AudioFrame.from_ndarray(data, format=format, layout=layout) + out_frames = self._resampler.resample(in_frame) + return [frame.to_ndarray() for frame in out_frames] + + #------------------------------------------------------------------------------ # SI #------------------------------------------------------------------------------ From 19a0e8e2be43b7601c06c2ec3da81562446f191a Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 13:05:58 -0400 Subject: [PATCH 08/50] Update hl2ss.py --- viewer/hl2ss.py | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/viewer/hl2ss.py b/viewer/hl2ss.py index 164e30372..93a056c3d 100644 --- a/viewer/hl2ss.py +++ b/viewer/hl2ss.py @@ -1193,6 +1193,13 @@ def get_next_packet(self): return data +class rx_raw_microphone(rx_microphone): + def get_next_packet(self): + data = super().get_next_packet() + data.payload = np.frombuffer(data.payload, dtype=np.int16).reshape((1, -1)) + return data + + #------------------------------------------------------------------------------ # Mode 2 Data Acquisition #------------------------------------------------------------------------------ From bc01e9ad3ab8171a123988fba66cde391b284212 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 13:41:44 -0400 Subject: [PATCH 09/50] Update hl2ss.py --- viewer/hl2ss.py | 33 +++++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/viewer/hl2ss.py b/viewer/hl2ss.py index 93a056c3d..86d629e07 100644 --- a/viewer/hl2ss.py +++ b/viewer/hl2ss.py @@ -1156,13 +1156,28 @@ def close(self): #------------------------------------------------------------------------------ class rx_raw_rm_vlc(rx_rm_vlc): + def __init__(self, host, port, chunk, mode, profile, bitrate): + super().__init__(host, port, chunk, mode, profile, bitrate) + + def open(self): + super().open() + def get_next_packet(self): data = super().get_next_packet() data.payload = np.frombuffer(data.payload, dtype=np.uint8).reshape(Parameters_RM_VLC.SHAPE) return data + + def close(self): + super().close() class rx_raw_rm_depth_ahat(rx_rm_depth_ahat): + def __init__(self, host, port, chunk, mode, profile, bitrate): + super().__init__(host, port, chunk, mode, profile, bitrate) + + def open(self): + super().open() + def get_next_packet(self): data = super().get_next_packet() depth = np.frombuffer(data.payload, dtype=np.uint16, offset=0, count=Parameters_RM_DEPTH_AHAT.PIXELS).reshape(Parameters_RM_DEPTH_AHAT.SHAPE) @@ -1170,6 +1185,9 @@ def get_next_packet(self): depth[depth >= 4090] = 0 data.payload = _RM_Depth_Frame(depth, ab) return data + + def close(self): + super().close() class rx_raw_pv(rx_pv): @@ -1186,18 +1204,33 @@ def __init__(self, host, port, chunk, mode, width, height, framerate, profile, b self.format = rx_raw_pv._cv2_nv12_format[format] self.stride = get_nv12_stride(self.width) + def open(self): + super().open() + def get_next_packet(self): data = super().get_next_packet() data.payload = unpack_pv(data.payload) data.payload.image = cv2.cvtColor(np.frombuffer(data.payload.image, dtype=np.uint8).reshape((int(self.height*3/2), self.stride))[:, :self.width], self.format) return data + + def close(self): + super().close() class rx_raw_microphone(rx_microphone): + def __init__(self, host, port, chunk, profile): + super().__init__(host, port, chunk, profile) + + def open(self): + super().open() + def get_next_packet(self): data = super().get_next_packet() data.payload = np.frombuffer(data.payload, dtype=np.int16).reshape((1, -1)) return data + + def close(self): + super().close() #------------------------------------------------------------------------------ From 2d96f0edc32723723cdf211e62ea4eb88c731dc5 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 13:41:49 -0400 Subject: [PATCH 10/50] Delete unity_test_twin_mesh.py --- viewer/unity_test_twin_mesh.py | 141 --------------------------------- 1 file changed, 141 deletions(-) delete mode 100644 viewer/unity_test_twin_mesh.py diff --git a/viewer/unity_test_twin_mesh.py b/viewer/unity_test_twin_mesh.py deleted file mode 100644 index 44a5983e6..000000000 --- a/viewer/unity_test_twin_mesh.py +++ /dev/null @@ -1,141 +0,0 @@ - -import multiprocessing as mp -import numpy as np -import cv2 -import hl2ss -import hl2ss_mp -import hl2ss_3dcv -import open3d as o3d -import rus - -host = '192.168.1.7' -calibration_path = '../calibration' - -width = 640 -height = 360 -framerate = 30 -profile = hl2ss.VideoProfile.H265_MAIN -bitrate = hl2ss.get_video_codec_bitrate(width, height, framerate, 1/142) -buffer_length = 10 - -# Integration parameters -voxel_length = 1/32 -sdf_trunc = 0.04 -max_depth = 4.0 - -if __name__ == '__main__': - hl2ss.start_subsystem_pv(host, hl2ss.StreamPort.PERSONAL_VIDEO) - - calibration_lt = hl2ss_3dcv.get_calibration_rm(host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, calibration_path) - - uv2xy = hl2ss_3dcv.compute_uv2xy(calibration_lt.intrinsics, hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT) - xy1, scale = hl2ss_3dcv.rm_depth_compute_rays(uv2xy, calibration_lt.scale) - - volume = o3d.pipelines.integration.ScalableTSDFVolume(voxel_length=voxel_length, sdf_trunc=sdf_trunc, color_type=o3d.pipelines.integration.TSDFVolumeColorType.RGB8) - intrinsics_depth = o3d.camera.PinholeCameraIntrinsic(hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT, calibration_lt.intrinsics[0, 0], calibration_lt.intrinsics[1, 1], calibration_lt.intrinsics[2, 0], calibration_lt.intrinsics[2, 1]) - - producer = hl2ss_mp.producer() - producer.configure_pv(True, host, hl2ss.StreamPort.PERSONAL_VIDEO, hl2ss.ChunkSize.PERSONAL_VIDEO, hl2ss.StreamMode.MODE_1, width, height, framerate, profile, bitrate, 'rgb24') - producer.configure_rm_depth_longthrow(True, host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.ChunkSize.RM_DEPTH_LONGTHROW, hl2ss.StreamMode.MODE_1, hl2ss.PngFilterMode.Paeth) - producer.initialize(hl2ss.StreamPort.PERSONAL_VIDEO, framerate * buffer_length) - producer.initialize(hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.Parameters_RM_DEPTH_LONGTHROW.FPS * buffer_length) - producer.start(hl2ss.StreamPort.PERSONAL_VIDEO) - producer.start(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) - - manager = mp.Manager() - consumer = hl2ss_mp.consumer() - sink_pv = consumer.create_sink(producer, hl2ss.StreamPort.PERSONAL_VIDEO, manager, None) - sink_depth = consumer.create_sink(producer, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, manager, ...) - sink_pv.get_attach_response() - sink_depth.get_attach_response() - - pv_intrinsics = hl2ss.create_pv_intrinsics_placeholder() - - while (True): - sink_depth.acquire() - - _, data_lt = sink_depth.get_most_recent_frame() - if ((data_lt is None) or (not hl2ss.is_valid_pose(data_lt.pose))): - continue - - _, data_pv = sink_pv.get_nearest(data_lt.timestamp) - if ((data_pv is None) or (not hl2ss.is_valid_pose(data_pv.pose))): - continue - - break - - sink_pv.detach() - sink_depth.detach() - producer.stop(hl2ss.StreamPort.PERSONAL_VIDEO) - producer.stop(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) - - depth = hl2ss_3dcv.rm_depth_undistort(data_lt.payload.depth, calibration_lt.undistort_map) - depth = hl2ss_3dcv.rm_depth_normalize(depth, scale) - color = data_pv.payload.image - pv_intrinsics = hl2ss.update_pv_intrinsics(pv_intrinsics, data_pv.payload.focal_length, data_pv.payload.principal_point) - - lt_points = hl2ss_3dcv.rm_depth_to_points(xy1, depth) - lt_to_world = hl2ss_3dcv.camera_to_rignode(calibration_lt.extrinsics) @ hl2ss_3dcv.reference_to_world(data_lt.pose) - world_to_lt = hl2ss_3dcv.world_to_reference(data_lt.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) - world_to_pv_image = hl2ss_3dcv.world_to_reference(data_pv.pose) @ hl2ss_3dcv.camera_to_image(pv_intrinsics) - world_points = hl2ss_3dcv.transform(lt_points, lt_to_world) - pv_uv = hl2ss_3dcv.project(world_points, world_to_pv_image) - color = cv2.remap(color, pv_uv[:, :, 0], pv_uv[:, :, 1], cv2.INTER_LINEAR) - - mask_uv = hl2ss_3dcv.slice_to_block((pv_uv[:, :, 0] < 0) | (pv_uv[:, :, 0] >= width) | (pv_uv[:, :, 1] < 0) | (pv_uv[:, :, 1] >= height)) - depth[mask_uv] = 0 - - color_image = o3d.geometry.Image(color) - depth_image = o3d.geometry.Image(depth) - rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(color_image, depth_image, depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) - - depth_world_to_camera = hl2ss_3dcv.world_to_reference(data_lt.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) - - volume.integrate(rgbd, intrinsics_depth, depth_world_to_camera.transpose()) - - mesh = volume.extract_triangle_mesh() - vertices = np.asarray(mesh.vertices) - tris = np.asarray(mesh.triangles) - l_vertices = vertices @ np.array([[1, 0, 0], [0, 1, 0], [0, 0, -1]], dtype=np.float32) - for i in range(0, tris.shape[0]): - a = tris[i, 0] - b = tris[i, 1] - c = tris[i, 2] - tris[i, 0] = c - tris[i, 1] = b - tris[i, 2] = a - mesh.vertices = o3d.utility.Vector3dVector(l_vertices) - mesh.triangles = o3d.utility.Vector3iVector(tris) - - print(f'Mesh triangles {tris.shape}') - - o3d.io.write_triangle_mesh('./data/mesh.obj', mesh) - - with open('./data/mesh.obj', mode='a') as file: - file.write('# end of file') - - with open('./data/mesh.obj', mode='rb') as file: - mesh_bytes = file.read() - - ipc_umq = hl2ss.ipc_umq(host, hl2ss.IPCPort.UNITY_MESSAGE_QUEUE) - ipc_umq.open() - - display_list = rus.command_buffer() - display_list.remove_all() - display_list.load_mesh(mesh_bytes) - - ipc_umq.push(display_list) - results = ipc_umq.pull(display_list) - key = results[1] - - display_list = rus.command_buffer() - display_list.set_world_transform(key, [0, 0, 0], [0, 0, 0, 1], [1, 1, 1]) - ipc_umq.push(display_list) - ipc_umq.pull(display_list) - - ipc_umq.close() - - print(f'Created object with id {key}') - - o3d.visualization.draw_geometries([mesh]) - From b0401341f72a4be187a4a60f5ccd1b93f8ebbad1 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 14:33:32 -0400 Subject: [PATCH 11/50] Update hl2ss.py --- viewer/hl2ss.py | 158 +++++++++++++++++++++--------------------------- 1 file changed, 70 insertions(+), 88 deletions(-) diff --git a/viewer/hl2ss.py b/viewer/hl2ss.py index 86d629e07..e6933eaa8 100644 --- a/viewer/hl2ss.py +++ b/viewer/hl2ss.py @@ -708,7 +708,7 @@ def get_video_codec_bitrate(width, height, fps, factor): # RM VLC Decoder #------------------------------------------------------------------------------ -class decode_rm_vlc: +class _decode_rm_vlc: def __init__(self, profile): self.profile = profile @@ -722,6 +722,18 @@ def decode(self, payload): return None +class _unpack_rm_vlc: + def create(self): + pass + + def decode(self, payload): + return np.frombuffer(payload, dtype=np.uint8).reshape(Parameters_RM_VLC.SHAPE) + + +def decode_rm_vlc(profile): + return _unpack_rm_vlc() if (profile == VideoProfile.RAW) else _decode_rm_vlc(profile) + + #------------------------------------------------------------------------------ # RM Depth Decoder #------------------------------------------------------------------------------ @@ -756,7 +768,7 @@ def _unpack_rm_depth_ahat_nv12_as_yuv420p(yuv): return _RM_Depth_Frame(y, ab) -class decode_rm_depth_ahat: +class _decode_rm_depth_ahat: def __init__(self, profile): self.profile = profile @@ -770,6 +782,21 @@ def decode(self, payload): return None +class _unpack_rm_depth_ahat: + def create(self): + pass + + def decode(self, payload): + depth = np.frombuffer(payload, dtype=np.uint16, offset=0, count=Parameters_RM_DEPTH_AHAT.PIXELS).reshape(Parameters_RM_DEPTH_AHAT.SHAPE) + ab = np.frombuffer(payload, dtype=np.uint16, offset=Parameters_RM_DEPTH_AHAT.PIXELS*_SIZEOF.WORD, count=Parameters_RM_DEPTH_AHAT.PIXELS).reshape(Parameters_RM_DEPTH_AHAT.SHAPE) + depth[depth >= 4090] = 0 + return _RM_Depth_Frame(depth, ab) + + +def decode_rm_depth_ahat(profile): + return _unpack_rm_depth_ahat() if (profile == VideoProfile.RAW) else _decode_rm_depth_ahat(profile) + + def decode_rm_depth_longthrow(payload): composite = cv2.imdecode(np.frombuffer(payload, dtype=np.uint8), cv2.IMREAD_UNCHANGED) h, w, _ = composite.shape @@ -839,11 +866,11 @@ def get_nv12_stride(width): return width + ((64 - (width & 63)) & 63) -class decode_pv: +class _decode_pv: def __init__(self, profile): self.profile = profile - def create(self): + def create(self, width, height): self._codec = av.CodecContext.create(get_video_codec_name(self.profile), 'r') def decode(self, payload, format): @@ -853,11 +880,36 @@ def decode(self, payload, format): return None +class _unpack_pv: + _cv2_nv12_format = { + 'rgb24' : cv2.COLOR_YUV2RGB_NV12, + 'bgr24' : cv2.COLOR_YUV2BGR_NV12, + 'rgba' : cv2.COLOR_YUV2RGBA_NV12, + 'bgra' : cv2.COLOR_YUV2BGRA_NV12, + 'gray8' : cv2.COLOR_YUV2GRAY_NV12, + 'nv12' : None + } + + def create(self, width, height): + self.width = width + self.height = height + self.stride = get_nv12_stride(width) + + def decode(self, payload, format): + image = np.frombuffer(payload, dtype=np.uint8).reshape((int(self.height*3/2), self.stride))[:, :self.width] + sf = _unpack_pv._cv2_nv12_format[format] + return image if (sf is None) else cv2.cvtColor(image, sf) + + +def decode_pv(profile): + return _unpack_pv() if (profile == VideoProfile.RAW) else _decode_pv(profile) + + #------------------------------------------------------------------------------ # Microphone Decoder #------------------------------------------------------------------------------ -class decode_microphone: +class _decode_microphone: def __init__(self, profile): self.profile = profile @@ -871,6 +923,18 @@ def decode(self, payload): return None +class _unpack_microphone: + def create(self): + pass + + def decode(self, payload): + return np.frombuffer(payload, dtype=np.int16).reshape((1, -1)) + + +def decode_microphone(profile): + return _unpack_microphone() if (profile == AudioProfile.RAW) else _decode_microphone(profile) + + #------------------------------------------------------------------------------ # SI Unpacker #------------------------------------------------------------------------------ @@ -1119,7 +1183,7 @@ def __init__(self, host, port, chunk, mode, width, height, framerate, profile, b self._codec = decode_pv(profile) def open(self): - self._codec.create() + self._codec.create(self.width, self.height) super().open() self.get_next_packet() @@ -1151,88 +1215,6 @@ def close(self): super().close() -#------------------------------------------------------------------------------ -# RAW Receivers -#------------------------------------------------------------------------------ - -class rx_raw_rm_vlc(rx_rm_vlc): - def __init__(self, host, port, chunk, mode, profile, bitrate): - super().__init__(host, port, chunk, mode, profile, bitrate) - - def open(self): - super().open() - - def get_next_packet(self): - data = super().get_next_packet() - data.payload = np.frombuffer(data.payload, dtype=np.uint8).reshape(Parameters_RM_VLC.SHAPE) - return data - - def close(self): - super().close() - - -class rx_raw_rm_depth_ahat(rx_rm_depth_ahat): - def __init__(self, host, port, chunk, mode, profile, bitrate): - super().__init__(host, port, chunk, mode, profile, bitrate) - - def open(self): - super().open() - - def get_next_packet(self): - data = super().get_next_packet() - depth = np.frombuffer(data.payload, dtype=np.uint16, offset=0, count=Parameters_RM_DEPTH_AHAT.PIXELS).reshape(Parameters_RM_DEPTH_AHAT.SHAPE) - ab = np.frombuffer(data.payload, dtype=np.uint16, offset=Parameters_RM_DEPTH_AHAT.PIXELS*_SIZEOF.WORD, count=Parameters_RM_DEPTH_AHAT.PIXELS).reshape(Parameters_RM_DEPTH_AHAT.SHAPE) - depth[depth >= 4090] = 0 - data.payload = _RM_Depth_Frame(depth, ab) - return data - - def close(self): - super().close() - - -class rx_raw_pv(rx_pv): - _cv2_nv12_format = { - 'rgb24' : cv2.COLOR_YUV2RGB_NV12, - 'bgr24' : cv2.COLOR_YUV2BGR_NV12, - 'rgba' : cv2.COLOR_YUV2RGBA_NV12, - 'bgra' : cv2.COLOR_YUV2BGRA_NV12, - 'gray8' : cv2.COLOR_YUV2GRAY_NV12, - } - - def __init__(self, host, port, chunk, mode, width, height, framerate, profile, bitrate, format): - super().__init__(host, port, chunk, mode, width, height, framerate, profile, bitrate) - self.format = rx_raw_pv._cv2_nv12_format[format] - self.stride = get_nv12_stride(self.width) - - def open(self): - super().open() - - def get_next_packet(self): - data = super().get_next_packet() - data.payload = unpack_pv(data.payload) - data.payload.image = cv2.cvtColor(np.frombuffer(data.payload.image, dtype=np.uint8).reshape((int(self.height*3/2), self.stride))[:, :self.width], self.format) - return data - - def close(self): - super().close() - - -class rx_raw_microphone(rx_microphone): - def __init__(self, host, port, chunk, profile): - super().__init__(host, port, chunk, profile) - - def open(self): - super().open() - - def get_next_packet(self): - data = super().get_next_packet() - data.payload = np.frombuffer(data.payload, dtype=np.int16).reshape((1, -1)) - return data - - def close(self): - super().close() - - #------------------------------------------------------------------------------ # Mode 2 Data Acquisition #------------------------------------------------------------------------------ From 056c56f590155127dde798be6042b7dd51059d49 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 14:50:24 -0400 Subject: [PATCH 12/50] delete raw clients --- viewer/client_microphone_raw.py | 64 ----------------------- viewer/client_pv_raw.py | 82 ------------------------------ viewer/client_rm_depth_ahat_raw.py | 72 -------------------------- viewer/client_rm_vlc_raw.py | 69 ------------------------- 4 files changed, 287 deletions(-) delete mode 100644 viewer/client_microphone_raw.py delete mode 100644 viewer/client_pv_raw.py delete mode 100644 viewer/client_rm_depth_ahat_raw.py delete mode 100644 viewer/client_rm_vlc_raw.py diff --git a/viewer/client_microphone_raw.py b/viewer/client_microphone_raw.py deleted file mode 100644 index 16d1c91c7..000000000 --- a/viewer/client_microphone_raw.py +++ /dev/null @@ -1,64 +0,0 @@ -#------------------------------------------------------------------------------ -# This script receives uncompressed microphone audio from the HoloLens and -# plays it. The main thread receives the data, decodes it, and puts the decoded -# audio samples in a queue. A second thread gets the samples from the queue and -# plays them. Audio stream configuration is fixed to 2 channels, 48000 Hz. -# Press esc to stop. -#------------------------------------------------------------------------------ - -from pynput import keyboard - -import hl2ss -import pyaudio -import queue -import threading - -# Settings -------------------------------------------------------------------- - -# HoloLens address -host = "192.168.1.7" - -# Port -port = hl2ss.StreamPort.MICROPHONE - -# Audio encoding profile -profile = hl2ss.AudioProfile.RAW - -#------------------------------------------------------------------------------ - -enable = True - -def pcmworker(pcmqueue): - global enable - p = pyaudio.PyAudio() - stream = p.open(format=pyaudio.paInt16, channels=hl2ss.Parameters_MICROPHONE.CHANNELS, rate=hl2ss.Parameters_MICROPHONE.SAMPLE_RATE, output=True) - stream.start_stream() - while (enable): - stream.write(pcmqueue.get()) - stream.stop_stream() - stream.close() - -def on_press(key): - global enable - enable = key != keyboard.Key.esc - return enable - -pcmqueue = queue.Queue() -thread = threading.Thread(target=pcmworker, args=(pcmqueue,)) -listener = keyboard.Listener(on_press=on_press) -thread.start() -listener.start() - -client = hl2ss.rx_microphone(host, port, hl2ss.ChunkSize.MICROPHONE, profile) -client.open() - -while (enable): - data = client.get_next_packet() - pcmqueue.put(bytes(data.payload)) - -client.close() - -enable = False -pcmqueue.put(b'') -thread.join() -listener.join() diff --git a/viewer/client_pv_raw.py b/viewer/client_pv_raw.py deleted file mode 100644 index 9760a4e5c..000000000 --- a/viewer/client_pv_raw.py +++ /dev/null @@ -1,82 +0,0 @@ -#------------------------------------------------------------------------------ -# This script receives uncompressed video from the HoloLens front RGB camera -# and plays it. The camera support various resolutions and framerates. See -# etc/hl2_capture_formats.txt for a list of supported formats. The default -# configuration is 1080p 30 FPS. The stream supports three operating modes: -# 0) video, 1) video + camera pose, 2) query calibration (single transfer). -# Press esc to stop. Note that the ahat stream cannot be used while the pv -# subsystem is on. -#------------------------------------------------------------------------------ - -from pynput import keyboard - -import time -import numpy as np -import cv2 -import hl2ss_imshow -import hl2ss - -# Settings -------------------------------------------------------------------- - -# HoloLens address -host = "192.168.1.7" - -# Port -port = hl2ss.StreamPort.PERSONAL_VIDEO - -# Operating mode -# 0: video -# 1: video + camera pose -# 2: query calibration (single transfer) -mode = hl2ss.StreamMode.MODE_1 - -# Camera parameters -width = 640 -height = 360 -framerate = 15 - -# Video encoding profile -profile = hl2ss.VideoProfile.RAW - -#------------------------------------------------------------------------------ - -hl2ss.start_subsystem_pv(host, port) - -if (mode == hl2ss.StreamMode.MODE_2): - data = hl2ss.download_calibration_pv(host, port, width, height, framerate) - print('Calibration') - print(data.focal_length) - print(data.principal_point) - print(data.radial_distortion) - print(data.tangential_distortion) - print(data.projection) - print(data.intrinsics) -else: - enable = True - - def on_press(key): - global enable - enable = key != keyboard.Key.esc - return enable - - listener = keyboard.Listener(on_press=on_press) - listener.start() - - client = hl2ss.rx_raw_pv(host, port, hl2ss.ChunkSize.PERSONAL_VIDEO, mode, width, height, framerate, profile, 1, 'bgr24') - client.open() - - stride = hl2ss.get_nv12_stride(width) - - while (enable): - data = client.get_next_packet() - - print('Pose at time {ts}'.format(ts=data.timestamp)) - print(data.pose) - - cv2.imshow('Video', data.payload.image) - cv2.waitKey(1) - - client.close() - listener.join() - -hl2ss.stop_subsystem_pv(host, port) diff --git a/viewer/client_rm_depth_ahat_raw.py b/viewer/client_rm_depth_ahat_raw.py deleted file mode 100644 index 8b3eaaf43..000000000 --- a/viewer/client_rm_depth_ahat_raw.py +++ /dev/null @@ -1,72 +0,0 @@ -#------------------------------------------------------------------------------ -# This script receives uncompressed video from the HoloLens depth camera in -# ahat mode and plays it. The resolution is 512x512 @ 45 FPS. The stream -# supports three operating modes: 0) video, 1) video + rig pose, 2) query -# calibration (single transfer). Press esc to stop. Depth and AB data are -# scaled for visibility. Note that 1) the ahat stream cannot be used while the -# pv subsystem is on, and 2) the ahat and long throw streams cannot be used -# simultaneously. -#------------------------------------------------------------------------------ - -from pynput import keyboard - -import numpy as np -import cv2 -import hl2ss_imshow -import hl2ss - -# Settings -------------------------------------------------------------------- - -# HoloLens address -host = "192.168.1.7" - -# Port -port = hl2ss.StreamPort.RM_DEPTH_AHAT - -# Operating mode -# 0: video -# 1: video + rig pose -# 2: query calibration (single transfer) -mode = hl2ss.StreamMode.MODE_1 - -# Video encoding profile -profile = hl2ss.VideoProfile.RAW - -#------------------------------------------------------------------------------ - -if (mode == hl2ss.StreamMode.MODE_2): - data = hl2ss.download_calibration_rm_depth_ahat(host, port) - print('Calibration data') - print(data.uv2xy.shape) - print(data.extrinsics) - print(data.scale) - print(data.alias) - print(data.undistort_map.shape) - print(data.intrinsics) - quit() - -enable = True - -def on_press(key): - global enable - enable = key != keyboard.Key.esc - return enable - -listener = keyboard.Listener(on_press=on_press) -listener.start() - -client = hl2ss.rx_raw_rm_depth_ahat(host, port, hl2ss.ChunkSize.RM_DEPTH_AHAT, mode, profile, 1) -client.open() - -while (enable): - data = client.get_next_packet() - - print('Pose at time {ts}'.format(ts=data.timestamp)) - print(data.pose) - - cv2.imshow('Depth', data.payload.depth / np.max(data.payload.depth)) - cv2.imshow('AB', data.payload.ab) # max ~ 12000 - cv2.waitKey(1) - -client.close() -listener.join() diff --git a/viewer/client_rm_vlc_raw.py b/viewer/client_rm_vlc_raw.py deleted file mode 100644 index cf466a20f..000000000 --- a/viewer/client_rm_vlc_raw.py +++ /dev/null @@ -1,69 +0,0 @@ -#------------------------------------------------------------------------------ -# This script receives uncompressed video from one of the HoloLens sideview -# grayscale cameras and plays it. The camera resolution is 640x480 @ 30 FPS. -# The stream supports three operating modes: 0) video, 1) video + rig pose, -# 2) query calibration (single transfer). Press esc to stop. -#------------------------------------------------------------------------------ - -from pynput import keyboard - -import numpy as np -import cv2 -import hl2ss_imshow -import hl2ss - -# Settings -------------------------------------------------------------------- - -# HoloLens address -host = "192.168.1.7" - -# Port -# Options: -# hl2ss.StreamPort.RM_VLC_LEFTFRONT -# hl2ss.StreamPort.RM_VLC_LEFTLEFT -# hl2ss.StreamPort.RM_VLC_RIGHTFRONT -# hl2ss.StreamPort.RM_VLC_RIGHTRIGHT -port = hl2ss.StreamPort.RM_VLC_LEFTFRONT - -# Operating mode -# 0: video -# 1: video + rig pose -# 2: query calibration (single transfer) -mode = hl2ss.StreamMode.MODE_1 - -# Video encoding profile -profile = hl2ss.VideoProfile.RAW - -#------------------------------------------------------------------------------ - -if (mode == hl2ss.StreamMode.MODE_2): - data = hl2ss.download_calibration_rm_vlc(host, port) - print('Calibration data') - print(data.uv2xy.shape) - print(data.extrinsics) - print(data.undistort_map.shape) - print(data.intrinsics) - quit() - -enable = True - -def on_press(key): - global enable - enable = key != keyboard.Key.esc - return enable - -listener = keyboard.Listener(on_press=on_press) -listener.start() - -client = hl2ss.rx_raw_rm_vlc(host, port, hl2ss.ChunkSize.RM_VLC, mode, profile, 1) -client.open() - -while (enable): - data = client.get_next_packet() - print('Pose at time {ts}'.format(ts=data.timestamp)) - print(data.pose) - cv2.imshow('Video', data.payload) - cv2.waitKey(1) - -client.close() -listener.join() From 20ed772a15a05c286541e2f6d8b700e3bbb5067c Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 14:50:32 -0400 Subject: [PATCH 13/50] Update client_microphone.py --- viewer/client_microphone.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/viewer/client_microphone.py b/viewer/client_microphone.py index 5a44afa5e..c09876b0c 100644 --- a/viewer/client_microphone.py +++ b/viewer/client_microphone.py @@ -27,12 +27,14 @@ #------------------------------------------------------------------------------ +audio_format = pyaudio.paInt16 if (profile == hl2ss.AudioProfile.RAW) else pyaudio.paFloat32 enable = True def pcmworker(pcmqueue): global enable + global audio_format p = pyaudio.PyAudio() - stream = p.open(format=pyaudio.paFloat32, channels=hl2ss.Parameters_MICROPHONE.CHANNELS, rate=hl2ss.Parameters_MICROPHONE.SAMPLE_RATE, output=True) + stream = p.open(format=audio_format, channels=hl2ss.Parameters_MICROPHONE.CHANNELS, rate=hl2ss.Parameters_MICROPHONE.SAMPLE_RATE, output=True) stream.start_stream() while (enable): stream.write(pcmqueue.get()) @@ -55,7 +57,10 @@ def on_press(key): while (enable): data = client.get_next_packet() - audio = hl2ss_utilities.microphone_planar_to_packed(data.payload) + if (profile != hl2ss.AudioProfile.RAW): + audio = hl2ss_utilities.microphone_planar_to_packed(data.payload) + else: + audio = data.payload pcmqueue.put(audio.tobytes()) client.close() From ac437815cd33bc268d58692c12458950129ab09c Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 14:51:35 -0400 Subject: [PATCH 14/50] Update hl2ss_io.py --- viewer/hl2ss_io.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/hl2ss_io.py b/viewer/hl2ss_io.py index 95bc5c665..214e72743 100644 --- a/viewer/hl2ss_io.py +++ b/viewer/hl2ss_io.py @@ -623,7 +623,7 @@ def __init__(self, filename, chunk, format): def open(self): super().open() self._codec = hl2ss.decode_pv(self.header.profile) - self._codec.create() + self._codec.create(self.header.width, self.header.height) self.read() def read(self): From 3171a90d654ecd034ddb458b2f9b81630a3090df Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:10:12 -0400 Subject: [PATCH 15/50] Update client_eet.py --- viewer/client_eet.py | 37 ++++++++++++++++++++++--------------- 1 file changed, 22 insertions(+), 15 deletions(-) diff --git a/viewer/client_eet.py b/viewer/client_eet.py index 98796d071..374458fdb 100644 --- a/viewer/client_eet.py +++ b/viewer/client_eet.py @@ -1,6 +1,5 @@ #------------------------------------------------------------------------------ # This script receives extended eye tracking data from the HoloLens. -# See https://learn.microsoft.com/en-us/windows/mixed-reality/develop/native/extended-eye-tracking-native #------------------------------------------------------------------------------ import hl2ss @@ -25,26 +24,34 @@ data = client.get_next_packet() data.payload = hl2ss.unpack_eet(data.payload) -print(data.timestamp) -print(data.payload._reserved) +# See +# https://learn.microsoft.com/en-us/windows/mixed-reality/develop/native/extended-eye-tracking-native +# for details +print(f'Tracking status at time {data.timestamp}') +print('Pose') +print(data.pose) + +print(f'Calibration valid: {data.payload.calibration_valid}') + +print('Combined eye gaze') +print(f'Valid: {data.payload.combined_ray_valid}') print(data.payload.combined_ray.origin) print(data.payload.combined_ray.direction) + +print('Left eye gaze') +print(f'Valid: {data.payload.left_ray_valid}') print(data.payload.left_ray.origin) print(data.payload.left_ray.direction) + +print('Right eye gaze') +print(f'Valid: {data.payload.right_ray_valid}') print(data.payload.right_ray.origin) print(data.payload.right_ray.direction) -print(data.payload.left_openness) -print(data.payload.right_openness) -print(data.payload.vergence_distance) - -print(data.payload.calibration_valid) -print(data.payload.combined_ray_valid) -print(data.payload.left_ray_valid) -print(data.payload.right_ray_valid) -print(data.payload.left_openness_valid) -print(data.payload.right_openness_valid) -print(data.payload.vergence_distance_valid) -print(data.pose) + +# "...not supported by HoloLens 2 at this time" +print(f'Left eye openness: Valid={data.payload.left_openness_valid} Value={data.payload.left_openness}') +print(f'Right eye openness: Valid={data.payload.right_openness_valid} Value={data.payload.right_openness}') +print(f'Vergence distance: Valid={data.payload.vergence_distance_valid} Value={data.payload.vergence_distance}') client.close() From 6b8ec2fd83936867c237b570af4cf99d627f2068 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:14:14 -0400 Subject: [PATCH 16/50] Update client_microphone.py --- viewer/client_microphone.py | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/viewer/client_microphone.py b/viewer/client_microphone.py index c09876b0c..121dff2e8 100644 --- a/viewer/client_microphone.py +++ b/viewer/client_microphone.py @@ -1,8 +1,8 @@ #------------------------------------------------------------------------------ -# This script receives AAC encoded microphone audio from the HoloLens and -# plays it. The main thread receives the data, decodes it, and puts the decoded -# audio samples in a queue. A second thread gets the samples from the queue and -# plays them. Audio stream configuration is fixed to 2 channels, 48000 Hz. +# This script receives microphone audio from the HoloLens and plays it. The +# main thread receives the data, decodes it, and puts the decoded audio samples +# in a queue. A second thread gets the samples from the queue and plays them. +# Audio stream configuration is fixed to 2 channels, 48000 Hz. # Press esc to stop. #------------------------------------------------------------------------------ @@ -27,6 +27,7 @@ #------------------------------------------------------------------------------ +# RAW format is s16 packed, AAC decoded format is f32 planar audio_format = pyaudio.paInt16 if (profile == hl2ss.AudioProfile.RAW) else pyaudio.paFloat32 enable = True @@ -57,10 +58,8 @@ def on_press(key): while (enable): data = client.get_next_packet() - if (profile != hl2ss.AudioProfile.RAW): - audio = hl2ss_utilities.microphone_planar_to_packed(data.payload) - else: - audio = data.payload + # RAW format is s16 packed, AAC decoded format is f32 planar + audio = hl2ss_utilities.microphone_planar_to_packed(data.payload) if (profile != hl2ss.AudioProfile.RAW) else data.payload pcmqueue.put(audio.tobytes()) client.close() From 6b910c0be1832bb02a0d3ad4003f8bcc123a6c64 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:26:16 -0400 Subject: [PATCH 17/50] renamed samples --- viewer/{client_pv_microphone.py => sample_pv_microphone.py} | 0 viewer/{client_pv_si.py => sample_pv_si.py} | 0 2 files changed, 0 insertions(+), 0 deletions(-) rename viewer/{client_pv_microphone.py => sample_pv_microphone.py} (100%) rename viewer/{client_pv_si.py => sample_pv_si.py} (100%) diff --git a/viewer/client_pv_microphone.py b/viewer/sample_pv_microphone.py similarity index 100% rename from viewer/client_pv_microphone.py rename to viewer/sample_pv_microphone.py diff --git a/viewer/client_pv_si.py b/viewer/sample_pv_si.py similarity index 100% rename from viewer/client_pv_si.py rename to viewer/sample_pv_si.py From ac9e377120f4cef2126ddaabbc174477f1c6a2bf Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:37:41 -0400 Subject: [PATCH 18/50] rename samples --- viewer/{client_rm_vlc_si.py => sample_rm_vlc_si.py} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename viewer/{client_rm_vlc_si.py => sample_rm_vlc_si.py} (100%) diff --git a/viewer/client_rm_vlc_si.py b/viewer/sample_rm_vlc_si.py similarity index 100% rename from viewer/client_rm_vlc_si.py rename to viewer/sample_rm_vlc_si.py From 4903e39a3d236140856ad49702930815b32ee247 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:41:36 -0400 Subject: [PATCH 19/50] Update client_eet.py --- viewer/client_eet.py | 19 +++---------------- 1 file changed, 3 insertions(+), 16 deletions(-) diff --git a/viewer/client_eet.py b/viewer/client_eet.py index 374458fdb..3a87f4639 100644 --- a/viewer/client_eet.py +++ b/viewer/client_eet.py @@ -31,23 +31,10 @@ print(f'Tracking status at time {data.timestamp}') print('Pose') print(data.pose) - print(f'Calibration valid: {data.payload.calibration_valid}') - -print('Combined eye gaze') -print(f'Valid: {data.payload.combined_ray_valid}') -print(data.payload.combined_ray.origin) -print(data.payload.combined_ray.direction) - -print('Left eye gaze') -print(f'Valid: {data.payload.left_ray_valid}') -print(data.payload.left_ray.origin) -print(data.payload.left_ray.direction) - -print('Right eye gaze') -print(f'Valid: {data.payload.right_ray_valid}') -print(data.payload.right_ray.origin) -print(data.payload.right_ray.direction) +print(f'Combined eye gaze: Valid={data.payload.combined_ray_valid} Origin={data.payload.combined_ray.origin} Direction={data.payload.combined_ray.direction}') +print(f'Left eye gaze: Valid={data.payload.left_ray_valid} Origin={data.payload.left_ray.origin} Direction={data.payload.left_ray.direction}') +print(f'Right eye gaze: Valid={data.payload.right_ray_valid} Origin={data.payload.right_ray.origin} Direction={data.payload.right_ray.direction}') # "...not supported by HoloLens 2 at this time" print(f'Left eye openness: Valid={data.payload.left_openness_valid} Value={data.payload.left_openness}') From b6ae84a0be29e05267d945761b592770f39fd01b Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:47:20 -0400 Subject: [PATCH 20/50] Update client_pv.py --- viewer/client_pv.py | 40 ++++++++++++++++++++++++---------------- 1 file changed, 24 insertions(+), 16 deletions(-) diff --git a/viewer/client_pv.py b/viewer/client_pv.py index bb4ac014f..ad5966ac4 100644 --- a/viewer/client_pv.py +++ b/viewer/client_pv.py @@ -1,11 +1,11 @@ #------------------------------------------------------------------------------ -# This script receives encoded video from the HoloLens front RGB camera and -# plays it. The camera support various resolutions and framerates. See -# etc/hl2_capture_formats.txt for a list of supported formats. The default -# configuration is 1080p 30 FPS. The stream supports three operating modes: -# 0) video, 1) video + camera pose, 2) query calibration (single transfer). -# Press esc to stop. Note that the ahat stream cannot be used while the pv -# subsystem is on. +# This script receives video from the HoloLens front RGB camera and plays it. +# The camera supports various resolutions and framerates. See +# https://github.com/jdibenes/hl2ss/blob/main/etc/pv_configurations.txt +# for a list of supported formats. The default configuration is 1080p 30 FPS. +# The stream supports three operating modes: 0) video, 1) video + camera pose, +# 2) query calibration (single transfer). +# Press esc to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -38,9 +38,15 @@ # Encoded stream average bits per second # Must be > 0 -bitrate = 5*1024*1024 +bitrate = hl2ss.get_video_codec_bitrate(width, height, framerate, hl2ss.get_video_codec_default_factor(profile)) # Decoded format +# Options include: +# 'bgr24' +# 'rgb24' +# 'bgra' +# 'rgba' +# 'gray8' decoded_format = 'bgr24' #------------------------------------------------------------------------------ @@ -50,11 +56,13 @@ if (mode == hl2ss.StreamMode.MODE_2): data = hl2ss.download_calibration_pv(host, port, width, height, framerate) print('Calibration') - print(data.focal_length) - print(data.principal_point) - print(data.radial_distortion) - print(data.tangential_distortion) + print(f'Focal length: {data.focal_length}') + print(f'Principal point: {data.principal_point}') + print(f'Radial distortion: {data.radial_distortion}') + print(f'Tangential distortion: {data.tangential_distortion}') + print('Projection') print(data.projection) + print('Intrinsics') print(data.intrinsics) else: enable = True @@ -72,12 +80,12 @@ def on_press(key): while (enable): data = client.get_next_packet() + print('Pose at time {ts}'.format(ts=data.timestamp)) print(data.pose) - print('Focal length') - print(data.payload.focal_length) - print('Principal point') - print(data.payload.principal_point) + print(f'Focal length: {data.payload.focal_length}') + print(f'Principal point: {data.payload.principal_point}') + cv2.imshow('Video', data.payload.image) cv2.waitKey(1) From a74cb40fe887435e6c53dbc71ce72fa88d02c147 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 15:54:06 -0400 Subject: [PATCH 21/50] Update client_rc.py --- viewer/client_rc.py | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/viewer/client_rc.py b/viewer/client_rc.py index 3bb7390da..37cbe69e3 100644 --- a/viewer/client_rc.py +++ b/viewer/client_rc.py @@ -64,17 +64,19 @@ client.open() version = client.get_application_version() -print('Installed version {v0}.{v1}.{v2}.{v3}'.format(v0=version[0], v1=version[1], v2=version[2], v3=version[3])) +print(f'Installed version {version[0]}.{version[1]}.{version[2]}.{version[3]}') -utc_offset = client.get_utc_offset(32) # Add this offset to timestamps to convert to utc -print('QPC timestamp to UTC offset is {offset} hundreds of nanoseconds'.format(offset=utc_offset)) +# Add this offset to timestamps to convert to utc (Windows FILETIME) +utc_offset = client.get_utc_offset(32) +print(f'QPC timestamp to UTC offset is {utc_offset} hundreds of nanoseconds') client.set_hs_marker_state(marker_state) # PV camera configuration pv_status = client.get_pv_subsystem_status() -print('PV subsystem is {status}'.format(status=('On' if pv_status else 'Off'))) +print(f'PV subsystem is {("On" if (pv_status) else "Off")}') +# Ignored if PV subsystem is Off client.set_pv_focus(focus_mode, auto_focus_range, manual_focus_distance, focus_value, driver_fallback) client.set_pv_video_temporal_denoising(video_temporal_denoising) client.set_pv_white_balance_preset(white_balance_preset) From a204d2de789d063a54a2ffddd6751644281e5e9a Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:05:47 -0400 Subject: [PATCH 22/50] Update client_rm_depth_ahat.py --- viewer/client_rm_depth_ahat.py | 31 +++++++++++++++++-------------- 1 file changed, 17 insertions(+), 14 deletions(-) diff --git a/viewer/client_rm_depth_ahat.py b/viewer/client_rm_depth_ahat.py index ac7c7e906..84ce315f3 100644 --- a/viewer/client_rm_depth_ahat.py +++ b/viewer/client_rm_depth_ahat.py @@ -1,11 +1,10 @@ #------------------------------------------------------------------------------ -# This script receives encoded video from the HoloLens depth camera in ahat -# mode and plays it. The resolution is 512x512 @ 45 FPS. The stream supports -# three operating modes: 0) video, 1) video + rig pose, 2) query calibration -# (single transfer). Press esc to stop. Depth and AB data are scaled for -# visibility. Note that 1) the ahat stream cannot be used while the pv -# subsystem is on, and 2) the ahat and long throw streams cannot be used -# simultaneously. +# This script receives video from the HoloLens depth camera in ahat mode and +# plays it. The resolution is 512x512 @ 45 FPS. The stream supports three +# operating modes: 0) video, 1) video + rig pose, 2) query calibration (single +# transfer). Depth and AB data are scaled for visibility. The ahat and long +# throw streams cannot be used simultaneously. +# Press esc to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -41,11 +40,15 @@ if (mode == hl2ss.StreamMode.MODE_2): data = hl2ss.download_calibration_rm_depth_ahat(host, port) print('Calibration data') - print(data.uv2xy.shape) + print('Image point to unit plane') + print(data.uv2xy) + print('Extrinsics') print(data.extrinsics) - print(data.scale) - print(data.alias) - print(data.undistort_map.shape) + print(f'Scale: {data.scale}') + print(f'Alias: {data.alias}') + print('Undistort map') + print(data.undistort_map) + print('Intrinsics (undistorted only)') print(data.intrinsics) quit() @@ -64,10 +67,10 @@ def on_press(key): while (enable): data = client.get_next_packet() - print('Pose at time {ts}'.format(ts=data.timestamp)) + print(f'Pose at time {data.timestamp}') print(data.pose) - cv2.imshow('Depth', data.payload.depth / np.max(data.payload.depth)) # Normalized for visibility - cv2.imshow('AB', data.payload.ab / np.max(data.payload.ab)) # Normalized for visibility + cv2.imshow('Depth', data.payload.depth / np.max(data.payload.depth)) # Scaled for visibility + cv2.imshow('AB', data.payload.ab / np.max(data.payload.ab)) # Scaled for visibility cv2.waitKey(1) client.close() From 4947761825c79b75815cd8e0debda8a47419dc5a Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:09:29 -0400 Subject: [PATCH 23/50] Update client_rm_depth_longthrow.py --- viewer/client_rm_depth_longthrow.py | 28 ++++++++++++++++------------ 1 file changed, 16 insertions(+), 12 deletions(-) diff --git a/viewer/client_rm_depth_longthrow.py b/viewer/client_rm_depth_longthrow.py index 63d8da875..8318b7d50 100644 --- a/viewer/client_rm_depth_longthrow.py +++ b/viewer/client_rm_depth_longthrow.py @@ -1,10 +1,10 @@ #------------------------------------------------------------------------------ -# This script receives encoded video from the HoloLens depth camera in long -# throw mode and plays it. The resolution is 320x288 @ 5 FPS. The stream -# supports three operating modes: 0) video, 1) video + rig pose, 2) query -# calibration (single transfer). Press esc to stop. Depth and AB data are -# scaled for visibility. Note that the ahat and long throw streams cannot be -# used simultaneously. +# This script receives video from the HoloLens depth camera in long throw mode +# and plays it. The resolution is 320x288 @ 5 FPS. The stream supports three +# operating modes: 0) video, 1) video + rig pose, 2) query calibration (single +# transfer). Depth and AB data are scaled for visibility. The ahat and long +# throw streams cannot be used simultaneously. +# Press esc to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -36,10 +36,14 @@ if (mode == hl2ss.StreamMode.MODE_2): data = hl2ss.download_calibration_rm_depth_longthrow(host, port) print('Calibration data') - print(data.uv2xy.shape) + print('Image point to unit plane') + print(data.uv2xy) + print('Extrinsics') print(data.extrinsics) - print(data.scale) - print(data.undistort_map.shape) + print(f'Scale: {data.scale}') + print('Undistort map') + print(data.undistort_map) + print('Intrinsics (undistorted only)') print(data.intrinsics) quit() @@ -58,10 +62,10 @@ def on_press(key): while (enable): data = client.get_next_packet() - print('Pose at time {ts}'.format(ts=data.timestamp)) + print(f'Pose at time {data.timestamp}') print(data.pose) - cv2.imshow('Depth', data.payload.depth / np.max(data.payload.depth)) # Normalized for visibility - cv2.imshow('AB', data.payload.ab / np.max(data.payload.ab)) # Normalized for visibility + cv2.imshow('Depth', data.payload.depth / np.max(data.payload.depth)) # Scaled for visibility + cv2.imshow('AB', data.payload.ab / np.max(data.payload.ab)) # Scaled for visibility cv2.waitKey(1) client.close() From 3efd02c5947de32d3e4294f7546f8625a440a2f5 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:20:27 -0400 Subject: [PATCH 24/50] Update client_rm_imu.py --- viewer/client_rm_imu.py | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/viewer/client_rm_imu.py b/viewer/client_rm_imu.py index 66f669025..dd5e6813d 100644 --- a/viewer/client_rm_imu.py +++ b/viewer/client_rm_imu.py @@ -1,12 +1,13 @@ #------------------------------------------------------------------------------ # This script receives IMU samples from the HoloLens and prints them. # Sensor details: -# Accelerometer: 93 samples per frame, sample rate ~1100 Hz -# Gyroscope: 315 samples per frame, sample rate ~6550 Hz -# Magnetometer: 11 samples per frame, sample rate ~50 Hz +# Accelerometer: 93 samples per frame, sample rate ~1100 Hz effective ~12 Hz +# Gyroscope: 315 samples per frame, sample rate ~7500 Hz effective ~24 Hz +# Magnetometer: 11 samples per frame, sample rate ~50 Hz effective ~5 Hz # The streams support three operating modes: 0) samples, 1) samples + rig pose, # 2) query calibration (single transfer), except for the magnetometer stream -# which does not support mode 2. Press esc to stop. +# which does not support mode 2. +# Press esc to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -43,6 +44,7 @@ if (mode == hl2ss.StreamMode.MODE_2): data = hl2ss.download_calibration_rm_imu(host, port) print('Calibration data') + print('Extrinsics') print(data.extrinsics) quit() @@ -61,12 +63,13 @@ def on_press(key): while (enable): data = client.get_next_packet() - print('Pose at time {ts}'.format(ts=data.timestamp)) + print(f'Pose at time {data.timestamp}') print(data.pose) imu_data = hl2ss.unpack_rm_imu(data.payload) count = imu_data.get_count() sample = imu_data.get_frame(0) - print(f'Got {count} samples at time {data.timestamp}, first sample is (ticks = {sample.vinyl_hup_ticks} | {sample.soc_ticks}, x = {sample.x}, y = {sample.y}, z = {sample.z}, temperature={sample.temperature})') + print(f'Got {count} samples at time {data.timestamp}') + print(f'First sample is (ticks = {sample.vinyl_hup_ticks} | {sample.soc_ticks}, x = {sample.x}, y = {sample.y}, z = {sample.z}, temperature={sample.temperature})') client.close() listener.join() From 16c06ee2eda0b190bad19f2fecc5c96ca263f6e6 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:26:12 -0400 Subject: [PATCH 25/50] rename samples --- ...o_2d_to_3d_segmentation.py => sample_2d_to_3d_segmentation.py} | 0 viewer/{demo_video.py => sample_video.py} | 0 2 files changed, 0 insertions(+), 0 deletions(-) rename viewer/{demo_2d_to_3d_segmentation.py => sample_2d_to_3d_segmentation.py} (100%) rename viewer/{demo_video.py => sample_video.py} (100%) diff --git a/viewer/demo_2d_to_3d_segmentation.py b/viewer/sample_2d_to_3d_segmentation.py similarity index 100% rename from viewer/demo_2d_to_3d_segmentation.py rename to viewer/sample_2d_to_3d_segmentation.py diff --git a/viewer/demo_video.py b/viewer/sample_video.py similarity index 100% rename from viewer/demo_video.py rename to viewer/sample_video.py From f39ced448a64a5011adedc06f569510e9094efcc Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:26:52 -0400 Subject: [PATCH 26/50] Update client_rm_vlc.py --- viewer/client_rm_vlc.py | 19 ++++++++++++------- 1 file changed, 12 insertions(+), 7 deletions(-) diff --git a/viewer/client_rm_vlc.py b/viewer/client_rm_vlc.py index 112a288f5..f366be8a7 100644 --- a/viewer/client_rm_vlc.py +++ b/viewer/client_rm_vlc.py @@ -1,8 +1,9 @@ #------------------------------------------------------------------------------ -# This script receives encoded video from one of the HoloLens sideview -# grayscale cameras and plays it. The camera resolution is 640x480 @ 30 FPS. -# The stream supports three operating modes: 0) video, 1) video + rig pose, -# 2) query calibration (single transfer). Press esc to stop. +# This script receives video from one of the HoloLens sideview grayscale +# cameras and plays it. The camera resolution is 640x480 @ 30 FPS. The stream +# supports three operating modes: 0) video, 1) video + rig pose, 2) query +# calibration (single transfer). +# Press esc to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -42,9 +43,13 @@ if (mode == hl2ss.StreamMode.MODE_2): data = hl2ss.download_calibration_rm_vlc(host, port) print('Calibration data') - print(data.uv2xy.shape) + print('Image point to unit plane') + print(data.uv2xy) + print('Extrinsics') print(data.extrinsics) - print(data.undistort_map.shape) + print('Undistort map') + print(data.undistort_map) + print('Intrinsics (undistorted only)') print(data.intrinsics) quit() @@ -63,7 +68,7 @@ def on_press(key): while (enable): data = client.get_next_packet() - print('Pose at time {ts}'.format(ts=data.timestamp)) + print(f'Pose at time {data.timestamp}') print(data.pose) cv2.imshow('Video', data.payload) cv2.waitKey(1) From 2fe16fdfafd1d0b64af5c71510a04edc342ebfda Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:36:45 -0400 Subject: [PATCH 27/50] Update client_si.py --- viewer/client_si.py | 36 +++++++++++++++++++----------------- 1 file changed, 19 insertions(+), 17 deletions(-) diff --git a/viewer/client_si.py b/viewer/client_si.py index b0c5d2a88..3a221efca 100644 --- a/viewer/client_si.py +++ b/viewer/client_si.py @@ -1,8 +1,6 @@ #------------------------------------------------------------------------------ # This script receives spatial input data from the HoloLens, which comprises: -# 1) Head pose, 2) Eye ray, 3) Hand tracking. The HoloLens sends this data at -# display framerate (60 Hz). If the display framerate drops, so does this -# stream. +# 1) Head pose, 2) Eye ray, 3) Hand tracking. Sample rate is 30 Hz. #------------------------------------------------------------------------------ import hl2ss @@ -23,14 +21,14 @@ data = client.get_next_packet() si = hl2ss.unpack_si(data.payload) -print('Tracking status at time {ts}'.format(ts=data.timestamp)) +print(f'Tracking status at time {data.timestamp}') if (si.is_valid_head_pose()): head_pose = si.get_head_pose() print('Head pose') - print(head_pose.position) - print(head_pose.forward) - print(head_pose.up) + print(f'Position: {head_pose.position}') + print(f'Forward: {head_pose.forward}') + print(f'Up: {head_pose.up}') # right = cross(up, -forward) # up => y, forward => -z, right => x else: @@ -39,19 +37,23 @@ if (si.is_valid_eye_ray()): eye_ray = si.get_eye_ray() print('Eye ray') - print(eye_ray.origin) - print(eye_ray.direction) + print(f'Origin: {eye_ray.origin}') + print(f'Direction: {eye_ray.direction}') else: print('No eye tracking data') +# See +# https://learn.microsoft.com/en-us/uwp/api/windows.perception.people.jointpose?view=winrt-22621 +# for hand data details + if (si.is_valid_hand_left()): hand_left = si.get_hand_left() pose = hand_left.get_joint_pose(hl2ss.SI_HandJointKind.Wrist) print('Left wrist pose') - print(pose.orientation) - print(pose.position) - print(pose.radius) - print(pose.accuracy) + print(f'Orientation: {pose.orientation}') # Quaternion: x, y, z, w + print(f'Position: {pose.position}') + print(f'Radius: {pose.radius}') + print(f'Accuracy: {pose.accuracy}') # 0: High, 1: Approximate else: print('No left hand data') @@ -59,10 +61,10 @@ hand_right = si.get_hand_right() pose = hand_right.get_joint_pose(hl2ss.SI_HandJointKind.Wrist) print('Right wrist pose') - print(pose.orientation) - print(pose.position) - print(pose.radius) - print(pose.accuracy) + print(f'Orientation: {pose.orientation}') # Quaternion: x, y, z, w + print(f'Position: {pose.position}') + print(f'Radius: {pose.radius}') + print(f'Accuracy: {pose.accuracy}') # 0: High, 1: Approximate else: print('No right hand data') From 747e940ede419c89acc554c43df0ebf04c4900b0 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:39:54 -0400 Subject: [PATCH 28/50] Update client_sm.py --- viewer/client_sm.py | 1 + 1 file changed, 1 insertion(+) diff --git a/viewer/client_sm.py b/viewer/client_sm.py index 101a6ab71..ba42bfeaf 100644 --- a/viewer/client_sm.py +++ b/viewer/client_sm.py @@ -77,6 +77,7 @@ mesh.unpack(vpf, tif, vnf) + # Surface timestamps are given in Windows FILETIME (utc) print(f'Task {index}: surface id {id_hex} @ {timestamp} has {mesh.vertex_positions.shape[0]} vertices {mesh.triangle_indices.shape[0]} triangles {mesh.vertex_normals.shape[0]} normals') mesh.vertex_positions[:, 0:3] = mesh.vertex_positions[:, 0:3] * mesh.vertex_position_scale From e3d7ae3fac86ac1dea975bb0a6dc3459c7fece51 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:47:27 -0400 Subject: [PATCH 29/50] Update client_su.py --- viewer/client_su.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/viewer/client_su.py b/viewer/client_su.py index 7e9e03a4f..d61331ab0 100644 --- a/viewer/client_su.py +++ b/viewer/client_su.py @@ -1,5 +1,6 @@ #------------------------------------------------------------------------------ -# This script downloads Scene Understanding data. +# This script downloads Scene Understanding data from the HoloLens and displays +# it. #------------------------------------------------------------------------------ import open3d as o3d @@ -28,7 +29,11 @@ get_quad = True get_meshes = True get_collider_meshes = True -guid_list = [] + +# To track surfaces between scenes +# Create a new scene using SU_Create.NewFromPrevious and add the GUID of the +# surface(s) of interest found in the previous scene +guid_list = [] #------------------------------------------------------------------------------ From ee5ff098d650dcf945874959f5bedbf92e116ac3 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:58:35 -0400 Subject: [PATCH 30/50] Update hl2ss.py --- viewer/hl2ss.py | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/viewer/hl2ss.py b/viewer/hl2ss.py index e6933eaa8..445d0be6c 100644 --- a/viewer/hl2ss.py +++ b/viewer/hl2ss.py @@ -1843,14 +1843,14 @@ class SU_Create: class SU_Kind: - Background = 0, - Wall = 1, - Floor = 2, - Ceiling = 3, - Platform = 4, - Unknown = 247, - World = 248, - CompletelyInferred = 249, + Background = 0 + Wall = 1 + Floor = 2 + Ceiling = 3 + Platform = 4 + Unknown = 247 + World = 248 + CompletelyInferred = 249 class su_task: From 814ca39b5afa4cb7b05d558b588aeda565c712c8 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 16:58:48 -0400 Subject: [PATCH 31/50] Update client_su.py --- viewer/client_su.py | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/viewer/client_su.py b/viewer/client_su.py index d61331ab0..a3ee43307 100644 --- a/viewer/client_su.py +++ b/viewer/client_su.py @@ -33,10 +33,22 @@ # To track surfaces between scenes # Create a new scene using SU_Create.NewFromPrevious and add the GUID of the # surface(s) of interest found in the previous scene +# If the surface is found in the new scene it will be returned guid_list = [] #------------------------------------------------------------------------------ +kind_color = { + hl2ss.SU_Kind.Background : [0, 0, 0], + hl2ss.SU_Kind.Ceiling : [0, 0, 1], + hl2ss.SU_Kind.CompletelyInferred : [0, 1, 0], + hl2ss.SU_Kind.Floor : [0, 1, 1], + hl2ss.SU_Kind.Platform : [1, 0, 0], + hl2ss.SU_Kind.Unknown : [1, 0, 1], + hl2ss.SU_Kind.Wall : [1, 1, 0], + hl2ss.SU_Kind.World : [1, 1, 1], +} + # Download data --------------------------------------------------------------- # See # https://learn.microsoft.com/en-us/windows/mixed-reality/develop/unity/scene-understanding-sdk @@ -90,6 +102,7 @@ open3d_mesh.vertices = o3d.utility.Vector3dVector((mesh.vertex_positions @ item.location[:3, :3]) + item.location[3, :3]) open3d_mesh.triangles = o3d.utility.Vector3iVector(mesh.triangle_indices) open3d_mesh.compute_vertex_normals() + open3d_mesh.paint_uniform_color(kind_color[int(item.kind)]) open3d_meshes.append(open3d_mesh) o3d.visualization.draw_geometries(open3d_meshes, mesh_show_back_face=True) From 2c2ba9b90f7b7f5b234fb76616c043d0efaa8235 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 17:05:24 -0400 Subject: [PATCH 32/50] Update client_vi.py --- viewer/client_vi.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/viewer/client_vi.py b/viewer/client_vi.py index bae01ab3e..5f304d463 100644 --- a/viewer/client_vi.py +++ b/viewer/client_vi.py @@ -1,5 +1,6 @@ #------------------------------------------------------------------------------ -# This script registers voice commands on the HoloLens. +# This script registers voice commands on the HoloLens and continously checks +# if any of the registered commands has been heard. # Press esc to stop. #------------------------------------------------------------------------------ @@ -46,12 +47,13 @@ def get_word(strings, index): client.create_recognizer() if (client.register_commands(True, strings)): print('Ready. Try saying any of the commands you defined.') - client.start() + client.start() while (enable): events = client.pop() for event in events: event.unpack() - print(f'Event: {get_word(strings, event.index)} {event.index} {event.confidence} {event.phrase_duration} {event.phrase_start_time} {event.raw_confidence}') + # Start timestamps are given in Windows FILETIME (utc) + print(f'Event: Command={get_word(strings, event.index)} Index={event.index} Confidence={event.confidence} Duration={event.phrase_duration} Start={event.phrase_start_time} RawConfidence={event.raw_confidence}') client.stop() client.clear() From 977aac6034f4ae1c4a7a76e7204c5e18b2e10966 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 17:13:56 -0400 Subject: [PATCH 33/50] Update client_su.py --- viewer/client_su.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/client_su.py b/viewer/client_su.py index a3ee43307..b85561d89 100644 --- a/viewer/client_su.py +++ b/viewer/client_su.py @@ -90,7 +90,7 @@ for item in result.items: item.unpack() - print(f'SceneObject {item.id.hex()} {item.kind} {item.orientation} {item.position} {item.alignment} {item.extents}') + print(f'SceneObject id={item.id.hex()} kind={item.kind} orientation={item.orientation} position={item.position} alignment={item.alignment} extents={item.extents}') print('Location') print(item.location) print(f'Meshes: {len(item.meshes)}') From d87891220b043eab9d6fac24d018cf02df2c560c Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 17:18:39 -0400 Subject: [PATCH 34/50] Update client_vi.py --- viewer/client_vi.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/viewer/client_vi.py b/viewer/client_vi.py index 5f304d463..5e28e4701 100644 --- a/viewer/client_vi.py +++ b/viewer/client_vi.py @@ -52,7 +52,9 @@ def get_word(strings, index): events = client.pop() for event in events: event.unpack() - # Start timestamps are given in Windows FILETIME (utc) + # See + # https://learn.microsoft.com/en-us/uwp/api/windows.media.speechrecognition.speechrecognitionresult?view=winrt-22621 + # for result details print(f'Event: Command={get_word(strings, event.index)} Index={event.index} Confidence={event.confidence} Duration={event.phrase_duration} Start={event.phrase_start_time} RawConfidence={event.raw_confidence}') client.stop() client.clear() From 1119d5183440b880a7a74d450dec0c0a60b52490 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 17:25:25 -0400 Subject: [PATCH 35/50] Update client_su.py --- viewer/client_su.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/client_su.py b/viewer/client_su.py index b85561d89..36c5fabfd 100644 --- a/viewer/client_su.py +++ b/viewer/client_su.py @@ -90,7 +90,7 @@ for item in result.items: item.unpack() - print(f'SceneObject id={item.id.hex()} kind={item.kind} orientation={item.orientation} position={item.position} alignment={item.alignment} extents={item.extents}') + print(f'SceneObject ID={item.id.hex()} Kind={item.kind} Orientation={item.orientation} Position={item.position} Alignment={item.alignment} Extents={item.extents}') print('Location') print(item.location) print(f'Meshes: {len(item.meshes)}') From 5697ed052720cd66fba195c0a32aee1b21c11304 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 17:39:07 -0400 Subject: [PATCH 36/50] Update hl2ss_io.py --- viewer/hl2ss_io.py | 109 +++++++++++++++++++++++++++++++++++++-------- 1 file changed, 91 insertions(+), 18 deletions(-) diff --git a/viewer/hl2ss_io.py b/viewer/hl2ss_io.py index 214e72743..be582c8bb 100644 --- a/viewer/hl2ss_io.py +++ b/viewer/hl2ss_io.py @@ -507,51 +507,124 @@ def _create_rd_eet(filename, chunk): # Reader Wrapper #------------------------------------------------------------------------------ -class _rd(hl2ss._context_manager): +class _rd_rm_vlc(hl2ss._context_manager): def __init__(self, filename, chunk): self.filename = filename self.chunk = chunk def open(self): - self._rd, self.header = self._create_rd(self.filename, self.chunk) + self._rd, self.header = _create_rd_rm_vlc(self.filename, self.chunk) def read(self): return self._rd.read() + + def close(self): + self._rd.close() + + +class _rd_rm_depth_ahat(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + + def open(self): + self._rd, self.header = _create_rd_rm_depth_ahat(self.filename, self.chunk) + def read(self): + return self._rd.read() + def close(self): self._rd.close() -class _rd_rm_vlc(_rd): - _create_rd = _create_rd_rm_vlc +class _rd_rm_depth_longthrow(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + def open(self): + self._rd, self.header = _create_rd_rm_depth_longthrow(self.filename, self.chunk) -class _rd_rm_depth_ahat(_rd): - _create_rd = _create_rd_rm_depth_ahat + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() -class _rd_rm_depth_longthrow(_rd): - _create_rd = _create_rd_rm_depth_longthrow +class _rd_rm_imu(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + def open(self): + self._rd, self.header = _create_rd_rm_imu(self.filename, self.chunk) -class _rd_rm_imu(_rd): - _create_rd = _create_rd_rm_imu + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() -class _rd_pv(_rd): - _create_rd = _create_rd_pv +class _rd_pv(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + def open(self): + self._rd, self.header = _create_rd_pv(self.filename, self.chunk) + + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() -class _rd_microphone(_rd): - _create_rd = _create_rd_microphone +class _rd_microphone(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk -class _rd_si(_rd): - _create_rd = _create_rd_si + def open(self): + self._rd, self.header = _create_rd_microphone(self.filename, self.chunk) + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() + + +class _rd_si(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + + def open(self): + self._rd, self.header = _create_rd_si(self.filename, self.chunk) -class _rd_eet(_rd): - _create_rd = _create_rd_eet + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() + + +class _rd_eet(hl2ss._context_manager): + def __init__(self, filename, chunk): + self.filename = filename + self.chunk = chunk + + def open(self): + self._rd, self.header = _create_rd_eet(self.filename, self.chunk) + + def read(self): + return self._rd.read() + + def close(self): + self._rd.close() #------------------------------------------------------------------------------ From 747e210ea153441f43d7fa26b40779a5da888c39 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:22:21 -0400 Subject: [PATCH 37/50] raw recording tests --- viewer/simple_player.py | 54 +++++++++++++++++++++++++++++++-------- viewer/simple_recorder.py | 39 ++++++++++++++++------------ 2 files changed, 67 insertions(+), 26 deletions(-) diff --git a/viewer/simple_player.py b/viewer/simple_player.py index 8633ec169..55b8a999c 100644 --- a/viewer/simple_player.py +++ b/viewer/simple_player.py @@ -10,7 +10,7 @@ path = './data' # Ports -port = hl2ss.StreamPort.RM_VLC_RIGHTFRONT +port = hl2ss.StreamPort.MICROPHONE ''' hl2ss.StreamPort.RM_VLC_LEFTFRONT, hl2ss.StreamPort.RM_VLC_LEFTLEFT, @@ -59,13 +59,18 @@ def display_pv(data): cv2.imshow('vlc', data.payload.image) cv2.waitKey(1) -microphone_buffer = np.empty((1,0), dtype=np.float32) -def display_microphone(data): + +def display_microphone_aac(data): global microphone_buffer print(f'Samples at time {data.timestamp}') microphone_buffer = np.hstack((microphone_buffer, hl2ss_utilities.microphone_planar_to_packed(data.payload))) +def display_microphone_raw(data): + global microphone_buffer + print(f'Samples at time {data.timestamp}') + microphone_buffer = np.hstack((microphone_buffer, data.payload)) + def display_si(data): si = hl2ss.unpack_si(data.payload) @@ -112,6 +117,33 @@ def display_si(data): else: print('No right hand data') +def display_eet(data): + data.payload = hl2ss.unpack_eet(data.payload) + + # See + # https://learn.microsoft.com/en-us/windows/mixed-reality/develop/native/extended-eye-tracking-native + # for details + + print(f'Tracking status at time {data.timestamp}') + print('Pose') + print(data.pose) + print(f'Calibration valid: {data.payload.calibration_valid}') + print(f'Combined eye gaze: Valid={data.payload.combined_ray_valid} Origin={data.payload.combined_ray.origin} Direction={data.payload.combined_ray.direction}') + print(f'Left eye gaze: Valid={data.payload.left_ray_valid} Origin={data.payload.left_ray.origin} Direction={data.payload.left_ray.direction}') + print(f'Right eye gaze: Valid={data.payload.right_ray_valid} Origin={data.payload.right_ray.origin} Direction={data.payload.right_ray.direction}') + + # "...not supported by HoloLens 2 at this time" + print(f'Left eye openness: Valid={data.payload.left_openness_valid} Value={data.payload.left_openness}') + print(f'Right eye openness: Valid={data.payload.right_openness_valid} Value={data.payload.right_openness}') + print(f'Vergence distance: Valid={data.payload.vergence_distance_valid} Value={data.payload.vergence_distance}') + + + +filename = os.path.join(path, f'{hl2ss.get_port_name(port)}.bin') +reader = hl2ss_io.create_rd(True, filename, hl2ss.ChunkSize.SINGLE_TRANSFER, 'bgr24') +reader.open() + +microphone_buffer = np.empty((1,0), dtype=np.int16 if (reader.header.profile == hl2ss.AudioProfile.RAW) else np.float32) display_method = { hl2ss.StreamPort.RM_VLC_LEFTFRONT : display_vlc, @@ -124,14 +156,11 @@ def display_si(data): hl2ss.StreamPort.RM_IMU_GYROSCOPE : display_imu, hl2ss.StreamPort.RM_IMU_MAGNETOMETER : display_imu, hl2ss.StreamPort.PERSONAL_VIDEO : display_pv, - hl2ss.StreamPort.MICROPHONE : display_microphone, - hl2ss.StreamPort.SPATIAL_INPUT : display_si + hl2ss.StreamPort.MICROPHONE : display_microphone_raw if (reader.header.profile == hl2ss.AudioProfile.RAW) else display_microphone_aac, + hl2ss.StreamPort.SPATIAL_INPUT : display_si, + hl2ss.StreamPort.EXTENDED_EYE_TRACKER : display_eet, } -filename = os.path.join(path, f'{hl2ss.get_port_name(port)}.bin') -reader = hl2ss_io.create_rd(True, filename, hl2ss.ChunkSize.SINGLE_TRANSFER, 'bgr24') -reader.open() - while (True): data = reader.read() if (data is None): @@ -140,8 +169,13 @@ def display_si(data): reader.close() +if (reader.header.port != hl2ss.StreamPort.MICROPHONE): + quit() + +print(reader.header.profile) + p = pyaudio.PyAudio() -stream = p.open(format=pyaudio.paFloat32, channels=hl2ss.Parameters_MICROPHONE.CHANNELS, rate=hl2ss.Parameters_MICROPHONE.SAMPLE_RATE, output=True) +stream = p.open(format=pyaudio.paInt16 if (reader.header.profile == hl2ss.AudioProfile.RAW) else pyaudio.paFloat32, channels=hl2ss.Parameters_MICROPHONE.CHANNELS, rate=hl2ss.Parameters_MICROPHONE.SAMPLE_RATE, output=True) stream.start_stream() stream.write(microphone_buffer.tobytes()) stream.stop_stream() diff --git a/viewer/simple_recorder.py b/viewer/simple_recorder.py index 7c35b8caf..9620e9252 100644 --- a/viewer/simple_recorder.py +++ b/viewer/simple_recorder.py @@ -1,5 +1,4 @@ -import multiprocessing as mp import time import os import hl2ss @@ -11,7 +10,7 @@ # Ports ports = [ - hl2ss.StreamPort.RM_VLC_LEFTFRONT, + #hl2ss.StreamPort.RM_VLC_LEFTFRONT, #hl2ss.StreamPort.RM_VLC_LEFTLEFT, #hl2ss.StreamPort.RM_VLC_RIGHTFRONT, #hl2ss.StreamPort.RM_VLC_RIGHTRIGHT, @@ -20,36 +19,43 @@ #hl2ss.StreamPort.RM_IMU_ACCELEROMETER, #hl2ss.StreamPort.RM_IMU_GYROSCOPE, #hl2ss.StreamPort.RM_IMU_MAGNETOMETER, - hl2ss.StreamPort.PERSONAL_VIDEO, + #hl2ss.StreamPort.PERSONAL_VIDEO, hl2ss.StreamPort.MICROPHONE, - #hl2ss.StreamPort.SPATIAL_INPUT + #hl2ss.StreamPort.SPATIAL_INPUT, + #hl2ss.StreamPort.EXTENDED_EYE_TRACKER, ] # RM VLC parameters vlc_mode = hl2ss.StreamMode.MODE_1 -vlc_profile = hl2ss.VideoProfile.H264_BASE -vlc_bitrate = 1*1024*1024 +vlc_profile = hl2ss.VideoProfile.RAW +vlc_bitrate = 2*1024*1024 # RM Depth AHAT parameters -ahat_mode = hl2ss.StreamMode.MODE_1 -ahat_profile = hl2ss.VideoProfile.H264_BASE +ahat_mode = hl2ss.StreamMode.MODE_1 +ahat_profile = hl2ss.VideoProfile.RAW ahat_bitrate = 8*1024*1024 # RM Depth Long Throw parameters -lt_mode = hl2ss.StreamMode.MODE_1 +lt_mode = hl2ss.StreamMode.MODE_1 lt_filter = hl2ss.PngFilterMode.Paeth # RM IMU parameters imu_mode = hl2ss.StreamMode.MODE_1 # PV parameters -pv_mode = hl2ss.StreamMode.MODE_1 -pv_width = 760 -pv_height = 428 +pv_mode = hl2ss.StreamMode.MODE_1 +pv_width = 760 +pv_height = 428 pv_framerate = 30 -pv_profile = hl2ss.VideoProfile.H264_BASE -pv_bitrate = 1*1024*1024 -pv_format = 'bgr24' +pv_profile = hl2ss.VideoProfile.RAW +pv_bitrate = 1*1024*1024 +pv_format = 'bgr24' + +# Microphone parameters +microphone_profile = hl2ss.AudioProfile.RAW + +# EET parameters +eet_fps = 90 # Maximum number of frames in buffer buffer_elements = 300 @@ -68,8 +74,9 @@ producer.configure_rm_imu(host, hl2ss.StreamPort.RM_IMU_GYROSCOPE, hl2ss.ChunkSize.RM_IMU_GYROSCOPE, imu_mode) producer.configure_rm_imu(host, hl2ss.StreamPort.RM_IMU_MAGNETOMETER, hl2ss.ChunkSize.RM_IMU_MAGNETOMETER, imu_mode) producer.configure_pv(False, host, hl2ss.StreamPort.PERSONAL_VIDEO, hl2ss.ChunkSize.PERSONAL_VIDEO, pv_mode, pv_width, pv_height, pv_framerate, pv_profile, pv_bitrate, pv_format) - producer.configure_microphone(False, host, hl2ss.StreamPort.MICROPHONE, hl2ss.ChunkSize.MICROPHONE, hl2ss.AudioProfile.AAC_24000) + producer.configure_microphone(False, host, hl2ss.StreamPort.MICROPHONE, hl2ss.ChunkSize.MICROPHONE, microphone_profile) producer.configure_si(host, hl2ss.StreamPort.SPATIAL_INPUT, hl2ss.ChunkSize.SPATIAL_INPUT) + producer.configure_eet(host, hl2ss.StreamPort.EXTENDED_EYE_TRACKER, hl2ss.ChunkSize.EXTENDED_EYE_TRACKER, eet_fps) for port in ports: producer.initialize(port, buffer_elements) From 10dcc1f03c6fa3c3f0f79e292a034b7dd0804979 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:25:39 -0400 Subject: [PATCH 38/50] rename rus --- viewer/{rus.py => hl2ss_rus.py} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename viewer/{rus.py => hl2ss_rus.py} (100%) diff --git a/viewer/rus.py b/viewer/hl2ss_rus.py similarity index 100% rename from viewer/rus.py rename to viewer/hl2ss_rus.py From 478a5fe3aa296e27634429a66c947ae2e35cf14c Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:26:24 -0400 Subject: [PATCH 39/50] rename rus 2 --- viewer/unity_demo_cube.py | 16 ++++++++-------- viewer/unity_demo_hud.py | 14 +++++++------- viewer/unity_demo_stickers.py | 10 +++++----- viewer/unity_demo_text.py | 12 ++++++------ 4 files changed, 26 insertions(+), 26 deletions(-) diff --git a/viewer/unity_demo_cube.py b/viewer/unity_demo_cube.py index 485129936..498a27016 100644 --- a/viewer/unity_demo_cube.py +++ b/viewer/unity_demo_cube.py @@ -6,7 +6,7 @@ from pynput import keyboard import hl2ss -import rus +import hl2ss_rus # Settings -------------------------------------------------------------------- @@ -45,15 +45,15 @@ def on_press(key): key = 0 -display_list = rus.command_buffer() +display_list = hl2ss_rus.command_buffer() display_list.begin_display_list() # Begin command sequence display_list.remove_all() # Remove all objects that were created remotely -display_list.create_primitive(rus.PrimitiveType.Cube) # Create a cube, server will return its id -display_list.set_target_mode(rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the cube +display_list.create_primitive(hl2ss_rus.PrimitiveType.Cube) # Create a cube, server will return its id +display_list.set_target_mode(hl2ss_rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the cube display_list.set_world_transform(key, position, rotation, scale) # Set the world transform of the cube display_list.set_color(key, rgba) # Set the color of the cube -display_list.set_active(key, rus.ActiveState.Active) # Make the cube visible -display_list.set_target_mode(rus.TargetMode.UseID) # Restore target mode +display_list.set_active(key, hl2ss_rus.ActiveState.Active) # Make the cube visible +display_list.set_target_mode(hl2ss_rus.TargetMode.UseID) # Restore target mode display_list.end_display_list() # End command sequence ipc.push(display_list) # Send commands to server results = ipc.pull(display_list) # Get results from server @@ -76,7 +76,7 @@ def on_press(key): position[2] = z - display_list = rus.command_buffer() + display_list = hl2ss_rus.command_buffer() display_list.begin_display_list() display_list.set_world_transform(key, position, rotation, scale) display_list.set_color(key, [z, 0, 1-z, 1-z]) # Semi-transparency is supported @@ -84,7 +84,7 @@ def on_press(key): ipc.push(display_list) results = ipc.pull(display_list) -command_buffer = rus.command_buffer() +command_buffer = hl2ss_rus.command_buffer() command_buffer.remove(key) # Destroy cube ipc.push(command_buffer) results = ipc.pull(command_buffer) diff --git a/viewer/unity_demo_hud.py b/viewer/unity_demo_hud.py index 3e501205b..06e183a07 100644 --- a/viewer/unity_demo_hud.py +++ b/viewer/unity_demo_hud.py @@ -7,7 +7,7 @@ import threading import hl2ss -import rus +import hl2ss_rus # Settings -------------------------------------------------------------------- @@ -57,15 +57,15 @@ def on_press(key): key = 0 -display_list = rus.command_buffer() +display_list = hl2ss_rus.command_buffer() display_list.begin_display_list() # Begin command sequence display_list.remove_all() # Remove all objects that were created remotely -display_list.create_primitive(rus.PrimitiveType.Quad) # Create a quad, server will return its id -display_list.set_target_mode(rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the quad +display_list.create_primitive(hl2ss_rus.PrimitiveType.Quad) # Create a quad, server will return its id +display_list.set_target_mode(hl2ss_rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the quad display_list.set_local_transform(key, position, rotation, scale) # Set the local transform of the cube display_list.set_texture(key, texture) # Set the texture of the quad -display_list.set_active(key, rus.ActiveState.Active) # Make the quad visible -display_list.set_target_mode(rus.TargetMode.UseID) # Restore target mode +display_list.set_active(key, hl2ss_rus.ActiveState.Active) # Make the quad visible +display_list.set_target_mode(hl2ss_rus.TargetMode.UseID) # Restore target mode display_list.end_display_list() # End command sequence ipc.push(display_list) # Send commands to server results = ipc.pull(display_list) # Get results from server @@ -73,7 +73,7 @@ def on_press(key): stop_event.wait() -command_buffer = rus.command_buffer() +command_buffer = hl2ss_rus.command_buffer() command_buffer.remove(key) # Destroy quad ipc.push(command_buffer) results = ipc.pull(command_buffer) diff --git a/viewer/unity_demo_stickers.py b/viewer/unity_demo_stickers.py index 208e0681b..ddd2cde6d 100644 --- a/viewer/unity_demo_stickers.py +++ b/viewer/unity_demo_stickers.py @@ -13,7 +13,7 @@ import hl2ss_imshow import hl2ss import hl2ss_3dcv -import rus +import hl2ss_rus import numpy as np # Settings -------------------------------------------------------------------- @@ -69,7 +69,7 @@ def on_release(key): key = 0 -command_buffer = rus.command_buffer() +command_buffer = hl2ss_rus.command_buffer() command_buffer.remove_all() ipc.push(command_buffer) results = ipc.pull(command_buffer) @@ -148,9 +148,9 @@ def on_release(key): rotation = [axis[0,0] * sin, axis[0,1] * sin, axis[0,2] * sin, cos] # Add quad to Unity scene - display_list = rus.command_buffer() + display_list = hl2ss_rus.command_buffer() display_list.begin_display_list() # Begin sequence - display_list.create_primitive(rus.PrimitiveType.Quad) # Create quad, returns id which can be used to modify its properties + display_list.create_primitive(hl2ss_rus.PrimitiveType.Quad) # Create quad, returns id which can be used to modify its properties display_list.set_target_mode(1) # Set server to use the last created object as target (this avoids waiting for the id) display_list.set_world_transform(key, centroid, rotation, scale) # Set the quad's world transform display_list.set_texture(key, image) # Set the quad's texture @@ -166,7 +166,7 @@ def on_release(key): print('Created quad with id {iid}'.format(iid=key)) -command_buffer = rus.command_buffer() +command_buffer = hl2ss_rus.command_buffer() command_buffer.remove_all() ipc.push(command_buffer) diff --git a/viewer/unity_demo_text.py b/viewer/unity_demo_text.py index 181814925..023b1d214 100644 --- a/viewer/unity_demo_text.py +++ b/viewer/unity_demo_text.py @@ -7,7 +7,7 @@ import threading import hl2ss -import rus +import hl2ss_rus # Settings -------------------------------------------------------------------- @@ -50,15 +50,15 @@ def on_press(key): key = 0 -display_list = rus.command_buffer() +display_list = hl2ss_rus.command_buffer() display_list.begin_display_list() # Begin command sequence display_list.remove_all() # Remove all objects that were created remotely display_list.create_text() # Create text object, server will return its id -display_list.set_target_mode(rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the text object +display_list.set_target_mode(hl2ss_rus.TargetMode.UseLast) # Set server to use the last created object as target, this avoids waiting for the id of the text object display_list.set_text(key, font_size, rgba, text) # Set text display_list.set_world_transform(key, position, rotation, [1, 1, 1]) # Set the world transform of the text object -display_list.set_active(key, rus.ActiveState.Active) # Make the text object visible -display_list.set_target_mode(rus.TargetMode.UseID) # Restore target mode +display_list.set_active(key, hl2ss_rus.ActiveState.Active) # Make the text object visible +display_list.set_target_mode(hl2ss_rus.TargetMode.UseID) # Restore target mode display_list.end_display_list() # End command sequence ipc.push(display_list) # Send commands to server results = ipc.pull(display_list) # Get results from server @@ -68,7 +68,7 @@ def on_press(key): stop_event.wait() -command_buffer = rus.command_buffer() +command_buffer = hl2ss_rus.command_buffer() command_buffer.remove(key) # Destroy text object ipc.push(command_buffer) results = ipc.pull(command_buffer) From bba09590aea089d479d8c7f33c577973cd4baedd Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:53:57 -0400 Subject: [PATCH 40/50] Update client_umq.py --- viewer/client_umq.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/client_umq.py b/viewer/client_umq.py index cead4f476..b3fd26854 100644 --- a/viewer/client_umq.py +++ b/viewer/client_umq.py @@ -28,7 +28,7 @@ def debug_message(self, text): # Command params: string encoded as utf-8 self.add(0xFFFFFFFE, text.encode('utf-8')) # Use the add method from hl2ss.umq_command_buffer to pack commands - # See rus.py and the unity_demo scripts for more examples. + # See hl2ss_rus.py and the unity_demo scripts for more examples. client = hl2ss.ipc_umq(host, hl2ss.IPCPort.UNITY_MESSAGE_QUEUE) # Create hl2ss client object From a5bda9d279969da3f595928c8678c4bfbb67438d Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:53:59 -0400 Subject: [PATCH 41/50] Update open3d_integrator_pv.py --- viewer/open3d_integrator_pv.py | 55 +++++++++++++++++++++++++++------- 1 file changed, 44 insertions(+), 11 deletions(-) diff --git a/viewer/open3d_integrator_pv.py b/viewer/open3d_integrator_pv.py index 725f98d38..313afa57e 100644 --- a/viewer/open3d_integrator_pv.py +++ b/viewer/open3d_integrator_pv.py @@ -2,7 +2,6 @@ # RGBD integration using Open3D. Color information comes from the front RGB # camera. Press space to stop. #------------------------------------------------------------------------------ -# TODO: USE CAMERA POSES FOR MORE ACCURATE REGISTRATION from pynput import keyboard @@ -19,15 +18,15 @@ # HoloLens address host = '192.168.1.7' +# Calibration path calibration_path = '../calibration' # Camera parameters -focus = 1000 width = 640 height = 360 framerate = 30 profile = hl2ss.VideoProfile.H265_MAIN -bitrate = 5*1024*1024 +bitrate = hl2ss.get_video_codec_bitrate(width, height, framerate, hl2ss.get_video_codec_default_factor(profile)) exposure_mode = hl2ss.PV_ExposureMode.Manual exposure = hl2ss.PV_ExposureValue.Max // 4 iso_speed_mode = hl2ss.PV_IsoSpeedMode.Manual @@ -45,6 +44,7 @@ #------------------------------------------------------------------------------ if __name__ == '__main__': + # Keyboard events --------------------------------------------------------- enable = True def on_press(key): @@ -55,13 +55,25 @@ def on_press(key): listener = keyboard.Listener(on_press=on_press) listener.start() + # Start PV Subsystem ------------------------------------------------------ hl2ss.start_subsystem_pv(host, hl2ss.StreamPort.PERSONAL_VIDEO) + # Wait for PV subsystem and fix exposure, iso speed, and white balance ---- + ipc_rc = hl2ss.ipc_rc(host, hl2ss.IPCPort.REMOTE_CONFIGURATION) + ipc_rc.open() + ipc_rc.wait_for_pv_subsystem(True) + ipc_rc.set_pv_exposure(exposure_mode, exposure) + ipc_rc.set_pv_iso_speed(iso_speed_mode, iso_speed_value) + ipc_rc.set_pv_white_balance_preset(white_balance) + ipc_rc.close() + + # Get RM Depth Long Throw calibration ------------------------------------- calibration_lt = hl2ss_3dcv.get_calibration_rm(host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, calibration_path) uv2xy = hl2ss_3dcv.compute_uv2xy(calibration_lt.intrinsics, hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT) xy1, scale = hl2ss_3dcv.rm_depth_compute_rays(uv2xy, calibration_lt.scale) + # Create Open3D integrator and visualizer --------------------------------- volume = o3d.pipelines.integration.ScalableTSDFVolume(voxel_length=voxel_length, sdf_trunc=sdf_trunc, color_type=o3d.pipelines.integration.TSDFVolumeColorType.RGB8) intrinsics_depth = o3d.camera.PinholeCameraIntrinsic(hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT, calibration_lt.intrinsics[0, 0], calibration_lt.intrinsics[1, 1], calibration_lt.intrinsics[2, 0], calibration_lt.intrinsics[2, 1]) @@ -69,6 +81,7 @@ def on_press(key): vis.create_window() first_pcd = True + # Start PV and RM Depth Long Throw streams -------------------------------- producer = hl2ss_mp.producer() producer.configure_pv(True, host, hl2ss.StreamPort.PERSONAL_VIDEO, hl2ss.ChunkSize.PERSONAL_VIDEO, hl2ss.StreamMode.MODE_1, width, height, framerate, profile, bitrate, 'rgb24') producer.configure_rm_depth_longthrow(True, host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.ChunkSize.RM_DEPTH_LONGTHROW, hl2ss.StreamMode.MODE_1, hl2ss.PngFilterMode.Paeth) @@ -77,20 +90,24 @@ def on_press(key): producer.start(hl2ss.StreamPort.PERSONAL_VIDEO) producer.start(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) - manager = mp.Manager() consumer = hl2ss_mp.consumer() + manager = mp.Manager() sink_pv = consumer.create_sink(producer, hl2ss.StreamPort.PERSONAL_VIDEO, manager, None) sink_depth = consumer.create_sink(producer, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, manager, ...) - sinks = [sink_pv, sink_depth] - - [sink.get_attach_response() for sink in sinks] + sink_pv.get_attach_response() + sink_depth.get_attach_response() + # Initialize PV intrinsics and extrinsics --------------------------------- pv_intrinsics = hl2ss.create_pv_intrinsics_placeholder() + pv_extrinsics = np.eye(4, 4, dtype=np.float32) + # Main Loop --------------------------------------------------------------- while (enable): + # Wait for RM Depth Long Throw frame ---------------------------------- sink_depth.acquire() + # Get RM Depth Long Throw frame and nearest (in time) PV frame -------- _, data_lt = sink_depth.get_most_recent_frame() if ((data_lt is None) or (not hl2ss.is_valid_pose(data_lt.pose))): continue @@ -99,15 +116,21 @@ def on_press(key): if ((data_pv is None) or (not hl2ss.is_valid_pose(data_pv.pose))): continue + # Preprocess frames --------------------------------------------------- depth = hl2ss_3dcv.rm_depth_undistort(data_lt.payload.depth, calibration_lt.undistort_map) depth = hl2ss_3dcv.rm_depth_normalize(depth, scale) color = data_pv.payload.image - pv_intrinsics = hl2ss.update_pv_intrinsics(pv_intrinsics, data_pv.payload.focal_length, data_pv.payload.principal_point) + # Update PV intrinsics ------------------------------------------------ + # PV intrinsics may change between frames due to autofocus + pv_intrinsics = hl2ss.update_pv_intrinsics(pv_intrinsics, data_pv.payload.focal_length, data_pv.payload.principal_point) + color_intrinsics, color_extrinsics = hl2ss_3dcv.pv_fix_calibration(pv_intrinsics, pv_extrinsics) + + # Generate aligned RGBD image ----------------------------------------- lt_points = hl2ss_3dcv.rm_depth_to_points(xy1, depth) lt_to_world = hl2ss_3dcv.camera_to_rignode(calibration_lt.extrinsics) @ hl2ss_3dcv.reference_to_world(data_lt.pose) world_to_lt = hl2ss_3dcv.world_to_reference(data_lt.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) - world_to_pv_image = hl2ss_3dcv.world_to_reference(data_pv.pose) @ hl2ss_3dcv.camera_to_image(pv_intrinsics) + world_to_pv_image = hl2ss_3dcv.world_to_reference(data_pv.pose) @ hl2ss_3dcv.rignode_to_camera(color_extrinsics) @ hl2ss_3dcv.camera_to_image(color_intrinsics) world_points = hl2ss_3dcv.transform(lt_points, lt_to_world) pv_uv = hl2ss_3dcv.project(world_points, world_to_pv_image) color = cv2.remap(color, pv_uv[:, :, 0], pv_uv[:, :, 1], cv2.INTER_LINEAR) @@ -115,12 +138,15 @@ def on_press(key): mask_uv = hl2ss_3dcv.slice_to_block((pv_uv[:, :, 0] < 0) | (pv_uv[:, :, 0] >= width) | (pv_uv[:, :, 1] < 0) | (pv_uv[:, :, 1] >= height)) depth[mask_uv] = 0 + # Convert to Open3D RGBD image ---------------------------------------- color_image = o3d.geometry.Image(color) depth_image = o3d.geometry.Image(depth) rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(color_image, depth_image, depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) + # Compute world to RM Depth Long Throw camera transformation matrix --- depth_world_to_camera = hl2ss_3dcv.world_to_reference(data_lt.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) + # Integrate RGBD and display point cloud ------------------------------ volume.integrate(rgbd, intrinsics_depth, depth_world_to_camera.transpose()) pcd_tmp = volume.extract_point_cloud() @@ -136,11 +162,18 @@ def on_press(key): vis.poll_events() vis.update_renderer() - [sink.detach() for sink in sinks] + # Stop PV and RM Depth Long Throw streams --------------------------------- + sink_pv.detach() + sink_depth.detach() producer.stop(hl2ss.StreamPort.PERSONAL_VIDEO) producer.stop(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) + + # Stop PV subsystem ------------------------------------------------------- + hl2ss.stop_subsystem_pv(host, hl2ss.StreamPort.PERSONAL_VIDEO) + + # Stop keyboard events ---------------------------------------------------- listener.join() + # Show final point cloud -------------------------------------------------- vis.run() - hl2ss.stop_subsystem_pv(host, hl2ss.StreamPort.PERSONAL_VIDEO) From 39f67bd3f6b040854c75ecd82e594805bd4e5d6d Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:55:55 -0400 Subject: [PATCH 42/50] Update open3d_integrator_pv.py --- viewer/open3d_integrator_pv.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/viewer/open3d_integrator_pv.py b/viewer/open3d_integrator_pv.py index 313afa57e..30168b25d 100644 --- a/viewer/open3d_integrator_pv.py +++ b/viewer/open3d_integrator_pv.py @@ -1,6 +1,7 @@ #------------------------------------------------------------------------------ # RGBD integration using Open3D. Color information comes from the front RGB -# camera. Press space to stop. +# camera. +# Press space to stop. #------------------------------------------------------------------------------ from pynput import keyboard From 87774945f74d4737b7710411bc9d381a7d9ba2ef Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 18:58:12 -0400 Subject: [PATCH 43/50] Update open3d_integrator_pv.py --- viewer/open3d_integrator_pv.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/viewer/open3d_integrator_pv.py b/viewer/open3d_integrator_pv.py index 30168b25d..a36a7910f 100644 --- a/viewer/open3d_integrator_pv.py +++ b/viewer/open3d_integrator_pv.py @@ -19,7 +19,7 @@ # HoloLens address host = '192.168.1.7' -# Calibration path +# Calibration path (must exist but can be empty) calibration_path = '../calibration' # Camera parameters @@ -69,6 +69,7 @@ def on_press(key): ipc_rc.close() # Get RM Depth Long Throw calibration ------------------------------------- + # Calibration data will be downloaded if it's not in the calibration folder calibration_lt = hl2ss_3dcv.get_calibration_rm(host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, calibration_path) uv2xy = hl2ss_3dcv.compute_uv2xy(calibration_lt.intrinsics, hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT) From 86e2227a962dc950152fc524a462948100716b35 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:08:40 -0400 Subject: [PATCH 44/50] Update open3d_integrator_rm_vlc.py --- viewer/open3d_integrator_rm_vlc.py | 59 +++++++++++++++++++++--------- 1 file changed, 41 insertions(+), 18 deletions(-) diff --git a/viewer/open3d_integrator_rm_vlc.py b/viewer/open3d_integrator_rm_vlc.py index 07f054732..be796ee23 100644 --- a/viewer/open3d_integrator_rm_vlc.py +++ b/viewer/open3d_integrator_rm_vlc.py @@ -1,6 +1,7 @@ #------------------------------------------------------------------------------ -# RGBD integration using Open3D. Color information comes from one of the -# sideview grayscale cameras. Press space to stop. +# RGBD integration using Open3D. "Color" information comes from one of the +# sideview grayscale cameras. +# Press space to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -17,12 +18,13 @@ # HoloLens address host = '192.168.1.7' +# Calibration path (must exist but can be empty) calibration_path = '../calibration' # Camera selection and parameters port = hl2ss.StreamPort.RM_VLC_LEFTFRONT -profile = hl2ss.VideoProfile.H264_HIGH -bitrate = 2*1024*1024 +profile = hl2ss.VideoProfile.H265_MAIN +bitrate = hl2ss.get_video_codec_bitrate(hl2ss.Parameters_RM_VLC.WIDTH, hl2ss.Parameters_RM_VLC.HEIGHT, hl2ss.Parameters_RM_VLC.FPS, hl2ss.get_video_codec_default_factor(profile)) # Buffer length in seconds buffer_length = 10 @@ -35,6 +37,7 @@ #------------------------------------------------------------------------------ if __name__ == '__main__': + # Keyboard events --------------------------------------------------------- enable = True def on_press(key): @@ -45,18 +48,23 @@ def on_press(key): listener = keyboard.Listener(on_press=on_press) listener.start() + # Get RM VLC and RM Depth Long Throw calibration -------------------------- + # Calibration data will be downloaded if it's not in the calibration folder calibration_vlc = hl2ss_3dcv.get_calibration_rm(host, port, calibration_path) calibration_lt = hl2ss_3dcv.get_calibration_rm(host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, calibration_path) uv2xy = hl2ss_3dcv.compute_uv2xy(calibration_lt.intrinsics, hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT) xy1, scale = hl2ss_3dcv.rm_depth_compute_rays(uv2xy, calibration_lt.scale) + # Create Open3D integrator and visualizer --------------------------------- volume = o3d.pipelines.integration.ScalableTSDFVolume(voxel_length=voxel_length, sdf_trunc=sdf_trunc, color_type=o3d.pipelines.integration.TSDFVolumeColorType.RGB8) intrinsics_depth = o3d.camera.PinholeCameraIntrinsic(hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT, calibration_lt.intrinsics[0, 0], calibration_lt.intrinsics[1, 1], calibration_lt.intrinsics[2, 0], calibration_lt.intrinsics[2, 1]) + vis = o3d.visualization.Visualizer() vis.create_window() first_pcd = True + # Start RM VLC and RM Depth Long Throw streams ---------------------------- producer = hl2ss_mp.producer() producer.configure_rm_vlc(True, host, port, hl2ss.ChunkSize.RM_VLC, hl2ss.StreamMode.MODE_1, profile, bitrate) producer.configure_rm_depth_longthrow(True, host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.ChunkSize.RM_DEPTH_LONGTHROW, hl2ss.StreamMode.MODE_1, hl2ss.PngFilterMode.Paeth) @@ -65,18 +73,20 @@ def on_press(key): producer.start(port) producer.start(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) - manager = mp.Manager() consumer = hl2ss_mp.consumer() + manager = mp.Manager() sink_vlc = consumer.create_sink(producer, port, manager, None) sink_depth = consumer.create_sink(producer, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, manager, ...) - sinks = [sink_vlc, sink_depth] - - [sink.get_attach_response() for sink in sinks] + sink_vlc.get_attach_response() + sink_depth.get_attach_response() + # Main Loop --------------------------------------------------------------- while (enable): + # Wait for RM Depth Long Throw frame ---------------------------------- sink_depth.acquire() + # Get RM Depth Long Throw frame and nearest (in time) RM VLC frame -------- _, data_depth = sink_depth.get_most_recent_frame() if ((data_depth is None) or (not hl2ss.is_valid_pose(data_depth.pose))): continue @@ -85,25 +95,33 @@ def on_press(key): if ((data_vlc is None) or (not hl2ss.is_valid_pose(data_vlc.pose))): continue + # Preprocess frames --------------------------------------------------- depth = hl2ss_3dcv.rm_depth_undistort(data_depth.payload.depth, calibration_lt.undistort_map) depth = hl2ss_3dcv.rm_depth_normalize(depth, scale) color = cv2.remap(data_vlc.payload, calibration_vlc.undistort_map[:, :, 0], calibration_vlc.undistort_map[:, :, 1], cv2.INTER_LINEAR) - lt_points = hl2ss_3dcv.rm_depth_to_points(xy1, depth) - lt_to_world = hl2ss_3dcv.camera_to_rignode(calibration_lt.extrinsics) @ hl2ss_3dcv.reference_to_world(data_depth.pose) - world_to_lt = hl2ss_3dcv.world_to_reference(data_depth.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) - world_to_pv_image = hl2ss_3dcv.world_to_reference(data_vlc.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_vlc.extrinsics) @ hl2ss_3dcv.camera_to_image(calibration_vlc.intrinsics) - world_points = hl2ss_3dcv.transform(lt_points, lt_to_world) - pv_uv = hl2ss_3dcv.project(world_points, world_to_pv_image) - color = cv2.remap(color, pv_uv[:, :, 0], pv_uv[:, :, 1], cv2.INTER_LINEAR) + # Generate aligned RGBD image ----------------------------------------- + lt_points = hl2ss_3dcv.rm_depth_to_points(xy1, depth) + lt_to_world = hl2ss_3dcv.camera_to_rignode(calibration_lt.extrinsics) @ hl2ss_3dcv.reference_to_world(data_depth.pose) + world_to_lt = hl2ss_3dcv.world_to_reference(data_depth.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) + world_to_vlc_image = hl2ss_3dcv.world_to_reference(data_vlc.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_vlc.extrinsics) @ hl2ss_3dcv.camera_to_image(calibration_vlc.intrinsics) + world_points = hl2ss_3dcv.transform(lt_points, lt_to_world) + vlc_uv = hl2ss_3dcv.project(world_points, world_to_vlc_image) + color = cv2.remap(color, vlc_uv[:, :, 0], vlc_uv[:, :, 1], cv2.INTER_LINEAR) - mask_uv = hl2ss_3dcv.slice_to_block((pv_uv[:, :, 0] < 0) | (pv_uv[:, :, 0] >= hl2ss.Parameters_RM_VLC.WIDTH) | (pv_uv[:, :, 1] < 0) | (pv_uv[:, :, 1] >= hl2ss.Parameters_RM_VLC.HEIGHT)) + mask_uv = hl2ss_3dcv.slice_to_block((vlc_uv[:, :, 0] < 0) | (vlc_uv[:, :, 0] >= hl2ss.Parameters_RM_VLC.WIDTH) | (vlc_uv[:, :, 1] < 0) | (vlc_uv[:, :, 1] >= hl2ss.Parameters_RM_VLC.HEIGHT)) depth[mask_uv] = 0 + # Convert to Open3D RGBD image ---------------------------------------- color = hl2ss_3dcv.rm_vlc_to_rgb(color) - rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(o3d.geometry.Image(color), o3d.geometry.Image(depth), depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) + color_image = o3d.geometry.Image(color) + depth_image = o3d.geometry.Image(depth) + rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(color_image, depth_image, depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) + + # Compute world to RM Depth Long Throw camera transformation matrix --- depth_world_to_camera = hl2ss_3dcv.world_to_reference(data_depth.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) + # Integrate RGBD and display point cloud ------------------------------ volume.integrate(rgbd, intrinsics_depth, depth_world_to_camera.transpose()) pcd_tmp = volume.extract_point_cloud() @@ -119,9 +137,14 @@ def on_press(key): vis.poll_events() vis.update_renderer() - [sink.detach() for sink in sinks] + # Stop RM VLC and RM Depth Long Throw streams ----------------------------- + sink_vlc.detach() + sink_depth.detach() producer.stop(port) producer.stop(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) + + # Stop keyboard events ---------------------------------------------------- listener.join() + # Show final point cloud -------------------------------------------------- vis.run() From e1b3fe5eaeb44545d66ccd265885c9630e75c6c0 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:12:51 -0400 Subject: [PATCH 45/50] Update open3d_integrator_rm_vlc.py --- viewer/open3d_integrator_rm_vlc.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/open3d_integrator_rm_vlc.py b/viewer/open3d_integrator_rm_vlc.py index be796ee23..886288b58 100644 --- a/viewer/open3d_integrator_rm_vlc.py +++ b/viewer/open3d_integrator_rm_vlc.py @@ -86,7 +86,7 @@ def on_press(key): # Wait for RM Depth Long Throw frame ---------------------------------- sink_depth.acquire() - # Get RM Depth Long Throw frame and nearest (in time) RM VLC frame -------- + # Get RM Depth Long Throw frame and nearest (in time) RM VLC frame ---- _, data_depth = sink_depth.get_most_recent_frame() if ((data_depth is None) or (not hl2ss.is_valid_pose(data_depth.pose))): continue From 7617878598439cce6d05363a71a7fc955ddcb112 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:15:54 -0400 Subject: [PATCH 46/50] Update open3d_integrator.py --- viewer/open3d_integrator.py | 37 ++++++++++++++++++++++++++++--------- 1 file changed, 28 insertions(+), 9 deletions(-) diff --git a/viewer/open3d_integrator.py b/viewer/open3d_integrator.py index a6ffbc784..86432994c 100644 --- a/viewer/open3d_integrator.py +++ b/viewer/open3d_integrator.py @@ -1,5 +1,6 @@ #------------------------------------------------------------------------------ -# RGBD integration using Open3D. Press space to stop. +# RGBD integration using Open3D. "Color" information come from AB component. +# Press space to stop. #------------------------------------------------------------------------------ from pynput import keyboard @@ -16,7 +17,7 @@ # HoloLens address host = '192.168.1.7' -# Calibration path +# Calibration path (must exist but can be empty) calibration_path = '../calibration' # Buffer length in seconds @@ -30,6 +31,7 @@ #------------------------------------------------------------------------------ if __name__ == '__main__': + # Keyboard events --------------------------------------------------------- enable = True def on_press(key): @@ -40,46 +42,59 @@ def on_press(key): listener = keyboard.Listener(on_press=on_press) listener.start() + # Get RM Depth Long Throw calibration ------------------------------------- + # Calibration data will be downloaded if it's not in the calibration folder calibration_lt = hl2ss_3dcv.get_calibration_rm(host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, calibration_path) uv2xy = hl2ss_3dcv.compute_uv2xy(calibration_lt.intrinsics, hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT) xy1, scale = hl2ss_3dcv.rm_depth_compute_rays(uv2xy, calibration_lt.scale) + # Create Open3D integrator and visualizer --------------------------------- volume = o3d.pipelines.integration.ScalableTSDFVolume(voxel_length=voxel_length, sdf_trunc=sdf_trunc, color_type=o3d.pipelines.integration.TSDFVolumeColorType.RGB8) intrinsics_depth = o3d.camera.PinholeCameraIntrinsic(hl2ss.Parameters_RM_DEPTH_LONGTHROW.WIDTH, hl2ss.Parameters_RM_DEPTH_LONGTHROW.HEIGHT, calibration_lt.intrinsics[0, 0], calibration_lt.intrinsics[1, 1], calibration_lt.intrinsics[2, 0], calibration_lt.intrinsics[2, 1]) + vis = o3d.visualization.Visualizer() vis.create_window() first_pcd = True + # Start RM Depth Long Throw stream ---------------------------------------- producer = hl2ss_mp.producer() producer.configure_rm_depth_longthrow(True, host, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.ChunkSize.RM_DEPTH_LONGTHROW, hl2ss.StreamMode.MODE_1, hl2ss.PngFilterMode.Paeth) producer.initialize(hl2ss.StreamPort.RM_DEPTH_LONGTHROW, hl2ss.Parameters_RM_DEPTH_LONGTHROW.FPS * buffer_length) producer.start(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) - manager = mp.Manager() consumer = hl2ss_mp.consumer() + manager = mp.Manager() sink_depth = consumer.create_sink(producer, hl2ss.StreamPort.RM_DEPTH_LONGTHROW, manager, ...) - sinks = [sink_depth] - - [sink.get_attach_response() for sink in sinks] + sink_depth.get_attach_response() + # Main Loop --------------------------------------------------------------- while (enable): + # Wait for RM Depth Long Throw frame ---------------------------------- sink_depth.acquire() + # Get RM Depth Long Throw frame --------------------------------------- _, data_depth = sink_depth.get_most_recent_frame() if ((data_depth is None) or (not hl2ss.is_valid_pose(data_depth.pose))): continue + # Preprocess frames --------------------------------------------------- depth = hl2ss_3dcv.rm_depth_undistort(data_depth.payload.depth, calibration_lt.undistort_map) depth = hl2ss_3dcv.rm_depth_normalize(depth, scale) color = cv2.remap(data_depth.payload.ab, calibration_lt.undistort_map[:, :, 0], calibration_lt.undistort_map[:, :, 1], cv2.INTER_LINEAR) + + # Convert to Open3D RGBD image ---------------------------------------- color = hl2ss_3dcv.rm_depth_to_uint8(color) color = hl2ss_3dcv.rm_depth_to_rgb(color) - - rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(o3d.geometry.Image(color), o3d.geometry.Image(depth), depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) + color_image = o3d.geometry.Image(color) + depth_image = o3d.geometry.Image(depth) + rgbd = o3d.geometry.RGBDImage.create_from_color_and_depth(color_image, depth_image, depth_scale=1, depth_trunc=max_depth, convert_rgb_to_intensity=False) + + # Compute world to RM Depth Long Throw camera transformation matrix --- depth_world_to_camera = hl2ss_3dcv.world_to_reference(data_depth.pose) @ hl2ss_3dcv.rignode_to_camera(calibration_lt.extrinsics) + # Integrate RGBD and display point cloud ------------------------------ volume.integrate(rgbd, intrinsics_depth, depth_world_to_camera.transpose()) pcd_tmp = volume.extract_point_cloud() @@ -95,8 +110,12 @@ def on_press(key): vis.poll_events() vis.update_renderer() - [sink.detach() for sink in sinks] + # Stop RM Depth Long Throw stream ----------------------------------------- + sink_depth.detach() producer.stop(hl2ss.StreamPort.RM_DEPTH_LONGTHROW) + + # Stop keyboard events ---------------------------------------------------- listener.join() + # Show final point cloud -------------------------------------------------- vis.run() From 1502a95a4d64ed2b2ddceecd4e67eb6bdef50e27 Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:19:44 -0400 Subject: [PATCH 47/50] rename samples --- viewer/{open3d_integrator.py => sample_integrator.py} | 0 viewer/{open3d_integrator_pv.py => sample_integrator_pv.py} | 0 .../{open3d_integrator_rm_vlc.py => sample_integrator_rm_vlc.py} | 0 3 files changed, 0 insertions(+), 0 deletions(-) rename viewer/{open3d_integrator.py => sample_integrator.py} (100%) rename viewer/{open3d_integrator_pv.py => sample_integrator_pv.py} (100%) rename viewer/{open3d_integrator_rm_vlc.py => sample_integrator_rm_vlc.py} (100%) diff --git a/viewer/open3d_integrator.py b/viewer/sample_integrator.py similarity index 100% rename from viewer/open3d_integrator.py rename to viewer/sample_integrator.py diff --git a/viewer/open3d_integrator_pv.py b/viewer/sample_integrator_pv.py similarity index 100% rename from viewer/open3d_integrator_pv.py rename to viewer/sample_integrator_pv.py diff --git a/viewer/open3d_integrator_rm_vlc.py b/viewer/sample_integrator_rm_vlc.py similarity index 100% rename from viewer/open3d_integrator_rm_vlc.py rename to viewer/sample_integrator_rm_vlc.py From 73e945cf32c758e7cbb606c199c6bc47287c80db Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:25:33 -0400 Subject: [PATCH 48/50] rename samples --- viewer/{open3d_pointcloud.py => sample_depth_to_pointcloud.py} | 1 - viewer/{open3d_rgbd.py => sample_rgbd.py} | 0 viewer/{open3d_rgbd_ahat_vlc.py => sample_rgbd_ahat_vlc.py} | 0 viewer/{open3d_rgbd_pv.py => sample_rgbd_pv.py} | 0 viewer/{open3d_rgbd_rm_vlc.py => sample_rgbd_rm_vlc.py} | 0 viewer/{opencv_stereo_rectify.py => sample_stereo_rectify.py} | 0 .../{opencv_stereo_rectify_pv.py => sample_stereo_rectify_pv.py} | 0 viewer/{open3d_viewer_si.py => sample_viewer_si.py} | 0 8 files changed, 1 deletion(-) rename viewer/{open3d_pointcloud.py => sample_depth_to_pointcloud.py} (98%) rename viewer/{open3d_rgbd.py => sample_rgbd.py} (100%) rename viewer/{open3d_rgbd_ahat_vlc.py => sample_rgbd_ahat_vlc.py} (100%) rename viewer/{open3d_rgbd_pv.py => sample_rgbd_pv.py} (100%) rename viewer/{open3d_rgbd_rm_vlc.py => sample_rgbd_rm_vlc.py} (100%) rename viewer/{opencv_stereo_rectify.py => sample_stereo_rectify.py} (100%) rename viewer/{opencv_stereo_rectify_pv.py => sample_stereo_rectify_pv.py} (100%) rename viewer/{open3d_viewer_si.py => sample_viewer_si.py} (100%) diff --git a/viewer/open3d_pointcloud.py b/viewer/sample_depth_to_pointcloud.py similarity index 98% rename from viewer/open3d_pointcloud.py rename to viewer/sample_depth_to_pointcloud.py index c3d1d2beb..7a464197d 100644 --- a/viewer/open3d_pointcloud.py +++ b/viewer/sample_depth_to_pointcloud.py @@ -6,7 +6,6 @@ import open3d as o3d import hl2ss import hl2ss_3dcv -import hl2ss_utilities # Settings -------------------------------------------------------------------- diff --git a/viewer/open3d_rgbd.py b/viewer/sample_rgbd.py similarity index 100% rename from viewer/open3d_rgbd.py rename to viewer/sample_rgbd.py diff --git a/viewer/open3d_rgbd_ahat_vlc.py b/viewer/sample_rgbd_ahat_vlc.py similarity index 100% rename from viewer/open3d_rgbd_ahat_vlc.py rename to viewer/sample_rgbd_ahat_vlc.py diff --git a/viewer/open3d_rgbd_pv.py b/viewer/sample_rgbd_pv.py similarity index 100% rename from viewer/open3d_rgbd_pv.py rename to viewer/sample_rgbd_pv.py diff --git a/viewer/open3d_rgbd_rm_vlc.py b/viewer/sample_rgbd_rm_vlc.py similarity index 100% rename from viewer/open3d_rgbd_rm_vlc.py rename to viewer/sample_rgbd_rm_vlc.py diff --git a/viewer/opencv_stereo_rectify.py b/viewer/sample_stereo_rectify.py similarity index 100% rename from viewer/opencv_stereo_rectify.py rename to viewer/sample_stereo_rectify.py diff --git a/viewer/opencv_stereo_rectify_pv.py b/viewer/sample_stereo_rectify_pv.py similarity index 100% rename from viewer/opencv_stereo_rectify_pv.py rename to viewer/sample_stereo_rectify_pv.py diff --git a/viewer/open3d_viewer_si.py b/viewer/sample_viewer_si.py similarity index 100% rename from viewer/open3d_viewer_si.py rename to viewer/sample_viewer_si.py From 01e8d310c2251e0e180fb526afd8362a0e9817bd Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:48:55 -0400 Subject: [PATCH 49/50] Update simple_player.py --- viewer/simple_player.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/viewer/simple_player.py b/viewer/simple_player.py index 55b8a999c..5eac24898 100644 --- a/viewer/simple_player.py +++ b/viewer/simple_player.py @@ -10,7 +10,7 @@ path = './data' # Ports -port = hl2ss.StreamPort.MICROPHONE +port = hl2ss.StreamPort.RM_VLC_RIGHTFRONT ''' hl2ss.StreamPort.RM_VLC_LEFTFRONT, hl2ss.StreamPort.RM_VLC_LEFTLEFT, From e1a607cb0838c0d6b817a9a4ab87c211922e03ec Mon Sep 17 00:00:00 2001 From: jdibenes <58836755+jdibenes@users.noreply.github.com> Date: Sat, 29 Apr 2023 19:49:00 -0400 Subject: [PATCH 50/50] Update simple_recorder.py --- viewer/simple_recorder.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/viewer/simple_recorder.py b/viewer/simple_recorder.py index 9620e9252..8118b8321 100644 --- a/viewer/simple_recorder.py +++ b/viewer/simple_recorder.py @@ -10,7 +10,7 @@ # Ports ports = [ - #hl2ss.StreamPort.RM_VLC_LEFTFRONT, + hl2ss.StreamPort.RM_VLC_LEFTFRONT, #hl2ss.StreamPort.RM_VLC_LEFTLEFT, #hl2ss.StreamPort.RM_VLC_RIGHTFRONT, #hl2ss.StreamPort.RM_VLC_RIGHTRIGHT, @@ -19,7 +19,7 @@ #hl2ss.StreamPort.RM_IMU_ACCELEROMETER, #hl2ss.StreamPort.RM_IMU_GYROSCOPE, #hl2ss.StreamPort.RM_IMU_MAGNETOMETER, - #hl2ss.StreamPort.PERSONAL_VIDEO, + hl2ss.StreamPort.PERSONAL_VIDEO, hl2ss.StreamPort.MICROPHONE, #hl2ss.StreamPort.SPATIAL_INPUT, #hl2ss.StreamPort.EXTENDED_EYE_TRACKER, @@ -27,12 +27,12 @@ # RM VLC parameters vlc_mode = hl2ss.StreamMode.MODE_1 -vlc_profile = hl2ss.VideoProfile.RAW +vlc_profile = hl2ss.VideoProfile.H264_MAIN vlc_bitrate = 2*1024*1024 # RM Depth AHAT parameters ahat_mode = hl2ss.StreamMode.MODE_1 -ahat_profile = hl2ss.VideoProfile.RAW +ahat_profile = hl2ss.VideoProfile.H264_MAIN ahat_bitrate = 8*1024*1024 # RM Depth Long Throw parameters @@ -47,12 +47,12 @@ pv_width = 760 pv_height = 428 pv_framerate = 30 -pv_profile = hl2ss.VideoProfile.RAW +pv_profile = hl2ss.VideoProfile.H264_MAIN pv_bitrate = 1*1024*1024 pv_format = 'bgr24' # Microphone parameters -microphone_profile = hl2ss.AudioProfile.RAW +microphone_profile = hl2ss.AudioProfile.AAC_24000 # EET parameters eet_fps = 90