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

Add the ZED Wrapper and Object Detector #3

Merged
merged 96 commits into from
Sep 19, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
96 commits
Select commit Hold shift + click to select a range
7782fff
starter files
jbrhm Sep 2, 2024
c244165
starter code
jbrhm Sep 2, 2024
ad2d273
finding zed sdk
jbrhm Sep 2, 2024
3d175e8
building zed sdk
jbrhm Sep 2, 2024
b2b97f7
y no build
jbrhm Sep 2, 2024
f197dcd
gets paramaters
jbrhm Sep 3, 2024
418b537
gets more aprameters
jbrhm Sep 3, 2024
267e919
why no callback
jbrhm Sep 3, 2024
7df671c
variants :)))))
jbrhm Sep 3, 2024
d861946
logic error
jbrhm Sep 3, 2024
288f67e
more params
jbrhm Sep 3, 2024
162ccbc
most init done
jbrhm Sep 3, 2024
36cd821
no build :(
jbrhm Sep 3, 2024
12e0638
almost builds
jbrhm Sep 3, 2024
560ec18
builds :)))))
jbrhm Sep 3, 2024
0a54992
wat svo?
jbrhm Sep 3, 2024
89147ac
builds
jbrhm Sep 5, 2024
0dbf0bb
AHHHHHHHH POINTERS AHHHHHHHHH
jbrhm Sep 5, 2024
e884847
STD MOVE FTW
jbrhm Sep 5, 2024
0d6a84e
undefined symbols :(
jbrhm Sep 5, 2024
ec4765e
undefined symbols :(
jbrhm Sep 5, 2024
a907977
Builds
jbrhm Sep 5, 2024
7209434
Runs
jbrhm Sep 5, 2024
5507127
Better includes
jbrhm Sep 5, 2024
a2bd36c
Works
jbrhm Sep 5, 2024
2499c37
62 Hz
jbrhm Sep 8, 2024
ffc00d7
begin porting voer obj detect
jbrhm Sep 8, 2024
1a7eba6
more errors
jbrhm Sep 8, 2024
e848112
learning lib builds
jbrhm Sep 8, 2024
c553f19
brah
jbrhm Sep 8, 2024
904ae74
bruhhhh
jbrhm Sep 12, 2024
295ae21
works
jbrhm Sep 12, 2024
051f860
better type safety
jbrhm Sep 13, 2024
4efccaa
got rid of compiler warnings
jbrhm Sep 14, 2024
2e6e95a
merged in master
jbrhm Sep 14, 2024
d23eb77
starter files
jbrhm Sep 2, 2024
b5cb2b5
starter code
jbrhm Sep 2, 2024
9ad086e
finding zed sdk
jbrhm Sep 2, 2024
73a12ce
building zed sdk
jbrhm Sep 2, 2024
b7eac39
y no build
jbrhm Sep 2, 2024
385bf85
gets paramaters
jbrhm Sep 3, 2024
9ca6322
gets more aprameters
jbrhm Sep 3, 2024
1bec7f5
why no callback
jbrhm Sep 3, 2024
f16bfa6
variants :)))))
jbrhm Sep 3, 2024
bf9a8b6
logic error
jbrhm Sep 3, 2024
fa31363
more params
jbrhm Sep 3, 2024
c8ed85d
most init done
jbrhm Sep 3, 2024
523c6e3
no build :(
jbrhm Sep 3, 2024
fab8db8
almost builds
jbrhm Sep 3, 2024
a2a9713
builds :)))))
jbrhm Sep 3, 2024
0e373f2
wat svo?
jbrhm Sep 3, 2024
35342c1
builds
jbrhm Sep 5, 2024
fbd4f32
AHHHHHHHH POINTERS AHHHHHHHHH
jbrhm Sep 5, 2024
4767e6d
STD MOVE FTW
jbrhm Sep 5, 2024
a43cec3
undefined symbols :(
jbrhm Sep 5, 2024
30003d6
undefined symbols :(
jbrhm Sep 5, 2024
ce4ef89
Builds
jbrhm Sep 5, 2024
c76de69
Runs
jbrhm Sep 5, 2024
86e74c8
Better includes
jbrhm Sep 5, 2024
9770d38
Works
jbrhm Sep 5, 2024
b2a97d7
62 Hz
jbrhm Sep 8, 2024
09704e9
begin porting voer obj detect
jbrhm Sep 8, 2024
f64d949
more errors
jbrhm Sep 8, 2024
8b19c94
learning lib builds
jbrhm Sep 8, 2024
3d1824d
brah
jbrhm Sep 8, 2024
44c679e
bruhhhh
jbrhm Sep 12, 2024
fc35e3a
works
jbrhm Sep 12, 2024
a7f55e2
better type safety
jbrhm Sep 13, 2024
e14237b
got rid of compiler warnings
jbrhm Sep 14, 2024
5728d67
Update zed_wrapper.bridge.cpp
jbrhm Sep 14, 2024
e438c2a
Updated CUDA
jbrhm Sep 15, 2024
14745ae
Updated CUDA
jbrhm Sep 15, 2024
eb7471b
Updated CUDA
jbrhm Sep 15, 2024
4e78274
Update zed_wrapper.bridge.cpp
jbrhm Sep 15, 2024
ba98358
CLANG FORMAT IS CRACCCCCCKKKKKKED bro :))))))))))))
jbrhm Sep 15, 2024
7cb4218
CLANG FORMAT IS CRACCCCCCKKKKKKED bro :))))))))))))
jbrhm Sep 15, 2024
f5d09a3
CLANG FORMAT IS CRACCCCCCKKKKKKED bro :))))))))))))
jbrhm Sep 15, 2024
6241883
CLANG FORMAT IS CRACCCCCCKKKKKKED bro :))))))))))))
jbrhm Sep 15, 2024
4a37c44
BLACK
jbrhm Sep 15, 2024
5fa616d
builds
jbrhm Sep 15, 2024
06c768f
no engine
jbrhm Sep 15, 2024
c3c555d
redundant
jbrhm Sep 15, 2024
4026fc9
Merge branch 'main' into JRA/ZED-Wrapper
jbrhm Sep 15, 2024
8ef1e63
Ashwin Suggestions
jbrhm Sep 16, 2024
beb2e9d
No more engine
jbrhm Sep 17, 2024
bfc4c79
Merge branch 'main' into JRA/ZED-Wrapper
jbrhm Sep 17, 2024
60be043
Updating compiler flags
jbrhm Sep 18, 2024
755f093
Final Refactoring
jbrhm Sep 18, 2024
f74ed16
od still works
jbrhm Sep 19, 2024
85eeecc
zed default params
jbrhm Sep 19, 2024
578f3bd
od default params
jbrhm Sep 19, 2024
6feb1d4
Works with new parameters
jbrhm Sep 19, 2024
2f3bb3d
Better pathing on zed launch
jbrhm Sep 19, 2024
fe6c546
Better launch file configuration
jbrhm Sep 19, 2024
a9ef7c3
Merge branch 'main' into JRA/ZED-Wrapper
jbrhm Sep 19, 2024
a990c0a
works
jbrhm Sep 19, 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
43 changes: 42 additions & 1 deletion CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ project(mrover VERSION 2025.0.0 LANGUAGES C CXX)
cmake_policy(SET CMP0148 OLD)

