diff --git a/CMakeLists.txt b/CMakeLists.txt index cfad1f58e61625bbdd0282891393e458f46ace8d..f1441f49c89b9b8ed4cb8c601783f99788820546 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -186,6 +186,10 @@ if (BUILD_VISION) add_subdirectory(applications/vision) endif() +if (HAVE_PCL) + add_subdirectory(applications/registration) +endif() + if (BUILD_RECONSTRUCT AND HAVE_PCL) add_subdirectory(applications/reconstruct) endif() diff --git a/applications/reconstruct/src/registration.cpp b/applications/reconstruct/src/registration.cpp index 70e2fb1da830faa5d5a5e53d58b1905a5ea4a535..ccb17ac5078eac6443b88c87fa8e5afce30e0e9c 100644 --- a/applications/reconstruct/src/registration.cpp +++ b/applications/reconstruct/src/registration.cpp @@ -233,7 +233,7 @@ Eigen::Matrix4f findTransformation(vector<PointCloud<PointXYZ>::Ptr> clouds_sour // score new transformation double score = 0.0; for (size_t j = 0; j < n_clouds; ++j) { - score += validate.validateTransformation(clouds_source[j], clouds_target[j], T); + score += validate.validateTransformation(clouds_source[j], clouds_target[j], T); // CHECK Is use of T here a mistake?? } score /= n_clouds; diff --git a/applications/reconstruct/src/virtual_source.cpp b/applications/reconstruct/src/virtual_source.cpp index 157b71a3d420ab08775c2842b2cb5bffd3aa6898..578b360907a04a8198395a778ca1073cd7ae94be 100644 --- a/applications/reconstruct/src/virtual_source.cpp +++ b/applications/reconstruct/src/virtual_source.cpp @@ -19,8 +19,8 @@ VirtualSource::VirtualSource(ftl::rgbd::Source *host) params_.fy = params_.fx; params_.width = rays_->value("width", 640); params_.height = rays_->value("height", 480); - params_.cx = params_.width / 2; - params_.cy = params_.height / 2; + params_.cx = -((double)params_.width / 2); + params_.cy = -((double)params_.height / 2); params_.maxDepth = rays_->value("max_depth", 10.0f); params_.minDepth = rays_->value("min_depth", 0.1f); @@ -43,8 +43,8 @@ bool VirtualSource::grab() { DepthCameraParams params; params.fx = params_.fx; params.fy = params_.fy; - params.mx = params_.cx; - params.my = params_.cy; + params.mx = -params_.cx; + params.my = -params_.cy; params.m_imageWidth = params_.width; params.m_imageHeight = params_.height; params.m_sensorDepthWorldMin = params_.minDepth; diff --git a/applications/registration/CMakeLists.txt b/applications/registration/CMakeLists.txt new file mode 100644 index 0000000000000000000000000000000000000000..f3bde265315d3f3c507692b405829d5d610bae18 --- /dev/null +++ b/applications/registration/CMakeLists.txt @@ -0,0 +1,12 @@ +set(REGSRC + src/main.cpp + src/manual.cpp + src/correspondances.cpp +) + +add_executable(ftl-register ${REGSRC}) + +target_include_directories(ftl-register PRIVATE src) + +target_link_libraries(ftl-register ftlcommon ftlnet ftlrgbd Threads::Threads ${OpenCV_LIBS}) + diff --git a/applications/registration/src/correspondances.cpp b/applications/registration/src/correspondances.cpp new file mode 100644 index 0000000000000000000000000000000000000000..58eb1c6e2cb9b7bad3873225ae0bbfe61a9d99c3 --- /dev/null +++ b/applications/registration/src/correspondances.cpp @@ -0,0 +1,82 @@ +#include "correspondances.hpp" +#include <nlohmann/json.hpp> + +using ftl::registration::Correspondances; +using std::string; +using std::map; +using std::vector; +using ftl::rgbd::Source; +using pcl::PointCloud; +using pcl::PointXYZ; +using pcl::PointXYZRGB; +using nlohmann::json; +using std::tuple; +using std::make_tuple; + + +Correspondances::Correspondances(Source *src, Source *targ) + : parent_(nullptr), targ_(targ), src_(src), + targ_cloud_(new PointCloud<PointXYZ>), + src_cloud_(new PointCloud<PointXYZ>), uptodate_(true) { + transform_.setIdentity(); +} + +Correspondances::Correspondances(Correspondances *parent, Source *src) + : parent_(parent), targ_(parent_->source()), src_(src), + targ_cloud_(new PointCloud<PointXYZ>), + src_cloud_(new PointCloud<PointXYZ>), uptodate_(true) { + transform_.setIdentity(); +} + +bool Correspondances::add(int tx, int ty, int sx, int sy) { + Eigen::Vector4f p1 = targ_->point(tx,ty); + Eigen::Vector4f p2 = src_->point(sx,sy); + PointXYZ pcl_p1; + pcl_p1.x = p1[0]; + pcl_p1.y = p1[1]; + pcl_p1.z = p1[2]; + + PointXYZ pcl_p2; + pcl_p2.x = p2[0]; + pcl_p2.y = p2[1]; + pcl_p2.z = p2[2]; + + if (pcl_p2.z >= 40.0f || pcl_p1.z >= 40.0f) { + LOG(WARNING) << "Bad point choosen"; + return false; + } + + targ_cloud_->push_back(pcl_p1); + src_cloud_->push_back(pcl_p2); + log_.push_back(make_tuple(tx,ty,sx,sy)); + + uptodate_ = false; + return true; +} + +void Correspondances::removeLast() { + uptodate_ = false; + targ_cloud_->erase(targ_cloud_->end()-1); + src_cloud_->erase(src_cloud_->end()-1); + log_.pop_back(); +} + +double Correspondances::estimateTransform() { + uptodate_ = true; + int n_points = targ_cloud_->size(); + + vector<int> idx(n_points); + for (int i = 0; i < n_points; i++) { idx[i] = i; } + + pcl::registration::TransformationValidationEuclidean<PointXYZ, PointXYZ> validate; + pcl::registration::TransformationEstimationSVD<PointXYZ,PointXYZ> svd; + + svd.estimateRigidTransformation(*src_cloud_, idx, *targ_cloud_, idx, transform_); + + return validate.validateTransformation(src_cloud_, targ_cloud_, transform_); +} + +Eigen::Matrix4f Correspondances::transform() { + if (!uptodate_) estimateTransform(); + return (parent_) ? parent_->transform() * transform_ : transform_; +} diff --git a/applications/registration/src/correspondances.hpp b/applications/registration/src/correspondances.hpp new file mode 100644 index 0000000000000000000000000000000000000000..2c09ab4d87f8760b2a7f0c3b50633f83e870ceab --- /dev/null +++ b/applications/registration/src/correspondances.hpp @@ -0,0 +1,78 @@ +#ifndef _FTL_REG_CORRESPONDANCES_HPP_ +#define _FTL_REG_CORRESPONDANCES_HPP_ + +#include <ftl/rgbd/source.hpp> +#include <nlohmann/json.hpp> + +#include <pcl/common/transforms.h> + +#include <pcl/registration/transformation_estimation_svd.h> +#include <pcl/registration/transformation_estimation_svd_scale.h> +#include <pcl/registration/transformation_validation.h> +#include <pcl/registration/transformation_validation_euclidean.h> + +#include <vector> +#include <string> +#include <map> +#include <tuple> + +namespace ftl { +namespace registration { + +/** + * Manage the identified set of correspondances between two sources. There may + * also be a parent correspondances object to support the chaining of + * transforms. + */ +class Correspondances { + public: + Correspondances(ftl::rgbd::Source *src, ftl::rgbd::Source *targ); + Correspondances(Correspondances *parent, ftl::rgbd::Source *src); + + ftl::rgbd::Source *source() { return src_; } + ftl::rgbd::Source *target() { return targ_; } + + /** + * Add a new correspondance point. The point will only be added if there is + * a valid depth value at that location. + * + * @param tx X-Coordinate in target image + * @param ty Y-Coordinate in target image + * @param sx X-Coordinate in source image + * @param sy Y-Coordinate in source image + * @return False if point was invalid and not added. + */ + bool add(int tx, int ty, int sx, int sy); + + void removeLast(); + + const std::vector<std::tuple<int,int,int,int>> &screenPoints() const { return log_; } + + /** + * Calculate a transform using current set of correspondances. + * + * @return Validation score of the transform. + */ + double estimateTransform(); + + /** + * Get the estimated transform. This includes any parent transforms to make + * it relative to root camera. + */ + Eigen::Matrix4f transform(); + + private: + Correspondances *parent_; + ftl::rgbd::Source *targ_; + ftl::rgbd::Source *src_; + pcl::PointCloud<pcl::PointXYZ>::Ptr targ_cloud_; + pcl::PointCloud<pcl::PointXYZ>::Ptr src_cloud_; + Eigen::Matrix4f transform_; + bool uptodate_; + std::vector<std::tuple<int,int,int,int>> log_; +}; + +} +} + +#endif // _FTL_REG_CORRESPONDANCES_HPP_ diff --git a/applications/registration/src/main.cpp b/applications/registration/src/main.cpp new file mode 100644 index 0000000000000000000000000000000000000000..904320ae8c62eef6b91ae3d82c7e652bf3e1d5da --- /dev/null +++ b/applications/registration/src/main.cpp @@ -0,0 +1,19 @@ +#include <loguru.hpp> +#include <ftl/configuration.hpp> +#include <string> + +#include "manual.hpp" + +using std::string; + +int main(int argc, char **argv) { + auto root = ftl::configure(argc, argv, "registration_default"); + + // TODO + ftl::registration::manual(root); + if (root->value("mode", string("manual") == "manual")) { + //ftl::registration::manual(root); + } + + return 0; +} diff --git a/applications/registration/src/manual.cpp b/applications/registration/src/manual.cpp new file mode 100644 index 0000000000000000000000000000000000000000..c9102f98ed92064cb00c089fc3dee92a311e8d2e --- /dev/null +++ b/applications/registration/src/manual.cpp @@ -0,0 +1,255 @@ +#include "manual.hpp" +#include "correspondances.hpp" + +#include <loguru.hpp> + +#include <ftl/net/universe.hpp> +#include <ftl/rgbd/source.hpp> + +#include <opencv2/core.hpp> +#include <opencv2/core/utility.hpp> +#include <opencv2/imgproc.hpp> +#include <opencv2/calib3d.hpp> +#include <opencv2/imgcodecs.hpp> +#include <opencv2/videoio.hpp> +#include <opencv2/highgui.hpp> + +#include <map> + +using std::string; +using std::vector; +using std::pair; +using std::map; + +using ftl::net::Universe; +using ftl::rgbd::Source; + +using cv::Mat; +using cv::Point; +using cv::Scalar; + +using namespace ftl::registration; + +using MouseAction = std::function<void(int, int, int, int)>; + +static void setMouseAction(const std::string& winName, const MouseAction &action) +{ + cv::setMouseCallback(winName, + [] (int event, int x, int y, int flags, void* userdata) { + (*(MouseAction*)userdata)(event, x, y, flags); + }, (void*)&action); +} + +static MouseAction tmouse; +static MouseAction smouse; + + +static void to_json(nlohmann::json &json, map<string, Eigen::Matrix4f> &transformations) { + for (auto &item : transformations) { + auto val = nlohmann::json::array(); + for(size_t i = 0; i < 16; i++) { val.push_back((float) item.second.data()[i]); } + json[item.first] = val; + } +} + +static bool saveTransformations(const string &path, map<string, Eigen::Matrix4f> &data) { + nlohmann::json data_json; + to_json(data_json, data); + std::ofstream file(path); + + if (!file.is_open()) { + LOG(ERROR) << "Error writing transformations to file " << path; + return false; + } + + file << std::setw(4) << data_json; + return true; +} + +static void build_correspondances(const vector<Source*> &sources, map<string, Correspondances*> &cs, int origin) { + Correspondances *last = nullptr; + + cs[sources[origin]->getURI()] = nullptr; + + for (int i=origin-1; i>=0; i--) { + if (last == nullptr) { + auto *c = new Correspondances(sources[i], sources[origin]); + last = c; + cs[sources[i]->getURI()] = c; + } else { + auto *c = new Correspondances(last, sources[i]); + last = c; + cs[sources[i]->getURI()] = c; + } + } + + last = nullptr; + + for (int i=origin+1; i<sources.size(); i++) { + if (last == nullptr) { + auto *c = new Correspondances(sources[i], sources[origin]); + last = c; + cs[sources[i]->getURI()] = c; + } else { + auto *c = new Correspondances(last, sources[i]); + last = c; + cs[sources[i]->getURI()] = c; + } + } +} + +void ftl::registration::manual(ftl::Configurable *root) { + using namespace cv; + + Universe *net = ftl::create<Universe>(root, "net"); + + net->start(); + net->waitConnections(); + + auto sources = ftl::createArray<Source>(root, "sources", net); + + int curtarget = 0; + bool freeze = false; + + vector<Mat> rgb; + vector<Mat> depth; + + if (sources.size() == 0) return; + + rgb.resize(sources.size()); + depth.resize(sources.size()); + + cv::namedWindow("Target", cv::WINDOW_KEEPRATIO); + cv::namedWindow("Source", cv::WINDOW_KEEPRATIO); + + //Correspondances c(sources[targsrc], sources[cursrc]); + map<string, Correspondances*> corrs; + build_correspondances(sources, corrs, root->value("origin", 0)); + + int lastTX = 0; + int lastTY = 0; + int lastSX = 0; + int lastSY = 0; + + tmouse = [&]( int event, int ux, int uy, int) { + if (event == 1) { // click + lastTX = ux; + lastTY = uy; + } + }; + setMouseAction("Target", tmouse); + + smouse = [&]( int event, int ux, int uy, int) { + if (event == 1) { // click + lastSX = ux; + lastSY = uy; + } + }; + setMouseAction("Source", smouse); + + bool depthtoggle = false; + double lastScore = 0.0; + + Correspondances *current = corrs[sources[curtarget]->getURI()]; + + while (ftl::running) { + if (!freeze) { + // Grab the latest frames from sources + for (int i=0; i<sources.size(); i++) { + if (sources[i]->isReady()) { + sources[i]->grab(); + } + } + } + + // Get the raw rgb and depth frames from sources + Mat ttrgb, trgb, tdepth, ttdepth; + Mat tsrgb, srgb, sdepth, tsdepth; + + if (current == nullptr) { + srgb = Mat(Size(640,480), CV_8UC3, Scalar(0,0,0)); + trgb = Mat(Size(640,480), CV_8UC3, Scalar(0,0,0)); + + putText(srgb, string("No correspondance for ") + sources[curtarget]->getURI(), Point(10,20), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + + if (!trgb.empty()) imshow("Target", trgb); + if (!srgb.empty()) imshow("Source", srgb); + } else if (current->source()->isReady() && current->target()->isReady()) { + current->source()->getFrames(tsrgb, tsdepth); + current->target()->getFrames(ttrgb, ttdepth); + tsrgb.copyTo(srgb); + ttrgb.copyTo(trgb); + ttdepth.convertTo(tdepth, CV_8U, 255.0f / 10.0f); + applyColorMap(tdepth, tdepth, cv::COLORMAP_JET); + tsdepth.convertTo(sdepth, CV_8U, 255.0f / 10.0f); + applyColorMap(sdepth, sdepth, cv::COLORMAP_JET); + + // Apply points and labels... + auto &points = current->screenPoints(); + for (auto &p : points) { + auto [tx,ty,sx,sy] = p; + drawMarker(srgb, Point(sx,sy), Scalar(0,255,0), MARKER_TILTED_CROSS); + drawMarker(sdepth, Point(sx,sy), Scalar(0,255,0), MARKER_TILTED_CROSS); + drawMarker(trgb, Point(tx,ty), Scalar(0,255,0), MARKER_TILTED_CROSS); + drawMarker(tdepth, Point(tx,ty), Scalar(0,255,0), MARKER_TILTED_CROSS); + } + + // Add most recent click position + drawMarker(srgb, Point(lastSX, lastSY), Scalar(0,0,255), MARKER_TILTED_CROSS); + drawMarker(trgb, Point(lastTX, lastTY), Scalar(0,0,255), MARKER_TILTED_CROSS); + drawMarker(sdepth, Point(lastSX, lastSY), Scalar(0,0,255), MARKER_TILTED_CROSS); + drawMarker(tdepth, Point(lastTX, lastTY), Scalar(0,0,255), MARKER_TILTED_CROSS); + + putText(srgb, string("Source: ") + current->source()->getURI(), Point(10,20), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + putText(trgb, string("Target: ") + current->target()->getURI(), Point(10,20), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + putText(srgb, string("Score: ") + std::to_string(lastScore), Point(10,40), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + putText(trgb, string("Score: ") + std::to_string(lastScore), Point(10,40), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + if (freeze) putText(srgb, string("Paused"), Point(10,50), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + if (freeze) putText(trgb, string("Paused"), Point(10,50), FONT_HERSHEY_PLAIN, 1.0, Scalar(0,0,255), 1); + + if (!trgb.empty()) imshow("Target", (depthtoggle) ? tdepth : trgb); + if (!srgb.empty()) imshow("Source", (depthtoggle) ? sdepth : srgb); + } + + int key = cv::waitKey(20); + if (key == 27) break; + else if (key >= 48 && key <= 57) { + curtarget = key-48; + if (curtarget >= sources.size()) curtarget = 0; + current = corrs[sources[curtarget]->getURI()]; + } else if (key == 'd') depthtoggle = !depthtoggle; + else if (key == 'a') { + if (current->add(lastTX,lastTY,lastSX,lastSY)) { + lastTX = lastTY = lastSX = lastSY = 0; + lastScore = current->estimateTransform(); + } + } else if (key == 'u') { + current->removeLast(); + lastScore = current->estimateTransform(); + } else if (key == 's') { + // Save + map<string, Eigen::Matrix4f> transforms; + //transforms[sources[targsrc]->getURI()] = Eigen::Matrix4f().setIdentity(); + //transforms[sources[cursrc]->getURI()] = c.transform(); + + for (auto x : corrs) { + if (x.second == nullptr) { + transforms[x.first] = Eigen::Matrix4f().setIdentity(); + } else { + transforms[x.first] = x.second->transform(); + } + } + + saveTransformations(root->value("output", string("./test.json")), transforms); + LOG(INFO) << "Saved!"; + } + else if (key == 32) freeze = !freeze; + } + + // store transformations in map<string Matrix4f> + + // TODO: (later) extract features and correspondencies from complete cloud + // and do ICP to find best possible transformation + + // saveTransformations(const string &path, map<string, Matrix4f> &data) +} diff --git a/applications/registration/src/manual.hpp b/applications/registration/src/manual.hpp new file mode 100644 index 0000000000000000000000000000000000000000..1b315cb038b22dc85adc1f0dc0b05ad4c772a920 --- /dev/null +++ b/applications/registration/src/manual.hpp @@ -0,0 +1,14 @@ +#ifndef _FTL_REGISTRATION_MANUAL_HPP_ +#define _FTL_REGISTRATION_MANUAL_HPP_ + +#include <ftl/configurable.hpp> + +namespace ftl { +namespace registration { + +void manual(ftl::Configurable *root); + +} +} + +#endif // _FTL_REGISTRATION_MANUAL_HPP_ diff --git a/components/rgbd-sources/src/source.cpp b/components/rgbd-sources/src/source.cpp index bc84322c3d95c320e429accb39227dcce70c9ffd..787dbdadaa51ef8efb1d9f3d1938c4bc302d2c81 100644 --- a/components/rgbd-sources/src/source.cpp +++ b/components/rgbd-sources/src/source.cpp @@ -149,8 +149,8 @@ void Source::getFrames(cv::Mat &rgb, cv::Mat &depth) { Eigen::Vector4f Source::point(uint ux, uint uy) { const auto ¶ms = parameters(); - const float x = ((float)ux-(float)params.cx) / (float)params.fx; - const float y = ((float)uy-(float)params.cy) / (float)params.fy; + const float x = ((float)ux+(float)params.cx) / (float)params.fx; + const float y = ((float)uy+(float)params.cy) / (float)params.fy; SHARED_LOCK(mutex_,lk); const float depth = depth_.at<float>(uy,ux); diff --git a/components/rgbd-sources/src/streamer.cpp b/components/rgbd-sources/src/streamer.cpp index 6d926ff8c7f4f94d9e83c341a86b875a41b3bdd0..b3ad77e6c1feff8da5c99d5ad2baaf16e812a109 100644 --- a/components/rgbd-sources/src/streamer.cpp +++ b/components/rgbd-sources/src/streamer.cpp @@ -266,7 +266,11 @@ void Streamer::_schedule() { // Grab job pool_.push([this,src](int id) { //StreamSource *src = sources_[uri]; - src->src->grab(); + try { + src->src->grab(); + } catch (...) { + LOG(ERROR) << "Exception when grabbing frame"; + } // CHECK (Nick) Can state be an atomic instead? //UNIQUE_LOCK(src->mutex, lk); diff --git a/config/config_nick.jsonc b/config/config_nick.jsonc index 60f7da97c4355043401bd01343e98c14232c1e3a..467d455c8712754bb8a07a20318d83fd97e4c59b 100644 --- a/config/config_nick.jsonc +++ b/config/config_nick.jsonc @@ -171,11 +171,13 @@ "registration_default": { "net": { - "peers": ["tcp://ftl-node-2:9001"] + "peers": [] }, "sources": [ - {"uri":"ftl://utu.fi/node2#vision_default/source"} - ] + {"uri":"file:///home/nick/Pictures/FTL/snap.tar.gz#0", "index": 0}, + {"uri":"file:///home/nick/Pictures/FTL/snap.tar.gz#1", "index": 1} + ], + "origin": 0 }, "reconstruction_rs": {