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.
353 lines
11 KiB
353 lines
11 KiB
2 months ago
|
// License: Apache 2.0. See LICENSE file in root directory.
|
||
|
// Copyright(c) 2017 Intel Corporation. All Rights Reserved.
|
||
|
|
||
|
#ifndef LIBREALSENSE_RS2_PROCESSING_GL_HPP
|
||
|
#define LIBREALSENSE_RS2_PROCESSING_GL_HPP
|
||
|
|
||
|
#include <librealsense2/rs.hpp>
|
||
|
#include "rs_processing_gl.h"
|
||
|
|
||
|
#include <memory>
|
||
|
|
||
|
namespace rs2
|
||
|
{
|
||
|
namespace gl
|
||
|
{
|
||
|
class pointcloud;
|
||
|
class yuy_to_rgb;
|
||
|
|
||
|
inline void shutdown_rendering()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_shutdown_rendering(RS2_API_VERSION, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
#ifdef _glfw3_h_
|
||
|
inline void init_rendering(bool use_glsl = true)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
glfw_binding binding{
|
||
|
nullptr,
|
||
|
&glfwWindowHint,
|
||
|
&glfwCreateWindow,
|
||
|
&glfwDestroyWindow,
|
||
|
&glfwMakeContextCurrent,
|
||
|
&glfwGetCurrentContext,
|
||
|
&glfwSwapInterval,
|
||
|
&glfwGetProcAddress
|
||
|
};
|
||
|
|
||
|
rs2_gl_init_rendering_glfw(RS2_API_VERSION, binding, use_glsl ? 1 : 0, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
#else
|
||
|
inline void init_rendering(bool use_glsl = true)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_init_rendering(RS2_API_VERSION, use_glsl ? 1 : 0, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
inline void init_processing(bool use_glsl = true)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_init_processing(RS2_API_VERSION, use_glsl ? 1 : 0, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
inline void shutdown_processing()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_shutdown_processing(RS2_API_VERSION, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
#ifdef _glfw3_h_
|
||
|
inline void init_processing(GLFWwindow* share_with, bool use_glsl = true)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
glfw_binding binding{
|
||
|
nullptr,
|
||
|
&glfwWindowHint,
|
||
|
&glfwCreateWindow,
|
||
|
&glfwDestroyWindow,
|
||
|
&glfwMakeContextCurrent,
|
||
|
&glfwGetCurrentContext,
|
||
|
&glfwSwapInterval,
|
||
|
&glfwGetProcAddress
|
||
|
};
|
||
|
|
||
|
rs2_gl_init_processing_glfw(RS2_API_VERSION, share_with, binding, use_glsl ? 1 : 0, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/**
|
||
|
* GPU-Frame extension can be used to recover underlying OpenGL textures from a GPU frame
|
||
|
* and to check if frame data is stored in CPU or GPU memory
|
||
|
*/
|
||
|
class gpu_frame : public frame
|
||
|
{
|
||
|
public:
|
||
|
gpu_frame(const frame& f)
|
||
|
: frame(f)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
if (!f || (rs2_gl_is_frame_extendable_to(f.get(), RS2_GL_EXTENSION_VIDEO_FRAME, &e) == 0 && !e))
|
||
|
{
|
||
|
reset();
|
||
|
}
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
uint32_t get_texture_id(unsigned int id = 0) const
|
||
|
{
|
||
|
rs2_error * e = nullptr;
|
||
|
auto r = rs2_gl_frame_get_texture_id(get(), id, &e);
|
||
|
error::handle(e);
|
||
|
return r;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* yuy_decoder can be used for YUY->RGB conversion
|
||
|
* Similar in functionality to rs2::yuy_decoder but performed on the GPU
|
||
|
*/
|
||
|
class yuy_decoder : public rs2::yuy_decoder
|
||
|
{
|
||
|
public:
|
||
|
yuy_decoder() : rs2::yuy_decoder(init()) { }
|
||
|
|
||
|
private:
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_yuy_decoder(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
error::handle(e);
|
||
|
|
||
|
// Redirect options API to the processing block
|
||
|
//options::operator=(pb);
|
||
|
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Similar in functionality (Y411->RGB8 conversion) to rs2::y411_decoder but performed on the GPU.
|
||
|
*/
|
||
|
class y411_decoder : public rs2::y411_decoder
|
||
|
{
|
||
|
public:
|
||
|
y411_decoder() : rs2::y411_decoder( init() ) {}
|
||
|
|
||
|
private:
|
||
|
static std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_y411_decoder(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
error::handle(e);
|
||
|
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Colorizer can be used for Depth->RGB conversion, including histogram equalization
|
||
|
* Similar in functionality to rs2::colorizer but performed on the GPU
|
||
|
*/
|
||
|
class colorizer : public rs2::colorizer
|
||
|
{
|
||
|
public:
|
||
|
colorizer() : rs2::colorizer(init()) { }
|
||
|
|
||
|
private:
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_colorizer(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
error::handle(e);
|
||
|
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class uploader : public rs2::filter
|
||
|
{
|
||
|
public:
|
||
|
uploader() : rs2::filter(init()) { }
|
||
|
|
||
|
private:
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_uploader(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
error::handle(e);
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Generating the 3D point cloud base on depth frame also create the mapped texture.
|
||
|
* Similar in functionality to rs2::pointcloud but performed on the GPU
|
||
|
*/
|
||
|
class pointcloud : public rs2::pointcloud
|
||
|
{
|
||
|
public:
|
||
|
pointcloud() : rs2::pointcloud(init()) {}
|
||
|
|
||
|
pointcloud(rs2_stream stream, int index = 0)
|
||
|
: rs2::pointcloud(init())
|
||
|
{
|
||
|
set_option(RS2_OPTION_STREAM_FILTER, float(stream));
|
||
|
set_option(RS2_OPTION_STREAM_INDEX_FILTER, float(index));
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
friend class context;
|
||
|
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_pointcloud(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
|
||
|
error::handle(e);
|
||
|
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Camera Renderer can be used to visualize underlying camera model, given a frame from the camera
|
||
|
* Based on official CAD files
|
||
|
* For proper rendering, set_matrix needs to be used to configure projection and view matrix
|
||
|
*/
|
||
|
class camera_renderer : public rs2::filter
|
||
|
{
|
||
|
public:
|
||
|
camera_renderer() : rs2::filter(init()) {}
|
||
|
|
||
|
void set_matrix(rs2_gl_matrix_type type, float* m4x4)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_set_matrix(get(), type, m4x4, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
static const auto OPTION_MOUSE_X = rs2_option(RS2_OPTION_COUNT + 1);
|
||
|
static const auto OPTION_MOUSE_Y = rs2_option(RS2_OPTION_COUNT + 2);
|
||
|
static const auto OPTION_MOUSE_PICK = rs2_option(RS2_OPTION_COUNT + 3);
|
||
|
static const auto OPTION_WAS_PICKED = rs2_option(RS2_OPTION_COUNT + 4);
|
||
|
static const auto OPTION_SELECTED = rs2_option(RS2_OPTION_COUNT + 5);
|
||
|
private:
|
||
|
friend class context;
|
||
|
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_camera_renderer(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
|
||
|
error::handle(e);
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Pointcloud renderer can be used to render librealsense points objects to screen
|
||
|
* It can render pointcloud as points (by setting proprietary OPTION_FILLED option to 0)
|
||
|
* or via connected polygons (where every three adjacent points produce a triangle)
|
||
|
* For proper rendering, set_matrix needs to be used to configure projection and view matrix
|
||
|
*/
|
||
|
class pointcloud_renderer : public rs2::filter
|
||
|
{
|
||
|
public:
|
||
|
pointcloud_renderer() : rs2::filter(init()) {}
|
||
|
|
||
|
void set_matrix(rs2_gl_matrix_type type, float* m4x4)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
rs2_gl_set_matrix(get(), type, m4x4, &e);
|
||
|
error::handle(e);
|
||
|
}
|
||
|
|
||
|
static const auto OPTION_FILLED = rs2_option(RS2_OPTION_COUNT + 1);
|
||
|
static const auto OPTION_SHADED = rs2_option(RS2_OPTION_COUNT + 2);
|
||
|
|
||
|
static const auto OPTION_MOUSE_X = rs2_option(RS2_OPTION_COUNT + 3);
|
||
|
static const auto OPTION_MOUSE_Y = rs2_option(RS2_OPTION_COUNT + 4);
|
||
|
static const auto OPTION_MOUSE_PICK = rs2_option(RS2_OPTION_COUNT + 5);
|
||
|
|
||
|
static const auto OPTION_PICKED_X = rs2_option(RS2_OPTION_COUNT + 6);
|
||
|
static const auto OPTION_PICKED_Y = rs2_option(RS2_OPTION_COUNT + 7);
|
||
|
static const auto OPTION_PICKED_Z = rs2_option(RS2_OPTION_COUNT + 8);
|
||
|
|
||
|
static const auto OPTION_PICKED_ID = rs2_option(RS2_OPTION_COUNT + 9);
|
||
|
static const auto OPTION_SELECTED = rs2_option(RS2_OPTION_COUNT + 10);
|
||
|
static const auto OPTION_ORIGIN_PICKED = rs2_option(RS2_OPTION_COUNT + 11);
|
||
|
|
||
|
static const auto OPTION_NORMAL_X = rs2_option(RS2_OPTION_COUNT + 12);
|
||
|
static const auto OPTION_NORMAL_Y = rs2_option(RS2_OPTION_COUNT + 13);
|
||
|
static const auto OPTION_NORMAL_Z = rs2_option(RS2_OPTION_COUNT + 14);
|
||
|
|
||
|
static const auto OPTION_SCALE_FACTOR = rs2_option(RS2_OPTION_COUNT + 15);
|
||
|
|
||
|
private:
|
||
|
friend class context;
|
||
|
|
||
|
std::shared_ptr<rs2_processing_block> init()
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_pointcloud_renderer(RS2_API_VERSION, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
|
||
|
error::handle(e);
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
*/
|
||
|
class align : public rs2::align
|
||
|
{
|
||
|
public:
|
||
|
align(rs2_stream to) : rs2::align(init(to)) {}
|
||
|
|
||
|
private:
|
||
|
friend class context;
|
||
|
|
||
|
std::shared_ptr<rs2_processing_block> init(rs2_stream to)
|
||
|
{
|
||
|
rs2_error* e = nullptr;
|
||
|
|
||
|
auto block = std::shared_ptr<rs2_processing_block>(
|
||
|
rs2_gl_create_align(RS2_API_VERSION, to, &e),
|
||
|
rs2_delete_processing_block);
|
||
|
|
||
|
error::handle(e);
|
||
|
|
||
|
return block;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
#endif // LIBREALSENSE_RS2_PROCESSING_GL_HPP
|