Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Speed [WP] Don't merge #20

Open
wants to merge 90 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
90 commits
Select commit Hold shift + click to select a range
fd235cc
Reuse globbed image paths
whoactuallycares Jul 24, 2024
83be7d5
Remove trace logging
whoactuallycares Jul 25, 2024
e2ba90f
Remove self.all_features, self.all_erros, self.charucos
whoactuallycares Jul 25, 2024
5be08c7
Remove self.collected_ids, self.collected_features, mark constant mem…
whoactuallycares Jul 25, 2024
c5d1869
Invert disabled camera condition
whoactuallycares Jul 25, 2024
5025db1
Remove self.width, self.height, merge calibration and image loading l…
whoactuallycares Jul 25, 2024
8cc9481
Remove self.calib_model
whoactuallycares Jul 25, 2024
5d2e0ab
Remove self.distortion_model
whoactuallycares Jul 25, 2024
67003e0
Remove self.cameraIntrinsics, self.cameraDistortion
whoactuallycares Jul 25, 2024
466b3c7
Remove redundant .keys()
whoactuallycares Jul 25, 2024
07b8b40
Move alibration into a temporary function in another scope
whoactuallycares Jul 25, 2024
1182a25
Split into smaller functions
whoactuallycares Jul 25, 2024
e57055f
Speedup initial getting and filtering of corners
whoactuallycares Jul 25, 2024
dc86578
Preparation for more multiprocessing, proxy class for possing aruco t…
whoactuallycares Jul 25, 2024
94b4166
Run individual ccm calibrations in multiple threads
whoactuallycares Jul 25, 2024
eb86106
Invert exit condition
whoactuallycares Jul 25, 2024
0fc2cb8
Invert move exit conditions
whoactuallycares Jul 25, 2024
c908edd
Move stereo calibration into it's own function
whoactuallycares Jul 25, 2024
8334368
Move intrinsic calibration back into class
whoactuallycares Jul 25, 2024
f4f51c4
Remove old unused functions
whoactuallycares Jul 27, 2024
82aeb04
Split calibration into data loading and calibration
whoactuallycares Jul 27, 2024
2459632
Change stereo calibration to calibrate one pair at a time
whoactuallycares Jul 27, 2024
a699a0a
Pass more variables via cam_info
whoactuallycares Jul 27, 2024
9c6496b
Move right cam disable check out of calibrate_stereo_pair
whoactuallycares Jul 27, 2024
1a373fc
Move main calibration functions out of class members
whoactuallycares Jul 27, 2024
e7e9478
Split calibrate_ccm_intrinsics into normal and per_ccm version
whoactuallycares Jul 27, 2024
3ece2f0
Move filtering features out of calibrate ccm function
whoactuallycares Jul 27, 2024
a8bc6b3
Split getting and filtering features
whoactuallycares Jul 29, 2024
b85039c
Split filtering into fisheye and non-fisheye
whoactuallycares Jul 29, 2024
cdb55bc
Split calibration into filterint and calibration
whoactuallycares Jul 29, 2024
e01601b
Simplify estimate_pose_and_filter
whoactuallycares Jul 29, 2024
9d9517f
Simplify arguments
whoactuallycares Jul 29, 2024
9fbe78f
Add parallelization back
whoactuallycares Jul 29, 2024
065e027
Type hints
whoactuallycares Jul 29, 2024
aac8d97
Fix parallelization
whoactuallycares Jul 31, 2024
b92640d
Cleanup imports
whoactuallycares Aug 2, 2024
e79962f
Split stereo calibration into multiple functions
whoactuallycares Aug 2, 2024
deec5d3
Move stereo_calibrate contents up
whoactuallycares Aug 2, 2024
5c6a264
Move stereo calibration into main function
whoactuallycares Aug 2, 2024
1f1ee94
Simplify arguments
whoactuallycares Aug 2, 2024
ddad1a4
Parallelize stereo calibration
whoactuallycares Aug 2, 2024
1daafc8
Return all data into board config
whoactuallycares Aug 5, 2024
533a97b
Remove rectification display
whoactuallycares Aug 6, 2024
4c58ec1
Remove unused functions
whoactuallycares Aug 6, 2024
c3706c3
Remove trailing spaces, tabsize to 2
whoactuallycares Aug 6, 2024
d24cb8f
Use enumerate()
whoactuallycares Aug 6, 2024
c79676e
Move methods out of class
whoactuallycares Aug 6, 2024
c0945b5
Add config structure
whoactuallycares Aug 6, 2024
4f593ff
Cleanup arguments
whoactuallycares Aug 6, 2024
9aa475b
Cleaner return values from ParallelTask
whoactuallycares Aug 6, 2024
b04704a
Slightly better typehints
whoactuallycares Aug 6, 2024
61fba81
Annotate camData
whoactuallycares Aug 6, 2024
5e1e4fb
Remove charuco_ids_to_objpoints
whoactuallycares Aug 6, 2024
1df473d
Refactor find_stereo_common_features
whoactuallycares Aug 6, 2024
bc24df0
Add __repr__, don't unpack dict, str and bytes in TaskGroup arguments
whoactuallycares Aug 14, 2024
a2c5fef
Thermal WIP
whoactuallycares Aug 15, 2024
790731b
Count TaskGroup as not finished if no tasks were created
whoactuallycares Aug 15, 2024
9e8fc73
Calibration with multiple charuco boards
whoactuallycares Aug 15, 2024
eec08f2
Comments and enums
whoactuallycares Aug 26, 2024
9a027e1
Remove unused features selection
whoactuallycares Aug 26, 2024
896338d
Fix conversion bug
whoactuallycares Aug 26, 2024
dc58edb
Remove explicit thermal switch cases, remove duplicate 'size' value, …
whoactuallycares Aug 26, 2024
a7eeca1
Adjust filterint so thermal cameras pass
whoactuallycares Aug 30, 2024
9fe41f9
Stereo fix for OAK-D-LITE in speed branch, needs testing
MaticTonin Sep 12, 2024
82b8b61
Merge pull request #22 from luxonis/speed_fixed_stereo_filtering
MaticTonin Sep 27, 2024
ad88194
Revert "Stereo fix for OAK-D-LITE in speed branch, needs testing"
MaticTonin Sep 27, 2024
41e745c
Merge pull request #24 from luxonis/revert-22-speed_fixed_stereo_filt…
MaticTonin Sep 27, 2024
d7bb717
Move types to separate file
whoactuallycares Nov 11, 2024
fe9e757
Merge commit 'd7bb7172' into speed
whoactuallycares Nov 11, 2024
890d037
Fix kwargs bug, remove unused functions, add debugging file
whoactuallycares Nov 11, 2024
6c9409c
Add debugging file
whoactuallycares Nov 11, 2024
bd886dc
Remove unused variables
whoactuallycares Nov 11, 2024
53d7bee
Remove unused variables
whoactuallycares Nov 11, 2024
41b3828
Typo
whoactuallycares Nov 11, 2024
b667263
Remove unused variable
whoactuallycares Nov 11, 2024
faf8fec
Fix segfault bug
whoactuallycares Nov 12, 2024
2231d56
Fix stereo calib not being produced
whoactuallycares Nov 14, 2024
cec3b6e
Add debug output parameter
whoactuallycares Nov 14, 2024
c79f3de
Stereo filter features
whoactuallycares Nov 15, 2024
792e6ee
Charuco board detection, reuse filtered images
whoactuallycares Nov 25, 2024
551f6a1
Fix bug in image loader
whoactuallycares Nov 25, 2024
dc16bd7
Add rectification display in debugging, enable multithreading
whoactuallycares Nov 28, 2024
e0f5f50
Fix incorrect corner passing when multiple calibration boards are used
whoactuallycares Dec 2, 2024
1b38689
Pass errors
whoactuallycares Dec 2, 2024
3f510d5
Return filtered charucos
whoactuallycares Dec 16, 2024
b06527a
Fix filtered corner saving
whoactuallycares Dec 17, 2024
fab3b5b
Automatically generate parallel worker calls by modifying the functio…
whoactuallycares Dec 20, 2024
280ec35
Formatting
whoactuallycares Dec 20, 2024
e9d24dc
Add comments
whoactuallycares Dec 20, 2024
b631ee0
Remove retval opening
whoactuallycares Dec 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
336 changes: 336 additions & 0 deletions calibration_debug.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,336 @@
import logging
logging.getLogger('matplotlib').setLevel(logging.WARNING)

