LCOV - code coverage report
Current view: top level - alg - gdal_interpolateatpoint.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 193 200 96.5 %
Date: 2025-01-18 12:42:00 Functions: 19 19 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  * $Id$
       3             :  *
       4             :  * Project:  GDAL DEM Interpolation
       5             :  * Purpose:  Interpolation algorithms with cache
       6             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       7             :  *
       8             :  ******************************************************************************
       9             :  * Copyright (c) 2001, Frank Warmerdam
      10             :  * Copyright (c) 2008-2012, Even Rouault <even dot rouault at spatialys.com>
      11             :  * Copyright (c) 2024, Javier Jimenez Shaw
      12             :  *
      13             :  * SPDX-License-Identifier: MIT
      14             :  ****************************************************************************/
      15             : 
      16             : #include "gdal_interpolateatpoint.h"
      17             : 
      18             : #include "gdalresamplingkernels.h"
      19             : 
      20             : #include "gdal_vectorx.h"
      21             : 
      22             : #include <algorithm>
      23             : #include <complex>
      24             : 
      25             : template <typename T> bool areEqualReal(double dfNoDataValue, T dfOut);
      26             : 
      27       16749 : template <> bool areEqualReal(double dfNoDataValue, double dfOut)
      28             : {
      29       16749 :     return ARE_REAL_EQUAL(dfNoDataValue, dfOut);
      30             : }
      31             : 
      32          76 : template <> bool areEqualReal(double dfNoDataValue, std::complex<double> dfOut)
      33             : {
      34          76 :     return ARE_REAL_EQUAL(dfNoDataValue, dfOut.real());
      35             : }
      36             : 
      37             : // Only valid for T = double or std::complex<double>
      38             : template <typename T>
      39      253682 : bool GDALInterpExtractValuesWindow(GDALRasterBand *pBand,
      40             :                                    std::unique_ptr<DoublePointsCache> &cache,
      41             :                                    gdal::Vector2i point,
      42             :                                    gdal::Vector2i dimensions, T *padfOut)
      43             : {
      44      253682 :     constexpr int BLOCK_SIZE = 64;
      45             : 
      46      253682 :     const int nX = point.x();
      47      253682 :     const int nY = point.y();
      48      253682 :     const int nWidth = dimensions.x();
      49      253682 :     const int nHeight = dimensions.y();
      50             : 
      51             :     // Request the DEM by blocks of BLOCK_SIZE * BLOCK_SIZE and put them
      52             :     // in cache
      53      253682 :     if (!cache)
      54          83 :         cache.reset(new DoublePointsCache{});
      55             : 
      56      253682 :     const int nXIters = (nX + nWidth - 1) / BLOCK_SIZE - nX / BLOCK_SIZE + 1;
      57      253682 :     const int nYIters = (nY + nHeight - 1) / BLOCK_SIZE - nY / BLOCK_SIZE + 1;
      58      253682 :     const int nRasterXSize = pBand->GetXSize();
      59      253682 :     const int nRasterYSize = pBand->GetYSize();
      60             :     const bool bIsComplex =
      61      253682 :         CPL_TO_BOOL(GDALDataTypeIsComplex(pBand->GetRasterDataType()));
      62             : 
      63      507800 :     for (int iY = 0; iY < nYIters; iY++)
      64             :     {
      65      254118 :         const int nBlockY = nY / BLOCK_SIZE + iY;
      66      254118 :         const int nReqYSize =
      67      254118 :             std::min(nRasterYSize - nBlockY * BLOCK_SIZE, BLOCK_SIZE);
      68      254118 :         const int nFirstLineInCachedBlock = (iY == 0) ? nY % BLOCK_SIZE : 0;
      69      254118 :         const int nFirstLineInOutput =
      70             :             (iY == 0) ? 0
      71         436 :                       : BLOCK_SIZE - (nY % BLOCK_SIZE) + (iY - 1) * BLOCK_SIZE;
      72      254990 :         const int nLinesToCopy = (nYIters == 1) ? nHeight
      73         436 :                                  : (iY == 0)    ? BLOCK_SIZE - (nY % BLOCK_SIZE)
      74         436 :                                  : (iY == nYIters - 1)
      75         436 :                                      ? 1 + (nY + nHeight - 1) % BLOCK_SIZE
      76             :                                      : BLOCK_SIZE;
      77      512285 :         for (int iX = 0; iX < nXIters; iX++)
      78             :         {
      79      258167 :             const int nBlockX = nX / BLOCK_SIZE + iX;
      80      258167 :             const int nReqXSize =
      81      258167 :                 std::min(nRasterXSize - nBlockX * BLOCK_SIZE, BLOCK_SIZE);
      82      258167 :             const uint64_t nKey =
      83      258167 :                 (static_cast<uint64_t>(nBlockY) << 32) | nBlockX;
      84      258167 :             const int nFirstColInCachedBlock = (iX == 0) ? nX % BLOCK_SIZE : 0;
      85      258167 :             const int nFirstColInOutput =
      86             :                 (iX == 0)
      87             :                     ? 0
      88        4049 :                     : BLOCK_SIZE - (nX % BLOCK_SIZE) + (iX - 1) * BLOCK_SIZE;
      89      266265 :             const int nColsToCopy = (nXIters == 1) ? nWidth
      90        4049 :                                     : (iX == 0) ? BLOCK_SIZE - (nX % BLOCK_SIZE)
      91        4049 :                                     : (iX == nXIters - 1)
      92        4049 :                                         ? 1 + (nX + nWidth - 1) % BLOCK_SIZE
      93             :                                         : BLOCK_SIZE;
      94             : 
      95             : #if 0
      96             :             CPLDebug("RPC", "nY=%d nX=%d nBlockY=%d nBlockX=%d "
      97             :                      "nFirstLineInCachedBlock=%d nFirstLineInOutput=%d nLinesToCopy=%d "
      98             :                      "nFirstColInCachedBlock=%d nFirstColInOutput=%d nColsToCopy=%d",
      99             :                      nY, nX, nBlockY, nBlockX, nFirstLineInCachedBlock, nFirstLineInOutput, nLinesToCopy,
     100             :                      nFirstColInCachedBlock, nFirstColInOutput, nColsToCopy);
     101             : #endif
     102             : 
     103      258167 :             constexpr int nTypeFactor = sizeof(T) / sizeof(double);
     104           0 :             std::shared_ptr<std::vector<double>> poValue;
     105      258167 :             if (!cache->tryGet(nKey, poValue))
     106             :             {
     107         722 :                 const GDALDataType eDataType =
     108             :                     bIsComplex ? GDT_CFloat64 : GDT_Float64;
     109         722 :                 const size_t nVectorSize =
     110         722 :                     size_t(nReqXSize) * nReqYSize * nTypeFactor;
     111         722 :                 poValue = std::make_shared<std::vector<double>>(nVectorSize);
     112         722 :                 CPLErr eErr = pBand->RasterIO(
     113             :                     GF_Read, nBlockX * BLOCK_SIZE, nBlockY * BLOCK_SIZE,
     114         722 :                     nReqXSize, nReqYSize, poValue->data(), nReqXSize, nReqYSize,
     115             :                     eDataType, 0, 0, nullptr);
     116         722 :                 if (eErr != CE_None)
     117             :                 {
     118           0 :                     return false;
     119             :                 }
     120         722 :                 cache->insert(nKey, poValue);
     121             :             }
     122             : 
     123      258167 :             double *padfAsDouble = reinterpret_cast<double *>(padfOut);
     124             :             // Compose the cached block to the final buffer
     125      773619 :             for (int j = 0; j < nLinesToCopy; j++)
     126             :             {
     127     1030908 :                 memcpy(padfAsDouble + ((nFirstLineInOutput + j) * nWidth +
     128      515388 :                                        nFirstColInOutput) *
     129             :                                           nTypeFactor,
     130      515452 :                        poValue->data() +
     131      515452 :                            ((nFirstLineInCachedBlock + j) * nReqXSize +
     132      515388 :                             nFirstColInCachedBlock) *
     133             :                                nTypeFactor,
     134      515452 :                        nColsToCopy * sizeof(T));
     135             :             }
     136             :         }
     137             :     }
     138             : 
     139             : #if 0
     140             :     CPLDebug("RPC_DEM", "DEM for %d,%d,%d,%d", nX, nY, nWidth, nHeight);
     141             :     for(int j = 0; j < nHeight; j++)
     142             :     {
     143             :         std::string osLine;
     144             :         for(int i = 0; i < nWidth; ++i )
     145             :         {
     146             :             if( !osLine.empty() )
     147             :                 osLine += ", ";
     148             :             osLine += std::to_string(padfOut[j * nWidth + i]);
     149             :         }
     150             :         CPLDebug("RPC_DEM", "%s", osLine.c_str());
     151             :     }
     152             : #endif
     153             : 
     154      253682 :     return true;
     155             : }
     156             : 
     157             : template <typename T>
     158      306655 : bool GDALInterpolateAtPointImpl(GDALRasterBand *pBand,
     159             :                                 GDALRIOResampleAlg eResampleAlg,
     160             :                                 std::unique_ptr<DoublePointsCache> &cache,
     161             :                                 const double dfXIn, const double dfYIn, T &out)
     162             : {
     163      306655 :     const gdal::Vector2i rasterSize{pBand->GetXSize(), pBand->GetYSize()};
     164      306655 :     const gdal::Vector2d inLoc{dfXIn, dfYIn};
     165             : 
     166      306655 :     int bGotNoDataValue = FALSE;
     167      306655 :     const double dfNoDataValue = pBand->GetNoDataValue(&bGotNoDataValue);
     168             : 
     169      560357 :     if (inLoc.x() < 0 || inLoc.x() > rasterSize.x() || inLoc.y() < 0 ||
     170      253702 :         inLoc.y() > rasterSize.y())
     171             :     {
     172       52973 :         return FALSE;
     173             :     }
     174             : 
     175             :     // Downgrade the interpolation algorithm if the image is too small
     176      253690 :     if ((rasterSize.x() < 4 || rasterSize.y() < 4) &&
     177           8 :         (eResampleAlg == GRIORA_CubicSpline || eResampleAlg == GRIORA_Cubic))
     178             :     {
     179           0 :         eResampleAlg = GRIORA_Bilinear;
     180             :     }
     181      253682 :     if ((rasterSize.x() < 2 || rasterSize.y() < 2) &&
     182           0 :         eResampleAlg == GRIORA_Bilinear)
     183             :     {
     184           0 :         eResampleAlg = GRIORA_NearestNeighbour;
     185             :     }
     186             : 
     187             :     auto outOfBorderCorrectionSimple =
     188      507228 :         [](int dNew, int nRasterSize, int nKernelsize)
     189             :     {
     190      507228 :         int dOutOfBorder = 0;
     191      507228 :         if (dNew < 0)
     192             :         {
     193       34586 :             dOutOfBorder = dNew;
     194             :         }
     195      507228 :         if (dNew + nKernelsize >= nRasterSize)
     196             :         {
     197       45557 :             dOutOfBorder = dNew + nKernelsize - nRasterSize;
     198             :         }
     199      507228 :         return dOutOfBorder;
     200             :     };
     201             : 
     202      253682 :     auto outOfBorderCorrection =
     203      507228 :         [&outOfBorderCorrectionSimple,
     204             :          &rasterSize](gdal::Vector2i input, int nKernelsize) -> gdal::Vector2i
     205             :     {
     206             :         return {
     207      253614 :             outOfBorderCorrectionSimple(input.x(), rasterSize.x(), nKernelsize),
     208      253614 :             outOfBorderCorrectionSimple(input.y(), rasterSize.y(),
     209      760842 :                                         nKernelsize)};
     210             :     };
     211             : 
     212             :     auto dragReadDataInBorderSimple =
     213     1049058 :         [](T *adfElevData, int dOutOfBorder, int nKernelSize, bool bIsX)
     214             :     {
     215      541826 :         while (dOutOfBorder < 0)
     216             :         {
     217      103848 :             for (int j = 0; j < nKernelSize; j++)
     218      138716 :                 for (int ii = 0; ii < nKernelSize - 1; ii++)
     219             :                 {
     220       69466 :                     const int i = nKernelSize - ii - 2;  // iterate in reverse
     221       69466 :                     const int row_src = bIsX ? j : i;
     222       69466 :                     const int row_dst = bIsX ? j : i + 1;
     223       69466 :                     const int col_src = bIsX ? i : j;
     224       69466 :                     const int col_dst = bIsX ? i + 1 : j;
     225       69466 :                     adfElevData[nKernelSize * row_dst + col_dst] =
     226       69466 :                         adfElevData[nKernelSize * row_src + col_src];
     227             :                 }
     228       34598 :             dOutOfBorder++;
     229             :         }
     230      541417 :         while (dOutOfBorder > 0)
     231             :         {
     232      102585 :             for (int j = 0; j < nKernelSize; j++)
     233      136864 :                 for (int i = 0; i < nKernelSize - 1; i++)
     234             :                 {
     235       68468 :                     const int row_src = bIsX ? j : i + 1;
     236       68468 :                     const int row_dst = bIsX ? j : i;
     237       68468 :                     const int col_src = bIsX ? i + 1 : j;
     238       68468 :                     const int col_dst = bIsX ? i : j;
     239       68468 :                     adfElevData[nKernelSize * row_dst + col_dst] =
     240       68468 :                         adfElevData[nKernelSize * row_src + col_src];
     241             :                 }
     242       34189 :             dOutOfBorder--;
     243             :         }
     244             :     };
     245     1014524 :     auto dragReadDataInBorder = [&dragReadDataInBorderSimple](
     246             :                                     T *adfElevData, gdal::Vector2i dOutOfBorder,
     247             :                                     int nKernelSize) -> void
     248             :     {
     249      253614 :         dragReadDataInBorderSimple(adfElevData, dOutOfBorder.x(), nKernelSize,
     250             :                                    true);
     251      253614 :         dragReadDataInBorderSimple(adfElevData, dOutOfBorder.y(), nKernelSize,
     252             :                                    false);
     253             :     };
     254             : 
     255      523943 :     auto applyBilinearKernel = [&](gdal::Vector2d dfDelta, T *adfValues,
     256             :                                    T &pdfRes) -> bool
     257             :     {
     258      253573 :         if (bGotNoDataValue)
     259             :         {
     260             :             // TODO: We could perhaps use a valid sample if there's one.
     261        4172 :             bool bFoundNoDataElev = false;
     262       20860 :             for (int k_i = 0; k_i < 4; k_i++)
     263             :             {
     264       16688 :                 if (areEqualReal(dfNoDataValue, adfValues[k_i]))
     265          16 :                     bFoundNoDataElev = true;
     266             :             }
     267        4172 :             if (bFoundNoDataElev)
     268             :             {
     269           4 :                 return FALSE;
     270             :             }
     271             :         }
     272      253569 :         const gdal::Vector2d dfDelta1 = 1.0 - dfDelta;
     273             : 
     274      253564 :         const T dfXZ1 =
     275      253569 :             adfValues[0] * dfDelta1.x() + adfValues[1] * dfDelta.x();
     276      253564 :         const T dfXZ2 =
     277      253569 :             adfValues[2] * dfDelta1.x() + adfValues[3] * dfDelta.x();
     278      253569 :         const T dfYZ = dfXZ1 * dfDelta1.y() + dfXZ2 * dfDelta.y();
     279             : 
     280      253569 :         pdfRes = dfYZ;
     281      253569 :         return TRUE;
     282             :     };
     283             : 
     284      255163 :     auto apply4x4Kernel = [&](gdal::Vector2d dfDelta, T *adfValues,
     285             :                               T &pdfRes) -> bool
     286             :     {
     287          41 :         T dfSumH = 0.0;
     288          41 :         T dfSumWeight = 0.0;
     289         205 :         for (int k_i = 0; k_i < 4; k_i++)
     290             :         {
     291             :             // Loop across the X axis.
     292         820 :             for (int k_j = 0; k_j < 4; k_j++)
     293             :             {
     294             :                 // Calculate the weight for the specified pixel according
     295             :                 // to the bicubic b-spline kernel we're using for
     296             :                 // interpolation.
     297         656 :                 const gdal::Vector2i dKernInd = {k_j - 1, k_i - 1};
     298         656 :                 const gdal::Vector2d fPoint = dKernInd.cast<double>() - dfDelta;
     299         656 :                 const double dfPixelWeight =
     300         656 :                     eResampleAlg == GDALRIOResampleAlg::GRIORA_CubicSpline
     301         320 :                         ? CubicSplineKernel(fPoint.x()) *
     302         320 :                               CubicSplineKernel(fPoint.y())
     303         336 :                         : CubicKernel(fPoint.x()) * CubicKernel(fPoint.y());
     304             : 
     305             :                 // Create a sum of all values
     306             :                 // adjusted for the pixel's calculated weight.
     307         656 :                 const T dfElev = adfValues[k_j + k_i * 4];
     308         656 :                 if (bGotNoDataValue && areEqualReal(dfNoDataValue, dfElev))
     309          64 :                     continue;
     310             : 
     311         112 :                 dfSumH += dfElev * dfPixelWeight;
     312         592 :                 dfSumWeight += dfPixelWeight;
     313             :             }
     314             :         }
     315          41 :         if (dfSumWeight == 0.0)
     316             :         {
     317           4 :             return FALSE;
     318             :         }
     319             : 
     320           7 :         pdfRes = dfSumH / dfSumWeight;
     321             : 
     322          37 :         return TRUE;
     323             :     };
     324             : 
     325      253682 :     if (eResampleAlg == GDALRIOResampleAlg::GRIORA_CubicSpline ||
     326      253662 :         eResampleAlg == GDALRIOResampleAlg::GRIORA_Cubic)
     327             :     {
     328             :         // Convert from upper left corner of pixel coordinates to center of
     329             :         // pixel coordinates:
     330          41 :         const gdal::Vector2d df = inLoc - 0.5;
     331          41 :         const gdal::Vector2i d = df.floor().template cast<int>();
     332          41 :         const gdal::Vector2d delta = df - d.cast<double>();
     333          41 :         const gdal::Vector2i dNew = d - 1;
     334          41 :         const int nKernelSize = 4;
     335             :         const gdal::Vector2i dOutOfBorder =
     336          41 :             outOfBorderCorrection(dNew, nKernelSize);
     337             : 
     338             :         // CubicSpline interpolation.
     339          41 :         T adfReadData[16] = {0.0};
     340          41 :         if (!GDALInterpExtractValuesWindow(pBand, cache, dNew - dOutOfBorder,
     341             :                                            {nKernelSize, nKernelSize},
     342             :                                            adfReadData))
     343             :         {
     344           0 :             return FALSE;
     345             :         }
     346          41 :         dragReadDataInBorder(adfReadData, dOutOfBorder, nKernelSize);
     347          41 :         if (!apply4x4Kernel(delta, adfReadData, out))
     348           4 :             return FALSE;
     349             : 
     350          37 :         return TRUE;
     351             :     }
     352      253641 :     else if (eResampleAlg == GDALRIOResampleAlg::GRIORA_Bilinear)
     353             :     {
     354             :         // Convert from upper left corner of pixel coordinates to center of
     355             :         // pixel coordinates:
     356      253573 :         const gdal::Vector2d df = inLoc - 0.5;
     357      253573 :         const gdal::Vector2i d = df.floor().template cast<int>();
     358      253573 :         const gdal::Vector2d delta = df - d.cast<double>();
     359      253573 :         const int nKernelSize = 2;
     360             :         const gdal::Vector2i dOutOfBorder =
     361      253573 :             outOfBorderCorrection(d, nKernelSize);
     362             : 
     363             :         // Bilinear interpolation.
     364      253573 :         T adfReadData[4] = {0.0};
     365      253573 :         if (!GDALInterpExtractValuesWindow(pBand, cache, d - dOutOfBorder,
     366             :                                            {nKernelSize, nKernelSize},
     367             :                                            adfReadData))
     368             :         {
     369           0 :             return FALSE;
     370             :         }
     371      253573 :         dragReadDataInBorder(adfReadData, dOutOfBorder, nKernelSize);
     372      253573 :         if (!applyBilinearKernel(delta, adfReadData, out))
     373           4 :             return FALSE;
     374             : 
     375      253569 :         return TRUE;
     376             :     }
     377             :     else
     378             :     {
     379          68 :         const gdal::Vector2i d = inLoc.cast<int>();
     380          68 :         T adfOut[1] = {};
     381         136 :         if (!GDALInterpExtractValuesWindow(pBand, cache, d, {1, 1}, adfOut) ||
     382          68 :             (bGotNoDataValue && areEqualReal(dfNoDataValue, adfOut[0])))
     383             :         {
     384           5 :             return FALSE;
     385             :         }
     386             : 
     387          63 :         out = adfOut[0];
     388             : 
     389          63 :         return TRUE;
     390             :     }
     391             : }
     392             : 
     393             : /************************************************************************/
     394             : /*                        GDALInterpolateAtPoint()                      */
     395             : /************************************************************************/
     396             : 
     397      306655 : bool GDALInterpolateAtPoint(GDALRasterBand *pBand,
     398             :                             GDALRIOResampleAlg eResampleAlg,
     399             :                             std::unique_ptr<DoublePointsCache> &cache,
     400             :                             const double dfXIn, const double dfYIn,
     401             :                             double *pdfOutputReal, double *pdfOutputImag)
     402             : {
     403             :     const bool bIsComplex =
     404      306655 :         CPL_TO_BOOL(GDALDataTypeIsComplex(pBand->GetRasterDataType()));
     405      306655 :     bool res = TRUE;
     406      306655 :     if (bIsComplex)
     407             :     {
     408          20 :         std::complex<double> out{};
     409          20 :         res = GDALInterpolateAtPointImpl(pBand, eResampleAlg, cache, dfXIn,
     410             :                                          dfYIn, out);
     411          20 :         *pdfOutputReal = out.real();
     412          20 :         if (pdfOutputImag)
     413          20 :             *pdfOutputImag = out.imag();
     414             :     }
     415             :     else
     416             :     {
     417      306635 :         double out{};
     418      306635 :         res = GDALInterpolateAtPointImpl(pBand, eResampleAlg, cache, dfXIn,
     419             :                                          dfYIn, out);
     420      306635 :         *pdfOutputReal = out;
     421      306635 :         if (pdfOutputImag)
     422          93 :             *pdfOutputImag = 0;
     423             :     }
     424      306655 :     return res;
     425             : }

Generated by: LCOV version 1.14