EVOLUTION-MANAGER
Edit File: gdaltransformer.cpp
/****************************************************************************** * * Project: Mapinfo Image Warper * Purpose: Implementation of one or more GDALTrasformerFunc types, including * the GenImgProj (general image reprojector) transformer. * Author: Frank Warmerdam, warmerdam@pobox.com * ****************************************************************************** * Copyright (c) 2002, i3 - information integration and imaging * Fort Collin, CO * Copyright (c) 2008-2013, Even Rouault <even dot rouault at mines-paris dot org> * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. ****************************************************************************/ #define DO_NOT_USE_DEBUG_BOOL // See TODO for bGCPUseOK. #include "cpl_port.h" #include "gdal_alg.h" #include "gdal_alg_priv.h" #include <climits> #include <cmath> #include <cstddef> #include <cstdlib> #include <cstring> #include <algorithm> #include "cpl_conv.h" #include "cpl_error.h" #include "cpl_list.h" #include "cpl_minixml.h" #include "cpl_multiproc.h" #include "cpl_string.h" #include "cpl_vsi.h" #include "gdal.h" #include "gdal_priv.h" #include "ogr_core.h" #include "ogr_spatialref.h" #include "ogr_srs_api.h" CPL_CVSID("$Id: gdaltransformer.cpp 39654 2017-07-24 19:24:32Z rouault $"); CPL_C_START void *GDALDeserializeGCPTransformer( CPLXMLNode *psTree ); void *GDALDeserializeTPSTransformer( CPLXMLNode *psTree ); void *GDALDeserializeGeoLocTransformer( CPLXMLNode *psTree ); void *GDALDeserializeRPCTransformer( CPLXMLNode *psTree ); CPL_C_END static CPLXMLNode *GDALSerializeReprojectionTransformer( void *pTransformArg ); static void *GDALDeserializeReprojectionTransformer( CPLXMLNode *psTree ); static CPLXMLNode *GDALSerializeGenImgProjTransformer( void *pTransformArg ); static void *GDALDeserializeGenImgProjTransformer( CPLXMLNode *psTree ); static void GDALRefreshGenImgProjTransformer(void* hTransformArg); static void * GDALCreateApproxTransformer2( GDALTransformerFunc pfnRawTransformer, void *pRawTransformerArg, double dfMaxErrorForward, double dfMaxErrorReverse ); /************************************************************************/ /* GDALTransformFunc */ /* */ /* Documentation for GDALTransformFunc typedef. */ /************************************************************************/ /*! \typedef int GDALTransformerFunc Generic signature for spatial point transformers. This function signature is used for a variety of functions that accept passed in functions used to transform point locations between two coordinate spaces. The GDALCreateGenImgProjTransformer(), GDALCreateReprojectionTransformer(), GDALCreateGCPTransformer() and GDALCreateApproxTransformer() functions can be used to prepare argument data for some built-in transformers. As well, applications can implement their own transformers to the following signature. \code typedef int (*GDALTransformerFunc)( void *pTransformerArg, int bDstToSrc, int nPointCount, double *x, double *y, double *z, int *panSuccess ); \endcode @param pTransformerArg application supplied callback data used by the transformer. @param bDstToSrc if TRUE the transformation will be from the destination coordinate space to the source coordinate system, otherwise the transformation will be from the source coordinate system to the destination coordinate system. @param nPointCount number of points in the x, y and z arrays. @param x input X coordinates. Results returned in same array. @param y input Y coordinates. Results returned in same array. @param z input Z coordinates. Results returned in same array. @param panSuccess array of ints in which success (TRUE) or failure (FALSE) flags are returned for the translation of each point. @return TRUE if the overall transformation succeeds (though some individual points may have failed) or FALSE if the overall transformation fails. */ /************************************************************************/ /* GDALSuggestedWarpOutput() */ /************************************************************************/ /** * Suggest output file size. * * This function is used to suggest the size, and georeferenced extents * appropriate given the indicated transformation and input file. It walks * the edges of the input file (approximately 20 sample points along each * edge) transforming into output coordinates in order to get an extents box. * * Then a resolution is computed with the intent that the length of the * distance from the top left corner of the output imagery to the bottom right * corner would represent the same number of pixels as in the source image. * Note that if the image is somewhat rotated the diagonal taken isn't of the * whole output bounding rectangle, but instead of the locations where the * top/left and bottom/right corners transform. The output pixel size is * always square. This is intended to approximately preserve the resolution * of the input data in the output file. * * The values returned in padfGeoTransformOut, pnPixels and pnLines are * the suggested number of pixels and lines for the output file, and the * geotransform relating those pixels to the output georeferenced coordinates. * * The trickiest part of using the function is ensuring that the * transformer created is from source file pixel/line coordinates to * output file georeferenced coordinates. This can be accomplished with * GDALCreateGenImgProjTransformer() by passing a NULL for the hDstDS. * * @param hSrcDS the input image (it is assumed the whole input images is * being transformed). * @param pfnTransformer the transformer function. * @param pTransformArg the callback data for the transformer function. * @param padfGeoTransformOut the array of six doubles in which the suggested * geotransform is returned. * @param pnPixels int in which the suggest pixel width of output is returned. * @param pnLines int in which the suggest pixel height of output is returned. * * @return CE_None if successful or CE_Failure otherwise. */ CPLErr CPL_STDCALL GDALSuggestedWarpOutput( GDALDatasetH hSrcDS, GDALTransformerFunc pfnTransformer, void *pTransformArg, double *padfGeoTransformOut, int *pnPixels, int *pnLines ) { VALIDATE_POINTER1( hSrcDS, "GDALSuggestedWarpOutput", CE_Failure ); double adfExtent[4] = {}; return GDALSuggestedWarpOutput2( hSrcDS, pfnTransformer, pTransformArg, padfGeoTransformOut, pnPixels, pnLines, adfExtent, 0 ); } static int GDALSuggestedWarpOutput2_MustAdjustForRightBorder( GDALTransformerFunc pfnTransformer, void *pTransformArg, double* padfExtent, CPL_UNUSED int nPixels, int nLines, double dfPixelSizeX, double dfPixelSizeY ) { double adfX[21] = {}; double adfY[21] = {}; const double dfMaxXOut = padfExtent[2]; const double dfMaxYOut = padfExtent[3]; // Take 20 steps. int nSamplePoints = 0; for( double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05 ) { // Ensure we end exactly at the end. if( dfRatio > 0.99 ) dfRatio = 1.0; // Along right. adfX[nSamplePoints] = dfMaxXOut; adfY[nSamplePoints] = dfMaxYOut - dfPixelSizeY * dfRatio * nLines; nSamplePoints++; } double adfZ[21] = {}; int abSuccess[21] = {}; bool bErr = false; if( !pfnTransformer( pTransformArg, TRUE, nSamplePoints, adfX, adfY, adfZ, abSuccess ) ) { bErr = true; } if( !bErr && !pfnTransformer( pTransformArg, FALSE, nSamplePoints, adfX, adfY, adfZ, abSuccess ) ) { bErr = true; } nSamplePoints = 0; int nBadCount = 0; for( double dfRatio = 0.0; !bErr && dfRatio <= 1.01; dfRatio += 0.05 ) { const double expected_x = dfMaxXOut; const double expected_y = dfMaxYOut - dfPixelSizeY * dfRatio * nLines; if( fabs(adfX[nSamplePoints] - expected_x) > dfPixelSizeX || fabs(adfY[nSamplePoints] - expected_y) > dfPixelSizeY ) nBadCount++; nSamplePoints++; } return nBadCount == nSamplePoints; } static int GDALSuggestedWarpOutput2_MustAdjustForBottomBorder( GDALTransformerFunc pfnTransformer, void *pTransformArg, double* padfExtent, int nPixels, CPL_UNUSED int nLines, double dfPixelSizeX, double dfPixelSizeY) { double adfX[21] = {}; double adfY[21] = {}; const double dfMinXOut = padfExtent[0]; const double dfMinYOut = padfExtent[1]; // Take 20 steps. int nSamplePoints = 0; for( double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05 ) { // Ensure we end exactly at the end. if( dfRatio > 0.99 ) dfRatio = 1.0; // Along right. adfX[nSamplePoints] = dfMinXOut + dfPixelSizeX * dfRatio * nPixels; adfY[nSamplePoints] = dfMinYOut; nSamplePoints++; } double adfZ[21] = {}; int abSuccess[21] = {}; bool bErr = false; if( !pfnTransformer( pTransformArg, TRUE, nSamplePoints, adfX, adfY, adfZ, abSuccess ) ) { bErr = true; } if( !bErr && !pfnTransformer( pTransformArg, FALSE, nSamplePoints, adfX, adfY, adfZ, abSuccess ) ) { bErr = true; } nSamplePoints = 0; int nBadCount = 0; for( double dfRatio = 0.0; !bErr && dfRatio <= 1.01; dfRatio += 0.05 ) { const double expected_x = dfMinXOut + dfPixelSizeX * dfRatio * nPixels; const double expected_y = dfMinYOut; if( fabs(adfX[nSamplePoints] - expected_x) > dfPixelSizeX || fabs(adfY[nSamplePoints] - expected_y) > dfPixelSizeY ) nBadCount++; nSamplePoints++; } return nBadCount == nSamplePoints; } /************************************************************************/ /* GDALSuggestedWarpOutput2() */ /************************************************************************/ /** * Suggest output file size. * * This function is used to suggest the size, and georeferenced extents * appropriate given the indicated transformation and input file. It walks * the edges of the input file (approximately 20 sample points along each * edge) transforming into output coordinates in order to get an extents box. * * Then a resolution is computed with the intent that the length of the * distance from the top left corner of the output imagery to the bottom right * corner would represent the same number of pixels as in the source image. * Note that if the image is somewhat rotated the diagonal taken isn't of the * whole output bounding rectangle, but instead of the locations where the * top/left and bottom/right corners transform. The output pixel size is * always square. This is intended to approximately preserve the resolution * of the input data in the output file. * * The values returned in padfGeoTransformOut, pnPixels and pnLines are * the suggested number of pixels and lines for the output file, and the * geotransform relating those pixels to the output georeferenced coordinates. * * The trickiest part of using the function is ensuring that the * transformer created is from source file pixel/line coordinates to * output file georeferenced coordinates. This can be accomplished with * GDALCreateGenImgProjTransformer() by passing a NULL for the hDstDS. * * @param hSrcDS the input image (it is assumed the whole input images is * being transformed). * @param pfnTransformer the transformer function. * @param pTransformArg the callback data for the transformer function. * @param padfGeoTransformOut the array of six doubles in which the suggested * geotransform is returned. * @param pnPixels int in which the suggest pixel width of output is returned. * @param pnLines int in which the suggest pixel height of output is returned. * @param padfExtent Four entry array to return extents as (xmin, ymin, xmax, * ymax). * @param nOptions Options, currently always zero. * * @return CE_None if successful or CE_Failure otherwise. */ CPLErr CPL_STDCALL GDALSuggestedWarpOutput2( GDALDatasetH hSrcDS, GDALTransformerFunc pfnTransformer, void *pTransformArg, double *padfGeoTransformOut, int *pnPixels, int *pnLines, double *padfExtent, CPL_UNUSED int nOptions ) { VALIDATE_POINTER1( hSrcDS, "GDALSuggestedWarpOutput2", CE_Failure ); /* -------------------------------------------------------------------- */ /* Setup sample points all around the edge of the input raster. */ /* -------------------------------------------------------------------- */ if( pfnTransformer == GDALGenImgProjTransform || pfnTransformer == GDALApproxTransform ) { // In case CHECK_WITH_INVERT_PROJ has been modified. GDALRefreshGenImgProjTransformer(pTransformArg); } const int nInXSize = GDALGetRasterXSize( hSrcDS ); const int nInYSize = GDALGetRasterYSize( hSrcDS ); const int N_PIXELSTEP = 50; int nSteps = static_cast<int>( static_cast<double>(std::min(nInYSize, nInXSize)) / N_PIXELSTEP + 0.5); if( nSteps < 20 ) nSteps = 20; nSteps = std::min(nSteps, 100); // TODO(rouault): How is this goto retry supposed to work? Added in r20537. // Does redoing the same malloc multiple times work? If it is needed, can // it be converted to a tigher while loop around the MALLOC3s and free? Is // the point to try with the full requested steps. Then, if there is not // enough memory, back off and try with just 20 steps? retry: int nSampleMax = (nSteps + 1)*(nSteps + 1); int *pabSuccess = NULL; double dfRatio = 0.0; double dfStep = 1.0 / nSteps; double *padfX = NULL; double *padfY = NULL; double *padfZ = NULL; double *padfXRevert = NULL; double *padfYRevert = NULL; double *padfZRevert = NULL; pabSuccess = static_cast<int *>( VSI_MALLOC3_VERBOSE(sizeof(int), nSteps + 1, nSteps + 1)); padfX = static_cast<double *>( VSI_MALLOC3_VERBOSE(sizeof(double) * 3, nSteps + 1, nSteps + 1)); padfXRevert = static_cast<double *>( VSI_MALLOC3_VERBOSE(sizeof(double) * 3, nSteps + 1, nSteps + 1)); if( pabSuccess == NULL || padfX == NULL || padfXRevert == NULL ) { CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); if( nSteps > 20 ) { nSteps = 20; goto retry; } return CE_Failure; } padfY = padfX + nSampleMax; padfZ = padfX + nSampleMax * 2; padfYRevert = padfXRevert + nSampleMax; padfZRevert = padfXRevert + nSampleMax * 2; // Take N_STEPS steps. for( int iStep = 0; iStep <= nSteps; iStep++ ) { dfRatio = (iStep == nSteps) ? 1.0 : iStep * dfStep; // Along top. padfX[iStep] = dfRatio * nInXSize; padfY[iStep] = 0.0; padfZ[iStep] = 0.0; // Along bottom. padfX[nSteps + 1 + iStep] = dfRatio * nInXSize; padfY[nSteps + 1 + iStep] = nInYSize; padfZ[nSteps + 1 + iStep] = 0.0; // Along left. padfX[2 * (nSteps + 1) + iStep] = 0.0; padfY[2 * (nSteps + 1) + iStep] = dfRatio * nInYSize; padfZ[2 * (nSteps + 1) + iStep] = 0.0; // Along right. padfX[3 * (nSteps + 1) + iStep] = nInXSize; padfY[3 * (nSteps + 1) + iStep] = dfRatio * nInYSize; padfZ[3 * (nSteps + 1) + iStep] = 0.0; } int nSamplePoints = 4 * (nSteps + 1); memset( pabSuccess, 1, sizeof(int) * nSampleMax ); /* -------------------------------------------------------------------- */ /* Transform them to the output coordinate system. */ /* -------------------------------------------------------------------- */ int nFailedCount = 0; if( !pfnTransformer( pTransformArg, FALSE, nSamplePoints, padfX, padfY, padfZ, pabSuccess ) ) { CPLError( CE_Failure, CPLE_AppDefined, "GDALSuggestedWarpOutput() failed because the passed " "transformer failed." ); CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); return CE_Failure; } for( int i = 0; i < nSamplePoints; i++ ) { if( !pabSuccess[i] ) nFailedCount++; } /* -------------------------------------------------------------------- */ /* Check if the computed target coordinates are revertable. */ /* If not, try the detailed grid sampling. */ /* -------------------------------------------------------------------- */ if( nFailedCount == 0 ) { memcpy(padfXRevert, padfX, nSamplePoints * sizeof(double)); memcpy(padfYRevert, padfY, nSamplePoints * sizeof(double)); memcpy(padfZRevert, padfZ, nSamplePoints * sizeof(double)); if( !pfnTransformer(pTransformArg, TRUE, nSamplePoints, padfXRevert, padfYRevert, padfZRevert, pabSuccess) ) { nFailedCount = 1; } else { for( int i = 0; nFailedCount == 0 && i < nSamplePoints; i++ ) { if( !pabSuccess[i] ) { nFailedCount++; break; } dfRatio = (i % (nSteps + 1)) * dfStep; if( dfRatio > 0.99 ) dfRatio = 1.0; double dfExpectedX = 0.0; double dfExpectedY = 0.0; if( i < nSteps + 1 ) { dfExpectedX = dfRatio * nInXSize; } else if( i < 2 * (nSteps + 1) ) { dfExpectedX = dfRatio * nInXSize; dfExpectedY = nInYSize; } else if( i < 3 * (nSteps + 1) ) { dfExpectedY = dfRatio * nInYSize; } else { dfExpectedX = nInXSize; dfExpectedY = dfRatio * nInYSize; } if( fabs(padfXRevert[i] - dfExpectedX) > nInXSize / static_cast<double>(nSteps) || fabs(padfYRevert[i] - dfExpectedY) > nInYSize / static_cast<double>(nSteps) ) nFailedCount++; } if( nFailedCount != 0 ) CPLDebug("WARP", "At least one point failed after revert transform"); } } else { CPLDebug("WARP", "At least one point failed after direct transform"); } /* -------------------------------------------------------------------- */ /* If any of the edge points failed to transform, we need to */ /* build a fairly detailed internal grid of points instead to */ /* help identify the area that is transformable. */ /* -------------------------------------------------------------------- */ if( nFailedCount > 0 ) { nSamplePoints = 0; // Take N_STEPS steps. for( int iStep = 0; iStep <= nSteps; iStep++ ) { dfRatio = (iStep == nSteps) ? 1.0 : iStep * dfStep; for( int iStep2 = 0; iStep2 <= nSteps; iStep2++ ) { const double dfRatio2 = iStep2 == nSteps ? 1.0 : iStep2 * dfStep; // From top to bottom, from left to right. padfX[nSamplePoints] = dfRatio2 * nInXSize; padfY[nSamplePoints] = dfRatio * nInYSize; padfZ[nSamplePoints] = 0.0; nSamplePoints++; } } CPLAssert( nSamplePoints == nSampleMax ); if( !pfnTransformer( pTransformArg, FALSE, nSamplePoints, padfX, padfY, padfZ, pabSuccess ) ) { CPLError(CE_Failure, CPLE_AppDefined, "GDALSuggestedWarpOutput() failed because the passed" "transformer failed."); CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); return CE_Failure; } } /* -------------------------------------------------------------------- */ /* Collect the bounds, ignoring any failed points. */ /* -------------------------------------------------------------------- */ double dfMinXOut = 0.0; double dfMinYOut = 0.0; double dfMaxXOut = 0.0; double dfMaxYOut = 0.0; bool bGotInitialPoint = false; nFailedCount = 0; for( int i = 0; i < nSamplePoints; i++ ) { int x_i = 0; int y_i = 0; if( nSamplePoints == nSampleMax ) { x_i = i % (nSteps + 1); y_i = i / (nSteps + 1); } else { if( i < 2 * (nSteps + 1 ) ) { x_i = i % (nSteps + 1); y_i = (i < nSteps + 1) ? 0 : nSteps; } else x_i = y_i = 0; } if( x_i > 0 && (pabSuccess[i-1] || pabSuccess[i]) ) { double x_out_before = padfX[i-1]; double x_out_after = padfX[i]; int nIter = 0; double x_in_before = static_cast<double>(x_i - 1) * nInXSize / nSteps; double x_in_after = static_cast<double>(x_i) * nInXSize / nSteps; int valid_before = pabSuccess[i-1]; int valid_after = pabSuccess[i]; // Detect discontinuity in target coordinates when the target x // coordinates change sign. This may be a false positive when the // target tx is around 0 Dichotomic search to reduce the interval // to near the discontinuity and get a better out extent. while( (!valid_before || !valid_after || x_out_before * x_out_after < 0.0) && nIter < 16 ) { double x = (x_in_before + x_in_after) / 2.0; double y = static_cast<double>(y_i) * nInYSize / nSteps; double z = 0.0; int bSuccess = TRUE; if( !pfnTransformer( pTransformArg, FALSE, 1, &x, &y, &z, &bSuccess ) || !bSuccess ) { if( !valid_before ) { x_in_before = (x_in_before + x_in_after) / 2.0; } else if( !valid_after ) { x_in_after = (x_in_before + x_in_after) / 2.0; } else { break; } } else { if( !bGotInitialPoint ) { bGotInitialPoint = true; dfMinXOut = x; dfMaxXOut = x; dfMinYOut = y; dfMaxYOut = y; } else { dfMinXOut = std::min(dfMinXOut, x); dfMinYOut = std::min(dfMinYOut, y); dfMaxXOut = std::max(dfMaxXOut, x); dfMaxYOut = std::max(dfMaxYOut, y); } if( !valid_before || x_out_before * x < 0 ) { valid_after = TRUE; x_in_after = (x_in_before + x_in_after) / 2.0; x_out_after = x; } else { valid_before = TRUE; x_out_before = x; x_in_before = (x_in_before + x_in_after) / 2.0; } } nIter++; } } if( !pabSuccess[i] ) { nFailedCount++; continue; } if( !bGotInitialPoint ) { bGotInitialPoint = true; dfMinXOut = padfX[i]; dfMaxXOut = padfX[i]; dfMinYOut = padfY[i]; dfMaxYOut = padfY[i]; } else { dfMinXOut = std::min(dfMinXOut, padfX[i]); dfMinYOut = std::min(dfMinYOut, padfY[i]); dfMaxXOut = std::max(dfMaxXOut, padfX[i]); dfMaxYOut = std::max(dfMaxYOut, padfY[i]); } } if( nFailedCount > nSamplePoints - 10 ) { CPLError(CE_Failure, CPLE_AppDefined, "Too many points (%d out of %d) failed to transform, " "unable to compute output bounds.", nFailedCount, nSamplePoints); CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); return CE_Failure; } if( nFailedCount > 0 ) CPLDebug("GDAL", "GDALSuggestedWarpOutput(): %d out of %d points failed to " "transform.", nFailedCount, nSamplePoints); /* -------------------------------------------------------------------- */ /* Compute the distance in "georeferenced" units from the top */ /* corner of the transformed input image to the bottom left */ /* corner of the transformed input. Use this distance to */ /* compute an approximate pixel size in the output */ /* georeferenced coordinates. */ /* -------------------------------------------------------------------- */ double dfDiagonalDist = 0.0; double dfDeltaX = 0.0; double dfDeltaY = 0.0; if( pabSuccess[0] && pabSuccess[nSamplePoints - 1] ) { dfDeltaX = padfX[nSamplePoints-1] - padfX[0]; dfDeltaY = padfY[nSamplePoints-1] - padfY[0]; // In some cases this can result in 0 values. See #5980 // Fallback to safer method in that case. } if( dfDeltaX == 0.0 || dfDeltaY == 0.0 ) { dfDeltaX = dfMaxXOut - dfMinXOut; dfDeltaY = dfMaxYOut - dfMinYOut; } dfDiagonalDist = sqrt( dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY ); /* -------------------------------------------------------------------- */ /* Compute a pixel size from this. */ /* -------------------------------------------------------------------- */ const double dfPixelSize = dfDiagonalDist / sqrt(static_cast<double>(nInXSize) * nInXSize + static_cast<double>(nInYSize) * nInYSize); const double dfPixels = (dfMaxXOut - dfMinXOut) / dfPixelSize; const double dfLines = (dfMaxYOut - dfMinYOut) / dfPixelSize; if( dfPixels > INT_MAX - 1 || dfLines > INT_MAX - 1 ) { CPLError( CE_Failure, CPLE_AppDefined, "Computed dimensions are too big : %.0f x %.0f", dfPixels + 0.5, dfLines + 0.5 ); CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); return CE_Failure; } *pnPixels = static_cast<int>(dfPixels + 0.5); *pnLines = static_cast<int>(dfLines + 0.5); double dfPixelSizeX = dfPixelSize; double dfPixelSizeY = dfPixelSize; double adfExtent[4] = {}; const double adfRatioArray[] = { 0.000, 0.001, 0.010, 0.100, 1.000 }; size_t nRetry = 0; /* -------------------------------------------------------------------- */ /* Check that the right border is not completely out of source */ /* image. If so, adjust the x pixel size a bit in the hope it will */ /* fit. */ /* -------------------------------------------------------------------- */ for( nRetry = 0; nRetry < CPL_ARRAYSIZE(adfRatioArray); nRetry++ ) { const double dfTryPixelSizeX = dfPixelSizeX - dfPixelSizeX * adfRatioArray[nRetry] / *pnPixels; adfExtent[0] = dfMinXOut; adfExtent[1] = dfMaxYOut - (*pnLines) * dfPixelSizeY; adfExtent[2] = dfMinXOut + (*pnPixels) * dfTryPixelSizeX; adfExtent[3] = dfMaxYOut; if( !GDALSuggestedWarpOutput2_MustAdjustForRightBorder( pfnTransformer, pTransformArg, adfExtent, *pnPixels, *pnLines, dfTryPixelSizeX, dfPixelSizeY) ) { dfPixelSizeX = dfTryPixelSizeX; break; } } /* -------------------------------------------------------------------- */ /* Check that the bottom border is not completely out of source */ /* image. If so, adjust the y pixel size a bit in the hope it will */ /* fit. */ /* -------------------------------------------------------------------- */ for( nRetry = 0; nRetry < CPL_ARRAYSIZE(adfRatioArray); nRetry++ ) { const double dfTryPixelSizeY = dfPixelSizeY - dfPixelSizeY * adfRatioArray[nRetry] / *pnLines; adfExtent[0] = dfMinXOut; adfExtent[1] = dfMaxYOut - (*pnLines) * dfTryPixelSizeY; adfExtent[2] = dfMinXOut + (*pnPixels) * dfPixelSizeX; adfExtent[3] = dfMaxYOut; if( !GDALSuggestedWarpOutput2_MustAdjustForBottomBorder( pfnTransformer, pTransformArg, adfExtent, *pnPixels, *pnLines, dfPixelSizeX, dfTryPixelSizeY) ) { dfPixelSizeY = dfTryPixelSizeY; break; } } /* -------------------------------------------------------------------- */ /* Recompute some bounds so that all return values are consistent */ /* -------------------------------------------------------------------- */ dfMaxXOut = dfMinXOut + (*pnPixels) * dfPixelSizeX; dfMinYOut = dfMaxYOut - (*pnLines) * dfPixelSizeY; /* -------------------------------------------------------------------- */ /* Return raw extents. */ /* -------------------------------------------------------------------- */ padfExtent[0] = dfMinXOut; padfExtent[1] = dfMinYOut; padfExtent[2] = dfMaxXOut; padfExtent[3] = dfMaxYOut; /* -------------------------------------------------------------------- */ /* Set the output geotransform. */ /* -------------------------------------------------------------------- */ padfGeoTransformOut[0] = dfMinXOut; padfGeoTransformOut[1] = dfPixelSizeX; padfGeoTransformOut[2] = 0.0; padfGeoTransformOut[3] = dfMaxYOut; padfGeoTransformOut[4] = 0.0; padfGeoTransformOut[5] = - dfPixelSizeY; CPLFree( padfX ); CPLFree( padfXRevert ); CPLFree( pabSuccess ); return CE_None; } /************************************************************************/ /* ==================================================================== */ /* GDALGenImgProjTransformer */ /* ==================================================================== */ /************************************************************************/ typedef struct { GDALTransformerInfo sTI; double adfSrcGeoTransform[6]; double adfSrcInvGeoTransform[6]; void *pSrcTransformArg; GDALTransformerFunc pSrcTransformer; void *pReprojectArg; GDALTransformerFunc pReproject; double adfDstGeoTransform[6]; double adfDstInvGeoTransform[6]; void *pDstTransformArg; GDALTransformerFunc pDstTransformer; } GDALGenImgProjTransformInfo; /************************************************************************/ /* GDALCreateSimilarGenImgProjTransformer() */ /************************************************************************/ static void * GDALCreateSimilarGenImgProjTransformer( void *hTransformArg, double dfRatioX, double dfRatioY ) { VALIDATE_POINTER1( hTransformArg, "GDALCreateSimilarGenImgProjTransformer", NULL ); GDALGenImgProjTransformInfo *psInfo = static_cast<GDALGenImgProjTransformInfo *>(hTransformArg); GDALGenImgProjTransformInfo *psClonedInfo = static_cast<GDALGenImgProjTransformInfo *>( CPLMalloc(sizeof(GDALGenImgProjTransformInfo))); memcpy(psClonedInfo, psInfo, sizeof(GDALGenImgProjTransformInfo)); if( psClonedInfo->pSrcTransformArg ) psClonedInfo->pSrcTransformArg = GDALCreateSimilarTransformer( psInfo->pSrcTransformArg, dfRatioX, dfRatioY ); else if( dfRatioX != 1.0 || dfRatioY != 1.0 ) { if( psClonedInfo->adfSrcGeoTransform[2] == 0.0 && psClonedInfo->adfSrcGeoTransform[4] == 0.0 ) { psClonedInfo->adfSrcGeoTransform[1] *= dfRatioX; psClonedInfo->adfSrcGeoTransform[5] *= dfRatioY; } else { // If the x and y ratios are not equal, then we cannot really // compute a geotransform. psClonedInfo->adfSrcGeoTransform[1] *= dfRatioX; psClonedInfo->adfSrcGeoTransform[2] *= dfRatioX; psClonedInfo->adfSrcGeoTransform[4] *= dfRatioX; psClonedInfo->adfSrcGeoTransform[5] *= dfRatioX; } if( !GDALInvGeoTransform( psClonedInfo->adfSrcGeoTransform, psClonedInfo->adfSrcInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); GDALDestroyGenImgProjTransformer( psClonedInfo ); return NULL; } } if( psClonedInfo->pReprojectArg ) psClonedInfo->pReprojectArg = GDALCloneTransformer( psInfo->pReprojectArg ); if( psClonedInfo->pDstTransformArg ) psClonedInfo->pDstTransformArg = GDALCloneTransformer( psInfo->pDstTransformArg ); return psClonedInfo; } /************************************************************************/ /* GDALCreateGenImgProjTransformer() */ /************************************************************************/ /** * Create image to image transformer. * * This function creates a transformation object that maps from pixel/line * coordinates on one image to pixel/line coordinates on another image. The * images may potentially be georeferenced in different coordinate systems, * and may used GCPs to map between their pixel/line coordinates and * georeferenced coordinates (as opposed to the default assumption that their * geotransform should be used). * * This transformer potentially performs three concatenated transformations. * * The first stage is from source image pixel/line coordinates to source * image georeferenced coordinates, and may be done using the geotransform, * or if not defined using a polynomial model derived from GCPs. If GCPs * are used this stage is accomplished using GDALGCPTransform(). * * The second stage is to change projections from the source coordinate system * to the destination coordinate system, assuming they differ. This is * accomplished internally using GDALReprojectionTransform(). * * The third stage is converting from destination image georeferenced * coordinates to destination image coordinates. This is done using the * destination image geotransform, or if not available, using a polynomial * model derived from GCPs. If GCPs are used this stage is accomplished using * GDALGCPTransform(). This stage is skipped if hDstDS is NULL when the * transformation is created. * * @param hSrcDS source dataset, or NULL. * @param pszSrcWKT the coordinate system for the source dataset. If NULL, * it will be read from the dataset itself. * @param hDstDS destination dataset (or NULL). * @param pszDstWKT the coordinate system for the destination dataset. If * NULL, and hDstDS not NULL, it will be read from the destination dataset. * @param bGCPUseOK TRUE if GCPs should be used if the geotransform is not * available on the source dataset (not destination). * @param dfGCPErrorThreshold ignored/deprecated. * @param nOrder the maximum order to use for GCP derived polynomials if * possible. Use 0 to autoselect, or -1 for thin plate splines. * * @return handle suitable for use GDALGenImgProjTransform(), and to be * deallocated with GDALDestroyGenImgProjTransformer(). */ void * GDALCreateGenImgProjTransformer( GDALDatasetH hSrcDS, const char *pszSrcWKT, GDALDatasetH hDstDS, const char *pszDstWKT, int bGCPUseOK, CPL_UNUSED double dfGCPErrorThreshold, int nOrder ) { char **papszOptions = NULL; if( pszSrcWKT != NULL ) papszOptions = CSLSetNameValue( papszOptions, "SRC_SRS", pszSrcWKT ); if( pszDstWKT != NULL ) papszOptions = CSLSetNameValue( papszOptions, "DST_SRS", pszDstWKT ); if( !bGCPUseOK ) papszOptions = CSLSetNameValue( papszOptions, "GCPS_OK", "FALSE" ); if( nOrder != 0 ) papszOptions = CSLSetNameValue( papszOptions, "MAX_GCP_ORDER", CPLString().Printf("%d", nOrder) ); void *pRet = GDALCreateGenImgProjTransformer2( hSrcDS, hDstDS, papszOptions ); CSLDestroy( papszOptions ); return pRet; } /************************************************************************/ /* InsertCenterLong() */ /* */ /* Insert a CENTER_LONG Extension entry on a GEOGCS to indicate */ /* the center longitude of the dataset for wrapping purposes. */ /************************************************************************/ static CPLString InsertCenterLong( GDALDatasetH hDS, CPLString osWKT ) { if( !STARTS_WITH_CI(osWKT.c_str(), "GEOGCS[") ) return osWKT; if( strstr(osWKT, "EXTENSION[\"CENTER_LONG") != NULL ) return osWKT; /* -------------------------------------------------------------------- */ /* For now we only do this if we have a geotransform since */ /* other forms require a bunch of extra work. */ /* -------------------------------------------------------------------- */ double adfGeoTransform[6] = {}; if( GDALGetGeoTransform( hDS, adfGeoTransform ) != CE_None ) return osWKT; /* -------------------------------------------------------------------- */ /* Compute min/max longitude based on testing the four corners. */ /* -------------------------------------------------------------------- */ const int nXSize = GDALGetRasterXSize( hDS ); const int nYSize = GDALGetRasterYSize( hDS ); const double dfMinLong = std::min(std::min(adfGeoTransform[0] + 0 * adfGeoTransform[1] + 0 * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + 0 * adfGeoTransform[2]), std::min(adfGeoTransform[0] + 0 * adfGeoTransform[1] + nYSize * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + nYSize * adfGeoTransform[2])); const double dfMaxLong = std::max(std::max(adfGeoTransform[0] + 0 * adfGeoTransform[1] + 0 * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + 0 * adfGeoTransform[2]), std::max(adfGeoTransform[0] + 0 * adfGeoTransform[1] + nYSize * adfGeoTransform[2], adfGeoTransform[0] + nXSize * adfGeoTransform[1] + nYSize * adfGeoTransform[2])); if( dfMaxLong - dfMinLong > 360.0 ) return osWKT; /* -------------------------------------------------------------------- */ /* Insert center long. */ /* -------------------------------------------------------------------- */ OGRSpatialReference oSRS( osWKT ); const double dfCenterLong = (dfMaxLong + dfMinLong) / 2.0; OGR_SRSNode *poExt = new OGR_SRSNode( "EXTENSION" ); poExt->AddChild( new OGR_SRSNode( "CENTER_LONG" ) ); poExt->AddChild( new OGR_SRSNode( CPLString().Printf("%g", dfCenterLong) )); oSRS.GetRoot()->AddChild( poExt->Clone() ); delete poExt; /* -------------------------------------------------------------------- */ /* Convert back to wkt. */ /* -------------------------------------------------------------------- */ char *pszWKT = NULL; oSRS.exportToWkt( &pszWKT ); osWKT = pszWKT; CPLFree( pszWKT ); return osWKT; } /************************************************************************/ /* GDALCreateGenImgProjTransformerInternal() */ /************************************************************************/ static GDALGenImgProjTransformInfo* GDALCreateGenImgProjTransformerInternal() { /* -------------------------------------------------------------------- */ /* Initialize the transform info. */ /* -------------------------------------------------------------------- */ GDALGenImgProjTransformInfo* psInfo = static_cast<GDALGenImgProjTransformInfo *>( CPLCalloc(sizeof(GDALGenImgProjTransformInfo), 1)); memcpy( psInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE) ); psInfo->sTI.pszClassName = "GDALGenImgProjTransformer"; psInfo->sTI.pfnTransform = GDALGenImgProjTransform; psInfo->sTI.pfnCleanup = GDALDestroyGenImgProjTransformer; psInfo->sTI.pfnSerialize = GDALSerializeGenImgProjTransformer; psInfo->sTI.pfnCreateSimilar = GDALCreateSimilarGenImgProjTransformer; return psInfo; } /************************************************************************/ /* GDALCreateGenImgProjTransformer2() */ /************************************************************************/ /** * Create image to image transformer. * * This function creates a transformation object that maps from pixel/line * coordinates on one image to pixel/line coordinates on another image. The * images may potentially be georeferenced in different coordinate systems, * and may used GCPs to map between their pixel/line coordinates and * georeferenced coordinates (as opposed to the default assumption that their * geotransform should be used). * * This transformer potentially performs three concatenated transformations. * * The first stage is from source image pixel/line coordinates to source * image georeferenced coordinates, and may be done using the geotransform, * or if not defined using a polynomial model derived from GCPs. If GCPs * are used this stage is accomplished using GDALGCPTransform(). * * The second stage is to change projections from the source coordinate system * to the destination coordinate system, assuming they differ. This is * accomplished internally using GDALReprojectionTransform(). * * The third stage is converting from destination image georeferenced * coordinates to destination image coordinates. This is done using the * destination image geotransform, or if not available, using a polynomial * model derived from GCPs. If GCPs are used this stage is accomplished using * GDALGCPTransform(). This stage is skipped if hDstDS is NULL when the * transformation is created. * * Supported Options (specified with the -to switch of gdalwarp for example): * <ul> * <li> SRC_SRS: WKT SRS to be used as an override for hSrcDS. * <li> DST_SRS: WKT SRS to be used as an override for hDstDS. * <li> GCPS_OK: If false, GCPs will not be used, default is TRUE. * <li> REFINE_MINIMUM_GCPS: The minimum amount of GCPs that should be available * after the refinement. * <li> REFINE_TOLERANCE: The tolerance that specifies when a GCP will be * eliminated. * <li> MAX_GCP_ORDER: the maximum order to use for GCP derived polynomials if * possible. The default is to autoselect based on the number of GCPs. * A value of -1 triggers use of Thin Plate Spline instead of polynomials. * <li> SRC_METHOD: may have a value which is one of GEOTRANSFORM, * GCP_POLYNOMIAL, GCP_TPS, GEOLOC_ARRAY, RPC to force only one geolocation * method to be considered on the source dataset. Will be used for pixel/line * to georef transformation on the source dataset. NO_GEOTRANSFORM can be * used to specify the identity geotransform (ungeoreference image) * <li> DST_METHOD: may have a value which is one of GEOTRANSFORM, * GCP_POLYNOMIAL, GCP_TPS, GEOLOC_ARRAY, RPC to force only one geolocation * method to be considered on the target dataset. Will be used for pixel/line * to georef transformation on the destination dataset. NO_GEOTRANSFORM can be * used to specify the identity geotransform (ungeoreference image) * <li> RPC_HEIGHT: A fixed height to be used with RPC calculations. * <li> RPC_DEM: The name of a DEM file to be used with RPC calculations. * <li> Other RPC related options. See GDALCreateRPCTransformer() * <li> INSERT_CENTER_LONG: May be set to FALSE to disable setting up a * CENTER_LONG value on the coordinate system to rewrap things around the * center of the image. * <li> SRC_APPROX_ERROR_IN_SRS_UNIT=err_threshold_in_SRS_units. (GDAL >= 2.2) Use an * approximate transformer for the source transformer. Must be defined together * with SRC_APPROX_ERROR_IN_PIXEL to be taken into account. * <li> SRC_APPROX_ERROR_IN_PIXEL=err_threshold_in_pixel. (GDAL >= 2.2) Use an * approximate transformer for the source transformer.. Must be defined together * with SRC_APPROX_ERROR_IN_SRS_UNIT to be taken into account. * <li> DST_APPROX_ERROR_IN_SRS_UNIT=err_threshold_in_SRS_units. (GDAL >= 2.2) Use an * approximate transformer for the destination transformer. Must be defined together * with DST_APPROX_ERROR_IN_PIXEL to be taken into account. * <li> DST_APPROX_ERROR_IN_PIXEL=err_threshold_in_pixel. (GDAL >= 2.2) Use an * approximate transformer for the destination transformer. Must be defined together * with DST_APPROX_ERROR_IN_SRS_UNIT to be taken into account. * <li> REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT=err_threshold_in_src_SRS_units. * (GDAL >= 2.2) Use an approximate transformer for the coordinate reprojection. * Must be used together with REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT to be taken * into account. * <li> REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT=err_threshold_in_dst_SRS_units. * (GDAL >= 2.2) Use an approximate transformer for the coordinate reprojection. * Must be used together with REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT to be taken * into account. * </ul> * * The use case for the *_APPROX_ERROR_* options is when defining an approximate * transformer on top of the GenImgProjTransformer globally is not practical. * Such a use case is when the source dataset has RPC with a RPC DEM. In such * case we don't want to use the approximate transformer on the RPC transformation, * as the RPC DEM generally involves non-linearities that the approximate * transformer will not detect. In such case, we must a non-approximated * GenImgProjTransformer, but it might be worthwile to use approximate sub- * transformers, for example on coordinate reprojection. For example if * warping from a source dataset with RPC to a destination dataset with * a UTM projection, since the inverse UTM transformation is rather costly. * In which case, one can use the REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT and * REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT options. * * @param hSrcDS source dataset, or NULL. * @param hDstDS destination dataset (or NULL). * @param papszOptions NULL-terminated list of string options (or NULL). * * @return handle suitable for use GDALGenImgProjTransform(), and to be * deallocated with GDALDestroyGenImgProjTransformer() or NULL on failure. */ void * GDALCreateGenImgProjTransformer2( GDALDatasetH hSrcDS, GDALDatasetH hDstDS, char **papszOptions ) { char **papszMD = NULL; GDALRPCInfo sRPCInfo; const char *pszMethod = CSLFetchNameValue( papszOptions, "SRC_METHOD" ); if( pszMethod == NULL ) pszMethod = CSLFetchNameValue( papszOptions, "METHOD" ); const char *pszSrcWKT = CSLFetchNameValue( papszOptions, "SRC_SRS" ); const char *pszDstWKT = CSLFetchNameValue( papszOptions, "DST_SRS" ); const char *pszValue = CSLFetchNameValue( papszOptions, "MAX_GCP_ORDER" ); const int nOrder = pszValue ? atoi(pszValue) : 0; pszValue = CSLFetchNameValue( papszOptions, "GCPS_OK" ); // TODO(schwehr): Why does this upset DEBUG_BOOL? const bool bGCPUseOK = pszValue ? CPLTestBool(pszValue) : true; pszValue = CSLFetchNameValue( papszOptions, "REFINE_MINIMUM_GCPS" ); const int nMinimumGcps = pszValue ? atoi(pszValue) : -1; pszValue = CSLFetchNameValue( papszOptions, "REFINE_TOLERANCE" ); const bool bRefine = pszValue != NULL; const double dfTolerance = pszValue ? CPLAtof(pszValue) : 0.0; /* -------------------------------------------------------------------- */ /* Initialize the transform info. */ /* -------------------------------------------------------------------- */ GDALGenImgProjTransformInfo *psInfo = GDALCreateGenImgProjTransformerInternal(); /* -------------------------------------------------------------------- */ /* Get forward and inverse geotransform for the source image. */ /* -------------------------------------------------------------------- */ if( hSrcDS == NULL || (pszMethod != NULL && EQUAL(pszMethod, "NO_GEOTRANSFORM")) ) { psInfo->adfSrcGeoTransform[0] = 0.0; psInfo->adfSrcGeoTransform[1] = 1.0; psInfo->adfSrcGeoTransform[2] = 0.0; psInfo->adfSrcGeoTransform[3] = 0.0; psInfo->adfSrcGeoTransform[4] = 0.0; psInfo->adfSrcGeoTransform[5] = 1.0; memcpy( psInfo->adfSrcInvGeoTransform, psInfo->adfSrcGeoTransform, sizeof(double) * 6 ); } else if( (pszMethod == NULL || EQUAL(pszMethod, "GEOTRANSFORM")) && GDALGetGeoTransform( hSrcDS, psInfo->adfSrcGeoTransform ) == CE_None && (psInfo->adfSrcGeoTransform[0] != 0.0 || psInfo->adfSrcGeoTransform[1] != 1.0 || psInfo->adfSrcGeoTransform[2] != 0.0 || psInfo->adfSrcGeoTransform[3] != 0.0 || psInfo->adfSrcGeoTransform[4] != 0.0 || std::abs(psInfo->adfSrcGeoTransform[5]) != 1.0) ) { if( !GDALInvGeoTransform( psInfo->adfSrcGeoTransform, psInfo->adfSrcInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } if( pszSrcWKT == NULL ) pszSrcWKT = GDALGetProjectionRef( hSrcDS ); } else if( bGCPUseOK && (pszMethod == NULL || EQUAL(pszMethod, "GCP_POLYNOMIAL") ) && GDALGetGCPCount( hSrcDS ) > 0 && nOrder >= 0 ) { if( bRefine ) { psInfo->pSrcTransformArg = GDALCreateGCPRefineTransformer( GDALGetGCPCount( hSrcDS ), GDALGetGCPs( hSrcDS ), nOrder, FALSE, dfTolerance, nMinimumGcps ); } else { psInfo->pSrcTransformArg = GDALCreateGCPTransformer( GDALGetGCPCount( hSrcDS ), GDALGetGCPs( hSrcDS ), nOrder, FALSE ); } if( psInfo->pSrcTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pSrcTransformer = GDALGCPTransform; if( pszSrcWKT == NULL ) pszSrcWKT = GDALGetGCPProjection( hSrcDS ); } else if( bGCPUseOK && GDALGetGCPCount( hSrcDS ) > 0 && nOrder <= 0 && (pszMethod == NULL || EQUAL(pszMethod, "GCP_TPS")) ) { psInfo->pSrcTransformArg = GDALCreateTPSTransformerInt( GDALGetGCPCount( hSrcDS ), GDALGetGCPs( hSrcDS ), FALSE, papszOptions); if( psInfo->pSrcTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pSrcTransformer = GDALTPSTransform; if( pszSrcWKT == NULL ) pszSrcWKT = GDALGetGCPProjection( hSrcDS ); } else if( (pszMethod == NULL || EQUAL(pszMethod, "RPC")) && (papszMD = GDALGetMetadata( hSrcDS, "RPC" )) != NULL && GDALExtractRPCInfo( papszMD, &sRPCInfo ) ) { psInfo->pSrcTransformArg = GDALCreateRPCTransformer( &sRPCInfo, FALSE, 0, papszOptions ); if( psInfo->pSrcTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pSrcTransformer = GDALRPCTransform; if( pszSrcWKT == NULL ) pszSrcWKT = SRS_WKT_WGS84; } else if( (pszMethod == NULL || EQUAL(pszMethod, "GEOLOC_ARRAY")) && (papszMD = GDALGetMetadata( hSrcDS, "GEOLOCATION" )) != NULL ) { psInfo->pSrcTransformArg = GDALCreateGeoLocTransformer( hSrcDS, papszMD, FALSE ); if( psInfo->pSrcTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pSrcTransformer = GDALGeoLocTransform; if( pszSrcWKT == NULL ) pszSrcWKT = CSLFetchNameValue( papszMD, "SRS" ); } else if( pszMethod != NULL ) { CPLError(CE_Failure, CPLE_AppDefined, "Unable to compute a %s based transformation between " "pixel/line and georeferenced coordinates for %s.", pszMethod, GDALGetDescription(hSrcDS)); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } else { CPLError(CE_Failure, CPLE_AppDefined, "The transformation is already \"north up\" or " "a transformation between pixel/line and georeferenced " "coordinates cannot be computed for %s. " "There is no affine transformation and no GCPs. " "Specify transformation option SRC_METHOD=NO_GEOTRANSFORM to " "bypass this check.", GDALGetDescription(hSrcDS)); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } /* -------------------------------------------------------------------- */ /* Handle optional source approximation transformer. */ /* -------------------------------------------------------------------- */ if( psInfo->pSrcTransformer ) { const char* pszSrcApproxErrorFwd = CSLFetchNameValue( papszOptions, "SRC_APPROX_ERROR_IN_SRS_UNIT" ); const char* pszSrcApproxErrorReverse = CSLFetchNameValue( papszOptions, "SRC_APPROX_ERROR_IN_PIXEL" ); if( pszSrcApproxErrorFwd &&pszSrcApproxErrorReverse ) { void* pArg = GDALCreateApproxTransformer2( psInfo->pSrcTransformer, psInfo->pSrcTransformArg, CPLAtof(pszSrcApproxErrorFwd), CPLAtof(pszSrcApproxErrorReverse) ); if( pArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pSrcTransformArg = pArg; psInfo->pSrcTransformer = GDALApproxTransform; GDALApproxTransformerOwnsSubtransformer(psInfo->pSrcTransformArg, TRUE); } } /* -------------------------------------------------------------------- */ /* Get forward and inverse geotransform for destination image. */ /* If we have no destination use a unit transform. */ /* -------------------------------------------------------------------- */ const char *pszDstMethod = CSLFetchNameValue( papszOptions, "DST_METHOD" ); if( !hDstDS || (pszDstMethod != NULL && EQUAL(pszDstMethod, "NO_GEOTRANSFORM")) ) { psInfo->adfDstGeoTransform[0] = 0.0; psInfo->adfDstGeoTransform[1] = 1.0; psInfo->adfDstGeoTransform[2] = 0.0; psInfo->adfDstGeoTransform[3] = 0.0; psInfo->adfDstGeoTransform[4] = 0.0; psInfo->adfDstGeoTransform[5] = 1.0; memcpy( psInfo->adfDstInvGeoTransform, psInfo->adfDstGeoTransform, sizeof(double) * 6 ); } else if( (pszDstMethod == NULL || EQUAL(pszDstMethod, "GEOTRANSFORM")) && GDALGetGeoTransform( hDstDS, psInfo->adfDstGeoTransform ) == CE_None) { if( pszDstWKT == NULL ) pszDstWKT = GDALGetProjectionRef( hDstDS ); if( !GDALInvGeoTransform( psInfo->adfDstGeoTransform, psInfo->adfDstInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } } else if( bGCPUseOK && (pszDstMethod == NULL || EQUAL(pszDstMethod, "GCP_POLYNOMIAL") ) && GDALGetGCPCount( hDstDS ) > 0 && nOrder >= 0 ) { if( bRefine ) { psInfo->pDstTransformArg = GDALCreateGCPRefineTransformer( GDALGetGCPCount( hDstDS ), GDALGetGCPs( hDstDS ), nOrder, FALSE, dfTolerance, nMinimumGcps ); } else { psInfo->pDstTransformArg = GDALCreateGCPTransformer( GDALGetGCPCount( hDstDS ), GDALGetGCPs( hDstDS ), nOrder, FALSE ); } if( psInfo->pDstTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pDstTransformer = GDALGCPTransform; if( pszDstWKT == NULL ) pszDstWKT = GDALGetGCPProjection( hDstDS ); } else if( bGCPUseOK && GDALGetGCPCount( hDstDS ) > 0 && nOrder <= 0 && (pszDstMethod == NULL || EQUAL(pszDstMethod, "GCP_TPS")) ) { psInfo->pDstTransformArg = GDALCreateTPSTransformerInt( GDALGetGCPCount( hDstDS ), GDALGetGCPs( hDstDS ), FALSE, papszOptions ); if( psInfo->pDstTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pDstTransformer = GDALTPSTransform; if( pszDstWKT == NULL ) pszDstWKT = GDALGetGCPProjection( hDstDS ); } else if( (pszDstMethod == NULL || EQUAL(pszDstMethod, "RPC")) && (papszMD = GDALGetMetadata( hDstDS, "RPC" )) != NULL && GDALExtractRPCInfo( papszMD, &sRPCInfo ) ) { psInfo->pDstTransformArg = GDALCreateRPCTransformer( &sRPCInfo, FALSE, 0, papszOptions ); if( psInfo->pDstTransformArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pDstTransformer = GDALRPCTransform; if( pszDstWKT == NULL ) pszDstWKT = SRS_WKT_WGS84; } else { CPLError(CE_Failure, CPLE_AppDefined, "Unable to compute a transformation between pixel/line " "and georeferenced coordinates for %s. " "There is no affine transformation and no GCPs. " "Specify transformation option DST_METHOD=NO_GEOTRANSFORM " "to bypass this check.", GDALGetDescription(hDstDS)); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } /* -------------------------------------------------------------------- */ /* Handle optional source approximation transformer. */ /* -------------------------------------------------------------------- */ if( psInfo->pDstTransformer ) { const char* pszDstApproxErrorFwd = CSLFetchNameValue( papszOptions, "DST_APPROX_ERROR_IN_PIXEL" ); const char* pszDstApproxErrorReverse = CSLFetchNameValue( papszOptions, "DST_APPROX_ERROR_IN_SRS_UNIT" ); if( pszDstApproxErrorFwd &&pszDstApproxErrorReverse ) { void* pArg = GDALCreateApproxTransformer2( psInfo->pDstTransformer, psInfo->pDstTransformArg, CPLAtof(pszDstApproxErrorFwd), CPLAtof(pszDstApproxErrorReverse) ); if( pArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pDstTransformArg = pArg; psInfo->pDstTransformer = GDALApproxTransform; GDALApproxTransformerOwnsSubtransformer(psInfo->pDstTransformArg, TRUE); } } /* -------------------------------------------------------------------- */ /* Setup reprojection. */ /* -------------------------------------------------------------------- */ CPLString osSrcWKT = pszSrcWKT ? pszSrcWKT : ""; CPLString osDstWKT = pszDstWKT ? pszDstWKT : ""; if( !osSrcWKT.empty() && !osDstWKT.empty() && !EQUAL(osSrcWKT, osDstWKT) ) { if( CPLFetchBool( papszOptions, "STRIP_VERT_CS", false ) ) { OGRSpatialReference oSRS; oSRS.SetFromUserInput(osSrcWKT); if( oSRS.IsCompound() ) { OGR_SRSNode* poNode = oSRS.GetRoot()->GetChild(1); if( poNode != NULL ) { char* pszWKT = NULL; poNode->exportToWkt(&pszWKT); osSrcWKT = pszWKT; CPLFree(pszWKT); } } oSRS.SetFromUserInput(osDstWKT); if( oSRS.IsCompound() ) { OGR_SRSNode* poNode = oSRS.GetRoot()->GetChild(1); if( poNode != NULL ) { char* pszWKT = NULL; poNode->exportToWkt(&pszWKT); osDstWKT = pszWKT; CPLFree(pszWKT); } } } } if( !osSrcWKT.empty() && !osDstWKT.empty() && !EQUAL(osSrcWKT, osDstWKT) ) { if( hSrcDS && CPLFetchBool( papszOptions, "INSERT_CENTER_LONG", true ) ) osSrcWKT = InsertCenterLong( hSrcDS, osSrcWKT ); psInfo->pReprojectArg = GDALCreateReprojectionTransformer( osSrcWKT, osDstWKT ); if( psInfo->pReprojectArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pReproject = GDALReprojectionTransform; /* -------------------------------------------------------------------- */ /* Handle optional reprojection approximation transformer. */ /* -------------------------------------------------------------------- */ const char* psApproxErrorFwd = CSLFetchNameValue( papszOptions, "REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT" ); const char* psApproxErrorReverse = CSLFetchNameValue( papszOptions, "REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT" ); if( psApproxErrorFwd && psApproxErrorReverse ) { void* pArg = GDALCreateApproxTransformer2( psInfo->pReproject, psInfo->pReprojectArg, CPLAtof(psApproxErrorFwd), CPLAtof(psApproxErrorReverse) ); if( pArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pReprojectArg = pArg; psInfo->pReproject = GDALApproxTransform; GDALApproxTransformerOwnsSubtransformer(psInfo->pReprojectArg, TRUE); } } return psInfo; } /************************************************************************/ /* GDALRefreshGenImgProjTransformer() */ /************************************************************************/ void GDALRefreshGenImgProjTransformer( void* hTransformArg ) { GDALGenImgProjTransformInfo *psInfo = static_cast<GDALGenImgProjTransformInfo *>( hTransformArg ); if( psInfo->pReprojectArg ) { CPLXMLNode* psXML = GDALSerializeTransformer(psInfo->pReproject, psInfo->pReprojectArg); GDALDestroyTransformer(psInfo->pReprojectArg); GDALDeserializeTransformer(psXML, &psInfo->pReproject, &psInfo->pReprojectArg); CPLDestroyXMLNode(psXML); } } /************************************************************************/ /* GDALCreateGenImgProjTransformer3() */ /************************************************************************/ /** * Create image to image transformer. * * This function creates a transformation object that maps from pixel/line * coordinates on one image to pixel/line coordinates on another image. The * images may potentially be georeferenced in different coordinate systems, * and may used GCPs to map between their pixel/line coordinates and * georeferenced coordinates (as opposed to the default assumption that their * geotransform should be used). * * This transformer potentially performs three concatenated transformations. * * The first stage is from source image pixel/line coordinates to source * image georeferenced coordinates, and may be done using the geotransform, * or if not defined using a polynomial model derived from GCPs. If GCPs * are used this stage is accomplished using GDALGCPTransform(). * * The second stage is to change projections from the source coordinate system * to the destination coordinate system, assuming they differ. This is * accomplished internally using GDALReprojectionTransform(). * * The third stage is converting from destination image georeferenced * coordinates to destination image coordinates. This is done using the * destination image geotransform, or if not available, using a polynomial * model derived from GCPs. If GCPs are used this stage is accomplished using * GDALGCPTransform(). This stage is skipped if hDstDS is NULL when the * transformation is created. * * @param pszSrcWKT source WKT (or NULL). * @param padfSrcGeoTransform source geotransform (or NULL). * @param pszDstWKT destination WKT (or NULL). * @param padfDstGeoTransform destination geotransform (or NULL). * * @return handle suitable for use GDALGenImgProjTransform(), and to be * deallocated with GDALDestroyGenImgProjTransformer() or NULL on failure. */ void * GDALCreateGenImgProjTransformer3( const char *pszSrcWKT, const double *padfSrcGeoTransform, const char *pszDstWKT, const double *padfDstGeoTransform ) { /* -------------------------------------------------------------------- */ /* Initialize the transform info. */ /* -------------------------------------------------------------------- */ GDALGenImgProjTransformInfo *psInfo = GDALCreateGenImgProjTransformerInternal(); /* -------------------------------------------------------------------- */ /* Get forward and inverse geotransform for the source image. */ /* -------------------------------------------------------------------- */ if( padfSrcGeoTransform ) { memcpy( psInfo->adfSrcGeoTransform, padfSrcGeoTransform, sizeof(psInfo->adfSrcGeoTransform) ); if( !GDALInvGeoTransform( psInfo->adfSrcGeoTransform, psInfo->adfSrcInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } } else { psInfo->adfSrcGeoTransform[0] = 0.0; psInfo->adfSrcGeoTransform[1] = 1.0; psInfo->adfSrcGeoTransform[2] = 0.0; psInfo->adfSrcGeoTransform[3] = 0.0; psInfo->adfSrcGeoTransform[4] = 0.0; psInfo->adfSrcGeoTransform[5] = 1.0; memcpy( psInfo->adfSrcInvGeoTransform, psInfo->adfSrcGeoTransform, sizeof(double) * 6 ); } /* -------------------------------------------------------------------- */ /* Setup reprojection. */ /* -------------------------------------------------------------------- */ if( pszSrcWKT != NULL && strlen(pszSrcWKT) > 0 && pszDstWKT != NULL && strlen(pszDstWKT) > 0 && !EQUAL(pszSrcWKT, pszDstWKT) ) { psInfo->pReprojectArg = GDALCreateReprojectionTransformer( pszSrcWKT, pszDstWKT ); if( psInfo->pReprojectArg == NULL ) { GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } psInfo->pReproject = GDALReprojectionTransform; } /* -------------------------------------------------------------------- */ /* Get forward and inverse geotransform for destination image. */ /* If we have no destination matrix use a unit transform. */ /* -------------------------------------------------------------------- */ if( padfDstGeoTransform ) { memcpy( psInfo->adfDstGeoTransform, padfDstGeoTransform, sizeof(psInfo->adfDstGeoTransform) ); if( !GDALInvGeoTransform( psInfo->adfDstGeoTransform, psInfo->adfDstInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); GDALDestroyGenImgProjTransformer( psInfo ); return NULL; } } else { psInfo->adfDstGeoTransform[0] = 0.0; psInfo->adfDstGeoTransform[1] = 1.0; psInfo->adfDstGeoTransform[2] = 0.0; psInfo->adfDstGeoTransform[3] = 0.0; psInfo->adfDstGeoTransform[4] = 0.0; psInfo->adfDstGeoTransform[5] = 1.0; memcpy( psInfo->adfDstInvGeoTransform, psInfo->adfDstGeoTransform, sizeof(double) * 6 ); } return psInfo; } /************************************************************************/ /* GDALSetGenImgProjTransformerDstGeoTransform() */ /************************************************************************/ /** * Set GenImgProj output geotransform. * * Normally the "destination geotransform", or transformation between * georeferenced output coordinates and pixel/line coordinates on the * destination file is extracted from the destination file by * GDALCreateGenImgProjTransformer() and stored in the GenImgProj private * info. However, sometimes it is inconvenient to have an output file * handle with appropriate geotransform information when creating the * transformation. For these cases, this function can be used to apply * the destination geotransform. * * @param hTransformArg the handle to update. * @param padfGeoTransform the destination geotransform to apply (six doubles). */ void GDALSetGenImgProjTransformerDstGeoTransform( void *hTransformArg, const double *padfGeoTransform ) { VALIDATE_POINTER0( hTransformArg, "GDALSetGenImgProjTransformerDstGeoTransform" ); GDALGenImgProjTransformInfo *psInfo = static_cast<GDALGenImgProjTransformInfo *>( hTransformArg ); memcpy( psInfo->adfDstGeoTransform, padfGeoTransform, sizeof(double) * 6 ); if( !GDALInvGeoTransform( psInfo->adfDstGeoTransform, psInfo->adfDstInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); } } /************************************************************************/ /* GDALDestroyGenImgProjTransformer() */ /************************************************************************/ /** * GenImgProjTransformer deallocator. * * This function is used to deallocate the handle created with * GDALCreateGenImgProjTransformer(). * * @param hTransformArg the handle to deallocate. */ void GDALDestroyGenImgProjTransformer( void *hTransformArg ) { if( hTransformArg == NULL ) return; GDALGenImgProjTransformInfo *psInfo = static_cast<GDALGenImgProjTransformInfo *>(hTransformArg); if( psInfo->pSrcTransformArg != NULL ) GDALDestroyTransformer( psInfo->pSrcTransformArg ); if( psInfo->pDstTransformArg != NULL ) GDALDestroyTransformer( psInfo->pDstTransformArg ); if( psInfo->pReprojectArg != NULL ) GDALDestroyTransformer( psInfo->pReprojectArg ); CPLFree( psInfo ); } /************************************************************************/ /* GDALGenImgProjTransform() */ /************************************************************************/ /** * Perform general image reprojection transformation. * * Actually performs the transformation setup in * GDALCreateGenImgProjTransformer(). This function matches the signature * required by the GDALTransformerFunc(), and more details on the arguments * can be found in that topic. */ #ifdef DEBUG_APPROX_TRANSFORMER int countGDALGenImgProjTransform = 0; #endif int GDALGenImgProjTransform( void *pTransformArgIn, int bDstToSrc, int nPointCount, double *padfX, double *padfY, double *padfZ, int *panSuccess ) { GDALGenImgProjTransformInfo *psInfo = (GDALGenImgProjTransformInfo *) pTransformArgIn; double *padfGeoTransform; GDALTransformerFunc pTransformer; void *pTransformArg; #ifdef DEBUG_APPROX_TRANSFORMER CPLAssert(nPointCount > 0); countGDALGenImgProjTransform += nPointCount; #endif for( int i = 0; i < nPointCount; i++ ) { panSuccess[i] = ( padfX[i] != HUGE_VAL && padfY[i] != HUGE_VAL ); } /* -------------------------------------------------------------------- */ /* Convert from src (dst) pixel/line to src (dst) */ /* georeferenced coordinates. */ /* -------------------------------------------------------------------- */ if( bDstToSrc ) { padfGeoTransform = psInfo->adfDstGeoTransform; pTransformArg = psInfo->pDstTransformArg; pTransformer = psInfo->pDstTransformer; } else { padfGeoTransform = psInfo->adfSrcGeoTransform; pTransformArg = psInfo->pSrcTransformArg; pTransformer = psInfo->pSrcTransformer; } if( pTransformArg != NULL ) { if( !pTransformer( pTransformArg, FALSE, nPointCount, padfX, padfY, padfZ, panSuccess ) ) return FALSE; } else { for( int i = 0; i < nPointCount; i++ ) { if( padfX[i] == HUGE_VAL || padfY[i] == HUGE_VAL ) { panSuccess[i] = FALSE; continue; } const double dfNewX = padfGeoTransform[0] + padfX[i] * padfGeoTransform[1] + padfY[i] * padfGeoTransform[2]; const double dfNewY = padfGeoTransform[3] + padfX[i] * padfGeoTransform[4] + padfY[i] * padfGeoTransform[5]; padfX[i] = dfNewX; padfY[i] = dfNewY; } } /* -------------------------------------------------------------------- */ /* Reproject if needed. */ /* -------------------------------------------------------------------- */ if( psInfo->pReprojectArg ) { if( !psInfo->pReproject( psInfo->pReprojectArg, bDstToSrc, nPointCount, padfX, padfY, padfZ, panSuccess ) ) return FALSE; } /* -------------------------------------------------------------------- */ /* Convert dst (src) georef coordinates back to pixel/line. */ /* -------------------------------------------------------------------- */ if( bDstToSrc ) { padfGeoTransform = psInfo->adfSrcInvGeoTransform; pTransformArg = psInfo->pSrcTransformArg; pTransformer = psInfo->pSrcTransformer; } else { padfGeoTransform = psInfo->adfDstInvGeoTransform; pTransformArg = psInfo->pDstTransformArg; pTransformer = psInfo->pDstTransformer; } if( pTransformArg != NULL ) { if( !pTransformer( pTransformArg, TRUE, nPointCount, padfX, padfY, padfZ, panSuccess ) ) return FALSE; } else { for( int i = 0; i < nPointCount; i++ ) { if( !panSuccess[i] ) continue; const double dfNewX = padfGeoTransform[0] + padfX[i] * padfGeoTransform[1] + padfY[i] * padfGeoTransform[2]; const double dfNewY = padfGeoTransform[3] + padfX[i] * padfGeoTransform[4] + padfY[i] * padfGeoTransform[5]; padfX[i] = dfNewX; padfY[i] = dfNewY; } } return TRUE; } /************************************************************************/ /* GDALSerializeGenImgProjTransformer() */ /************************************************************************/ static CPLXMLNode * GDALSerializeGenImgProjTransformer( void *pTransformArg ) { GDALGenImgProjTransformInfo *psInfo = static_cast<GDALGenImgProjTransformInfo *>(pTransformArg); CPLXMLNode *psTree = CPLCreateXMLNode( NULL, CXT_Element, "GenImgProjTransformer" ); char szWork[200] = {}; /* -------------------------------------------------------------------- */ /* Handle source transformation. */ /* -------------------------------------------------------------------- */ if( psInfo->pSrcTransformArg != NULL ) { CPLXMLNode *psTransformer = GDALSerializeTransformer( psInfo->pSrcTransformer, psInfo->pSrcTransformArg); if( psTransformer != NULL ) { CPLXMLNode *psTransformerContainer = CPLCreateXMLNode( psTree, CXT_Element, CPLSPrintf("Src%s", psTransformer->pszValue) ); CPLAddXMLChild( psTransformerContainer, psTransformer ); } } /* -------------------------------------------------------------------- */ /* Handle source geotransforms. */ /* -------------------------------------------------------------------- */ else { CPLsnprintf( szWork, sizeof(szWork), "%.18g,%.18g,%.18g,%.18g,%.18g,%.18g", psInfo->adfSrcGeoTransform[0], psInfo->adfSrcGeoTransform[1], psInfo->adfSrcGeoTransform[2], psInfo->adfSrcGeoTransform[3], psInfo->adfSrcGeoTransform[4], psInfo->adfSrcGeoTransform[5] ); CPLCreateXMLElementAndValue( psTree, "SrcGeoTransform", szWork ); CPLsnprintf( szWork, sizeof(szWork), "%.18g,%.18g,%.18g,%.18g,%.18g,%.18g", psInfo->adfSrcInvGeoTransform[0], psInfo->adfSrcInvGeoTransform[1], psInfo->adfSrcInvGeoTransform[2], psInfo->adfSrcInvGeoTransform[3], psInfo->adfSrcInvGeoTransform[4], psInfo->adfSrcInvGeoTransform[5] ); CPLCreateXMLElementAndValue( psTree, "SrcInvGeoTransform", szWork ); } /* -------------------------------------------------------------------- */ /* Handle dest transformation. */ /* -------------------------------------------------------------------- */ if( psInfo->pDstTransformArg != NULL ) { CPLXMLNode *psTransformer = GDALSerializeTransformer( psInfo->pDstTransformer, psInfo->pDstTransformArg); if( psTransformer != NULL ) { CPLXMLNode *psTransformerContainer = CPLCreateXMLNode( psTree, CXT_Element, CPLSPrintf("Dst%s", psTransformer->pszValue) ); CPLAddXMLChild( psTransformerContainer, psTransformer ); } } /* -------------------------------------------------------------------- */ /* Handle destination geotransforms. */ /* -------------------------------------------------------------------- */ else { CPLsnprintf( szWork, sizeof(szWork), "%.18g,%.18g,%.18g,%.18g,%.18g,%.18g", psInfo->adfDstGeoTransform[0], psInfo->adfDstGeoTransform[1], psInfo->adfDstGeoTransform[2], psInfo->adfDstGeoTransform[3], psInfo->adfDstGeoTransform[4], psInfo->adfDstGeoTransform[5] ); CPLCreateXMLElementAndValue( psTree, "DstGeoTransform", szWork ); CPLsnprintf( szWork, sizeof(szWork), "%.18g,%.18g,%.18g,%.18g,%.18g,%.18g", psInfo->adfDstInvGeoTransform[0], psInfo->adfDstInvGeoTransform[1], psInfo->adfDstInvGeoTransform[2], psInfo->adfDstInvGeoTransform[3], psInfo->adfDstInvGeoTransform[4], psInfo->adfDstInvGeoTransform[5] ); CPLCreateXMLElementAndValue( psTree, "DstInvGeoTransform", szWork ); } /* -------------------------------------------------------------------- */ /* Do we have a reprojection transformer? */ /* -------------------------------------------------------------------- */ if( psInfo->pReprojectArg != NULL ) { CPLXMLNode *psTransformerContainer = CPLCreateXMLNode( psTree, CXT_Element, "ReprojectTransformer" ); CPLXMLNode *psTransformer = GDALSerializeTransformer( psInfo->pReproject, psInfo->pReprojectArg ); if( psTransformer != NULL ) CPLAddXMLChild( psTransformerContainer, psTransformer ); } return psTree; } /************************************************************************/ /* GDALDeserializeGeoTransform() */ /************************************************************************/ static void GDALDeserializeGeoTransform(const char* pszGT, double adfGeoTransform[6]) { CPLsscanf( pszGT, "%lf,%lf,%lf,%lf,%lf,%lf", adfGeoTransform + 0, adfGeoTransform + 1, adfGeoTransform + 2, adfGeoTransform + 3, adfGeoTransform + 4, adfGeoTransform + 5 ); } /************************************************************************/ /* GDALDeserializeGenImgProjTransformer() */ /************************************************************************/ void *GDALDeserializeGenImgProjTransformer( CPLXMLNode *psTree ) { /* -------------------------------------------------------------------- */ /* Initialize the transform info. */ /* -------------------------------------------------------------------- */ GDALGenImgProjTransformInfo *psInfo = GDALCreateGenImgProjTransformerInternal(); /* -------------------------------------------------------------------- */ /* SrcGeotransform */ /* -------------------------------------------------------------------- */ if( CPLGetXMLNode( psTree, "SrcGeoTransform" ) != NULL ) { GDALDeserializeGeoTransform( CPLGetXMLValue( psTree, "SrcGeoTransform", "" ), psInfo->adfSrcGeoTransform ); if( CPLGetXMLNode( psTree, "SrcInvGeoTransform" ) != NULL ) { GDALDeserializeGeoTransform( CPLGetXMLValue( psTree, "SrcInvGeoTransform", "" ), psInfo->adfSrcInvGeoTransform ); } else { if( !GDALInvGeoTransform( psInfo->adfSrcGeoTransform, psInfo->adfSrcInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); } } } /* -------------------------------------------------------------------- */ /* Src Transform */ /* -------------------------------------------------------------------- */ else { for( CPLXMLNode* psIter = psTree->psChild; psIter != NULL; psIter = psIter->psNext ) { if( psIter->eType == CXT_Element && STARTS_WITH_CI(psIter->pszValue, "Src") ) { GDALDeserializeTransformer( psIter->psChild, &psInfo->pSrcTransformer, &psInfo->pSrcTransformArg ); } } } /* -------------------------------------------------------------------- */ /* DstGeotransform */ /* -------------------------------------------------------------------- */ if( CPLGetXMLNode( psTree, "DstGeoTransform" ) != NULL ) { GDALDeserializeGeoTransform( CPLGetXMLValue( psTree, "DstGeoTransform", "" ), psInfo->adfDstGeoTransform); if( CPLGetXMLNode( psTree, "DstInvGeoTransform" ) != NULL ) { GDALDeserializeGeoTransform( CPLGetXMLValue( psTree, "DstInvGeoTransform", "" ), psInfo->adfDstInvGeoTransform); } else { if( !GDALInvGeoTransform( psInfo->adfDstGeoTransform, psInfo->adfDstInvGeoTransform ) ) { CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform"); } } } /* -------------------------------------------------------------------- */ /* Dst Transform */ /* -------------------------------------------------------------------- */ else { for( CPLXMLNode* psIter = psTree->psChild; psIter != NULL; psIter = psIter->psNext ) { if( psIter->eType == CXT_Element && STARTS_WITH_CI(psIter->pszValue, "Dst") ) { GDALDeserializeTransformer( psIter->psChild, &psInfo->pDstTransformer, &psInfo->pDstTransformArg ); break; } } } /* -------------------------------------------------------------------- */ /* Reproject transformer */ /* -------------------------------------------------------------------- */ CPLXMLNode* psSubtree = CPLGetXMLNode( psTree, "ReprojectTransformer" ); if( psSubtree != NULL && psSubtree->psChild != NULL ) { GDALDeserializeTransformer( psSubtree->psChild, &psInfo->pReproject, &psInfo->pReprojectArg ); } return psInfo; } /************************************************************************/ /* ==================================================================== */ /* GDALReprojectionTransformer */ /* ==================================================================== */ /************************************************************************/ typedef struct { GDALTransformerInfo sTI; OGRCoordinateTransformation *poForwardTransform; OGRCoordinateTransformation *poReverseTransform; } GDALReprojectionTransformInfo; /************************************************************************/ /* GDALCreateReprojectionTransformer() */ /************************************************************************/ /** * Create reprojection transformer. * * Creates a callback data structure suitable for use with * GDALReprojectionTransformation() to represent a transformation from * one geographic or projected coordinate system to another. On input * the coordinate systems are described in OpenGIS WKT format. * * Internally the OGRCoordinateTransformation object is used to implement * the reprojection. * * @param pszSrcWKT the coordinate system for the source coordinate system. * @param pszDstWKT the coordinate system for the destination coordinate * system. * * @return Handle for use with GDALReprojectionTransform(), or NULL if the * system fails to initialize the reprojection. **/ void *GDALCreateReprojectionTransformer( const char *pszSrcWKT, const char *pszDstWKT ) { /* -------------------------------------------------------------------- */ /* Ingest the SRS definitions. */ /* -------------------------------------------------------------------- */ OGRSpatialReference oSrcSRS; if( oSrcSRS.importFromWkt( (char **) &pszSrcWKT ) != OGRERR_NONE ) { CPLError( CE_Failure, CPLE_AppDefined, "Failed to import coordinate system `%s'.", pszSrcWKT ); return NULL; } OGRSpatialReference oDstSRS; if( oDstSRS.importFromWkt( (char **) &pszDstWKT ) != OGRERR_NONE ) { CPLError( CE_Failure, CPLE_AppDefined, "Failed to import coordinate system `%s'.", pszSrcWKT ); return NULL; } /* -------------------------------------------------------------------- */ /* Build the forward coordinate transformation. */ /* -------------------------------------------------------------------- */ OGRCoordinateTransformation *poForwardTransform = OGRCreateCoordinateTransformation(&oSrcSRS, &oDstSRS); if( poForwardTransform == NULL ) // OGRCreateCoordinateTransformation() will report errors on its own. return NULL; /* -------------------------------------------------------------------- */ /* Create a structure to hold the transform info, and also */ /* build reverse transform. We assume that if the forward */ /* transform can be created, then so can the reverse one. */ /* -------------------------------------------------------------------- */ GDALReprojectionTransformInfo *psInfo = static_cast<GDALReprojectionTransformInfo *>( CPLCalloc(sizeof(GDALReprojectionTransformInfo), 1)); psInfo->poForwardTransform = poForwardTransform; psInfo->poReverseTransform = OGRCreateCoordinateTransformation(&oDstSRS, &oSrcSRS); memcpy( psInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE) ); psInfo->sTI.pszClassName = "GDALReprojectionTransformer"; psInfo->sTI.pfnTransform = GDALReprojectionTransform; psInfo->sTI.pfnCleanup = GDALDestroyReprojectionTransformer; psInfo->sTI.pfnSerialize = GDALSerializeReprojectionTransformer; return psInfo; } /************************************************************************/ /* GDALDestroyReprojectionTransformer() */ /************************************************************************/ /** * Destroy reprojection transformation. * * @param pTransformArg the transformation handle returned by * GDALCreateReprojectionTransformer(). */ void GDALDestroyReprojectionTransformer( void *pTransformArg ) { if( pTransformArg == NULL ) return; GDALReprojectionTransformInfo *psInfo = (GDALReprojectionTransformInfo *) pTransformArg; if( psInfo->poForwardTransform ) delete psInfo->poForwardTransform; if( psInfo->poReverseTransform ) delete psInfo->poReverseTransform; CPLFree( psInfo ); } /************************************************************************/ /* GDALReprojectionTransform() */ /************************************************************************/ /** * Perform reprojection transformation. * * Actually performs the reprojection transformation described in * GDALCreateReprojectionTransformer(). This function matches the * GDALTransformerFunc() signature. Details of the arguments are described * there. */ int GDALReprojectionTransform( void *pTransformArg, int bDstToSrc, int nPointCount, double *padfX, double *padfY, double *padfZ, int *panSuccess ) { GDALReprojectionTransformInfo *psInfo = (GDALReprojectionTransformInfo *) pTransformArg; int bSuccess; if( bDstToSrc ) bSuccess = psInfo->poReverseTransform->TransformEx( nPointCount, padfX, padfY, padfZ, panSuccess ); else bSuccess = psInfo->poForwardTransform->TransformEx( nPointCount, padfX, padfY, padfZ, panSuccess ); return bSuccess; } /************************************************************************/ /* GDALSerializeReprojectionTransformer() */ /************************************************************************/ static CPLXMLNode * GDALSerializeReprojectionTransformer( void *pTransformArg ) { CPLXMLNode *psTree; GDALReprojectionTransformInfo *psInfo = (GDALReprojectionTransformInfo *) pTransformArg; psTree = CPLCreateXMLNode( NULL, CXT_Element, "ReprojectionTransformer" ); /* -------------------------------------------------------------------- */ /* Handle SourceCS. */ /* -------------------------------------------------------------------- */ OGRSpatialReference *poSRS; char *pszWKT = NULL; poSRS = psInfo->poForwardTransform->GetSourceCS(); poSRS->exportToWkt( &pszWKT ); CPLCreateXMLElementAndValue( psTree, "SourceSRS", pszWKT ); CPLFree( pszWKT ); /* -------------------------------------------------------------------- */ /* Handle DestinationCS. */ /* -------------------------------------------------------------------- */ poSRS = psInfo->poForwardTransform->GetTargetCS(); poSRS->exportToWkt( &pszWKT ); CPLCreateXMLElementAndValue( psTree, "TargetSRS", pszWKT ); CPLFree( pszWKT ); return psTree; } /************************************************************************/ /* GDALDeserializeReprojectionTransformer() */ /************************************************************************/ static void * GDALDeserializeReprojectionTransformer( CPLXMLNode *psTree ) { const char *pszSourceSRS = CPLGetXMLValue( psTree, "SourceSRS", NULL ); const char *pszTargetSRS = CPLGetXMLValue( psTree, "TargetSRS", NULL ); char *pszSourceWKT = NULL, *pszTargetWKT = NULL; void *pResult = NULL; if( pszSourceSRS != NULL ) { OGRSpatialReference oSRS; if( oSRS.SetFromUserInput( pszSourceSRS ) == OGRERR_NONE ) oSRS.exportToWkt( &pszSourceWKT ); } if( pszTargetSRS != NULL ) { OGRSpatialReference oSRS; if( oSRS.SetFromUserInput( pszTargetSRS ) == OGRERR_NONE ) oSRS.exportToWkt( &pszTargetWKT ); } if( pszSourceWKT != NULL && pszTargetWKT != NULL ) { pResult = GDALCreateReprojectionTransformer( pszSourceWKT, pszTargetWKT ); } else { CPLError(CE_Failure, CPLE_AppDefined, "ReprojectionTransformer definition missing either " "SourceSRS or TargetSRS definition."); } CPLFree( pszSourceWKT ); CPLFree( pszTargetWKT ); return pResult; } /************************************************************************/ /* ==================================================================== */ /* Approximate transformer. */ /* ==================================================================== */ /************************************************************************/ typedef struct { GDALTransformerInfo sTI; GDALTransformerFunc pfnBaseTransformer; void *pBaseCBData; double dfMaxErrorForward; double dfMaxErrorReverse; int bOwnSubtransformer; } ApproxTransformInfo; /************************************************************************/ /* GDALCreateSimilarApproxTransformer() */ /************************************************************************/ static void * GDALCreateSimilarApproxTransformer( void *hTransformArg, double dfSrcRatioX, double dfSrcRatioY ) { VALIDATE_POINTER1( hTransformArg, "GDALCreateSimilarApproxTransformer", NULL ); ApproxTransformInfo *psInfo = static_cast<ApproxTransformInfo *>(hTransformArg); ApproxTransformInfo *psClonedInfo = static_cast<ApproxTransformInfo *>( CPLMalloc(sizeof(ApproxTransformInfo))); memcpy(psClonedInfo, psInfo, sizeof(ApproxTransformInfo)); if( psClonedInfo->pBaseCBData ) { psClonedInfo->pBaseCBData = GDALCreateSimilarTransformer( psInfo->pBaseCBData, dfSrcRatioX, dfSrcRatioY ); if( psClonedInfo->pBaseCBData == NULL ) { CPLFree(psClonedInfo); return NULL; } } psClonedInfo->bOwnSubtransformer = TRUE; return psClonedInfo; } /************************************************************************/ /* GDALSerializeApproxTransformer() */ /************************************************************************/ static CPLXMLNode * GDALSerializeApproxTransformer( void *pTransformArg ) { CPLXMLNode *psTree; ApproxTransformInfo *psInfo = (ApproxTransformInfo *) pTransformArg; psTree = CPLCreateXMLNode( NULL, CXT_Element, "ApproxTransformer" ); /* -------------------------------------------------------------------- */ /* Attach max error. */ /* -------------------------------------------------------------------- */ if( psInfo->dfMaxErrorForward == psInfo->dfMaxErrorReverse ) { CPLCreateXMLElementAndValue( psTree, "MaxError", CPLString().Printf("%g", psInfo->dfMaxErrorForward) ); } else { CPLCreateXMLElementAndValue( psTree, "MaxErrorForward", CPLString().Printf("%g", psInfo->dfMaxErrorForward) ); CPLCreateXMLElementAndValue( psTree, "MaxErrorReverse", CPLString().Printf("%g", psInfo->dfMaxErrorReverse) ); } /* -------------------------------------------------------------------- */ /* Capture underlying transformer. */ /* -------------------------------------------------------------------- */ CPLXMLNode *psTransformerContainer = CPLCreateXMLNode( psTree, CXT_Element, "BaseTransformer" ); CPLXMLNode *psTransformer = GDALSerializeTransformer( psInfo->pfnBaseTransformer, psInfo->pBaseCBData ); if( psTransformer != NULL ) CPLAddXMLChild( psTransformerContainer, psTransformer ); return psTree; } /************************************************************************/ /* GDALCreateApproxTransformer() */ /************************************************************************/ /** * Create an approximating transformer. * * This function creates a context for an approximated transformer. Basically * a high precision transformer is supplied as input and internally linear * approximations are computed to generate results to within a defined * precision. * * The approximation is actually done at the point where GDALApproxTransform() * calls are made, and depend on the assumption that the roughly linear. The * first and last point passed in must be the extreme values and the * intermediate values should describe a curve between the end points. The * approximator transforms and center using the approximate transformer, and * then compares the true middle transformed value to a linear approximation * based on the end points. If the error is within the supplied threshold * then the end points are used to linearly approximate all the values * otherwise the inputs points are split into two smaller sets, and the * function recursively called till a sufficiently small set of points if found * that the linear approximation is OK, or that all the points are exactly * computed. * * This function is very suitable for approximating transformation results * from output pixel/line space to input coordinates for warpers that operate * on one input scanline at a time. Care should be taken using it in other * circumstances as little internal validation is done, in order to keep things * fast. * * @param pfnBaseTransformer the high precision transformer which should be * approximated. * @param pBaseTransformArg the callback argument for the high precision * transformer. * @param dfMaxError the maximum cartesian error in the "output" space that * is to be accepted in the linear approximation. * * @return callback pointer suitable for use with GDALApproxTransform(). It * should be deallocated with GDALDestroyApproxTransformer(). */ void *GDALCreateApproxTransformer( GDALTransformerFunc pfnBaseTransformer, void *pBaseTransformArg, double dfMaxError) { return GDALCreateApproxTransformer2(pfnBaseTransformer, pBaseTransformArg, dfMaxError, dfMaxError); } static void *GDALCreateApproxTransformer2( GDALTransformerFunc pfnBaseTransformer, void *pBaseTransformArg, double dfMaxErrorForward, double dfMaxErrorReverse) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>( CPLMalloc(sizeof(ApproxTransformInfo))); psATInfo->pfnBaseTransformer = pfnBaseTransformer; psATInfo->pBaseCBData = pBaseTransformArg; psATInfo->dfMaxErrorForward = dfMaxErrorForward; psATInfo->dfMaxErrorReverse = dfMaxErrorReverse; psATInfo->bOwnSubtransformer = FALSE; memcpy(psATInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)); psATInfo->sTI.pszClassName = "GDALApproxTransformer"; psATInfo->sTI.pfnTransform = GDALApproxTransform; psATInfo->sTI.pfnCleanup = GDALDestroyApproxTransformer; psATInfo->sTI.pfnSerialize = GDALSerializeApproxTransformer; psATInfo->sTI.pfnCreateSimilar = GDALCreateSimilarApproxTransformer; return psATInfo; } /************************************************************************/ /* GDALApproxTransformerOwnsSubtransformer() */ /************************************************************************/ /** Set bOwnSubtransformer flag */ void GDALApproxTransformerOwnsSubtransformer( void *pCBData, int bOwnFlag ) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pCBData); psATInfo->bOwnSubtransformer = bOwnFlag; } /************************************************************************/ /* GDALDestroyApproxTransformer() */ /************************************************************************/ /** * Cleanup approximate transformer. * * Deallocates the resources allocated by GDALCreateApproxTransformer(). * * @param pCBData callback data originally returned by * GDALCreateApproxTransformer(). */ void GDALDestroyApproxTransformer( void * pCBData ) { if( pCBData == NULL) return; ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pCBData); if( psATInfo->bOwnSubtransformer ) GDALDestroyTransformer( psATInfo->pBaseCBData ); CPLFree( pCBData ); } /************************************************************************/ /* GDALApproxTransformInternal() */ /************************************************************************/ static int GDALApproxTransformInternal( void *pCBData, int bDstToSrc, int nPoints, double *x, double *y, double *z, int *panSuccess, // SME = Start, Middle, End. const double xSMETransformed[3], const double ySMETransformed[3], const double zSMETransformed[3] ) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pCBData); int bSuccess = FALSE; // TODO(schwehr): Split into each case. const int nMiddle = (nPoints - 1) / 2; #ifdef notdef_sanify_check { double x2[3] = { x[0], x[nMiddle], x[nPoints-1] }; double y2[3] = { y[0], y[nMiddle], y[nPoints-1] }; double z2[3] = { z[0], z[nMiddle], z[nPoints-1] }; int anSuccess2[3] = {}; bSuccess = psATInfo->pfnBaseTransformer( psATInfo->pBaseCBData, bDstToSrc, 3, x2, y2, z2, anSuccess2 ); CPLAssert(bSuccess); CPLAssert(anSuccess2[0]); CPLAssert(anSuccess2[1]); CPLAssert(anSuccess2[2]); CPLAssert(x2[0] == xSMETransformed[0]); CPLAssert(y2[0] == ySMETransformed[0]); CPLAssert(z2[0] == zSMETransformed[0]); CPLAssert(x2[1] == xSMETransformed[1]); CPLAssert(y2[1] == ySMETransformed[1]); CPLAssert(z2[1] == zSMETransformed[1]); CPLAssert(x2[2] == xSMETransformed[2]); CPLAssert(y2[2] == ySMETransformed[2]); CPLAssert(z2[2] == zSMETransformed[2]); } #endif #ifdef DEBUG_APPROX_TRANSFORMER fprintf(stderr, "start (%.3f,%.3f) -> (%.3f,%.3f)\n",/*ok*/ x[0], y[0], xSMETransformed[0], ySMETransformed[0]); fprintf(stderr, "middle (%.3f,%.3f) -> (%.3f,%.3f)\n",/*ok*/ x[nMiddle], y[nMiddle], xSMETransformed[1], ySMETransformed[1]); fprintf(stderr, "end (%.3f,%.3f) -> (%.3f,%.3f)\n",/*ok*/ x[nPoints-1], y[nPoints-1], xSMETransformed[2], ySMETransformed[2]); #endif /* -------------------------------------------------------------------- */ /* Is the error at the middle acceptable relative to an */ /* interpolation of the middle position? */ /* -------------------------------------------------------------------- */ const double dfDeltaX = (xSMETransformed[2] - xSMETransformed[0]) / (x[nPoints-1] - x[0]); const double dfDeltaY = (ySMETransformed[2] - ySMETransformed[0]) / (x[nPoints-1] - x[0]); const double dfDeltaZ = (zSMETransformed[2] - zSMETransformed[0]) / (x[nPoints-1] - x[0]); const double dfError = fabs((xSMETransformed[0] + dfDeltaX * (x[nMiddle] - x[0])) - xSMETransformed[1]) + fabs((ySMETransformed[0] + dfDeltaY * (x[nMiddle] - x[0])) - ySMETransformed[1]); const double dfMaxError = (bDstToSrc) ? psATInfo->dfMaxErrorReverse : psATInfo->dfMaxErrorForward; if( dfError > dfMaxError ) { #if DEBUG_VERBOSE CPLDebug( "GDAL", "ApproxTransformer - " "error %g over threshold %g, subdivide %d points.", dfError, dfMaxError, nPoints ); #endif double xMiddle[3] = { x[(nMiddle - 1) / 2], x[nMiddle - 1], x[nMiddle + (nPoints - nMiddle - 1) / 2] }; double yMiddle[3] = { y[(nMiddle - 1) / 2], y[nMiddle - 1], y[nMiddle + (nPoints - nMiddle - 1) / 2] }; double zMiddle[3] = { z[(nMiddle - 1) / 2], z[nMiddle - 1 ], z[nMiddle + (nPoints - nMiddle - 1) / 2] }; const bool bUseBaseTransformForHalf1 = nMiddle <= 5 || y[0] != y[nMiddle-1] || y[0] != y[(nMiddle - 1) / 2] || x[0] == x[nMiddle-1] || x[0] == x[(nMiddle - 1) / 2]; const bool bUseBaseTransformForHalf2 = nPoints - nMiddle <= 5 || y[nMiddle] != y[nPoints-1] || y[nMiddle] != y[nMiddle + (nPoints - nMiddle - 1) / 2] || x[nMiddle] == x[nPoints-1] || x[nMiddle] == x[nMiddle + (nPoints - nMiddle - 1) / 2]; int anSuccess2[3] = {}; if( !bUseBaseTransformForHalf1 && !bUseBaseTransformForHalf2 ) bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 3, xMiddle, yMiddle, zMiddle, anSuccess2 ); else if( !bUseBaseTransformForHalf1 ) { bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 2, xMiddle, yMiddle, zMiddle, anSuccess2 ); anSuccess2[2] = TRUE; } else if( !bUseBaseTransformForHalf2 ) { bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 1, xMiddle + 2, yMiddle + 2, zMiddle + 2, anSuccess2 + 2 ); anSuccess2[0] = TRUE; anSuccess2[1] = TRUE; } else { bSuccess = FALSE; } if( !bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2] ) { bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1, z + 1, panSuccess + 1); bSuccess &= psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2, x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1, panSuccess + nMiddle + 1); x[0] = xSMETransformed[0]; y[0] = ySMETransformed[0]; z[0] = zSMETransformed[0]; panSuccess[0] = TRUE; x[nMiddle] = xSMETransformed[1]; y[nMiddle] = ySMETransformed[1]; z[nMiddle] = zSMETransformed[1]; panSuccess[nMiddle] = TRUE; x[nPoints-1] = xSMETransformed[2]; y[nPoints-1] = ySMETransformed[2]; z[nPoints-1] = zSMETransformed[2]; panSuccess[nPoints-1] = TRUE; return bSuccess; } double x2[3] = {}; double y2[3] = {}; double z2[3] = {}; if( !bUseBaseTransformForHalf1 ) { x2[0] = xSMETransformed[0]; y2[0] = ySMETransformed[0]; z2[0] = zSMETransformed[0]; x2[1] = xMiddle[0]; y2[1] = yMiddle[0]; z2[1] = zMiddle[0]; x2[2] = xMiddle[1]; y2[2] = yMiddle[1]; z2[2] = zMiddle[1]; bSuccess = GDALApproxTransformInternal( psATInfo, bDstToSrc, nMiddle, x, y, z, panSuccess, x2, y2, z2); } else { bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1, z + 1, panSuccess + 1 ); x[0] = xSMETransformed[0]; y[0] = ySMETransformed[0]; z[0] = zSMETransformed[0]; panSuccess[0] = TRUE; } if( !bSuccess ) return FALSE; if( !bUseBaseTransformForHalf2 ) { x2[0] = xSMETransformed[1]; y2[0] = ySMETransformed[1]; z2[0] = zSMETransformed[1]; x2[1] = xMiddle[2]; y2[1] = yMiddle[2]; z2[1] = zMiddle[2]; x2[2] = xSMETransformed[2]; y2[2] = ySMETransformed[2]; z2[2] = zSMETransformed[2]; bSuccess = GDALApproxTransformInternal(psATInfo, bDstToSrc, nPoints - nMiddle, x+nMiddle, y+nMiddle, z+nMiddle, panSuccess+nMiddle, x2, y2, z2); } else { bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2, x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1, panSuccess+nMiddle+1 ); x[nMiddle] = xSMETransformed[1]; y[nMiddle] = ySMETransformed[1]; z[nMiddle] = zSMETransformed[1]; panSuccess[nMiddle] = TRUE; x[nPoints-1] = xSMETransformed[2]; y[nPoints-1] = ySMETransformed[2]; z[nPoints-1] = zSMETransformed[2]; panSuccess[nPoints-1] = TRUE; } if( !bSuccess ) return FALSE; return TRUE; } /* -------------------------------------------------------------------- */ /* Error is OK since this is just used to compute output bounds */ /* of newly created file for gdalwarper. So just use affine */ /* approximation of the reverse transform. Eventually we */ /* should implement iterative searching to find a result within */ /* our error threshold. */ /* NOTE: the above comment is not true: gdalwarp uses approximator */ /* also to compute the source pixel of each target pixel. */ /* -------------------------------------------------------------------- */ for( int i = nPoints-1; i >= 0; i-- ) { #ifdef check_error double xtemp = x[i]; double ytemp = y[i]; double ztemp = z[i]; double x_ori = xtemp; double y_ori = ytemp; int btemp = FALSE; psATInfo->pfnBaseTransformer( psATInfo->pBaseCBData, bDstToSrc, 1, &xtemp, &ytemp, &ztemp, &btemp); #endif const double dfDist = (x[i] - x[0]); x[i] = xSMETransformed[0] + dfDeltaX * dfDist; y[i] = ySMETransformed[0] + dfDeltaY * dfDist; z[i] = zSMETransformed[0] + dfDeltaZ * dfDist; #ifdef check_error const double dfError2 = fabs(x[i] - xtemp) + fabs(y[i] - ytemp); if( dfError2 > 4 /*10 * dfMaxError*/ ) { /*ok*/printf("Error = %f on (%f, %f)\n", dfError2, x_ori, y_ori); } #endif panSuccess[i] = TRUE; } return TRUE; } /************************************************************************/ /* GDALApproxTransform() */ /************************************************************************/ /** * Perform approximate transformation. * * Actually performs the approximate transformation described in * GDALCreateApproxTransformer(). This function matches the * GDALTransformerFunc() signature. Details of the arguments are described * there. */ int GDALApproxTransform( void *pCBData, int bDstToSrc, int nPoints, double *x, double *y, double *z, int *panSuccess ) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pCBData); double x2[3] = {}; double y2[3] = {}; double z2[3] = {}; int anSuccess2[3] = {}; int bSuccess; const int nMiddle = (nPoints - 1) / 2; /* -------------------------------------------------------------------- */ /* Bail if our preconditions are not met, or if error is not */ /* acceptable. */ /* -------------------------------------------------------------------- */ int bRet = FALSE; if( y[0] != y[nPoints-1] || y[0] != y[nMiddle] || x[0] == x[nPoints-1] || x[0] == x[nMiddle] || (psATInfo->dfMaxErrorForward == 0.0 && psATInfo->dfMaxErrorReverse == 0.0) || nPoints <= 5 ) { bRet = psATInfo->pfnBaseTransformer( psATInfo->pBaseCBData, bDstToSrc, nPoints, x, y, z, panSuccess ); goto end; } /* -------------------------------------------------------------------- */ /* Transform first, last and middle point. */ /* -------------------------------------------------------------------- */ x2[0] = x[0]; y2[0] = y[0]; z2[0] = z[0]; x2[1] = x[nMiddle]; y2[1] = y[nMiddle]; z2[1] = z[nMiddle]; x2[2] = x[nPoints-1]; y2[2] = y[nPoints-1]; z2[2] = z[nPoints-1]; bSuccess = psATInfo->pfnBaseTransformer( psATInfo->pBaseCBData, bDstToSrc, 3, x2, y2, z2, anSuccess2 ); if( !bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2] ) { bRet = psATInfo->pfnBaseTransformer( psATInfo->pBaseCBData, bDstToSrc, nPoints, x, y, z, panSuccess ); goto end; } bRet = GDALApproxTransformInternal( pCBData, bDstToSrc, nPoints, x, y, z, panSuccess, x2, y2, z2 ); end: #ifdef DEBUG_APPROX_TRANSFORMER for( int i = 0; i < nPoints; i++ ) fprintf(stderr, "[%d] (%.10f,%.10f) %d\n",/*ok*/ i, x[i], y[i], panSuccess[i]); #endif return bRet; } /************************************************************************/ /* GDALDeserializeApproxTransformer() */ /************************************************************************/ static void * GDALDeserializeApproxTransformer( CPLXMLNode *psTree ) { double dfMaxErrorForward = 0.25; double dfMaxErrorReverse = 0.25; const char* pszMaxError = CPLGetXMLValue( psTree, "MaxError", NULL); if( pszMaxError != NULL ) { dfMaxErrorForward = CPLAtof(pszMaxError); dfMaxErrorReverse = dfMaxErrorForward; } const char* pszMaxErrorForward = CPLGetXMLValue( psTree, "MaxErrorForward", NULL); if( pszMaxErrorForward != NULL ) { dfMaxErrorForward = CPLAtof(pszMaxErrorForward); } const char* pszMaxErrorReverse = CPLGetXMLValue( psTree, "MaxErrorReverse", NULL); if( pszMaxErrorForward != NULL ) { dfMaxErrorReverse = CPLAtof(pszMaxErrorReverse); } GDALTransformerFunc pfnBaseTransform = NULL; void *pBaseCBData = NULL; CPLXMLNode *psContainer = CPLGetXMLNode( psTree, "BaseTransformer" ); if( psContainer != NULL && psContainer->psChild != NULL ) { GDALDeserializeTransformer( psContainer->psChild, &pfnBaseTransform, &pBaseCBData ); } if( pfnBaseTransform == NULL ) { CPLError( CE_Failure, CPLE_AppDefined, "Cannot get base transform for approx transformer." ); return NULL; } void *pApproxCBData = GDALCreateApproxTransformer2( pfnBaseTransform, pBaseCBData, dfMaxErrorForward, dfMaxErrorReverse ); GDALApproxTransformerOwnsSubtransformer( pApproxCBData, TRUE ); return pApproxCBData; } /************************************************************************/ /* GDALApplyGeoTransform() */ /************************************************************************/ /** * Apply GeoTransform to x/y coordinate. * * Applies the following computation, converting a (pixel, line) coordinate * into a georeferenced (geo_x, geo_y) location. * * *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1] * + dfLine * padfGeoTransform[2]; * *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4] * + dfLine * padfGeoTransform[5]; * * @param padfGeoTransform Six coefficient GeoTransform to apply. * @param dfPixel Input pixel position. * @param dfLine Input line position. * @param pdfGeoX output location where geo_x (easting/longitude) * location is placed. * @param pdfGeoY output location where geo_y (northing/latitude) * location is placed. */ void CPL_STDCALL GDALApplyGeoTransform( double *padfGeoTransform, double dfPixel, double dfLine, double *pdfGeoX, double *pdfGeoY ) { *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1] + dfLine * padfGeoTransform[2]; *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4] + dfLine * padfGeoTransform[5]; } /************************************************************************/ /* GDALInvGeoTransform() */ /************************************************************************/ /** * Invert Geotransform. * * This function will invert a standard 3x2 set of GeoTransform coefficients. * This converts the equation from being pixel to geo to being geo to pixel. * * @param gt_in Input geotransform (six doubles - unaltered). * @param gt_out Output geotransform (six doubles - updated). * * @return TRUE on success or FALSE if the equation is uninvertable. */ int CPL_STDCALL GDALInvGeoTransform( double *gt_in, double *gt_out ) { // Special case - no rotation - to avoid computing determinate // and potential precision issues. if( gt_in[2] == 0.0 && gt_in[4] == 0.0 && gt_in[1] != 0.0 && gt_in[5] != 0.0 ) { /*X = gt_in[0] + x * gt_in[1] Y = gt_in[3] + y * gt_in[5] --> x = -gt_in[0] / gt_in[1] + (1 / gt_in[1]) * X y = -gt_in[3] / gt_in[5] + (1 / gt_in[5]) * Y */ gt_out[0] = -gt_in[0] / gt_in[1]; gt_out[1] = 1.0 / gt_in[1]; gt_out[2] = 0.0; gt_out[3] = -gt_in[3] / gt_in[5]; gt_out[4] = 0.0; gt_out[5] = 1.0 / gt_in[5]; return 1; } // Assume a 3rd row that is [1 0 0]. // Compute determinate. const double det = gt_in[1] * gt_in[5] - gt_in[2] * gt_in[4]; if( fabs(det) < 0.000000000000001 ) return 0; const double inv_det = 1.0 / det; // Compute adjoint, and divide by determinate. gt_out[1] = gt_in[5] * inv_det; gt_out[4] = -gt_in[4] * inv_det; gt_out[2] = -gt_in[2] * inv_det; gt_out[5] = gt_in[1] * inv_det; gt_out[0] = ( gt_in[2] * gt_in[3] - gt_in[0] * gt_in[5]) * inv_det; gt_out[3] = (-gt_in[1] * gt_in[3] + gt_in[0] * gt_in[4]) * inv_det; return 1; } /************************************************************************/ /* GDALSerializeTransformer() */ /************************************************************************/ CPLXMLNode *GDALSerializeTransformer( GDALTransformerFunc /* pfnFunc */, void *pTransformArg ) { VALIDATE_POINTER1( pTransformArg, "GDALSerializeTransformer", NULL ); GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "Attempt to serialize non-GTI2 transformer." ); return NULL; } else if( psInfo->pfnSerialize == NULL ) { CPLError( CE_Failure, CPLE_AppDefined, "No serialization function available for this transformer." ); return NULL; } return psInfo->pfnSerialize( pTransformArg ); } /************************************************************************/ /* GDALRegisterTransformDeserializer() */ /************************************************************************/ static CPLList* psListDeserializer = NULL; static CPLMutex* hDeserializerMutex = NULL; typedef struct { char* pszTransformName; GDALTransformerFunc pfnTransformerFunc; GDALTransformDeserializeFunc pfnDeserializeFunc; } TransformDeserializerInfo; void * GDALRegisterTransformDeserializer( const char* pszTransformName, GDALTransformerFunc pfnTransformerFunc, GDALTransformDeserializeFunc pfnDeserializeFunc ) { TransformDeserializerInfo* psInfo = static_cast<TransformDeserializerInfo *>( CPLMalloc(sizeof(TransformDeserializerInfo))); psInfo->pszTransformName = CPLStrdup(pszTransformName); psInfo->pfnTransformerFunc = pfnTransformerFunc; psInfo->pfnDeserializeFunc = pfnDeserializeFunc; CPLMutexHolderD(&hDeserializerMutex); psListDeserializer = CPLListInsert(psListDeserializer, psInfo, 0); return psInfo; } /************************************************************************/ /* GDALUnregisterTransformDeserializer() */ /************************************************************************/ void GDALUnregisterTransformDeserializer( void* pData ) { CPLMutexHolderD(&hDeserializerMutex); CPLList* psList = psListDeserializer; CPLList* psLast = NULL; while( psList ) { if( psList->pData == pData ) { TransformDeserializerInfo* psInfo = static_cast<TransformDeserializerInfo *>(pData); CPLFree(psInfo->pszTransformName); CPLFree(pData); if( psLast ) psLast->psNext = psList->psNext; else psListDeserializer = NULL; CPLFree(psList); break; } psLast = psList; psList = psList->psNext; } } /************************************************************************/ /* GDALUnregisterTransformDeserializer() */ /************************************************************************/ void GDALCleanupTransformDeserializerMutex() { if( hDeserializerMutex != NULL ) { CPLDestroyMutex(hDeserializerMutex); hDeserializerMutex = NULL; } } /************************************************************************/ /* GDALDeserializeTransformer() */ /************************************************************************/ CPLErr GDALDeserializeTransformer( CPLXMLNode *psTree, GDALTransformerFunc *ppfnFunc, void **ppTransformArg ) { *ppfnFunc = NULL; *ppTransformArg = NULL; CPLErrorReset(); if( psTree == NULL || psTree->eType != CXT_Element ) CPLError( CE_Failure, CPLE_AppDefined, "Malformed element in GDALDeserializeTransformer" ); else if( EQUAL(psTree->pszValue, "GenImgProjTransformer") ) { *ppfnFunc = GDALGenImgProjTransform; *ppTransformArg = GDALDeserializeGenImgProjTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "ReprojectionTransformer") ) { *ppfnFunc = GDALReprojectionTransform; *ppTransformArg = GDALDeserializeReprojectionTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "GCPTransformer") ) { *ppfnFunc = GDALGCPTransform; *ppTransformArg = GDALDeserializeGCPTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "TPSTransformer") ) { *ppfnFunc = GDALTPSTransform; *ppTransformArg = GDALDeserializeTPSTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "GeoLocTransformer") ) { *ppfnFunc = GDALGeoLocTransform; *ppTransformArg = GDALDeserializeGeoLocTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "RPCTransformer") ) { *ppfnFunc = GDALRPCTransform; *ppTransformArg = GDALDeserializeRPCTransformer( psTree ); } else if( EQUAL(psTree->pszValue, "ApproxTransformer") ) { *ppfnFunc = GDALApproxTransform; *ppTransformArg = GDALDeserializeApproxTransformer( psTree ); } else { GDALTransformDeserializeFunc pfnDeserializeFunc = NULL; { CPLMutexHolderD(&hDeserializerMutex); CPLList* psList = psListDeserializer; while( psList ) { TransformDeserializerInfo* psInfo = (TransformDeserializerInfo*)psList->pData; if( strcmp(psInfo->pszTransformName, psTree->pszValue) == 0 ) { *ppfnFunc = psInfo->pfnTransformerFunc; pfnDeserializeFunc = psInfo->pfnDeserializeFunc; break; } psList = psList->psNext; } } if( pfnDeserializeFunc != NULL ) { *ppTransformArg = pfnDeserializeFunc( psTree ); } else { CPLError( CE_Failure, CPLE_AppDefined, "Unrecognized element '%s' GDALDeserializeTransformer", psTree->pszValue ); } } return CPLGetLastErrorType(); } /************************************************************************/ /* GDALDestroyTransformer() */ /************************************************************************/ void GDALDestroyTransformer( void *pTransformArg ) { if( pTransformArg == NULL ) return; GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "Attempt to destroy non-GTI2 transformer." ); return; } psInfo->pfnCleanup( pTransformArg ); } /************************************************************************/ /* GDALUseTransformer() */ /************************************************************************/ int GDALUseTransformer( void *pTransformArg, int bDstToSrc, int nPointCount, double *x, double *y, double *z, int *panSuccess ) { GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "Attempt to use non-GTI2 transformer." ); return FALSE; } return psInfo->pfnTransform(pTransformArg, bDstToSrc, nPointCount, x, y, z, panSuccess); } /************************************************************************/ /* GDALCloneTransformer() */ /************************************************************************/ void *GDALCloneTransformer( void *pTransformArg ) { VALIDATE_POINTER1( pTransformArg, "GDALCloneTransformer", NULL ); GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "Attempt to clone non-GTI2 transformer." ); return NULL; } if( psInfo->pfnCreateSimilar != NULL ) { return psInfo->pfnCreateSimilar(psInfo, 1.0, 1.0); } if( psInfo->pfnSerialize == NULL ) { CPLError( CE_Failure, CPLE_AppDefined, "No serialization function available for this transformer." ); return NULL; } CPLXMLNode* pSerialized = psInfo->pfnSerialize( pTransformArg ); if( pSerialized == NULL ) return NULL; GDALTransformerFunc pfnTransformer = NULL; void *pClonedTransformArg = NULL; if( GDALDeserializeTransformer( pSerialized, &pfnTransformer, &pClonedTransformArg ) != CE_None ) { CPLDestroyXMLNode(pSerialized); CPLFree(pClonedTransformArg); return NULL; } CPLDestroyXMLNode(pSerialized); return pClonedTransformArg; } /************************************************************************/ /* GDALCreateSimilarTransformer() */ /************************************************************************/ void* GDALCreateSimilarTransformer( void* pTransformArg, double dfRatioX, double dfRatioY ) { VALIDATE_POINTER1( pTransformArg, "GDALCreateSimilarTransformer", NULL ); GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError( CE_Failure, CPLE_AppDefined, "Attempt to call CreateSimilar on a non-GTI2 transformer." ); return NULL; } if( psInfo->pfnCreateSimilar == NULL ) { CPLError(CE_Failure, CPLE_AppDefined, "No CreateSimilar function available for this transformer."); return NULL; } return psInfo->pfnCreateSimilar(psInfo, dfRatioX, dfRatioY); } /************************************************************************/ /* GDALSetTransformerDstGeoTransform() */ /************************************************************************/ /** * Set ApproxTransformer or GenImgProj output geotransform. * * This is a layer above GDALSetGenImgProjTransformerDstGeoTransform() that * checks that the passed hTransformArg is compatible. * * Normally the "destination geotransform", or transformation between * georeferenced output coordinates and pixel/line coordinates on the * destination file is extracted from the destination file by * GDALCreateGenImgProjTransformer() and stored in the GenImgProj private * info. However, sometimes it is inconvenient to have an output file * handle with appropriate geotransform information when creating the * transformation. For these cases, this function can be used to apply * the destination geotransform. * * @param pTransformArg the handle to update. * @param padfGeoTransform the destination geotransform to apply (six doubles). */ void GDALSetTransformerDstGeoTransform( void *pTransformArg, const double *padfGeoTransform ) { VALIDATE_POINTER0( pTransformArg, "GDALSetTransformerDstGeoTransform" ); GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError(CE_Failure, CPLE_AppDefined, "Attempt to call GDALSetTransformerDstGeoTransform on " "a non-GTI2 transformer."); return; } if( EQUAL(psInfo->pszClassName, "GDALApproxTransformer") ) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pTransformArg); psInfo = static_cast<GDALTransformerInfo *>(psATInfo->pBaseCBData); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError(CE_Failure, CPLE_AppDefined, "Attempt to call GDALSetTransformerDstGeoTransform on " "a non-GTI2 transformer."); return; } } if( EQUAL(psInfo->pszClassName, "GDALGenImgProjTransformer") ) { GDALSetGenImgProjTransformerDstGeoTransform(psInfo, padfGeoTransform); } } /************************************************************************/ /* GDALGetTransformerDstGeoTransform() */ /************************************************************************/ /** * Get ApproxTransformer or GenImgProj output geotransform. * * @param pTransformArg transformer handle. * @param padfGeoTransform (output) the destination geotransform to return (six doubles). */ void GDALGetTransformerDstGeoTransform( void *pTransformArg, double *padfGeoTransform ) { VALIDATE_POINTER0( pTransformArg, "GDALSetTransformerDstGeoTransform" ); GDALTransformerInfo *psInfo = static_cast<GDALTransformerInfo *>(pTransformArg); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError(CE_Failure, CPLE_AppDefined, "Attempt to call GDALGetTransformerDstGeoTransform on " "a non-GTI2 transformer."); return; } if( EQUAL(psInfo->pszClassName, "GDALApproxTransformer") ) { ApproxTransformInfo *psATInfo = static_cast<ApproxTransformInfo *>(pTransformArg); psInfo = static_cast<GDALTransformerInfo *>(psATInfo->pBaseCBData); if( psInfo == NULL || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE, strlen(GDAL_GTI2_SIGNATURE)) != 0 ) { CPLError(CE_Failure, CPLE_AppDefined, "Attempt to call GDALGetTransformerDstGeoTransform on " "a non-GTI2 transformer."); return; } } if( EQUAL(psInfo->pszClassName, "GDALGenImgProjTransformer") ) { GDALGenImgProjTransformInfo *psGenImgProjInfo = reinterpret_cast<GDALGenImgProjTransformInfo *>( psInfo ); memcpy( padfGeoTransform, psGenImgProjInfo->adfDstGeoTransform, sizeof(double) * 6 ); } }