import matplotlib.pyplot as plt
from collections import deque
from pathlib import Path
from .types import *
import numpy as np
import glob
import cv2

# Helper functions

def scale_intrinsics(intrinsics, originalShape, destShape):
scale = destShape[1] / originalShape[1] # scale on width
scale_mat = np.array([[scale, 0, 0], [0, scale, 0], [0, 0, 1]])
scaled_intrinsics = np.matmul(scale_mat, intrinsics)
""" print("Scaled height offset : {}".format(
(originalShape[0] * scale - destShape[0]) / 2))
print("Scaled width offset : {}".format(
(originalShape[1] * scale - destShape[1]) / 2)) """
scaled_intrinsics[1][2] -= (originalShape[0] # c_y - along height of the image
* scale - destShape[0]) / 2
scaled_intrinsics[0][2] -= (originalShape[1] # c_x width of the image
* scale - destShape[1]) / 2
#if self.traceLevel == 3 or self.traceLevel == 10:
print('original_intrinsics')
print(intrinsics)
print('scaled_intrinsics')
print(scaled_intrinsics)

return scaled_intrinsics

def scale_image(img, scaled_res):
expected_height = img.shape[0]*(scaled_res[1]/img.shape[1])
#if self.traceLevel == 2 or self.traceLevel == 10:
print("Expected Height: {}".format(expected_height))

