TSlam is built on top of UcoSLAM for augmented carpentry research. The main features are:
- Better API for use.
- Using STag
- Can do map fusion (merging one map into another).
- By indicating it's in instance mode, the system stop running global optimization and only keeps a fixed number of new added keyframes, which smooth the overall experience.
- Add CLAHE for preprocessing.
- Several bugs are resolved.
- we add a reconstruction pipeline to obtain digital models of the physically tagged objects
June 20
- Replaced 3rd party library ArUco to version
3.1.15
to avoid the uglified code in the original repo. - Added
Stag
in this repo, editedCMakeLists.txt
,cmake/dependencies.cmake
, andcmake/options.cmake
to make it work.
- Markers are detected in
src/utils/frameextractor.cpp
, line 670
auto markers=_mdetector->detect(Iinfo.im_org); // Iinfo.im_org is cv::Mat
- Marker's attribute (in
src/utils/markerdetector.h
):
// unique marker id
uint32_t id;
// three dimentional points of the marker wrt its center
// (This is something like [-0.5 * marker_size, 0.5 * marker_size, 0 ... ])
std::vector<cv::Point3f> points3d;
// original corners in the image
std::vector<cv::Point2f> corners;
// optional info about the marker
// (This attribute is probably not used)
std::string info;
- However, the tracking of feature points disappears at some point, not sure why.
This problem has been figured out. It's probably because the vocabulary is no specified. Also, the unconsistency of the marker size may be the reason that cause the distortion.
June 4
-
I tried to used the
-noKeyPoints
flag to map the timber, but the program would crash. The solution is to downgrade the UcoSLAM from 1.2.4 to 1.1.0, which is tagged "stable" in the releasing website. However, the result is unpromising (worse than the result with key points). -
Tried with smaller object and slightly bigger tags (2cm vs 2.3cm), works pretty well!
-
Debug command
gdb --args ./tslam_monocular '/home/tpp/UCOSlam-IBOIS/result/STag23mm_smallCube/use.mp4' '/home/tpp/UCOSlam-IBOIS/result/calibration_pixel3.yml' '-map' '/home/tpp/UCOSlam-IBOIS/result/STag23mm_smallCube/markers.map' "-markerSize" "0.023"
Function | 1.2.4 | 1.1.0 |
---|---|---|
Mapping w/o keypoints | X | O |
Mapping w/ keypoints | O | slow |
Tracking | Yes, but Not w/ GUI | X |
So now, we go back to 1.2.4. This table may be wrong :P
-
It lost tracking (the camera position is not recovered) when keypoints are not matching (but marker is detected). This should be fixed.
- Done the sticker generation (
stag_util/sticker_generator.py
) - Post processing now exports an .ply
- Seems to come out when inconstency introduced in the scene, I guess it's doing optimization on the map.
-
During Mapping
- ✔️ 2-phase mapping, which reduce this situation, but may still happen
Made a python script to do the experiment. It should be converted into the C++ version later.
- Move the code cause the lag into another thread, and let the main thread keep going => but this probably causes problem
- This mostly happens when the timber is moved, so
- Mapping with only the markers: tried, the accuracy is not good
- Sementic segmentaion which get rid of the background
- ✔️ 2-phase mapping, which reduce this situation, but may still happen
-
During intance
- Not updating Map -> lower accuracy
- Move the code cause the lag into another thread, and let the main thread keep going
- ✔️ Just cancel it
This has been tackled down by adding a system-scale parameter "-localizeOnly", which disables the optimization process.
-
- Takes two tag maps (the exported .yml file) as input
- Estimate the affine transformation matrix using OpenCV
Map 1 | Map 2 |
---|---|
- During instancing, key frames will be keep adding into the map, causing the FPS drop after a long run.
- In
src/utils/mapmanager.cpp:1594
, the functionset<uint32_t> MapManager::keyFrameCulling()
is modified. During instancing, the system will keep a fixed amount of number of new inserted keyframes, which is mainly used when the camera is going close to the timber and can't see a marker anymore. - In the test case, where the original map has 220 key frames and 20 new inserted key frames are preserved (which results in a total of 240 key frames), the FPS remains ~40 after 10 mins.
Image 33000 fps=36.002 21.0113 draw=53.9523 tracked Image 33100 fps=47.5574 24.0895 draw=52.6297 tracked Image 33200 fps=37.6424 21.3407 draw=52.7255 tracked Image 33300 fps=41.806 22.4592 draw=52.3648 tracked Image 33400 fps=40.5782 21.8395 draw=50.4677 tracked Image 33500 fps=45.1767 22.8656 draw=49.971 tracked Image 33600 fps=43.7453 22.5114 draw=49.983 tracked Image 33700 fps=36.9097 20.708 draw=51.1201 tracked Image 33800 fps=42.6364 22.2163 draw=50.3684 tracked Image 33900 fps=40.427 22.5163 draw=55.1343 tracked Image 34000 fps=41.9248 22.2568 draw=51.4437 tracked
Jul 25
- Usage:
build/utils/tslam_conbine_map 1.map 2.map combined.map
- In
map.h/map.cpp
, a new instanceMap::projectTo(Map anotherMap)
is added, which estimates the transformation matrix based on the duplicated tags and perform the transformation. - Result
- The beam is curved, both the original map and the combined result.
- Scanned Map
Map 1 | Map 2 |
---|---|
- Combined Map
Front | Up | Side |
---|---|---|
- The result is better
- Scanned Map
Map 1 | Map 2 |
---|---|
- Combined Map
Front | Up | Side |
---|---|---|
- Rolling-shutter Calibration: https://github.com/ethz-asl/kalibr/wiki/Rolling-Shutter-Camera-calibration
- We hit 100k! (But last time it crashed in 106XXX with a segmentation fault in the 3rd party library fbow :P)
Aug 01
- The map utilities (merge & optimization) is now a member function of Map.
- The optimization process is the main cause of the lagging problem.
- Actually there's a system-scale parameter "-enableLoopClosure" that controls this.
Mapping View | Mapped result |
---|---|
Mapping View | Mapped result |
---|---|
We will use only the .yaml
file since the point cloud obtained from the mapping phase has no definition. Mostly the points are concentrated where the fiducial markers are.
Added a small visualizer for debugging and dev.
Here's the workflow for the reconstruction: [To be added]
-
To get the intersections of the planes we switched to an AABB instead of OBB. It's simpler and faster for calculating the polygon intersections for all the tags. Here's the preliminary result:
We are building now the testing framework for the reconstruction part. The testing framework exploits doctest and the unit tests are composed of:
- unit testing of each geometric solver step with boolean check
- value check for accuracy of comparison vertex-vertex with the ground truth synthetically generated model
Ideally, this has to be done with a series of very different geometries to test the robustness of the reconstruction method.
⚠️ compatibility bug GTest-Open3d: there is an error by linking Open3d and GTest that we tried at first to the same executable (it does not happen with other libraries i.e. Eigen):/usr/bin/ld: CMakeFiles/test_gsolver.dir/test_gsolver.cc.o: in function `GSolverTest_TestHelloWorld_Test::TestBody()': test_gsolver.cc:(.text+0x12d): undefined reference to `testing::internal::GetBoolAssertionFailureMessage(testing::AssertionResult const&, char const*, char const*, char const*)' collect2: error: ld returned 1 exit status gmake[2]: *** [tests/CMakeFiles/test_gsolver.dir/build.make:152: tests/test_gsolver] Error 1 gmake[1]: *** [CMakeFiles/Makefile2:917: tests/CMakeFiles/test_gsolver.dir/all] Error 2 gmake: *** [Makefile:91: all] Error 2
To overcome this we decided to go for the header only library doctest.
The main base code is over but some things are left to do (see TODO.md
updated list). The major achievement is the precision at which faces of the timber are detected now. Of course this will need some adaptations once we will test with the generated synthetic data.
The new overhaul reconstruction and modified gsolver:
*The precise face matching of the reconstructed mesh:"
We change strategy to obtain the pool of all possible polygons. First, we intersect the polygons obtained from the intersection with the AABB among them. The splitting segments generated by the extremes given by the intersection on each plane are used to detect all the possible polygons describing the volume. The tassellation of the intersecting segments is obtained with CGAL/Segmentation_2d.hh
toolkit.
In order to use the tassellation we had to apply a plane-toXYplane transform to apply the CGAL toolkit only on 2d
The results are quite solid but we are now facing the problem of the selection of candidate faces for notched timber in the next step. See the image below:
Here's the list of next TODOs:
- check if planes are actually well assigned to each new tassellated polygons
- in real world data tolerance needs to be adaptative and permissive for the process of considering tag's centers as part of the plane
- internal polygons need to be effectivly culled out to leave only the external polygons.
- faces might not have tags. To obviate this, we should find a mechanism to maximize the existing tags (e.g., extending them with a radius or circle)
In order to select the face composing the actual volume we restructured the previous selectionag's algorithm into the 3 following filtering checks:
- (i) an adaptative tolerance: in order to find the thershold distance to consider a point belonging to a polygon's plane, we consider the median with tolerance (see code). It adapts from polygon to polygon;
- (ii) additional selection criteria by normal comparison: the filtering of closest tag's centers might not be enough in case of internal polygons or corner scenarios. In those cases we check the tag's center normal with the normal of the plane. If too different we pass.
- (iii) is the tag's center inside the polygon: the last check consists in projecting the filtered tag's center so far to the plane, if they are inside the polygon we select our face as a candidate.
⚠️ The currentmost important to the system is if there is an eligible face, which has no sticker attached to the timber face. In this case it will discared. The algorithm should be reinforced by covering also this case.
A basic top api interface is implemented, see the file tslam_reconstructor.hh
.
We also tried to give it a shot for very complex geometries such as this one but without success.
real object | reconstruct geo |
---|---|
We need testing for the next CI but also the refinement of the current reconstruction algorithm. We are using DocTest
for testing and we test both synthetic and reals scans maps (.yml
) with expected ground-truth data.
We changed the way we cluster the tags and now we implement a normal-based clustering from Cilantro. By adding a refinement by planes of the group of tags the tag grouping result quite robust.
The current challenge is the face selection after the tassellation. It might happen that tags are not present to validate the face, hence the polygon will not be selected and we will have a hole.
A different approach needs to be found for selecting eligible faces
Today we got a general stand up meeting with Yves and Julien. The development is successfull but we need to focus now on the evaluation of the TSLAM. This means that we need to cut out some features from the road map, fix parameters, do a due diligence on the evaluation of SLAMS and esstablish a scientific protocol for the evaluation.
We implemented a new method to check wether a polygon is a face or not by checking if a tag is contained in it. It is robust and the algorithm has been parametrized to detect multiple faces. Here we can see that we are able to now to describe complex shapes.
notched piece | complex piece |
---|---|
What we are missing now is a resilient refiner to fill the holes with e.g. one method in CGAL. In general this will be a limitation of the system to have markers to cover the entire piece. The rule to avoid holes is not clear at the moment.
update 22-02-2023
: we tried the triangulate_refine_and_fair_hole()
func in CGAL 5.5.1 but it is not giving the expected results. An issue is open on CGAL github.
The hole filling from CGAL is implemented and it patch some of the missing holes. Nonetheless, some non-manifold faces are still produced. A refinement of the hole patcher needs to be implemented.
notched piece | complex piece |
---|---|
The bug about clipping and perspective projection has been fixed. Mesh view is now working properly.
We started to preparing the evaluation pipeline and the set up for the evaluation campaign.
Test on "no key point detection" mode. From the screenshot we can see that in tag-only mode, the drift decreased.
with keypoints | without keypoints |
---|---|
Current TODOs:
-
(urgent) a piece with double stripes on edges each is not reconstructing correctly, check the bug out;
-
(urgent) establish an evaluation protocol for the tslam
-
(urgent) find a better algorithm for selecting candidate polygon faces
-
(urgent) solve reconstruction for more complex objects
-
(urgent) delete all dependecies of old UCOSlam and code labels replaced with TSLam
-
(urgent) integrate the reconstruction as a library in the main TSlam base code
-
(medium) TSLAM: adjust visualizer for overlapping mesh
-
(mdeium) TSLAM: establish scientific protocol for evaluation with optitrack
-
(medium) in reconstruction: the check mesh sanity is not wrokgin properly
-
(medium) in reconstruction: we need to implement a hole filler
-
(medium) set up CI/CD
-
(medium) for reconstruction sub-program: get rid of open3d and integrate a mesh I/O pipeline
-
(medium) package and integrate the new version of tslam in AC
-
(optional) implement the a
.xac
(xml based) format ints_geometric_solver.hh
andtslam_reconstructor.hh
; -
for reconstruction: write test units