Неправильная карта диспаратности с использованием LIBELAS и алгоритма пост-фильтрации

Mat generateDisparityMap(Mat& left, Mat& right)
{
  Mat lb, rb;
  if (left.empty() || right.empty())
    return left;
  cvtColor(left, lb, CV_BGR2GRAY);
  cvtColor(right, rb, CV_BGR2GRAY);

  const Size imsize = lb.size();
  const int32_t dims[3] = { imsize.width,imsize.height,imsize.width };

  Mat leftdpf = Mat::zeros(imsize, CV_32F);
  Mat rightdpf = Mat::zeros(imsize, CV_32F);

  Elas::parameters param(Elas::MIDDLEBURY);
  param.postprocess_only_left = true;
  Elas elas(param);

  elas.process(lb.data, rb.data, leftdpf.ptr<float>(0), rightdpf.ptr<float>
    (0), dims);


  Mat show = Mat(left.rows, left.cols, CV_8UC1, Scalar(0));
  leftdpf.convertTo(show, CV_8U, 5.);

  int   max_disp = -1;

  for (int i = 0; i < imsize.width; i++) {
    for (int j = 0; j < imsize.height; j++) {
      if (show.at<uchar>(j,i) > max_disp) 
        max_disp = show.at<uchar>(j,i);
    }
  }

  for (int i = 0; i < imsize.width; i++) {
    for (int j = 0; j < imsize.height; j++) {
      show.at<uchar>(j,i) =
      (int)max(255.0*(float)show.at<uchar>(j,i)/max_disp,0.0);
    }
  }

  //return dmap;
  return show;
  //return show;
}

Пожалуйста, посмотрите на изображение карты диспаратности, сгенерированной моим кодом, показанной в ссылке.

Я видел результаты, полученные библиотекой LIBELAS онлайн, и они казались идеальными. Мой код работает без ошибок, но я получаю смутно искаженные карты, пожалуйста, дайте мне знать о любых изменениях в моем коде. Я использую Visual Studio 2017 IDE с библиотеками opencv 3.3.0.

РЕДАКТИРОВАТЬ Я попытался использовать код, чтобы найти несоответствие, указанное в ссылке https://github.com/opencv/opencv_contrib/blob/master/modules/ximgproc/samples/disparity_filtering.cpp. Однако карта диспаратности в некоторых областях неверна. Некоторые объекты вдали от камеры выглядят ярче, чем близкие объекты. Я попытался вычислить фактическую глубину путем умножения значения диспаратности на калибровочную матрицу Q. Рассчитанная глубина далека от реальных значений, измеренных. Я уверен, что матрица Q верна, так как мое исправленное изображение кажется хорошим. Мое значение квадрата для калибровки также было точным (0,05 метра). Мое несоответствие изображено в указанной ссылке https://photos.app.goo.gl/YWPc6yq7XAmUpkk62.

Это добавленный код для расчета фактической глубины по отфильтрованному изображению диспаратности, хранящемуся в Filter_disp_vis.

fs1["Q"] >> Q;
    Mat Image;
    Mat V = Mat(4, 1, CV_64FC1);
    Mat pos = Mat(4, 1, CV_64FC1);
    vector< Point3d > points;
    //float fMaxDistance = static_cast<float>((1. / Q.at<double>(3, 2)) * Q.at<double>(2, 3));
    //filtered_disp_vis.convertTo(filtered_disp_vis, CV_64FC1, 1.0 / 16.0, 0.0);
    //imshow("filtered disparity", filtered_disp_vis);


    // outputDisparityValue is single 16-bit value from disparityMap
    // DISP_SCALE = 16
    //float fDisparity = outputDisparityValue / (float)StereoMatcher::DISP_SCALE;
    //float fDistance = fMaxDistance / fDisparity;
    reprojectImageTo3D(filtered_disp_vis, Image, Q, false, CV_32F);
    //cout << Image;
    for (int i = 0; i < filtered_disp_vis.cols; i++)
    {
    for (int j = 0; j < filtered_disp_vis.rows; j++)
    {
    int d = filtered_disp_vis.at<uchar>(j, i);
    //filtered_disp_vis.convertTo(filtered_disp_vis, CV_32F, 1.0 / 16.0, 0.0);



    //int l = img_left.at<uchar>(j, i);
    //cout << "(" << j << "," << i << ")" << "=" << d;
    //out << endl;

    // if low disparity, then ignore
    /*if (d < 2) {
    continue;
    }*/
    // V is the vector to be multiplied to Q to get
    // the 3D homogenous coordinates of the image point
    V.at<double>(0, 0) = (double)(i);
    V.at<double>(1, 0) = (double)(j);
    V.at<double>(2, 0) = (double)d;
    V.at<double>(3, 0) = 1.;
    pos = Q * V; // 3D homogeneous coordinate
    double X = pos.at<double>(0, 0) / pos.at<double>(3, 0);
    double Y = pos.at<double>(1, 0) / pos.at<double>(3, 0);
    double Z = pos.at<double>(2, 0) / pos.at<double>(3, 0);

    if (i == 446 && j == 362)
    {
    cout << "(" << j << "," << i << ")" << " =   ";

    cout << X << " " << Y << " " << Z << " " << d;
    cout << endl;
    }

    Mat point3d_cam = Mat(3, 1, CV_64FC1);
    point3d_cam.at<double>(0, 0) = X;
    point3d_cam.at<double>(1, 0) = Y;
    point3d_cam.at<double>(2, 0) = Z;
    // transform 3D point from camera frame to robot frame
    //Mat point3d_robot = XR * point3d_cam + XT;
    points.push_back(Point3d(point3d_cam));
    }

Куда я иду не так? Будем весьма благодарны за любые изменения в моем фрагменте или другие рекомендации для получения правильных карт диспаратности с точными значениями глубины.

1 ответ

Я думаю, что это не проблема LIBELAS, а проблема конверсии. Я не уверен в диапазоне ваших результирующих изображений. Но обычно не очень хорошая идея конвертировать напрямую CV_32F в CV_8U, вы потеряете информацию, и это также будет зависеть от диапазона...

Кроме того, вы нормализуете значения после преобразования его в 8U, это может вызвать проблемы, так как вы потеряли информацию, и тогда максимальное значение может быть неправильным.

Если это только для отображения, вы можете использовать функцию нормализации из OpenCV.

cv::Mat show;
cv::normalize(leftdpf, show, 0, 255, cv::NORM_MINMAX, CV_8U);

Это позволит отобразить изображение с типом CV_8U со значениями, нормализованными для соответствия диапазону (0-255).

Другие вопросы по тегам