if not (img.shape[0] == scaled_res[0] and img.shape[1] == scaled_res[1]):
if int(expected_height) == scaled_res[0]:
# resizing to have both stereo and rgb to have same
# resolution to capture extrinsics of the rgb-right camera
img = cv2.resize(img, (scaled_res[1], scaled_res[0]),
interpolation=cv2.INTER_CUBIC)
return img
else:
# resizing and cropping to have both stereo and rgb to have same resolution
# to calculate extrinsics of the rgb-right camera
scale_width = scaled_res[1]/img.shape[1]
dest_res = (
int(img.shape[1] * scale_width), int(img.shape[0] * scale_width))
img = cv2.resize(
img, dest_res, interpolation=cv2.INTER_CUBIC)
if img.shape[0] < scaled_res[0]:
raise RuntimeError("resizeed height of rgb is smaller than required. {0} < {1}".format(
img.shape[0], scaled_res[0]))
# print(gray.shape[0] - req_resolution[0])
del_height = (img.shape[0] - scaled_res[0]) // 2
# gray = gray[: req_resolution[0], :]
img = img[del_height: del_height + scaled_res[0], :]
return img
else:
return img

def detect_charuco_board(config: CharucoBoard, image: np.array):
arucoParams = cv2.aruco.DetectorParameters_create()
arucoParams.minMarkerDistanceRate = 0.01
corners, ids, rejectedImgPoints = cv2.aruco.detectMarkers(image, config.dictionary, parameters=arucoParams) # First, detect markers
marker_corners, marker_ids, refusd, recoverd = cv2.aruco.refineDetectedMarkers(image, config.board, corners, ids, rejectedCorners=rejectedImgPoints)
# If found, add object points, image points (after refining them)
if len(marker_corners) > 0:
ret, corners, ids = cv2.aruco.interpolateCornersCharuco(marker_corners,marker_ids,image, config.board, minMarkers = 1)
return ret, corners, ids, marker_corners, marker_ids
else:
return None, None, None, None, None

