You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
192 lines
6.6 KiB
192 lines
6.6 KiB
3 months ago
|
# License: Apache 2.0. See LICENSE file in root directory.
|
||
|
# Copyright(c) 2023 Intel Corporation. All Rights Reserved.
|
||
|
|
||
|
# test:device D400*
|
||
|
|
||
|
import pyrealsense2 as rs
|
||
|
from rspy import test, log
|
||
|
from rspy import tests_wrapper as tw
|
||
|
import os
|
||
|
import math
|
||
|
|
||
|
# Start depth + color streams and go through the frame to make sure it is showing a depth image
|
||
|
# Color stream is only used to display the way the camera is facing
|
||
|
# Verify that the frame does indeed have variance - therefore it is showing a depth image
|
||
|
|
||
|
# In debug mode, an image with the frames found will be displayed and saved
|
||
|
# Make sure you have the required packages installed
|
||
|
DEBUG_MODE = False
|
||
|
|
||
|
# Defines how far in cm do pixels have to be, to be considered in a different distance
|
||
|
# for example, 10 for 10cm, will define the range 100-109 cm as one (as 100)
|
||
|
DETAIL_LEVEL = 10
|
||
|
FRAMES_TO_CHECK = 30
|
||
|
|
||
|
dev = test.find_first_device_or_exit()
|
||
|
tw.start_wrapper( dev )
|
||
|
|
||
|
cfg = rs.config()
|
||
|
cfg.enable_stream(rs.stream.depth, rs.format.z16, 30)
|
||
|
if DEBUG_MODE:
|
||
|
cfg.enable_stream(rs.stream.color, rs.format.bgr8, 30)
|
||
|
|
||
|
|
||
|
def display_image(img):
|
||
|
"""
|
||
|
Display a given image and exits when the x button or esc key are pressed
|
||
|
"""
|
||
|
import cv2
|
||
|
|
||
|
window_title = "Output Stream"
|
||
|
cv2.imshow(window_title, img)
|
||
|
while cv2.getWindowProperty(window_title, cv2.WND_PROP_VISIBLE) > 0:
|
||
|
k = cv2.waitKey(33)
|
||
|
if k == 27: # Esc key to stop
|
||
|
cv2.destroyAllWindows()
|
||
|
break
|
||
|
elif k == -1: # normally -1 returned
|
||
|
pass
|
||
|
|
||
|
|
||
|
def frames_to_image(depth, color, save, display, laser_enabled):
|
||
|
"""
|
||
|
This function gets depth and color frames, transforms them to an image (numpy array)
|
||
|
and then save and/or display
|
||
|
|
||
|
If color frame is given, it will also concatenate it with the depth frame before doing the given action
|
||
|
"""
|
||
|
import numpy as np
|
||
|
import cv2
|
||
|
|
||
|
colorizer = rs.colorizer()
|
||
|
depth_image = np.asanyarray(colorizer.colorize(depth).get_data())
|
||
|
img = depth_image
|
||
|
|
||
|
if color: # if color frame was successfully captured, merge it and the depth frame
|
||
|
from scipy.ndimage import zoom
|
||
|
color_image = np.asanyarray(color.get_data())
|
||
|
depth_rows, _, _ = depth_image.shape
|
||
|
color_rows, _, _ = color_image.shape
|
||
|
# resize the image with the higher resolution to look like the smaller one
|
||
|
if depth_rows < color_rows:
|
||
|
color_image = zoom(color_image, (depth_rows / color_rows, depth_rows / color_rows, 1))
|
||
|
elif color_rows < depth_rows:
|
||
|
depth_image = zoom(depth_image, (color_rows / depth_rows, color_rows / depth_rows, 1))
|
||
|
img = np.concatenate((depth_image, color_image), axis=1)
|
||
|
|
||
|
if save:
|
||
|
file_name = f"output_stream{'_laser_on' if laser_enabled else '_laser_off'}.png"
|
||
|
log.i("Saved image in", os.getcwd() + "\\" + file_name)
|
||
|
cv2.imwrite(file_name, img)
|
||
|
if display:
|
||
|
display_image(img)
|
||
|
|
||
|
|
||
|
def get_frames(config, laser_enabled):
|
||
|
pipeline = rs.pipeline()
|
||
|
pipeline_profile = pipeline.start(config)
|
||
|
|
||
|
sensor = pipeline_profile.get_device().first_depth_sensor()
|
||
|
if laser_enabled and sensor.supports(rs.option.laser_power):
|
||
|
sensor.set_option(rs.option.laser_power, sensor.get_option_range(rs.option.laser_power).max)
|
||
|
sensor.set_option(rs.option.emitter_enabled, 1 if laser_enabled else 0)
|
||
|
|
||
|
# to get a proper image, we sometimes need to wait a few frames, like when the camera is facing a light source
|
||
|
frames = pipeline.wait_for_frames()
|
||
|
for i in range(30):
|
||
|
frames = pipeline.wait_for_frames()
|
||
|
depth = frames.get_depth_frame()
|
||
|
color = frames.get_color_frame()
|
||
|
pipeline.stop()
|
||
|
return depth, color
|
||
|
|
||
|
|
||
|
def round_to_units(num):
|
||
|
"""
|
||
|
Input: Distance of a certain point, in meters
|
||
|
Output: Distance according to the given detail level, in cm
|
||
|
"""
|
||
|
in_cm = round(num, 2) * 100 # convert to cm
|
||
|
return math.floor(in_cm / DETAIL_LEVEL) * DETAIL_LEVEL # rounds the distance according to the given unit
|
||
|
|
||
|
|
||
|
def sort_dict(my_dict):
|
||
|
my_keys = list(my_dict.keys())
|
||
|
my_keys.sort()
|
||
|
sorted_dict = {i: my_dict[i] for i in my_keys}
|
||
|
return sorted_dict
|
||
|
|
||
|
|
||
|
def get_distances(depth):
|
||
|
MAX_METERS = 10 # max distance that can be detected, in meters
|
||
|
dists = {}
|
||
|
total = 0
|
||
|
for y in range(depth.get_height()):
|
||
|
for x in range(depth.get_width()):
|
||
|
dist = depth.get_distance(x, y)
|
||
|
if dist >= MAX_METERS: # out of bounds, assuming it is a junk value
|
||
|
continue
|
||
|
|
||
|
dist = round_to_units(dist) # round according to DETAIL_LEVEL
|
||
|
|
||
|
if dists.get(dist) is not None:
|
||
|
dists[dist] += 1
|
||
|
else:
|
||
|
dists[dist] = 1
|
||
|
total += 1
|
||
|
|
||
|
dists = sort_dict(dists) # for debug convenience
|
||
|
log.d("Distances detected in frame are:", dists)
|
||
|
return dists, total
|
||
|
|
||
|
|
||
|
def is_depth_meaningful(config, laser_enabled=True, save_image=False, show_image=False):
|
||
|
"""
|
||
|
Checks if the camera is showing a frame with a meaningful depth.
|
||
|
DETAIL_LEVEL is setting how close distances need to be, to be considered the same
|
||
|
|
||
|
returns true if frame shows meaningful depth
|
||
|
"""
|
||
|
|
||
|
depth, color = get_frames(config, laser_enabled)
|
||
|
|
||
|
if not depth:
|
||
|
log.f("Error getting depth frame")
|
||
|
return False
|
||
|
if DEBUG_MODE and not color:
|
||
|
log.e("Error getting color frame")
|
||
|
|
||
|
dists, total = get_distances(depth)
|
||
|
|
||
|
# save or display image (only possible through manual debugging)
|
||
|
if save_image or show_image:
|
||
|
frames_to_image(depth, color, save_image, show_image, laser_enabled)
|
||
|
|
||
|
# Goes over the distances found, and checks if any distance is the same on more than 90% of the pixels
|
||
|
meaningful_depth = True
|
||
|
for key in dists:
|
||
|
if dists[key] > total*0.9:
|
||
|
meaningful_depth = False
|
||
|
break
|
||
|
num_blank_pixels = dists[0]
|
||
|
return meaningful_depth, num_blank_pixels
|
||
|
|
||
|
|
||
|
################################################################################################
|
||
|
|
||
|
test.start("Testing depth frame - laser ON -", dev.get_info(rs.camera_info.name))
|
||
|
is_there_depth = False
|
||
|
max_black_pixels = float('inf')
|
||
|
|
||
|
# we perform the check on a few different frames to make sure we get the best indication if we have depth
|
||
|
for frame_num in range(FRAMES_TO_CHECK):
|
||
|
result, laser_black_pixels = is_depth_meaningful(cfg, laser_enabled=True, save_image=DEBUG_MODE, show_image=DEBUG_MODE)
|
||
|
is_there_depth = is_there_depth or result # we check if we found depth at any frame checked
|
||
|
max_black_pixels = min(max_black_pixels, laser_black_pixels)
|
||
|
|
||
|
test.check(is_there_depth is True)
|
||
|
test.finish()
|
||
|
|
||
|
tw.stop_wrapper( dev )
|
||
|
test.print_results_and_exit()
|