mirror of
https://github.com/opencv/opencv.git
synced 2025-08-06 14:36:36 +08:00
Merge pull request #25308 from dkurt:not_normalized_findHomography
Not-normalized output from findHomography #25308 ### Pull Request Readiness Checklist resolves https://github.com/opencv/opencv/issues/25133 resolves https://github.com/opencv/opencv/issues/4834 resolves https://github.com/opencv/opencv/issues/22166 resolves https://github.com/opencv/opencv/issues/18592 See details at https://github.com/opencv/opencv/wiki/How_to_contribute#making-a-good-pull-request - [x] I agree to contribute to the project under Apache 2 License. - [x] To the best of my knowledge, the proposed patch is not based on a code under GPL or another license that is incompatible with OpenCV - [x] The PR is proposed to the proper branch - [x] There is a reference to the original bug report and related work - [x] There is accuracy test, performance test and test data in opencv_extra repository, if applicable Patch to opencv_extra has the same branch name. - [x] The feature is well documented and sample code can be built with the project CMake
This commit is contained in:
parent
12e2cc9502
commit
b1e01970ef
@ -726,8 +726,8 @@ correctly only when there are more than 50% of inliers. Finally, if there are no
|
||||
noise is rather small, use the default method (method=0).
|
||||
|
||||
The function is used to find initial intrinsic and extrinsic matrices. Homography matrix is
|
||||
determined up to a scale. Thus, it is normalized so that \f$h_{33}=1\f$. Note that whenever an \f$H\f$ matrix
|
||||
cannot be estimated, an empty one will be returned.
|
||||
determined up to a scale. If \f$h_{33}\f$ is non-zero, the matrix is normalized so that \f$h_{33}=1\f$.
|
||||
@note Whenever an \f$H\f$ matrix cannot be estimated, an empty one will be returned.
|
||||
|
||||
@sa
|
||||
getAffineTransform, estimateAffine2D, estimateAffinePartial2D, getPerspectiveTransform, warpPerspective,
|
||||
|
@ -49,6 +49,13 @@
|
||||
namespace cv
|
||||
{
|
||||
|
||||
static inline double scaleFor(double x){
|
||||
return (std::fabs(x) > std::numeric_limits<float>::epsilon()) ? 1./x : 1.;
|
||||
}
|
||||
static inline float scaleFor(float x){
|
||||
return (std::fabs(x) > std::numeric_limits<float>::epsilon()) ? 1.f/x : 1.f;
|
||||
}
|
||||
|
||||
/**
|
||||
* This class estimates a homography \f$H\in \mathbb{R}^{3\times 3}\f$
|
||||
* between \f$\mathbf{x} \in \mathbb{R}^3\f$ and
|
||||
@ -177,8 +184,7 @@ public:
|
||||
eigen( _LtL, matW, matV );
|
||||
_Htemp = _invHnorm*_H0;
|
||||
_H0 = _Htemp*_Hnorm2;
|
||||
_H0.convertTo(_model, _H0.type(), 1./_H0.at<double>(2,2) );
|
||||
|
||||
_H0.convertTo(_model, _H0.type(), scaleFor(_H0.at<double>(2,2)));
|
||||
return 1;
|
||||
}
|
||||
|
||||
@ -197,14 +203,14 @@ public:
|
||||
const Point2f* M = m1.ptr<Point2f>();
|
||||
const Point2f* m = m2.ptr<Point2f>();
|
||||
const double* H = model.ptr<double>();
|
||||
float Hf[] = { (float)H[0], (float)H[1], (float)H[2], (float)H[3], (float)H[4], (float)H[5], (float)H[6], (float)H[7] };
|
||||
float Hf[] = { (float)H[0], (float)H[1], (float)H[2], (float)H[3], (float)H[4], (float)H[5], (float)H[6], (float)H[7], (float)H[8] };
|
||||
|
||||
_err.create(count, 1, CV_32F);
|
||||
float* err = _err.getMat().ptr<float>();
|
||||
|
||||
for( i = 0; i < count; i++ )
|
||||
{
|
||||
float ww = 1.f/(Hf[6]*M[i].x + Hf[7]*M[i].y + 1.f);
|
||||
float ww = 1.f/(Hf[6]*M[i].x + Hf[7]*M[i].y + Hf[8]);
|
||||
float dx = (Hf[0]*M[i].x + Hf[1]*M[i].y + Hf[2])*ww - m[i].x;
|
||||
float dy = (Hf[3]*M[i].x + Hf[4]*M[i].y + Hf[5])*ww - m[i].y;
|
||||
err[i] = dx*dx + dy*dy;
|
||||
@ -231,8 +237,9 @@ public:
|
||||
if( _Jac.needed())
|
||||
{
|
||||
_Jac.create(count*2, param.rows, CV_64F);
|
||||
_Jac.setTo(0.);
|
||||
J = _Jac.getMat();
|
||||
CV_Assert( J.isContinuous() && J.cols == 8 );
|
||||
CV_Assert( J.isContinuous() && J.cols == 9 );
|
||||
}
|
||||
|
||||
const Point2f* M = src.ptr<Point2f>();
|
||||
@ -244,7 +251,7 @@ public:
|
||||
for( i = 0; i < count; i++ )
|
||||
{
|
||||
double Mx = M[i].x, My = M[i].y;
|
||||
double ww = h[6]*Mx + h[7]*My + 1.;
|
||||
double ww = h[6]*Mx + h[7]*My + h[8];
|
||||
ww = fabs(ww) > DBL_EPSILON ? 1./ww : 0;
|
||||
double xi = (h[0]*Mx + h[1]*My + h[2])*ww;
|
||||
double yi = (h[3]*Mx + h[4]*My + h[5])*ww;
|
||||
@ -254,9 +261,7 @@ public:
|
||||
if( Jptr )
|
||||
{
|
||||
Jptr[0] = Mx*ww; Jptr[1] = My*ww; Jptr[2] = ww;
|
||||
Jptr[3] = Jptr[4] = Jptr[5] = 0.;
|
||||
Jptr[6] = -Mx*ww*xi; Jptr[7] = -My*ww*xi;
|
||||
Jptr[8] = Jptr[9] = Jptr[10] = 0.;
|
||||
Jptr[11] = Mx*ww; Jptr[12] = My*ww; Jptr[13] = ww;
|
||||
Jptr[14] = -Mx*ww*yi; Jptr[15] = -My*ww*yi;
|
||||
|
||||
@ -419,7 +424,7 @@ cv::Mat cv::findHomography( InputArray _points1, InputArray _points2,
|
||||
dst = dst1;
|
||||
if( method == RANSAC || method == LMEDS )
|
||||
cb->runKernel( src, dst, H );
|
||||
Mat H8(8, 1, CV_64F, H.ptr<double>());
|
||||
Mat H8(9, 1, CV_64F, H.ptr<double>());
|
||||
LMSolver::create(makePtr<HomographyRefineCallback>(src, dst), 10)->run(H8);
|
||||
}
|
||||
}
|
||||
@ -1002,14 +1007,6 @@ void cv::computeCorrespondEpilines( InputArray _points, int whichImage,
|
||||
}
|
||||
}
|
||||
|
||||
static inline double scaleFor(double x){
|
||||
return (std::fabs(x) > std::numeric_limits<float>::epsilon()) ? 1./x : 1.;
|
||||
}
|
||||
static inline float scaleFor(float x){
|
||||
return (std::fabs(x) > std::numeric_limits<float>::epsilon()) ? 1.f/x : 1.f;
|
||||
}
|
||||
|
||||
|
||||
void cv::convertPointsFromHomogeneous( InputArray _src, OutputArray _dst )
|
||||
{
|
||||
CV_INSTRUMENT_REGION();
|
||||
|
@ -704,4 +704,27 @@ TEST(Calib3d_Homography, minPoints)
|
||||
EXPECT_THROW(findHomography(p1, p2, RANSAC, 0.01, mask), cv::Exception);
|
||||
}
|
||||
|
||||
TEST(Calib3d_Homography, not_normalized)
|
||||
{
|
||||
Mat_<double> p1({5, 2}, {-1, -1, -2, -2, -1, 1, -2, 2, -1, 0});
|
||||
Mat_<double> p2({5, 2}, {0, -1, -1, -1, 0, 0, -1, 0, 0, -0.5});
|
||||
Mat_<double> ref({3, 3}, {
|
||||
0.74276086, 0., 0.74276086,
|
||||
0.18569022, 0.18569022, 0.,
|
||||
-0.37138043, 0., 0.
|
||||
});
|
||||
|
||||
for (int method : std::vector<int>({0, RANSAC, LMEDS}))
|
||||
{
|
||||
Mat h = findHomography(p1, p2, method);
|
||||
for (auto it = h.begin<double>(); it != h.end<double>(); ++it) {
|
||||
ASSERT_FALSE(cvIsNaN(*it)) << format("method %d\nResult:\n", method) << h;
|
||||
}
|
||||
if (h.at<double>(0, 0) * ref.at<double>(0, 0) < 0) {
|
||||
h *= -1;
|
||||
}
|
||||
ASSERT_LE(cv::norm(h, ref, NORM_INF), 1e-8) << format("method %d\nResult:\n", method) << h;
|
||||
}
|
||||
}
|
||||
|
||||
}} // namespace
|
||||
|
Loading…
Reference in New Issue
Block a user