def display_rectification(image_data_pairs, images_corners_l, images_corners_r, image_epipolar_color, isHorizontal):
print("Displaying Stereo Pair for visual inspection. Press the [ESC] key to exit.")
colors = [(0, 255 , 0), (0, 0, 255)]
for idx, image_data_pair in enumerate(image_data_pairs):
if isHorizontal:
img_concat = cv2.hconcat(
[image_data_pair[0], image_data_pair[1]])
for left_pt, right_pt, colorMode in zip(images_corners_l[idx], images_corners_r[idx], image_epipolar_color[idx]):
cv2.line(img_concat,
(int(left_pt[0][0]), int(left_pt[0][1])), (int(right_pt[0][0]) + image_data_pair[0].shape[1], int(right_pt[0][1])),
colors[colorMode], 1)
else:
img_concat = cv2.vconcat(
[image_data_pair[0], image_data_pair[1]])
for left_pt, right_pt, colorMode in zip(images_corners_l[idx], images_corners_r[idx], image_epipolar_color[idx]):
cv2.line(img_concat,
(int(left_pt[0][0]), int(left_pt[0][1])), (int(right_pt[0][0]), int(right_pt[0][1]) + image_data_pair[0].shape[0]),
colors[colorMode], 1)

img_concat = cv2.resize(img_concat, (0, 0), fx=0.8, fy=0.8)

# show image
cv2.imshow('Stereo Pair', img_concat)
while True:
k = cv2.waitKey(1)
if k == 27: # Esc key to stop
break

cv2.destroyWindow('Stereo Pair')

# Debugging functions

def debug_epipolar_charuco(left_cam_info: CameraData, right_cam_info: CameraData, left_board: CharucoBoard, right_board: CharucoBoard, t, r_l, r_r):
images_left = glob.glob(left_cam_info['images_path'] + '/*.png')
images_right = glob.glob(right_cam_info['images_path'] + '/*.png')
images_left.sort()
print(images_left)
images_right.sort()
assert len(images_left) != 0, "ERROR: Images not read correctly"
assert len(images_right) != 0, "ERROR: Images not read correctly"
isHorizontal = np.absolute(t[0]) > np.absolute(t[1])

scale = None
scale_req = False
frame_left_shape = cv2.imread(images_left[0], 0).shape
frame_right_shape = cv2.imread(images_right[0], 0).shape
scalable_res = frame_left_shape
scaled_res = frame_right_shape
if frame_right_shape[0] < frame_left_shape[0] and frame_right_shape[1] < frame_left_shape[1]:
scale_req = True
scale = frame_right_shape[1] / frame_left_shape[1]
elif frame_right_shape[0] > frame_left_shape[0] and frame_right_shape[1] > frame_left_shape[1]:
scale_req = True
scale = frame_left_shape[1] / frame_right_shape[1]
scalable_res = frame_right_shape
scaled_res = frame_left_shape

if scale_req:
scaled_height = scale * scalable_res[0]
diff = scaled_height - scaled_res[0]
if diff < 0:
scaled_res = (int(scaled_height), scaled_res[1])
#if self.traceLevel == 3 or self.traceLevel == 10:
print(
f'Is scale Req: {scale_req}\n scale value: {scale} \n scalable Res: {scalable_res} \n scale Res: {scaled_res}')
print("Original res Left :{}".format(frame_left_shape))
print("Original res Right :{}".format(frame_right_shape))
# print("Scale res :{}".format(scaled_res))

M_l = left_cam_info['intrinsics']
M_r = right_cam_info['intrinsics']
M_lp = scale_intrinsics(M_l, frame_left_shape, scaled_res)
M_rp = scale_intrinsics(M_r, frame_right_shape, scaled_res)

criteria = (cv2.TERM_CRITERIA_EPS +
cv2.TERM_CRITERIA_MAX_ITER, 10000, 0.00001)