set(CMAKE_CXX_STANDARD 23)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

# Generate compile_commands.json for clangd language server.
# Can be used by VSCode, CLion, VIM, etc.
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
Expand Down Expand Up @@ -210,6 +210,7 @@ ament_target_dependencies(lie rclcpp geometry_msgs tf2 tf2_ros)

mrover_add_header_only_library(units units)
mrover_add_header_only_library(loop_profiler loop_profiler)
mrover_add_header_only_library(parameter_utils parameter_utils)

# Simulator

Expand All @@ -226,6 +227,46 @@ target_include_directories(simulator SYSTEM PRIVATE ${BULLET_INCLUDE_DIRS} ${OPE

# Perception

find_package(ZED QUIET)
find_package(CUDA QUIET)
find_package(OpenCV REQUIRED)
if(ZED_FOUND AND CUDA_FOUND)
enable_language(CUDA)
# CUDA Compile Options
add_library(cuda_compiler_flags INTERFACE)
target_compile_options(cuda_compiler_flags INTERFACE
-Wno-pedantic
-Wno-deprecated
-Wno-unused-parameter
-diag-suppress=815
-diag-suppress=780
-Wno-deprecated-copy
-Wno-unused-command-line-argument
-Wno-ignored-qualifiers
-Wno-sometimes-uninitialized
)

# ZED Wrapper
mrover_add_node(zed perception/zed_wrapper/*.c* perception/zed_wrapper/pch.hpp)
target_compile_options(zed PRIVATE $<$<COMPILE_LANGUAGE:CXX>:-std=c++20>)
target_link_libraries(zed parameter_utils lie MANIF::manif ${CUDA_LIBRARIES} loop_profiler cuda_compiler_flags)
ament_target_dependencies(zed rclcpp sensor_msgs ZED CUDA tf2 tf2_ros)

# Learning Library
# TODO(john): Update to use the new API
mrover_add_library(tensorrt tensorrt/*.c* tensorrt)
target_compile_options(tensorrt PRIVATE -Wno-deprecated-declarations $<$<COMPILE_LANGUAGE:CUDA>:-std=c++17>)
target_include_directories(tensorrt PRIVATE ${CMAKE_CUDA_TOOLKIT_INCLUDE_DIRECTORIES})
target_link_libraries(tensorrt PRIVATE opencv_core opencv_dnn opencv_imgproc lie nvinfer nvonnxparser tbb cuda_compiler_flags)

# Object Detector
mrover_add_node(object_detector perception/object_detector/*.c* src/perception/object_detector/pch.hpp)
target_link_libraries(object_detector opencv_core opencv_dnn opencv_imgproc lie tbb tensorrt opencv_imgcodecs opencv_highgui loop_profiler parameter_utils cuda_compiler_flags)
ament_target_dependencies(object_detector rclcpp sensor_msgs CUDA tf2 tf2_ros)
else()
message("ZED not found...")
endif()

mrover_add_node(tag_detector perception/tag_detector/*.cpp perception/tag_detector/pch.hpp)
ament_target_dependencies(tag_detector rclcpp tf2 tf2_ros)
target_link_libraries(tag_detector lie opencv_core opencv_aruco opencv_imgproc loop_profiler)
Expand Down
6 changes: 5 additions & 1 deletion build.sh
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,14 @@ set -euxo pipefail
# Build in the colcon workspace, not the package
pushd ../..

# set C/CXX compilers
# Set C/C++ compilers
export CC=clang
export CXX=clang++

# Set CUDA compilers
export CUDAHOSTCXX=g++-9
export CUDACXX=/usr/local/cuda-12.3/bin/nvcc

# TODO (ali): add build configs for debug vs release
colcon build \
--cmake-args -G Ninja -W no-dev -DCMAKE_BUILD_TYPE=RelWithDebInfo \
Expand Down
18 changes: 18 additions & 0 deletions config/object_detector.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@

# All units are in SI
# ===================
# Time: second, hz
# Angle: radian
# Distance: meter

/object_detector:
ros__parameters:
camera_frame: "zed_left_camera_frame"
world_frame: "map"
increment_weight: 2
decrement_weight: 1
hitcount_threshold: 5
hitcount_max: 10
model_name: "Large-Dataset"
model_score_threshold: 0.75
model_nms_threshold: 0.5
22 changes: 22 additions & 0 deletions config/zed.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
# All units are in SI
# ===================
# Time: second, hz
# Angle: radian
# Distance: meter

/zed_wrapper:
ros__parameters:
depth_confidence: 70
serial_number: -1
grab_target_fps: 60
texture_confidence: 100
image_width: 1280
image_height: 720
svo_file: ""
use_depth_stabilization: false
grab_resolution: "HD720"
depth_mode: "PERFORMANCE"
depth_maximum_distance: 12.0
use_builtin_visual_odom: false
use_pose_smoothing: true
use_area_memory: true
1 change: 1 addition & 0 deletions data/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
*.engine
3 changes: 3 additions & 0 deletions data/Large-Dataset.onnx
Git LFS file not shown
28 changes: 28 additions & 0 deletions launch/perception.launch.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
from pathlib import Path

from ament_index_python import get_package_share_directory

from launch import LaunchDescription
from launch.actions import DeclareLaunchArgument
from launch.substitutions import LaunchConfiguration
from launch_ros.actions import Node, ComposableNodeContainer
from launch_ros.descriptions import ComposableNode
from launch.conditions import LaunchConfigurationEquals


def generate_launch_description():
zed_node = Node(
package="mrover",
executable="zed",
name="zed_wrapper",
parameters=[Path(get_package_share_directory("mrover"), "config", "zed.yaml")],
)

object_detector_node = Node(
package="mrover",
executable="object_detector",
name="object_detector",
parameters=[Path(get_package_share_directory("mrover"), "config", "object_detector.yaml")],
)

return LaunchDescription([zed_node, object_detector_node])
104 changes: 104 additions & 0 deletions parameter_utils/parameter.hpp
jbrhm marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
#include <vector>
#include <string>
#include <variant>
#include <exception>
#include <rclcpp/rclcpp.hpp>

// Overloads pattern for visit
template<class... Ts> struct overload : Ts... { using Ts::operator()...; };
jbrhm marked this conversation as resolved.
Show resolved Hide resolved
template<class... Ts> overload(Ts...) -> overload<Ts...>;


namespace mrover {
class ParameterWrapper {
private:
static inline std::shared_ptr<rclcpp::ParameterCallbackHandle> cbHande;

public:
rclcpp::ParameterType mType;

std::string mParamDescriptor;

std::variant<int*, std::string*, bool*, double*, float*> mData;

std::variant<int, std::string, bool, double, float> mDefaultValue;

ParameterWrapper(std::string paramDescriptor, int& variable, int defaultValue = 0) : mType{rclcpp::ParameterType::PARAMETER_INTEGER}, mParamDescriptor{std::move(paramDescriptor)}, mData{&variable}, mDefaultValue{defaultValue}{}

ParameterWrapper(std::string paramDescriptor, std::string& variable, std::string defaultValue = "") : mType{rclcpp::ParameterType::PARAMETER_STRING}, mParamDescriptor{std::move(paramDescriptor)}, mData{&variable}, mDefaultValue{std::move(defaultValue)}{}

ParameterWrapper(std::string paramDescriptor, bool& variable, bool defaultValue = false) : mType{rclcpp::ParameterType::PARAMETER_BOOL}, mParamDescriptor{std::move(paramDescriptor)}, mData{&variable}, mDefaultValue{defaultValue}{}

ParameterWrapper(std::string paramDescriptor, double& variable, double defaultValue = 0.0) : mType{rclcpp::ParameterType::PARAMETER_DOUBLE}, mParamDescriptor{std::move(paramDescriptor)}, mData{&variable}, mDefaultValue{defaultValue}{}

ParameterWrapper(std::string paramDescriptor, float& variable, float defaultValue = 0.0) : mType{rclcpp::ParameterType::PARAMETER_DOUBLE}, mParamDescriptor{std::move(paramDescriptor)}, mData{&variable}, mDefaultValue{defaultValue}{}

void visit(rclcpp::Node* node){
std::visit(overload{
[&](int* arg){
try{
*arg = static_cast<int>(node->get_parameter(mParamDescriptor).as_int());
}catch(rclcpp::exceptions::ParameterUninitializedException& e){
try{
*arg = std::get<int>(mDefaultValue);
}catch (std::bad_variant_access const& ex){
throw std::runtime_error("Bad Variant Access: Type not int");
}
}
},
[&](std::string* arg){
try{
*arg = node->get_parameter(mParamDescriptor).as_string();
}catch(rclcpp::exceptions::ParameterUninitializedException& e){
try{
*arg = std::get<std::string>(mDefaultValue);
}catch (std::bad_variant_access const& ex){
throw std::runtime_error("Bad Variant Access: Type not std::string");
}
}
},
[&](bool* arg){
try{
*arg = node->get_parameter(mParamDescriptor).as_bool();
}catch(rclcpp::exceptions::ParameterUninitializedException& e){
try{
*arg = std::get<bool>(mDefaultValue);
}catch (std::bad_variant_access const& ex){
throw std::runtime_error("Bad Variant Access: Type not bool");
}
}
},
[&](double* arg){
try{
*arg = node->get_parameter(mParamDescriptor).as_double();
}catch(rclcpp::exceptions::ParameterUninitializedException& e){
try{
*arg = std::get<double>(mDefaultValue);
}catch (std::bad_variant_access const& ex){
throw std::runtime_error("Bad Variant Access: Type not double");
}
}
},
[&](float* arg){
try{
*arg = static_cast<float>(node->get_parameter(mParamDescriptor).as_double());
}catch(rclcpp::exceptions::ParameterUninitializedException& e){
try{
*arg = std::get<float>(mDefaultValue);
}catch (std::bad_variant_access const& ex){
throw std::runtime_error("Bad Variant Access: Type not float");
}
}
}
}, mData);
}

static inline auto declareParameters(rclcpp::Node* node, std::vector<ParameterWrapper>& params) -> void{
RCLCPP_INFO(rclcpp::get_logger("param_logger"), "Declaring %zu parameters...", params.size());
for(auto& param : params){
node->declare_parameter(param.mParamDescriptor, param.mType);
param.visit(node);
}
}
};
};
47 changes: 47 additions & 0 deletions perception/object_detector/object_detector.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#include "object_detector.hpp"

namespace mrover {

ObjectDetectorBase::ObjectDetectorBase() : rclcpp::Node(NODE_NAME), mLoopProfiler{get_logger()} {

std::vector<ParameterWrapper> params{
{"camera_frame", mCameraFrame, "zed_left_camera_frame"},
{"world_frame", mWorldFrame, "map"},
{"increment_weight", mObjIncrementWeight, 2},
{"decrement_weight", mObjDecrementWeight, 1},
{"hitcount_threshold", mObjHitThreshold, 5},
{"hitcount_max", mObjMaxHitcount, 10},
{"model_name", mModelName, "Large-Dataset"},
{"model_score_threshold", mModelScoreThreshold, 0.75},
{"model_nms_threshold", mModelNmsThreshold, 0.5}
};

ParameterWrapper::declareParameters(this, params);

std::filesystem::path packagePath = std::filesystem::path{ament_index_cpp::get_package_prefix("mrover")} / ".." / ".." / "src" / "mrover";

RCLCPP_INFO_STREAM(get_logger(), "Opening Model " << mModelName);

RCLCPP_INFO_STREAM(get_logger(), "Found package path " << packagePath);

mTensorRT = TensortRT{mModelName, packagePath.string()};

mDebugImgPub = create_publisher<sensor_msgs::msg::Image>("object_detector/debug_img", 1);

RCLCPP_INFO_STREAM(get_logger(), std::format("Object detector initialized with model: {} and thresholds: {} and {}", mModelName, mModelScoreThreshold, mModelNmsThreshold));
}

StereoObjectDetector::StereoObjectDetector() {
mSensorSub = create_subscription<sensor_msgs::msg::PointCloud2>("/camera/left/points", 1, [this](sensor_msgs::msg::PointCloud2::UniquePtr const& msg) {
StereoObjectDetector::pointCloudCallback(msg);
});
}
} // namespace mrover


auto main(int argc, char** argv) -> int {
rclcpp::init(argc, argv);
rclcpp::spin(std::make_shared<mrover::StereoObjectDetector>());
rclcpp::shutdown();
return EXIT_SUCCESS;
}
67 changes: 67 additions & 0 deletions perception/object_detector/object_detector.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
#pragma once

#include "pch.hpp"

namespace mrover {

class ObjectDetectorBase : public rclcpp::Node {

protected:
static constexpr char const* NODE_NAME = "object_detector";

std::unique_ptr<tf2_ros::Buffer> mTfBuffer = std::make_unique<tf2_ros::Buffer>(get_clock());
std::shared_ptr<tf2_ros::TransformBroadcaster> mTfBroadcaster = std::make_shared<tf2_ros::TransformBroadcaster>(this);
std::shared_ptr<tf2_ros::TransformListener> mTfListener = std::make_shared<tf2_ros::TransformListener>(*mTfBuffer);

std::string mCameraFrame;
std::string mWorldFrame;

std::string mModelName;

LoopProfiler mLoopProfiler;

TensortRT mTensorRT;

cv::Mat mRgbImage, mImageBlob;
sensor_msgs::msg::Image mDetectionsImageMessage;

rclcpp::Publisher<sensor_msgs::msg::Image>::SharedPtr mDebugImgPub;
rclcpp::Subscription<sensor_msgs::msg::PointCloud2>::SharedPtr mSensorSub;

// TODO(quintin): Do not hard code exactly two classes
std::vector<int> mObjectHitCounts{0, 0};

int mObjIncrementWeight{};
int mObjDecrementWeight{};
int mObjHitThreshold{};
int mObjMaxHitcount{};
float mModelScoreThreshold{};
float mModelNmsThreshold{};

auto spiralSearchForValidPoint(sensor_msgs::msg::PointCloud2::UniquePtr const& cloudPtr,
std::size_t u, std::size_t v,
std::size_t width, std::size_t height) const -> std::optional<SE3d>;

auto updateHitsObject(sensor_msgs::msg::PointCloud2::UniquePtr const& msg,
std::span<Detection const> detections,
cv::Size const& imageSize = {640, 640}) -> void;

auto publishDetectedObjects(cv::InputArray image) -> void;

static auto drawDetectionBoxes(cv::InputOutputArray image, std::span<Detection const> detections) -> void;

public:
explicit ObjectDetectorBase();

~ObjectDetectorBase() override = default;
};

class StereoObjectDetector final : public ObjectDetectorBase {
public:
explicit StereoObjectDetector();

static auto convertPointCloudToRGB(sensor_msgs::msg::PointCloud2::UniquePtr const& msg, cv::Mat const& image) -> void;

auto pointCloudCallback(sensor_msgs::msg::PointCloud2::UniquePtr const& msg) -> void;
};
} // namespace mrover
Loading