From b9623539a5d3933d4f7bc84c1ef5b393826f852d Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sat, 10 Jun 2017 19:14:23 +0200 Subject: [PATCH 01/53] first commit. Added photometric calibration class and a dummy function --- modules/pcalib/CMakeLists.txt | 2 + modules/pcalib/README.md | 2 + modules/pcalib/include/opencv2/pcalib.hpp | 66 ++++++++++++++++ modules/pcalib/src/pcalib.cpp | 93 +++++++++++++++++++++++ modules/pcalib/src/precomp.hpp | 51 +++++++++++++ 5 files changed, 214 insertions(+) create mode 100644 modules/pcalib/CMakeLists.txt create mode 100644 modules/pcalib/README.md create mode 100644 modules/pcalib/include/opencv2/pcalib.hpp create mode 100644 modules/pcalib/src/pcalib.cpp create mode 100644 modules/pcalib/src/precomp.hpp diff --git a/modules/pcalib/CMakeLists.txt b/modules/pcalib/CMakeLists.txt new file mode 100644 index 000000000..e7cb2450d --- /dev/null +++ b/modules/pcalib/CMakeLists.txt @@ -0,0 +1,2 @@ +set(the_description "Photometric Calibration") +ocv_define_module(pcalib opencv_core opencv_imgproc opencv_calib3d opencv_features2d opencv_highgui WRAP python) diff --git a/modules/pcalib/README.md b/modules/pcalib/README.md new file mode 100644 index 000000000..3d7db7867 --- /dev/null +++ b/modules/pcalib/README.md @@ -0,0 +1,2 @@ +Photometric Calibration +================================================ diff --git a/modules/pcalib/include/opencv2/pcalib.hpp b/modules/pcalib/include/opencv2/pcalib.hpp new file mode 100644 index 000000000..d37787a85 --- /dev/null +++ b/modules/pcalib/include/opencv2/pcalib.hpp @@ -0,0 +1,66 @@ +/*M/////////////////////////////////////////////////////////////////////////////////////// + // + // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. + // + // By downloading, copying, installing or using the software you agree to this license. + // If you do not agree to this license, do not download, install, + // copy or use the software. + // + // + // License Agreement + // For Open Source Computer Vision Library + // + // Copyright (C) 2014, OpenCV Foundation, all rights reserved. + // Third party copyrights are property of their respective owners. + // + // Redistribution and use in source and binary forms, with or without modification, + // are permitted provided that the following conditions are met: + // + // * Redistribution's of source code must retain the above copyright notice, + // this list of conditions and the following disclaimer. + // + // * Redistribution's in binary form must reproduce the above copyright notice, + // this list of conditions and the following disclaimer in the documentation + // and/or other materials provided with the distribution. + // + // * The name of the copyright holders may not be used to endorse or promote products + // derived from this software without specific prior written permission. + // + // This software is provided by the copyright holders and contributors "as is" and + // any express or implied warranties, including, but not limited to, the implied + // warranties of merchantability and fitness for a particular purpose are disclaimed. + // In no event shall the Intel Corporation or contributors be liable for any direct, + // indirect, incidental, special, exemplary, or consequential damages + // (including, but not limited to, procurement of substitute goods or services; + // loss of use, data, or profits; or business interruption) however caused + // and on any theory of liability, whether in contract, strict liability, + // or tort (including negligence or otherwise) arising in any way out of + // the use of this software, even if advised of the possibility of such damage. + // + //M*/ + +#ifndef __OPENCV_PCALIB_HPP__ +#define __OPENCV_PCALIB_HPP__ + +#include +#include +#include + +#include + +namespace cv{ namespace pcalib{ + +//! @addtogroup pcalib +//! @{ + +class CV_EXPORTS PhotometricCalibrator : public Algorithm +{ +public: + bool validImgs(std::vector &inputImgs, std::vector &exposureTime); +}; + +//! @} + +}} // namespace pcalib, cv + +#endif diff --git a/modules/pcalib/src/pcalib.cpp b/modules/pcalib/src/pcalib.cpp new file mode 100644 index 000000000..21205850d --- /dev/null +++ b/modules/pcalib/src/pcalib.cpp @@ -0,0 +1,93 @@ +/*M/////////////////////////////////////////////////////////////////////////////////////// + // + // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. + // + // By downloading, copying, installing or using the software you agree to this license. + // If you do not agree to this license, do not download, install, + // copy or use the software. + // + // + // License Agreement + // For Open Source Computer Vision Library + // + // Copyright (C) 2014, OpenCV Foundation, all rights reserved. + // Third party copyrights are property of their respective owners. + // + // Redistribution and use in source and binary forms, with or without modification, + // are permitted provided that the following conditions are met: + // + // * Redistribution's of source code must retain the above copyright notice, + // this list of conditions and the following disclaimer. + // + // * Redistribution's in binary form must reproduce the above copyright notice, + // this list of conditions and the following disclaimer in the documentation + // and/or other materials provided with the distribution. + // + // * The name of the copyright holders may not be used to endorse or promote products + // derived from this software without specific prior written permission. + // + // This software is provided by the copyright holders and contributors "as is" and + // any express or implied warranties, including, but not limited to, the implied + // warranties of merchantability and fitness for a particular purpose are disclaimed. + // In no event shall the Intel Corporation or contributors be liable for any direct, + // indirect, incidental, special, exemplary, or consequential damages + // (including, but not limited to, procurement of substitute goods or services; + // loss of use, data, or profits; or business interruption) however caused + // and on any theory of liability, whether in contract, strict liability, + // or tort (including negligence or otherwise) arising in any way out of + // the use of this software, even if advised of the possibility of such damage. + // + //M*/ + +#ifndef __OPENCV_PCALIB_CPP__ +#define __OPENCV_PCALIB_CPP__ +#ifdef __cplusplus + +#include "precomp.hpp" +#include "opencv2/pcalib.hpp" + +#include + +#include +#include + +namespace cv{ namespace pcalib{ + +using namespace std; + +bool PhotometricCalibrator::validImgs(std::vector &inputImgs, std::vector &exposureTime) +{ + if(inputImgs.empty() || exposureTime.empty() || inputImgs.size() != exposureTime.size()) + return false; + + int width, height = 0; + for(size_t i = 0; i < inputImgs.size(); ++ i) + { + Mat img; + img = inputImgs[i]; + if(img.type() != CV_8U) + { + cout<<"The type of the image should be CV_8U!"< +#include "opencv2/imgproc.hpp" +#include "opencv2/highgui.hpp" +#include "opencv2/video/tracking.hpp" +#include +#endif From ef6cebeb987ca1bf28686b4c71b688bc5e32dab5 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 12 Jun 2017 09:16:50 +0200 Subject: [PATCH 02/53] Add initialization of variables and delete empty line. --- modules/pcalib/src/pcalib.cpp | 5 ++--- modules/pcalib/src/precomp.hpp | 1 - 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/modules/pcalib/src/pcalib.cpp b/modules/pcalib/src/pcalib.cpp index 21205850d..3fdfd03f6 100644 --- a/modules/pcalib/src/pcalib.cpp +++ b/modules/pcalib/src/pcalib.cpp @@ -60,7 +60,7 @@ bool PhotometricCalibrator::validImgs(std::vector &inputImgs, std::vector< if(inputImgs.empty() || exposureTime.empty() || inputImgs.size() != exposureTime.size()) return false; - int width, height = 0; + int width = 0, height = 0; for(size_t i = 0; i < inputImgs.size(); ++ i) { Mat img; @@ -89,5 +89,4 @@ bool PhotometricCalibrator::validImgs(std::vector &inputImgs, std::vector< }} // namespace pcalib, cv #endif // __OPENCV_PCALIB_CPP__ -#endif // cplusplus - +#endif // cplusplus \ No newline at end of file diff --git a/modules/pcalib/src/precomp.hpp b/modules/pcalib/src/precomp.hpp index 9cc6a060d..d30df22e2 100644 --- a/modules/pcalib/src/precomp.hpp +++ b/modules/pcalib/src/precomp.hpp @@ -46,6 +46,5 @@ #include #include "opencv2/imgproc.hpp" #include "opencv2/highgui.hpp" -#include "opencv2/video/tracking.hpp" #include #endif From d4d89c67cf8983eec76f73298c2f4af5768fa0ee Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Tue, 20 Jun 2017 11:55:39 +0200 Subject: [PATCH 03/53] add Reader class to read the imgaes used for photometric calibration --- modules/pcalib/include/opencv2/pcalib.hpp | 3 + .../pcalib/include/opencv2/pcalib/Reader.hpp | 33 ++++++++ modules/pcalib/src/Reader.cpp | 81 +++++++++++++++++++ modules/pcalib/src/pcalib.cpp | 5 -- 4 files changed, 117 insertions(+), 5 deletions(-) create mode 100644 modules/pcalib/include/opencv2/pcalib/Reader.hpp create mode 100644 modules/pcalib/src/Reader.cpp diff --git a/modules/pcalib/include/opencv2/pcalib.hpp b/modules/pcalib/include/opencv2/pcalib.hpp index d37787a85..43b792180 100644 --- a/modules/pcalib/include/opencv2/pcalib.hpp +++ b/modules/pcalib/include/opencv2/pcalib.hpp @@ -47,6 +47,9 @@ #include #include +#include +#include +#include namespace cv{ namespace pcalib{ diff --git a/modules/pcalib/include/opencv2/pcalib/Reader.hpp b/modules/pcalib/include/opencv2/pcalib/Reader.hpp new file mode 100644 index 000000000..4324a5c71 --- /dev/null +++ b/modules/pcalib/include/opencv2/pcalib/Reader.hpp @@ -0,0 +1,33 @@ +#ifndef _OPENCV_READER_HPP +#define _OPENCV_READER_HPP + +#include "opencv2/pcalib.hpp" + +namespace cv { namespace pcalib{ + +class Reader +{ +public: + Reader(std::string folderPath, std::string timesPath); + + unsigned long getNumImages(); + + double getTimestamp(unsigned long id); + + float getExposureTime(unsigned long id); + + +private: + inline void loadTimestamps(std::string timesFile); + + std::vector files; + std::vector timeStamps; + std::vector exposureTimes; + + int width, height; + + String path; +}; + +}} // namespace cv pcalib +#endif //_OPENCV_READER_HPP diff --git a/modules/pcalib/src/Reader.cpp b/modules/pcalib/src/Reader.cpp new file mode 100644 index 000000000..846c92e3a --- /dev/null +++ b/modules/pcalib/src/Reader.cpp @@ -0,0 +1,81 @@ +// +// Created by 杨楠 on 17/6/14. +// + +#include "opencv2/pcalib/Reader.hpp" +#include "precomp.hpp" + +namespace cv{ namespace pcalib{ + +unsigned long Reader::getNumImages() +{ + return (unsigned long)files.size(); +} + +void Reader::loadTimestamps(std::string timesFile) +{ + std::ifstream timesStream; + timesStream.open(timesFile.c_str()); + timeStamps.clear(); + exposureTimes.clear(); + while (!timesStream.eof() && timesStream.good()) + { + char buf[1000]; + timesStream.getline(buf, 1000); + + int id = 0; + double timeStamp = 0.0; + float exposureTime = 0.0; + + CV_Assert(3 == scanf(buf, "%d %lf %f", &id, &timeStamp, &exposureTime)); + + timeStamps.push_back(timeStamp); + exposureTimes.push_back(exposureTime); + } + timesStream.close(); + + CV_Assert(timeStamps.size() == getNumImages() && exposureTimes.size() == getNumImages()); +} + +Reader::Reader(std::string folderPath, std::string timesPath) +{ + String cvFolderPath(folderPath); + glob(cvFolderPath, files); + CV_Assert(files.size() > 0); + std::sort(files.begin(), files.end()); + loadTimestamps(timesPath); + + width = 0; + height = 0; + + for(unsigned long i = 0; i < files.size(); ++i) + { + Mat img = imread(files[i]); + CV_Assert(img.type() == CV_8U); + if(0 == i) + { + width = img.cols; + height = img.rows; + } + else + { + CV_Assert(width == img.cols && height == img.rows); + } + } + + std::cout< - -#include -#include - namespace cv{ namespace pcalib{ using namespace std; From e42de3a44e9568c1b754a930687404904ae408fe Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 21 Jun 2017 09:36:18 +0200 Subject: [PATCH 04/53] fix quotes, space and guards: https://github.com/opencv/opencv_contrib/pull/1219 commented by @grace- --- modules/pcalib/include/opencv2/pcalib.hpp | 7 +++---- modules/pcalib/src/Reader.cpp | 4 ++-- modules/pcalib/src/pcalib.cpp | 9 +-------- modules/pcalib/src/precomp.hpp | 2 +- 4 files changed, 7 insertions(+), 15 deletions(-) diff --git a/modules/pcalib/include/opencv2/pcalib.hpp b/modules/pcalib/include/opencv2/pcalib.hpp index 43b792180..3ae385771 100644 --- a/modules/pcalib/include/opencv2/pcalib.hpp +++ b/modules/pcalib/include/opencv2/pcalib.hpp @@ -42,16 +42,15 @@ #ifndef __OPENCV_PCALIB_HPP__ #define __OPENCV_PCALIB_HPP__ -#include -#include -#include +#include "opencv2/core.hpp" +#include "opencv2/imgproc.hpp" #include #include #include #include -namespace cv{ namespace pcalib{ +namespace cv { namespace pcalib{ //! @addtogroup pcalib //! @{ diff --git a/modules/pcalib/src/Reader.cpp b/modules/pcalib/src/Reader.cpp index 846c92e3a..1c1ec26bf 100644 --- a/modules/pcalib/src/Reader.cpp +++ b/modules/pcalib/src/Reader.cpp @@ -2,10 +2,10 @@ // Created by 杨楠 on 17/6/14. // -#include "opencv2/pcalib/Reader.hpp" #include "precomp.hpp" +#include "opencv2/pcalib/Reader.hpp" -namespace cv{ namespace pcalib{ +namespace cv { namespace pcalib{ unsigned long Reader::getNumImages() { diff --git a/modules/pcalib/src/pcalib.cpp b/modules/pcalib/src/pcalib.cpp index 4034744d4..c3a50cfe5 100644 --- a/modules/pcalib/src/pcalib.cpp +++ b/modules/pcalib/src/pcalib.cpp @@ -39,10 +39,6 @@ // //M*/ -#ifndef __OPENCV_PCALIB_CPP__ -#define __OPENCV_PCALIB_CPP__ -#ifdef __cplusplus - #include "precomp.hpp" #include "opencv2/pcalib.hpp" @@ -81,7 +77,4 @@ bool PhotometricCalibrator::validImgs(std::vector &inputImgs, std::vector< return true; } -}} // namespace pcalib, cv - -#endif // __OPENCV_PCALIB_CPP__ -#endif // cplusplus \ No newline at end of file +}} // namespace pcalib, cv \ No newline at end of file diff --git a/modules/pcalib/src/precomp.hpp b/modules/pcalib/src/precomp.hpp index d30df22e2..b30949b3d 100644 --- a/modules/pcalib/src/precomp.hpp +++ b/modules/pcalib/src/precomp.hpp @@ -43,7 +43,7 @@ #ifndef __OPENCV_PRECOMP_H__ #define __OPENCV_PRECOMP_H__ -#include +#include "opencv2/core.hpp" #include "opencv2/imgproc.hpp" #include "opencv2/highgui.hpp" #include From a8cb141e9f64560c9aef7bf39181b96d81b81f1a Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 21 Jun 2017 10:01:43 +0200 Subject: [PATCH 05/53] change the module name to photometric_calib --- modules/pcalib/CMakeLists.txt | 2 -- modules/photometric_calib/CMakeLists.txt | 2 ++ modules/{pcalib => photometric_calib}/README.md | 0 .../include/opencv2/photometric_calib.hpp} | 12 ++++++------ .../include/opencv2/photometric_calib}/Reader.hpp | 8 ++++---- modules/{pcalib => photometric_calib}/src/Reader.cpp | 8 ++++---- .../src/photometric_calib.cpp} | 6 +++--- .../{pcalib => photometric_calib}/src/precomp.hpp | 2 +- 8 files changed, 20 insertions(+), 20 deletions(-) delete mode 100644 modules/pcalib/CMakeLists.txt create mode 100644 modules/photometric_calib/CMakeLists.txt rename modules/{pcalib => photometric_calib}/README.md (100%) rename modules/{pcalib/include/opencv2/pcalib.hpp => photometric_calib/include/opencv2/photometric_calib.hpp} (92%) rename modules/{pcalib/include/opencv2/pcalib => photometric_calib/include/opencv2/photometric_calib}/Reader.hpp (76%) rename modules/{pcalib => photometric_calib}/src/Reader.cpp (91%) rename modules/{pcalib/src/pcalib.cpp => photometric_calib/src/photometric_calib.cpp} (96%) rename modules/{pcalib => photometric_calib}/src/precomp.hpp (99%) diff --git a/modules/pcalib/CMakeLists.txt b/modules/pcalib/CMakeLists.txt deleted file mode 100644 index e7cb2450d..000000000 --- a/modules/pcalib/CMakeLists.txt +++ /dev/null @@ -1,2 +0,0 @@ -set(the_description "Photometric Calibration") -ocv_define_module(pcalib opencv_core opencv_imgproc opencv_calib3d opencv_features2d opencv_highgui WRAP python) diff --git a/modules/photometric_calib/CMakeLists.txt b/modules/photometric_calib/CMakeLists.txt new file mode 100644 index 000000000..951c173d5 --- /dev/null +++ b/modules/photometric_calib/CMakeLists.txt @@ -0,0 +1,2 @@ +set(the_description "Photometric Calibration") +ocv_define_module(photometric_calib opencv_core opencv_imgproc opencv_calib3d opencv_features2d opencv_highgui WRAP python) \ No newline at end of file diff --git a/modules/pcalib/README.md b/modules/photometric_calib/README.md similarity index 100% rename from modules/pcalib/README.md rename to modules/photometric_calib/README.md diff --git a/modules/pcalib/include/opencv2/pcalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp similarity index 92% rename from modules/pcalib/include/opencv2/pcalib.hpp rename to modules/photometric_calib/include/opencv2/photometric_calib.hpp index 3ae385771..291639f2e 100644 --- a/modules/pcalib/include/opencv2/pcalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -39,8 +39,8 @@ // //M*/ -#ifndef __OPENCV_PCALIB_HPP__ -#define __OPENCV_PCALIB_HPP__ +#ifndef __OPENCV_PHOTOMETRIC_CALIB_HPP__ +#define __OPENCV_PHOTOMETRIC_CALIB_HPP__ #include "opencv2/core.hpp" #include "opencv2/imgproc.hpp" @@ -50,9 +50,9 @@ #include #include -namespace cv { namespace pcalib{ +namespace cv { namespace photometric_calib{ -//! @addtogroup pcalib +//! @addtogroup photometric_calib //! @{ class CV_EXPORTS PhotometricCalibrator : public Algorithm @@ -63,6 +63,6 @@ public: //! @} -}} // namespace pcalib, cv +}} // namespace photometric_calib, cv -#endif +#endif \ No newline at end of file diff --git a/modules/pcalib/include/opencv2/pcalib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp similarity index 76% rename from modules/pcalib/include/opencv2/pcalib/Reader.hpp rename to modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 4324a5c71..a573134e3 100644 --- a/modules/pcalib/include/opencv2/pcalib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -1,9 +1,9 @@ #ifndef _OPENCV_READER_HPP #define _OPENCV_READER_HPP -#include "opencv2/pcalib.hpp" +#include "opencv2/photometric_calib.hpp" -namespace cv { namespace pcalib{ +namespace cv { namespace photometric_calib{ class Reader { @@ -29,5 +29,5 @@ private: String path; }; -}} // namespace cv pcalib -#endif //_OPENCV_READER_HPP +}} // namespace cv photometric_calib +#endif //_OPENCV_READER_HPP \ No newline at end of file diff --git a/modules/pcalib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp similarity index 91% rename from modules/pcalib/src/Reader.cpp rename to modules/photometric_calib/src/Reader.cpp index 1c1ec26bf..55a12a633 100644 --- a/modules/pcalib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -1,11 +1,11 @@ // -// Created by 杨楠 on 17/6/14. +// Created by Nan Yang on 17/6/14. // #include "precomp.hpp" -#include "opencv2/pcalib/Reader.hpp" +#include "opencv2/photometric_calib/Reader.hpp" -namespace cv { namespace pcalib{ +namespace cv { namespace photometric_calib{ unsigned long Reader::getNumImages() { @@ -78,4 +78,4 @@ float Reader::getExposureTime(unsigned long id) return exposureTimes[id]; } -}} \ No newline at end of file +}} // namespace photometric_calib, cv \ No newline at end of file diff --git a/modules/pcalib/src/pcalib.cpp b/modules/photometric_calib/src/photometric_calib.cpp similarity index 96% rename from modules/pcalib/src/pcalib.cpp rename to modules/photometric_calib/src/photometric_calib.cpp index c3a50cfe5..1ba0297bb 100644 --- a/modules/pcalib/src/pcalib.cpp +++ b/modules/photometric_calib/src/photometric_calib.cpp @@ -40,9 +40,9 @@ //M*/ #include "precomp.hpp" -#include "opencv2/pcalib.hpp" +#include "opencv2/photometric_calib.hpp" -namespace cv{ namespace pcalib{ +namespace cv{ namespace photometric_calib{ using namespace std; @@ -77,4 +77,4 @@ bool PhotometricCalibrator::validImgs(std::vector &inputImgs, std::vector< return true; } -}} // namespace pcalib, cv \ No newline at end of file +}} // namespace photometric_calib, cv \ No newline at end of file diff --git a/modules/pcalib/src/precomp.hpp b/modules/photometric_calib/src/precomp.hpp similarity index 99% rename from modules/pcalib/src/precomp.hpp rename to modules/photometric_calib/src/precomp.hpp index b30949b3d..0ecb09d66 100644 --- a/modules/pcalib/src/precomp.hpp +++ b/modules/photometric_calib/src/precomp.hpp @@ -47,4 +47,4 @@ #include "opencv2/imgproc.hpp" #include "opencv2/highgui.hpp" #include -#endif +#endif \ No newline at end of file From a3cb3d9462265b7c51a157ad6b6381425c651fc4 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 26 Jun 2017 21:21:35 +0200 Subject: [PATCH 06/53] Add brief README.md --- modules/photometric_calib/README.md | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/modules/photometric_calib/README.md b/modules/photometric_calib/README.md index 3d7db7867..ba725c2b6 100644 --- a/modules/photometric_calib/README.md +++ b/modules/photometric_calib/README.md @@ -1,2 +1,8 @@ Photometric Calibration ================================================ + +Implementation of non-parametric photometric calibration algorithm proposed by J. Engel et al.: + +1. Camera Response Function Calibration +2. Vignette Calibration +3. Photometric Distortion Remover \ No newline at end of file From b34ec27d250704338d4200ca9d7c89a11900ac63 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 26 Jun 2017 21:21:58 +0200 Subject: [PATCH 07/53] Add .bib file. --- modules/photometric_calib/doc/photometric_calib.bib | 9 +++++++++ 1 file changed, 9 insertions(+) create mode 100644 modules/photometric_calib/doc/photometric_calib.bib diff --git a/modules/photometric_calib/doc/photometric_calib.bib b/modules/photometric_calib/doc/photometric_calib.bib new file mode 100644 index 000000000..044d3ddca --- /dev/null +++ b/modules/photometric_calib/doc/photometric_calib.bib @@ -0,0 +1,9 @@ +@InProceedings{engel2016monodataset, + author = "J. Engel and V. Usenko and D. Cremers", + title = "A Photometrically Calibrated Benchmark For Monocular Visual Odometry", + booktitle = "arXiv:1607.02555", + arXiv = " arXiv:1607.02555", + year = "2016", + month = "July", + keywords={mono-ds,dso} +} \ No newline at end of file From 5ef6a788693065f930cd8b785d3902f8e230f486 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 26 Jun 2017 21:26:13 +0200 Subject: [PATCH 08/53] Fixed problems commented by @vrabaud --- .../include/opencv2/photometric_calib.hpp | 48 +++-------------- .../opencv2/photometric_calib/Reader.hpp | 19 +++++-- modules/photometric_calib/src/Reader.cpp | 54 ++++++++++--------- .../src/photometric_calib.cpp | 45 ++-------------- modules/photometric_calib/src/precomp.hpp | 44 ++------------- 5 files changed, 56 insertions(+), 154 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp index 291639f2e..7b748c276 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -1,43 +1,6 @@ -/*M/////////////////////////////////////////////////////////////////////////////////////// - // - // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. - // - // By downloading, copying, installing or using the software you agree to this license. - // If you do not agree to this license, do not download, install, - // copy or use the software. - // - // - // License Agreement - // For Open Source Computer Vision Library - // - // Copyright (C) 2014, OpenCV Foundation, all rights reserved. - // Third party copyrights are property of their respective owners. - // - // Redistribution and use in source and binary forms, with or without modification, - // are permitted provided that the following conditions are met: - // - // * Redistribution's of source code must retain the above copyright notice, - // this list of conditions and the following disclaimer. - // - // * Redistribution's in binary form must reproduce the above copyright notice, - // this list of conditions and the following disclaimer in the documentation - // and/or other materials provided with the distribution. - // - // * The name of the copyright holders may not be used to endorse or promote products - // derived from this software without specific prior written permission. - // - // This software is provided by the copyright holders and contributors "as is" and - // any express or implied warranties, including, but not limited to, the implied - // warranties of merchantability and fitness for a particular purpose are disclaimed. - // In no event shall the Intel Corporation or contributors be liable for any direct, - // indirect, incidental, special, exemplary, or consequential damages - // (including, but not limited to, procurement of substitute goods or services; - // loss of use, data, or profits; or business interruption) however caused - // and on any theory of liability, whether in contract, strict liability, - // or tort (including negligence or otherwise) arising in any way out of - // the use of this software, even if advised of the possibility of such damage. - // - //M*/ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. #ifndef __OPENCV_PHOTOMETRIC_CALIB_HPP__ #define __OPENCV_PHOTOMETRIC_CALIB_HPP__ @@ -50,6 +13,9 @@ #include #include +/** @defgroup photometric_calib Photometric Calibration +*/ + namespace cv { namespace photometric_calib{ //! @addtogroup photometric_calib @@ -58,7 +24,7 @@ namespace cv { namespace photometric_calib{ class CV_EXPORTS PhotometricCalibrator : public Algorithm { public: - bool validImgs(std::vector &inputImgs, std::vector &exposureTime); + bool validImgs(const std::vector &inputImgs, const std::vector &exposureTime); }; //! @} diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index a573134e3..9b31dfc6b 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -1,3 +1,7 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + #ifndef _OPENCV_READER_HPP #define _OPENCV_READER_HPP @@ -5,20 +9,23 @@ namespace cv { namespace photometric_calib{ +//! @addtogroup photometric_calib +//! @{ + class Reader { public: - Reader(std::string folderPath, std::string timesPath); + Reader(const std::string &folderPath, const std::string ×Path); - unsigned long getNumImages(); + unsigned long getNumImages() const; - double getTimestamp(unsigned long id); + double getTimestamp(unsigned long id) const; - float getExposureTime(unsigned long id); + float getExposureTime(unsigned long id) const; private: - inline void loadTimestamps(std::string timesFile); + inline void loadTimestamps(const std::string ×File); std::vector files; std::vector timeStamps; @@ -29,5 +36,7 @@ private: String path; }; +//! @} + }} // namespace cv photometric_calib #endif //_OPENCV_READER_HPP \ No newline at end of file diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 55a12a633..53f2df44d 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -1,43 +1,47 @@ -// -// Created by Nan Yang on 17/6/14. -// +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. #include "precomp.hpp" #include "opencv2/photometric_calib/Reader.hpp" namespace cv { namespace photometric_calib{ -unsigned long Reader::getNumImages() +unsigned long Reader::getNumImages() const { return (unsigned long)files.size(); } -void Reader::loadTimestamps(std::string timesFile) +void Reader::loadTimestamps(const std::string ×File) { - std::ifstream timesStream; - timesStream.open(timesFile.c_str()); + CV_Assert(timesFile.substr(timesFile.find_last_of(".") + 1) == "yaml" || timesFile.substr(timesFile.find_last_of(".") + 1) == "yml"); + + FileStorage timeFile; + timeFile.open(timesFile, FileStorage::READ); timeStamps.clear(); exposureTimes.clear(); - while (!timesStream.eof() && timesStream.good()) - { - char buf[1000]; - timesStream.getline(buf, 1000); - int id = 0; - double timeStamp = 0.0; - float exposureTime = 0.0; + CV_Assert(timeFile.isOpened()); - CV_Assert(3 == scanf(buf, "%d %lf %f", &id, &timeStamp, &exposureTime)); + FileNode timeStampNode = timeFile["times"]; + FileNode exposureTimeNode = timeFile["exposures"]; - timeStamps.push_back(timeStamp); - exposureTimes.push_back(exposureTime); - } - timesStream.close(); + CV_Assert(timeStampNode.type() == FileNode::SEQ && exposureTimeNode.type() == FileNode::SEQ); + + FileNodeIterator itTs = timeStampNode.begin(), itTsEnd = timeStampNode.end(); + FileNodeIterator itEt = exposureTimeNode.begin(), itEtEnd = exposureTimeNode.end(); + + for (; itTs != itTsEnd; ++itTs) + timeStamps.push_back((double)*itTs); + for (; itEt != itEtEnd; ++itEt) + exposureTimes.push_back((float)*itEt); + + timeFile.release(); CV_Assert(timeStamps.size() == getNumImages() && exposureTimes.size() == getNumImages()); } -Reader::Reader(std::string folderPath, std::string timesPath) +Reader::Reader(const std::string &folderPath, const std::string ×Path) { String cvFolderPath(folderPath); glob(cvFolderPath, files); @@ -48,11 +52,11 @@ Reader::Reader(std::string folderPath, std::string timesPath) width = 0; height = 0; - for(unsigned long i = 0; i < files.size(); ++i) + for(size_t i = 0; i < files.size(); ++i) { Mat img = imread(files[i]); CV_Assert(img.type() == CV_8U); - if(0 == i) + if(i == 0) { width = img.cols; height = img.rows; @@ -62,17 +66,15 @@ Reader::Reader(std::string folderPath, std::string timesPath) CV_Assert(width == img.cols && height == img.rows); } } - - std::cout< &inputImgs, std::vector &exposureTime) +bool PhotometricCalibrator::validImgs(const std::vector &inputImgs, const std::vector &exposureTime) { if(inputImgs.empty() || exposureTime.empty() || inputImgs.size() != exposureTime.size()) return false; diff --git a/modules/photometric_calib/src/precomp.hpp b/modules/photometric_calib/src/precomp.hpp index 0ecb09d66..5d7c2b7dc 100644 --- a/modules/photometric_calib/src/precomp.hpp +++ b/modules/photometric_calib/src/precomp.hpp @@ -1,44 +1,6 @@ -/*M/////////////////////////////////////////////////////////////////////////////////////// -// -// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING. -// -// By downloading, copying, installing or using the software you agree to this license. -// If you do not agree to this license, do not download, install, -// copy or use the software. -// -// -// License Agreement -// For Open Source Computer Vision Library -// -// Copyright (C) 2000-2008, Intel Corporation, all rights reserved. -// Copyright (C) 2009, Willow Garage Inc., all rights reserved. -// Third party copyrights are property of their respective owners. -// -// Redistribution and use in source and binary forms, with or without modification, -// are permitted provided that the following conditions are met: -// -// * Redistribution's of source code must retain the above copyright notice, -// this list of conditions and the following disclaimer. -// -// * Redistribution's in binary form must reproduce the above copyright notice, -// this list of conditions and the following disclaimer in the documentation -// and/or other materials provided with the distribution. -// -// * The name of the copyright holders may not be used to endorse or promote products -// derived from this software without specific prior written permission. -// -// This software is provided by the copyright holders and contributors "as is" and -// any express or implied warranties, including, but not limited to, the implied -// warranties of merchantability and fitness for a particular purpose are disclaimed. -// In no event shall the Intel Corporation or contributors be liable for any direct, -// indirect, incidental, special, exemplary, or consequential damages -// (including, but not limited to, procurement of substitute goods or services; -// loss of use, data, or profits; or business interruption) however caused -// and on any theory of liability, whether in contract, strict liability, -// or tort (including negligence or otherwise) arising in any way out of -// the use of this software, even if advised of the possibility of such damage. -// -//M*/ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. #ifndef __OPENCV_PRECOMP_H__ #define __OPENCV_PRECOMP_H__ From 1a94d7ed5c7bf1be63ad24958f8905eeebe2ea4b Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Thu, 29 Jun 2017 13:37:38 +0200 Subject: [PATCH 09/53] change exposureTime to exposureDuration. --- .../include/opencv2/photometric_calib/Reader.hpp | 4 ++-- modules/photometric_calib/src/Reader.cpp | 13 +++++++------ 2 files changed, 9 insertions(+), 8 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 9b31dfc6b..63636f9ab 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -21,7 +21,7 @@ public: double getTimestamp(unsigned long id) const; - float getExposureTime(unsigned long id) const; + float getExposureDuration(unsigned long id) const; private: @@ -29,7 +29,7 @@ private: std::vector files; std::vector timeStamps; - std::vector exposureTimes; + std::vector exposureDurations;//All the exposure duration for images int width, height; diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 53f2df44d..09063e483 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -19,7 +19,7 @@ void Reader::loadTimestamps(const std::string ×File) FileStorage timeFile; timeFile.open(timesFile, FileStorage::READ); timeStamps.clear(); - exposureTimes.clear(); + exposureDurations.clear(); CV_Assert(timeFile.isOpened()); @@ -34,11 +34,11 @@ void Reader::loadTimestamps(const std::string ×File) for (; itTs != itTsEnd; ++itTs) timeStamps.push_back((double)*itTs); for (; itEt != itEtEnd; ++itEt) - exposureTimes.push_back((float)*itEt); + exposureDurations.push_back((float)*itEt); timeFile.release(); - CV_Assert(timeStamps.size() == getNumImages() && exposureTimes.size() == getNumImages()); + CV_Assert(timeStamps.size() == getNumImages() && exposureDurations.size() == getNumImages()); } Reader::Reader(const std::string &folderPath, const std::string ×Path) @@ -74,10 +74,11 @@ double Reader::getTimestamp(unsigned long id) const return timeStamps[id]; } -float Reader::getExposureTime(unsigned long id) const +float Reader::getExposureDuration(unsigned long id) const { - CV_Assert(id < exposureTimes.size()); - return exposureTimes[id]; + CV_Assert(id < exposureDurations.size()); + return exposureDurations[id]; +} } }} // namespace photometric_calib, cv \ No newline at end of file From 0456657e911658403496a654174d8e1ae2c7d0df Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Fri, 30 Jun 2017 13:00:58 +0200 Subject: [PATCH 10/53] Delete useless included header files --- .../photometric_calib/include/opencv2/photometric_calib.hpp | 3 --- .../include/opencv2/photometric_calib/Reader.hpp | 3 +++ modules/photometric_calib/src/photometric_calib.cpp | 1 + 3 files changed, 4 insertions(+), 3 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp index 7b748c276..b45cfff13 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -9,9 +9,6 @@ #include "opencv2/imgproc.hpp" #include -#include -#include -#include /** @defgroup photometric_calib Photometric Calibration */ diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 63636f9ab..d432639a3 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -7,6 +7,9 @@ #include "opencv2/photometric_calib.hpp" +#include +#include + namespace cv { namespace photometric_calib{ //! @addtogroup photometric_calib diff --git a/modules/photometric_calib/src/photometric_calib.cpp b/modules/photometric_calib/src/photometric_calib.cpp index 3595cd1db..a83f66a97 100644 --- a/modules/photometric_calib/src/photometric_calib.cpp +++ b/modules/photometric_calib/src/photometric_calib.cpp @@ -4,6 +4,7 @@ #include "precomp.hpp" #include "opencv2/photometric_calib.hpp" +#include namespace cv{ namespace photometric_calib{ From de6d4a83c2c81b49e53d688eaa269cc61b409dae Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Fri, 30 Jun 2017 13:13:39 +0200 Subject: [PATCH 11/53] changed some properties names and added some getters and setters --- .../opencv2/photometric_calib/Reader.hpp | 11 ++++-- modules/photometric_calib/src/Reader.cpp | 37 +++++++++++++------ 2 files changed, 33 insertions(+), 15 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index d432639a3..ea2765386 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -26,17 +26,20 @@ public: float getExposureDuration(unsigned long id) const; + int getWidth() const; + int getHeight() const; + const std::string &getFolderPath() const; private: inline void loadTimestamps(const std::string ×File); - std::vector files; - std::vector timeStamps; + std::vector images; //All the names/paths of images + std::vector timeStamps; //All the Unix Time Stamps of images std::vector exposureDurations;//All the exposure duration for images - int width, height; + int _width, _height;//The image width and height. All the images should be of the same size. - String path; + std::string _folderPath; }; //! @} diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 09063e483..bdee14b2a 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -9,11 +9,12 @@ namespace cv { namespace photometric_calib{ unsigned long Reader::getNumImages() const { - return (unsigned long)files.size(); + return (unsigned long)images.size(); } void Reader::loadTimestamps(const std::string ×File) { + // check the extension of the time file. CV_Assert(timesFile.substr(timesFile.find_last_of(".") + 1) == "yaml" || timesFile.substr(timesFile.find_last_of(".") + 1) == "yml"); FileStorage timeFile; @@ -44,28 +45,31 @@ void Reader::loadTimestamps(const std::string ×File) Reader::Reader(const std::string &folderPath, const std::string ×Path) { String cvFolderPath(folderPath); - glob(cvFolderPath, files); - CV_Assert(files.size() > 0); - std::sort(files.begin(), files.end()); + glob(cvFolderPath, images); + CV_Assert(images.size() > 0); + std::sort(images.begin(), images.end()); loadTimestamps(timesPath); - width = 0; - height = 0; + _width = 0; + _height = 0; - for(size_t i = 0; i < files.size(); ++i) + // images should be of CV_8U and same size + for(size_t i = 0; i < images.size(); ++i) { - Mat img = imread(files[i]); + Mat img = imread(images[i]); CV_Assert(img.type() == CV_8U); if(i == 0) { - width = img.cols; - height = img.rows; + _width = img.cols; + _height = img.rows; } else { - CV_Assert(width == img.cols && height == img.rows); + CV_Assert(_width == img.cols && _height == img.rows); } } + + _folderPath = folderPath; } double Reader::getTimestamp(unsigned long id) const @@ -79,6 +83,17 @@ float Reader::getExposureDuration(unsigned long id) const CV_Assert(id < exposureDurations.size()); return exposureDurations[id]; } + +int Reader::getWidth() const { + return _width; +} + +int Reader::getHeight() const { + return _height; +} + +const std::string &Reader::getFolderPath() const { + return _folderPath; } }} // namespace photometric_calib, cv \ No newline at end of file From 616c25ef6caa3b6250bd73729f7cb0024715661d Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Fri, 30 Jun 2017 13:21:48 +0200 Subject: [PATCH 12/53] add Doxygen documentation --- .../opencv2/photometric_calib/Reader.hpp | 34 +++++++++++++++++++ 1 file changed, 34 insertions(+) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index ea2765386..fc8d879a2 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -15,15 +15,45 @@ namespace cv { namespace photometric_calib{ //! @addtogroup photometric_calib //! @{ +/*! + * @brief Class for reading the sequence used for photometric calibration. Both the folder path of the sequence + * and the path of time file should be provided. The images of the sequence should be of format CV_8U. The time + * file should be .yaml or .yml. In the time file, the timestamps and exposure duration of the corresponding images + * of the sequence should be provided. + * + * The image paths are stored in std::vector images, timestamps are stored in std::vector timeStamps, + * exposure duration is stored in std::vector exposureTimes + */ + class Reader { public: + /*! + * @brief Constructor + * @param folderPath the path of folder which contains the images + * @param timesPath the path of time file + */ Reader(const std::string &folderPath, const std::string ×Path); + /*! + * @return the amount of images loaded + */ unsigned long getNumImages() const; + + /*! + * @brief Given the id of the image and return its timestamp value. id is in fact just the index of the image in the + * vector contains all the images. + * @param id + * @return timestamp of the id^th image. + */ double getTimestamp(unsigned long id) const; + /*! + * @brief Given the id of the image and return its exposure duration when is was taken. + * @param id + * @return exposure duration of the image. + */ float getExposureDuration(unsigned long id) const; int getWidth() const; @@ -31,6 +61,10 @@ public: const std::string &getFolderPath() const; private: + /*! + * @brief Load timestamps and exposure duration. + * @param timesFile + */ inline void loadTimestamps(const std::string ×File); std::vector images; //All the names/paths of images From b933912a879e5b265fd4ab253e05f7e3875025ce Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Fri, 30 Jun 2017 18:39:33 +0200 Subject: [PATCH 13/53] Add CV_EXPORTS to Reader class. --- .../include/opencv2/photometric_calib/Reader.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index fc8d879a2..d0d7bcb5b 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -25,7 +25,7 @@ namespace cv { namespace photometric_calib{ * exposure duration is stored in std::vector exposureTimes */ -class Reader +class CV_EXPORTS Reader { public: /*! From b1ee8581ef930d856737e051381e401cfa9279b0 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 3 Jul 2017 11:12:06 +0200 Subject: [PATCH 14/53] Read image with IMREAD_GRAYSCALE and add getImage() method --- .../include/opencv2/photometric_calib/Reader.hpp | 8 ++++++++ modules/photometric_calib/src/Reader.cpp | 8 +++++++- 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index d0d7bcb5b..081629178 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -41,6 +41,14 @@ public: unsigned long getNumImages() const; + /*! + * @brief Given the id of the image and return the image. id is in fact just the index of the image in the + * vector contains all the images. + * @param id + * @return Mat of the id^th image. + */ + Mat getImage(unsigned long id) const; + /*! * @brief Given the id of the image and return its timestamp value. id is in fact just the index of the image in the * vector contains all the images. diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index bdee14b2a..36bb98fdf 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -56,7 +56,7 @@ Reader::Reader(const std::string &folderPath, const std::string ×Path) // images should be of CV_8U and same size for(size_t i = 0; i < images.size(); ++i) { - Mat img = imread(images[i]); + Mat img = imread(images[i], IMREAD_GRAYSCALE); CV_Assert(img.type() == CV_8U); if(i == 0) { @@ -72,6 +72,12 @@ Reader::Reader(const std::string &folderPath, const std::string ×Path) _folderPath = folderPath; } +Mat Reader::getImage(unsigned long id) const +{ + CV_Assert(id < images.size()); + return imread(images[id], IMREAD_GRAYSCALE); +} + double Reader::getTimestamp(unsigned long id) const { CV_Assert(id < timeStamps.size()); From 5c04c0295d1fcbdc95e1b74fdcb3d8d0895c9ed6 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 3 Jul 2017 15:31:02 +0200 Subject: [PATCH 15/53] Camera response function(Gamma function) remover. --- .../photometric_calib/GammaRemover.hpp | 41 ++++++++++ .../photometric_calib/src/GammaRemover.cpp | 78 +++++++++++++++++++ 2 files changed, 119 insertions(+) create mode 100644 modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp create mode 100644 modules/photometric_calib/src/GammaRemover.cpp diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp new file mode 100644 index 000000000..be6893cbb --- /dev/null +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -0,0 +1,41 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#ifndef _OPENCV_GAMMAREMOVER_HPP +#define _OPENCV_GAMMAREMOVER_HPP + +#include "opencv2/photometric_calib.hpp" + +namespace cv { namespace photometric_calib { + +class CV_EXPORTS GammaRemover +{ +public: + GammaRemover(const std::string &gammaPath, int w_, int h_); + + Mat getUnGammaImageMat(Mat inputIm); + void getUnGammaImageArr(Mat inputIm, std::vector &outImVec); + + inline float* getG() + { + if(!validGamma) return nullptr; + else return G; + }; + inline float* getGInv() + { + if(!validGamma) return nullptr; + else return GInv; + }; + +private: + float G[256]; + float GInv[256]; + int w,h; + bool validGamma; +}; + + +}} + +#endif //_OPENCV__GAMMAREMOVER_HPP diff --git a/modules/photometric_calib/src/GammaRemover.cpp b/modules/photometric_calib/src/GammaRemover.cpp new file mode 100644 index 000000000..642e86e6f --- /dev/null +++ b/modules/photometric_calib/src/GammaRemover.cpp @@ -0,0 +1,78 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#include "precomp.hpp" +#include "opencv2/photometric_calib/GammaRemover.hpp" + +namespace cv { namespace photometric_calib { + +GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) +{ + validGamma = false; + w = w_; + h = h_; + + // check the extension of the time file. + CV_Assert(gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yaml" || gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yml"); + + FileStorage gammaFile; + gammaFile.open(gammaPath, FileStorage::READ); + CV_Assert(gammaFile.isOpened()); + + FileNode gammaNode = gammaFile["gamma"]; + CV_Assert(gammaNode.type() == FileNode::SEQ); + FileNodeIterator itS = gammaNode.begin(), itE = gammaNode.end(); + std::vector GInvVec; + for (; itS != itE; ++itS) + GInvVec.push_back((float)*itS); + CV_Assert(GInvVec.size() == 256); + + for(int i=0;i<256;i++) GInv[i] = GInvVec[i]; + for(int i=0;i<255;i++) + { + CV_Assert(GInv[i+1] > GInv[i]); + } + float min = GInv[0]; + float max = GInv[255]; + for(int i=0;i<256;i++) GInv[i] = 255.0 * (GInv[i] - min) / (max-min); + for(int i=1;i<255;i++) + { + for(int s=1;s<255;s++) + { + if(GInv[s] <= i && GInv[s+1] >= i) + { + G[i] = s+(i - GInv[s]) / (GInv[s+1]-GInv[s]); + break; + } + } + } + G[0] = 0; + G[255] = 255; + gammaFile.release(); + validGamma=true; +} + +Mat GammaRemover::getUnGammaImageMat(Mat inputIm) +{ + CV_Assert(validGamma); + uchar *inputImArr = inputIm.data; + float outImArr[w*h]; + for (int i = 0; i < w * h; ++i) + { + outImArr[i] = GInv[inputImArr[i]]; + } + Mat _outIm(h, w, CV_32F, outImArr); + Mat outIm = _outIm * (1/255.0f); + return outIm; +} + +void GammaRemover::getUnGammaImageArr(Mat inputIm, std::vector &outImVec) +{ + CV_Assert(validGamma); + uchar *inputImArr = inputIm.data; + CV_Assert(outImVec.size() == (unsigned long)w * h); + for(int i = 0; i < w * h; i++) outImVec[i] = GInv[inputImArr[i]]; +} + +}} From bb415db8a9f7badce1d5bc5b4a94d79a48191272 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 3 Jul 2017 15:52:36 +0200 Subject: [PATCH 16/53] fix cast type. --- modules/photometric_calib/src/GammaRemover.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/photometric_calib/src/GammaRemover.cpp b/modules/photometric_calib/src/GammaRemover.cpp index 642e86e6f..d146bde74 100644 --- a/modules/photometric_calib/src/GammaRemover.cpp +++ b/modules/photometric_calib/src/GammaRemover.cpp @@ -35,7 +35,7 @@ GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) } float min = GInv[0]; float max = GInv[255]; - for(int i=0;i<256;i++) GInv[i] = 255.0 * (GInv[i] - min) / (max-min); + for(int i=0;i<256;i++) GInv[i] = (float) (255.0 * (GInv[i] - min) / (max - min)); for(int i=1;i<255;i++) { for(int s=1;s<255;s++) From 80c227dc3fdf5273ce31a50c31bcc9c9dca797cf Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 3 Jul 2017 16:48:34 +0200 Subject: [PATCH 17/53] fix Variable Length Array for MSVS. --- modules/photometric_calib/src/GammaRemover.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/modules/photometric_calib/src/GammaRemover.cpp b/modules/photometric_calib/src/GammaRemover.cpp index d146bde74..a2820eb16 100644 --- a/modules/photometric_calib/src/GammaRemover.cpp +++ b/modules/photometric_calib/src/GammaRemover.cpp @@ -57,13 +57,14 @@ Mat GammaRemover::getUnGammaImageMat(Mat inputIm) { CV_Assert(validGamma); uchar *inputImArr = inputIm.data; - float outImArr[w*h]; + float *outImArr = new float[w*h]; for (int i = 0; i < w * h; ++i) { outImArr[i] = GInv[inputImArr[i]]; } Mat _outIm(h, w, CV_32F, outImArr); Mat outIm = _outIm * (1/255.0f); + delete[] outImArr; return outIm; } From 7c55a71c2688a81b0f65f06ead557933f2996d6d Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Tue, 4 Jul 2017 09:19:34 +0200 Subject: [PATCH 18/53] nullptr -> 0 --- .../include/opencv2/photometric_calib/GammaRemover.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index be6893cbb..1de1188bc 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -19,12 +19,12 @@ public: inline float* getG() { - if(!validGamma) return nullptr; + if(!validGamma) return 0; else return G; }; inline float* getGInv() { - if(!validGamma) return nullptr; + if(!validGamma) return 0; else return GInv; }; From 09b17673c60bc182fa7aca9e335761ab54407ead Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Tue, 4 Jul 2017 19:06:58 +0200 Subject: [PATCH 19/53] Change method name from *Arr to *Vec --- .../include/opencv2/photometric_calib/GammaRemover.hpp | 2 +- modules/photometric_calib/src/GammaRemover.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index 1de1188bc..9429fe8c4 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -15,7 +15,7 @@ public: GammaRemover(const std::string &gammaPath, int w_, int h_); Mat getUnGammaImageMat(Mat inputIm); - void getUnGammaImageArr(Mat inputIm, std::vector &outImVec); + void getUnGammaImageVec(Mat inputIm, std::vector &outImVec); inline float* getG() { diff --git a/modules/photometric_calib/src/GammaRemover.cpp b/modules/photometric_calib/src/GammaRemover.cpp index a2820eb16..b83b921c4 100644 --- a/modules/photometric_calib/src/GammaRemover.cpp +++ b/modules/photometric_calib/src/GammaRemover.cpp @@ -68,7 +68,7 @@ Mat GammaRemover::getUnGammaImageMat(Mat inputIm) return outIm; } -void GammaRemover::getUnGammaImageArr(Mat inputIm, std::vector &outImVec) +void GammaRemover::getUnGammaImageVec(Mat inputIm, std::vector &outImVec) { CV_Assert(validGamma); uchar *inputImArr = inputIm.data; From 5ff47e5465c6853c5b4a123631afc54e366d16cd Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Tue, 4 Jul 2017 19:07:27 +0200 Subject: [PATCH 20/53] Add class VignetteRemover --- .../photometric_calib/VignetteRemover.hpp | 31 +++++++ .../photometric_calib/src/VignetteRemover.cpp | 81 +++++++++++++++++++ 2 files changed, 112 insertions(+) create mode 100644 modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp create mode 100644 modules/photometric_calib/src/VignetteRemover.cpp diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp new file mode 100644 index 000000000..6279262d7 --- /dev/null +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -0,0 +1,31 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#ifndef _OPENCV_VIGNETTEREMOVER_HPP +#define _OPENCV_VIGNETTEREMOVER_HPP + +#include "opencv2/photometric_calib.hpp" + +namespace cv { namespace photometric_calib { + +class CV_EXPORTS VignetteRemover +{ +public: + VignetteRemover(const std::string &vignettePath, int w_, int h_); + ~VignetteRemover(); + + Mat getUnVignetteImageMat(std::vector &unGammaImVec); + void getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec); + +private: + float* vignetteMap; + float* vignetteMapInv; + int w,h; + bool validVignette; +}; + +}} + + +#endif //_OPENCV_VIGNETTEREMOVER_HPP diff --git a/modules/photometric_calib/src/VignetteRemover.cpp b/modules/photometric_calib/src/VignetteRemover.cpp new file mode 100644 index 000000000..f83eb435f --- /dev/null +++ b/modules/photometric_calib/src/VignetteRemover.cpp @@ -0,0 +1,81 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#include "precomp.hpp" +#include "opencv2/photometric_calib/VignetteRemover.hpp" + +namespace cv { namespace photometric_calib{ + +VignetteRemover::VignetteRemover(const std::string &vignettePath, int w_, int h_) +{ + CV_Assert(vignettePath != ""); + + validVignette=false; + vignetteMap=0; + vignetteMapInv=0; + w = w_; + h = h_; + + + Mat vignetteMat = imread(vignettePath, IMREAD_UNCHANGED); + vignetteMap = new float[w * h]; + vignetteMapInv = new float[w * h]; + + CV_Assert(vignetteMat.rows == h && vignetteMat.cols == w); + + CV_Assert(vignetteMat.type() == CV_8U || vignetteMat.type() == CV_16U); + if(vignetteMat.type() == CV_8U) + { + float maxV=0; + for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); + + for(int i=0;i(i) / maxV; + } + + else + { + float maxV=0; + for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); + + for(int i=0;i(i) / maxV; + } + + for(int i=0;i &unGammaImVec) +{ + std::vector _outImVec(w * h); + getUnVignetteImageVec(unGammaImVec, _outImVec); + + Mat _outIm(h, w, CV_32F, &_outImVec[0]); + Mat outIm = _outIm * (1/255.0f); + return outIm; +} + +void VignetteRemover::getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec) +{ + CV_Assert(validVignette); + CV_Assert(outImVec.size() == (unsigned long)w * h); + for (int i = 0; i < w * h; ++i) + outImVec[i] = unGammaImVec[i] * vignetteMapInv[i]; +} + +VignetteRemover::~VignetteRemover() { + if(vignetteMap != 0) + delete[] vignetteMap; + if(vignetteMapInv != 0) + delete[] vignetteMapInv; +} + + +}} From 54d2c931094e61657eb90d90deed2c1526e754ba Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 10 Jul 2017 13:55:40 +0200 Subject: [PATCH 21/53] Specifying image format when reading images from a folder. --- .../include/opencv2/photometric_calib/GammaRemover.hpp | 1 - .../include/opencv2/photometric_calib/Reader.hpp | 3 ++- modules/photometric_calib/src/Reader.cpp | 10 +++++++++- 3 files changed, 11 insertions(+), 3 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index 9429fe8c4..101330db7 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -35,7 +35,6 @@ private: bool validGamma; }; - }} #endif //_OPENCV__GAMMAREMOVER_HPP diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 081629178..38cd2ead0 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -31,9 +31,10 @@ public: /*! * @brief Constructor * @param folderPath the path of folder which contains the images + * @param imageExt the format of the input images, e.g., jpg or png. * @param timesPath the path of time file */ - Reader(const std::string &folderPath, const std::string ×Path); + Reader(const std::string &folderPath, const std::string &imageExt, const std::string ×Path); /*! * @return the amount of images loaded diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 36bb98fdf..379f05990 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -42,9 +42,17 @@ void Reader::loadTimestamps(const std::string ×File) CV_Assert(timeStamps.size() == getNumImages() && exposureDurations.size() == getNumImages()); } -Reader::Reader(const std::string &folderPath, const std::string ×Path) +Reader::Reader(const std::string &folderPath, const std::string &imageExt, const std::string ×Path) { String cvFolderPath(folderPath); + +#if defined WIN32 || defined _WIN32 || defined WINCE + *cvFolderPath.end() == '\\' ? cvFolderPath = cvFolderPath : cvFolderPath += '\\'; +#else + *cvFolderPath.end() == '/' ? cvFolderPath = cvFolderPath : cvFolderPath += '/'; +#endif + + cvFolderPath += ("*." + imageExt); glob(cvFolderPath, images); CV_Assert(images.size() > 0); std::sort(images.begin(), images.end()); From c077ecc7515cdfc19844165167e8a1e655d77b7a Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 12 Jul 2017 09:23:45 +0200 Subject: [PATCH 22/53] Small modifications. Add new member property and method to Reader class. --- .../include/opencv2/photometric_calib/Reader.hpp | 2 ++ modules/photometric_calib/src/Reader.cpp | 5 +++++ 2 files changed, 7 insertions(+) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 38cd2ead0..610982ade 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -68,6 +68,7 @@ public: int getWidth() const; int getHeight() const; const std::string &getFolderPath() const; + const std::string &getTimeFilePath() const; private: /*! @@ -83,6 +84,7 @@ private: int _width, _height;//The image width and height. All the images should be of the same size. std::string _folderPath; + std::string _timeFilePath; }; //! @} diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 379f05990..5ff00c12c 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -40,6 +40,7 @@ void Reader::loadTimestamps(const std::string ×File) timeFile.release(); CV_Assert(timeStamps.size() == getNumImages() && exposureDurations.size() == getNumImages()); + _timeFilePath = timesFile; } Reader::Reader(const std::string &folderPath, const std::string &imageExt, const std::string ×Path) @@ -110,4 +111,8 @@ const std::string &Reader::getFolderPath() const { return _folderPath; } +const std::string &Reader::getTimeFilePath() const { + return _timeFilePath; +} + }} // namespace photometric_calib, cv \ No newline at end of file From 1bc68006df754bf625b164173e65743ce4fc1ad9 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 12 Jul 2017 09:29:46 +0200 Subject: [PATCH 23/53] Add class ResponseCalib: draft of camera response function calibrator. --- .../photometric_calib/ResponseCalib.hpp | 47 +++ .../photometric_calib/src/ResponseCalib.cpp | 310 ++++++++++++++++++ 2 files changed, 357 insertions(+) create mode 100644 modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp create mode 100644 modules/photometric_calib/src/ResponseCalib.cpp diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp new file mode 100644 index 000000000..ca3aaf22d --- /dev/null +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -0,0 +1,47 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#ifndef _OPENCV_RESPONSECALIB_HPP +#define _OPENCV_RESPONSECALIB_HPP + +#include "opencv2/photometric_calib.hpp" +#include "opencv2/photometric_calib/Reader.hpp" + +namespace cv { namespace photometric_calib { + +class CV_EXPORTS ResponseCalib +{ +public: + ResponseCalib(std::string folderPath, std::string timePath); + ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames); + + void plotE(const double* E, int w, int h, const std::string &saveTo=""); + + Vec2d rmse(const double* G, const double* E, const std::vector &exposureVec, const std::vector &dataVec, int wh); + + void plotG(const double* G, const std::string &saveTo=""); + + void calib(); + + inline const std::string& getImageFolderPath () const + { + CV_Assert(imageReader); + return imageReader -> getFolderPath(); + } + inline const std::string& getTimeFilePath () const + { + CV_Assert(imageReader); + return imageReader -> getTimeFilePath(); + } + +private: + int leakPadding; + int nIts; + int skipFrames; + Reader *imageReader; +}; + +}} + +#endif //_OPENCV_RESPONSECALIB_HPP diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp new file mode 100644 index 000000000..30247fee4 --- /dev/null +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -0,0 +1,310 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#include "precomp.hpp" +#include "opencv2/photometric_calib/ResponseCalib.hpp" + +#include +#include +#include + +namespace cv { namespace photometric_calib{ + +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath) : leakPadding(2), nIts(10), skipFrames(1) +{ + imageReader = new Reader(folderPath, "png", timePath); +} + +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames) : + leakPadding(leakPadding), nIts(nIts), skipFrames(skipFrames) +{ + imageReader = new Reader(folderPath, "png", timePath); +} + +Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector &exposureVec, const std::vector &dataVec, + int wh) +{ + long double e = 0; // yeah - these will be sums of a LOT of values, so we need super high precision. + long double num = 0; + + unsigned long n = dataVec.size(); + for(size_t i = 0; i < n; i++) + { + for(size_t k = 0; k < wh; k++) + { + if(dataVec[i][k] == 255) continue; + double r = G[dataVec[i][k]] - exposureVec[i] * E[k]; + if(!std::isfinite(r)) continue; + e += r * r * 1e-10; + num++; + } + } + + return Vec2d((double)(1e5 * sqrtl((e/num))), (double)num); +} + +void ResponseCalib::plotE(const double* E, int w, int h, const std::string &saveTo="") +{ + + // try to find some good color scaling for plotting. + double offset = 20; + double min = 1e10, max = -1e10; + + double Emin = 1e10, Emax = -1e10; + + for(size_t i = 0; i < w * h; i++) + { + double le = log(E[i] + offset); + if(le < min) min = le; + if(le > max) max = le; + + if(E[i] < Emin) Emin = E[i]; + if(E[i] > Emax) Emax = E[i]; + } + + cv::Mat EImg = cv::Mat(h,w,CV_8UC3); + cv::Mat EImg16 = cv::Mat(h,w,CV_16U); + + for(int i=0;i(i) = color; + EImg16.at(i) = (ushort)(255 * 255 * (E[i] - Emin) / (Emax - Emin)); + } + + std::cout<<"Irradiance "< max) max = G[i]; + } + + for(int i=0;i<256;i++) + { + double val = 256*(G[i]-min) / (max-min); + for(int k=0;k<256;k++) + { + if(val < k) + GImg.at(k,i) = (float)(k - val); + } + } + + std::cout<<"Inv. Response "< exposureDurationVec; + std::vector dataVec; + + for(size_t i = 0 ; i < imageReader->getNumImages(); i += skipFrames) + { + cv::Mat img = imageReader->getImage(i); + if(img.rows==0 || img.cols==0) continue; + CV_Assert(img.type() == CV_8U); + w = img.cols; + h = img.rows; + + uchar *data = new uchar[w*h]; + memcpy(data, img.data, w*h); + dataVec.push_back(data); + exposureDurationVec.push_back((double)(imageReader->getExposureDuration(i))); + unsigned char* data2 = new unsigned char[w*h]; + + for (int i = 0; i < leakPadding; ++i) + { + memcpy(data2, data, w*h); + for (int y = 1; y < h - 1; ++y) + { + for (int x = 1; x < w - 1; ++x) + { + if(data[x+y*w] == 255) + { + data2[x+1 + w*(y+1)] = 255; + data2[x+1 + w*(y )] = 255; + data2[x+1 + w*(y-1)] = 255; + + data2[x + w*(y+1)] = 255; + data2[x + w*(y )] = 255; + data2[x + w*(y-1)] = 255; + + data2[x-1 + w*(y+1)] = 255; + data2[x-1 + w*(y )] = 255; + data2[x-1 + w*(y-1)] = 255; + } + } + } + memcpy(data, data2, w*h); + } + delete[] data2; + } + n = dataVec.size(); + + double* E = new double[w*h]; // scene irradiance + double* En = new double[w*h]; // scene irradiance + double* G = new double[256]; // inverse response function + + // set starting scene irradiance to mean of all images. + memset(E,0,sizeof(double)*w*h); + memset(En,0,sizeof(double)*w*h); + memset(G,0,sizeof(double)*256); + + for(int i=0;i 1) G[i] = G[i-1] + (G[i-1]-G[i-2]); + } + delete[] GSum; + delete[] GNum; + printf("optG RMSE = %f! \t", rmse(G, E, exposureDurationVec, dataVec, w*h )[0]); + + char buf[1000]; snprintf(buf, 1000, "G-%d.png", it+1); + plotG(G, buf); + } + + + + + + if(optE) + { + // optimize scene irradiance function. + double* ESum = new double[w*h]; + double* ENum = new double[w*h]; + memset(ESum,0,w*h*sizeof(double)); + memset(ENum,0,w*h*sizeof(double)); + for(int i=0;i Date: Wed, 12 Jul 2017 09:43:21 +0200 Subject: [PATCH 24/53] fix shadow declaration. --- .../photometric_calib/ResponseCalib.hpp | 10 +++++----- .../photometric_calib/src/ResponseCalib.cpp | 18 +++++++++--------- 2 files changed, 14 insertions(+), 14 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp index ca3aaf22d..c2c434baa 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -16,11 +16,11 @@ public: ResponseCalib(std::string folderPath, std::string timePath); ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames); - void plotE(const double* E, int w, int h, const std::string &saveTo=""); + void plotE(const double* E, int w, int h, const std::string &saveTo); Vec2d rmse(const double* G, const double* E, const std::vector &exposureVec, const std::vector &dataVec, int wh); - void plotG(const double* G, const std::string &saveTo=""); + void plotG(const double* G, const std::string &saveTo); void calib(); @@ -36,9 +36,9 @@ public: } private: - int leakPadding; - int nIts; - int skipFrames; + int _leakPadding; + int _nIts; + int _skipFrames; Reader *imageReader; }; diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 30247fee4..a5b329311 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -11,13 +11,13 @@ namespace cv { namespace photometric_calib{ -ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath) : leakPadding(2), nIts(10), skipFrames(1) +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath) : _leakPadding(2), _nIts(10), _skipFrames(1) { imageReader = new Reader(folderPath, "png", timePath); } ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames) : - leakPadding(leakPadding), nIts(nIts), skipFrames(skipFrames) + _leakPadding(leakPadding), _nIts(nIts), _skipFrames(skipFrames) { imageReader = new Reader(folderPath, "png", timePath); } @@ -31,7 +31,7 @@ Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector exposureDurationVec; std::vector dataVec; - for(size_t i = 0 ; i < imageReader->getNumImages(); i += skipFrames) + for(size_t i = 0 ; i < imageReader->getNumImages(); i += _skipFrames) { cv::Mat img = imageReader->getImage(i); if(img.rows==0 || img.cols==0) continue; @@ -142,7 +142,7 @@ void ResponseCalib::calib() exposureDurationVec.push_back((double)(imageReader->getExposureDuration(i))); unsigned char* data2 = new unsigned char[w*h]; - for (int i = 0; i < leakPadding; ++i) + for (int j = 0; j < _leakPadding; ++j) { memcpy(data2, data, w*h); for (int y = 1; y < h - 1; ++y) @@ -205,7 +205,7 @@ void ResponseCalib::calib() bool optE = true; bool optG = true; - for(int it=0;it Date: Wed, 12 Jul 2017 09:57:50 +0200 Subject: [PATCH 25/53] add math.h header file. --- modules/photometric_calib/src/ResponseCalib.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index a5b329311..51ac84353 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -5,9 +5,9 @@ #include "precomp.hpp" #include "opencv2/photometric_calib/ResponseCalib.hpp" -#include #include #include +#include namespace cv { namespace photometric_calib{ From 345f7fede8b2297ff0cb1cf5c166189f4ac4899b Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 12 Jul 2017 10:06:47 +0200 Subject: [PATCH 26/53] Android doesn't support long double, so use sqrt instead of sqrtl. According to http://en.cppreference.com/w/c/numeric/math/sqrt, sqrt() is defined as a type-generic macro. --- modules/photometric_calib/src/ResponseCalib.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 51ac84353..3cfb989a7 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -41,7 +41,8 @@ Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector Date: Wed, 12 Jul 2017 14:13:40 +0200 Subject: [PATCH 27/53] fix type conversion problem. --- modules/photometric_calib/src/ResponseCalib.cpp | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 3cfb989a7..04dd94302 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -28,7 +28,7 @@ Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector exposureDurationVec; std::vector dataVec; - for(size_t i = 0 ; i < imageReader->getNumImages(); i += _skipFrames) + for(unsigned long i = 0 ; i < imageReader->getNumImages(); i += _skipFrames) { cv::Mat img = imageReader->getImage(i); if(img.rows==0 || img.cols==0) continue; @@ -181,7 +182,7 @@ void ResponseCalib::calib() memset(En,0,sizeof(double)*w*h); memset(G,0,sizeof(double)*256); - for(int i=0;i Date: Tue, 18 Jul 2017 11:04:55 +0200 Subject: [PATCH 28/53] Create class VignetteCalib --- .../photometric_calib/VignetteCalib.hpp | 43 +++++++++++++++++++ .../photometric_calib/src/VignetteCalib.cpp | 28 ++++++++++++ 2 files changed, 71 insertions(+) create mode 100644 modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp create mode 100644 modules/photometric_calib/src/VignetteCalib.cpp diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp new file mode 100644 index 000000000..9d6a2ff88 --- /dev/null +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -0,0 +1,43 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#ifndef _OPENCV_VIGNETTECALIB_HPP +#define _OPENCV_VIGNETTECALIB_HPP + +#include "opencv2/photometric_calib.hpp" +#include "opencv2/photometric_calib/Reader.hpp" + +namespace cv { namespace photometric_calib { + + +class CV_EXPORTS VignetteCalib +{ +public: + VignetteCalib(std::string folderPath, std::string timePath); + VignetteCalib(std::string folderPath, std::string timePath, int imageSkip, int maxIterations, int outlierTh, + int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); + +private: + int _imageSkip; + int _maxIterations; + int _outlierTh; + + // grid width for template image. + int _gridWidth; + int _gridHeight; + + // width of grid relative to marker (fac times marker size) + float _facW; + float _facH; + + // remove pixel with absolute gradient larger than this from the optimization. + int _maxAbsGrad; + + Reader *imageReader; +}; + +}} + + +#endif //_OPENCV_VIGNETTECALIB_HPP diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp new file mode 100644 index 000000000..94d0b851b --- /dev/null +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -0,0 +1,28 @@ +// This file is part of OpenCV project. +// It is subject to the license terms in the LICENSE file found in the top-level directory +// of this distribution and at http://opencv.org/license.html. + +#include "precomp.hpp" +#include "opencv2/photometric_calib/VignetteCalib.hpp" + +#include +#include + +namespace cv { namespace photometric_calib{ + + +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath) : + _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), + _maxAbsGrad(255) +{ + imageReader = new Reader(folderPath, "png", timePath); +} + +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, int imageSkip, int maxIterations, + int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : + _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), _gridHeight(gridHeight), + _facW(facW), _facH(facH), _maxAbsGrad(maxAbsGrad) +{ + imageReader = new Reader(folderPath, "png", timePath); +} +}}// namespace photometric_calib, cv \ No newline at end of file From 2556e8cc868b898e768ae21d27c7e00f9d50f8c2 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 30 Jul 2017 18:17:46 +0200 Subject: [PATCH 29/53] add support header files for camera undistortion and aruco marker detector for vignette calibration --- modules/photometric_calib/src/precomp.hpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/modules/photometric_calib/src/precomp.hpp b/modules/photometric_calib/src/precomp.hpp index 5d7c2b7dc..274b99475 100644 --- a/modules/photometric_calib/src/precomp.hpp +++ b/modules/photometric_calib/src/precomp.hpp @@ -8,5 +8,7 @@ #include "opencv2/core.hpp" #include "opencv2/imgproc.hpp" #include "opencv2/highgui.hpp" +#include "opencv2/aruco.hpp" +#include "opencv2/calib3d/calib3d.hpp" #include #endif \ No newline at end of file From a981654d7442f504299179d54fc2aaeb6a5ddefe Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 30 Jul 2017 18:19:23 +0200 Subject: [PATCH 30/53] finish interfaces of VignetteCalib class --- .../opencv2/photometric_calib/VignetteCalib.hpp | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 9d6a2ff88..58cd73ef6 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -7,6 +7,7 @@ #include "opencv2/photometric_calib.hpp" #include "opencv2/photometric_calib/Reader.hpp" +#include "opencv2/photometric_calib/GammaRemover.hpp" namespace cv { namespace photometric_calib { @@ -14,10 +15,18 @@ namespace cv { namespace photometric_calib { class CV_EXPORTS VignetteCalib { public: - VignetteCalib(std::string folderPath, std::string timePath); - VignetteCalib(std::string folderPath, std::string timePath, int imageSkip, int maxIterations, int outlierTh, + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile); + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); + virtual ~VignetteCalib(); + + //EIGEN_ALWAYS_INLINE float getInterpolatedElement(const float* const mat, const float x, const float y, const int width) + float getInterpolatedElement(const float* const mat, const float x, const float y, const int width); + void displayImage(float* I, int w, int h, std::string name); + void displayImageV(float* I, int w, int h, std::string name); + void calib(); + private: int _imageSkip; int _maxIterations; @@ -34,7 +43,11 @@ private: // remove pixel with absolute gradient larger than this from the optimization. int _maxAbsGrad; + Matx33f _cameraMatrix; + Mat _distCoeffs; + Reader *imageReader; + GammaRemover *gammaRemover; }; }} From b2233121a352c8089050c56d3e5ed91073df230e Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 30 Jul 2017 18:20:24 +0200 Subject: [PATCH 31/53] finish implementation of VignetteCalibration roughly. Still a lot to be improved. --- .../photometric_calib/src/VignetteCalib.cpp | 479 +++++++++++++++++- 1 file changed, 477 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 94d0b851b..392229aa9 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -7,22 +7,497 @@ #include #include +#include namespace cv { namespace photometric_calib{ -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath) : +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile) : _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), _maxAbsGrad(255) { imageReader = new Reader(folderPath, "png", timePath); + // check the extension of the camera file. + CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); + FileStorage cameraStorage(cameraFile, FileStorage::READ); + cameraStorage["cameraMatrix"] >> _cameraMatrix; + cameraStorage["distCoeffs"] >> _distCoeffs; + gammaRemover = new GammaRemover(gammaFile, imageReader->getWidth(), imageReader->getHeight()); } -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, int imageSkip, int maxIterations, +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), _gridHeight(gridHeight), _facW(facW), _facH(facH), _maxAbsGrad(maxAbsGrad) { imageReader = new Reader(folderPath, "png", timePath); + // check the extension of the camera file. + CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); + FileStorage cameraStorage(cameraFile, FileStorage::READ); + cameraStorage["cameraMatrix"] >> _cameraMatrix; + cameraStorage["distCoeffs"] >> _distCoeffs; + gammaRemover = new GammaRemover(gammaFile, imageReader->getWidth(), imageReader->getHeight()); +} + +float VignetteCalib::getInterpolatedElement(const float *const mat, const float x, const float y, const int width) +{ + int ix = (int)x; + int iy = (int)y; + float dx = x - ix; + float dy = y - iy; + float dxdy = dx*dy; + const float* bp = mat +ix+iy*width; + + float res = dxdy * bp[1+width] + + (dy-dxdy) * bp[width] + + (dx-dxdy) * bp[1] + + (1-dx-dy+dxdy) * bp[0]; + + return res; +} + +void VignetteCalib::displayImage(float *I, int w, int h, std::string name) +{ + float vmin=1e10; + float vmax=-1e10; + + for(int i=0;i I[i]) vmin = I[i]; + if(vmax < I[i]) vmax = I[i]; + } + + cv::Mat img = cv::Mat(h,w,CV_8UC3); + + for(int i=0;i(i) = cv::Vec3b(0,0,255); + else img.at(i) = cv::Vec3b(255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin)); + } + + printf("plane image values %f - %f!\n", vmin, vmax); + cv::imshow(name, img); + cv::imwrite("vignetteCalibResult/plane.png", img); +} + +void VignetteCalib::displayImageV(float *I, int w, int h, std::string name) +{ + cv::Mat img = cv::Mat(h,w,CV_8UC3); + for(int i=0;i(i) = cv::Vec3b(0,0,255); + else + { + float c = 254*I[i]; + img.at(i) = cv::Vec3b(c,c,c); + } + + } + cv::imshow(name, img); +} + +void VignetteCalib::calib() +{ + if(-1 == system("rm -rf vignetteCalibResult")) printf("could not delete old vignetteCalibResult folder!\n"); + if(-1 == system("mkdir vignetteCalibResult")) printf("could not delete old vignetteCalibResult folder!\n"); + + // affine map from plane cordinates to grid coordinates. + Mat test2(3,3,CV_8U); + Matx33f test3(test2); + Matx33f K_p2idx = Matx33f::eye(); + Mat1f test = Mat1f::eye(3,3); + K_p2idx(0,0) = _gridWidth / _facW; + K_p2idx(1,1) = _gridHeight / _facH; + K_p2idx(0,2) = _gridWidth / 2; + K_p2idx(1,2) = _gridHeight / 2; + Matx33f K_p2idx_inverse = K_p2idx.inv(); + + int wO, hO; + wO = imageReader->getWidth(); + hO = imageReader->getHeight(); + int wI = wO, hI = hO; + + Ptr parameters = aruco::DetectorParameters::create(); + Ptr dictionary = aruco::getPredefinedDictionary(cv::aruco::DICT_6X6_250); + + std::vector images; + std::vector p2imgX; + std::vector p2imgY; + + float meanExposure = 0.f; + for(size_t i=0; i < imageReader->getNumImages(); ++i) + { + std::vector markerIds; + std::vector< std::vector > markerCorners, rejectedCandidates; + + Mat oriImg = imageReader->getImage(i); + Mat undisImg; + undistort(oriImg, undisImg, _cameraMatrix, _distCoeffs); + + aruco::detectMarkers(undisImg, dictionary, markerCorners, markerIds, parameters, rejectedCandidates); + + if(markerCorners.size() != 1) continue; + + std::vector ptsP; + std::vector ptsI; + ptsI.push_back(Point2f(markerCorners[0][0].x, markerCorners[0][0].y)); + ptsI.push_back(Point2f(markerCorners[0][1].x, markerCorners[0][1].y)); + ptsI.push_back(Point2f(markerCorners[0][2].x, markerCorners[0][2].y)); + ptsI.push_back(Point2f(markerCorners[0][3].x, markerCorners[0][3].y)); + ptsP.push_back(Point2f(-0.5,0.5)); + ptsP.push_back(Point2f(0.5,0.5)); + ptsP.push_back(Point2f(0.5,-0.5)); + ptsP.push_back(Point2f(-0.5,-0.5)); + + Mat Hcv = findHomography(ptsP, ptsI); +// Eigen::Matrix3f H; +// H(0,0) = Hcv.at(0,0); +// H(0,1) = Hcv.at(0,1); +// H(0,2) = Hcv.at(0,2); +// H(1,0) = Hcv.at(1,0); +// H(1,1) = Hcv.at(1,1); +// H(1,2) = Hcv.at(1,2); +// H(2,0) = Hcv.at(2,0); +// H(2,1) = Hcv.at(2,1); +// H(2,2) = Hcv.at(2,2); + Matx33f H(Hcv); + + std::vector imgRaw(wI * hI); + gammaRemover->getUnGammaImageVec(imageReader->getImage(i), imgRaw); + + float* plane2imgX = new float[_gridWidth*_gridHeight]; + float* plane2imgY = new float[_gridWidth*_gridHeight]; + + Matx33f HK = H * K_p2idx_inverse; + + int idx=0; + for(int y=0;y<_gridHeight;y++) + for(int x=0;x<_gridWidth;x++) + { + //Eigen::Vector3f pp = HK*Eigen::Vector3f(x,y,1); + Vec3f pp = HK * Vec3f(x,y,1); + plane2imgX[idx] = pp[0] / pp[2]; + plane2imgY[idx] = pp[1] / pp[2]; + idx++; + } + + //reader->getUndistorter()->distortCoordinates(plane2imgX, plane2imgY, gw*gh); + + float expDuration; + expDuration = (imageReader->getExposureDuration(i) == 0 ? 1 : imageReader->getExposureDuration(i)); + float* image = new float[wI*hI]; + for(int y=0; y _maxAbsGrad) { image[x+y*wI] = NAN; image[x+deltax+(y+deltay)*wI]=NAN; } + } + } + + images.push_back(image); + + // debug-plot. + cv::Mat dbgImg(hI, wI, CV_8UC3); + for(int i=0;i(i) = cv::Vec3b(imgRaw[i], imgRaw[i], imgRaw[i]); + + for(int x=0; x<=_gridWidth;x+=200) + for(int y=0; y<=_gridHeight;y+=10) + { + int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; + int idxT = (x<_gridWidth ? x : _gridWidth-1)+((y+10)<_gridHeight ? (y+10) : _gridHeight-1)*_gridWidth; + + int u_dS = plane2imgX[idxS]+0.5; + int v_dS = plane2imgY[idxS]+0.5; + + int u_dT = plane2imgX[idxT]+0.5; + int v_dT = plane2imgY[idxT]+0.5; + + if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT1 && v_d >1 && u_dgetNumImages(); + + float* planeColor = new float[_gridWidth*_gridHeight]; + float* planeColorFF = new float[_gridWidth*_gridHeight]; + float* planeColorFC = new float[_gridWidth*_gridHeight]; + float* vignetteFactor = new float[hI*wI]; + float* vignetteFactorTT = new float[hI*wI]; + float* vignetteFactorCT = new float[hI*wI]; + + // initialize vignette factors to 1. + for(int i=0;i oth2) + { + E += oth2; + R ++; + continue; + } + + + planeColorFF[pi] += fac*fac; + planeColorFC[pi] += color*fac; + + if(cvIsNaN(planeColor[pi])) continue; + E += residual; + R ++; + } + } + + for(int pi=0;pi<_gridWidth*_gridWidth;pi++) // for all plane points + { + if(planeColorFF[pi] < 1) + planeColor[pi]=NAN; + else + planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; + } + displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); + + printf("%f residual terms => %f\n", R, sqrtf(E/R)); + + // ================================ optimize vignette ======================================= + memset(vignetteFactorTT,0,hI*wI*sizeof(float)); + memset(vignetteFactorCT,0,hI*wI*sizeof(float)); + E=0;R=0; + + for(int img=0;img oth2) + { + E += oth2; + R ++; + continue; + } + + + int ix = (int)x; + int iy = (int)y; + float dx = x - ix; + float dy = y - iy; + float dxdy = dx*dy; + + vignetteFactorTT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorPlane*colorPlane; + vignetteFactorTT[ix+iy*wI + 1] += (dx-dxdy) * colorPlane*colorPlane; + vignetteFactorTT[ix+iy*wI + wI] += (dy-dxdy) * colorPlane*colorPlane; + vignetteFactorTT[ix+iy*wI + 1+wI] += dxdy * colorPlane*colorPlane; + + vignetteFactorCT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorImage*colorPlane; + vignetteFactorCT[ix+iy*wI + 1] += (dx-dxdy) * colorImage*colorPlane; + vignetteFactorCT[ix+iy*wI + wI] += (dy-dxdy) * colorImage*colorPlane; + vignetteFactorCT[ix+iy*wI + 1+wI] += dxdy * colorImage*colorPlane; + + if(cvIsNaN(fac)) continue; + E += residual; + R ++; + } + } + + float maxFac=0; + for(int pi=0;pimaxFac) maxFac=vignetteFactor[pi]; + } + } + + printf("%f residual terms => %f\n", R, sqrtf(E/R)); + + // normalize to vignette max. factor 1. + for(int pi=0;pi0 && !cvIsNaN(vignetteFactorCT[idx+1-wI])) {sum += vignetteFactorCT[idx+1-wI]; num++;} + + if(y0 && !cvIsNaN(vignetteFactorCT[idx-wI])) {sum += vignetteFactorCT[idx-wI]; num++;} + + if(y0 && !cvIsNaN(vignetteFactorCT[idx-1+wI])) {sum += vignetteFactorCT[idx-1+wI]; num++;} + if(x>0 && !cvIsNaN(vignetteFactorCT[idx-1])) {sum += vignetteFactorCT[idx-1]; num++;} + if(y>0 && x>0 && !cvIsNaN(vignetteFactorCT[idx-1-wI])) {sum += vignetteFactorCT[idx-1-wI]; num++;} + + if(num>0) vignetteFactorTT[idx] = sum/num; + } + } + } + + { + displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); + cv::Mat wrap = cv::Mat(hI, wI, CV_32F, vignetteFactorTT)*254.9*254.9; + cv::Mat wrap16; + wrap.convertTo(wrap16, CV_16U,1,0); + cv::imwrite("vignetteCalibResult/vignetteSmoothed.png", wrap16); + cv::waitKey(50); + } + { + displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); + cv::Mat wrap = cv::Mat(hI, wI, CV_32F, vignetteFactor)*254.9*254.9; + cv::Mat wrap16; + wrap.convertTo(wrap16, CV_16U,1,0); + cv::imwrite("vignetteCalibResult/vignette.png", wrap16); + cv::waitKey(50); + } + } + } + + logFile.flush(); + logFile.close(); + + delete[] planeColor; + delete[] planeColorFF; + delete[] planeColorFC; + delete[] vignetteFactor; + delete[] vignetteFactorTT; + delete[] vignetteFactorCT; + + for(int i=0;i Date: Wed, 9 Aug 2017 14:34:51 +0200 Subject: [PATCH 32/53] Use proper module and remove redundant headers. --- modules/photometric_calib/CMakeLists.txt | 2 +- modules/photometric_calib/src/precomp.hpp | 3 +-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/modules/photometric_calib/CMakeLists.txt b/modules/photometric_calib/CMakeLists.txt index 951c173d5..696dfc065 100644 --- a/modules/photometric_calib/CMakeLists.txt +++ b/modules/photometric_calib/CMakeLists.txt @@ -1,2 +1,2 @@ set(the_description "Photometric Calibration") -ocv_define_module(photometric_calib opencv_core opencv_imgproc opencv_calib3d opencv_features2d opencv_highgui WRAP python) \ No newline at end of file +ocv_define_module(photometric_calib opencv_core opencv_aruco opencv_calib3d opencv_highgui WRAP python) \ No newline at end of file diff --git a/modules/photometric_calib/src/precomp.hpp b/modules/photometric_calib/src/precomp.hpp index 274b99475..39d1b844b 100644 --- a/modules/photometric_calib/src/precomp.hpp +++ b/modules/photometric_calib/src/precomp.hpp @@ -6,9 +6,8 @@ #define __OPENCV_PRECOMP_H__ #include "opencv2/core.hpp" -#include "opencv2/imgproc.hpp" #include "opencv2/highgui.hpp" #include "opencv2/aruco.hpp" -#include "opencv2/calib3d/calib3d.hpp" +#include "opencv2/calib3d.hpp" #include #endif \ No newline at end of file From 044bb5ea7ccf7befa67a7bca2fa34359b1bb4fa4 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 9 Aug 2017 14:35:24 +0200 Subject: [PATCH 33/53] Fix some bugs. --- .../photometric_calib/VignetteCalib.hpp | 2 +- .../photometric_calib/src/VignetteCalib.cpp | 63 ++++++++++--------- 2 files changed, 35 insertions(+), 30 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 58cd73ef6..ca892bdaf 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -43,7 +43,7 @@ private: // remove pixel with absolute gradient larger than this from the optimization. int _maxAbsGrad; - Matx33f _cameraMatrix; + Mat _cameraMatrix; Mat _distCoeffs; Reader *imageReader; diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 392229aa9..a52bfca2a 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -67,35 +67,35 @@ void VignetteCalib::displayImage(float *I, int w, int h, std::string name) if(vmax < I[i]) vmax = I[i]; } - cv::Mat img = cv::Mat(h,w,CV_8UC3); + Mat img = Mat(h,w,CV_8UC3); for(int i=0;i(i) = cv::Vec3b(0,0,255); - else img.at(i) = cv::Vec3b(255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin)); + if(cvIsNaN(I[i]) == 1) img.at(i) = Vec3b(0,0,255); + else img.at(i) = Vec3b(255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin)); } printf("plane image values %f - %f!\n", vmin, vmax); - cv::imshow(name, img); - cv::imwrite("vignetteCalibResult/plane.png", img); + imshow(name, img); + imwrite("vignetteCalibResult/plane.png", img); } void VignetteCalib::displayImageV(float *I, int w, int h, std::string name) { - cv::Mat img = cv::Mat(h,w,CV_8UC3); + Mat img = Mat(h,w,CV_8UC3); for(int i=0;i(i) = cv::Vec3b(0,0,255); + img.at(i) = Vec3b(0,0,255); else { float c = 254*I[i]; - img.at(i) = cv::Vec3b(c,c,c); + img.at(i) = Vec3b(c,c,c); } } - cv::imshow(name, img); + imshow(name, img); } void VignetteCalib::calib() @@ -104,8 +104,6 @@ void VignetteCalib::calib() if(-1 == system("mkdir vignetteCalibResult")) printf("could not delete old vignetteCalibResult folder!\n"); // affine map from plane cordinates to grid coordinates. - Mat test2(3,3,CV_8U); - Matx33f test3(test2); Matx33f K_p2idx = Matx33f::eye(); Mat1f test = Mat1f::eye(3,3); K_p2idx(0,0) = _gridWidth / _facW; @@ -120,13 +118,19 @@ void VignetteCalib::calib() int wI = wO, hI = hO; Ptr parameters = aruco::DetectorParameters::create(); - Ptr dictionary = aruco::getPredefinedDictionary(cv::aruco::DICT_6X6_250); + Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_6X6_250); std::vector images; std::vector p2imgX; std::vector p2imgY; float meanExposure = 0.f; + for(size_t i=0;igetNumImages();i+=_imageSkip) + meanExposure+=imageReader->getExposureDuration(i); + meanExposure = meanExposure/imageReader->getNumImages(); + + if(meanExposure==0) meanExposure = 1; + for(size_t i=0; i < imageReader->getNumImages(); ++i) { std::vector markerIds; @@ -205,9 +209,9 @@ void VignetteCalib::calib() images.push_back(image); // debug-plot. - cv::Mat dbgImg(hI, wI, CV_8UC3); - for(int i=0;i(i) = cv::Vec3b(imgRaw[i], imgRaw[i], imgRaw[i]); + Mat dbgImg(hI, wI, CV_8UC3); + for(int j=0;j(j) = Vec3b(imgRaw[j], imgRaw[j], imgRaw[j]); for(int x=0; x<=_gridWidth;x+=200) for(int y=0; y<=_gridHeight;y+=10) @@ -222,7 +226,8 @@ void VignetteCalib::calib() int v_dT = plane2imgY[idxT]+0.5; if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT Date: Wed, 9 Aug 2017 15:36:00 +0200 Subject: [PATCH 34/53] Fix warnings on Win and Android --- .../photometric_calib/src/VignetteCalib.cpp | 93 +++++++++---------- 1 file changed, 46 insertions(+), 47 deletions(-) diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index a52bfca2a..dd982bf75 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -7,7 +7,6 @@ #include #include -#include namespace cv { namespace photometric_calib{ @@ -73,10 +72,10 @@ void VignetteCalib::displayImage(float *I, int w, int h, std::string name) { //isnanf? isnan? if(cvIsNaN(I[i]) == 1) img.at(i) = Vec3b(0,0,255); - else img.at(i) = Vec3b(255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin),255*(I[i]-vmin) / (vmax-vmin)); + else img.at(i) = Vec3b((uchar)(255*(I[i]-vmin) / (vmax-vmin)),(uchar)(255*(I[i]-vmin) / (vmax-vmin)),(uchar)(255*(I[i]-vmin) / (vmax-vmin))); } - printf("plane image values %f - %f!\n", vmin, vmax); + std::cout<<"plane image values " << vmin << " - " << vmax << "!" <(i) = Vec3b(c,c,c); + img.at(i) = Vec3b((uchar)c,(uchar)c,(uchar)c); } } @@ -100,16 +99,16 @@ void VignetteCalib::displayImageV(float *I, int w, int h, std::string name) void VignetteCalib::calib() { - if(-1 == system("rm -rf vignetteCalibResult")) printf("could not delete old vignetteCalibResult folder!\n"); - if(-1 == system("mkdir vignetteCalibResult")) printf("could not delete old vignetteCalibResult folder!\n"); + if(-1 == system("rm -rf vignetteCalibResult")) std::cout<< "could not delete old vignetteCalibResult folder!" << std::endl; + if(-1 == system("mkdir vignetteCalibResult")) std::cout<< "could not delete old vignetteCalibResult folder" << std::endl; // affine map from plane cordinates to grid coordinates. Matx33f K_p2idx = Matx33f::eye(); Mat1f test = Mat1f::eye(3,3); K_p2idx(0,0) = _gridWidth / _facW; K_p2idx(1,1) = _gridHeight / _facH; - K_p2idx(0,2) = _gridWidth / 2; - K_p2idx(1,2) = _gridHeight / 2; + K_p2idx(0,2) = _gridWidth / 2.f; + K_p2idx(1,2) = _gridHeight / 2.f; Matx33f K_p2idx_inverse = K_p2idx.inv(); int wO, hO; @@ -125,13 +124,13 @@ void VignetteCalib::calib() std::vector p2imgY; float meanExposure = 0.f; - for(size_t i=0;igetNumImages();i+=_imageSkip) + for(unsigned long i=0;igetNumImages();i+=_imageSkip) meanExposure+=imageReader->getExposureDuration(i); meanExposure = meanExposure/imageReader->getNumImages(); if(meanExposure==0) meanExposure = 1; - for(size_t i=0; i < imageReader->getNumImages(); ++i) + for(unsigned long i=0; i < imageReader->getNumImages(); ++i) { std::vector markerIds; std::vector< std::vector > markerCorners, rejectedCandidates; @@ -181,7 +180,7 @@ void VignetteCalib::calib() for(int x=0;x<_gridWidth;x++) { //Eigen::Vector3f pp = HK*Eigen::Vector3f(x,y,1); - Vec3f pp = HK * Vec3f(x,y,1); + Vec3f pp = HK * Vec3f((float)x,(float)y,1); plane2imgX[idx] = pp[0] / pp[2]; plane2imgY[idx] = pp[1] / pp[2]; idx++; @@ -211,7 +210,7 @@ void VignetteCalib::calib() // debug-plot. Mat dbgImg(hI, wI, CV_8UC3); for(int j=0;j(j) = Vec3b(imgRaw[j], imgRaw[j], imgRaw[j]); + dbgImg.at(j) = Vec3b((uchar)imgRaw[j], (uchar)imgRaw[j], (uchar)imgRaw[j]); for(int x=0; x<=_gridWidth;x+=200) for(int y=0; y<=_gridHeight;y+=10) @@ -219,11 +218,11 @@ void VignetteCalib::calib() int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; int idxT = (x<_gridWidth ? x : _gridWidth-1)+((y+10)<_gridHeight ? (y+10) : _gridHeight-1)*_gridWidth; - int u_dS = plane2imgX[idxS]+0.5; - int v_dS = plane2imgY[idxS]+0.5; + int u_dS = (int)lround((plane2imgX[idxS]+0.5)); + int v_dS = (int)lround((plane2imgY[idxS]+0.5)); - int u_dT = plane2imgX[idxT]+0.5; - int v_dT = plane2imgY[idxT]+0.5; + int u_dT = (int)lround((plane2imgX[idxT]+0.5)); + int v_dT = (int)lround((plane2imgY[idxT]+0.5)); if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT1 && v_d >1 && u_dgetNumImages(); + unsigned long n = imageReader->getNumImages(); float* planeColor = new float[_gridWidth*_gridHeight]; float* planeColorFF = new float[_gridWidth*_gridHeight]; @@ -305,7 +304,7 @@ void VignetteCalib::calib() E=0;R=0; // for each plane pixel, it's optimum is at sum(CF)/sum(FF) - for(int img=0;img oth2) @@ -334,7 +333,7 @@ void VignetteCalib::calib() planeColorFF[pi] += fac*fac; planeColorFC[pi] += color*fac; - if(cvIsNaN(planeColor[pi])) continue; + if(cvIsNaN(planeColor[pi]) == 1) continue; E += residual; R ++; } @@ -349,14 +348,14 @@ void VignetteCalib::calib() } displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); - printf("%f residual terms => %f\n", R, sqrtf(E/R)); + std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; // ================================ optimize vignette ======================================= memset(vignetteFactorTT,0,hI*wI*sizeof(float)); memset(vignetteFactorCT,0,hI*wI*sizeof(float)); E=0;R=0; - for(int img=0;img oth2) @@ -400,7 +399,7 @@ void VignetteCalib::calib() vignetteFactorCT[ix+iy*wI + wI] += (dy-dxdy) * colorImage*colorPlane; vignetteFactorCT[ix+iy*wI + 1+wI] += dxdy * colorImage*colorPlane; - if(cvIsNaN(fac)) continue; + if(cvIsNaN(fac) == 1) continue; E += residual; R ++; } @@ -418,7 +417,7 @@ void VignetteCalib::calib() } } - printf("%f residual terms => %f\n", R, sqrtf(E/R)); + std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; // normalize to vignette max. factor 1. for(int pi=0;pi0 && !cvIsNaN(vignetteFactorCT[idx+1-wI])) {sum += vignetteFactorCT[idx+1-wI]; num++;} + if(x0 && cvIsNaN(vignetteFactorCT[idx+1-wI]) != 1) {sum += vignetteFactorCT[idx+1-wI]; num++;} - if(y0 && !cvIsNaN(vignetteFactorCT[idx-wI])) {sum += vignetteFactorCT[idx-wI]; num++;} + if(y0 && cvIsNaN(vignetteFactorCT[idx-wI]) != 1) {sum += vignetteFactorCT[idx-wI]; num++;} - if(y0 && !cvIsNaN(vignetteFactorCT[idx-1+wI])) {sum += vignetteFactorCT[idx-1+wI]; num++;} - if(x>0 && !cvIsNaN(vignetteFactorCT[idx-1])) {sum += vignetteFactorCT[idx-1]; num++;} - if(y>0 && x>0 && !cvIsNaN(vignetteFactorCT[idx-1-wI])) {sum += vignetteFactorCT[idx-1-wI]; num++;} + if(y0 && cvIsNaN(vignetteFactorCT[idx-1+wI]) != 1) {sum += vignetteFactorCT[idx-1+wI]; num++;} + if(x>0 && cvIsNaN(vignetteFactorCT[idx-1]) != 1) {sum += vignetteFactorCT[idx-1]; num++;} + if(y>0 && x>0 && cvIsNaN(vignetteFactorCT[idx-1-wI]) != 1) {sum += vignetteFactorCT[idx-1-wI]; num++;} if(num>0) vignetteFactorTT[idx] = sum/num; } @@ -492,7 +491,7 @@ void VignetteCalib::calib() delete[] vignetteFactorTT; delete[] vignetteFactorCT; - for(int i=0;i Date: Wed, 16 Aug 2017 18:46:49 +0200 Subject: [PATCH 35/53] Delete useless code and clean up header files. --- .../include/opencv2/photometric_calib.hpp | 24 +++---------- .../photometric_calib/GammaRemover.hpp | 2 +- .../opencv2/photometric_calib/Reader.hpp | 2 +- .../photometric_calib/ResponseCalib.hpp | 1 - .../photometric_calib/VignetteCalib.hpp | 2 +- .../photometric_calib/VignetteRemover.hpp | 2 +- .../src/photometric_calib.cpp | 36 ------------------- modules/photometric_calib/src/precomp.hpp | 1 + 8 files changed, 10 insertions(+), 60 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp index b45cfff13..efbd13f18 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -5,27 +5,13 @@ #ifndef __OPENCV_PHOTOMETRIC_CALIB_HPP__ #define __OPENCV_PHOTOMETRIC_CALIB_HPP__ -#include "opencv2/core.hpp" -#include "opencv2/imgproc.hpp" - -#include - -/** @defgroup photometric_calib Photometric Calibration -*/ +#include "opencv2/photometric_calib/Reader.hpp" +#include "opencv2/photometric_calib/GammaRemover.hpp" +#include "opencv2/photometric_calib/VignetteRemover.hpp" +#include "opencv2/photometric_calib/ResponseCalib.hpp" +#include "opencv2/photometric_calib/VignetteCalib.hpp" namespace cv { namespace photometric_calib{ - -//! @addtogroup photometric_calib -//! @{ - -class CV_EXPORTS PhotometricCalibrator : public Algorithm -{ -public: - bool validImgs(const std::vector &inputImgs, const std::vector &exposureTime); -}; - -//! @} - }} // namespace photometric_calib, cv #endif \ No newline at end of file diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index 101330db7..3b688ff1d 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -5,7 +5,7 @@ #ifndef _OPENCV_GAMMAREMOVER_HPP #define _OPENCV_GAMMAREMOVER_HPP -#include "opencv2/photometric_calib.hpp" +#include "opencv2/core.hpp" namespace cv { namespace photometric_calib { diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 610982ade..9f78b1f56 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -5,7 +5,7 @@ #ifndef _OPENCV_READER_HPP #define _OPENCV_READER_HPP -#include "opencv2/photometric_calib.hpp" +#include "opencv2/core.hpp" #include #include diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp index c2c434baa..aa80547ec 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -5,7 +5,6 @@ #ifndef _OPENCV_RESPONSECALIB_HPP #define _OPENCV_RESPONSECALIB_HPP -#include "opencv2/photometric_calib.hpp" #include "opencv2/photometric_calib/Reader.hpp" namespace cv { namespace photometric_calib { diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index ca892bdaf..66bbe6a29 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -5,7 +5,7 @@ #ifndef _OPENCV_VIGNETTECALIB_HPP #define _OPENCV_VIGNETTECALIB_HPP -#include "opencv2/photometric_calib.hpp" +#include "opencv2/core.hpp" #include "opencv2/photometric_calib/Reader.hpp" #include "opencv2/photometric_calib/GammaRemover.hpp" diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp index 6279262d7..c78098c78 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -5,7 +5,7 @@ #ifndef _OPENCV_VIGNETTEREMOVER_HPP #define _OPENCV_VIGNETTEREMOVER_HPP -#include "opencv2/photometric_calib.hpp" +#include "opencv2/core.hpp" namespace cv { namespace photometric_calib { diff --git a/modules/photometric_calib/src/photometric_calib.cpp b/modules/photometric_calib/src/photometric_calib.cpp index a83f66a97..d7331d444 100644 --- a/modules/photometric_calib/src/photometric_calib.cpp +++ b/modules/photometric_calib/src/photometric_calib.cpp @@ -3,42 +3,6 @@ // of this distribution and at http://opencv.org/license.html. #include "precomp.hpp" -#include "opencv2/photometric_calib.hpp" -#include namespace cv{ namespace photometric_calib{ - -using namespace std; - -bool PhotometricCalibrator::validImgs(const std::vector &inputImgs, const std::vector &exposureTime) -{ - if(inputImgs.empty() || exposureTime.empty() || inputImgs.size() != exposureTime.size()) - return false; - - int width = 0, height = 0; - for(size_t i = 0; i < inputImgs.size(); ++ i) - { - Mat img; - img = inputImgs[i]; - if(img.type() != CV_8U) - { - cout<<"The type of the image should be CV_8U!"< From 1cda04366a2be95ef180b61e155195c04e7b1754 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Wed, 16 Aug 2017 18:48:24 +0200 Subject: [PATCH 36/53] Add useful std output for response calibration. --- .../photometric_calib/src/ResponseCalib.cpp | 74 ++++++++++++------- 1 file changed, 49 insertions(+), 25 deletions(-) diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 04dd94302..335a567ff 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -41,7 +41,7 @@ Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vectorgetImage(i); if(img.rows==0 || img.cols==0) continue; CV_Assert(img.type() == CV_8U); + + if((w!=0 && w != img.cols) || img.cols==0) + { + std::cout<<"Width mismatch!"<getExposureDuration(i))); - unsigned char* data2 = new unsigned char[w*h]; + unsigned char* data2 = new unsigned char[w * h]; for (int j = 0; j < _leakPadding; ++j) { - memcpy(data2, data, w*h); + memcpy(data2, data, w * h); for (int y = 1; y < h - 1; ++y) { for (int x = 1; x < w - 1; ++x) { - if(data[x+y*w] == 255) + if(data[x + y * w] == 255) { data2[x+1 + w*(y+1)] = 255; data2[x+1 + w*(y )] = 255; @@ -167,11 +180,13 @@ void ResponseCalib::calib() } } } - memcpy(data, data2, w*h); + memcpy(data, data2, w * h); } delete[] data2; } n = dataVec.size(); + std::cout<<"Loaded "< Date: Wed, 16 Aug 2017 18:49:12 +0200 Subject: [PATCH 37/53] Add doxygen to GammaRemover and VignetteRemover. --- .../photometric_calib/GammaRemover.hpp | 34 +++++++++++++++++++ .../photometric_calib/VignetteRemover.hpp | 25 ++++++++++++++ 2 files changed, 59 insertions(+) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index 3b688ff1d..5df5ea36c 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -9,19 +9,53 @@ namespace cv { namespace photometric_calib { +//! @addtogroup photometric_calib +//! @{ + +/*! + * @brief Class for removing the camera response function (mostly gamma function) when provided with pcalib file. + * + */ + class CV_EXPORTS GammaRemover { public: + /*! + * @brief Constructor + * @param gammaPath the path of pcalib file of which the format should be .yaml or .yml + * @param w_ the width of input image + * @param h_ the height of input image + */ GammaRemover(const std::string &gammaPath, int w_, int h_); + /*! + * @brief get irradiance image in the form of cv::Mat. Convenient for display, etc. + * @param inputIm + * @return + */ Mat getUnGammaImageMat(Mat inputIm); + + /*! + * @brief get irradiance image in the form of std::vector. Convenient for optimization or SLAM. + * @param inputIm + * @param outImVec + */ void getUnGammaImageVec(Mat inputIm, std::vector &outImVec); + /*! + * @brief get gamma function. + * @return + */ inline float* getG() { if(!validGamma) return 0; else return G; }; + + /*! + * @brief get inverse gamma function + * @return + */ inline float* getGInv() { if(!validGamma) return 0; diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp index c78098c78..8fc4a2555 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -9,13 +9,38 @@ namespace cv { namespace photometric_calib { +//! @addtogroup photometric_calib +//! @{ + +/*! + * @brief Class for removing the vignetting artifact when provided with vignetting file. + * + */ + class CV_EXPORTS VignetteRemover { public: + /*! + * @brief Constructor + * @param vignettePath the path of vignetting file + * @param w_ the width of input image + * @param h_ the height of input image + */ VignetteRemover(const std::string &vignettePath, int w_, int h_); ~VignetteRemover(); + /*! + * @brief get vignetting-removed image in form of cv::Mat. + * @param unGammaImVec the irradiance image. + * @return + */ Mat getUnVignetteImageMat(std::vector &unGammaImVec); + + /*! + * @brief get vignetting-removed image in form of std::vector. + * @param unGammaImVec the irradiance image. + * @param outImVec the vignetting-removed image vector. + */ void getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec); private: From 15ca7a99bb4e8c5af948e11b45ff071c070b643c Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 20 Aug 2017 16:08:45 +0200 Subject: [PATCH 38/53] Add imageFormat parameter for response and vignetting calibration constructor class. --- .../include/opencv2/photometric_calib/ResponseCalib.hpp | 4 ++-- .../include/opencv2/photometric_calib/VignetteCalib.hpp | 4 ++-- modules/photometric_calib/src/ResponseCalib.cpp | 8 ++++---- modules/photometric_calib/src/VignetteCalib.cpp | 8 ++++---- 4 files changed, 12 insertions(+), 12 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp index aa80547ec..6bb2c5d13 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -12,8 +12,8 @@ namespace cv { namespace photometric_calib { class CV_EXPORTS ResponseCalib { public: - ResponseCalib(std::string folderPath, std::string timePath); - ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames); + ResponseCalib(std::string folderPath, std::string timePath, std::string imageFormat); + ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, std::string imageFormat); void plotE(const double* E, int w, int h, const std::string &saveTo); diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 66bbe6a29..1053f3a6d 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -15,9 +15,9 @@ namespace cv { namespace photometric_calib { class CV_EXPORTS VignetteCalib { public: - VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile); + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat); VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, int outlierTh, - int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); + int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad, std::string imageFormat); virtual ~VignetteCalib(); diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 335a567ff..8dc8ff5cc 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -11,15 +11,15 @@ namespace cv { namespace photometric_calib{ -ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath) : _leakPadding(2), _nIts(10), _skipFrames(1) +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, std::string imageFormat) : _leakPadding(2), _nIts(10), _skipFrames(1) { - imageReader = new Reader(folderPath, "png", timePath); + imageReader = new Reader(folderPath, imageFormat, timePath); } -ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames) : +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, std::string imageFormat) : _leakPadding(leakPadding), _nIts(nIts), _skipFrames(skipFrames) { - imageReader = new Reader(folderPath, "png", timePath); + imageReader = new Reader(folderPath, imageFormat, timePath); } Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector &exposureVec, const std::vector &dataVec, diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index dd982bf75..520e87395 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -11,11 +11,11 @@ namespace cv { namespace photometric_calib{ -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile) : +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat) : _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), _maxAbsGrad(255) { - imageReader = new Reader(folderPath, "png", timePath); + imageReader = new Reader(folderPath, imageFormat, timePath); // check the extension of the camera file. CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); FileStorage cameraStorage(cameraFile, FileStorage::READ); @@ -25,11 +25,11 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: } VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, - int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : + int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad, std::string imageFormat) : _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), _gridHeight(gridHeight), _facW(facW), _facH(facH), _maxAbsGrad(maxAbsGrad) { - imageReader = new Reader(folderPath, "png", timePath); + imageReader = new Reader(folderPath, imageFormat, timePath); // check the extension of the camera file. CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); FileStorage cameraStorage(cameraFile, FileStorage::READ); From 78f3e56df63fb7c92dbcd12c0327d73a57435113 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 20 Aug 2017 21:44:08 +0200 Subject: [PATCH 39/53] Memory-efficient vignette calibration implementation. --- .../photometric_calib/VignetteCalib.hpp | 14 +- .../photometric_calib/src/VignetteCalib.cpp | 460 +++++++++--------- 2 files changed, 239 insertions(+), 235 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 1053f3a6d..749d477ad 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -15,17 +15,20 @@ namespace cv { namespace photometric_calib { class CV_EXPORTS VignetteCalib { public: - VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat); - VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, int outlierTh, - int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad, std::string imageFormat); + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent); + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, int maxIterations, int outlierTh, + int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); virtual ~VignetteCalib(); //EIGEN_ALWAYS_INLINE float getInterpolatedElement(const float* const mat, const float x, const float y, const int width) float getInterpolatedElement(const float* const mat, const float x, const float y, const int width); + float calMeanExposureTime(); void displayImage(float* I, int w, int h, std::string name); void displayImageV(float* I, int w, int h, std::string name); + bool preCalib(unsigned long id, float*& image, float*& plane2imgX, float*& plane2imgY); void calib(); + void calibFast(); private: int _imageSkip; @@ -45,9 +48,14 @@ private: Mat _cameraMatrix; Mat _distCoeffs; + Matx33f _K_p2idx; + Matx33f _K_p2idx_inverse; Reader *imageReader; GammaRemover *gammaRemover; + + float _meanExposure; + bool _silent; }; }} diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 520e87395..224eeca9f 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -11,7 +11,7 @@ namespace cv { namespace photometric_calib{ -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat) : +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent) : _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), _maxAbsGrad(255) { @@ -22,10 +22,22 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: cameraStorage["cameraMatrix"] >> _cameraMatrix; cameraStorage["distCoeffs"] >> _distCoeffs; gammaRemover = new GammaRemover(gammaFile, imageReader->getWidth(), imageReader->getHeight()); + + // affine map from plane cordinates to grid coordinates. + _K_p2idx = Matx33f::eye(); + _K_p2idx(0,0) = _gridWidth / _facW; + _K_p2idx(1,1) = _gridHeight / _facH; + _K_p2idx(0,2) = _gridWidth / 2.f; + _K_p2idx(1,2) = _gridHeight / 2.f; + _K_p2idx_inverse = _K_p2idx.inv(); + + _meanExposure = calMeanExposureTime(); + + _silent = silent; } -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, int imageSkip, int maxIterations, - int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad, std::string imageFormat) : +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, int maxIterations, + int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), _gridHeight(gridHeight), _facW(facW), _facH(facH), _maxAbsGrad(maxAbsGrad) { @@ -36,6 +48,18 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: cameraStorage["cameraMatrix"] >> _cameraMatrix; cameraStorage["distCoeffs"] >> _distCoeffs; gammaRemover = new GammaRemover(gammaFile, imageReader->getWidth(), imageReader->getHeight()); + + // affine map from plane cordinates to grid coordinates. + _K_p2idx = Matx33f::eye(); + _K_p2idx(0,0) = _gridWidth / _facW; + _K_p2idx(1,1) = _gridHeight / _facH; + _K_p2idx(0,2) = _gridWidth / 2.f; + _K_p2idx(1,2) = _gridHeight / 2.f; + _K_p2idx_inverse = _K_p2idx.inv(); + + _meanExposure = calMeanExposureTime(); + + _silent = silent; } float VignetteCalib::getInterpolatedElement(const float *const mat, const float x, const float y, const int width) @@ -76,6 +100,7 @@ void VignetteCalib::displayImage(float *I, int w, int h, std::string name) } std::cout<<"plane image values " << vmin << " - " << vmax << "!" <getWidth(); - hO = imageReader->getHeight(); - int wI = wO, hI = hO; - - Ptr parameters = aruco::DetectorParameters::create(); - Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_6X6_250); - - std::vector images; - std::vector p2imgX; - std::vector p2imgY; - float meanExposure = 0.f; for(unsigned long i=0;igetNumImages();i+=_imageSkip) meanExposure+=imageReader->getExposureDuration(i); meanExposure = meanExposure/imageReader->getNumImages(); - if(meanExposure==0) meanExposure = 1; + return meanExposure; +} - for(unsigned long i=0; i < imageReader->getNumImages(); ++i) - { - std::vector markerIds; - std::vector< std::vector > markerCorners, rejectedCandidates; - - Mat oriImg = imageReader->getImage(i); - Mat undisImg; - undistort(oriImg, undisImg, _cameraMatrix, _distCoeffs); - - aruco::detectMarkers(undisImg, dictionary, markerCorners, markerIds, parameters, rejectedCandidates); - - if(markerCorners.size() != 1) continue; - - std::vector ptsP; - std::vector ptsI; - ptsI.push_back(Point2f(markerCorners[0][0].x, markerCorners[0][0].y)); - ptsI.push_back(Point2f(markerCorners[0][1].x, markerCorners[0][1].y)); - ptsI.push_back(Point2f(markerCorners[0][2].x, markerCorners[0][2].y)); - ptsI.push_back(Point2f(markerCorners[0][3].x, markerCorners[0][3].y)); - ptsP.push_back(Point2f(-0.5,0.5)); - ptsP.push_back(Point2f(0.5,0.5)); - ptsP.push_back(Point2f(0.5,-0.5)); - ptsP.push_back(Point2f(-0.5,-0.5)); - - Mat Hcv = findHomography(ptsP, ptsI); -// Eigen::Matrix3f H; -// H(0,0) = Hcv.at(0,0); -// H(0,1) = Hcv.at(0,1); -// H(0,2) = Hcv.at(0,2); -// H(1,0) = Hcv.at(1,0); -// H(1,1) = Hcv.at(1,1); -// H(1,2) = Hcv.at(1,2); -// H(2,0) = Hcv.at(2,0); -// H(2,1) = Hcv.at(2,1); -// H(2,2) = Hcv.at(2,2); - Matx33f H(Hcv); - - std::vector imgRaw(wI * hI); - gammaRemover->getUnGammaImageVec(imageReader->getImage(i), imgRaw); - - float* plane2imgX = new float[_gridWidth*_gridHeight]; - float* plane2imgY = new float[_gridWidth*_gridHeight]; - - Matx33f HK = H * K_p2idx_inverse; - - int idx=0; - for(int y=0;y<_gridHeight;y++) - for(int x=0;x<_gridWidth;x++) - { - //Eigen::Vector3f pp = HK*Eigen::Vector3f(x,y,1); - Vec3f pp = HK * Vec3f((float)x,(float)y,1); - plane2imgX[idx] = pp[0] / pp[2]; - plane2imgY[idx] = pp[1] / pp[2]; - idx++; - } +bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX, float*& plane2imgY) +{ + int wI = imageReader->getWidth(); + int hI = imageReader->getHeight(); - //reader->getUndistorter()->distortCoordinates(plane2imgX, plane2imgY, gw*gh); + std::vector markerIds; + std::vector< std::vector > markerCorners, rejectedCandidates; - float expDuration; - expDuration = (imageReader->getExposureDuration(i) == 0 ? 1 : imageReader->getExposureDuration(i)); - float* image = new float[wI*hI]; - for(int y=0; ygetImage(id); + Mat undisImg; + undistort(oriImg, undisImg, _cameraMatrix, _distCoeffs); - for(int y=2; y _maxAbsGrad) { image[x+y*wI] = NAN; image[x+deltax+(y+deltay)*wI]=NAN; } - } - } + Ptr parameters = aruco::DetectorParameters::create(); + Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_ARUCO_ORIGINAL); + + aruco::detectMarkers(undisImg, dictionary, markerCorners, markerIds, parameters, rejectedCandidates); + if(markerCorners.size() != 1) + return false; + + std::vector ptsP; + std::vector ptsI; + ptsI.push_back(Point2f(markerCorners[0][0].x, markerCorners[0][0].y)); + ptsI.push_back(Point2f(markerCorners[0][1].x, markerCorners[0][1].y)); + ptsI.push_back(Point2f(markerCorners[0][2].x, markerCorners[0][2].y)); + ptsI.push_back(Point2f(markerCorners[0][3].x, markerCorners[0][3].y)); + ptsP.push_back(Point2f(-0.5,0.5)); + ptsP.push_back(Point2f(0.5,0.5)); + ptsP.push_back(Point2f(0.5,-0.5)); + ptsP.push_back(Point2f(-0.5,-0.5)); + + Matx33f H = findHomography(ptsP, ptsI); + + std::vector imgRaw(wI * hI); + gammaRemover->getUnGammaImageVec(imageReader->getImage(id), imgRaw); + + plane2imgX = new float[_gridWidth*_gridHeight]; + plane2imgY = new float[_gridWidth*_gridHeight]; + + Matx33f HK = H * _K_p2idx_inverse; + + int idx=0; + for(int y=0;y<_gridHeight;y++) { + for (int x = 0; x < _gridWidth; x++) { + Vec3f pp = HK * Vec3f((float) x, (float) y, 1); + plane2imgX[idx] = pp[0] / pp[2]; + plane2imgY[idx] = pp[1] / pp[2]; + idx++; + } + } - images.push_back(image); + float expDuration; + expDuration = (imageReader->getExposureDuration(id) == 0 ? 1 : imageReader->getExposureDuration(id)); + image = new float[wI*hI]; + for(int y=0; y(j) = Vec3b((uchar)imgRaw[j], (uchar)imgRaw[j], (uchar)imgRaw[j]); + for(int y=2; y _maxAbsGrad) + { + image[x+y*wI] = NAN; image[x+deltax+(y+deltay)*wI]=NAN; + } + } + } + } - for(int x=0; x<=_gridWidth;x+=200) - for(int y=0; y<=_gridHeight;y+=10) + for(int x=0; x<_gridWidth;x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) { - int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; - int idxT = (x<_gridWidth ? x : _gridWidth-1)+((y+10)<_gridHeight ? (y+10) : _gridHeight-1)*_gridWidth; + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } - int u_dS = (int)lround((plane2imgX[idxS]+0.5)); - int v_dS = (int)lround((plane2imgY[idxS]+0.5)); + if(!_silent) + { + // debug-plot. + Mat dbgImg(hI, wI, CV_8UC3); + for(int j=0;j(j) = Vec3b((uchar)imgRaw[j], (uchar)imgRaw[j], (uchar)imgRaw[j]); - int u_dT = (int)lround((plane2imgX[idxT]+0.5)); - int v_dT = (int)lround((plane2imgY[idxT]+0.5)); + for(int x=0; x<=_gridWidth;x+=200) + for(int y=0; y<=_gridHeight;y+=10) + { + int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; + int idxT = (x<_gridWidth ? x : _gridWidth-1)+((y+10)<_gridHeight ? (y+10) : _gridHeight-1)*_gridWidth; - if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT1 && v_d >1 && u_d=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dTgetWidth(); + hO = imageReader->getHeight(); + int wI = wO, hI = hO; + // log file std::ofstream logFile; logFile.open("vignetteCalibResult/log.txt", std::ios::trunc | std::ios::out); logFile.precision(15); + // number of images in total unsigned long n = imageReader->getNumImages(); float* planeColor = new float[_gridWidth*_gridHeight]; @@ -293,12 +307,16 @@ void VignetteCalib::calib() double E=0; double R=0; + // optimization begins for(int it=0;it<_maxIterations;it++) { + std::cout<<"Iteration "< oth2) { @@ -328,15 +344,15 @@ void VignetteCalib::calib() R ++; continue; } - - planeColorFF[pi] += fac*fac; planeColorFC[pi] += color*fac; - if(cvIsNaN(planeColor[pi]) == 1) continue; E += residual; R ++; } + delete[] plane2imgX; + delete[] plane2imgY; + delete[] image; } for(int pi=0;pi<_gridWidth*_gridWidth;pi++) // for all plane points @@ -346,34 +362,32 @@ void VignetteCalib::calib() else planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; } - displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); - + if(!_silent) + displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; // ================================ optimize vignette ======================================= + std::cout<<"Optimize Vignette..."< oth2) { @@ -381,30 +395,27 @@ void VignetteCalib::calib() R ++; continue; } - - int ix = (int)x; int iy = (int)y; float dx = x - ix; float dy = y - iy; float dxdy = dx*dy; - vignetteFactorTT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorPlane*colorPlane; vignetteFactorTT[ix+iy*wI + 1] += (dx-dxdy) * colorPlane*colorPlane; vignetteFactorTT[ix+iy*wI + wI] += (dy-dxdy) * colorPlane*colorPlane; vignetteFactorTT[ix+iy*wI + 1+wI] += dxdy * colorPlane*colorPlane; - vignetteFactorCT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorImage*colorPlane; vignetteFactorCT[ix+iy*wI + 1] += (dx-dxdy) * colorImage*colorPlane; vignetteFactorCT[ix+iy*wI + wI] += (dy-dxdy) * colorImage*colorPlane; vignetteFactorCT[ix+iy*wI + 1+wI] += dxdy * colorImage*colorPlane; - if(cvIsNaN(fac) == 1) continue; E += residual; R ++; } + delete[] plane2imgX; + delete[] plane2imgY; + delete[] image; } - float maxFac=0; for(int pi=0;pimaxFac) maxFac=vignetteFactor[pi]; } } - std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; // normalize to vignette max. factor 1. for(int pi=0;pi0 && cvIsNaN(vignetteFactorCT[idx+1-wI]) != 1) {sum += vignetteFactorCT[idx+1-wI]; num++;} - - if(y0 && cvIsNaN(vignetteFactorCT[idx-wI]) != 1) {sum += vignetteFactorCT[idx-wI]; num++;} - - if(y0 && cvIsNaN(vignetteFactorCT[idx-1+wI]) != 1) {sum += vignetteFactorCT[idx-1+wI]; num++;} - if(x>0 && cvIsNaN(vignetteFactorCT[idx-1]) != 1) {sum += vignetteFactorCT[idx-1]; num++;} - if(y>0 && x>0 && cvIsNaN(vignetteFactorCT[idx-1-wI]) != 1) {sum += vignetteFactorCT[idx-1-wI]; num++;} - - if(num>0) vignetteFactorTT[idx] = sum/num; - } - } - } + float sum=0, num=0; + if(x0 && cvIsNaN(vignetteFactorCT[idx+1-wI]) != 1) {sum += vignetteFactorCT[idx+1-wI]; num++;} - { - displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); - Mat wrap = Mat(hI, wI, CV_32F, vignetteFactorTT)*254.9*254.9; - Mat wrap16; - wrap.convertTo(wrap16, CV_16U,1,0); - imwrite("vignetteCalibResult/vignetteSmoothed.png", wrap16); - waitKey(50); - } - { - displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); - Mat wrap = Mat(hI, wI, CV_32F, vignetteFactor)*254.9*254.9; - Mat wrap16; - wrap.convertTo(wrap16, CV_16U,1,0); - imwrite("vignetteCalibResult/vignette.png", wrap16); - waitKey(50); - } + if(y0 && cvIsNaN(vignetteFactorCT[idx-wI]) != 1) {sum += vignetteFactorCT[idx-wI]; num++;} + + if(y0 && cvIsNaN(vignetteFactorCT[idx-1+wI]) != 1) {sum += vignetteFactorCT[idx-1+wI]; num++;} + if(x>0 && cvIsNaN(vignetteFactorCT[idx-1]) != 1) {sum += vignetteFactorCT[idx-1]; num++;} + if(y>0 && x>0 && cvIsNaN(vignetteFactorCT[idx-1-wI]) != 1) {sum += vignetteFactorCT[idx-1-wI]; num++;} + + if(num>0) vignetteFactorTT[idx] = sum/num; + } + } } + + // ================================ Store Vignette Image ======================================= + if(!_silent) + displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); + Mat wrapSmoothed = Mat(hI, wI, CV_32F, vignetteFactorTT)*254.9*254.9; + Mat wrapSmoothed16; + wrapSmoothed.convertTo(wrapSmoothed16, CV_16U,1,0); + imwrite("vignetteCalibResult/vignetteSmoothed.png", wrapSmoothed16); + waitKey(50); + + if(!_silent) + displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); + Mat wrap = Mat(hI, wI, CV_32F, vignetteFactor)*254.9*254.9; + Mat wrap16; + wrap.convertTo(wrap16, CV_16U,1,0); + imwrite("vignetteCalibResult/vignette.png", wrap16); + waitKey(50); } logFile.flush(); @@ -490,15 +490,11 @@ void VignetteCalib::calib() delete[] vignetteFactor; delete[] vignetteFactorTT; delete[] vignetteFactorCT; - - for(unsigned long i=0;i Date: Sun, 20 Aug 2017 21:54:05 +0200 Subject: [PATCH 40/53] Code reformat. --- .../include/opencv2/photometric_calib.hpp | 6 +- .../photometric_calib/GammaRemover.hpp | 24 +- .../opencv2/photometric_calib/Reader.hpp | 10 +- .../photometric_calib/ResponseCalib.hpp | 26 +- .../photometric_calib/VignetteCalib.hpp | 28 +- .../photometric_calib/VignetteRemover.hpp | 14 +- .../photometric_calib/src/GammaRemover.cpp | 39 +- modules/photometric_calib/src/Reader.cpp | 35 +- .../photometric_calib/src/ResponseCalib.cpp | 261 +++++---- .../photometric_calib/src/VignetteCalib.cpp | 512 +++++++++++------- .../photometric_calib/src/VignetteRemover.cpp | 64 ++- .../src/photometric_calib.cpp | 6 +- modules/photometric_calib/src/precomp.hpp | 1 + 13 files changed, 610 insertions(+), 416 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp index efbd13f18..e1c9d5052 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -11,7 +11,9 @@ #include "opencv2/photometric_calib/ResponseCalib.hpp" #include "opencv2/photometric_calib/VignetteCalib.hpp" -namespace cv { namespace photometric_calib{ -}} // namespace photometric_calib, cv +namespace cv { +namespace photometric_calib { +} // namespace photometric_calib +} // namespace cv #endif \ No newline at end of file diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp index 5df5ea36c..37b8a8b33 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/GammaRemover.hpp @@ -7,7 +7,8 @@ #include "opencv2/core.hpp" -namespace cv { namespace photometric_calib { +namespace cv { +namespace photometric_calib { //! @addtogroup photometric_calib //! @{ @@ -46,29 +47,34 @@ public: * @brief get gamma function. * @return */ - inline float* getG() + inline float *getG() { - if(!validGamma) return 0; - else return G; + if (!validGamma) + { return 0; } + else + { return G; } }; /*! * @brief get inverse gamma function * @return */ - inline float* getGInv() + inline float *getGInv() { - if(!validGamma) return 0; - else return GInv; + if (!validGamma) + { return 0; } + else + { return GInv; } }; private: float G[256]; float GInv[256]; - int w,h; + int w, h; bool validGamma; }; -}} +} // namespace photometric_calib +} // namespace cv #endif //_OPENCV__GAMMAREMOVER_HPP diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp index 9f78b1f56..b36ad9ccc 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/Reader.hpp @@ -10,7 +10,8 @@ #include #include -namespace cv { namespace photometric_calib{ +namespace cv { +namespace photometric_calib { //! @addtogroup photometric_calib //! @{ @@ -66,8 +67,11 @@ public: float getExposureDuration(unsigned long id) const; int getWidth() const; + int getHeight() const; + const std::string &getFolderPath() const; + const std::string &getTimeFilePath() const; private: @@ -89,5 +93,7 @@ private: //! @} -}} // namespace cv photometric_calib +} // namespace photometric_calib +} // namespace cv + #endif //_OPENCV_READER_HPP \ No newline at end of file diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp index 6bb2c5d13..31103182c 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -7,31 +7,36 @@ #include "opencv2/photometric_calib/Reader.hpp" -namespace cv { namespace photometric_calib { +namespace cv { +namespace photometric_calib { class CV_EXPORTS ResponseCalib { public: ResponseCalib(std::string folderPath, std::string timePath, std::string imageFormat); - ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, std::string imageFormat); - void plotE(const double* E, int w, int h, const std::string &saveTo); + ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, + std::string imageFormat); - Vec2d rmse(const double* G, const double* E, const std::vector &exposureVec, const std::vector &dataVec, int wh); + void plotE(const double *E, int w, int h, const std::string &saveTo); - void plotG(const double* G, const std::string &saveTo); + Vec2d rmse(const double *G, const double *E, const std::vector &exposureVec, + const std::vector &dataVec, int wh); + + void plotG(const double *G, const std::string &saveTo); void calib(); - inline const std::string& getImageFolderPath () const + inline const std::string &getImageFolderPath() const { CV_Assert(imageReader); - return imageReader -> getFolderPath(); + return imageReader->getFolderPath(); } - inline const std::string& getTimeFilePath () const + + inline const std::string &getTimeFilePath() const { CV_Assert(imageReader); - return imageReader -> getTimeFilePath(); + return imageReader->getTimeFilePath(); } private: @@ -41,6 +46,7 @@ private: Reader *imageReader; }; -}} +} // namespace photometric_calib +} // namespace cv #endif //_OPENCV_RESPONSECALIB_HPP diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 749d477ad..44c651773 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -9,25 +9,35 @@ #include "opencv2/photometric_calib/Reader.hpp" #include "opencv2/photometric_calib/GammaRemover.hpp" -namespace cv { namespace photometric_calib { +namespace cv { +namespace photometric_calib { class CV_EXPORTS VignetteCalib { public: - VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent); - VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, int maxIterations, int outlierTh, + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, + std::string imageFormat, bool silent); + + VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, + std::string imageFormat, bool silent, int imageSkip, int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); virtual ~VignetteCalib(); //EIGEN_ALWAYS_INLINE float getInterpolatedElement(const float* const mat, const float x, const float y, const int width) - float getInterpolatedElement(const float* const mat, const float x, const float y, const int width); + float getInterpolatedElement(const float *const mat, const float x, const float y, const int width); + float calMeanExposureTime(); - void displayImage(float* I, int w, int h, std::string name); - void displayImageV(float* I, int w, int h, std::string name); - bool preCalib(unsigned long id, float*& image, float*& plane2imgX, float*& plane2imgY); + + void displayImage(float *I, int w, int h, std::string name); + + void displayImageV(float *I, int w, int h, std::string name); + + bool preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY); + void calib(); + void calibFast(); private: @@ -58,7 +68,7 @@ private: bool _silent; }; -}} - +} // namespace photometric_calib +} // namespace cv #endif //_OPENCV_VIGNETTECALIB_HPP diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp index 8fc4a2555..c041804ca 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -7,7 +7,8 @@ #include "opencv2/core.hpp" -namespace cv { namespace photometric_calib { +namespace cv { +namespace photometric_calib { //! @addtogroup photometric_calib //! @{ @@ -27,6 +28,7 @@ public: * @param h_ the height of input image */ VignetteRemover(const std::string &vignettePath, int w_, int h_); + ~VignetteRemover(); /*! @@ -44,13 +46,13 @@ public: void getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec); private: - float* vignetteMap; - float* vignetteMapInv; - int w,h; + float *vignetteMap; + float *vignetteMapInv; + int w, h; bool validVignette; }; -}} - +} // namespace photometric_calib +} // namespace cv #endif //_OPENCV_VIGNETTEREMOVER_HPP diff --git a/modules/photometric_calib/src/GammaRemover.cpp b/modules/photometric_calib/src/GammaRemover.cpp index b83b921c4..cd8daad6f 100644 --- a/modules/photometric_calib/src/GammaRemover.cpp +++ b/modules/photometric_calib/src/GammaRemover.cpp @@ -5,7 +5,8 @@ #include "precomp.hpp" #include "opencv2/photometric_calib/GammaRemover.hpp" -namespace cv { namespace photometric_calib { +namespace cv { +namespace photometric_calib { GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) { @@ -14,7 +15,8 @@ GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) h = h_; // check the extension of the time file. - CV_Assert(gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yaml" || gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yml"); + CV_Assert(gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yaml" || + gammaPath.substr(gammaPath.find_last_of(".") + 1) == "yml"); FileStorage gammaFile; gammaFile.open(gammaPath, FileStorage::READ); @@ -25,24 +27,26 @@ GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) FileNodeIterator itS = gammaNode.begin(), itE = gammaNode.end(); std::vector GInvVec; for (; itS != itE; ++itS) - GInvVec.push_back((float)*itS); + { + GInvVec.push_back((float) *itS); + } CV_Assert(GInvVec.size() == 256); - for(int i=0;i<256;i++) GInv[i] = GInvVec[i]; - for(int i=0;i<255;i++) + for (int i = 0; i < 256; i++) GInv[i] = GInvVec[i]; + for (int i = 0; i < 255; i++) { - CV_Assert(GInv[i+1] > GInv[i]); + CV_Assert(GInv[i + 1] > GInv[i]); } float min = GInv[0]; float max = GInv[255]; - for(int i=0;i<256;i++) GInv[i] = (float) (255.0 * (GInv[i] - min) / (max - min)); - for(int i=1;i<255;i++) + for (int i = 0; i < 256; i++) GInv[i] = (float) (255.0 * (GInv[i] - min) / (max - min)); + for (int i = 1; i < 255; i++) { - for(int s=1;s<255;s++) + for (int s = 1; s < 255; s++) { - if(GInv[s] <= i && GInv[s+1] >= i) + if (GInv[s] <= i && GInv[s + 1] >= i) { - G[i] = s+(i - GInv[s]) / (GInv[s+1]-GInv[s]); + G[i] = s + (i - GInv[s]) / (GInv[s + 1] - GInv[s]); break; } } @@ -50,20 +54,20 @@ GammaRemover::GammaRemover(const std::string &gammaPath, int w_, int h_) G[0] = 0; G[255] = 255; gammaFile.release(); - validGamma=true; + validGamma = true; } Mat GammaRemover::getUnGammaImageMat(Mat inputIm) { CV_Assert(validGamma); uchar *inputImArr = inputIm.data; - float *outImArr = new float[w*h]; + float *outImArr = new float[w * h]; for (int i = 0; i < w * h; ++i) { outImArr[i] = GInv[inputImArr[i]]; } Mat _outIm(h, w, CV_32F, outImArr); - Mat outIm = _outIm * (1/255.0f); + Mat outIm = _outIm * (1 / 255.0f); delete[] outImArr; return outIm; } @@ -72,8 +76,9 @@ void GammaRemover::getUnGammaImageVec(Mat inputIm, std::vector &outImVec) { CV_Assert(validGamma); uchar *inputImArr = inputIm.data; - CV_Assert(outImVec.size() == (unsigned long)w * h); - for(int i = 0; i < w * h; i++) outImVec[i] = GInv[inputImArr[i]]; + CV_Assert(outImVec.size() == (unsigned long) w * h); + for (int i = 0; i < w * h; i++) outImVec[i] = GInv[inputImArr[i]]; } -}} +} // namespace photometric_calib +} // namespace cv diff --git a/modules/photometric_calib/src/Reader.cpp b/modules/photometric_calib/src/Reader.cpp index 5ff00c12c..02e88524a 100644 --- a/modules/photometric_calib/src/Reader.cpp +++ b/modules/photometric_calib/src/Reader.cpp @@ -5,17 +5,19 @@ #include "precomp.hpp" #include "opencv2/photometric_calib/Reader.hpp" -namespace cv { namespace photometric_calib{ +namespace cv { +namespace photometric_calib { unsigned long Reader::getNumImages() const { - return (unsigned long)images.size(); + return (unsigned long) images.size(); } void Reader::loadTimestamps(const std::string ×File) { // check the extension of the time file. - CV_Assert(timesFile.substr(timesFile.find_last_of(".") + 1) == "yaml" || timesFile.substr(timesFile.find_last_of(".") + 1) == "yml"); + CV_Assert(timesFile.substr(timesFile.find_last_of(".") + 1) == "yaml" || + timesFile.substr(timesFile.find_last_of(".") + 1) == "yml"); FileStorage timeFile; timeFile.open(timesFile, FileStorage::READ); @@ -33,9 +35,13 @@ void Reader::loadTimestamps(const std::string ×File) FileNodeIterator itEt = exposureTimeNode.begin(), itEtEnd = exposureTimeNode.end(); for (; itTs != itTsEnd; ++itTs) - timeStamps.push_back((double)*itTs); + { + timeStamps.push_back((double) *itTs); + } for (; itEt != itEtEnd; ++itEt) - exposureDurations.push_back((float)*itEt); + { + exposureDurations.push_back((float) *itEt); + } timeFile.release(); @@ -63,11 +69,11 @@ Reader::Reader(const std::string &folderPath, const std::string &imageExt, const _height = 0; // images should be of CV_8U and same size - for(size_t i = 0; i < images.size(); ++i) + for (size_t i = 0; i < images.size(); ++i) { Mat img = imread(images[i], IMREAD_GRAYSCALE); CV_Assert(img.type() == CV_8U); - if(i == 0) + if (i == 0) { _width = img.cols; _height = img.rows; @@ -99,20 +105,25 @@ float Reader::getExposureDuration(unsigned long id) const return exposureDurations[id]; } -int Reader::getWidth() const { +int Reader::getWidth() const +{ return _width; } -int Reader::getHeight() const { +int Reader::getHeight() const +{ return _height; } -const std::string &Reader::getFolderPath() const { +const std::string &Reader::getFolderPath() const +{ return _folderPath; } -const std::string &Reader::getTimeFilePath() const { +const std::string &Reader::getTimeFilePath() const +{ return _timeFilePath; } -}} // namespace photometric_calib, cv \ No newline at end of file +} // namespace photometric_calib +} // namespace cv \ No newline at end of file diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 8dc8ff5cc..038afb694 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -9,43 +9,48 @@ #include #include -namespace cv { namespace photometric_calib{ +namespace cv { +namespace photometric_calib { -ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, std::string imageFormat) : _leakPadding(2), _nIts(10), _skipFrames(1) +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, std::string imageFormat) : _leakPadding(2), + _nIts(10), + _skipFrames(1) { imageReader = new Reader(folderPath, imageFormat, timePath); } -ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, std::string imageFormat) : +ResponseCalib::ResponseCalib(std::string folderPath, std::string timePath, int leakPadding, int nIts, int skipFrames, + std::string imageFormat) : _leakPadding(leakPadding), _nIts(nIts), _skipFrames(skipFrames) { imageReader = new Reader(folderPath, imageFormat, timePath); } -Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector &exposureVec, const std::vector &dataVec, +Vec2d ResponseCalib::rmse(const double *G, const double *E, const std::vector &exposureVec, + const std::vector &dataVec, int wh) { - long double e = 0; // yeah - these will be sums of a LOT of values, so we need super high precision. + long double e = 0; // yeah - these will be sums of a LOT of values, so we need super high precision. long double num = 0; size_t n = dataVec.size(); - for(size_t i = 0; i < n; i++) + for (size_t i = 0; i < n; i++) { - for(int k = 0; k < wh; k++) + for (int k = 0; k < wh; k++) { - if(dataVec[i][k] == 255) continue; + if (dataVec[i][k] == 255) continue; double r = G[dataVec[i][k]] - exposureVec[i] * E[k]; - if(!std::isfinite(r)) continue; + if (!std::isfinite(r)) continue; e += r * r * 1e-10; num++; } } //return Eigen::Vector2d(1e5*sqrtl((e/num)), (double)num); - return Vec2d((double)(1e5 *sqrt((e/num))), (double)num); + return Vec2d((double) (1e5 * sqrt((e / num))), (double) num); } -void ResponseCalib::plotE(const double* E, int w, int h, const std::string &saveTo) +void ResponseCalib::plotE(const double *E, int w, int h, const std::string &saveTo) { // try to find some good color scaling for plotting. @@ -54,98 +59,101 @@ void ResponseCalib::plotE(const double* E, int w, int h, const std::string &save double Emin = 1e10, Emax = -1e10; - for(int i = 0; i < w * h; i++) + for (int i = 0; i < w * h; i++) { double le = log(E[i] + offset); - if(le < min) min = le; - if(le > max) max = le; + if (le < min) min = le; + if (le > max) max = le; - if(E[i] < Emin) Emin = E[i]; - if(E[i] > Emax) Emax = E[i]; + if (E[i] < Emin) Emin = E[i]; + if (E[i] > Emax) Emax = E[i]; } - cv::Mat EImg = cv::Mat(h,w,CV_8UC3); - cv::Mat EImg16 = cv::Mat(h,w,CV_16U); + cv::Mat EImg = cv::Mat(h, w, CV_8UC3); + cv::Mat EImg16 = cv::Mat(h, w, CV_16U); - for(int i=0;i(i) = color; - EImg16.at(i) = (ushort)(255 * 255 * (E[i] - Emin) / (Emax - Emin)); + EImg16.at(i) = (ushort) (255 * 255 * (E[i] - Emin) / (Emax - Emin)); } - std::cout<<"Irradiance "< max) max = G[i]; + if (G[i] < min) min = G[i]; + if (G[i] > max) max = G[i]; } - for(int i=0;i<256;i++) + for (int i = 0; i < 256; i++) { - double val = 256*(G[i]-min) / (max-min); - for(int k=0;k<256;k++) + double val = 256 * (G[i] - min) / (max - min); + for (int k = 0; k < 256; k++) { - if(val < k) - GImg.at(k,i) = (float)(k - val); + if (val < k) + { + GImg.at(k, i) = (float) (k - val); + } } } - std::cout<<"Inv. Response "< exposureDurationVec; std::vector dataVec; - for(unsigned long i = 0 ; i < imageReader->getNumImages(); i += _skipFrames) + for (unsigned long i = 0; i < imageReader->getNumImages(); i += _skipFrames) { cv::Mat img = imageReader->getImage(i); - if(img.rows==0 || img.cols==0) continue; + if (img.rows == 0 || img.cols == 0) continue; CV_Assert(img.type() == CV_8U); - if((w!=0 && w != img.cols) || img.cols==0) + if ((w != 0 && w != img.cols) || img.cols == 0) { - std::cout<<"Width mismatch!"<getExposureDuration(i))); + exposureDurationVec.push_back((double) (imageReader->getExposureDuration(i))); - unsigned char* data2 = new unsigned char[w * h]; + unsigned char *data2 = new unsigned char[w * h]; for (int j = 0; j < _leakPadding; ++j) { memcpy(data2, data, w * h); @@ -164,19 +172,19 @@ void ResponseCalib::calib() { for (int x = 1; x < w - 1; ++x) { - if(data[x + y * w] == 255) + if (data[x + y * w] == 255) { - data2[x+1 + w*(y+1)] = 255; - data2[x+1 + w*(y )] = 255; - data2[x+1 + w*(y-1)] = 255; + data2[x + 1 + w * (y + 1)] = 255; + data2[x + 1 + w * (y)] = 255; + data2[x + 1 + w * (y - 1)] = 255; - data2[x + w*(y+1)] = 255; - data2[x + w*(y )] = 255; - data2[x + w*(y-1)] = 255; + data2[x + w * (y + 1)] = 255; + data2[x + w * (y)] = 255; + data2[x + w * (y - 1)] = 255; - data2[x-1 + w*(y+1)] = 255; - data2[x-1 + w*(y )] = 255; - data2[x-1 + w*(y-1)] = 255; + data2[x - 1 + w * (y + 1)] = 255; + data2[x - 1 + w * (y)] = 255; + data2[x - 1 + w * (y - 1)] = 255; } } } @@ -185,122 +193,130 @@ void ResponseCalib::calib() delete[] data2; } n = dataVec.size(); - std::cout<<"Loaded "< 1) G[i] = G[i-1] + (G[i-1]-G[i-2]); + if (!std::isfinite(G[i]) && i > 1) G[i] = G[i - 1] + (G[i - 1] - G[i - 2]); } delete[] GSum; delete[] GNum; - printf("optG RMSE = %f! \t", rmse(G, E, exposureDurationVec, dataVec, w*h )[0]); + printf("optG RMSE = %f! \t", rmse(G, E, exposureDurationVec, dataVec, w * h)[0]); - char buf[1000]; snprintf(buf, 1000, "photoCalibResult/G-%02d.png", it+1); + char buf[1000]; + snprintf(buf, 1000, "photoCalibResult/G-%02d.png", it + 1); plotG(G, buf); } - if(optE) + if (optE) { // optimize scene irradiance function. - double* ESum = new double[w*h]; - double* ENum = new double[w*h]; - memset(ESum,0,w*h*sizeof(double)); - memset(ENum,0,w*h*sizeof(double)); - for(size_t i=0;i #include -namespace cv { namespace photometric_calib{ +namespace cv { +namespace photometric_calib { -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent) : +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, + std::string gammaFile, std::string imageFormat, bool silent) : _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), _maxAbsGrad(255) { imageReader = new Reader(folderPath, imageFormat, timePath); // check the extension of the camera file. - CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); + CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || + cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); FileStorage cameraStorage(cameraFile, FileStorage::READ); cameraStorage["cameraMatrix"] >> _cameraMatrix; cameraStorage["distCoeffs"] >> _distCoeffs; @@ -25,10 +28,10 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: // affine map from plane cordinates to grid coordinates. _K_p2idx = Matx33f::eye(); - _K_p2idx(0,0) = _gridWidth / _facW; - _K_p2idx(1,1) = _gridHeight / _facH; - _K_p2idx(0,2) = _gridWidth / 2.f; - _K_p2idx(1,2) = _gridHeight / 2.f; + _K_p2idx(0, 0) = _gridWidth / _facW; + _K_p2idx(1, 1) = _gridHeight / _facH; + _K_p2idx(0, 2) = _gridWidth / 2.f; + _K_p2idx(1, 2) = _gridHeight / 2.f; _K_p2idx_inverse = _K_p2idx.inv(); _meanExposure = calMeanExposureTime(); @@ -36,14 +39,18 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: _silent = silent; } -VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, int maxIterations, +VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, + std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, + int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : - _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), _gridHeight(gridHeight), + _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), + _gridHeight(gridHeight), _facW(facW), _facH(facH), _maxAbsGrad(maxAbsGrad) { imageReader = new Reader(folderPath, imageFormat, timePath); // check the extension of the camera file. - CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); + CV_Assert(cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yaml" || + cameraFile.substr(cameraFile.find_last_of(".") + 1) == "yml"); FileStorage cameraStorage(cameraFile, FileStorage::READ); cameraStorage["cameraMatrix"] >> _cameraMatrix; cameraStorage["distCoeffs"] >> _distCoeffs; @@ -51,10 +58,10 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: // affine map from plane cordinates to grid coordinates. _K_p2idx = Matx33f::eye(); - _K_p2idx(0,0) = _gridWidth / _facW; - _K_p2idx(1,1) = _gridHeight / _facH; - _K_p2idx(0,2) = _gridWidth / 2.f; - _K_p2idx(1,2) = _gridHeight / 2.f; + _K_p2idx(0, 0) = _gridWidth / _facW; + _K_p2idx(1, 1) = _gridHeight / _facH; + _K_p2idx(0, 2) = _gridWidth / 2.f; + _K_p2idx(1, 2) = _gridHeight / 2.f; _K_p2idx_inverse = _K_p2idx.inv(); _meanExposure = calMeanExposureTime(); @@ -64,42 +71,48 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: float VignetteCalib::getInterpolatedElement(const float *const mat, const float x, const float y, const int width) { - int ix = (int)x; - int iy = (int)y; + int ix = (int) x; + int iy = (int) y; float dx = x - ix; float dy = y - iy; - float dxdy = dx*dy; - const float* bp = mat +ix+iy*width; + float dxdy = dx * dy; + const float *bp = mat + ix + iy * width; - float res = dxdy * bp[1+width] - + (dy-dxdy) * bp[width] - + (dx-dxdy) * bp[1] - + (1-dx-dy+dxdy) * bp[0]; + float res = dxdy * bp[1 + width] + + (dy - dxdy) * bp[width] + + (dx - dxdy) * bp[1] + + (1 - dx - dy + dxdy) * bp[0]; return res; } void VignetteCalib::displayImage(float *I, int w, int h, std::string name) { - float vmin=1e10; - float vmax=-1e10; + float vmin = 1e10; + float vmax = -1e10; - for(int i=0;i I[i]) vmin = I[i]; - if(vmax < I[i]) vmax = I[i]; + if (vmin > I[i]) vmin = I[i]; + if (vmax < I[i]) vmax = I[i]; } - Mat img = Mat(h,w,CV_8UC3); + Mat img = Mat(h, w, CV_8UC3); - for(int i=0;i(i) = Vec3b(0,0,255); - else img.at(i) = Vec3b((uchar)(255*(I[i]-vmin) / (vmax-vmin)),(uchar)(255*(I[i]-vmin) / (vmax-vmin)),(uchar)(255*(I[i]-vmin) / (vmax-vmin))); + if (cvIsNaN(I[i]) == 1) + { img.at(i) = Vec3b(0, 0, 255); } + else + { + img.at(i) = Vec3b((uchar) (255 * (I[i] - vmin) / (vmax - vmin)), + (uchar) (255 * (I[i] - vmin) / (vmax - vmin)), + (uchar) (255 * (I[i] - vmin) / (vmax - vmin))); + } } - std::cout<<"plane image values " << vmin << " - " << vmax << "!" <(i) = Vec3b(0,0,255); + if (cvIsNaN(I[i]) == 1) + { + img.at(i) = Vec3b(0, 0, 255); + } else { - float c = 254*I[i]; - img.at(i) = Vec3b((uchar)c,(uchar)c,(uchar)c); + float c = 254 * I[i]; + img.at(i) = Vec3b((uchar) c, (uchar) c, (uchar) c); } } @@ -125,20 +140,22 @@ void VignetteCalib::displayImageV(float *I, int w, int h, std::string name) float VignetteCalib::calMeanExposureTime() { float meanExposure = 0.f; - for(unsigned long i=0;igetNumImages();i+=_imageSkip) - meanExposure+=imageReader->getExposureDuration(i); - meanExposure = meanExposure/imageReader->getNumImages(); - if(meanExposure==0) meanExposure = 1; + for (unsigned long i = 0; i < imageReader->getNumImages(); i += _imageSkip) + { + meanExposure += imageReader->getExposureDuration(i); + } + meanExposure = meanExposure / imageReader->getNumImages(); + if (meanExposure == 0) meanExposure = 1; return meanExposure; } -bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX, float*& plane2imgY) +bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY) { int wI = imageReader->getWidth(); int hI = imageReader->getHeight(); std::vector markerIds; - std::vector< std::vector > markerCorners, rejectedCandidates; + std::vector > markerCorners, rejectedCandidates; Mat oriImg = imageReader->getImage(id); Mat undisImg; @@ -148,8 +165,10 @@ bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_ARUCO_ORIGINAL); aruco::detectMarkers(undisImg, dictionary, markerCorners, markerIds, parameters, rejectedCandidates); - if(markerCorners.size() != 1) + if (markerCorners.size() != 1) + { return false; + } std::vector ptsP; std::vector ptsI; @@ -157,24 +176,26 @@ bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX ptsI.push_back(Point2f(markerCorners[0][1].x, markerCorners[0][1].y)); ptsI.push_back(Point2f(markerCorners[0][2].x, markerCorners[0][2].y)); ptsI.push_back(Point2f(markerCorners[0][3].x, markerCorners[0][3].y)); - ptsP.push_back(Point2f(-0.5,0.5)); - ptsP.push_back(Point2f(0.5,0.5)); - ptsP.push_back(Point2f(0.5,-0.5)); - ptsP.push_back(Point2f(-0.5,-0.5)); + ptsP.push_back(Point2f(-0.5, 0.5)); + ptsP.push_back(Point2f(0.5, 0.5)); + ptsP.push_back(Point2f(0.5, -0.5)); + ptsP.push_back(Point2f(-0.5, -0.5)); Matx33f H = findHomography(ptsP, ptsI); std::vector imgRaw(wI * hI); gammaRemover->getUnGammaImageVec(imageReader->getImage(id), imgRaw); - plane2imgX = new float[_gridWidth*_gridHeight]; - plane2imgY = new float[_gridWidth*_gridHeight]; + plane2imgX = new float[_gridWidth * _gridHeight]; + plane2imgY = new float[_gridWidth * _gridHeight]; Matx33f HK = H * _K_p2idx_inverse; - int idx=0; - for(int y=0;y<_gridHeight;y++) { - for (int x = 0; x < _gridWidth; x++) { + int idx = 0; + for (int y = 0; y < _gridHeight; y++) + { + for (int x = 0; x < _gridWidth; x++) + { Vec3f pp = HK * Vec3f((float) x, (float) y, 1); plane2imgX[idx] = pp[0] / pp[2]; plane2imgY[idx] = pp[1] / pp[2]; @@ -184,31 +205,34 @@ bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX float expDuration; expDuration = (imageReader->getExposureDuration(id) == 0 ? 1 : imageReader->getExposureDuration(id)); - image = new float[wI*hI]; - for(int y=0; y _maxAbsGrad) + if (fabsf(image[x + y * wI] - image[x + deltax + (y + deltay) * wI]) > _maxAbsGrad) { - image[x+y*wI] = NAN; image[x+deltax+(y+deltay)*wI]=NAN; + image[x + y * wI] = NAN; + image[x + deltax + (y + deltay) * wI] = NAN; } } + } } } - for(int x=0; x<_gridWidth;x++) + for (int x = 0; x < _gridWidth; x++) { for (int y = 0; y < _gridHeight; y++) { @@ -222,55 +246,69 @@ bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX } } - if(!_silent) + if (!_silent) + { + // debug-plot. + Mat dbgImg(hI, wI, CV_8UC3); + for (int j = 0; j < wI * hI; j++) { - // debug-plot. - Mat dbgImg(hI, wI, CV_8UC3); - for(int j=0;j(j) = Vec3b((uchar)imgRaw[j], (uchar)imgRaw[j], (uchar)imgRaw[j]); + dbgImg.at(j) = Vec3b((uchar) imgRaw[j], (uchar) imgRaw[j], (uchar) imgRaw[j]); + } - for(int x=0; x<=_gridWidth;x+=200) - for(int y=0; y<=_gridHeight;y+=10) - { - int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; - int idxT = (x<_gridWidth ? x : _gridWidth-1)+((y+10)<_gridHeight ? (y+10) : _gridHeight-1)*_gridWidth; + for (int x = 0; x <= _gridWidth; x += 200) + { + for (int y = 0; y <= _gridHeight; y += 10) + { + int idxS = (x < _gridWidth ? x : _gridWidth - 1) + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; + int idxT = (x < _gridWidth ? x : _gridWidth - 1) + + ((y + 10) < _gridHeight ? (y + 10) : _gridHeight - 1) * _gridWidth; - int u_dS = (int)lround((plane2imgX[idxS]+0.5)); - int v_dS = (int)lround((plane2imgY[idxS]+0.5)); + int u_dS = (int) lround((plane2imgX[idxS] + 0.5)); + int v_dS = (int) lround((plane2imgY[idxS] + 0.5)); - int u_dT = (int)lround((plane2imgX[idxT]+0.5)); - int v_dT = (int)lround((plane2imgY[idxT]+0.5)); + int u_dT = (int) lround((plane2imgX[idxT] + 0.5)); + int v_dT = (int) lround((plane2imgY[idxT] + 0.5)); - if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT= 0 && v_dS >= 0 && u_dS < wI && v_dS < hI && u_dT >= 0 && v_dT >= 0 && u_dT < wI && + v_dT < hI) + { + line(dbgImg, Point(u_dS, v_dS), Point(u_dT, v_dT), Scalar(0, 0, 255), 10, LINE_AA); } + } + } - for(int x=0; x<=_gridWidth;x+=10) - for(int y=0; y<=_gridHeight;y+=200) - { - int idxS = (x<_gridWidth ? x : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; - int idxT = ((x+10)<_gridWidth ? (x+10) : _gridWidth-1)+(y<_gridHeight ? y : _gridHeight-1)*_gridWidth; + for (int x = 0; x <= _gridWidth; x += 10) + { + for (int y = 0; y <= _gridHeight; y += 200) + { + int idxS = (x < _gridWidth ? x : _gridWidth - 1) + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; + int idxT = ((x + 10) < _gridWidth ? (x + 10) : _gridWidth - 1) + + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; - int u_dS = (int)lround(plane2imgX[idxS]+0.5); - int v_dS = (int)lround(plane2imgY[idxS]+0.5); + int u_dS = (int) lround(plane2imgX[idxS] + 0.5); + int v_dS = (int) lround(plane2imgY[idxS] + 0.5); - int u_dT = (int)lround(plane2imgX[idxT]+0.5); - int v_dT = (int)lround(plane2imgY[idxT]+0.5); + int u_dT = (int) lround(plane2imgX[idxT] + 0.5); + int v_dT = (int) lround(plane2imgY[idxT] + 0.5); - if(u_dS>=0 && v_dS >=0 && u_dS=0 && v_dT >=0 && u_dT= 0 && v_dS >= 0 && u_dS < wI && v_dS < hI && u_dT >= 0 && v_dT >= 0 && u_dT < wI && + v_dT < hI) + { + line(dbgImg, Point(u_dS, v_dS), Point(u_dT, v_dT), Scalar(0, 0, 255), 10, LINE_AA); } + } + } - imshow("inRaw",dbgImg); - waitKey(1); + imshow("inRaw", dbgImg); + waitKey(1); - if(rand()%40==0) - { - char buf[1000]; - snprintf(buf,1000,"vignetteCalibResult/img%u.png",(unsigned)id); - imwrite(buf, dbgImg); - } + if (rand() % 40 == 0) + { + char buf[1000]; + snprintf(buf, 1000, "vignetteCalibResult/img%u.png", (unsigned) id); + imwrite(buf, dbgImg); } + } return true; } @@ -278,8 +316,14 @@ bool VignetteCalib::preCalib(unsigned long id, float*& image, float*& plane2imgX void VignetteCalib::calib() { // Create folder for vignette calibration - if(-1 == system("rm -rf vignetteCalibResult")) std::cout<< "could not delete old vignetteCalibResult folder!" << std::endl; - if(-1 == system("mkdir vignetteCalibResult")) std::cout<< "could not delete old vignetteCalibResult folder" << std::endl; + if (-1 == system("rm -rf vignetteCalibResult")) + { + std::cout << "could not delete old vignetteCalibResult folder!" << std::endl; + } + if (-1 == system("mkdir vignetteCalibResult")) + { + std::cout << "could not delete old vignetteCalibResult folder" << std::endl; + } int wO, hO; wO = imageReader->getWidth(); @@ -294,189 +338,247 @@ void VignetteCalib::calib() // number of images in total unsigned long n = imageReader->getNumImages(); - float* planeColor = new float[_gridWidth*_gridHeight]; - float* planeColorFF = new float[_gridWidth*_gridHeight]; - float* planeColorFC = new float[_gridWidth*_gridHeight]; - float* vignetteFactor = new float[hI*wI]; - float* vignetteFactorTT = new float[hI*wI]; - float* vignetteFactorCT = new float[hI*wI]; + float *planeColor = new float[_gridWidth * _gridHeight]; + float *planeColorFF = new float[_gridWidth * _gridHeight]; + float *planeColorFC = new float[_gridWidth * _gridHeight]; + float *vignetteFactor = new float[hI * wI]; + float *vignetteFactorTT = new float[hI * wI]; + float *vignetteFactorCT = new float[hI * wI]; // initialize vignette factors to 1. - for(int i=0;i oth2) + if (cvIsNaN(fac) == 1) continue; + if (cvIsNaN(color) == 1) continue; + double residual = (double) ((color - planeColor[pi] * fac) * (color - planeColor[pi] * fac)); + if (abs(residual) > oth2) { E += oth2; - R ++; + R++; continue; } - planeColorFF[pi] += fac*fac; - planeColorFC[pi] += color*fac; - if(cvIsNaN(planeColor[pi]) == 1) continue; + planeColorFF[pi] += fac * fac; + planeColorFC[pi] += color * fac; + if (cvIsNaN(planeColor[pi]) == 1) continue; E += residual; - R ++; + R++; } delete[] plane2imgX; delete[] plane2imgY; delete[] image; } - for(int pi=0;pi<_gridWidth*_gridWidth;pi++) // for all plane points + for (int pi = 0; pi < _gridWidth * _gridWidth; pi++) // for all plane points { - if(planeColorFF[pi] < 1) - planeColor[pi]=NAN; + if (planeColorFF[pi] < 1) + { + planeColor[pi] = NAN; + } else + { planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; + } } - if(!_silent) + if (!_silent) + { displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); - std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; + } + std::cout << R << " residual terms => " << sqrt(E / R) << std::endl; // ================================ optimize vignette ======================================= - std::cout<<"Optimize Vignette..."< oth2) + if (cvIsNaN(colorPlane) == 1) continue; + if (cvIsNaN(colorImage) == 1) continue; + double residual = (double) ((colorImage - colorPlane * fac) * (colorImage - colorPlane * fac)); + if (abs(residual) > oth2) { E += oth2; - R ++; + R++; continue; } - int ix = (int)x; - int iy = (int)y; + int ix = (int) x; + int iy = (int) y; float dx = x - ix; float dy = y - iy; - float dxdy = dx*dy; - vignetteFactorTT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorPlane*colorPlane; - vignetteFactorTT[ix+iy*wI + 1] += (dx-dxdy) * colorPlane*colorPlane; - vignetteFactorTT[ix+iy*wI + wI] += (dy-dxdy) * colorPlane*colorPlane; - vignetteFactorTT[ix+iy*wI + 1+wI] += dxdy * colorPlane*colorPlane; - vignetteFactorCT[ix+iy*wI + 0] += (1-dx-dy+dxdy) * colorImage*colorPlane; - vignetteFactorCT[ix+iy*wI + 1] += (dx-dxdy) * colorImage*colorPlane; - vignetteFactorCT[ix+iy*wI + wI] += (dy-dxdy) * colorImage*colorPlane; - vignetteFactorCT[ix+iy*wI + 1+wI] += dxdy * colorImage*colorPlane; - if(cvIsNaN(fac) == 1) continue; + float dxdy = dx * dy; + vignetteFactorTT[ix + iy * wI + 0] += (1 - dx - dy + dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + 1] += (dx - dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + wI] += (dy - dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + 1 + wI] += dxdy * colorPlane * colorPlane; + vignetteFactorCT[ix + iy * wI + 0] += (1 - dx - dy + dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + 1] += (dx - dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + wI] += (dy - dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + 1 + wI] += dxdy * colorImage * colorPlane; + if (cvIsNaN(fac) == 1) continue; E += residual; - R ++; + R++; } delete[] plane2imgX; delete[] plane2imgY; delete[] image; } - float maxFac=0; - for(int pi=0;pimaxFac) maxFac=vignetteFactor[pi]; + if (vignetteFactor[pi] > maxFac) maxFac = vignetteFactor[pi]; } } - std::cout << R << " residual terms => " << sqrt(E/R) << std::endl; + std::cout << R << " residual terms => " << sqrt(E / R) << std::endl; // normalize to vignette max. factor 1. - for(int pi=0;pi0 && cvIsNaN(vignetteFactorCT[idx+1-wI]) != 1) {sum += vignetteFactorCT[idx+1-wI]; num++;} - - if(y0 && cvIsNaN(vignetteFactorCT[idx-wI]) != 1) {sum += vignetteFactorCT[idx-wI]; num++;} - - if(y0 && cvIsNaN(vignetteFactorCT[idx-1+wI]) != 1) {sum += vignetteFactorCT[idx-1+wI]; num++;} - if(x>0 && cvIsNaN(vignetteFactorCT[idx-1]) != 1) {sum += vignetteFactorCT[idx-1]; num++;} - if(y>0 && x>0 && cvIsNaN(vignetteFactorCT[idx-1-wI]) != 1) {sum += vignetteFactorCT[idx-1-wI]; num++;} - - if(num>0) vignetteFactorTT[idx] = sum/num; + float sum = 0, num = 0; + if (x < wI - 1 && y < hI - 1 && cvIsNaN(vignetteFactorCT[idx + 1 + wI]) != 1) + { + sum += vignetteFactorCT[idx + 1 + wI]; + num++; + } + if (x < wI - 1 && cvIsNaN(vignetteFactorCT[idx + 1]) != 1) + { + sum += vignetteFactorCT[idx + 1]; + num++; + } + if (x < wI - 1 && y > 0 && cvIsNaN(vignetteFactorCT[idx + 1 - wI]) != 1) + { + sum += vignetteFactorCT[idx + 1 - wI]; + num++; + } + + if (y < hI - 1 && cvIsNaN(vignetteFactorCT[idx + wI]) != 1) + { + sum += vignetteFactorCT[idx + wI]; + num++; + } + if (cvIsNaN(vignetteFactorCT[idx]) != 1) + { + sum += vignetteFactorCT[idx]; + num++; + } + if (y > 0 && cvIsNaN(vignetteFactorCT[idx - wI]) != 1) + { + sum += vignetteFactorCT[idx - wI]; + num++; + } + + if (y < hI - 1 && x > 0 && cvIsNaN(vignetteFactorCT[idx - 1 + wI]) != 1) + { + sum += vignetteFactorCT[idx - 1 + wI]; + num++; + } + if (x > 0 && cvIsNaN(vignetteFactorCT[idx - 1]) != 1) + { + sum += vignetteFactorCT[idx - 1]; + num++; + } + if (y > 0 && x > 0 && cvIsNaN(vignetteFactorCT[idx - 1 - wI]) != 1) + { + sum += vignetteFactorCT[idx - 1 - wI]; + num++; + } + + if (num > 0) vignetteFactorTT[idx] = sum / num; } } + } } // ================================ Store Vignette Image ======================================= - if(!_silent) + if (!_silent) + { displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); - Mat wrapSmoothed = Mat(hI, wI, CV_32F, vignetteFactorTT)*254.9*254.9; + } + Mat wrapSmoothed = Mat(hI, wI, CV_32F, vignetteFactorTT) * 254.9 * 254.9; Mat wrapSmoothed16; - wrapSmoothed.convertTo(wrapSmoothed16, CV_16U,1,0); + wrapSmoothed.convertTo(wrapSmoothed16, CV_16U, 1, 0); imwrite("vignetteCalibResult/vignetteSmoothed.png", wrapSmoothed16); waitKey(50); - if(!_silent) + if (!_silent) + { displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); - Mat wrap = Mat(hI, wI, CV_32F, vignetteFactor)*254.9*254.9; + } + Mat wrap = Mat(hI, wI, CV_32F, vignetteFactor) * 254.9 * 254.9; Mat wrap16; - wrap.convertTo(wrap16, CV_16U,1,0); + wrap.convertTo(wrap16, CV_16U, 1, 0); imwrite("vignetteCalibResult/vignette.png", wrap16); waitKey(50); } @@ -500,4 +602,6 @@ VignetteCalib::~VignetteCalib() delete imageReader; delete gammaRemover; } -}}// namespace photometric_calib, cv \ No newline at end of file + +} // namespace photometric_calib +} // namespace cv \ No newline at end of file diff --git a/modules/photometric_calib/src/VignetteRemover.cpp b/modules/photometric_calib/src/VignetteRemover.cpp index f83eb435f..2ccc55912 100644 --- a/modules/photometric_calib/src/VignetteRemover.cpp +++ b/modules/photometric_calib/src/VignetteRemover.cpp @@ -5,15 +5,16 @@ #include "precomp.hpp" #include "opencv2/photometric_calib/VignetteRemover.hpp" -namespace cv { namespace photometric_calib{ +namespace cv { +namespace photometric_calib { VignetteRemover::VignetteRemover(const std::string &vignettePath, int w_, int h_) { CV_Assert(vignettePath != ""); - validVignette=false; - vignetteMap=0; - vignetteMapInv=0; + validVignette = false; + vignetteMap = 0; + vignetteMapInv = 0; w = w_; h = h_; @@ -25,28 +26,38 @@ VignetteRemover::VignetteRemover(const std::string &vignettePath, int w_, int h_ CV_Assert(vignetteMat.rows == h && vignetteMat.cols == w); CV_Assert(vignetteMat.type() == CV_8U || vignetteMat.type() == CV_16U); - if(vignetteMat.type() == CV_8U) + if (vignetteMat.type() == CV_8U) { - float maxV=0; - for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); - - for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); + } + + for (int i = 0; i < w * h; i++) + { vignetteMap[i] = vignetteMat.at(i) / maxV; + } } else { - float maxV=0; - for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); - - for(int i=0;i(i) > maxV) maxV = vignetteMat.at(i); + } + + for (int i = 0; i < w * h; i++) + { vignetteMap[i] = vignetteMat.at(i) / maxV; + } } - for(int i=0;i &unGammaImVec) getUnVignetteImageVec(unGammaImVec, _outImVec); Mat _outIm(h, w, CV_32F, &_outImVec[0]); - Mat outIm = _outIm * (1/255.0f); + Mat outIm = _outIm * (1 / 255.0f); return outIm; } void VignetteRemover::getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec) { CV_Assert(validVignette); - CV_Assert(outImVec.size() == (unsigned long)w * h); + CV_Assert(outImVec.size() == (unsigned long) w * h); for (int i = 0; i < w * h; ++i) + { outImVec[i] = unGammaImVec[i] * vignetteMapInv[i]; + } } -VignetteRemover::~VignetteRemover() { - if(vignetteMap != 0) +VignetteRemover::~VignetteRemover() +{ + if (vignetteMap != 0) + { delete[] vignetteMap; - if(vignetteMapInv != 0) + } + if (vignetteMapInv != 0) + { delete[] vignetteMapInv; + } } - -}} +} // namespace photometric_calib +} // namespace cv diff --git a/modules/photometric_calib/src/photometric_calib.cpp b/modules/photometric_calib/src/photometric_calib.cpp index d7331d444..69f5564d8 100644 --- a/modules/photometric_calib/src/photometric_calib.cpp +++ b/modules/photometric_calib/src/photometric_calib.cpp @@ -4,5 +4,7 @@ #include "precomp.hpp" -namespace cv{ namespace photometric_calib{ -}} // namespace photometric_calib, cv \ No newline at end of file +namespace cv { +namespace photometric_calib { +} +} // namespace photometric_calib, cv \ No newline at end of file diff --git a/modules/photometric_calib/src/precomp.hpp b/modules/photometric_calib/src/precomp.hpp index 1cdc18e82..7c8ae91d9 100644 --- a/modules/photometric_calib/src/precomp.hpp +++ b/modules/photometric_calib/src/precomp.hpp @@ -11,4 +11,5 @@ #include "opencv2/aruco.hpp" #include "opencv2/calib3d.hpp" #include + #endif \ No newline at end of file From 67729326cc243209b53411d1bd8bec9960dded64 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Sun, 20 Aug 2017 22:19:02 +0200 Subject: [PATCH 41/53] Time-efficient vignette calibration implementation. --- .../photometric_calib/src/VignetteCalib.cpp | 460 +++++++++++++++++- 1 file changed, 458 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 81bade2ae..0e90a75b1 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -181,7 +181,7 @@ bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX ptsP.push_back(Point2f(0.5, -0.5)); ptsP.push_back(Point2f(-0.5, -0.5)); - Matx33f H = findHomography(ptsP, ptsI); + Matx33f H(findHomography(ptsP, ptsI)); std::vector imgRaw(wI * hI); gammaRemover->getUnGammaImageVec(imageReader->getImage(id), imgRaw); @@ -595,7 +595,463 @@ void VignetteCalib::calib() } void VignetteCalib::calibFast() -{} +{ + if (-1 == system("rm -rf vignetteCalibResult")) + { + std::cout << "could not delete old vignetteCalibResult folder!" << std::endl; + } + if (-1 == system("mkdir vignetteCalibResult")) + { + std::cout << "could not delete old vignetteCalibResult folder" << std::endl; + } + + int wO, hO; + wO = imageReader->getWidth(); + hO = imageReader->getHeight(); + int wI = wO, hI = hO; + + Ptr parameters = aruco::DetectorParameters::create(); + Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_6X6_250); + + std::vector images; + std::vector p2imgX; + std::vector p2imgY; + + for (unsigned long i = 0; i < imageReader->getNumImages(); ++i) + { + std::vector markerIds; + std::vector > markerCorners, rejectedCandidates; + + Mat oriImg = imageReader->getImage(i); + Mat undisImg; + undistort(oriImg, undisImg, _cameraMatrix, _distCoeffs); + + aruco::detectMarkers(undisImg, dictionary, markerCorners, markerIds, parameters, rejectedCandidates); + + if (markerCorners.size() != 1) continue; + + std::vector ptsP; + std::vector ptsI; + ptsI.push_back(Point2f(markerCorners[0][0].x, markerCorners[0][0].y)); + ptsI.push_back(Point2f(markerCorners[0][1].x, markerCorners[0][1].y)); + ptsI.push_back(Point2f(markerCorners[0][2].x, markerCorners[0][2].y)); + ptsI.push_back(Point2f(markerCorners[0][3].x, markerCorners[0][3].y)); + ptsP.push_back(Point2f(-0.5, 0.5)); + ptsP.push_back(Point2f(0.5, 0.5)); + ptsP.push_back(Point2f(0.5, -0.5)); + ptsP.push_back(Point2f(-0.5, -0.5)); + + Matx33f H(findHomography(ptsP, ptsI)); + + std::vector imgRaw(wI * hI); + gammaRemover->getUnGammaImageVec(imageReader->getImage(i), imgRaw); + + float *plane2imgX = new float[_gridWidth * _gridHeight]; + float *plane2imgY = new float[_gridWidth * _gridHeight]; + + Matx33f HK = H * _K_p2idx_inverse; + + int idx = 0; + for (int y = 0; y < _gridHeight; y++) + { + for (int x = 0; x < _gridWidth; x++) + { + Vec3f pp = HK * Vec3f((float) x, (float) y, 1); + plane2imgX[idx] = pp[0] / pp[2]; + plane2imgY[idx] = pp[1] / pp[2]; + idx++; + } + } + + float expDuration; + expDuration = (imageReader->getExposureDuration(i) == 0 ? 1 : imageReader->getExposureDuration(i)); + float *image = new float[wI * hI]; + for (int y = 0; y < hI; y++) + { + for (int x = 0; x < wI; x++) + { + image[x + y * wI] = _meanExposure * imgRaw[x + y * wI] / expDuration; + } + } + + for (int y = 2; y < hI - 2; y++) + { + for (int x = 2; x < wI - 2; x++) + { + for (int deltax = -2; deltax < 3; deltax++) + { + for (int deltay = -2; deltay < 3; deltay++) + { + if (fabsf(image[x + y * wI] - image[x + deltax + (y + deltay) * wI]) > _maxAbsGrad) + { + image[x + y * wI] = NAN; + image[x + deltax + (y + deltay) * wI] = NAN; + } + } + } + } + } + + images.push_back(image); + + for (int x = 0; x < _gridWidth; x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) + { + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } + + if (!_silent) + { + // debug-plot. + Mat dbgImg(hI, wI, CV_8UC3); + for (int j = 0; j < wI * hI; j++) + { + dbgImg.at(j) = Vec3b((uchar) imgRaw[j], (uchar) imgRaw[j], (uchar) imgRaw[j]); + } + + for (int x = 0; x <= _gridWidth; x += 200) + { + for (int y = 0; y <= _gridHeight; y += 10) + { + int idxS = (x < _gridWidth ? x : _gridWidth - 1) + + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; + int idxT = (x < _gridWidth ? x : _gridWidth - 1) + + ((y + 10) < _gridHeight ? (y + 10) : _gridHeight - 1) * _gridWidth; + + int u_dS = (int) lround((plane2imgX[idxS] + 0.5)); + int v_dS = (int) lround((plane2imgY[idxS] + 0.5)); + + int u_dT = (int) lround((plane2imgX[idxT] + 0.5)); + int v_dT = (int) lround((plane2imgY[idxT] + 0.5)); + + if (u_dS >= 0 && v_dS >= 0 && u_dS < wI && v_dS < hI && u_dT >= 0 && v_dT >= 0 && u_dT < wI && + v_dT < hI) + { + line(dbgImg, Point(u_dS, v_dS), Point(u_dT, v_dT), Scalar(0, 0, 255), 10, LINE_AA); + } + } + } + + for (int x = 0; x <= _gridWidth; x += 10) + { + for (int y = 0; y <= _gridHeight; y += 200) + { + int idxS = (x < _gridWidth ? x : _gridWidth - 1) + + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; + int idxT = ((x + 10) < _gridWidth ? (x + 10) : _gridWidth - 1) + + (y < _gridHeight ? y : _gridHeight - 1) * _gridWidth; + + int u_dS = (int) lround(plane2imgX[idxS] + 0.5); + int v_dS = (int) lround(plane2imgY[idxS] + 0.5); + + int u_dT = (int) lround(plane2imgX[idxT] + 0.5); + int v_dT = (int) lround(plane2imgY[idxT] + 0.5); + + if (u_dS >= 0 && v_dS >= 0 && u_dS < wI && v_dS < hI && u_dT >= 0 && v_dT >= 0 && u_dT < wI && + v_dT < hI) + { + line(dbgImg, Point(u_dS, v_dS), Point(u_dT, v_dT), Scalar(0, 0, 255), 10, LINE_AA); + } + } + } + + imshow("inRaw", dbgImg); + + if (rand() % 40 == 0) + { + char buf[1000]; + snprintf(buf, 1000, "vignetteCalibResult/img%u.png", (unsigned) i); + imwrite(buf, dbgImg); + } + + waitKey(1); + } + + p2imgX.push_back(plane2imgX); + p2imgY.push_back(plane2imgY); + + } + + std::ofstream logFile; + logFile.open("vignetteCalibResult/log.txt", std::ios::trunc | std::ios::out); + logFile.precision(15); + + unsigned long n = imageReader->getNumImages(); + + float *planeColor = new float[_gridWidth * _gridHeight]; + float *planeColorFF = new float[_gridWidth * _gridHeight]; + float *planeColorFC = new float[_gridWidth * _gridHeight]; + float *vignetteFactor = new float[hI * wI]; + float *vignetteFactorTT = new float[hI * wI]; + float *vignetteFactorCT = new float[hI * wI]; + + // initialize vignette factors to 1. + for (int i = 0; i < hI * wI; i++) vignetteFactor[i] = 1; + + double E = 0; + double R = 0; + + for (int it = 0; it < _maxIterations; it++) + { + int oth2 = _outlierTh * _outlierTh; + if (it < _maxIterations / 2) oth2 = 10000 * 10000; + + // ============================ optimize planeColor ================================ + memset(planeColorFF, 0, _gridWidth * _gridHeight * sizeof(float)); + memset(planeColorFC, 0, _gridWidth * _gridHeight * sizeof(float)); + E = 0; + R = 0; + + // for each plane pixel, it's optimum is at sum(CF)/sum(FF) + for (unsigned long img = 0; img < n; img++) // for all images + { + float *plane2imgX = p2imgX[img]; + float *plane2imgY = p2imgY[img]; + float *image = images[img]; + + for (int pi = 0; pi < _gridWidth * _gridHeight; pi++) // for all plane points + { + if (cvIsNaN(plane2imgX[pi]) == 1) continue; + + // get vignetted color at that point, and add to build average. + float color = getInterpolatedElement(image, plane2imgX[pi], plane2imgY[pi], wI); + float fac = getInterpolatedElement(vignetteFactor, plane2imgX[pi], plane2imgY[pi], wI); + + if (cvIsNaN(fac) == 1) continue; + if (cvIsNaN(color) == 1) continue; + + double residual = (double) ((color - planeColor[pi] * fac) * (color - planeColor[pi] * fac)); + if (abs(residual) > oth2) + { + E += oth2; + R++; + continue; + } + + + planeColorFF[pi] += fac * fac; + planeColorFC[pi] += color * fac; + + if (cvIsNaN(planeColor[pi]) == 1) continue; + E += residual; + R++; + } + } + + for (int pi = 0; pi < _gridWidth * _gridWidth; pi++) // for all plane points + { + if (planeColorFF[pi] < 1) + { + planeColor[pi] = NAN; + } + else + { + planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; + } + } + if (!_silent) + { + displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); + } + + std::cout << R << " residual terms => " << sqrt(E / R) << std::endl; + + // ================================ optimize vignette ======================================= + memset(vignetteFactorTT, 0, hI * wI * sizeof(float)); + memset(vignetteFactorCT, 0, hI * wI * sizeof(float)); + E = 0; + R = 0; + + for (unsigned long img = 0; img < n; img++) // for all images + { + float *plane2imgX = p2imgX[img]; + float *plane2imgY = p2imgY[img]; + float *image = images[img]; + + for (int pi = 0; pi < _gridWidth * _gridWidth; pi++) // for all plane points + { + if (cvIsNaN(plane2imgX[pi]) == 1) continue; + float x = plane2imgX[pi]; + float y = plane2imgY[pi]; + + float colorImage = getInterpolatedElement(image, x, y, wI); + float fac = getInterpolatedElement(vignetteFactor, x, y, wI); + float colorPlane = planeColor[pi]; + + if (cvIsNaN(colorPlane) == 1) continue; + if (cvIsNaN(colorImage) == 1) continue; + + double residual = (double) ((colorImage - colorPlane * fac) * (colorImage - colorPlane * fac)); + if (abs(residual) > oth2) + { + E += oth2; + R++; + continue; + } + + + int ix = (int) x; + int iy = (int) y; + float dx = x - ix; + float dy = y - iy; + float dxdy = dx * dy; + + vignetteFactorTT[ix + iy * wI + 0] += (1 - dx - dy + dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + 1] += (dx - dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + wI] += (dy - dxdy) * colorPlane * colorPlane; + vignetteFactorTT[ix + iy * wI + 1 + wI] += dxdy * colorPlane * colorPlane; + + vignetteFactorCT[ix + iy * wI + 0] += (1 - dx - dy + dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + 1] += (dx - dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + wI] += (dy - dxdy) * colorImage * colorPlane; + vignetteFactorCT[ix + iy * wI + 1 + wI] += dxdy * colorImage * colorPlane; + + if (cvIsNaN(fac) == 1) continue; + E += residual; + R++; + } + } + + float maxFac = 0; + for (int pi = 0; pi < hI * wI; pi++) // for all plane points + { + if (vignetteFactorTT[pi] < 1) + { + vignetteFactor[pi] = NAN; + } + else + { + vignetteFactor[pi] = vignetteFactorCT[pi] / vignetteFactorTT[pi]; + if (vignetteFactor[pi] > maxFac) maxFac = vignetteFactor[pi]; + } + } + + std::cout << R << " residual terms => " << sqrt(E / R) << std::endl; + + // normalize to vignette max. factor 1. + for (int pi = 0; pi < hI * wI; pi++) + { + vignetteFactor[pi] /= maxFac; + } + + logFile << it << " " << n << " " << R << " " << sqrt(E / R) << "\n"; + + // dilate & smoothe vignette by 4 pixel for output. + // does not change anything in the optimization; uses vignetteFactorTT and vignetteFactorCT for temporary storing + memcpy(vignetteFactorTT, vignetteFactor, sizeof(float) * hI * wI); + for (int dilit = 0; dilit < 4; dilit++) + { + memcpy(vignetteFactorCT, vignetteFactorTT, sizeof(float) * hI * wI); + for (int y = 0; y < hI; y++) + { + for (int x = 0; x < wI; x++) + { + int idx = x + y * wI; + { + float sum = 0, num = 0; + if (x < wI - 1 && y < hI - 1 && cvIsNaN(vignetteFactorCT[idx + 1 + wI]) != 1) + { + sum += vignetteFactorCT[idx + 1 + wI]; + num++; + } + if (x < wI - 1 && cvIsNaN(vignetteFactorCT[idx + 1]) != 1) + { + sum += vignetteFactorCT[idx + 1]; + num++; + } + if (x < wI - 1 && y > 0 && cvIsNaN(vignetteFactorCT[idx + 1 - wI]) != 1) + { + sum += vignetteFactorCT[idx + 1 - wI]; + num++; + } + + if (y < hI - 1 && cvIsNaN(vignetteFactorCT[idx + wI]) != 1) + { + sum += vignetteFactorCT[idx + wI]; + num++; + } + if (cvIsNaN(vignetteFactorCT[idx]) != 1) + { + sum += vignetteFactorCT[idx]; + num++; + } + if (y > 0 && cvIsNaN(vignetteFactorCT[idx - wI]) != 1) + { + sum += vignetteFactorCT[idx - wI]; + num++; + } + + if (y < hI - 1 && x > 0 && cvIsNaN(vignetteFactorCT[idx - 1 + wI]) != 1) + { + sum += vignetteFactorCT[idx - 1 + wI]; + num++; + } + if (x > 0 && cvIsNaN(vignetteFactorCT[idx - 1]) != 1) + { + sum += vignetteFactorCT[idx - 1]; + num++; + } + if (y > 0 && x > 0 && cvIsNaN(vignetteFactorCT[idx - 1 - wI]) != 1) + { + sum += vignetteFactorCT[idx - 1 - wI]; + num++; + } + + if (num > 0) vignetteFactorTT[idx] = sum / num; + } + } + } + } + + // ================================ Store Vignette Image ======================================= + if (!_silent) + { + displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); + } + Mat wrapSmoothed = Mat(hI, wI, CV_32F, vignetteFactorTT) * 254.9 * 254.9; + Mat wrapSmoothed16; + wrapSmoothed.convertTo(wrapSmoothed16, CV_16U, 1, 0); + imwrite("vignetteCalibResult/vignetteSmoothed.png", wrapSmoothed16); + waitKey(50); + + if (!_silent) + { + displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); + } + Mat wrap = Mat(hI, wI, CV_32F, vignetteFactor) * 254.9 * 254.9; + Mat wrap16; + wrap.convertTo(wrap16, CV_16U, 1, 0); + imwrite("vignetteCalibResult/vignette.png", wrap16); + waitKey(50); + } + + logFile.flush(); + logFile.close(); + + delete[] planeColor; + delete[] planeColorFF; + delete[] planeColorFC; + delete[] vignetteFactor; + delete[] vignetteFactorTT; + delete[] vignetteFactorCT; + + for (unsigned long i = 0; i < n; i++) + { + delete[] images[i]; + delete[] p2imgX[i]; + delete[] p2imgY[i]; + } +} + VignetteCalib::~VignetteCalib() { From 5a5ed6bbf1cb1d1cd45e8e611ee67d5dae3dcbf5 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 11:32:18 +0200 Subject: [PATCH 42/53] Add debug mode for response calibration. --- .../photometric_calib/ResponseCalib.hpp | 2 +- .../photometric_calib/src/ResponseCalib.cpp | 34 ++++++++++++------- 2 files changed, 23 insertions(+), 13 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp index 31103182c..e3657966b 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/ResponseCalib.hpp @@ -25,7 +25,7 @@ public: void plotG(const double *G, const std::string &saveTo); - void calib(); + void calib(bool debug); inline const std::string &getImageFolderPath() const { diff --git a/modules/photometric_calib/src/ResponseCalib.cpp b/modules/photometric_calib/src/ResponseCalib.cpp index 038afb694..6b16bfca3 100644 --- a/modules/photometric_calib/src/ResponseCalib.cpp +++ b/modules/photometric_calib/src/ResponseCalib.cpp @@ -121,7 +121,7 @@ void ResponseCalib::plotG(const double *G, const std::string &saveTo) { if (val < k) { - GImg.at(k, i) = (float) (k - val); + GImg.at(255 - k, i) = (float) (k - val); } } } @@ -132,7 +132,7 @@ void ResponseCalib::plotG(const double *G, const std::string &saveTo) std::cout << "Saved: " << saveTo << std::endl; } -void ResponseCalib::calib() +void ResponseCalib::calib(bool debug) { int w = 0, h = 0; size_t n = 0; @@ -140,6 +140,7 @@ void ResponseCalib::calib() std::vector exposureDurationVec; std::vector dataVec; + std::cout << "Preprocessing for response calibration... " << std::endl; for (unsigned long i = 0; i < imageReader->getNumImages(); i += _skipFrames) { cv::Mat img = imageReader->getImage(i); @@ -235,8 +236,11 @@ void ResponseCalib::calib() logFile.precision(15); std::cout << "Initial RMSE = " << rmse(G, E, exposureDurationVec, dataVec, w * h)[0] << "!" << std::endl; - plotE(E, w, h, "photoCalibResult/E-0"); - cv::waitKey(100); + if (debug) + { + plotE(E, w, h, "photoCalibResult/E-0"); + cv::waitKey(100); + } bool optE = true; bool optG = true; @@ -270,9 +274,12 @@ void ResponseCalib::calib() delete[] GNum; printf("optG RMSE = %f! \t", rmse(G, E, exposureDurationVec, dataVec, w * h)[0]); - char buf[1000]; - snprintf(buf, 1000, "photoCalibResult/G-%02d.png", it + 1); - plotG(G, buf); + if (debug) + { + char buf[1000]; + snprintf(buf, 1000, "photoCalibResult/G-%02d.png", it + 1); + plotG(G, buf); + } } if (optE) @@ -302,9 +309,12 @@ void ResponseCalib::calib() delete[] ESum; printf("OptE RMSE = %f! \t", rmse(G, E, exposureDurationVec, dataVec, w * h)[0]); - char buf[1000]; - snprintf(buf, 1000, "photoCalibResult/E-%02d", it + 1); - plotE(E, w, h, buf); + if (debug) + { + char buf[1000]; + snprintf(buf, 1000, "photoCalibResult/E-%02d", it + 1); + plotE(E, w, h, buf); + } } // rescale such that maximum response is 255 (fairly arbitrary choice). @@ -314,9 +324,9 @@ void ResponseCalib::calib() E[i] *= rescaleFactor; if (i < 256) G[i] *= rescaleFactor; } - //Eigen::Vector2d err = rmse(G, E, exposureVec, dataVec, w*h ); + Vec2d err = rmse(G, E, exposureDurationVec, dataVec, w * h); - printf("Rescaled RMSE = %f! \trescale with %f!\n", err[0], rescaleFactor); + printf("Rescaled RMSE = %f! \trescale with %f!\n\n", err[0], rescaleFactor); logFile << it << " " << n << " " << err[1] << " " << err[0] << "\n"; From ceb2af688b80259e0945cf3da63bebc621a3dc9b Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 11:32:50 +0200 Subject: [PATCH 43/53] Add sample code for response calibration. --- .../samples/response_calibration.cpp | 56 +++++++++++++++++++ 1 file changed, 56 insertions(+) create mode 100644 modules/photometric_calib/samples/response_calibration.cpp diff --git a/modules/photometric_calib/samples/response_calibration.cpp b/modules/photometric_calib/samples/response_calibration.cpp new file mode 100644 index 000000000..9a2ae9565 --- /dev/null +++ b/modules/photometric_calib/samples/response_calibration.cpp @@ -0,0 +1,56 @@ +#include +#include +#include +#include + +#include "opencv2/opencv.hpp" +#include "opencv2/core.hpp" +#include "opencv2/highgui.hpp" +#include "opencv2/photometric_calib.hpp" +#include "opencv2/imgproc/imgproc.hpp" + +using namespace std; +using namespace cv; + +int main(int argc, char **argv) +{ + // Please down load the sample dataset from: + // https://www.dropbox.com/s/5x48uhc7k2bgjcj/GSoC2017_PhotometricCalib_Sample_Data.zip?dl=0 + // By unzipping the file, you would get a folder named /GSoC2017_PhotometricCalib_Sample_Data which contains 4 subfolders: + // response_calib, response remover, vignette_calib, vignette_remover + // in this sample, we will use the data in the folder response_calib and response_remover + + // Prefix for the data, e.g. /Users/Yelen/GSoC2017_PhotometricCalib_Sample + string userPrefix = "/Users/Yelen/GSoC2017_PhotometricCalib_Sample_Data/"; + // The path for the images used for response calibration + string imageFolderPath = userPrefix + "response_calib/images"; + // The yaml file which contains the timestamps and exposure times for each image used for camera response calibration + string timePath = userPrefix + "response_calib/times.yaml"; + + // Construct a photometric_calib::ResponseCalib object by giving path of image, path of time file and specify the format of images + photometric_calib::ResponseCalib resCal(imageFolderPath, timePath, "jpg"); + + // Debug mode will generate some temporary data + bool debug = true; + // Calibration of camera response function begins + resCal.calib(debug); + + // The result and some intermediate data are stored in the folder ./photoCalibResult in which + // pcalib.yaml is the camera response function file + // Since we are using debug mode, we can visualize the response function: + Mat invRes = imread("./photoCalibResult/G-10.png", CV_LOAD_IMAGE_UNCHANGED); + // As shown as Fig.3 in the paper from J.Engel, et al. in the paper A Photometrically Calibrated Benchmark For Monocular Visual Odometry + imshow("Inverse Response Function", invRes); + waitKey(0); + + // To see the response-calibrated image, we can use GammaRemover + Mat oriImg = imread(imageFolderPath + "/00480.jpg", CV_LOAD_IMAGE_UNCHANGED); + photometric_calib::GammaRemover gammaRemover("./photoCalibResult/pcalib.yaml", oriImg.cols, oriImg.rows); + Mat caliImg = gammaRemover.getUnGammaImageMat(oriImg); + imshow("Original Image", oriImg); + waitKey(0); + imshow("Gamma Removed Image", caliImg); + waitKey(0); + + return 0; +} \ No newline at end of file From c1d3f5c427f926a822fe38372111424dd3b6a225 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:08:44 +0200 Subject: [PATCH 44/53] Better visualization for the sample code of response calibration. --- modules/photometric_calib/samples/response_calibration.cpp | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/modules/photometric_calib/samples/response_calibration.cpp b/modules/photometric_calib/samples/response_calibration.cpp index 9a2ae9565..3d1312b85 100644 --- a/modules/photometric_calib/samples/response_calibration.cpp +++ b/modules/photometric_calib/samples/response_calibration.cpp @@ -40,15 +40,18 @@ int main(int argc, char **argv) // Since we are using debug mode, we can visualize the response function: Mat invRes = imread("./photoCalibResult/G-10.png", CV_LOAD_IMAGE_UNCHANGED); // As shown as Fig.3 in the paper from J.Engel, et al. in the paper A Photometrically Calibrated Benchmark For Monocular Visual Odometry + namedWindow( "Inverse Response Function", WINDOW_AUTOSIZE ); imshow("Inverse Response Function", invRes); - waitKey(0); // To see the response-calibrated image, we can use GammaRemover Mat oriImg = imread(imageFolderPath + "/00480.jpg", CV_LOAD_IMAGE_UNCHANGED); photometric_calib::GammaRemover gammaRemover("./photoCalibResult/pcalib.yaml", oriImg.cols, oriImg.rows); Mat caliImg = gammaRemover.getUnGammaImageMat(oriImg); + + // Visualization + namedWindow( "Original Image", WINDOW_AUTOSIZE ); imshow("Original Image", oriImg); - waitKey(0); + namedWindow( "Gamma Removed Image", WINDOW_AUTOSIZE ); imshow("Gamma Removed Image", caliImg); waitKey(0); From ff2187c6950803e555fce1e1cd89317474d3a7c2 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:12:03 +0200 Subject: [PATCH 45/53] Change parameter bool silent of the constructor to the parameter of method calib() and calibFast(). And also change 'silent' to 'debug'. --- .../photometric_calib/VignetteCalib.hpp | 11 +- .../photometric_calib/src/VignetteCalib.cpp | 129 +++++++++++------- 2 files changed, 86 insertions(+), 54 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp index 44c651773..4308d7074 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteCalib.hpp @@ -17,10 +17,10 @@ class CV_EXPORTS VignetteCalib { public: VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, - std::string imageFormat, bool silent); + std::string imageFormat); VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, std::string gammaFile, - std::string imageFormat, bool silent, int imageSkip, int maxIterations, int outlierTh, + std::string imageFormat, int imageSkip, int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad); virtual ~VignetteCalib(); @@ -34,11 +34,11 @@ public: void displayImageV(float *I, int w, int h, std::string name); - bool preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY); + bool preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY, bool debug); - void calib(); + void calib(bool debug); - void calibFast(); + void calibFast(bool debug); private: int _imageSkip; @@ -65,7 +65,6 @@ private: GammaRemover *gammaRemover; float _meanExposure; - bool _silent; }; } // namespace photometric_calib diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 0e90a75b1..1cc62540f 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -13,7 +13,7 @@ namespace photometric_calib { VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, - std::string gammaFile, std::string imageFormat, bool silent) : + std::string gammaFile, std::string imageFormat) : _imageSkip(1), _maxIterations(20), _outlierTh(15), _gridWidth(1000), _gridHeight(1000), _facW(5), _facH(5), _maxAbsGrad(255) { @@ -35,12 +35,10 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: _K_p2idx_inverse = _K_p2idx.inv(); _meanExposure = calMeanExposureTime(); - - _silent = silent; } VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std::string cameraFile, - std::string gammaFile, std::string imageFormat, bool silent, int imageSkip, + std::string gammaFile, std::string imageFormat, int imageSkip, int maxIterations, int outlierTh, int gridWidth, int gridHeight, float facW, float facH, int maxAbsGrad) : _imageSkip(imageSkip), _maxIterations(maxIterations), _outlierTh(outlierTh), _gridWidth(gridWidth), @@ -65,8 +63,6 @@ VignetteCalib::VignetteCalib(std::string folderPath, std::string timePath, std:: _K_p2idx_inverse = _K_p2idx.inv(); _meanExposure = calMeanExposureTime(); - - _silent = silent; } float VignetteCalib::getInterpolatedElement(const float *const mat, const float x, const float y, const int width) @@ -149,7 +145,7 @@ float VignetteCalib::calMeanExposureTime() return meanExposure; } -bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY) +bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX, float *&plane2imgY, bool debug) { int wI = imageReader->getWidth(); int hI = imageReader->getHeight(); @@ -232,21 +228,7 @@ bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX } } - for (int x = 0; x < _gridWidth; x++) - { - for (int y = 0; y < _gridHeight; y++) - { - int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); - int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); - if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) - { - plane2imgX[x + y * _gridWidth] = NAN; - plane2imgY[x + y * _gridWidth] = NAN; - } - } - } - - if (!_silent) + if (debug) { // debug-plot. Mat dbgImg(hI, wI, CV_8UC3); @@ -299,6 +281,20 @@ bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX } } + for (int x = 0; x < _gridWidth; x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) + { + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } + imshow("inRaw", dbgImg); waitKey(1); @@ -310,10 +306,27 @@ bool VignetteCalib::preCalib(unsigned long id, float *&image, float *&plane2imgX } } + else + { + for (int x = 0; x < _gridWidth; x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) + { + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } + } + return true; } -void VignetteCalib::calib() +void VignetteCalib::calib(bool debug) { // Create folder for vignette calibration if (-1 == system("rm -rf vignetteCalibResult")) @@ -372,7 +385,7 @@ void VignetteCalib::calib() float *plane2imgX = NULL; float *plane2imgY = NULL; float *image = NULL; - if (!preCalib(img, image, plane2imgX, plane2imgY)) + if (!preCalib(img, image, plane2imgX, plane2imgY, debug)) { continue; } @@ -413,7 +426,7 @@ void VignetteCalib::calib() planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; } } - if (!_silent) + if (debug) { displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); } @@ -430,7 +443,7 @@ void VignetteCalib::calib() float *plane2imgX = NULL; float *plane2imgY = NULL; float *image = NULL; - if (!preCalib(img, image, plane2imgX, plane2imgY)) + if (!preCalib(img, image, plane2imgX, plane2imgY, debug)) { continue; } @@ -562,7 +575,7 @@ void VignetteCalib::calib() } // ================================ Store Vignette Image ======================================= - if (!_silent) + if (debug) { displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); } @@ -572,7 +585,7 @@ void VignetteCalib::calib() imwrite("vignetteCalibResult/vignetteSmoothed.png", wrapSmoothed16); waitKey(50); - if (!_silent) + if (debug) { displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); } @@ -594,8 +607,10 @@ void VignetteCalib::calib() delete[] vignetteFactorCT; } -void VignetteCalib::calibFast() +void VignetteCalib::calibFast(bool debug) { + std::cout<<"Fast mode! This requires large memory (10GB+)!"< 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) - { - plane2imgX[x + y * _gridWidth] = NAN; - plane2imgY[x + y * _gridWidth] = NAN; - } - } - } - - if (!_silent) + if (debug) { // debug-plot. Mat dbgImg(hI, wI, CV_8UC3); @@ -764,6 +764,21 @@ void VignetteCalib::calibFast() } } + for (int x = 0; x < _gridWidth; x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) + { + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } + imshow("inRaw", dbgImg); if (rand() % 40 == 0) @@ -776,6 +791,24 @@ void VignetteCalib::calibFast() waitKey(1); } + else + { + for (int x = 0; x < _gridWidth; x++) + { + for (int y = 0; y < _gridHeight; y++) + { + int u_d = (int) lround(plane2imgX[x + y * _gridWidth] + 0.5); + int v_d = (int) lround(plane2imgY[x + y * _gridWidth] + 0.5); + + if (!(u_d > 1 && v_d > 1 && u_d < wI - 2 && v_d < hI - 2)) + { + plane2imgX[x + y * _gridWidth] = NAN; + plane2imgY[x + y * _gridWidth] = NAN; + } + } + } + } + p2imgX.push_back(plane2imgX); p2imgY.push_back(plane2imgY); @@ -858,7 +891,7 @@ void VignetteCalib::calibFast() planeColor[pi] = planeColorFC[pi] / planeColorFF[pi]; } } - if (!_silent) + if (debug) { displayImage(planeColor, _gridWidth, _gridWidth, "Plane"); } @@ -1013,7 +1046,7 @@ void VignetteCalib::calibFast() } // ================================ Store Vignette Image ======================================= - if (!_silent) + if (debug) { displayImageV(vignetteFactorTT, wI, hI, "VignetteSmoothed"); } @@ -1023,7 +1056,7 @@ void VignetteCalib::calibFast() imwrite("vignetteCalibResult/vignetteSmoothed.png", wrapSmoothed16); waitKey(50); - if (!_silent) + if (debug) { displayImageV(vignetteFactor, wI, hI, "VignetteOrg"); } From 963b29b04697fda0fbcfc684f5049ff3ee925abd Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:12:38 +0200 Subject: [PATCH 46/53] Fix bug of calibFast(). --- modules/photometric_calib/src/VignetteCalib.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/modules/photometric_calib/src/VignetteCalib.cpp b/modules/photometric_calib/src/VignetteCalib.cpp index 1cc62540f..75cd1f96d 100644 --- a/modules/photometric_calib/src/VignetteCalib.cpp +++ b/modules/photometric_calib/src/VignetteCalib.cpp @@ -626,12 +626,13 @@ void VignetteCalib::calibFast(bool debug) int wI = wO, hI = hO; Ptr parameters = aruco::DetectorParameters::create(); - Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_6X6_250); + Ptr dictionary = aruco::getPredefinedDictionary(aruco::DICT_ARUCO_ORIGINAL); std::vector images; std::vector p2imgX; std::vector p2imgY; + std::cout<<"Preprocessing images..."<getNumImages(); ++i) { std::vector markerIds; From 3e419531a7df58e3ce436f8240c4063e4ad97658 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:13:12 +0200 Subject: [PATCH 47/53] More convenient VignetteRemover. --- .../opencv2/photometric_calib/VignetteRemover.hpp | 7 ++++--- modules/photometric_calib/src/VignetteRemover.cpp | 14 ++++++++++---- 2 files changed, 14 insertions(+), 7 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp index c041804ca..97b7e8f09 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -27,7 +27,7 @@ public: * @param w_ the width of input image * @param h_ the height of input image */ - VignetteRemover(const std::string &vignettePath, int w_, int h_); + VignetteRemover(const std::string &vignettePath, const std::string &pcalibPath, int w_, int h_); ~VignetteRemover(); @@ -36,18 +36,19 @@ public: * @param unGammaImVec the irradiance image. * @return */ - Mat getUnVignetteImageMat(std::vector &unGammaImVec); + Mat getUnVignetteImageMat(Mat oriImMat); /*! * @brief get vignetting-removed image in form of std::vector. * @param unGammaImVec the irradiance image. * @param outImVec the vignetting-removed image vector. */ - void getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec); + void getUnVignetteImageVec(Mat oriImMat, std::vector &outImVec); private: float *vignetteMap; float *vignetteMapInv; + std::string _pcalibPath; int w, h; bool validVignette; }; diff --git a/modules/photometric_calib/src/VignetteRemover.cpp b/modules/photometric_calib/src/VignetteRemover.cpp index 2ccc55912..cabee946b 100644 --- a/modules/photometric_calib/src/VignetteRemover.cpp +++ b/modules/photometric_calib/src/VignetteRemover.cpp @@ -4,13 +4,15 @@ #include "precomp.hpp" #include "opencv2/photometric_calib/VignetteRemover.hpp" +#include "opencv2/photometric_calib/GammaRemover.hpp" namespace cv { namespace photometric_calib { -VignetteRemover::VignetteRemover(const std::string &vignettePath, int w_, int h_) +VignetteRemover::VignetteRemover(const std::string &vignettePath, const std::string &pcalibPath, int w_, int h_) { CV_Assert(vignettePath != ""); + CV_Assert(pcalibPath != ""); validVignette = false; vignetteMap = 0; @@ -59,24 +61,28 @@ VignetteRemover::VignetteRemover(const std::string &vignettePath, int w_, int h_ vignetteMapInv[i] = 1.0f / vignetteMap[i]; } + _pcalibPath = pcalibPath; validVignette = true; } -Mat VignetteRemover::getUnVignetteImageMat(std::vector &unGammaImVec) +Mat VignetteRemover::getUnVignetteImageMat(Mat oriImMat) { std::vector _outImVec(w * h); - getUnVignetteImageVec(unGammaImVec, _outImVec); + getUnVignetteImageVec(oriImMat, _outImVec); Mat _outIm(h, w, CV_32F, &_outImVec[0]); Mat outIm = _outIm * (1 / 255.0f); return outIm; } -void VignetteRemover::getUnVignetteImageVec(const std::vector &unGammaImVec, std::vector &outImVec) +void VignetteRemover::getUnVignetteImageVec(Mat oriImMat, std::vector &outImVec) { CV_Assert(validVignette); CV_Assert(outImVec.size() == (unsigned long) w * h); + photometric_calib::GammaRemover gammaRemover(_pcalibPath, w, h); + std::vector unGammaImVec(w * h); + gammaRemover.getUnGammaImageVec(oriImMat, unGammaImVec); for (int i = 0; i < w * h; ++i) { outImVec[i] = unGammaImVec[i] * vignetteMapInv[i]; From ff5a312f977eb9767bdeb6aa4787e6950c0c9f23 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:13:38 +0200 Subject: [PATCH 48/53] Sample code of vignette calibration. --- .../samples/vignette_calibration.cpp | 74 +++++++++++++++++++ 1 file changed, 74 insertions(+) create mode 100644 modules/photometric_calib/samples/vignette_calibration.cpp diff --git a/modules/photometric_calib/samples/vignette_calibration.cpp b/modules/photometric_calib/samples/vignette_calibration.cpp new file mode 100644 index 000000000..0fbe81cea --- /dev/null +++ b/modules/photometric_calib/samples/vignette_calibration.cpp @@ -0,0 +1,74 @@ +#include +#include +#include +#include + +#include "opencv2/opencv.hpp" +#include "opencv2/core.hpp" +#include "opencv2/highgui.hpp" +#include "opencv2/photometric_calib.hpp" +#include "opencv2/imgproc/imgproc.hpp" + +using namespace std; +using namespace cv; + +int main(int argc, char** argv) +{ + // Please down load the sample dataset from: + // https://www.dropbox.com/s/5x48uhc7k2bgjcj/GSoC2017_PhotometricCalib_Sample_Data.zip?dl=0 + // By unzipping the file, you would get a folder named /GSoC2017_PhotometricCalib_Sample_Data which contains 2 subfolders: + // response_calib, vignette_calib + // in this sample, we will use the data in the folder vignette_calib + + // Prefix for the data, e.g. /Users/Yelen/GSoC2017_PhotometricCalib_Sample + string userPrefix = "/Users/Yelen/GSoC2017_PhotometricCalib_Sample_Data/"; + // The path for the images used for response calibration + string imageFolderPath = userPrefix + "vignette_calib/images"; + // The yaml file which contains the timestamps and exposure times for each image used for vignette calibration + string timePath = userPrefix + "vignette_calib/times.yaml"; + // The yaml file which contains the camera intrinsics and extrinsics. + // Note that the images are already rectified, so the distortion parameters are 0s + string cameraPath = userPrefix + "vignette_calib/camera.yaml"; + // The pcalib file. Vignette calibration can be performed only when provided with pcalib file. + // We use the identical pcalib.yaml file generated by response_calibration.cpp + // You can refer to the code in response_calibration.cpp for details + string gammaPath = userPrefix + "vignette_calib/pcalib.yaml"; + + // Construct a photometric_calib::VignetteCalib object by giving path of image, path of time file, camera parameter file, pcalib file and specify the format of images + photometric_calib::VignetteCalib vigCal(imageFolderPath, timePath, cameraPath, gammaPath, "jpg"); + + // Debug mode will visualize the optimization process and generate some temporary data + bool debug = true; + // Calibration of camera response function begins + vigCal.calib(debug); + + // You can also use fast mode, but with much memory (potentially with 10GB+) + // vigCal.calibFast(debug); + + // The result and some intermediate data are stored in the folder ./vignetteCalibResult in which + // vignette.png and vignetteSmoothed.png are the vignette images. + // In practice, vignetteSomoothed.png is used, since it doesn't have the black boarders. + Mat vigSmoothed = imread("./vignetteCalibResult/vignetteSmoothed.png", CV_LOAD_IMAGE_UNCHANGED); + // As shown as Fig.4 in the paper from J.Engel, et al. in the paper A Photometrically Calibrated Benchmark For Monocular Visual Odometry + namedWindow( "Vignette Smoothed", WINDOW_AUTOSIZE ); + imshow("Vignette Smoothed", vigSmoothed); + + // To see the vignette-calibrated image, we can use VignetteRemover + Mat oriImg = imread(imageFolderPath + "/00480.jpg", CV_LOAD_IMAGE_UNCHANGED); + photometric_calib::GammaRemover gammaRemover(gammaPath, oriImg.cols, oriImg.rows); + photometric_calib::VignetteRemover vignetteRemover("./vignetteCalibResult/vignetteSmoothed.png", gammaPath, oriImg.cols, oriImg.rows); + Mat resCaliImg = gammaRemover.getUnGammaImageMat(oriImg); + Mat vigCaliImg = vignetteRemover.getUnVignetteImageMat(oriImg); + + // Visualization + namedWindow( "Original Image", WINDOW_AUTOSIZE ); + imshow("Original Image", oriImg); + namedWindow( "Gamma Removed Image", WINDOW_AUTOSIZE ); + imshow("Gamma Removed Image", resCaliImg); + namedWindow( "Vignette Removed Image", WINDOW_AUTOSIZE ); + imshow("Vignette Removed Image", vigCaliImg); + + waitKey(0); + + return 0; +} \ No newline at end of file From 6e97cc20591ffd373e84198b2ef89fc53a22bb4e Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 14:24:11 +0200 Subject: [PATCH 49/53] Fix bug with header files of sample codes --- .../samples/response_calibration.cpp | 13 +++---------- .../samples/vignette_calibration.cpp | 13 +++---------- 2 files changed, 6 insertions(+), 20 deletions(-) diff --git a/modules/photometric_calib/samples/response_calibration.cpp b/modules/photometric_calib/samples/response_calibration.cpp index 3d1312b85..aec40fbda 100644 --- a/modules/photometric_calib/samples/response_calibration.cpp +++ b/modules/photometric_calib/samples/response_calibration.cpp @@ -1,18 +1,11 @@ -#include -#include -#include -#include - -#include "opencv2/opencv.hpp" #include "opencv2/core.hpp" #include "opencv2/highgui.hpp" #include "opencv2/photometric_calib.hpp" -#include "opencv2/imgproc/imgproc.hpp" using namespace std; using namespace cv; -int main(int argc, char **argv) +int main() { // Please down load the sample dataset from: // https://www.dropbox.com/s/5x48uhc7k2bgjcj/GSoC2017_PhotometricCalib_Sample_Data.zip?dl=0 @@ -38,13 +31,13 @@ int main(int argc, char **argv) // The result and some intermediate data are stored in the folder ./photoCalibResult in which // pcalib.yaml is the camera response function file // Since we are using debug mode, we can visualize the response function: - Mat invRes = imread("./photoCalibResult/G-10.png", CV_LOAD_IMAGE_UNCHANGED); + Mat invRes = imread("./photoCalibResult/G-10.png", IMREAD_UNCHANGED); // As shown as Fig.3 in the paper from J.Engel, et al. in the paper A Photometrically Calibrated Benchmark For Monocular Visual Odometry namedWindow( "Inverse Response Function", WINDOW_AUTOSIZE ); imshow("Inverse Response Function", invRes); // To see the response-calibrated image, we can use GammaRemover - Mat oriImg = imread(imageFolderPath + "/00480.jpg", CV_LOAD_IMAGE_UNCHANGED); + Mat oriImg = imread(imageFolderPath + "/00480.jpg", IMREAD_UNCHANGED); photometric_calib::GammaRemover gammaRemover("./photoCalibResult/pcalib.yaml", oriImg.cols, oriImg.rows); Mat caliImg = gammaRemover.getUnGammaImageMat(oriImg); diff --git a/modules/photometric_calib/samples/vignette_calibration.cpp b/modules/photometric_calib/samples/vignette_calibration.cpp index 0fbe81cea..b47b109be 100644 --- a/modules/photometric_calib/samples/vignette_calibration.cpp +++ b/modules/photometric_calib/samples/vignette_calibration.cpp @@ -1,18 +1,11 @@ -#include -#include -#include -#include - -#include "opencv2/opencv.hpp" #include "opencv2/core.hpp" #include "opencv2/highgui.hpp" #include "opencv2/photometric_calib.hpp" -#include "opencv2/imgproc/imgproc.hpp" using namespace std; using namespace cv; -int main(int argc, char** argv) +int main() { // Please down load the sample dataset from: // https://www.dropbox.com/s/5x48uhc7k2bgjcj/GSoC2017_PhotometricCalib_Sample_Data.zip?dl=0 @@ -48,13 +41,13 @@ int main(int argc, char** argv) // The result and some intermediate data are stored in the folder ./vignetteCalibResult in which // vignette.png and vignetteSmoothed.png are the vignette images. // In practice, vignetteSomoothed.png is used, since it doesn't have the black boarders. - Mat vigSmoothed = imread("./vignetteCalibResult/vignetteSmoothed.png", CV_LOAD_IMAGE_UNCHANGED); + Mat vigSmoothed = imread("./vignetteCalibResult/vignetteSmoothed.png", IMREAD_UNCHANGED); // As shown as Fig.4 in the paper from J.Engel, et al. in the paper A Photometrically Calibrated Benchmark For Monocular Visual Odometry namedWindow( "Vignette Smoothed", WINDOW_AUTOSIZE ); imshow("Vignette Smoothed", vigSmoothed); // To see the vignette-calibrated image, we can use VignetteRemover - Mat oriImg = imread(imageFolderPath + "/00480.jpg", CV_LOAD_IMAGE_UNCHANGED); + Mat oriImg = imread(imageFolderPath + "/00480.jpg", IMREAD_UNCHANGED); photometric_calib::GammaRemover gammaRemover(gammaPath, oriImg.cols, oriImg.rows); photometric_calib::VignetteRemover vignetteRemover("./vignetteCalibResult/vignetteSmoothed.png", gammaPath, oriImg.cols, oriImg.rows); Mat resCaliImg = gammaRemover.getUnGammaImageMat(oriImg); From 16ea96b6453c1da9dbd9e32eb991a4cced868676 Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 21 Aug 2017 19:14:11 +0200 Subject: [PATCH 50/53] Update Doxygen. --- .../include/opencv2/photometric_calib/VignetteRemover.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp index 97b7e8f09..c03ab2754 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib/VignetteRemover.hpp @@ -24,6 +24,7 @@ public: /*! * @brief Constructor * @param vignettePath the path of vignetting file + * @param pcalibPath the path of pcalib file * @param w_ the width of input image * @param h_ the height of input image */ @@ -33,14 +34,13 @@ public: /*! * @brief get vignetting-removed image in form of cv::Mat. - * @param unGammaImVec the irradiance image. - * @return + * @param oriImMat the image to be calibrated. */ Mat getUnVignetteImageMat(Mat oriImMat); /*! * @brief get vignetting-removed image in form of std::vector. - * @param unGammaImVec the irradiance image. + * @param oriImMat the image to be calibrated. * @param outImVec the vignetting-removed image vector. */ void getUnVignetteImageVec(Mat oriImMat, std::vector &outImVec); From d7375e6cc434fa7446388938225bbae6a9a027fc Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 28 Aug 2017 10:01:17 +0200 Subject: [PATCH 51/53] Add test header file --- .../photometric_calib/test/test_precomp.hpp | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) create mode 100644 modules/photometric_calib/test/test_precomp.hpp diff --git a/modules/photometric_calib/test/test_precomp.hpp b/modules/photometric_calib/test/test_precomp.hpp new file mode 100644 index 000000000..e832810b2 --- /dev/null +++ b/modules/photometric_calib/test/test_precomp.hpp @@ -0,0 +1,18 @@ +#ifdef __GNUC__ +# pragma GCC diagnostic ignored "-Wmissing-declarations" +# if defined __clang__ || defined __APPLE__ +# pragma GCC diagnostic ignored "-Wmissing-prototypes" +# pragma GCC diagnostic ignored "-Wextra" +# endif +#endif + +#ifndef __OPENCV_TEST_PRECOMP_HPP__ +#define __OPENCV_TEST_PRECOMP_HPP__ + +#include +#include "opencv2/ts.hpp" +#include "opencv2/imgproc.hpp" +#include "opencv2/photometric_calib.hpp" +#include "opencv2/highgui.hpp" + +#endif From 9653dde00926687a68ffe3778cfe481c8d7bc5df Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 28 Aug 2017 10:01:32 +0200 Subject: [PATCH 52/53] Add documentation. --- .../include/opencv2/photometric_calib.hpp | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/modules/photometric_calib/include/opencv2/photometric_calib.hpp b/modules/photometric_calib/include/opencv2/photometric_calib.hpp index e1c9d5052..2b5bfb624 100644 --- a/modules/photometric_calib/include/opencv2/photometric_calib.hpp +++ b/modules/photometric_calib/include/opencv2/photometric_calib.hpp @@ -11,6 +11,15 @@ #include "opencv2/photometric_calib/ResponseCalib.hpp" #include "opencv2/photometric_calib/VignetteCalib.hpp" +/** + * @defgroup photometric_calib Photometric Calibration + * The photometric_calib contains photomeric calibration algorithm proposed by Jakob Engel. \n + * The implementation is totally based on the paper \cite engel2016monodataset. \n + * Photometric calibration aimed at removing the camera response function and vitnetting artefact, + * by which the tracking and image alignment algorithms based on direct methods can be improved significantly. \n + * For details please refer to \cite engel2016monodataset. + */ + namespace cv { namespace photometric_calib { } // namespace photometric_calib From 37633553f4f4bc09758974c1e1c206b02392e2ba Mon Sep 17 00:00:00 2001 From: Nan Yang Date: Mon, 28 Aug 2017 10:47:16 +0200 Subject: [PATCH 53/53] Add test main --- modules/photometric_calib/test/test_main.cpp | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 modules/photometric_calib/test/test_main.cpp diff --git a/modules/photometric_calib/test/test_main.cpp b/modules/photometric_calib/test/test_main.cpp new file mode 100644 index 000000000..6f9ac2e0d --- /dev/null +++ b/modules/photometric_calib/test/test_main.cpp @@ -0,0 +1,3 @@ +#include "test_precomp.hpp" + +CV_TEST_MAIN("")