# TODO(Sachin): Observe Images by adding visualization
# TODO(Sachin): Check if the stetch is only in calibration Images
print('Original intrinsics ....')
print(f"L {M_lp}")
print(f"R: {M_rp}")
#if self.traceLevel == 3 or self.traceLevel == 10:
print(f'Width and height is {scaled_res[::-1]}')
# kScaledL, _ = cv2.getOptimalNewCameraMatrix(M_r, d_r, scaled_res[::-1], 0)
# kScaledL, _ = cv2.getOptimalNewCameraMatrix(M_r, d_l, scaled_res[::-1], 0)
# kScaledR, _ = cv2.getOptimalNewCameraMatrix(M_r, d_r, scaled_res[::-1], 0)
kScaledR = kScaledL = M_rp

# if self.cameraModel != 'perspective':
# kScaledR = cv2.fisheye.estimateNewCameraMatrixForUndistortRectify(M_r, d_r, scaled_res[::-1], np.eye(3), fov_scale=1.1)
# kScaledL = kScaledR


print('Intrinsics from the getOptimalNewCameraMatrix/Original ....')
print(f"L: {kScaledL}")
print(f"R: {kScaledR}")
oldEpipolarError = None
epQueue = deque()
movePos = True

left_calib_model = left_cam_info['calib_model']
right_calib_model = right_cam_info['calib_model']
d_l = left_cam_info['dist_coeff']
d_r = left_cam_info['dist_coeff']
print(left_calib_model, right_calib_model)
if left_calib_model == 'perspective':
mapx_l, mapy_l = cv2.initUndistortRectifyMap(
M_lp, d_l, r_l, kScaledL, scaled_res[::-1], cv2.CV_32FC1)
else:
mapx_l, mapy_l = cv2.fisheye.initUndistortRectifyMap(
M_lp, d_l, r_l, kScaledL, scaled_res[::-1], cv2.CV_32FC1)
if right_calib_model == "perspective":
mapx_r, mapy_r = cv2.initUndistortRectifyMap(
M_rp, d_r, r_r, kScaledR, scaled_res[::-1], cv2.CV_32FC1)
else:
mapx_r, mapy_r = cv2.fisheye.initUndistortRectifyMap(
M_rp, d_r, r_r, kScaledR, scaled_res[::-1], cv2.CV_32FC1)

image_data_pairs = []
for image_left, image_right in zip(images_left, images_right):
# read images
img_l = cv2.imread(image_left, 0)
img_r = cv2.imread(image_right, 0)

img_l = scale_image(img_l, scaled_res)
img_r = scale_image(img_r, scaled_res)
# print(img_l.shape)
# print(img_r.shape)

# warp right image
# img_l = cv2.warpPerspective(img_l, self.H1, img_l.shape[::-1],
# cv2.INTER_CUBIC +
# cv2.WARP_FILL_OUTLIERS +
# cv2.WARP_INVERSE_MAP)

# img_r = cv2.warpPerspective(img_r, self.H2, img_r.shape[::-1],
# cv2.INTER_CUBIC +
# cv2.WARP_FILL_OUTLIERS +
# cv2.WARP_INVERSE_MAP)

img_l = cv2.remap(img_l, mapx_l, mapy_l, cv2.INTER_LINEAR)
img_r = cv2.remap(img_r, mapx_r, mapy_r, cv2.INTER_LINEAR)

image_data_pairs.append((img_l, img_r))

#if self.traceLevel == 4 or self.traceLevel == 5 or self.traceLevel == 10:
#cv2.imshow("undistorted-Left", img_l)
#cv2.imshow("undistorted-right", img_r)
# print(f'image path - {im}')
# print(f'Image Undistorted Size {undistorted_img.shape}')
# k = cv2.waitKey(0)
# if k == 27: # Esc key to stop
# break
#if self.traceLevel == 4 or self.traceLevel == 5 or self.traceLevel == 10:
# cv2.destroyWindow("undistorted-Left")
# cv2.destroyWindow("undistorted-right")
# compute metrics
imgpoints_r = []
imgpoints_l = []
image_epipolar_color = []
# new_imagePairs = [])
for i, image_data_pair in enumerate(image_data_pairs):
res2_l = detect_charuco_board(left_board, image_data_pair[0])
res2_r = detect_charuco_board(right_board, image_data_pair[1])

# if self.traceLevel == 2 or self.traceLevel == 4 or self.traceLevel == 10:


img_concat = cv2.hconcat([image_data_pair[0], image_data_pair[1]])
img_concat = cv2.cvtColor(img_concat, cv2.COLOR_GRAY2RGB)
line_row = 0
while line_row < img_concat.shape[0]:
cv2.line(img_concat,
(0, line_row), (img_concat.shape[1], line_row),
(0, 255, 0), 1)
line_row += 30

cv2.imshow('Stereo Pair', img_concat)
while True:
k = cv2.waitKey(1)
if k == 27: # Esc key to stop
break

if res2_l[1] is not None and res2_r[2] is not None and len(res2_l[1]) > 3 and len(res2_r[1]) > 3:

cv2.cornerSubPix(image_data_pair[0], res2_l[1],
winSize=(5, 5),
zeroZone=(-1, -1),
criteria=criteria)
cv2.cornerSubPix(image_data_pair[1], res2_r[1],
winSize=(5, 5),
zeroZone=(-1, -1),
criteria=criteria)

# termination criteria
img_pth_right = Path(images_right[i])
img_pth_left = Path(images_left[i])
org = (100, 50)
# cv2.imshow('ltext', lText)
# cv2.waitKey(0)
localError = 0
corners_l = []
corners_r = []
for j in range(len(res2_l[2])):
idx = np.where(res2_r[2] == res2_l[2][j])
if idx[0].size == 0:
continue
corners_l.append(res2_l[1][j])
corners_r.append(res2_r[1][idx])
#if len(corners_l) == 0 or len(corners_r) == 0:
#continue

imgpoints_l.append(corners_l)
imgpoints_r.append(corners_r)
epi_error_sum = 0
corner_epipolar_color = []
for l_pt, r_pt in zip(corners_l, corners_r):
if isHorizontal:
curr_epipolar_error = abs(l_pt[0][1] - r_pt[0][1])
else:
curr_epipolar_error = abs(l_pt[0][0] - r_pt[0][0])
if curr_epipolar_error >= 1:
corner_epipolar_color.append(1)
else:
corner_epipolar_color.append(0)
epi_error_sum += curr_epipolar_error
localError = epi_error_sum / len(corners_l)
image_epipolar_color.append(corner_epipolar_color)
#if self.traceLevel == 2 or self.traceLevel == 3 or self.traceLevel == 4 or self.traceLevel == 10:
print("Epipolar Error per image on host in " + img_pth_right.name + " : " +
str(localError))
else:
print('Numer of corners is in left -> and right ->')
imgpoints_l.append([])
imgpoints_r.append([])
continue
return -1
lText = cv2.putText(cv2.cvtColor(image_data_pair[0],cv2.COLOR_GRAY2RGB), img_pth_left.name, org, cv2.FONT_HERSHEY_SIMPLEX, 1, (2, 0, 255), 2, cv2.LINE_AA)
rText = cv2.putText(cv2.cvtColor(image_data_pair[1],cv2.COLOR_GRAY2RGB), img_pth_right.name + " Error: " + str(localError), org, cv2.FONT_HERSHEY_SIMPLEX, 1, (2, 0, 255), 2, cv2.LINE_AA)
image_data_pairs[i] = (lText, rText)


epi_error_sum = 0
total_corners = 0
for corners_l, corners_r in zip(imgpoints_l, imgpoints_r):
total_corners += len(corners_l)
for l_pt, r_pt in zip(corners_l, corners_r):
if isHorizontal:
epi_error_sum += abs(l_pt[0][1] - r_pt[0][1])
else:
epi_error_sum += abs(l_pt[0][0] - r_pt[0][0])

avg_epipolar = epi_error_sum / total_corners
print("Average Epipolar Error is : " + str(avg_epipolar))

if True or enable_rectification_disp:
display_rectification(image_data_pairs, imgpoints_l, imgpoints_r, image_epipolar_color, isHorizontal)

return avg_epipolar
Loading