Coverage Report

Created: 2025-11-16 06:25

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/gdal/alg/gdaltransformer.cpp
Line
Count
Source
1
/******************************************************************************
2
 *
3
 * Project:  Mapinfo Image Warper
4
 * Purpose:  Implementation of one or more GDALTrasformerFunc types, including
5
 *           the GenImgProj (general image reprojector) transformer.
6
 * Author:   Frank Warmerdam, warmerdam@pobox.com
7
 *
8
 ******************************************************************************
9
 * Copyright (c) 2002, i3 - information integration and imaging
10
 *                          Fort Collin, CO
11
 * Copyright (c) 2008-2013, Even Rouault <even dot rouault at spatialys.com>
12
 * Copyright (c) 2021, CLS
13
 *
14
 * SPDX-License-Identifier: MIT
15
 ****************************************************************************/
16
17
#include "cpl_port.h"
18
#include "gdal_alg.h"
19
#include "gdal_alg_priv.h"
20
21
#include <climits>
22
#include <cmath>
23
#include <cstddef>
24
#include <cstdlib>
25
#include <cstring>
26
27
#include <algorithm>
28
#include <limits>
29
#include <utility>
30
31
#include "cpl_conv.h"
32
#include "cpl_error.h"
33
#include "cpl_list.h"
34
#include "cpl_minixml.h"
35
#include "cpl_multiproc.h"
36
#include "cpl_string.h"
37
#include "cpl_vsi.h"
38
#include "gdal.h"
39
#include "gdal_priv.h"
40
#include "ogr_core.h"
41
#include "ogr_spatialref.h"
42
#include "ogr_srs_api.h"
43
44
CPL_C_START
45
void *GDALDeserializeGCPTransformer(CPLXMLNode *psTree);
46
void *GDALDeserializeTPSTransformer(CPLXMLNode *psTree);
47
void *GDALDeserializeGeoLocTransformer(CPLXMLNode *psTree);
48
void *GDALDeserializeRPCTransformer(CPLXMLNode *psTree);
49
void *GDALDeserializeHomographyTransformer(CPLXMLNode *psTree);
50
CPL_C_END
51
52
static CPLXMLNode *GDALSerializeReprojectionTransformer(void *pTransformArg);
53
static void *GDALDeserializeReprojectionTransformer(CPLXMLNode *psTree);
54
55
static CPLXMLNode *GDALSerializeGenImgProjTransformer(void *pTransformArg);
56
static void *GDALDeserializeGenImgProjTransformer(CPLXMLNode *psTree);
57
58
static void *GDALCreateApproxTransformer2(GDALTransformerFunc pfnRawTransformer,
59
                                          void *pRawTransformerArg,
60
                                          double dfMaxErrorForward,
61
                                          double dfMaxErrorReverse);
62
63
/************************************************************************/
64
/*                            GDALIsTransformer()                       */
65
/************************************************************************/
66
67
bool GDALIsTransformer(void *hTransformerArg, const char *pszClassName)
68
0
{
69
0
    if (!hTransformerArg)
70
0
        return false;
71
    // All transformers should have a GDALTransformerInfo member as their first members
72
0
    GDALTransformerInfo *psInfo =
73
0
        static_cast<GDALTransformerInfo *>(hTransformerArg);
74
0
    return memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
75
0
                  strlen(GDAL_GTI2_SIGNATURE)) == 0 &&
76
0
           strcmp(psInfo->pszClassName, pszClassName) == 0;
77
0
}
78
79
/************************************************************************/
80
/*                          GDALTransformFunc                           */
81
/*                                                                      */
82
/*      Documentation for GDALTransformFunc typedef.                    */
83
/************************************************************************/
84
85
/*!
86
87
\typedef typedef int (*GDALTransformerFunc)( void *pTransformerArg, int
88
bDstToSrc, int nPointCount, double *x, double *y, double *z, int *panSuccess );
89
90
Generic signature for spatial point transformers.
91
92
This function signature is used for a variety of functions that accept
93
passed in functions used to transform point locations between two coordinate
94
spaces.
95
96
The GDALCreateGenImgProjTransformer(), GDALCreateReprojectionTransformerEx(),
97
GDALCreateGCPTransformer() and GDALCreateApproxTransformer() functions can
98
be used to prepare argument data for some built-in transformers.  As well,
99
applications can implement their own transformers to the following signature.
100
101
\code
102
typedef int
103
(*GDALTransformerFunc)( void *pTransformerArg,
104
                        int bDstToSrc, int nPointCount,
105
                        double *x, double *y, double *z, int *panSuccess );
106
\endcode
107
108
@param pTransformerArg application supplied callback data used by the
109
transformer.
110
111
@param bDstToSrc if TRUE the transformation will be from the destination
112
coordinate space to the source coordinate system, otherwise the transformation
113
will be from the source coordinate system to the destination coordinate system.
114
115
@param nPointCount number of points in the x, y and z arrays.
116
117
@param[in,out] x input X coordinates.  Results returned in same array.
118
119
@param[in,out] y input Y coordinates.  Results returned in same array.
120
121
@param[in,out] z input Z coordinates.  Results returned in same array.
122
123
@param[out] panSuccess array of ints in which success (TRUE) or failure (FALSE)
124
flags are returned for the translation of each point. Must not be NULL.
125
126
@return TRUE if all points have been successfully transformed (changed in 3.11,
127
previously was TRUE if some points have been successfully transformed)
128
129
*/
130
131
/************************************************************************/
132
/*                      GDALSuggestedWarpOutput()                       */
133
/************************************************************************/
134
135
/**
136
 * Suggest output file size.
137
 *
138
 * This function is used to suggest the size, and georeferenced extents
139
 * appropriate given the indicated transformation and input file.  It walks
140
 * the edges of the input file (approximately 20 sample points along each
141
 * edge) transforming into output coordinates in order to get an extents box.
142
 *
143
 * Then a resolution is computed with the intent that the length of the
144
 * distance from the top left corner of the output imagery to the bottom right
145
 * corner would represent the same number of pixels as in the source image.
146
 * Note that if the image is somewhat rotated the diagonal taken isn't of the
147
 * whole output bounding rectangle, but instead of the locations where the
148
 * top/left and bottom/right corners transform.  The output pixel size is
149
 * always square.  This is intended to approximately preserve the resolution
150
 * of the input data in the output file.
151
 *
152
 * The values returned in padfGeoTransformOut, pnPixels and pnLines are
153
 * the suggested number of pixels and lines for the output file, and the
154
 * geotransform relating those pixels to the output georeferenced coordinates.
155
 *
156
 * The trickiest part of using the function is ensuring that the
157
 * transformer created is from source file pixel/line coordinates to
158
 * output file georeferenced coordinates.  This can be accomplished with
159
 * GDALCreateGenImgProjTransformer() by passing a NULL for the hDstDS.
160
 *
161
 * @param hSrcDS the input image (it is assumed the whole input image is
162
 * being transformed).
163
 * @param pfnTransformer the transformer function.
164
 * @param pTransformArg the callback data for the transformer function.
165
 * @param padfGeoTransformOut the array of six doubles in which the suggested
166
 * geotransform is returned.
167
 * @param pnPixels int in which the suggest pixel width of output is returned.
168
 * @param pnLines int in which the suggest pixel height of output is returned.
169
 *
170
 * @return CE_None if successful or CE_Failure otherwise.
171
 */
172
173
CPLErr CPL_STDCALL GDALSuggestedWarpOutput(GDALDatasetH hSrcDS,
174
                                           GDALTransformerFunc pfnTransformer,
175
                                           void *pTransformArg,
176
                                           double *padfGeoTransformOut,
177
                                           int *pnPixels, int *pnLines)
178
179
0
{
180
0
    VALIDATE_POINTER1(hSrcDS, "GDALSuggestedWarpOutput", CE_Failure);
181
182
0
    double adfExtent[4] = {};
183
184
0
    return GDALSuggestedWarpOutput2(hSrcDS, pfnTransformer, pTransformArg,
185
0
                                    padfGeoTransformOut, pnPixels, pnLines,
186
0
                                    adfExtent, 0);
187
0
}
188
189
static bool GDALSuggestedWarpOutput2_MustAdjustForRightBorder(
190
    GDALTransformerFunc pfnTransformer, void *pTransformArg, double *padfExtent,
191
    int /* nPixels*/, int nLines, double dfPixelSizeX, double dfPixelSizeY)
192
0
{
193
0
    double adfX[21] = {};
194
0
    double adfY[21] = {};
195
196
0
    const double dfMaxXOut = padfExtent[2];
197
0
    const double dfMaxYOut = padfExtent[3];
198
199
    // Take 20 steps.
200
0
    int nSamplePoints = 0;
201
0
    for (double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05)
202
0
    {
203
        // Ensure we end exactly at the end.
204
0
        if (dfRatio > 0.99)
205
0
            dfRatio = 1.0;
206
207
        // Along right.
208
0
        adfX[nSamplePoints] = dfMaxXOut;
209
0
        adfY[nSamplePoints] = dfMaxYOut - dfPixelSizeY * dfRatio * nLines;
210
0
        nSamplePoints++;
211
0
    }
212
0
    double adfZ[21] = {};
213
214
0
    int abSuccess[21] = {};
215
216
0
    pfnTransformer(pTransformArg, TRUE, nSamplePoints, adfX, adfY, adfZ,
217
0
                   abSuccess);
218
219
0
    int abSuccess2[21] = {};
220
221
0
    pfnTransformer(pTransformArg, FALSE, nSamplePoints, adfX, adfY, adfZ,
222
0
                   abSuccess2);
223
224
0
    nSamplePoints = 0;
225
0
    int nBadCount = 0;
226
0
    for (double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05)
227
0
    {
228
0
        const double expected_x = dfMaxXOut;
229
0
        const double expected_y = dfMaxYOut - dfPixelSizeY * dfRatio * nLines;
230
0
        if (!abSuccess[nSamplePoints] || !abSuccess2[nSamplePoints] ||
231
0
            fabs(adfX[nSamplePoints] - expected_x) > dfPixelSizeX ||
232
0
            fabs(adfY[nSamplePoints] - expected_y) > dfPixelSizeY)
233
0
        {
234
0
            nBadCount++;
235
0
        }
236
0
        nSamplePoints++;
237
0
    }
238
239
0
    return nBadCount == nSamplePoints;
240
0
}
241
242
static bool GDALSuggestedWarpOutput2_MustAdjustForBottomBorder(
243
    GDALTransformerFunc pfnTransformer, void *pTransformArg, double *padfExtent,
244
    int nPixels, int /* nLines */, double dfPixelSizeX, double dfPixelSizeY)
245
0
{
246
0
    double adfX[21] = {};
247
0
    double adfY[21] = {};
248
249
0
    const double dfMinXOut = padfExtent[0];
250
0
    const double dfMinYOut = padfExtent[1];
251
252
    // Take 20 steps.
253
0
    int nSamplePoints = 0;
254
0
    for (double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05)
255
0
    {
256
        // Ensure we end exactly at the end.
257
0
        if (dfRatio > 0.99)
258
0
            dfRatio = 1.0;
259
260
        // Along right.
261
0
        adfX[nSamplePoints] = dfMinXOut + dfPixelSizeX * dfRatio * nPixels;
262
0
        adfY[nSamplePoints] = dfMinYOut;
263
0
        nSamplePoints++;
264
0
    }
265
0
    double adfZ[21] = {};
266
267
0
    int abSuccess[21] = {};
268
269
0
    pfnTransformer(pTransformArg, TRUE, nSamplePoints, adfX, adfY, adfZ,
270
0
                   abSuccess);
271
272
0
    int abSuccess2[21] = {};
273
274
0
    pfnTransformer(pTransformArg, FALSE, nSamplePoints, adfX, adfY, adfZ,
275
0
                   abSuccess2);
276
277
0
    nSamplePoints = 0;
278
0
    int nBadCount = 0;
279
0
    for (double dfRatio = 0.0; dfRatio <= 1.01; dfRatio += 0.05)
280
0
    {
281
0
        const double expected_x = dfMinXOut + dfPixelSizeX * dfRatio * nPixels;
282
0
        const double expected_y = dfMinYOut;
283
0
        if (!abSuccess[nSamplePoints] || !abSuccess2[nSamplePoints] ||
284
0
            fabs(adfX[nSamplePoints] - expected_x) > dfPixelSizeX ||
285
0
            fabs(adfY[nSamplePoints] - expected_y) > dfPixelSizeY)
286
0
        {
287
0
            nBadCount++;
288
0
        }
289
0
        nSamplePoints++;
290
0
    }
291
292
0
    return nBadCount == nSamplePoints;
293
0
}
294
295
/************************************************************************/
296
/*                      GDALSuggestedWarpOutput2()                      */
297
/************************************************************************/
298
299
/**
300
 * Suggest output file size.
301
 *
302
 * This function is used to suggest the size, and georeferenced extents
303
 * appropriate given the indicated transformation and input file.  It walks
304
 * the edges of the input file (approximately 20 sample points along each
305
 * edge) transforming into output coordinates in order to get an extents box.
306
 *
307
 * Then a resolution is computed with the intent that the length of the
308
 * distance from the top left corner of the output imagery to the bottom right
309
 * corner would represent the same number of pixels as in the source image.
310
 * Note that if the image is somewhat rotated the diagonal taken isn't of the
311
 * whole output bounding rectangle, but instead of the locations where the
312
 * top/left and bottom/right corners transform.  The output pixel size is
313
 * always square.  This is intended to approximately preserve the resolution
314
 * of the input data in the output file.
315
 *
316
 * The values returned in padfGeoTransformOut, pnPixels and pnLines are
317
 * the suggested number of pixels and lines for the output file, and the
318
 * geotransform relating those pixels to the output georeferenced coordinates.
319
 *
320
 * The trickiest part of using the function is ensuring that the
321
 * transformer created is from source file pixel/line coordinates to
322
 * output file georeferenced coordinates.  This can be accomplished with
323
 * GDALCreateGenImgProjTransformer() by passing a NULL for the hDstDS.
324
 *
325
 * @param hSrcDS the input image (it is assumed the whole input image is
326
 * being transformed).
327
 * @param pfnTransformer the transformer function.
328
 * @param pTransformArg the callback data for the transformer function.
329
 * @param padfGeoTransformOut the array of six doubles in which the suggested
330
 * geotransform is returned.
331
 * @param pnPixels int in which the suggest pixel width of output is returned.
332
 * @param pnLines int in which the suggest pixel height of output is returned.
333
 * @param padfExtent Four entry array to return extents as (xmin, ymin, xmax,
334
 * ymax).
335
 * @param nOptions Options flags. Zero or GDAL_SWO_ROUND_UP_SIZE  to ask *pnPixels
336
 * and *pnLines to be rounded up instead of being rounded to the closes integer, or
337
 * GDAL_SWO_FORCE_SQUARE_PIXEL to indicate that the generated pixel size is a square.
338
 *
339
 * @return CE_None if successful or CE_Failure otherwise.
340
 */
341
342
CPLErr CPL_STDCALL GDALSuggestedWarpOutput2(GDALDatasetH hSrcDS,
343
                                            GDALTransformerFunc pfnTransformer,
344
                                            void *pTransformArg,
345
                                            double *padfGeoTransformOut,
346
                                            int *pnPixels, int *pnLines,
347
                                            double *padfExtent, int nOptions)
348
0
{
349
0
    VALIDATE_POINTER1(hSrcDS, "GDALSuggestedWarpOutput2", CE_Failure);
350
351
0
    const bool bIsGDALGenImgProjTransform{
352
0
        pTransformArg &&
353
0
        GDALIsTransformer(pTransformArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME)};
354
355
    /* -------------------------------------------------------------------- */
356
    /*      Setup sample points all around the edge of the input raster.    */
357
    /* -------------------------------------------------------------------- */
358
0
    if (bIsGDALGenImgProjTransform)
359
0
    {
360
        // In case CHECK_WITH_INVERT_PROJ has been modified.
361
0
        GDALRefreshGenImgProjTransformer(pTransformArg);
362
0
    }
363
0
    else if (GDALIsTransformer(pTransformArg,
364
0
                               GDAL_APPROX_TRANSFORMER_CLASS_NAME))
365
0
    {
366
        // In case CHECK_WITH_INVERT_PROJ has been modified.
367
0
        GDALRefreshApproxTransformer(pTransformArg);
368
0
    }
369
370
0
    const int nInXSize = GDALGetRasterXSize(hSrcDS);
371
0
    const int nInYSize = GDALGetRasterYSize(hSrcDS);
372
373
    /* ------------------------------------------------------------- */
374
    /* Special case for warping on the same (or null) CRS.           */
375
    /* ------------------------------------------------------------- */
376
0
    if ((!nOptions || (nOptions & GDAL_SWO_FORCE_SQUARE_PIXEL) == 0) &&
377
0
        pTransformArg && bIsGDALGenImgProjTransform)
378
0
    {
379
0
        const GDALGenImgProjTransformInfo *psInfo =
380
0
            static_cast<const GDALGenImgProjTransformInfo *>(pTransformArg);
381
382
0
        if (!psInfo->sSrcParams.pTransformer &&
383
0
            !psInfo->bHasCustomTransformationPipeline &&
384
0
            !psInfo->sDstParams.pTransformer &&
385
0
            psInfo->sSrcParams.adfGeoTransform[2] == 0 &&
386
0
            psInfo->sSrcParams.adfGeoTransform[4] == 0 &&
387
0
            psInfo->sDstParams.adfGeoTransform[0] == 0 &&
388
0
            psInfo->sDstParams.adfGeoTransform[1] == 1 &&
389
0
            psInfo->sDstParams.adfGeoTransform[2] == 0 &&
390
0
            psInfo->sDstParams.adfGeoTransform[3] == 0 &&
391
0
            psInfo->sDstParams.adfGeoTransform[4] == 0 &&
392
0
            psInfo->sDstParams.adfGeoTransform[5] == 1)
393
0
        {
394
0
            const OGRSpatialReference *poSourceCRS = nullptr;
395
0
            const OGRSpatialReference *poTargetCRS = nullptr;
396
397
0
            if (psInfo->pReprojectArg)
398
0
            {
399
0
                const GDALReprojectionTransformInfo *psRTI =
400
0
                    static_cast<const GDALReprojectionTransformInfo *>(
401
0
                        psInfo->pReprojectArg);
402
0
                poSourceCRS = psRTI->poForwardTransform->GetSourceCS();
403
0
                poTargetCRS = psRTI->poForwardTransform->GetTargetCS();
404
0
            }
405
406
0
            if ((!poSourceCRS && !poTargetCRS) ||
407
0
                (poSourceCRS && poTargetCRS &&
408
0
                 poSourceCRS->IsSame(poTargetCRS)))
409
0
            {
410
411
0
                const bool bNorthUp{psInfo->sSrcParams.adfGeoTransform[5] <
412
0
                                    0.0};
413
414
0
                memcpy(padfGeoTransformOut, psInfo->sSrcParams.adfGeoTransform,
415
0
                       sizeof(double) * 6);
416
417
0
                if (!bNorthUp)
418
0
                {
419
0
                    padfGeoTransformOut[3] = padfGeoTransformOut[3] +
420
0
                                             nInYSize * padfGeoTransformOut[5];
421
0
                    padfGeoTransformOut[5] = -padfGeoTransformOut[5];
422
0
                }
423
424
0
                *pnPixels = nInXSize;
425
0
                *pnLines = nInYSize;
426
427
                // Calculate extent from hSrcDS
428
0
                if (padfExtent)
429
0
                {
430
0
                    padfExtent[0] = psInfo->sSrcParams.adfGeoTransform[0];
431
0
                    padfExtent[1] =
432
0
                        psInfo->sSrcParams.adfGeoTransform[3] +
433
0
                        nInYSize * psInfo->sSrcParams.adfGeoTransform[5];
434
0
                    padfExtent[2] =
435
0
                        psInfo->sSrcParams.adfGeoTransform[0] +
436
0
                        nInXSize * psInfo->sSrcParams.adfGeoTransform[1];
437
0
                    padfExtent[3] = psInfo->sSrcParams.adfGeoTransform[3];
438
0
                    if (!bNorthUp)
439
0
                    {
440
0
                        std::swap(padfExtent[1], padfExtent[3]);
441
0
                    }
442
0
                }
443
0
                return CE_None;
444
0
            }
445
0
        }
446
0
    }
447
448
0
    const int N_PIXELSTEP = 50;
449
0
    int nSteps = static_cast<int>(
450
0
        static_cast<double>(std::min(nInYSize, nInXSize)) / N_PIXELSTEP + 0.5);
451
0
    if (nSteps < 20)
452
0
        nSteps = 20;
453
0
    else if (nSteps > 100)
454
0
        nSteps = 100;
455
456
    // TODO(rouault): How is this goto retry supposed to work?  Added in r20537.
457
    // Does redoing the same malloc multiple times work?  If it is needed, can
458
    // it be converted to a tigher while loop around the MALLOC3s and free?  Is
459
    // the point to try with the full requested steps.  Then, if there is not
460
    // enough memory, back off and try with just 20 steps?
461
0
retry:
462
0
    int nStepsPlusOne = nSteps + 1;
463
0
    int nSampleMax = nStepsPlusOne * nStepsPlusOne;
464
465
0
    double dfStep = 1.0 / nSteps;
466
0
    double *padfY = nullptr;
467
0
    double *padfZ = nullptr;
468
0
    double *padfYRevert = nullptr;
469
0
    double *padfZRevert = nullptr;
470
471
0
    int *pabSuccess = static_cast<int *>(
472
0
        VSI_MALLOC3_VERBOSE(sizeof(int), nStepsPlusOne, nStepsPlusOne));
473
0
    double *padfX = static_cast<double *>(
474
0
        VSI_MALLOC3_VERBOSE(sizeof(double) * 3, nStepsPlusOne, nStepsPlusOne));
475
0
    double *padfXRevert = static_cast<double *>(
476
0
        VSI_MALLOC3_VERBOSE(sizeof(double) * 3, nStepsPlusOne, nStepsPlusOne));
477
0
    if (pabSuccess == nullptr || padfX == nullptr || padfXRevert == nullptr)
478
0
    {
479
0
        CPLFree(padfX);
480
0
        CPLFree(padfXRevert);
481
0
        CPLFree(pabSuccess);
482
0
        if (nSteps > 20)
483
0
        {
484
0
            nSteps = 20;
485
0
            goto retry;
486
0
        }
487
0
        return CE_Failure;
488
0
    }
489
490
0
    padfY = padfX + nSampleMax;
491
0
    padfZ = padfX + nSampleMax * 2;
492
0
    padfYRevert = padfXRevert + nSampleMax;
493
0
    padfZRevert = padfXRevert + nSampleMax * 2;
494
495
    // Take N_STEPS steps.
496
0
    for (int iStep = 0; iStep <= nSteps; iStep++)
497
0
    {
498
0
        double dfRatio = (iStep == nSteps) ? 1.0 : iStep * dfStep;
499
0
        int iStep2 = iStep;
500
501
        // Along top.
502
0
        padfX[iStep2] = dfRatio * nInXSize;
503
0
        padfY[iStep2] = 0.0;
504
0
        padfZ[iStep2] = 0.0;
505
506
        // Along bottom.
507
0
        iStep2 += nStepsPlusOne;
508
0
        padfX[iStep2] = dfRatio * nInXSize;
509
0
        padfY[iStep2] = nInYSize;
510
0
        padfZ[iStep2] = 0.0;
511
512
        // Along left.
513
0
        iStep2 += nStepsPlusOne;
514
0
        padfX[iStep2] = 0.0;
515
0
        padfY[iStep2] = dfRatio * nInYSize;
516
0
        padfZ[iStep2] = 0.0;
517
518
        // Along right.
519
0
        iStep2 += nStepsPlusOne;
520
0
        padfX[iStep2] = nInXSize;
521
0
        padfY[iStep2] = dfRatio * nInYSize;
522
0
        padfZ[iStep2] = 0.0;
523
0
    }
524
525
0
    int nSamplePoints = 4 * nStepsPlusOne;
526
527
0
    memset(pabSuccess, 1, sizeof(int) * nSampleMax);
528
529
    /* -------------------------------------------------------------------- */
530
    /*      Transform them to the output coordinate system.                 */
531
    /* -------------------------------------------------------------------- */
532
0
    {
533
0
        CPLTurnFailureIntoWarningBackuper oErrorsToWarnings{};
534
0
        pfnTransformer(pTransformArg, FALSE, nSamplePoints, padfX, padfY, padfZ,
535
0
                       pabSuccess);
536
0
    }
537
0
    constexpr int SIGN_FINAL_UNINIT = -2;
538
0
    constexpr int SIGN_FINAL_INVALID = 0;
539
0
    int iSignDiscontinuity = SIGN_FINAL_UNINIT;
540
0
    int nFailedCount = 0;
541
0
    const int iSignArray[2] = {-1, 1};
542
0
    for (int i = 0; i < nSamplePoints; i++)
543
0
    {
544
0
        if (pabSuccess[i])
545
0
        {
546
            // Fix for https://trac.osgeo.org/gdal/ticket/7243
547
            // where echo "-2050000.000 2050000.000" |
548
            //              gdaltransform -s_srs EPSG:3411 -t_srs EPSG:4326
549
            // gives "-180 63.691332898492"
550
            // but we would rather like 180
551
0
            if (iSignDiscontinuity == 1 || iSignDiscontinuity == -1)
552
0
            {
553
0
                if (!((iSignDiscontinuity * padfX[i] > 0 &&
554
0
                       iSignDiscontinuity * padfX[i] <= 180.0) ||
555
0
                      (fabs(padfX[i] - iSignDiscontinuity * -180.0) < 1e-8)))
556
0
                {
557
0
                    iSignDiscontinuity = SIGN_FINAL_INVALID;
558
0
                }
559
0
            }
560
0
            else if (iSignDiscontinuity == SIGN_FINAL_UNINIT)
561
0
            {
562
0
                for (const auto &iSign : iSignArray)
563
0
                {
564
0
                    if ((iSign * padfX[i] > 0 && iSign * padfX[i] <= 180.0) ||
565
0
                        (fabs(padfX[i] - iSign * -180.0) < 1e-8))
566
0
                    {
567
0
                        iSignDiscontinuity = iSign;
568
0
                        break;
569
0
                    }
570
0
                }
571
0
                if (iSignDiscontinuity == SIGN_FINAL_UNINIT)
572
0
                {
573
0
                    iSignDiscontinuity = SIGN_FINAL_INVALID;
574
0
                }
575
0
            }
576
0
        }
577
0
        else
578
0
        {
579
0
            nFailedCount++;
580
0
        }
581
0
    }
582
583
0
    if (iSignDiscontinuity == 1 || iSignDiscontinuity == -1)
584
0
    {
585
0
        for (int i = 0; i < nSamplePoints; i++)
586
0
        {
587
0
            if (pabSuccess[i])
588
0
            {
589
0
                if (fabs(padfX[i] - iSignDiscontinuity * -180.0) < 1e-8)
590
0
                {
591
0
                    double axTemp[2] = {iSignDiscontinuity * -180.0,
592
0
                                        iSignDiscontinuity * 180.0};
593
0
                    double ayTemp[2] = {padfY[i], padfY[i]};
594
0
                    double azTemp[2] = {padfZ[i], padfZ[i]};
595
0
                    int abSuccess[2] = {FALSE, FALSE};
596
0
                    CPLTurnFailureIntoWarningBackuper oErrorsToWarnings{};
597
0
                    if (pfnTransformer(pTransformArg, TRUE, 2, axTemp, ayTemp,
598
0
                                       azTemp, abSuccess) &&
599
0
                        fabs(axTemp[0] - axTemp[1]) < 1e-8 &&
600
0
                        fabs(ayTemp[0] - ayTemp[1]) < 1e-8)
601
0
                    {
602
0
                        padfX[i] = iSignDiscontinuity * 180.0;
603
0
                    }
604
0
                }
605
0
            }
606
0
        }
607
0
    }
608
609
    /* -------------------------------------------------------------------- */
610
    /*      Check if the computed target coordinates are revertable.        */
611
    /*      If not, try the detailed grid sampling.                         */
612
    /* -------------------------------------------------------------------- */
613
0
    if (nFailedCount)
614
0
    {
615
0
        CPLDebug("WARP", "At least one point failed after direct transform");
616
0
    }
617
0
    else
618
0
    {
619
0
        memcpy(padfXRevert, padfX, nSamplePoints * sizeof(double));
620
0
        memcpy(padfYRevert, padfY, nSamplePoints * sizeof(double));
621
0
        memcpy(padfZRevert, padfZ, nSamplePoints * sizeof(double));
622
0
        {
623
0
            CPLTurnFailureIntoWarningBackuper oErrorsToWarnings{};
624
0
            pfnTransformer(pTransformArg, TRUE, nSamplePoints, padfXRevert,
625
0
                           padfYRevert, padfZRevert, pabSuccess);
626
0
        }
627
628
0
        for (int i = 0; nFailedCount == 0 && i < nSamplePoints; i++)
629
0
        {
630
0
            if (!pabSuccess[i])
631
0
            {
632
0
                nFailedCount++;
633
0
                break;
634
0
            }
635
636
0
            double dfRatio = (i % nStepsPlusOne) * dfStep;
637
0
            if (dfRatio > 0.99)
638
0
                dfRatio = 1.0;
639
640
0
            double dfExpectedX = 0.0;
641
0
            double dfExpectedY = 0.0;
642
0
            if (i < nStepsPlusOne)
643
0
            {
644
0
                dfExpectedX = dfRatio * nInXSize;
645
0
            }
646
0
            else if (i < 2 * nStepsPlusOne)
647
0
            {
648
0
                dfExpectedX = dfRatio * nInXSize;
649
0
                dfExpectedY = nInYSize;
650
0
            }
651
0
            else if (i < 3 * nStepsPlusOne)
652
0
            {
653
0
                dfExpectedY = dfRatio * nInYSize;
654
0
            }
655
0
            else
656
0
            {
657
0
                dfExpectedX = nInXSize;
658
0
                dfExpectedY = dfRatio * nInYSize;
659
0
            }
660
661
0
            if (fabs(padfXRevert[i] - dfExpectedX) >
662
0
                    nInXSize / static_cast<double>(nSteps) ||
663
0
                fabs(padfYRevert[i] - dfExpectedY) >
664
0
                    nInYSize / static_cast<double>(nSteps))
665
0
                nFailedCount++;
666
0
        }
667
0
        if (nFailedCount != 0)
668
0
            CPLDebug("WARP",
669
0
                     "At least one point failed after revert transform");
670
0
    }
671
672
    /* -------------------------------------------------------------------- */
673
    /*      If any of the edge points failed to transform, we need to       */
674
    /*      build a fairly detailed internal grid of points instead to      */
675
    /*      help identify the area that is transformable.                   */
676
    /* -------------------------------------------------------------------- */
677
0
    if (nFailedCount)
678
0
    {
679
0
        nSamplePoints = 0;
680
681
        // Take N_STEPS steps.
682
0
        for (int iStep = 0; iStep <= nSteps; iStep++)
683
0
        {
684
0
            double dfRatio = (iStep == nSteps) ? 1.0 : iStep * dfStep;
685
686
0
            for (int iStep2 = 0; iStep2 <= nSteps; iStep2++)
687
0
            {
688
0
                const double dfRatio2 =
689
0
                    iStep2 == nSteps ? 1.0 : iStep2 * dfStep;
690
691
                // From top to bottom, from left to right.
692
0
                padfX[nSamplePoints] = dfRatio2 * nInXSize;
693
0
                padfY[nSamplePoints] = dfRatio * nInYSize;
694
0
                padfZ[nSamplePoints] = 0.0;
695
0
                nSamplePoints++;
696
0
            }
697
0
        }
698
699
0
        CPLAssert(nSamplePoints == nSampleMax);
700
701
0
        {
702
0
            CPLTurnFailureIntoWarningBackuper oErrorsToWarnings{};
703
0
            pfnTransformer(pTransformArg, FALSE, nSamplePoints, padfX, padfY,
704
0
                           padfZ, pabSuccess);
705
0
        }
706
0
    }
707
708
    /* -------------------------------------------------------------------- */
709
    /*      Collect the bounds, ignoring any failed points.                 */
710
    /* -------------------------------------------------------------------- */
711
0
    double dfMinXOut = 0.0;
712
0
    double dfMinYOut = 0.0;
713
0
    double dfMaxXOut = 0.0;
714
0
    double dfMaxYOut = 0.0;
715
0
    bool bGotInitialPoint = false;
716
717
0
    nFailedCount = 0;
718
0
    for (int i = 0; i < nSamplePoints; i++)
719
0
    {
720
0
        int x_i = 0;
721
0
        int y_i = 0;
722
723
0
        if (nSamplePoints == nSampleMax)
724
0
        {
725
0
            x_i = i % nStepsPlusOne;
726
0
            y_i = i / nStepsPlusOne;
727
0
        }
728
0
        else
729
0
        {
730
0
            if (i < 2 * nStepsPlusOne)
731
0
            {
732
0
                x_i = i % nStepsPlusOne;
733
0
                y_i = (i < nStepsPlusOne) ? 0 : nSteps;
734
0
            }
735
0
        }
736
737
0
        if (x_i > 0 && (pabSuccess[i - 1] || pabSuccess[i]))
738
0
        {
739
0
            double x_out_before = padfX[i - 1];
740
0
            double x_out_after = padfX[i];
741
0
            int nIter = 0;
742
0
            double x_in_before =
743
0
                static_cast<double>(x_i - 1) * nInXSize / nSteps;
744
0
            double x_in_after = static_cast<double>(x_i) * nInXSize / nSteps;
745
0
            int invalid_before = !(pabSuccess[i - 1]);
746
0
            int invalid_after = !(pabSuccess[i]);
747
748
            // Detect discontinuity in target coordinates when the target x
749
            // coordinates change sign. This may be a false positive when the
750
            // target tx is around 0 Dichotomic search to reduce the interval
751
            // to near the discontinuity and get a better out extent.
752
0
            while ((invalid_before || invalid_after ||
753
0
                    x_out_before * x_out_after < 0.0) &&
754
0
                   nIter < 16)
755
0
            {
756
0
                double x = (x_in_before + x_in_after) / 2.0;
757
0
                double y = static_cast<double>(y_i) * nInYSize / nSteps;
758
0
                double z = 0.0;
759
0
                int bSuccess = TRUE;
760
0
                if (pfnTransformer(pTransformArg, FALSE, 1, &x, &y, &z,
761
0
                                   &bSuccess) &&
762
0
                    bSuccess)
763
0
                {
764
0
                    if (bGotInitialPoint)
765
0
                    {
766
0
                        dfMinXOut = std::min(dfMinXOut, x);
767
0
                        dfMinYOut = std::min(dfMinYOut, y);
768
0
                        dfMaxXOut = std::max(dfMaxXOut, x);
769
0
                        dfMaxYOut = std::max(dfMaxYOut, y);
770
0
                    }
771
0
                    else
772
0
                    {
773
0
                        bGotInitialPoint = true;
774
0
                        dfMinXOut = x;
775
0
                        dfMaxXOut = x;
776
0
                        dfMinYOut = y;
777
0
                        dfMaxYOut = y;
778
0
                    }
779
780
0
                    if (invalid_before || x_out_before * x < 0)
781
0
                    {
782
0
                        invalid_after = FALSE;
783
0
                        x_in_after = (x_in_before + x_in_after) / 2.0;
784
0
                        x_out_after = x;
785
0
                    }
786
0
                    else
787
0
                    {
788
0
                        invalid_before = FALSE;
789
0
                        x_out_before = x;
790
0
                        x_in_before = (x_in_before + x_in_after) / 2.0;
791
0
                    }
792
0
                }
793
0
                else
794
0
                {
795
0
                    if (invalid_before)
796
0
                    {
797
0
                        x_in_before = (x_in_before + x_in_after) / 2.0;
798
0
                    }
799
0
                    else if (invalid_after)
800
0
                    {
801
0
                        x_in_after = (x_in_before + x_in_after) / 2.0;
802
0
                    }
803
0
                    else
804
0
                    {
805
0
                        break;
806
0
                    }
807
0
                }
808
0
                nIter++;
809
0
            }
810
0
        }
811
812
0
        if (!pabSuccess[i])
813
0
        {
814
0
            nFailedCount++;
815
0
            continue;
816
0
        }
817
818
0
        if (bGotInitialPoint)
819
0
        {
820
0
            dfMinXOut = std::min(dfMinXOut, padfX[i]);
821
0
            dfMinYOut = std::min(dfMinYOut, padfY[i]);
822
0
            dfMaxXOut = std::max(dfMaxXOut, padfX[i]);
823
0
            dfMaxYOut = std::max(dfMaxYOut, padfY[i]);
824
0
        }
825
0
        else
826
0
        {
827
0
            bGotInitialPoint = true;
828
0
            dfMinXOut = padfX[i];
829
0
            dfMaxXOut = padfX[i];
830
0
            dfMinYOut = padfY[i];
831
0
            dfMaxYOut = padfY[i];
832
0
        }
833
0
    }
834
835
0
    if (nFailedCount > nSamplePoints - 10)
836
0
    {
837
0
        CPLError(CE_Failure, CPLE_AppDefined,
838
0
                 "Too many points (%d out of %d) failed to transform, "
839
0
                 "unable to compute output bounds.",
840
0
                 nFailedCount, nSamplePoints);
841
842
0
        CPLFree(padfX);
843
0
        CPLFree(padfXRevert);
844
0
        CPLFree(pabSuccess);
845
846
0
        return CE_Failure;
847
0
    }
848
849
0
    if (nFailedCount)
850
0
        CPLDebug("GDAL",
851
0
                 "GDALSuggestedWarpOutput(): %d out of %d points failed to "
852
0
                 "transform.",
853
0
                 nFailedCount, nSamplePoints);
854
855
0
    bool bIsGeographicCoordsDeg = false;
856
0
    if (bIsGDALGenImgProjTransform)
857
0
    {
858
0
        const GDALGenImgProjTransformInfo *pGIPTI =
859
0
            static_cast<const GDALGenImgProjTransformInfo *>(pTransformArg);
860
0
        if (pGIPTI->sSrcParams.pTransformer == GDALGeoLocTransform &&
861
0
            pGIPTI->sDstParams.pTransformer == nullptr &&
862
0
            pGIPTI->sDstParams.adfGeoTransform[0] == 0 &&
863
0
            pGIPTI->sDstParams.adfGeoTransform[1] == 1 &&
864
0
            pGIPTI->sDstParams.adfGeoTransform[2] == 0 &&
865
0
            pGIPTI->sDstParams.adfGeoTransform[3] == 0 &&
866
0
            pGIPTI->sDstParams.adfGeoTransform[4] == 0 &&
867
0
            pGIPTI->sDstParams.adfGeoTransform[5] == 1)
868
0
        {
869
            /* --------------------------------------------------------------------
870
             */
871
            /*      Special case for geolocation array, to quickly find the
872
             * bounds. */
873
            /* --------------------------------------------------------------------
874
             */
875
0
            const GDALGeoLocTransformInfo *pGLTI =
876
0
                static_cast<const GDALGeoLocTransformInfo *>(
877
0
                    pGIPTI->sSrcParams.pTransformArg);
878
879
0
            if (pGIPTI->pReproject == nullptr)
880
0
            {
881
0
                const char *pszGLSRS =
882
0
                    CSLFetchNameValue(pGLTI->papszGeolocationInfo, "SRS");
883
0
                if (pszGLSRS == nullptr)
884
0
                {
885
0
                    bIsGeographicCoordsDeg = true;
886
0
                }
887
0
                else
888
0
                {
889
0
                    OGRSpatialReference oSRS;
890
0
                    if (oSRS.SetFromUserInput(pszGLSRS) == OGRERR_NONE &&
891
0
                        oSRS.IsGeographic() &&
892
0
                        std::fabs(oSRS.GetAngularUnits() -
893
0
                                  CPLAtof(SRS_UA_DEGREE_CONV)) < 1e-9)
894
0
                    {
895
0
                        bIsGeographicCoordsDeg = true;
896
0
                    }
897
0
                }
898
0
            }
899
900
0
            for (const auto &xy :
901
0
                 {std::pair<double, double>(pGLTI->dfMinX, pGLTI->dfYAtMinX),
902
0
                  std::pair<double, double>(pGLTI->dfXAtMinY, pGLTI->dfMinY),
903
0
                  std::pair<double, double>(pGLTI->dfMaxX, pGLTI->dfYAtMaxX),
904
0
                  std::pair<double, double>(pGLTI->dfXAtMaxY, pGLTI->dfMaxY)})
905
0
            {
906
0
                double x = xy.first;
907
0
                double y = xy.second;
908
0
                if (pGLTI->bSwapXY)
909
0
                {
910
0
                    std::swap(x, y);
911
0
                }
912
0
                double xOut = std::numeric_limits<double>::quiet_NaN();
913
0
                double yOut = std::numeric_limits<double>::quiet_NaN();
914
0
                if (pGIPTI->pReproject == nullptr ||
915
0
                    pGIPTI->pReproject(pGIPTI->pReprojectArg, false, 1, &x, &y,
916
0
                                       nullptr, nullptr))
917
0
                {
918
0
                    xOut = x;
919
0
                    yOut = y;
920
0
                }
921
0
                dfMinXOut = std::min(dfMinXOut, xOut);
922
0
                dfMinYOut = std::min(dfMinYOut, yOut);
923
0
                dfMaxXOut = std::max(dfMaxXOut, xOut);
924
0
                dfMaxYOut = std::max(dfMaxYOut, yOut);
925
0
            }
926
0
        }
927
0
        else if (pGIPTI->sSrcParams.pTransformer == nullptr &&
928
0
                 pGIPTI->sDstParams.pTransformer == nullptr &&
929
0
                 pGIPTI->pReproject == GDALReprojectionTransform &&
930
0
                 pGIPTI->sDstParams.adfGeoTransform[0] == 0 &&
931
0
                 pGIPTI->sDstParams.adfGeoTransform[1] == 1 &&
932
0
                 pGIPTI->sDstParams.adfGeoTransform[2] == 0 &&
933
0
                 pGIPTI->sDstParams.adfGeoTransform[3] == 0 &&
934
0
                 pGIPTI->sDstParams.adfGeoTransform[4] == 0 &&
935
0
                 pGIPTI->sDstParams.adfGeoTransform[5] == 1)
936
0
        {
937
            /* ------------------------------------------------------------- */
938
            /* Special case for warping using source geotransform and        */
939
            /* reprojection to deal with the poles.                          */
940
            /* ------------------------------------------------------------- */
941
0
            const GDALReprojectionTransformInfo *psRTI =
942
0
                static_cast<const GDALReprojectionTransformInfo *>(
943
0
                    pGIPTI->pReprojectArg);
944
0
            const OGRSpatialReference *poSourceCRS =
945
0
                psRTI->poForwardTransform->GetSourceCS();
946
0
            const OGRSpatialReference *poTargetCRS =
947
0
                psRTI->poForwardTransform->GetTargetCS();
948
0
            if (poTargetCRS != nullptr &&
949
0
                psRTI->poReverseTransform != nullptr &&
950
0
                poTargetCRS->IsGeographic() &&
951
0
                fabs(poTargetCRS->GetAngularUnits() -
952
0
                     CPLAtof(SRS_UA_DEGREE_CONV)) < 1e-9 &&
953
0
                (!poSourceCRS || !poSourceCRS->IsGeographic()))
954
0
            {
955
0
                bIsGeographicCoordsDeg = true;
956
957
0
                std::unique_ptr<CPLConfigOptionSetter> poSetter;
958
0
                if (pGIPTI->bCheckWithInvertPROJ)
959
0
                {
960
                    // CHECK_WITH_INVERT_PROJ=YES prevent reliable
961
                    // transformation of poles.
962
0
                    poSetter = std::make_unique<CPLConfigOptionSetter>(
963
0
                        "CHECK_WITH_INVERT_PROJ", "NO", false);
964
0
                    GDALRefreshGenImgProjTransformer(pTransformArg);
965
                    // GDALRefreshGenImgProjTransformer() has invalidated psRTI
966
0
                    psRTI = static_cast<const GDALReprojectionTransformInfo *>(
967
0
                        pGIPTI->pReprojectArg);
968
0
                }
969
970
0
                for (const auto &sign : iSignArray)
971
0
                {
972
0
                    double X = 0.0;
973
0
                    const double Yinit = 90.0 * sign;
974
0
                    double Y = Yinit;
975
0
                    if (psRTI->poReverseTransform->Transform(1, &X, &Y))
976
0
                    {
977
0
                        const auto invGT =
978
0
                            pGIPTI->sSrcParams.adfInvGeoTransform;
979
0
                        const double x = invGT[0] + X * invGT[1] + Y * invGT[2];
980
0
                        const double y = invGT[3] + X * invGT[4] + Y * invGT[5];
981
0
                        constexpr double EPSILON = 1e-5;
982
0
                        if (x >= -EPSILON && x <= nInXSize + EPSILON &&
983
0
                            y >= -EPSILON && y <= nInYSize + EPSILON)
984
0
                        {
985
0
                            if (psRTI->poForwardTransform->Transform(1, &X,
986
0
                                                                     &Y) &&
987
0
                                fabs(Y - Yinit) <= 1e-6)
988
0
                            {
989
0
                                bool bMinXMaxXSet = false;
990
0
                                if (poSourceCRS)
991
0
                                {
992
0
                                    const char *pszProjection =
993
0
                                        poSourceCRS->GetAttrValue("PROJECTION");
994
0
                                    if (pszProjection &&
995
0
                                        EQUAL(pszProjection,
996
0
                                              SRS_PT_ORTHOGRAPHIC))
997
0
                                    {
998
0
                                        const double dfLon0 =
999
0
                                            poSourceCRS->GetNormProjParm(
1000
0
                                                SRS_PP_CENTRAL_MERIDIAN, 0.0);
1001
0
                                        dfMinXOut = dfLon0 - 90;
1002
0
                                        dfMaxXOut = dfLon0 + 90;
1003
0
                                        bMinXMaxXSet = true;
1004
0
                                    }
1005
0
                                }
1006
0
                                if (!bMinXMaxXSet)
1007
0
                                {
1008
0
                                    dfMinXOut = -180;
1009
0
                                    dfMaxXOut = 180;
1010
0
                                }
1011
0
                                if (sign < 0)
1012
0
                                    dfMinYOut = Yinit;
1013
0
                                else
1014
0
                                    dfMaxYOut = Yinit;
1015
0
                            }
1016
0
                        }
1017
0
                    }
1018
0
                }
1019
1020
0
                if (poSetter)
1021
0
                {
1022
0
                    poSetter.reset();
1023
0
                    GDALRefreshGenImgProjTransformer(pTransformArg);
1024
0
                    pGIPTI = static_cast<const GDALGenImgProjTransformInfo *>(
1025
0
                        pTransformArg);
1026
0
                    psRTI = static_cast<const GDALReprojectionTransformInfo *>(
1027
0
                        pGIPTI->pReprojectArg);
1028
0
                    poSourceCRS = psRTI->poForwardTransform->GetSourceCS();
1029
0
                    poTargetCRS = psRTI->poForwardTransform->GetTargetCS();
1030
0
                }
1031
0
            }
1032
1033
            // Use TransformBounds() to handle more particular cases
1034
0
            if (poSourceCRS != nullptr && poTargetCRS != nullptr &&
1035
0
                pGIPTI->sSrcParams.adfGeoTransform[1] != 0 &&
1036
0
                pGIPTI->sSrcParams.adfGeoTransform[2] == 0 &&
1037
0
                pGIPTI->sSrcParams.adfGeoTransform[4] == 0 &&
1038
0
                pGIPTI->sSrcParams.adfGeoTransform[5] != 0)
1039
0
            {
1040
0
                const double dfULX = pGIPTI->sSrcParams.adfGeoTransform[0];
1041
0
                const double dfULY = pGIPTI->sSrcParams.adfGeoTransform[3];
1042
0
                const double dfLRX =
1043
0
                    dfULX + pGIPTI->sSrcParams.adfGeoTransform[1] * nInXSize;
1044
0
                const double dfLRY =
1045
0
                    dfULY + pGIPTI->sSrcParams.adfGeoTransform[5] * nInYSize;
1046
0
                const double dfMinSrcX = std::min(dfULX, dfLRX);
1047
0
                const double dfMinSrcY = std::min(dfULY, dfLRY);
1048
0
                const double dfMaxSrcX = std::max(dfULX, dfLRX);
1049
0
                const double dfMaxSrcY = std::max(dfULY, dfLRY);
1050
0
                double dfTmpMinXOut = std::numeric_limits<double>::max();
1051
0
                double dfTmpMinYOut = std::numeric_limits<double>::max();
1052
0
                double dfTmpMaxXOut = std::numeric_limits<double>::min();
1053
0
                double dfTmpMaxYOut = std::numeric_limits<double>::min();
1054
0
                if (psRTI->poForwardTransform->TransformBounds(
1055
0
                        dfMinSrcX, dfMinSrcY, dfMaxSrcX, dfMaxSrcY,
1056
0
                        &dfTmpMinXOut, &dfTmpMinYOut, &dfTmpMaxXOut,
1057
0
                        &dfTmpMaxYOut,
1058
0
                        2))  // minimum number of points as we already have a
1059
                             // logic above to sample
1060
0
                {
1061
0
                    dfMinXOut = std::min(dfMinXOut, dfTmpMinXOut);
1062
0
                    dfMinYOut = std::min(dfMinYOut, dfTmpMinYOut);
1063
0
                    dfMaxXOut = std::max(dfMaxXOut, dfTmpMaxXOut);
1064
0
                    dfMaxYOut = std::max(dfMaxYOut, dfTmpMaxYOut);
1065
0
                }
1066
0
            }
1067
0
        }
1068
0
    }
1069
1070
    /* -------------------------------------------------------------------- */
1071
    /*      Compute the distance in "georeferenced" units from the top      */
1072
    /*      corner of the transformed input image to the bottom left        */
1073
    /*      corner of the transformed input.  Use this distance to          */
1074
    /*      compute an approximate pixel size in the output                 */
1075
    /*      georeferenced coordinates.                                      */
1076
    /* -------------------------------------------------------------------- */
1077
0
    double dfDiagonalDist = 0.0;
1078
0
    double dfDeltaX = 0.0;
1079
0
    double dfDeltaY = 0.0;
1080
1081
0
    if (pabSuccess[0] && pabSuccess[nSamplePoints - 1])
1082
0
    {
1083
0
        dfDeltaX = padfX[nSamplePoints - 1] - padfX[0];
1084
0
        dfDeltaY = padfY[nSamplePoints - 1] - padfY[0];
1085
        // In some cases this can result in 0 values. See #5980
1086
        // Fallback to safer method in that case.
1087
0
    }
1088
0
    if (dfDeltaX == 0.0 || dfDeltaY == 0.0)
1089
0
    {
1090
0
        dfDeltaX = dfMaxXOut - dfMinXOut;
1091
0
        dfDeltaY = dfMaxYOut - dfMinYOut;
1092
0
    }
1093
1094
0
    dfDiagonalDist = sqrt(dfDeltaX * dfDeltaX + dfDeltaY * dfDeltaY);
1095
1096
    /* -------------------------------------------------------------------- */
1097
    /*      Compute a pixel size from this.                                 */
1098
    /* -------------------------------------------------------------------- */
1099
0
    double dfPixelSize =
1100
0
        dfDiagonalDist / sqrt(static_cast<double>(nInXSize) * nInXSize +
1101
0
                              static_cast<double>(nInYSize) * nInYSize);
1102
1103
0
    double dfPixels = (dfMaxXOut - dfMinXOut) / dfPixelSize;
1104
0
    double dfLines = (dfMaxYOut - dfMinYOut) / dfPixelSize;
1105
1106
0
    const int knIntMaxMinusOne = std::numeric_limits<int>::max() - 1;
1107
0
    if (dfPixels > knIntMaxMinusOne && dfLines <= dfPixels)
1108
0
    {
1109
0
        dfPixels = knIntMaxMinusOne;
1110
0
        dfPixelSize = (dfMaxXOut - dfMinXOut) / dfPixels;
1111
0
        dfLines = (dfMaxYOut - dfMinYOut) / dfPixelSize;
1112
0
    }
1113
0
    else if (dfLines > knIntMaxMinusOne)
1114
0
    {
1115
0
        dfLines = knIntMaxMinusOne;
1116
0
        dfPixelSize = (dfMaxYOut - dfMinYOut) / dfLines;
1117
0
        dfPixels = (dfMaxXOut - dfMinXOut) / dfPixelSize;
1118
0
    }
1119
1120
0
    if (dfPixels > knIntMaxMinusOne || dfLines > knIntMaxMinusOne)
1121
0
    {
1122
0
        CPLError(CE_Failure, CPLE_AppDefined,
1123
0
                 "Computed dimensions are too big : %.0f x %.0f",
1124
0
                 dfPixels + 0.5, dfLines + 0.5);
1125
1126
0
        CPLFree(padfX);
1127
0
        CPLFree(padfXRevert);
1128
0
        CPLFree(pabSuccess);
1129
1130
0
        return CE_Failure;
1131
0
    }
1132
1133
0
    if ((nOptions & GDAL_SWO_ROUND_UP_SIZE) != 0)
1134
0
    {
1135
0
        constexpr double EPS = 1e-5;
1136
0
        *pnPixels = static_cast<int>(std::ceil(dfPixels - EPS));
1137
0
        *pnLines = static_cast<int>(std::ceil(dfLines - EPS));
1138
0
    }
1139
0
    else
1140
0
    {
1141
0
        *pnPixels = static_cast<int>(dfPixels + 0.5);
1142
0
        *pnLines = static_cast<int>(dfLines + 0.5);
1143
0
    }
1144
1145
0
    double dfPixelSizeX = dfPixelSize;
1146
0
    double dfPixelSizeY = dfPixelSize;
1147
1148
0
    const double adfRatioArray[] = {0.000, 0.001, 0.010, 0.100, 1.000};
1149
1150
    /* -------------------------------------------------------------------- */
1151
    /*      Check that the right border is not completely out of source     */
1152
    /*      image. If so, adjust the x pixel size a bit in the hope it will */
1153
    /*      fit.                                                            */
1154
    /* -------------------------------------------------------------------- */
1155
0
    for (const auto &dfRatio : adfRatioArray)
1156
0
    {
1157
0
        const double dfTryPixelSizeX =
1158
0
            dfPixelSizeX - dfPixelSizeX * dfRatio / *pnPixels;
1159
0
        double adfExtent[4] = {dfMinXOut, dfMaxYOut - (*pnLines) * dfPixelSizeY,
1160
0
                               dfMinXOut + (*pnPixels) * dfTryPixelSizeX,
1161
0
                               dfMaxYOut};
1162
0
        if (!GDALSuggestedWarpOutput2_MustAdjustForRightBorder(
1163
0
                pfnTransformer, pTransformArg, adfExtent, *pnPixels, *pnLines,
1164
0
                dfTryPixelSizeX, dfPixelSizeY))
1165
0
        {
1166
0
            dfPixelSizeX = dfTryPixelSizeX;
1167
0
            break;
1168
0
        }
1169
0
    }
1170
1171
    /* -------------------------------------------------------------------- */
1172
    /*      Check that the bottom border is not completely out of source    */
1173
    /*      image. If so, adjust the y pixel size a bit in the hope it will */
1174
    /*      fit.                                                            */
1175
    /* -------------------------------------------------------------------- */
1176
0
    for (const auto &dfRatio : adfRatioArray)
1177
0
    {
1178
0
        const double dfTryPixelSizeY =
1179
0
            dfPixelSizeY - dfPixelSizeY * dfRatio / *pnLines;
1180
0
        double adfExtent[4] = {
1181
0
            dfMinXOut, dfMaxYOut - (*pnLines) * dfTryPixelSizeY,
1182
0
            dfMinXOut + (*pnPixels) * dfPixelSizeX, dfMaxYOut};
1183
0
        if (!GDALSuggestedWarpOutput2_MustAdjustForBottomBorder(
1184
0
                pfnTransformer, pTransformArg, adfExtent, *pnPixels, *pnLines,
1185
0
                dfPixelSizeX, dfTryPixelSizeY))
1186
0
        {
1187
0
            dfPixelSizeY = dfTryPixelSizeY;
1188
0
            break;
1189
0
        }
1190
0
    }
1191
1192
    /* -------------------------------------------------------------------- */
1193
    /*      Recompute some bounds so that all return values are consistent  */
1194
    /* -------------------------------------------------------------------- */
1195
0
    double dfMaxXOutNew = dfMinXOut + (*pnPixels) * dfPixelSizeX;
1196
0
    if (bIsGeographicCoordsDeg &&
1197
0
        ((dfMaxXOut <= 180 && dfMaxXOutNew > 180) || dfMaxXOut == 180))
1198
0
    {
1199
0
        dfMaxXOut = 180;
1200
0
        dfPixelSizeX = (dfMaxXOut - dfMinXOut) / *pnPixels;
1201
0
    }
1202
0
    else
1203
0
    {
1204
0
        dfMaxXOut = dfMaxXOutNew;
1205
0
    }
1206
1207
0
    double dfMinYOutNew = dfMaxYOut - (*pnLines) * dfPixelSizeY;
1208
0
    if (bIsGeographicCoordsDeg && dfMinYOut >= -90 && dfMinYOutNew < -90)
1209
0
    {
1210
0
        dfMinYOut = -90;
1211
0
        dfPixelSizeY = (dfMaxYOut - dfMinYOut) / *pnLines;
1212
0
    }
1213
0
    else
1214
0
    {
1215
0
        dfMinYOut = dfMinYOutNew;
1216
0
    }
1217
1218
    /* -------------------------------------------------------------------- */
1219
    /*      Return raw extents.                                             */
1220
    /* -------------------------------------------------------------------- */
1221
0
    padfExtent[0] = dfMinXOut;
1222
0
    padfExtent[1] = dfMinYOut;
1223
0
    padfExtent[2] = dfMaxXOut;
1224
0
    padfExtent[3] = dfMaxYOut;
1225
1226
    /* -------------------------------------------------------------------- */
1227
    /*      Set the output geotransform.                                    */
1228
    /* -------------------------------------------------------------------- */
1229
0
    padfGeoTransformOut[0] = dfMinXOut;
1230
0
    padfGeoTransformOut[1] = dfPixelSizeX;
1231
0
    padfGeoTransformOut[2] = 0.0;
1232
0
    padfGeoTransformOut[3] = dfMaxYOut;
1233
0
    padfGeoTransformOut[4] = 0.0;
1234
0
    padfGeoTransformOut[5] = -dfPixelSizeY;
1235
1236
0
    CPLFree(padfX);
1237
0
    CPLFree(padfXRevert);
1238
0
    CPLFree(pabSuccess);
1239
1240
0
    return CE_None;
1241
0
}
1242
1243
/************************************************************************/
1244
/*                    GetCurrentCheckWithInvertPROJ()                   */
1245
/************************************************************************/
1246
1247
static bool GetCurrentCheckWithInvertPROJ()
1248
0
{
1249
0
    return CPLTestBool(CPLGetConfigOption("CHECK_WITH_INVERT_PROJ", "NO"));
1250
0
}
1251
1252
/************************************************************************/
1253
/*               GDALCreateGenImgProjTransformerInternal()              */
1254
/************************************************************************/
1255
1256
static void *GDALCreateSimilarGenImgProjTransformer(void *hTransformArg,
1257
                                                    double dfRatioX,
1258
                                                    double dfRatioY);
1259
1260
static GDALGenImgProjTransformInfo *GDALCreateGenImgProjTransformerInternal()
1261
0
{
1262
    /* -------------------------------------------------------------------- */
1263
    /*      Initialize the transform info.                                  */
1264
    /* -------------------------------------------------------------------- */
1265
0
    GDALGenImgProjTransformInfo *psInfo =
1266
0
        static_cast<GDALGenImgProjTransformInfo *>(
1267
0
            CPLCalloc(sizeof(GDALGenImgProjTransformInfo), 1));
1268
1269
0
    memcpy(psInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE,
1270
0
           strlen(GDAL_GTI2_SIGNATURE));
1271
0
    psInfo->sTI.pszClassName = GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME;
1272
0
    psInfo->sTI.pfnTransform = GDALGenImgProjTransform;
1273
0
    psInfo->sTI.pfnCleanup = GDALDestroyGenImgProjTransformer;
1274
0
    psInfo->sTI.pfnSerialize = GDALSerializeGenImgProjTransformer;
1275
0
    psInfo->sTI.pfnCreateSimilar = GDALCreateSimilarGenImgProjTransformer;
1276
1277
0
    psInfo->bCheckWithInvertPROJ = GetCurrentCheckWithInvertPROJ();
1278
0
    psInfo->bHasCustomTransformationPipeline = false;
1279
1280
0
    return psInfo;
1281
0
}
1282
1283
/************************************************************************/
1284
/*                GDALCreateSimilarGenImgProjTransformer()              */
1285
/************************************************************************/
1286
1287
static void *GDALCreateSimilarGenImgProjTransformer(void *hTransformArg,
1288
                                                    double dfRatioX,
1289
                                                    double dfRatioY)
1290
0
{
1291
0
    VALIDATE_POINTER1(hTransformArg, "GDALCreateSimilarGenImgProjTransformer",
1292
0
                      nullptr);
1293
1294
0
    GDALGenImgProjTransformInfo *psInfo =
1295
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
1296
1297
0
    GDALGenImgProjTransformInfo *psClonedInfo =
1298
0
        GDALCreateGenImgProjTransformerInternal();
1299
1300
0
    memcpy(psClonedInfo, psInfo, sizeof(GDALGenImgProjTransformInfo));
1301
1302
0
    psClonedInfo->bCheckWithInvertPROJ = GetCurrentCheckWithInvertPROJ();
1303
1304
0
    if (psClonedInfo->sSrcParams.pTransformArg)
1305
0
        psClonedInfo->sSrcParams.pTransformArg = GDALCreateSimilarTransformer(
1306
0
            psInfo->sSrcParams.pTransformArg, dfRatioX, dfRatioY);
1307
0
    else if (dfRatioX != 1.0 || dfRatioY != 1.0)
1308
0
    {
1309
0
        if (psClonedInfo->sSrcParams.adfGeoTransform[2] == 0.0 &&
1310
0
            psClonedInfo->sSrcParams.adfGeoTransform[4] == 0.0)
1311
0
        {
1312
0
            psClonedInfo->sSrcParams.adfGeoTransform[1] *= dfRatioX;
1313
0
            psClonedInfo->sSrcParams.adfGeoTransform[5] *= dfRatioY;
1314
0
        }
1315
0
        else
1316
0
        {
1317
            // If the x and y ratios are not equal, then we cannot really
1318
            // compute a geotransform.
1319
0
            psClonedInfo->sSrcParams.adfGeoTransform[1] *= dfRatioX;
1320
0
            psClonedInfo->sSrcParams.adfGeoTransform[2] *= dfRatioX;
1321
0
            psClonedInfo->sSrcParams.adfGeoTransform[4] *= dfRatioX;
1322
0
            psClonedInfo->sSrcParams.adfGeoTransform[5] *= dfRatioX;
1323
0
        }
1324
0
        if (!GDALInvGeoTransform(psClonedInfo->sSrcParams.adfGeoTransform,
1325
0
                                 psClonedInfo->sSrcParams.adfInvGeoTransform))
1326
0
        {
1327
0
            CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform");
1328
0
            GDALDestroyGenImgProjTransformer(psClonedInfo);
1329
0
            return nullptr;
1330
0
        }
1331
0
    }
1332
1333
0
    if (psClonedInfo->pReprojectArg)
1334
0
        psClonedInfo->pReprojectArg =
1335
0
            GDALCloneTransformer(psInfo->pReprojectArg);
1336
1337
0
    if (psClonedInfo->sDstParams.pTransformArg)
1338
0
        psClonedInfo->sDstParams.pTransformArg =
1339
0
            GDALCloneTransformer(psInfo->sDstParams.pTransformArg);
1340
1341
0
    return psClonedInfo;
1342
0
}
1343
1344
/************************************************************************/
1345
/*                  GDALCreateGenImgProjTransformer()                   */
1346
/************************************************************************/
1347
1348
/**
1349
 * Create image to image transformer.
1350
 *
1351
 * This function creates a transformation object that maps from pixel/line
1352
 * coordinates on one image to pixel/line coordinates on another image.  The
1353
 * images may potentially be georeferenced in different coordinate systems,
1354
 * and may used GCPs to map between their pixel/line coordinates and
1355
 * georeferenced coordinates (as opposed to the default assumption that their
1356
 * geotransform should be used).
1357
 *
1358
 * This transformer potentially performs three concatenated transformations.
1359
 *
1360
 * The first stage is from source image pixel/line coordinates to source
1361
 * image georeferenced coordinates, and may be done using the geotransform,
1362
 * or if not defined using a polynomial model derived from GCPs.  If GCPs
1363
 * are used this stage is accomplished using GDALGCPTransform().
1364
 *
1365
 * The second stage is to change projections from the source coordinate system
1366
 * to the destination coordinate system, assuming they differ.  This is
1367
 * accomplished internally using GDALReprojectionTransform().
1368
 *
1369
 * The third stage is converting from destination image georeferenced
1370
 * coordinates to destination image coordinates.  This is done using the
1371
 * destination image geotransform, or if not available, using a polynomial
1372
 * model derived from GCPs. If GCPs are used this stage is accomplished using
1373
 * GDALGCPTransform().  This stage is skipped if hDstDS is NULL when the
1374
 * transformation is created.
1375
 *
1376
 * @param hSrcDS source dataset, or NULL.
1377
 * @param pszSrcWKT the coordinate system for the source dataset.  If NULL,
1378
 * it will be read from the dataset itself.
1379
 * @param hDstDS destination dataset (or NULL).
1380
 * @param pszDstWKT the coordinate system for the destination dataset.  If
1381
 * NULL, and hDstDS not NULL, it will be read from the destination dataset.
1382
 * @param bGCPUseOK TRUE if GCPs should be used if the geotransform is not
1383
 * available on the source dataset (not destination).
1384
 * @param dfGCPErrorThreshold ignored/deprecated.
1385
 * @param nOrder the maximum order to use for GCP derived polynomials if
1386
 * possible.  Use 0 to autoselect, or -1 for thin plate splines.
1387
 *
1388
 * @return handle suitable for use GDALGenImgProjTransform(), and to be
1389
 * deallocated with GDALDestroyGenImgProjTransformer().
1390
 */
1391
1392
void *GDALCreateGenImgProjTransformer(GDALDatasetH hSrcDS,
1393
                                      const char *pszSrcWKT,
1394
                                      GDALDatasetH hDstDS,
1395
                                      const char *pszDstWKT, int bGCPUseOK,
1396
                                      CPL_UNUSED double dfGCPErrorThreshold,
1397
                                      int nOrder)
1398
0
{
1399
0
    char **papszOptions = nullptr;
1400
1401
0
    if (pszSrcWKT != nullptr)
1402
0
        papszOptions = CSLSetNameValue(papszOptions, "SRC_SRS", pszSrcWKT);
1403
0
    if (pszDstWKT != nullptr)
1404
0
        papszOptions = CSLSetNameValue(papszOptions, "DST_SRS", pszDstWKT);
1405
0
    if (!bGCPUseOK)
1406
0
        papszOptions = CSLSetNameValue(papszOptions, "GCPS_OK", "FALSE");
1407
0
    if (nOrder != 0)
1408
0
        papszOptions = CSLSetNameValue(papszOptions, "MAX_GCP_ORDER",
1409
0
                                       CPLString().Printf("%d", nOrder));
1410
1411
0
    void *pRet = GDALCreateGenImgProjTransformer2(hSrcDS, hDstDS, papszOptions);
1412
0
    CSLDestroy(papszOptions);
1413
1414
0
    return pRet;
1415
0
}
1416
1417
/************************************************************************/
1418
/*                          InsertCenterLong()                          */
1419
/*                                                                      */
1420
/*      Insert a CENTER_LONG Extension entry on a GEOGCS to indicate    */
1421
/*      the center longitude of the dataset for wrapping purposes.      */
1422
/************************************************************************/
1423
1424
static void InsertCenterLong(GDALDatasetH hDS, const OGRSpatialReference *poSRS,
1425
                             const OGRSpatialReference *poDstSRS,
1426
                             const char *pszTargetExtent,
1427
                             CPLStringList &aosOptions)
1428
1429
0
{
1430
0
    if (!poSRS->IsGeographic() || std::fabs(poSRS->GetAngularUnits() -
1431
0
                                            CPLAtof(SRS_UA_DEGREE_CONV)) > 1e-9)
1432
0
    {
1433
0
        return;
1434
0
    }
1435
1436
0
    if (poSRS->GetExtension(nullptr, "CENTER_LONG"))
1437
0
        return;
1438
1439
    /* -------------------------------------------------------------------- */
1440
    /*      For now we only do this if we have a geotransform since         */
1441
    /*      other forms require a bunch of extra work.                      */
1442
    /* -------------------------------------------------------------------- */
1443
0
    double adfGeoTransform[6] = {};
1444
1445
0
    if (GDALGetGeoTransform(hDS, adfGeoTransform) != CE_None)
1446
0
        return;
1447
1448
    /* -------------------------------------------------------------------- */
1449
    /*      Compute min/max longitude based on testing the four corners.    */
1450
    /* -------------------------------------------------------------------- */
1451
0
    const int nXSize = GDALGetRasterXSize(hDS);
1452
0
    const int nYSize = GDALGetRasterYSize(hDS);
1453
1454
0
    const double dfMinLong =
1455
0
        std::min(std::min(adfGeoTransform[0] + 0 * adfGeoTransform[1] +
1456
0
                              0 * adfGeoTransform[2],
1457
0
                          adfGeoTransform[0] + nXSize * adfGeoTransform[1] +
1458
0
                              0 * adfGeoTransform[2]),
1459
0
                 std::min(adfGeoTransform[0] + 0 * adfGeoTransform[1] +
1460
0
                              nYSize * adfGeoTransform[2],
1461
0
                          adfGeoTransform[0] + nXSize * adfGeoTransform[1] +
1462
0
                              nYSize * adfGeoTransform[2]));
1463
0
    const double dfMaxLong =
1464
0
        std::max(std::max(adfGeoTransform[0] + 0 * adfGeoTransform[1] +
1465
0
                              0 * adfGeoTransform[2],
1466
0
                          adfGeoTransform[0] + nXSize * adfGeoTransform[1] +
1467
0
                              0 * adfGeoTransform[2]),
1468
0
                 std::max(adfGeoTransform[0] + 0 * adfGeoTransform[1] +
1469
0
                              nYSize * adfGeoTransform[2],
1470
0
                          adfGeoTransform[0] + nXSize * adfGeoTransform[1] +
1471
0
                              nYSize * adfGeoTransform[2]));
1472
1473
    // If the raster covers more than 360 degree, give up,
1474
    // except is the target SRS is geographic and crossing the antimeridian
1475
0
    if (dfMaxLong - dfMinLong > 360.0)
1476
0
    {
1477
0
        const CPLStringList aosTE(CSLTokenizeString2(pszTargetExtent, ",", 0));
1478
0
        if (aosTE.size() == 4 && poDstSRS->IsGeographic() &&
1479
0
            std::fabs(poDstSRS->GetAngularUnits() -
1480
0
                      CPLAtof(SRS_UA_DEGREE_CONV)) <= 1e-9 &&
1481
0
            ((CPLAtof(aosTE[0]) >= -179 && CPLAtof(aosTE[0]) < 180 &&
1482
0
              CPLAtof(aosTE[2]) > 180) ||
1483
0
             (CPLAtof(aosTE[0]) < -180 && CPLAtof(aosTE[2]) > -180 &&
1484
0
              CPLAtof(aosTE[2]) <= 179)))
1485
0
        {
1486
            // insert CENTER_LONG
1487
0
        }
1488
0
        else
1489
0
        {
1490
0
            return;
1491
0
        }
1492
0
    }
1493
1494
    /* -------------------------------------------------------------------- */
1495
    /*      Insert center long.                                             */
1496
    /* -------------------------------------------------------------------- */
1497
0
    const double dfCenterLong = (dfMaxLong + dfMinLong) / 2.0;
1498
0
    aosOptions.SetNameValue("CENTER_LONG", CPLSPrintf("%g", dfCenterLong));
1499
0
}
1500
1501
/************************************************************************/
1502
/*                      GDALComputeAreaOfInterest()                     */
1503
/************************************************************************/
1504
1505
bool GDALComputeAreaOfInterest(const OGRSpatialReference *poSRS,
1506
                               double adfGT[6], int nXSize, int nYSize,
1507
                               double &dfWestLongitudeDeg,
1508
                               double &dfSouthLatitudeDeg,
1509
                               double &dfEastLongitudeDeg,
1510
                               double &dfNorthLatitudeDeg)
1511
0
{
1512
0
    bool ret = false;
1513
1514
0
    if (!poSRS)
1515
0
        return false;
1516
1517
0
    OGRSpatialReference oSrcSRSHoriz(*poSRS);
1518
0
    if (oSrcSRSHoriz.IsCompound())
1519
0
    {
1520
0
        oSrcSRSHoriz.StripVertical();
1521
0
    }
1522
1523
0
    OGRSpatialReference *poGeog = oSrcSRSHoriz.CloneGeogCS();
1524
0
    if (poGeog)
1525
0
    {
1526
0
        poGeog->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
1527
0
        poGeog->SetAngularUnits(SRS_UA_DEGREE, CPLAtof(SRS_UA_DEGREE_CONV));
1528
1529
0
        auto poCT = OGRCreateCoordinateTransformation(&oSrcSRSHoriz, poGeog);
1530
0
        if (poCT)
1531
0
        {
1532
0
            poCT->SetEmitErrors(false);
1533
1534
0
            double x[4], y[4];
1535
0
            x[0] = adfGT[0];
1536
0
            y[0] = adfGT[3];
1537
0
            x[1] = adfGT[0] + nXSize * adfGT[1];
1538
0
            y[1] = adfGT[3];
1539
0
            x[2] = adfGT[0];
1540
0
            y[2] = adfGT[3] + nYSize * adfGT[5];
1541
0
            x[3] = x[1];
1542
0
            y[3] = y[2];
1543
0
            int validity[4] = {false, false, false, false};
1544
0
            poCT->Transform(4, x, y, nullptr, validity);
1545
0
            dfWestLongitudeDeg = std::numeric_limits<double>::max();
1546
0
            dfSouthLatitudeDeg = std::numeric_limits<double>::max();
1547
0
            dfEastLongitudeDeg = -std::numeric_limits<double>::max();
1548
0
            dfNorthLatitudeDeg = -std::numeric_limits<double>::max();
1549
0
            for (int i = 0; i < 4; i++)
1550
0
            {
1551
0
                if (validity[i])
1552
0
                {
1553
0
                    ret = true;
1554
0
                    dfWestLongitudeDeg = std::min(dfWestLongitudeDeg, x[i]);
1555
0
                    dfSouthLatitudeDeg = std::min(dfSouthLatitudeDeg, y[i]);
1556
0
                    dfEastLongitudeDeg = std::max(dfEastLongitudeDeg, x[i]);
1557
0
                    dfNorthLatitudeDeg = std::max(dfNorthLatitudeDeg, y[i]);
1558
0
                }
1559
0
            }
1560
0
            if (validity[0] && validity[1] && x[0] > x[1])
1561
0
            {
1562
0
                dfWestLongitudeDeg = x[0];
1563
0
                dfEastLongitudeDeg = x[1];
1564
0
            }
1565
0
            if (ret && std::fabs(dfWestLongitudeDeg) <= 180 &&
1566
0
                std::fabs(dfEastLongitudeDeg) <= 180 &&
1567
0
                std::fabs(dfSouthLatitudeDeg) <= 90 &&
1568
0
                std::fabs(dfNorthLatitudeDeg) <= 90)
1569
0
            {
1570
0
                CPLDebug("GDAL", "Computing area of interest: %g, %g, %g, %g",
1571
0
                         dfWestLongitudeDeg, dfSouthLatitudeDeg,
1572
0
                         dfEastLongitudeDeg, dfNorthLatitudeDeg);
1573
0
            }
1574
0
            else
1575
0
            {
1576
0
                CPLDebug("GDAL", "Could not compute area of interest");
1577
0
                dfWestLongitudeDeg = 0;
1578
0
                dfSouthLatitudeDeg = 0;
1579
0
                dfEastLongitudeDeg = 0;
1580
0
                dfNorthLatitudeDeg = 0;
1581
0
            }
1582
0
            OGRCoordinateTransformation::DestroyCT(poCT);
1583
0
        }
1584
1585
0
        delete poGeog;
1586
0
    }
1587
1588
0
    return ret;
1589
0
}
1590
1591
bool GDALComputeAreaOfInterest(const OGRSpatialReference *poSRS, double dfX1,
1592
                               double dfY1, double dfX2, double dfY2,
1593
                               double &dfWestLongitudeDeg,
1594
                               double &dfSouthLatitudeDeg,
1595
                               double &dfEastLongitudeDeg,
1596
                               double &dfNorthLatitudeDeg)
1597
0
{
1598
0
    bool ret = false;
1599
1600
0
    if (!poSRS)
1601
0
        return false;
1602
1603
0
    OGRSpatialReference oSrcSRSHoriz(*poSRS);
1604
0
    if (oSrcSRSHoriz.IsCompound())
1605
0
    {
1606
0
        oSrcSRSHoriz.StripVertical();
1607
0
    }
1608
1609
0
    OGRSpatialReference *poGeog = oSrcSRSHoriz.CloneGeogCS();
1610
0
    if (poGeog)
1611
0
    {
1612
0
        poGeog->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
1613
1614
0
        auto poCT = OGRCreateCoordinateTransformation(&oSrcSRSHoriz, poGeog);
1615
0
        if (poCT)
1616
0
        {
1617
0
            double x[4], y[4];
1618
0
            x[0] = dfX1;
1619
0
            y[0] = dfY1;
1620
0
            x[1] = dfX2;
1621
0
            y[1] = dfY1;
1622
0
            x[2] = dfX1;
1623
0
            y[2] = dfY2;
1624
0
            x[3] = dfX2;
1625
0
            y[3] = dfY2;
1626
0
            int validity[4] = {false, false, false, false};
1627
0
            poCT->Transform(4, x, y, nullptr, validity);
1628
0
            dfWestLongitudeDeg = std::numeric_limits<double>::max();
1629
0
            dfSouthLatitudeDeg = std::numeric_limits<double>::max();
1630
0
            dfEastLongitudeDeg = -std::numeric_limits<double>::max();
1631
0
            dfNorthLatitudeDeg = -std::numeric_limits<double>::max();
1632
0
            for (int i = 0; i < 4; i++)
1633
0
            {
1634
0
                if (validity[i])
1635
0
                {
1636
0
                    ret = true;
1637
0
                    dfWestLongitudeDeg = std::min(dfWestLongitudeDeg, x[i]);
1638
0
                    dfSouthLatitudeDeg = std::min(dfSouthLatitudeDeg, y[i]);
1639
0
                    dfEastLongitudeDeg = std::max(dfEastLongitudeDeg, x[i]);
1640
0
                    dfNorthLatitudeDeg = std::max(dfNorthLatitudeDeg, y[i]);
1641
0
                }
1642
0
            }
1643
0
            if (validity[0] && validity[1] && (dfX1 - dfX2) * (x[0] - x[1]) < 0)
1644
0
            {
1645
0
                dfWestLongitudeDeg = x[0];
1646
0
                dfEastLongitudeDeg = x[1];
1647
0
            }
1648
0
            if (ret)
1649
0
            {
1650
0
                CPLDebug("GDAL", "Computing area of interest: %g, %g, %g, %g",
1651
0
                         dfWestLongitudeDeg, dfSouthLatitudeDeg,
1652
0
                         dfEastLongitudeDeg, dfNorthLatitudeDeg);
1653
0
            }
1654
0
            else
1655
0
            {
1656
0
                CPLDebug("GDAL", "Could not compute area of interest");
1657
0
                dfWestLongitudeDeg = 0;
1658
0
                dfSouthLatitudeDeg = 0;
1659
0
                dfEastLongitudeDeg = 0;
1660
0
                dfNorthLatitudeDeg = 0;
1661
0
            }
1662
0
            delete poCT;
1663
0
        }
1664
1665
0
        delete poGeog;
1666
0
    }
1667
1668
0
    return ret;
1669
0
}
1670
1671
/************************************************************************/
1672
/*                    GDALGCPAntimeridianUnwrap()                       */
1673
/************************************************************************/
1674
1675
/* Deal with discontinuties of dfGCPX longitudes around the anti-meridian.
1676
 * Cf https://github.com/OSGeo/gdal/issues/8371
1677
 */
1678
static void GDALGCPAntimeridianUnwrap(int nGCPCount, GDAL_GCP *pasGCPList,
1679
                                      const OGRSpatialReference &oSRS,
1680
                                      CSLConstList papszOptions)
1681
0
{
1682
0
    const char *pszGCPAntimeridianUnwrap =
1683
0
        CSLFetchNameValueDef(papszOptions, "GCP_ANTIMERIDIAN_UNWRAP", "AUTO");
1684
0
    const bool bForced = EQUAL(pszGCPAntimeridianUnwrap, "YES") ||
1685
0
                         EQUAL(pszGCPAntimeridianUnwrap, "ON") ||
1686
0
                         EQUAL(pszGCPAntimeridianUnwrap, "TRUE") ||
1687
0
                         EQUAL(pszGCPAntimeridianUnwrap, "1");
1688
0
    if (bForced || (!oSRS.IsEmpty() && oSRS.IsGeographic() &&
1689
0
                    fabs(oSRS.GetAngularUnits(nullptr) -
1690
0
                         CPLAtof(SRS_UA_DEGREE_CONV)) < 1e-8 &&
1691
0
                    EQUAL(pszGCPAntimeridianUnwrap, "AUTO")))
1692
0
    {
1693
0
        if (!bForced)
1694
0
        {
1695
            // Proceed to unwrapping only if the longitudes are within
1696
            // [-180, -170] or [170, 180]
1697
0
            for (int i = 0; i < nGCPCount; ++i)
1698
0
            {
1699
0
                const double dfLongAbs = fabs(pasGCPList[i].dfGCPX);
1700
0
                if (dfLongAbs > 180 || dfLongAbs < 170)
1701
0
                {
1702
0
                    return;
1703
0
                }
1704
0
            }
1705
0
        }
1706
1707
0
        bool bDone = false;
1708
0
        for (int i = 0; i < nGCPCount; ++i)
1709
0
        {
1710
0
            if (pasGCPList[i].dfGCPX < 0)
1711
0
            {
1712
0
                if (!bDone)
1713
0
                {
1714
0
                    bDone = true;
1715
0
                    CPLDebug("WARP", "GCP longitude unwrapping");
1716
0
                }
1717
0
                pasGCPList[i].dfGCPX += 360;
1718
0
            }
1719
0
        }
1720
0
    }
1721
0
}
1722
1723
/************************************************************************/
1724
/*              GDALGetGenImgProjTranformerOptionList()                 */
1725
/************************************************************************/
1726
1727
/** Return a XML string describing options accepted by
1728
 * GDALCreateGenImgProjTransformer2().
1729
 *
1730
 * @since 3.11
1731
 */
1732
const char *GDALGetGenImgProjTranformerOptionList(void)
1733
0
{
1734
0
    return "<OptionList>"
1735
0
           "<Option name='SRC_SRS' type='string' description='WKT SRS, or any "
1736
0
           "string recognized by OGRSpatialReference::SetFromUserInput(), to "
1737
0
           "be used as an override for CRS of input dataset'/>"
1738
0
           "<Option name='DST_SRS' type='string' description='WKT SRS, or any "
1739
0
           "string recognized by OGRSpatialReference::SetFromUserInput(), to "
1740
0
           "be used as an override for CRS of output dataset'/>"
1741
0
           "<Option name='PROMOTE_TO_3D' type='boolean' description='"
1742
0
           "Whether to promote SRC_SRS / DST_SRS to 3D.' "
1743
0
           "default='NO'/>"
1744
0
           "<Option name='COORDINATE_OPERATION' type='string' description='"
1745
0
           "Coordinate operation, as a PROJ or WKT string, used as an override "
1746
0
           "over the normally computed pipeline. The pipeline must take into "
1747
0
           "account the axis order of the source and target SRS.'/>"
1748
0
           "<Option name='ALLOW_BALLPARK' type='boolean' description='"
1749
0
           "Whether ballpark coordinate operations are allowed.' "
1750
0
           "default='YES'/>"
1751
0
           "<Option name='ONLY_BEST' type='string-select' "
1752
0
           "description='"
1753
0
           "By default (at least in the PROJ 9.x series), PROJ may use "
1754
0
           "coordinate operations that are not the \"best\" if resources "
1755
0
           "(typically grids) needed to use them are missing. It will then "
1756
0
           "fallback to other coordinate operations that have a lesser "
1757
0
           "accuracy, for example using Helmert transformations, or in the "
1758
0
           "absence of such operations, to ones with potential very rough "
1759
0
           " accuracy, using \"ballpark\" transformations (see "
1760
0
           "https://proj.org/glossary.html). "
1761
0
           "When calling this method with YES, PROJ will only consider the "
1762
0
           "\"best\" operation, and error out (at Transform() time) if they "
1763
0
           "cannot be used. This method may be used together with "
1764
0
           "ALLOW_BALLPARK=NO to only allow best operations that have a known "
1765
0
           "accuracy. Note that this method has no effect on PROJ versions "
1766
0
           "before 9.2. The default value for this option can be also set with "
1767
0
           "the PROJ_ONLY_BEST_DEFAULT environment variable, or with the "
1768
0
           "\"only_best_default\" setting of proj.ini. Setting "
1769
0
           "ONLY_BEST=YES/NO overrides such default value' default='AUTO'>"
1770
0
           "  <Value>AUTO</Value>"
1771
0
           "  <Value>YES</Value>"
1772
0
           "  <Value>NO</Value>"
1773
0
           "</Option>"
1774
0
           "<Option name='COORDINATE_EPOCH' type='float' description='"
1775
0
           "Coordinate epoch, expressed as a decimal year. Useful for "
1776
0
           "time-dependent coordinate operations.'/>"
1777
0
           "<Option name='SRC_COORDINATE_EPOCH' type='float' description='"
1778
0
           "Coordinate epoch of source CRS, expressed as a decimal year. "
1779
0
           "Useful for time-dependent coordinate operations.'/>"
1780
0
           "<Option name='DST_COORDINATE_EPOCH' type='float' description='"
1781
0
           "Coordinate epoch of target CRS, expressed as a decimal year. "
1782
0
           "Useful for time-dependent coordinate operations.'/>"
1783
0
           "<Option name='GCPS_OK' type='boolean' description='"
1784
0
           "Allow use of GCPs.' default='YES'/>"
1785
0
           "<Option name='REFINE_MINIMUM_GCPS' type='int' description='"
1786
0
           "The minimum amount of GCPs that should be available after the "
1787
0
           "refinement'/>"
1788
0
           "<Option name='REFINE_TOLERANCE' type='float' description='"
1789
0
           "The tolerance that specifies when a GCP will be eliminated.'/>"
1790
0
           "<Option name='MAX_GCP_ORDER' type='int' description='"
1791
0
           "The maximum order to use for GCP derived polynomials if possible. "
1792
0
           "The default is to autoselect based on the number of GCPs. A value "
1793
0
           "of -1 triggers use of Thin Plate Spline instead of polynomials.'/>"
1794
0
           "<Option name='GCP_ANTIMERIDIAN_UNWRAP' type='string-select' "
1795
0
           "description='"
1796
0
           "Whether to \"unwrap\" longitudes of ground control points that "
1797
0
           "span the antimeridian. For datasets with GCPs in "
1798
0
           "longitude/latitude coordinate space spanning the antimeridian, "
1799
0
           "longitudes will have a discontinuity on +/- 180 deg, and will "
1800
0
           "result in a subset of the GCPs with longitude in the [-180,-170] "
1801
0
           "range and another subset in [170, 180]. By default (AUTO), that "
1802
0
           "situation will be detected and longitudes in [-180,-170] will be "
1803
0
           "shifted to [180, 190] to get a continuous set. This option can be "
1804
0
           "set to YES to force that behavior (useful if no SRS information is "
1805
0
           "available), or to NO to disable it.' default='AUTO'>"
1806
0
           "  <Value>AUTO</Value>"
1807
0
           "  <Value>YES</Value>"
1808
0
           "  <Value>NO</Value>"
1809
0
           "</Option>"
1810
0
           "<Option name='SRC_METHOD' alias='METHOD' type='string-select' "
1811
0
           "description='"
1812
0
           "Force only one geolocation method to be considered on the source "
1813
0
           "dataset. Will be used for pixel/line to georef transformation on "
1814
0
           "the source dataset. NO_GEOTRANSFORM can be used to specify the "
1815
0
           "identity geotransform (ungeoreferenced image)'>"
1816
0
           "  <Value>GEOTRANSFORM</Value>"
1817
0
           "  <Value>GCP_POLYNOMIAL</Value>"
1818
0
           "  <Value>GCP_TPS</Value>"
1819
0
           "  <Value>GCP_HOMOGRAPHY</Value>"
1820
0
           "  <Value>GEOLOC_ARRAY</Value>"
1821
0
           "  <Value>RPC</Value>"
1822
0
           "  <Value>NO_GEOTRANSFORM</Value>"
1823
0
           "</Option>"
1824
0
           "<Option name='DST_METHOD' type='string-select' description='"
1825
0
           "Force only one geolocation method to be considered on the target "
1826
0
           "dataset. Will be used for pixel/line to georef transformation on "
1827
0
           "the targe dataset. NO_GEOTRANSFORM can be used to specify the "
1828
0
           "identity geotransform (ungeoreferenced image)'>"
1829
0
           "  <Value>GEOTRANSFORM</Value>"
1830
0
           "  <Value>GCP_POLYNOMIAL</Value>"
1831
0
           "  <Value>GCP_TPS</Value>"
1832
0
           "  <Value>GCP_HOMOGRAPHY</Value>"
1833
0
           "  <Value>GEOLOC_ARRAY</Value>"
1834
0
           "  <Value>RPC</Value>"
1835
0
           "  <Value>NO_GEOTRANSFORM</Value>"
1836
0
           "</Option>"
1837
0
           "<Option name='RPC_HEIGHT' type='float' description='"
1838
0
           "A fixed height to be used with RPC calculations. If RPC_HEIGHT and "
1839
0
           "RPC_DEM are not specified but that the RPC metadata domain contains"
1840
0
           " a HEIGHT_DEFAULT item (for example, the DIMAP driver may fill it),"
1841
0
           "this value will be used as the RPC_HEIGHT. Otherwise, if none of "
1842
0
           "RPC_HEIGHT and RPC_DEM are specified as transformer options and "
1843
0
           "if HEIGHT_DEFAULT is no available, a height of 0 will be used.'/>"
1844
0
           "<Option name='RPC_DEM' type='string' description='"
1845
0
           "Name of a GDAL dataset (a DEM file typically) used to extract "
1846
0
           "elevation offsets from. In this situation the Z passed into the "
1847
0
           "transformation function is assumed to be height above ground. "
1848
0
           "This option should be used in replacement of RPC_HEIGHT to provide "
1849
0
           "a way of defining a non uniform ground for the target scene.'/>"
1850
0
           "<Option name='RPC_HEIGHT_SCALE' type='float' description='"
1851
0
           "Factor used to multiply heights above ground. Useful when "
1852
0
           "elevation offsets of the DEM are not expressed in meters.'/>"
1853
0
           "<Option name='RPC_DEMINTERPOLATION' type='string-select' "
1854
0
           "description='DEM interpolation method' default='BILINEAR'>"
1855
0
           "  <Value>NEAR</Value>"
1856
0
           "  <Value>BILINEAR</Value>"
1857
0
           "  <Value>CUBIC</Value>"
1858
0
           "</Option>"
1859
0
           "<Option name='RPC_DEM_MISSING_VALUE' type='float' description='"
1860
0
           "Value of DEM height that must be used in case the DEM has nodata "
1861
0
           "value at the sampling point, or if its extent does not cover the "
1862
0
           "requested coordinate. When not specified, missing values will "
1863
0
           "cause a failed transform.'/>"
1864
0
           "<Option name='RPC_DEM_SRS' type='string' description='"
1865
0
           "WKT SRS, or any string recognized by "
1866
0
           "OGRSpatialReference::SetFromUserInput(), to be used as an "
1867
0
           "override for DEM SRS. Useful if DEM SRS does not have an explicit "
1868
0
           "vertical component.'/>"
1869
0
           "<Option name='RPC_DEM_APPLY_VDATUM_SHIFT' type='boolean' "
1870
0
           "description='"
1871
0
           "Whether the vertical component of a compound SRS for the DEM "
1872
0
           "should be used (when it is present). This is useful so as to "
1873
0
           "be able to transform the raw values from the DEM expressed with "
1874
0
           "respect to a geoid to the heights with respect to the WGS84 "
1875
0
           "ellipsoid. When this is enabled, the GTIFF_REPORT_COMPD_CS "
1876
0
           "configuration option will be also set temporarily so as to get "
1877
0
           "the vertical information from GeoTIFF files.' default='YES'/>"
1878
0
           "<Option name='RPC_PIXEL_ERROR_THRESHOLD' type='float' description='"
1879
0
           "Overrides the dfPixErrThreshold parameter, i.e. the error "
1880
0
           "(measured in pixels) allowed in the iterative solution of "
1881
0
           "pixel/line to lat/long computations (the other way is always "
1882
0
           "exact given the equations).'/>"
1883
0
           "<Option name='RPC_MAX_ITERATIONS' type='int' description='"
1884
0
           "Maximum number of iterations allowed in the iterative solution of "
1885
0
           "pixel/line to lat/long computations. Default value is 10 in the "
1886
0
           "absence of a DEM, or 20 if there is a DEM.'/>"
1887
0
           "<Option name='RPC_FOOTPRINT' type='string' description='"
1888
0
           "WKT or GeoJSON polygon (in long / lat coordinate space) with a "
1889
0
           "validity footprint for the RPC. Any coordinate transformation that "
1890
0
           "goes from or arrive outside this footprint will be considered "
1891
0
           "invalid. This* is useful in situations where the RPC values become "
1892
0
           "highly unstable outside of the area on which they have been "
1893
0
           "computed for, potentially leading to undesirable \"echoes\" / "
1894
0
           "false positives. This requires GDAL to be built against GEOS..'/>"
1895
0
           "<Option name='RPC_MAX_ITERATIONS' type='int' description='"
1896
0
           "Maximum number of iterations allowed in the iterative solution of "
1897
0
           "pixel/line to lat/long computations. Default value is 10 in the "
1898
0
           "absence of a DEM, or 20 if there is a DEM.'/>"
1899
0
           "<Option name='INSERT_CENTER_LONG' type='boolean' description='"
1900
0
           "May be set to FALSE to disable setting up a CENTER_LONG value on "
1901
0
           "the coordinate system to rewrap things around the center of the "
1902
0
           "image.' default='YES'/>"
1903
0
           "<Option name='SRC_APPROX_ERROR_IN_SRS_UNIT' type='float' "
1904
0
           "description='"
1905
0
           "Use an approximate transformer for the source transformer. Must be "
1906
0
           "defined together with SRC_APPROX_ERROR_IN_PIXEL to be taken into "
1907
0
           "account.'/>"
1908
0
           "<Option name='SRC_APPROX_ERROR_IN_PIXEL' type='float' "
1909
0
           "description='"
1910
0
           "Use an approximate transformer for the source transformer. Must be "
1911
0
           "defined together with SRC_APPROX_ERROR_IN_SRS_UNIT to be taken "
1912
0
           "into "
1913
0
           "account.'/>"
1914
0
           "<Option name='DST_APPROX_ERROR_IN_SRS_UNIT' type='float' "
1915
0
           "description='"
1916
0
           "Use an approximate transformer for the target transformer. Must be "
1917
0
           "defined together with DST_APPROX_ERROR_IN_PIXEL to be taken into "
1918
0
           "account.'/>"
1919
0
           "<Option name='DST_APPROX_ERROR_IN_PIXEL' type='float' "
1920
0
           "description='"
1921
0
           "Use an approximate transformer for the target transformer. Must be "
1922
0
           "defined together with DST_APPROX_ERROR_IN_SRS_UNIT to be taken "
1923
0
           "into "
1924
0
           "account.'/>"
1925
0
           "<Option name='REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT' "
1926
0
           "type='float' "
1927
0
           "description='"
1928
0
           "Use an approximate transformer for the coordinate reprojection. "
1929
0
           "Must be used together with "
1930
0
           "REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT to be taken into "
1931
0
           "account.'/>"
1932
0
           "<Option name='REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT' "
1933
0
           "type='float' "
1934
0
           "description='"
1935
0
           "Use an approximate transformer for the coordinate reprojection. "
1936
0
           "Must be used together with "
1937
0
           "REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT to be taken into "
1938
0
           "account.'/>"
1939
0
           "<Option name='AREA_OF_INTEREST' type='string' "
1940
0
           "description='"
1941
0
           "Area of interest, as "
1942
0
           "west_lon_deg,south_lat_deg,east_lon_deg,north_lat_deg, used to "
1943
0
           "compute the best coordinate operation between the source and "
1944
0
           "target SRS. If not specified, the bounding box of the source "
1945
0
           "raster will be used.'/>"
1946
0
           "<Option name='TARGET_EXTENT' type='string' "
1947
0
           "description='Target extent as minx,miny,maxx,maxy expressed in "
1948
0
           "target SRS.'/>"
1949
0
           "<Option name='GEOLOC_BACKMAP_OVERSAMPLE_FACTOR' type='float' "
1950
0
           "min='0.1' max='2' description='"
1951
0
           "Oversample factor used to derive the size of the \"backmap\" used "
1952
0
           "for geolocation array transformers.' default='1.3'/>"
1953
0
           "<Option name='GEOLOC_USE_TEMP_DATASETS' type='boolean' "
1954
0
           "description='"
1955
0
           "Whether temporary GeoTIFF datasets should be used to store the "
1956
0
           "backmap. The default is NO, that is to use in-memory arrays, "
1957
0
           "unless the number of pixels of the geolocation array is greater "
1958
0
           "than 16 megapixels.' default='NO'/>"
1959
0
           "<Option name='GEOLOC_ARRAY' alias='SRC_GEOLOC_ARRAY' type='string' "
1960
0
           "description='"
1961
0
           "Name of a GDAL dataset containing a geolocation array and "
1962
0
           "associated metadata. This is an alternative to having geolocation "
1963
0
           "information described in the GEOLOCATION metadata domain of the "
1964
0
           "source dataset. The dataset specified may have a GEOLOCATION "
1965
0
           "metadata domain containing appropriate metadata, however default "
1966
0
           "values are assigned for all omitted items. X_BAND defaults to 1 "
1967
0
           "and Y_BAND to 2, however the dataset must contain exactly 2 bands. "
1968
0
           "PIXEL_OFFSET and LINE_OFFSET default to 0. PIXEL_STEP and "
1969
0
           "LINE_STEP default to the ratio of the width/height of the source "
1970
0
           "dataset divided by the with/height of the geolocation array. "
1971
0
           "SRS defaults to the spatial reference system of the geolocation "
1972
0
           "array dataset, if set, otherwise WGS84 is used. "
1973
0
           "GEOREFERENCING_CONVENTION is selected from the main metadata "
1974
0
           "domain if it is omitted from the GEOLOCATION domain, and if not "
1975
0
           "available TOP_LEFT_CORNER is assigned as a default. "
1976
0
           "If GEOLOC_ARRAY is set SRC_METHOD defaults to GEOLOC_ARRAY.'/>"
1977
0
           "<Option name='DST_GEOLOC_ARRAY' type='string' "
1978
0
           "description='"
1979
0
           "Name of a GDAL dataset that contains at least 2 bands with the X "
1980
0
           "and Y geolocation bands. This is an alternative to having "
1981
0
           "geolocation information described in the GEOLOCATION metadata "
1982
0
           "domain of the destination dataset. See SRC_GEOLOC_ARRAY "
1983
0
           "description for details, assumptions, and defaults. If this "
1984
0
           "option is set, DST_METHOD=GEOLOC_ARRAY will be assumed if not "
1985
0
           "set.'/>"
1986
0
           "<Option name='GEOLOC_NORMALIZE_LONGITUDE_MINUS_180_PLUS_180' "
1987
0
           "type='boolean' "
1988
0
           "description='"
1989
0
           "Force geolocation longitudes into -180,180 when longitude/latitude "
1990
0
           "is the coordinate system of the geolocation arrays' default='NO'>"
1991
0
           "  <Value>YES</Value>"
1992
0
           "  <Value>NO</Value>"
1993
0
           "</Option>"
1994
0
           "<Option name='NUM_THREADS' type='string' "
1995
0
           "description='Number of threads to use'/>"
1996
0
           "</OptionList>";
1997
0
}
1998
1999
/************************************************************************/
2000
/*                  GDALCreateGenImgProjTransformer2()                  */
2001
/************************************************************************/
2002
2003
/* clang-format off */
2004
/**
2005
 * Create image to image transformer.
2006
 *
2007
 * This function creates a transformation object that maps from pixel/line
2008
 * coordinates on one image to pixel/line coordinates on another image.  The
2009
 * images may potentially be georeferenced in different coordinate systems,
2010
 * and may used GCPs to map between their pixel/line coordinates and
2011
 * georeferenced coordinates (as opposed to the default assumption that their
2012
 * geotransform should be used).
2013
 *
2014
 * This transformer potentially performs three concatenated transformations.
2015
 *
2016
 * The first stage is from source image pixel/line coordinates to source
2017
 * image georeferenced coordinates, and may be done using the geotransform,
2018
 * or if not defined using a polynomial model derived from GCPs.  If GCPs
2019
 * are used this stage is accomplished using GDALGCPTransform().
2020
 *
2021
 * The second stage is to change projections from the source coordinate system
2022
 * to the destination coordinate system, assuming they differ.  This is
2023
 * accomplished internally using GDALReprojectionTransform().
2024
 *
2025
 * The third stage is converting from destination image georeferenced
2026
 * coordinates to destination image coordinates.  This is done using the
2027
 * destination image geotransform, or if not available, using a polynomial
2028
 * model derived from GCPs. If GCPs are used this stage is accomplished using
2029
 * GDALGCPTransform().  This stage is skipped if hDstDS is NULL when the
2030
 * transformation is created.
2031
 *
2032
 * Supported Options (specified with the -to switch of gdalwarp for example):
2033
 * <ul>
2034
 * <li> SRC_SRS: WKT SRS, or any string recognized by
2035
 * OGRSpatialReference::SetFromUserInput(), to be used as an override for
2036
 * hSrcDS.</li>
2037
 * <li> DST_SRS: WKT SRS, or any string recognized by
2038
 * OGRSpatialReference::SetFromUserInput(),  to be used as an override for
2039
 * hDstDS.
2040
 * </li>
2041
 * <li>PROMOTE_TO_3D=YES/NO: whether to promote SRC_SRS / DST_SRS to 3D.
2042
 * Default is NO</li>
2043
 * <li> COORDINATE_OPERATION: (GDAL &gt;= 3.0) Coordinate operation, as
2044
 * a PROJ or WKT string, used as an override over the normally computed
2045
 * pipeline. The pipeline must take into account the axis order of the source
2046
 * and target SRS.
2047
 * </li>
2048
 * <li> ALLOW_BALLPARK=YES/NO: (GDAL &gt;= 3.11) Whether ballpark coordinate
2049
 * operations are allowed. Defaults to YES.</li>
2050
 * <li> ONLY_BEST=YES/NO/AUTO: (GDAL &gt;= 3.11) By default (at least in the
2051
 * PROJ 9.x series), PROJ may use coordinate
2052
 * operations that are not the "best" if resources (typically grids) needed
2053
 * to use them are missing. It will then fallback to other coordinate operations
2054
 * that have a lesser accuracy, for example using Helmert transformations,
2055
 * or in the absence of such operations, to ones with potential very rought
2056
 * accuracy, using "ballpark" transformations
2057
 * (see https://proj.org/glossary.html).
2058
 * When calling this method with YES, PROJ will only consider the
2059
 * "best" operation, and error out (at Transform() time) if they cannot be
2060
 * used.
2061
 * This method may be used together with ALLOW_BALLPARK=NO to
2062
 * only allow best operations that have a known accuracy.
2063
 * Note that this method has no effect on PROJ versions before 9.2.
2064
 * The default value for this option can be also set with the
2065
 * PROJ_ONLY_BEST_DEFAULT environment variable, or with the "only_best_default"
2066
 * setting of proj.ini. Calling SetOnlyBest() overrides such default value.</li>
2067
 * <li> COORDINATE_EPOCH: (GDAL &gt;= 3.0) Coordinate epoch,
2068
 * expressed as a decimal year. Useful for time-dependent coordinate operations.
2069
 * </li>
2070
 * <li> SRC_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch of source CRS,
2071
 * expressed as a decimal year. Useful for time-dependent coordinate operations.
2072
 * </li>
2073
 * <li> DST_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch of target CRS,
2074
 * expressed as a decimal year. Useful for time-dependent coordinate operations.
2075
 * </li>
2076
 * <li> GCPS_OK: If false, GCPs will not be used, default is TRUE.
2077
 * </li>
2078
 * <li> REFINE_MINIMUM_GCPS: The minimum amount of GCPs that should be available
2079
 * after the refinement.
2080
 * </li>
2081
 * <li> REFINE_TOLERANCE: The tolerance that specifies when a GCP will be
2082
 * eliminated.
2083
 * </li>
2084
 * <li> MAX_GCP_ORDER: the maximum order to use for GCP derived polynomials if
2085
 * possible.  The default is to autoselect based on the number of GCPs.
2086
 * A value of -1 triggers use of Thin Plate Spline instead of polynomials.
2087
 * </li>
2088
 * <li>GCP_ANTIMERIDIAN_UNWRAP=AUTO/YES/NO. (GDAL &gt;= 3.8) Whether to
2089
 * "unwrap" longitudes of ground control points that span the antimeridian.
2090
 * For datasets with GCPs in longitude/latitude coordinate space spanning the
2091
 * antimeridian, longitudes will have a discontinuity on +/- 180 deg, and
2092
 * will result in a subset of the GCPs with longitude in the [-180,-170] range
2093
 * and another subset in [170, 180]. By default (AUTO), that situation will be
2094
 * detected and longitudes in [-180,-170] will be shifted to [180, 190] to get
2095
 * a continuous set. This option can be set to YES to force that behavior
2096
 * (useful if no SRS information is available), or to NO to disable it.
2097
 * </li>
2098
 * <li> SRC_METHOD: may have a value which is one of GEOTRANSFORM, GCP_HOMOGRAPHY,
2099
 * GCP_POLYNOMIAL, GCP_TPS, GEOLOC_ARRAY, RPC to force only one geolocation
2100
 * method to be considered on the source dataset. Will be used for pixel/line
2101
 * to georef transformation on the source dataset. NO_GEOTRANSFORM can be
2102
 * used to specify the identity geotransform (ungeoreferenced image)
2103
 * </li>
2104
 * <li> DST_METHOD: may have a value which is one of GEOTRANSFORM,
2105
 * GCP_POLYNOMIAL, GCP_HOMOGRAPHY, GCP_TPS, GEOLOC_ARRAY (added in 3.5), RPC to
2106
 * force only one
2107
 * geolocation method to be considered on the target dataset.  Will be used for
2108
 * pixel/line to georef transformation on the destination dataset.
2109
 * NO_GEOTRANSFORM can be used to specify the identity geotransform
2110
 * (ungeoreferenced image)
2111
 * </li>
2112
 * <li> RPC_HEIGHT: A fixed height to be used with RPC
2113
 * calculations. If RPC_HEIGHT and RPC_DEM are not specified but that the RPC
2114
 * metadata domain contains a HEIGHT_DEFAULT item (for example, the DIMAP driver
2115
 * may fill it), this value will be used as the RPC_HEIGHT. Otherwise, if none
2116
 * of RPC_HEIGHT and RPC_DEM are specified as transformer
2117
 * options and if HEIGHT_DEFAULT is no available, a height of 0 will be used.
2118
 * </li>
2119
 * <li> RPC_DEM: The name of a DEM file to be used with RPC
2120
 * calculations. See GDALCreateRPCTransformerV2() for more details.
2121
 * </li>
2122
 * <li> Other RPC related options. See GDALCreateRPCTransformerV2()
2123
 * </li>
2124
 * <li>
2125
 * INSERT_CENTER_LONG: May be set to FALSE to disable setting up a CENTER_LONG
2126
 * value on the coordinate system to rewrap things around the center of the
2127
 * image.
2128
 * </li>
2129
 * <li> SRC_APPROX_ERROR_IN_SRS_UNIT=err_threshold_in_SRS_units. (GDAL
2130
 * &gt;= 2.2) Use an approximate transformer for the source transformer. Must be
2131
 * defined together with SRC_APPROX_ERROR_IN_PIXEL to be taken into account.
2132
 * </li>
2133
 * <li> SRC_APPROX_ERROR_IN_PIXEL=err_threshold_in_pixel. (GDAL &gt;= 2.2) Use
2134
 * an approximate transformer for the source transformer.. Must be defined
2135
 * together with SRC_APPROX_ERROR_IN_SRS_UNIT to be taken into account.
2136
 * </li>
2137
 * <li>
2138
 * DST_APPROX_ERROR_IN_SRS_UNIT=err_threshold_in_SRS_units. (GDAL &gt;= 2.2) Use
2139
 * an approximate transformer for the destination transformer. Must be defined
2140
 * together with DST_APPROX_ERROR_IN_PIXEL to be taken into account.
2141
 * </li>
2142
 * <li>
2143
 * DST_APPROX_ERROR_IN_PIXEL=err_threshold_in_pixel. (GDAL &gt;= 2.2) Use an
2144
 * approximate transformer for the destination transformer. Must be defined
2145
 * together with DST_APPROX_ERROR_IN_SRS_UNIT to be taken into account.
2146
 * </li>
2147
 * <li>
2148
 * REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT=err_threshold_in_src_SRS_units.
2149
 * (GDAL &gt;= 2.2) Use an approximate transformer for the coordinate
2150
 * reprojection. Must be used together with
2151
 * REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT to be taken into account.
2152
 * </li>
2153
 * <li>
2154
 * REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT=err_threshold_in_dst_SRS_units.
2155
 * (GDAL &gt;= 2.2) Use an approximate transformer for the coordinate
2156
 * reprojection. Must be used together with
2157
 * REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT to be taken into account.
2158
 * </li>
2159
 * <li>
2160
 * AREA_OF_INTEREST=west_lon_deg,south_lat_deg,east_lon_deg,north_lat_deg. (GDAL
2161
 * &gt;= 3.0) Area of interest, used to compute the best coordinate operation
2162
 * between the source and target SRS. If not specified, the bounding box of the
2163
 * source raster will be used.
2164
 * </li>
2165
 * <li> GEOLOC_BACKMAP_OVERSAMPLE_FACTOR=[0.1,2]. (GDAL &gt;= 3.5) Oversample
2166
 * factor used to derive the size of the "backmap" used for geolocation array
2167
 * transformers. Default value is 1.3.
2168
 * </li>
2169
 * <li> GEOLOC_USE_TEMP_DATASETS=YES/NO.
2170
 * (GDAL &gt;= 3.5) Whether temporary GeoTIFF datasets should be used to store
2171
 * the backmap. The default is NO, that is to use in-memory arrays, unless the
2172
 * number of pixels of the geolocation array is greater than 16 megapixels.
2173
 * </li>
2174
 * <li>
2175
 * GEOLOC_ARRAY/SRC_GEOLOC_ARRAY=filename. (GDAL &gt;= 3.5.2) Name of a GDAL
2176
 * dataset containing a geolocation array and associated metadata. This is an
2177
 * alternative to having geolocation information described in the GEOLOCATION
2178
 * metadata domain of the source dataset. The dataset specified may have a
2179
 * GEOLOCATION metadata domain containing appropriate metadata, however default
2180
 * values are assigned for all omitted items. X_BAND defaults to 1 and Y_BAND to
2181
 * 2, however the dataset must contain exactly 2 bands. PIXEL_OFFSET and
2182
 * LINE_OFFSET default to 0. PIXEL_STEP and LINE_STEP default to the ratio of
2183
 * the width/height of the source dataset divided by the with/height of the
2184
 * geolocation array. SRS defaults to the geolocation array dataset's spatial
2185
 * reference system if set, otherwise WGS84 is used.
2186
 * GEOREFERENCING_CONVENTION is selected from the main metadata domain if it
2187
 * is omitted from the GEOLOCATION domain, and if not available
2188
 * TOP_LEFT_CORNER is assigned as a default.
2189
 * If GEOLOC_ARRAY is set SRC_METHOD
2190
 * defaults to GEOLOC_ARRAY.
2191
 * </li>
2192
 * <li>DST_GEOLOC_ARRAY=filename. (GDAL &gt;= 3.5.2) Name of a
2193
 * GDAL dataset that contains at least 2 bands with the X and Y geolocation
2194
 * bands. This is an alternative to having geolocation information described in
2195
 * the GEOLOCATION metadata domain of the destination dataset. See
2196
 * SRC_GEOLOC_ARRAY description for details, assumptions, and defaults. If this
2197
 * option is set, DST_METHOD=GEOLOC_ARRAY will be assumed if not set.
2198
 * </li>
2199
 * <li>GEOLOC_NORMALIZE_LONGITUDE_MINUS_180_PLUS_180=YES/NO. (GDAL &gt;= 3.12.0)
2200
 * Whether to force geolocation longitudes into -180,180 when longitude/latitude is
2201
 * the coordinate system of the geolocation arrays. The default is to enable this mode
2202
 * when the values in the geolocation array are in the -180,180, otherwise NO.
2203
 * </li>
2204
 * </ul>
2205
 *
2206
 * The use case for the *_APPROX_ERROR_* options is when defining an approximate
2207
 * transformer on top of the GenImgProjTransformer globally is not practical.
2208
 * Such a use case is when the source dataset has RPC with a RPC DEM. In such
2209
 * case we don't want to use the approximate transformer on the RPC
2210
 * transformation, as the RPC DEM generally involves non-linearities that the
2211
 * approximate transformer will not detect. In such case, we must a
2212
 * non-approximated GenImgProjTransformer, but it might be worthwhile to use
2213
 * approximate sub- transformers, for example on coordinate reprojection. For
2214
 * example if warping from a source dataset with RPC to a destination dataset
2215
 * with a UTM projection, since the inverse UTM transformation is rather costly.
2216
 * In which case, one can use the REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT and
2217
 * REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT options.
2218
 *
2219
 * The list of supported options can also be programmatically obtained with
2220
 * GDALGetGenImgProjTranformerOptionList().
2221
 *
2222
 * @param hSrcDS source dataset, or NULL.
2223
 * @param hDstDS destination dataset (or NULL).
2224
 * @param papszOptions NULL-terminated list of string options (or NULL).
2225
 *
2226
 * @return handle suitable for use GDALGenImgProjTransform(), and to be
2227
 * deallocated with GDALDestroyGenImgProjTransformer() or NULL on failure.
2228
 */
2229
/* clang-format on */
2230
2231
void *GDALCreateGenImgProjTransformer2(GDALDatasetH hSrcDS, GDALDatasetH hDstDS,
2232
                                       CSLConstList papszOptions)
2233
2234
0
{
2235
0
    GDALValidateOptions(GDALGetGenImgProjTranformerOptionList(), papszOptions,
2236
0
                        "option", "transformer options");
2237
2238
0
    double dfWestLongitudeDeg = 0.0;
2239
0
    double dfSouthLatitudeDeg = 0.0;
2240
0
    double dfEastLongitudeDeg = 0.0;
2241
0
    double dfNorthLatitudeDeg = 0.0;
2242
0
    bool bHasAreaOfInterest = false;
2243
0
    if (const char *pszAreaOfInterest =
2244
0
            CSLFetchNameValue(papszOptions, "AREA_OF_INTEREST"))
2245
0
    {
2246
0
        const CPLStringList aosTokens(
2247
0
            CSLTokenizeString2(pszAreaOfInterest, ", ", 0));
2248
0
        if (aosTokens.size() == 4)
2249
0
        {
2250
0
            dfWestLongitudeDeg = CPLAtof(aosTokens[0]);
2251
0
            dfSouthLatitudeDeg = CPLAtof(aosTokens[1]);
2252
0
            dfEastLongitudeDeg = CPLAtof(aosTokens[2]);
2253
0
            dfNorthLatitudeDeg = CPLAtof(aosTokens[3]);
2254
0
            bHasAreaOfInterest = true;
2255
0
        }
2256
0
    }
2257
2258
0
    const char *pszCO = CSLFetchNameValue(papszOptions, "COORDINATE_OPERATION");
2259
2260
0
    const auto SetAxisMapping =
2261
0
        [papszOptions](OGRSpatialReference &oSRS, const char *pszPrefix)
2262
0
    {
2263
0
        const char *pszMapping = CSLFetchNameValue(
2264
0
            papszOptions, std::string(pszPrefix)
2265
0
                              .append("_DATA_AXIS_TO_SRS_AXIS_MAPPING")
2266
0
                              .c_str());
2267
0
        if (pszMapping)
2268
0
        {
2269
0
            CPLStringList aosTokens(CSLTokenizeString2(pszMapping, ",", 0));
2270
0
            std::vector<int> anMapping;
2271
0
            for (int i = 0; i < aosTokens.size(); ++i)
2272
0
                anMapping.push_back(atoi(aosTokens[i]));
2273
0
            oSRS.SetDataAxisToSRSAxisMapping(anMapping);
2274
0
        }
2275
0
        else
2276
0
        {
2277
0
            const char *pszStrategy = CSLFetchNameValueDef(
2278
0
                papszOptions,
2279
0
                std::string(pszPrefix).append("_AXIS_MAPPING_STRATEGY").c_str(),
2280
0
                "TRADITIONAL_GIS_ORDER");
2281
0
            if (EQUAL(pszStrategy, "TRADITIONAL_GIS_ORDER"))
2282
0
                oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
2283
0
            else if (EQUAL(pszStrategy, "AUTHORITY_COMPLIANT"))
2284
0
                oSRS.SetAxisMappingStrategy(OAMS_AUTHORITY_COMPLIANT);
2285
0
            else
2286
0
            {
2287
0
                CPLError(CE_Warning, CPLE_AppDefined,
2288
0
                         "Unrecognized value '%s' for %s", pszStrategy,
2289
0
                         std::string(pszPrefix)
2290
0
                             .append("_AXIS_MAPPING_STRATEGY")
2291
0
                             .c_str());
2292
0
                return false;
2293
0
            }
2294
0
        }
2295
0
        return true;
2296
0
    };
2297
2298
    /* -------------------------------------------------------------------- */
2299
    /*      Initialize the transform info.                                  */
2300
    /* -------------------------------------------------------------------- */
2301
0
    GDALGenImgProjTransformInfo *psInfo =
2302
0
        GDALCreateGenImgProjTransformerInternal();
2303
2304
0
    const auto DealWithForwardOrInverse =
2305
0
        [bHasAreaOfInterest, &dfWestLongitudeDeg, &dfSouthLatitudeDeg,
2306
0
         &dfEastLongitudeDeg, &dfNorthLatitudeDeg, pszCO, papszOptions,
2307
0
         &SetAxisMapping](GDALGenImgProjTransformPart &part, GDALDatasetH hDS,
2308
0
                          const char *pszPrefix, OGRSpatialReference &oSRS,
2309
0
                          bool &bCanUseGeoTransform)
2310
0
    {
2311
0
        const int nOrder =
2312
0
            atoi(CSLFetchNameValueDef(papszOptions, "MAX_GCP_ORDER", "0"));
2313
2314
0
        const bool bGCPUseOK =
2315
0
            CPLTestBool(CSLFetchNameValueDef(papszOptions, "GCPS_OK", "YES"));
2316
0
        const int nMinimumGcps = atoi(
2317
0
            CSLFetchNameValueDef(papszOptions, "REFINE_MINIMUM_GCPS", "-1"));
2318
2319
0
        const char *pszRefineTolerance =
2320
0
            CSLFetchNameValue(papszOptions, "REFINE_TOLERANCE");
2321
0
        const bool bRefine = pszRefineTolerance != nullptr;
2322
0
        const double dfTolerance =
2323
0
            pszRefineTolerance ? CPLAtof(pszRefineTolerance) : 0.0;
2324
2325
0
        const std::string osSRSOptionName =
2326
0
            std::string(pszPrefix).append("_SRS");
2327
0
        const char *pszSRS =
2328
0
            CSLFetchNameValue(papszOptions, osSRSOptionName.c_str());
2329
0
        if (pszSRS)
2330
0
        {
2331
0
            if (pszSRS[0] != '\0' &&
2332
0
                oSRS.SetFromUserInput(pszSRS) != OGRERR_NONE)
2333
0
            {
2334
0
                CPLError(CE_Failure, CPLE_AppDefined,
2335
0
                         "Failed to import coordinate system `%s'.", pszSRS);
2336
0
                return false;
2337
0
            }
2338
0
            if (!SetAxisMapping(oSRS, osSRSOptionName.c_str()))
2339
0
                return false;
2340
0
        }
2341
2342
0
        CSLConstList papszMD = nullptr;
2343
0
        GDALRPCInfoV2 sRPCInfo;
2344
2345
0
        bCanUseGeoTransform = false;
2346
2347
0
        const char *pszMethod = CSLFetchNameValue(
2348
0
            papszOptions, std::string(pszPrefix).append("_METHOD").c_str());
2349
0
        if (!pszMethod && EQUAL(pszPrefix, "SRC"))
2350
0
            pszMethod = CSLFetchNameValue(papszOptions, "METHOD");
2351
2352
0
        const char *pszGeolocArray = CSLFetchNameValue(
2353
0
            papszOptions,
2354
0
            std::string(pszPrefix).append("_GEOLOC_ARRAY").c_str());
2355
0
        if (!pszGeolocArray && EQUAL(pszPrefix, "SRC"))
2356
0
            pszGeolocArray = CSLFetchNameValue(papszOptions, "GEOLOC_ARRAY");
2357
0
        if (!pszMethod && pszGeolocArray != nullptr)
2358
0
            pszMethod = "GEOLOC_ARRAY";
2359
2360
        /* -------------------------------------------------------------------- */
2361
        /*      Get forward and inverse geotransform for the source image.      */
2362
        /* -------------------------------------------------------------------- */
2363
0
        if (hDS == nullptr ||
2364
0
            (pszMethod != nullptr && EQUAL(pszMethod, "NO_GEOTRANSFORM")))
2365
0
        {
2366
0
            part.adfGeoTransform[0] = 0.0;
2367
0
            part.adfGeoTransform[1] = 1.0;
2368
0
            part.adfGeoTransform[2] = 0.0;
2369
0
            part.adfGeoTransform[3] = 0.0;
2370
0
            part.adfGeoTransform[4] = 0.0;
2371
0
            part.adfGeoTransform[5] = 1.0;
2372
0
            memcpy(part.adfInvGeoTransform, part.adfGeoTransform,
2373
0
                   sizeof(double) * 6);
2374
0
        }
2375
0
        else if ((pszMethod == nullptr || EQUAL(pszMethod, "GEOTRANSFORM")) &&
2376
0
                 GDALGetGeoTransform(hDS, part.adfGeoTransform) == CE_None)
2377
0
        {
2378
0
            if (!GDALInvGeoTransform(part.adfGeoTransform,
2379
0
                                     part.adfInvGeoTransform))
2380
0
            {
2381
0
                CPLError(CE_Failure, CPLE_AppDefined,
2382
0
                         "Cannot invert geotransform");
2383
0
                return false;
2384
0
            }
2385
0
            if (pszSRS == nullptr)
2386
0
            {
2387
0
                auto hSRS = GDALGetSpatialRef(hDS);
2388
0
                if (hSRS)
2389
0
                    oSRS = *(OGRSpatialReference::FromHandle(hSRS));
2390
0
            }
2391
0
            if (EQUAL(pszPrefix, "SRC"))
2392
0
            {
2393
0
                if (!bHasAreaOfInterest && pszCO == nullptr && !oSRS.IsEmpty())
2394
0
                {
2395
0
                    GDALComputeAreaOfInterest(
2396
0
                        &oSRS, part.adfGeoTransform, GDALGetRasterXSize(hDS),
2397
0
                        GDALGetRasterYSize(hDS), dfWestLongitudeDeg,
2398
0
                        dfSouthLatitudeDeg, dfEastLongitudeDeg,
2399
0
                        dfNorthLatitudeDeg);
2400
0
                }
2401
0
                bCanUseGeoTransform = true;
2402
0
            }
2403
0
        }
2404
0
        else if (bGCPUseOK &&
2405
0
                 ((pszMethod == nullptr && GDALGetGCPCount(hDS) >= 4 &&
2406
0
                   GDALGetGCPCount(hDS) < 6) ||
2407
0
                  (pszMethod != nullptr &&
2408
0
                   EQUAL(pszMethod, "GCP_HOMOGRAPHY"))) &&
2409
0
                 GDALGetGCPCount(hDS) > 0)
2410
0
        {
2411
0
            if (pszSRS == nullptr)
2412
0
            {
2413
0
                auto hSRS = GDALGetGCPSpatialRef(hDS);
2414
0
                if (hSRS)
2415
0
                    oSRS = *(OGRSpatialReference::FromHandle(hSRS));
2416
0
            }
2417
2418
0
            const auto nGCPCount = GDALGetGCPCount(hDS);
2419
0
            auto pasGCPList = GDALDuplicateGCPs(nGCPCount, GDALGetGCPs(hDS));
2420
0
            GDALGCPAntimeridianUnwrap(nGCPCount, pasGCPList, oSRS,
2421
0
                                      papszOptions);
2422
2423
0
            part.pTransformArg =
2424
0
                GDALCreateHomographyTransformerFromGCPs(nGCPCount, pasGCPList);
2425
2426
0
            GDALDeinitGCPs(nGCPCount, pasGCPList);
2427
0
            CPLFree(pasGCPList);
2428
2429
0
            if (part.pTransformArg == nullptr)
2430
0
            {
2431
0
                return false;
2432
0
            }
2433
0
            part.pTransformer = GDALHomographyTransform;
2434
0
        }
2435
0
        else if (bGCPUseOK &&
2436
0
                 (pszMethod == nullptr || EQUAL(pszMethod, "GCP_POLYNOMIAL")) &&
2437
0
                 GDALGetGCPCount(hDS) > 0 && nOrder >= 0)
2438
0
        {
2439
0
            if (pszSRS == nullptr)
2440
0
            {
2441
0
                auto hSRS = GDALGetGCPSpatialRef(hDS);
2442
0
                if (hSRS)
2443
0
                    oSRS = *(OGRSpatialReference::FromHandle(hSRS));
2444
0
            }
2445
2446
0
            const auto nGCPCount = GDALGetGCPCount(hDS);
2447
0
            auto pasGCPList = GDALDuplicateGCPs(nGCPCount, GDALGetGCPs(hDS));
2448
0
            GDALGCPAntimeridianUnwrap(nGCPCount, pasGCPList, oSRS,
2449
0
                                      papszOptions);
2450
2451
0
            if (bRefine)
2452
0
            {
2453
0
                part.pTransformArg = GDALCreateGCPRefineTransformer(
2454
0
                    nGCPCount, pasGCPList, nOrder, FALSE, dfTolerance,
2455
0
                    nMinimumGcps);
2456
0
            }
2457
0
            else
2458
0
            {
2459
0
                part.pTransformArg = GDALCreateGCPTransformer(
2460
0
                    nGCPCount, pasGCPList, nOrder, FALSE);
2461
0
            }
2462
2463
0
            GDALDeinitGCPs(nGCPCount, pasGCPList);
2464
0
            CPLFree(pasGCPList);
2465
2466
0
            if (part.pTransformArg == nullptr)
2467
0
            {
2468
0
                return false;
2469
0
            }
2470
0
            part.pTransformer = GDALGCPTransform;
2471
0
        }
2472
2473
0
        else if (bGCPUseOK && GDALGetGCPCount(hDS) > 0 && nOrder <= 0 &&
2474
0
                 (pszMethod == nullptr || EQUAL(pszMethod, "GCP_TPS")))
2475
0
        {
2476
0
            if (pszSRS == nullptr)
2477
0
            {
2478
0
                auto hSRS = GDALGetGCPSpatialRef(hDS);
2479
0
                if (hSRS)
2480
0
                    oSRS = *(OGRSpatialReference::FromHandle(hSRS));
2481
0
            }
2482
2483
0
            const auto nGCPCount = GDALGetGCPCount(hDS);
2484
0
            auto pasGCPList = GDALDuplicateGCPs(nGCPCount, GDALGetGCPs(hDS));
2485
0
            GDALGCPAntimeridianUnwrap(nGCPCount, pasGCPList, oSRS,
2486
0
                                      papszOptions);
2487
2488
0
            part.pTransformArg = GDALCreateTPSTransformerInt(
2489
0
                nGCPCount, pasGCPList, FALSE, papszOptions);
2490
2491
0
            GDALDeinitGCPs(nGCPCount, pasGCPList);
2492
0
            CPLFree(pasGCPList);
2493
2494
0
            if (part.pTransformArg == nullptr)
2495
0
            {
2496
0
                return false;
2497
0
            }
2498
0
            part.pTransformer = GDALTPSTransform;
2499
0
        }
2500
2501
0
        else if ((pszMethod == nullptr || EQUAL(pszMethod, "RPC")) &&
2502
0
                 (papszMD = GDALGetMetadata(hDS, "RPC")) != nullptr &&
2503
0
                 GDALExtractRPCInfoV2(papszMD, &sRPCInfo))
2504
0
        {
2505
0
            CPLStringList aosOptions(papszOptions);
2506
0
            if (!CSLFetchNameValue(papszOptions, "RPC_HEIGHT") &&
2507
0
                !CSLFetchNameValue(papszOptions, "RPC_DEM"))
2508
0
            {
2509
0
                if (const char *pszHEIGHT_DEFAULT =
2510
0
                        CSLFetchNameValue(papszMD, "HEIGHT_DEFAULT"))
2511
0
                {
2512
0
                    CPLDebug("GDAL",
2513
0
                             "For %s, using RPC_HEIGHT = HEIGHT_DEFAULT = %s",
2514
0
                             pszPrefix, pszHEIGHT_DEFAULT);
2515
0
                    aosOptions.SetNameValue("RPC_HEIGHT", pszHEIGHT_DEFAULT);
2516
0
                }
2517
0
            }
2518
0
            part.pTransformArg = GDALCreateRPCTransformerV2(&sRPCInfo, FALSE, 0,
2519
0
                                                            aosOptions.List());
2520
0
            if (part.pTransformArg == nullptr)
2521
0
            {
2522
0
                return false;
2523
0
            }
2524
0
            part.pTransformer = GDALRPCTransform;
2525
0
            if (pszSRS == nullptr)
2526
0
            {
2527
0
                oSRS.SetFromUserInput(SRS_WKT_WGS84_LAT_LONG);
2528
0
                oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
2529
0
            }
2530
0
        }
2531
2532
0
        else if ((pszMethod == nullptr || EQUAL(pszMethod, "GEOLOC_ARRAY")) &&
2533
0
                 ((papszMD = GDALGetMetadata(hDS, "GEOLOCATION")) != nullptr ||
2534
0
                  pszGeolocArray != nullptr))
2535
0
        {
2536
0
            CPLStringList aosGeolocMD;  // keep in this scope
2537
0
            if (pszGeolocArray != nullptr)
2538
0
            {
2539
0
                if (papszMD != nullptr)
2540
0
                {
2541
0
                    CPLError(
2542
0
                        CE_Warning, CPLE_AppDefined,
2543
0
                        "Both GEOLOCATION metadata domain on the source "
2544
0
                        "dataset "
2545
0
                        "and [%s_]GEOLOC_ARRAY transformer option are set. "
2546
0
                        "Only using the later.",
2547
0
                        pszPrefix);
2548
0
                }
2549
0
                aosGeolocMD = GDALCreateGeolocationMetadata(
2550
0
                    hDS, pszGeolocArray,
2551
0
                    /* bIsSource= */ EQUAL(pszPrefix, "SRC"));
2552
0
                if (aosGeolocMD.empty())
2553
0
                {
2554
0
                    return false;
2555
0
                }
2556
0
                papszMD = aosGeolocMD.List();
2557
0
            }
2558
2559
0
            part.pTransformArg = GDALCreateGeoLocTransformerEx(
2560
0
                hDS, papszMD, FALSE, nullptr, papszOptions);
2561
0
            if (part.pTransformArg == nullptr)
2562
0
            {
2563
0
                return false;
2564
0
            }
2565
0
            part.pTransformer = GDALGeoLocTransform;
2566
0
            if (pszSRS == nullptr)
2567
0
            {
2568
0
                pszSRS = CSLFetchNameValue(papszMD, "SRS");
2569
0
                if (pszSRS)
2570
0
                {
2571
0
                    oSRS.SetFromUserInput(pszSRS);
2572
0
                    oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
2573
0
                }
2574
0
            }
2575
0
        }
2576
2577
0
        else if (pszMethod != nullptr && EQUAL(pszPrefix, "SRC"))
2578
0
        {
2579
0
            CPLError(CE_Failure, CPLE_AppDefined,
2580
0
                     "Unable to compute a %s based transformation between "
2581
0
                     "pixel/line and georeferenced coordinates for %s.",
2582
0
                     pszMethod, GDALGetDescription(hDS));
2583
2584
0
            return false;
2585
0
        }
2586
2587
0
        else
2588
0
        {
2589
0
            CPLError(CE_Failure, CPLE_AppDefined,
2590
0
                     "Unable to compute a transformation between pixel/line "
2591
0
                     "and georeferenced coordinates for %s. "
2592
0
                     "There is no affine transformation and no GCPs. "
2593
0
                     "Specify transformation option %s_METHOD=NO_GEOTRANSFORM "
2594
0
                     "to bypass this check.",
2595
0
                     GDALGetDescription(hDS), pszPrefix);
2596
2597
0
            return false;
2598
0
        }
2599
2600
        /* ---------------------------------------------------------------- */
2601
        /*      Handle optional source approximation transformer.           */
2602
        /* ---------------------------------------------------------------- */
2603
0
        if (part.pTransformer)
2604
0
        {
2605
0
            const char *pszApproxErrorFwd = CSLFetchNameValue(
2606
0
                papszOptions, std::string(pszPrefix)
2607
0
                                  .append("_APPROX_ERROR_IN_SRS_UNIT")
2608
0
                                  .c_str());
2609
0
            const char *pszApproxErrorReverse = CSLFetchNameValue(
2610
0
                papszOptions, std::string(pszPrefix)
2611
0
                                  .append("_APPROX_ERROR_IN_PIXEL")
2612
0
                                  .c_str());
2613
0
            if (pszApproxErrorFwd && pszApproxErrorReverse)
2614
0
            {
2615
0
                void *pArg = GDALCreateApproxTransformer2(
2616
0
                    part.pTransformer, part.pTransformArg,
2617
0
                    CPLAtof(pszApproxErrorFwd), CPLAtof(pszApproxErrorReverse));
2618
0
                if (pArg == nullptr)
2619
0
                {
2620
0
                    return false;
2621
0
                }
2622
0
                part.pTransformArg = pArg;
2623
0
                part.pTransformer = GDALApproxTransform;
2624
0
                GDALApproxTransformerOwnsSubtransformer(part.pTransformArg,
2625
0
                                                        TRUE);
2626
0
            }
2627
0
        }
2628
2629
0
        return true;
2630
0
    };
2631
2632
    /* -------------------------------------------------------------------- */
2633
    /*      Get forward and inverse geotransform for the source image.      */
2634
    /* -------------------------------------------------------------------- */
2635
0
    bool bCanUseSrcGeoTransform = false;
2636
0
    OGRSpatialReference oSrcSRS;
2637
0
    if (!DealWithForwardOrInverse(psInfo->sSrcParams, hSrcDS, "SRC", oSrcSRS,
2638
0
                                  bCanUseSrcGeoTransform))
2639
0
    {
2640
0
        GDALDestroyGenImgProjTransformer(psInfo);
2641
0
        return nullptr;
2642
0
    }
2643
2644
    /* -------------------------------------------------------------------- */
2645
    /*      Get forward and inverse geotransform for destination image.     */
2646
    /*      If we have no destination use a unit transform.                 */
2647
    /* -------------------------------------------------------------------- */
2648
0
    bool bIgnored = false;
2649
0
    OGRSpatialReference oDstSRS;
2650
0
    if (!DealWithForwardOrInverse(psInfo->sDstParams, hDstDS, "DST", oDstSRS,
2651
0
                                  bIgnored))
2652
0
    {
2653
0
        GDALDestroyGenImgProjTransformer(psInfo);
2654
0
        return nullptr;
2655
0
    }
2656
2657
    /* -------------------------------------------------------------------- */
2658
    /*      Setup reprojection.                                             */
2659
    /* -------------------------------------------------------------------- */
2660
2661
0
    if (CPLFetchBool(papszOptions, "@STRIP_VERT_CS", false))
2662
0
    {
2663
0
        if (oSrcSRS.IsCompound())
2664
0
        {
2665
0
            oSrcSRS.StripVertical();
2666
0
        }
2667
0
        if (oDstSRS.IsCompound())
2668
0
        {
2669
0
            oDstSRS.StripVertical();
2670
0
        }
2671
0
    }
2672
2673
0
    const bool bMayInsertCenterLong =
2674
0
        (bCanUseSrcGeoTransform && !oSrcSRS.IsEmpty() && hSrcDS &&
2675
0
         CPLFetchBool(papszOptions, "INSERT_CENTER_LONG", true));
2676
0
    const char *pszSrcCoordEpoch =
2677
0
        CSLFetchNameValue(papszOptions, "SRC_COORDINATE_EPOCH");
2678
0
    const char *pszDstCoordEpoch =
2679
0
        CSLFetchNameValue(papszOptions, "DST_COORDINATE_EPOCH");
2680
0
    if ((!oSrcSRS.IsEmpty() && !oDstSRS.IsEmpty() &&
2681
0
         (pszSrcCoordEpoch || pszDstCoordEpoch || !oSrcSRS.IsSame(&oDstSRS) ||
2682
0
          (oSrcSRS.IsGeographic() && bMayInsertCenterLong))) ||
2683
0
        pszCO)
2684
0
    {
2685
0
        CPLStringList aosOptions;
2686
2687
0
        if (bMayInsertCenterLong)
2688
0
        {
2689
0
            InsertCenterLong(hSrcDS, &oSrcSRS, &oDstSRS,
2690
0
                             CSLFetchNameValue(papszOptions, "TARGET_EXTENT"),
2691
0
                             aosOptions);
2692
0
        }
2693
2694
0
        if (CPLFetchBool(papszOptions, "PROMOTE_TO_3D", false))
2695
0
        {
2696
0
            oSrcSRS.PromoteTo3D(nullptr);
2697
0
            oDstSRS.PromoteTo3D(nullptr);
2698
0
        }
2699
2700
0
        if (!(dfWestLongitudeDeg == 0.0 && dfSouthLatitudeDeg == 0.0 &&
2701
0
              dfEastLongitudeDeg == 0.0 && dfNorthLatitudeDeg == 0.0))
2702
0
        {
2703
0
            aosOptions.SetNameValue(
2704
0
                "AREA_OF_INTEREST",
2705
0
                CPLSPrintf("%.16g,%.16g,%.16g,%.16g", dfWestLongitudeDeg,
2706
0
                           dfSouthLatitudeDeg, dfEastLongitudeDeg,
2707
0
                           dfNorthLatitudeDeg));
2708
0
        }
2709
0
        if (pszCO)
2710
0
        {
2711
0
            aosOptions.SetNameValue("COORDINATE_OPERATION", pszCO);
2712
0
        }
2713
2714
0
        const char *pszCoordEpoch =
2715
0
            CSLFetchNameValue(papszOptions, "COORDINATE_EPOCH");
2716
0
        if (pszCoordEpoch)
2717
0
        {
2718
0
            aosOptions.SetNameValue("COORDINATE_EPOCH", pszCoordEpoch);
2719
0
        }
2720
2721
0
        if (pszSrcCoordEpoch)
2722
0
        {
2723
0
            aosOptions.SetNameValue("SRC_COORDINATE_EPOCH", pszSrcCoordEpoch);
2724
0
            oSrcSRS.SetCoordinateEpoch(CPLAtof(pszSrcCoordEpoch));
2725
0
        }
2726
2727
0
        if (pszDstCoordEpoch)
2728
0
        {
2729
0
            aosOptions.SetNameValue("DST_COORDINATE_EPOCH", pszDstCoordEpoch);
2730
0
            oDstSRS.SetCoordinateEpoch(CPLAtof(pszDstCoordEpoch));
2731
0
        }
2732
2733
0
        if (const char *pszAllowBallpark =
2734
0
                CSLFetchNameValue(papszOptions, "ALLOW_BALLPARK"))
2735
0
        {
2736
0
            aosOptions.SetNameValue("ALLOW_BALLPARK", pszAllowBallpark);
2737
0
        }
2738
2739
0
        if (const char *pszOnlyBest =
2740
0
                CSLFetchNameValue(papszOptions, "ONLY_BEST"))
2741
0
        {
2742
0
            aosOptions.SetNameValue("ONLY_BEST", pszOnlyBest);
2743
0
        }
2744
2745
0
        psInfo->pReprojectArg = GDALCreateReprojectionTransformerEx(
2746
0
            !oSrcSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oSrcSRS)
2747
0
                               : nullptr,
2748
0
            !oDstSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oDstSRS)
2749
0
                               : nullptr,
2750
0
            aosOptions.List());
2751
2752
0
        if (pszCO)
2753
0
        {
2754
0
            psInfo->bHasCustomTransformationPipeline = true;
2755
0
        }
2756
2757
0
        if (psInfo->pReprojectArg == nullptr)
2758
0
        {
2759
0
            GDALDestroyGenImgProjTransformer(psInfo);
2760
0
            return nullptr;
2761
0
        }
2762
0
        psInfo->pReproject = GDALReprojectionTransform;
2763
2764
        /* --------------------------------------------------------------------
2765
         */
2766
        /*      Handle optional reprojection approximation transformer. */
2767
        /* --------------------------------------------------------------------
2768
         */
2769
0
        const char *psApproxErrorFwd = CSLFetchNameValue(
2770
0
            papszOptions, "REPROJECTION_APPROX_ERROR_IN_DST_SRS_UNIT");
2771
0
        const char *psApproxErrorReverse = CSLFetchNameValue(
2772
0
            papszOptions, "REPROJECTION_APPROX_ERROR_IN_SRC_SRS_UNIT");
2773
0
        if (psApproxErrorFwd && psApproxErrorReverse)
2774
0
        {
2775
0
            void *pArg = GDALCreateApproxTransformer2(
2776
0
                psInfo->pReproject, psInfo->pReprojectArg,
2777
0
                CPLAtof(psApproxErrorFwd), CPLAtof(psApproxErrorReverse));
2778
0
            if (pArg == nullptr)
2779
0
            {
2780
0
                GDALDestroyGenImgProjTransformer(psInfo);
2781
0
                return nullptr;
2782
0
            }
2783
0
            psInfo->pReprojectArg = pArg;
2784
0
            psInfo->pReproject = GDALApproxTransform;
2785
0
            GDALApproxTransformerOwnsSubtransformer(psInfo->pReprojectArg,
2786
0
                                                    TRUE);
2787
0
        }
2788
0
    }
2789
2790
0
    return psInfo;
2791
0
}
2792
2793
/************************************************************************/
2794
/*                  GDALRefreshGenImgProjTransformer()                  */
2795
/************************************************************************/
2796
2797
void GDALRefreshGenImgProjTransformer(void *hTransformArg)
2798
0
{
2799
0
    GDALGenImgProjTransformInfo *psInfo =
2800
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
2801
2802
0
    if (psInfo->pReprojectArg &&
2803
0
        psInfo->bCheckWithInvertPROJ != GetCurrentCheckWithInvertPROJ())
2804
0
    {
2805
0
        psInfo->bCheckWithInvertPROJ = !psInfo->bCheckWithInvertPROJ;
2806
2807
0
        CPLXMLNode *psXML =
2808
0
            GDALSerializeTransformer(psInfo->pReproject, psInfo->pReprojectArg);
2809
0
        GDALDestroyTransformer(psInfo->pReprojectArg);
2810
0
        GDALDeserializeTransformer(psXML, &psInfo->pReproject,
2811
0
                                   &psInfo->pReprojectArg);
2812
0
        CPLDestroyXMLNode(psXML);
2813
0
    }
2814
0
}
2815
2816
/************************************************************************/
2817
/*                  GDALCreateGenImgProjTransformer3()                  */
2818
/************************************************************************/
2819
2820
/**
2821
 * Create image to image transformer.
2822
 *
2823
 * This function creates a transformation object that maps from pixel/line
2824
 * coordinates on one image to pixel/line coordinates on another image.  The
2825
 * images may potentially be georeferenced in different coordinate systems,
2826
 * and may used GCPs to map between their pixel/line coordinates and
2827
 * georeferenced coordinates (as opposed to the default assumption that their
2828
 * geotransform should be used).
2829
 *
2830
 * This transformer potentially performs three concatenated transformations.
2831
 *
2832
 * The first stage is from source image pixel/line coordinates to source
2833
 * image georeferenced coordinates, and may be done using the geotransform,
2834
 * or if not defined using a polynomial model derived from GCPs.  If GCPs
2835
 * are used this stage is accomplished using GDALGCPTransform().
2836
 *
2837
 * The second stage is to change projections from the source coordinate system
2838
 * to the destination coordinate system, assuming they differ.  This is
2839
 * accomplished internally using GDALReprojectionTransform().
2840
 *
2841
 * The third stage is converting from destination image georeferenced
2842
 * coordinates to destination image coordinates.  This is done using the
2843
 * destination image geotransform, or if not available, using a polynomial
2844
 * model derived from GCPs. If GCPs are used this stage is accomplished using
2845
 * GDALGCPTransform().  This stage is skipped if hDstDS is NULL when the
2846
 * transformation is created.
2847
 *
2848
 * @param pszSrcWKT source WKT (or NULL).
2849
 * @param padfSrcGeoTransform source geotransform (or NULL).
2850
 * @param pszDstWKT destination WKT (or NULL).
2851
 * @param padfDstGeoTransform destination geotransform (or NULL).
2852
 *
2853
 * @return handle suitable for use GDALGenImgProjTransform(), and to be
2854
 * deallocated with GDALDestroyGenImgProjTransformer() or NULL on failure.
2855
 */
2856
2857
void *GDALCreateGenImgProjTransformer3(const char *pszSrcWKT,
2858
                                       const double *padfSrcGeoTransform,
2859
                                       const char *pszDstWKT,
2860
                                       const double *padfDstGeoTransform)
2861
2862
0
{
2863
0
    OGRSpatialReference oSrcSRS;
2864
0
    if (pszSrcWKT)
2865
0
    {
2866
0
        oSrcSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
2867
0
        if (pszSrcWKT[0] != '\0' &&
2868
0
            oSrcSRS.importFromWkt(pszSrcWKT) != OGRERR_NONE)
2869
0
        {
2870
0
            CPLError(CE_Failure, CPLE_AppDefined,
2871
0
                     "Failed to import coordinate system `%s'.", pszSrcWKT);
2872
0
            return nullptr;
2873
0
        }
2874
0
    }
2875
2876
0
    OGRSpatialReference oDstSRS;
2877
0
    if (pszDstWKT)
2878
0
    {
2879
0
        oDstSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
2880
0
        if (pszDstWKT[0] != '\0' &&
2881
0
            oDstSRS.importFromWkt(pszDstWKT) != OGRERR_NONE)
2882
0
        {
2883
0
            CPLError(CE_Failure, CPLE_AppDefined,
2884
0
                     "Failed to import coordinate system `%s'.", pszDstWKT);
2885
0
            return nullptr;
2886
0
        }
2887
0
    }
2888
0
    return GDALCreateGenImgProjTransformer4(
2889
0
        OGRSpatialReference::ToHandle(&oSrcSRS), padfSrcGeoTransform,
2890
0
        OGRSpatialReference::ToHandle(&oDstSRS), padfDstGeoTransform, nullptr);
2891
0
}
2892
2893
/************************************************************************/
2894
/*                  GDALCreateGenImgProjTransformer4()                  */
2895
/************************************************************************/
2896
2897
/**
2898
 * Create image to image transformer.
2899
 *
2900
 * Similar to GDALCreateGenImgProjTransformer3(), except that it takes
2901
 * OGRSpatialReferenceH objects and options.
2902
 * The options are the ones supported by GDALCreateReprojectionTransformerEx()
2903
 *
2904
 * @since GDAL 3.0
2905
 */
2906
void *GDALCreateGenImgProjTransformer4(OGRSpatialReferenceH hSrcSRS,
2907
                                       const double *padfSrcGeoTransform,
2908
                                       OGRSpatialReferenceH hDstSRS,
2909
                                       const double *padfDstGeoTransform,
2910
                                       const char *const *papszOptions)
2911
0
{
2912
    /* -------------------------------------------------------------------- */
2913
    /*      Initialize the transform info.                                  */
2914
    /* -------------------------------------------------------------------- */
2915
0
    GDALGenImgProjTransformInfo *psInfo =
2916
0
        GDALCreateGenImgProjTransformerInternal();
2917
2918
    /* -------------------------------------------------------------------- */
2919
    /*      Get forward and inverse geotransform for the source image.      */
2920
    /* -------------------------------------------------------------------- */
2921
2922
0
    const auto SetParams =
2923
0
        [](GDALGenImgProjTransformPart &part, const double *padfGT)
2924
0
    {
2925
0
        if (padfGT)
2926
0
        {
2927
0
            memcpy(part.adfGeoTransform, padfGT, sizeof(part.adfGeoTransform));
2928
0
            if (!GDALInvGeoTransform(part.adfGeoTransform,
2929
0
                                     part.adfInvGeoTransform))
2930
0
            {
2931
0
                CPLError(CE_Failure, CPLE_AppDefined,
2932
0
                         "Cannot invert geotransform");
2933
0
                return false;
2934
0
            }
2935
0
        }
2936
0
        else
2937
0
        {
2938
0
            part.adfGeoTransform[0] = 0.0;
2939
0
            part.adfGeoTransform[1] = 1.0;
2940
0
            part.adfGeoTransform[2] = 0.0;
2941
0
            part.adfGeoTransform[3] = 0.0;
2942
0
            part.adfGeoTransform[4] = 0.0;
2943
0
            part.adfGeoTransform[5] = 1.0;
2944
0
            memcpy(part.adfInvGeoTransform, part.adfGeoTransform,
2945
0
                   sizeof(double) * 6);
2946
0
        }
2947
0
        return true;
2948
0
    };
2949
2950
0
    if (!SetParams(psInfo->sSrcParams, padfSrcGeoTransform))
2951
0
    {
2952
0
        GDALDestroyGenImgProjTransformer(psInfo);
2953
0
        return nullptr;
2954
0
    }
2955
2956
    /* -------------------------------------------------------------------- */
2957
    /*      Setup reprojection.                                             */
2958
    /* -------------------------------------------------------------------- */
2959
0
    OGRSpatialReference *poSrcSRS = OGRSpatialReference::FromHandle(hSrcSRS);
2960
0
    OGRSpatialReference *poDstSRS = OGRSpatialReference::FromHandle(hDstSRS);
2961
0
    if (!poSrcSRS->IsEmpty() && !poDstSRS->IsEmpty() &&
2962
0
        !poSrcSRS->IsSame(poDstSRS))
2963
0
    {
2964
0
        psInfo->pReprojectArg =
2965
0
            GDALCreateReprojectionTransformerEx(hSrcSRS, hDstSRS, papszOptions);
2966
0
        if (psInfo->pReprojectArg == nullptr)
2967
0
        {
2968
0
            GDALDestroyGenImgProjTransformer(psInfo);
2969
0
            return nullptr;
2970
0
        }
2971
0
        psInfo->pReproject = GDALReprojectionTransform;
2972
0
    }
2973
2974
    /* -------------------------------------------------------------------- */
2975
    /*      Get forward and inverse geotransform for destination image.     */
2976
    /*      If we have no destination matrix use a unit transform.          */
2977
    /* -------------------------------------------------------------------- */
2978
0
    if (!SetParams(psInfo->sDstParams, padfDstGeoTransform))
2979
0
    {
2980
0
        GDALDestroyGenImgProjTransformer(psInfo);
2981
0
        return nullptr;
2982
0
    }
2983
2984
0
    return psInfo;
2985
0
}
2986
2987
/************************************************************************/
2988
/*            GDALSetGenImgProjTransformerDstGeoTransform()             */
2989
/************************************************************************/
2990
2991
/**
2992
 * Set GenImgProj output geotransform.
2993
 *
2994
 * Normally the "destination geotransform", or transformation between
2995
 * georeferenced output coordinates and pixel/line coordinates on the
2996
 * destination file is extracted from the destination file by
2997
 * GDALCreateGenImgProjTransformer() and stored in the GenImgProj private
2998
 * info.  However, sometimes it is inconvenient to have an output file
2999
 * handle with appropriate geotransform information when creating the
3000
 * transformation.  For these cases, this function can be used to apply
3001
 * the destination geotransform.
3002
 *
3003
 * @param hTransformArg the handle to update.
3004
 * @param padfGeoTransform the destination geotransform to apply (six doubles).
3005
 */
3006
3007
void GDALSetGenImgProjTransformerDstGeoTransform(void *hTransformArg,
3008
                                                 const double *padfGeoTransform)
3009
3010
0
{
3011
0
    VALIDATE_POINTER0(hTransformArg,
3012
0
                      "GDALSetGenImgProjTransformerDstGeoTransform");
3013
3014
0
    GDALGenImgProjTransformInfo *psInfo =
3015
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
3016
3017
0
    memcpy(psInfo->sDstParams.adfGeoTransform, padfGeoTransform,
3018
0
           sizeof(double) * 6);
3019
0
    if (!GDALInvGeoTransform(psInfo->sDstParams.adfGeoTransform,
3020
0
                             psInfo->sDstParams.adfInvGeoTransform))
3021
0
    {
3022
0
        CPLError(CE_Failure, CPLE_AppDefined, "Cannot invert geotransform");
3023
0
    }
3024
0
}
3025
3026
/************************************************************************/
3027
/*                  GDALDestroyGenImgProjTransformer()                  */
3028
/************************************************************************/
3029
3030
/**
3031
 * GenImgProjTransformer deallocator.
3032
 *
3033
 * This function is used to deallocate the handle created with
3034
 * GDALCreateGenImgProjTransformer().
3035
 *
3036
 * @param hTransformArg the handle to deallocate.
3037
 */
3038
3039
void GDALDestroyGenImgProjTransformer(void *hTransformArg)
3040
3041
0
{
3042
0
    if (hTransformArg == nullptr)
3043
0
        return;
3044
3045
0
    GDALGenImgProjTransformInfo *psInfo =
3046
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
3047
3048
0
    if (psInfo->sSrcParams.pTransformArg != nullptr)
3049
0
        GDALDestroyTransformer(psInfo->sSrcParams.pTransformArg);
3050
3051
0
    if (psInfo->sDstParams.pTransformArg != nullptr)
3052
0
        GDALDestroyTransformer(psInfo->sDstParams.pTransformArg);
3053
3054
0
    if (psInfo->pReprojectArg != nullptr)
3055
0
        GDALDestroyTransformer(psInfo->pReprojectArg);
3056
3057
0
    CPLFree(psInfo);
3058
0
}
3059
3060
/************************************************************************/
3061
/*                      GDALGenImgProjTransform()                       */
3062
/************************************************************************/
3063
3064
/**
3065
 * Perform general image reprojection transformation.
3066
 *
3067
 * Actually performs the transformation setup in
3068
 * GDALCreateGenImgProjTransformer().  This function matches the signature
3069
 * required by the GDALTransformerFunc(), and more details on the arguments
3070
 * can be found in that topic.
3071
 */
3072
3073
#ifdef DEBUG_APPROX_TRANSFORMER
3074
int countGDALGenImgProjTransform = 0;
3075
#endif
3076
3077
int GDALGenImgProjTransform(void *pTransformArgIn, int bDstToSrc,
3078
                            int nPointCount, double *padfX, double *padfY,
3079
                            double *padfZ, int *panSuccess)
3080
0
{
3081
0
    GDALGenImgProjTransformInfo *psInfo =
3082
0
        static_cast<GDALGenImgProjTransformInfo *>(pTransformArgIn);
3083
3084
#ifdef DEBUG_APPROX_TRANSFORMER
3085
    CPLAssert(nPointCount > 0);
3086
    countGDALGenImgProjTransform += nPointCount;
3087
#endif
3088
3089
0
    for (int i = 0; i < nPointCount; i++)
3090
0
    {
3091
0
        panSuccess[i] = (padfX[i] != HUGE_VAL && padfY[i] != HUGE_VAL);
3092
0
    }
3093
3094
0
    int ret = TRUE;
3095
3096
    /* -------------------------------------------------------------------- */
3097
    /*      Convert from src (dst) pixel/line to src (dst)                  */
3098
    /*      georeferenced coordinates.                                      */
3099
    /* -------------------------------------------------------------------- */
3100
0
    {
3101
0
        const auto params = bDstToSrc ? psInfo->sDstParams : psInfo->sSrcParams;
3102
0
        const double *padfGeoTransform = params.adfGeoTransform;
3103
0
        void *pTransformArg = params.pTransformArg;
3104
0
        GDALTransformerFunc pTransformer = params.pTransformer;
3105
3106
0
        if (pTransformArg != nullptr)
3107
0
        {
3108
0
            if (!pTransformer(pTransformArg, FALSE, nPointCount, padfX, padfY,
3109
0
                              padfZ, panSuccess))
3110
0
                ret = FALSE;
3111
0
        }
3112
0
        else
3113
0
        {
3114
0
            for (int i = 0; i < nPointCount; i++)
3115
0
            {
3116
0
                if (!panSuccess[i])
3117
0
                    continue;
3118
3119
0
                const double dfNewX = padfGeoTransform[0] +
3120
0
                                      padfX[i] * padfGeoTransform[1] +
3121
0
                                      padfY[i] * padfGeoTransform[2];
3122
0
                const double dfNewY = padfGeoTransform[3] +
3123
0
                                      padfX[i] * padfGeoTransform[4] +
3124
0
                                      padfY[i] * padfGeoTransform[5];
3125
3126
0
                padfX[i] = dfNewX;
3127
0
                padfY[i] = dfNewY;
3128
0
            }
3129
0
        }
3130
0
    }
3131
3132
    /* -------------------------------------------------------------------- */
3133
    /*      Reproject if needed.                                            */
3134
    /* -------------------------------------------------------------------- */
3135
0
    if (psInfo->pReprojectArg)
3136
0
    {
3137
0
        if (!psInfo->pReproject(psInfo->pReprojectArg, bDstToSrc, nPointCount,
3138
0
                                padfX, padfY, padfZ, panSuccess))
3139
0
            ret = FALSE;
3140
0
    }
3141
3142
    /* -------------------------------------------------------------------- */
3143
    /*      Convert dst (src) georef coordinates back to pixel/line.        */
3144
    /* -------------------------------------------------------------------- */
3145
0
    {
3146
0
        const auto params = bDstToSrc ? psInfo->sSrcParams : psInfo->sDstParams;
3147
0
        const double *padfInvGeoTransform = params.adfInvGeoTransform;
3148
0
        void *pTransformArg = params.pTransformArg;
3149
0
        GDALTransformerFunc pTransformer = params.pTransformer;
3150
3151
0
        if (pTransformArg != nullptr)
3152
0
        {
3153
0
            if (!pTransformer(pTransformArg, TRUE, nPointCount, padfX, padfY,
3154
0
                              padfZ, panSuccess))
3155
0
                ret = FALSE;
3156
0
        }
3157
0
        else
3158
0
        {
3159
0
            for (int i = 0; i < nPointCount; i++)
3160
0
            {
3161
0
                if (!panSuccess[i])
3162
0
                    continue;
3163
3164
0
                const double dfNewX = padfInvGeoTransform[0] +
3165
0
                                      padfX[i] * padfInvGeoTransform[1] +
3166
0
                                      padfY[i] * padfInvGeoTransform[2];
3167
0
                const double dfNewY = padfInvGeoTransform[3] +
3168
0
                                      padfX[i] * padfInvGeoTransform[4] +
3169
0
                                      padfY[i] * padfInvGeoTransform[5];
3170
3171
0
                padfX[i] = dfNewX;
3172
0
                padfY[i] = dfNewY;
3173
0
            }
3174
0
        }
3175
0
    }
3176
3177
0
    return ret;
3178
0
}
3179
3180
/************************************************************************/
3181
/*              GDALTransformLonLatToDestGenImgProjTransformer()        */
3182
/************************************************************************/
3183
3184
int GDALTransformLonLatToDestGenImgProjTransformer(void *hTransformArg,
3185
                                                   double *pdfX, double *pdfY)
3186
0
{
3187
0
    GDALGenImgProjTransformInfo *psInfo =
3188
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
3189
3190
0
    if (psInfo->pReprojectArg == nullptr ||
3191
0
        psInfo->pReproject != GDALReprojectionTransform)
3192
0
        return false;
3193
3194
0
    GDALReprojectionTransformInfo *psReprojInfo =
3195
0
        static_cast<GDALReprojectionTransformInfo *>(psInfo->pReprojectArg);
3196
0
    if (psReprojInfo->poForwardTransform == nullptr ||
3197
0
        psReprojInfo->poForwardTransform->GetSourceCS() == nullptr)
3198
0
        return false;
3199
3200
0
    double z = 0;
3201
0
    int success = true;
3202
0
    auto poSourceCRS = psReprojInfo->poForwardTransform->GetSourceCS();
3203
0
    if (poSourceCRS->IsGeographic() &&
3204
0
        std::fabs(poSourceCRS->GetAngularUnits() -
3205
0
                  CPLAtof(SRS_UA_DEGREE_CONV)) < 1e-9)
3206
0
    {
3207
        // Optimization to avoid creating a OGRCoordinateTransformation
3208
0
        OGRAxisOrientation eSourceFirstAxisOrient = OAO_Other;
3209
0
        poSourceCRS->GetAxis(nullptr, 0, &eSourceFirstAxisOrient);
3210
0
        const auto &mapping = poSourceCRS->GetDataAxisToSRSAxisMapping();
3211
0
        if ((mapping[0] == 2 && eSourceFirstAxisOrient == OAO_East) ||
3212
0
            (mapping[0] == 1 && eSourceFirstAxisOrient != OAO_East))
3213
0
        {
3214
0
            std::swap(*pdfX, *pdfY);
3215
0
        }
3216
0
    }
3217
0
    else
3218
0
    {
3219
0
        auto poLongLat =
3220
0
            std::unique_ptr<OGRSpatialReference>(poSourceCRS->CloneGeogCS());
3221
0
        if (poLongLat == nullptr)
3222
0
            return false;
3223
0
        poLongLat->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3224
3225
0
        const bool bCurrentCheckWithInvertProj =
3226
0
            GetCurrentCheckWithInvertPROJ();
3227
0
        if (!bCurrentCheckWithInvertProj)
3228
0
            CPLSetThreadLocalConfigOption("CHECK_WITH_INVERT_PROJ", "YES");
3229
0
        auto poCT = std::unique_ptr<OGRCoordinateTransformation>(
3230
0
            OGRCreateCoordinateTransformation(poLongLat.get(), poSourceCRS));
3231
0
        if (!bCurrentCheckWithInvertProj)
3232
0
            CPLSetThreadLocalConfigOption("CHECK_WITH_INVERT_PROJ", nullptr);
3233
0
        if (poCT == nullptr)
3234
0
            return false;
3235
3236
0
        poCT->SetEmitErrors(false);
3237
0
        if (!poCT->Transform(1, pdfX, pdfY))
3238
0
            return false;
3239
3240
0
        if (!psInfo->pReproject(psInfo->pReprojectArg, false, 1, pdfX, pdfY, &z,
3241
0
                                &success) ||
3242
0
            !success)
3243
0
        {
3244
0
            return false;
3245
0
        }
3246
0
    }
3247
3248
0
    double *padfGeoTransform = psInfo->sDstParams.adfInvGeoTransform;
3249
0
    void *pTransformArg = psInfo->sDstParams.pTransformArg;
3250
0
    GDALTransformerFunc pTransformer = psInfo->sDstParams.pTransformer;
3251
0
    if (pTransformArg != nullptr)
3252
0
    {
3253
0
        if (!pTransformer(pTransformArg, TRUE, 1, pdfX, pdfY, &z, &success) ||
3254
0
            !success)
3255
0
        {
3256
0
            return false;
3257
0
        }
3258
0
    }
3259
0
    else
3260
0
    {
3261
0
        const double dfNewX = padfGeoTransform[0] +
3262
0
                              pdfX[0] * padfGeoTransform[1] +
3263
0
                              pdfY[0] * padfGeoTransform[2];
3264
0
        const double dfNewY = padfGeoTransform[3] +
3265
0
                              pdfX[0] * padfGeoTransform[4] +
3266
0
                              pdfY[0] * padfGeoTransform[5];
3267
3268
0
        pdfX[0] = dfNewX;
3269
0
        pdfY[0] = dfNewY;
3270
0
    }
3271
3272
0
    return true;
3273
0
}
3274
3275
/************************************************************************/
3276
/*                 GDALSerializeGenImgProjTransformer()                 */
3277
/************************************************************************/
3278
3279
static CPLXMLNode *GDALSerializeGenImgProjTransformer(void *pTransformArg)
3280
3281
0
{
3282
0
    GDALGenImgProjTransformInfo *psInfo =
3283
0
        static_cast<GDALGenImgProjTransformInfo *>(pTransformArg);
3284
3285
0
    CPLXMLNode *psTree =
3286
0
        CPLCreateXMLNode(nullptr, CXT_Element, "GenImgProjTransformer");
3287
3288
0
    const auto SerializePart =
3289
0
        [psTree](const char *pszPrefix, const GDALGenImgProjTransformPart &part)
3290
0
    {
3291
0
        char szWork[200] = {};
3292
3293
        /* ------------------------------------------------------------- */
3294
        /*      Handle transformation.                                   */
3295
        /* ------------------------------------------------------------- */
3296
0
        if (part.pTransformArg != nullptr)
3297
0
        {
3298
0
            CPLXMLNode *psTransformer =
3299
0
                GDALSerializeTransformer(part.pTransformer, part.pTransformArg);
3300
0
            if (psTransformer != nullptr)
3301
0
            {
3302
0
                CPLXMLNode *psTransformerContainer = CPLCreateXMLNode(
3303
0
                    psTree, CXT_Element,
3304
0
                    CPLSPrintf("%s%s", pszPrefix, psTransformer->pszValue));
3305
3306
0
                CPLAddXMLChild(psTransformerContainer, psTransformer);
3307
0
            }
3308
0
        }
3309
3310
        /* ------------------------------------------------------------- */
3311
        /*      Handle geotransforms.                                    */
3312
        /* ------------------------------------------------------------- */
3313
0
        else
3314
0
        {
3315
0
            CPLsnprintf(szWork, sizeof(szWork),
3316
0
                        "%.17g,%.17g,%.17g,%.17g,%.17g,%.17g",
3317
0
                        part.adfGeoTransform[0], part.adfGeoTransform[1],
3318
0
                        part.adfGeoTransform[2], part.adfGeoTransform[3],
3319
0
                        part.adfGeoTransform[4], part.adfGeoTransform[5]);
3320
0
            CPLCreateXMLElementAndValue(
3321
0
                psTree, CPLSPrintf("%sGeoTransform", pszPrefix), szWork);
3322
3323
0
            CPLsnprintf(szWork, sizeof(szWork),
3324
0
                        "%.17g,%.17g,%.17g,%.17g,%.17g,%.17g",
3325
0
                        part.adfInvGeoTransform[0], part.adfInvGeoTransform[1],
3326
0
                        part.adfInvGeoTransform[2], part.adfInvGeoTransform[3],
3327
0
                        part.adfInvGeoTransform[4], part.adfInvGeoTransform[5]);
3328
0
            CPLCreateXMLElementAndValue(
3329
0
                psTree, CPLSPrintf("%sInvGeoTransform", pszPrefix), szWork);
3330
0
        }
3331
0
    };
3332
3333
0
    SerializePart("Src", psInfo->sSrcParams);
3334
0
    SerializePart("Dst", psInfo->sDstParams);
3335
3336
    /* -------------------------------------------------------------------- */
3337
    /*      Do we have a reprojection transformer?                          */
3338
    /* -------------------------------------------------------------------- */
3339
0
    if (psInfo->pReprojectArg != nullptr)
3340
0
    {
3341
3342
0
        CPLXMLNode *psTransformerContainer =
3343
0
            CPLCreateXMLNode(psTree, CXT_Element, "ReprojectTransformer");
3344
3345
0
        CPLXMLNode *psTransformer =
3346
0
            GDALSerializeTransformer(psInfo->pReproject, psInfo->pReprojectArg);
3347
0
        if (psTransformer != nullptr)
3348
0
            CPLAddXMLChild(psTransformerContainer, psTransformer);
3349
0
    }
3350
3351
0
    return psTree;
3352
0
}
3353
3354
/************************************************************************/
3355
/*                    GDALDeserializeGeoTransform()                     */
3356
/************************************************************************/
3357
3358
static void GDALDeserializeGeoTransform(const char *pszGT,
3359
                                        double adfGeoTransform[6])
3360
0
{
3361
0
    CPLsscanf(pszGT, "%lf,%lf,%lf,%lf,%lf,%lf", adfGeoTransform + 0,
3362
0
              adfGeoTransform + 1, adfGeoTransform + 2, adfGeoTransform + 3,
3363
0
              adfGeoTransform + 4, adfGeoTransform + 5);
3364
0
}
3365
3366
/************************************************************************/
3367
/*                GDALDeserializeGenImgProjTransformer()                */
3368
/************************************************************************/
3369
3370
void *GDALDeserializeGenImgProjTransformer(CPLXMLNode *psTree)
3371
3372
0
{
3373
    /* -------------------------------------------------------------------- */
3374
    /*      Initialize the transform info.                                  */
3375
    /* -------------------------------------------------------------------- */
3376
0
    GDALGenImgProjTransformInfo *psInfo =
3377
0
        GDALCreateGenImgProjTransformerInternal();
3378
3379
0
    const auto DeserializePart =
3380
0
        [psTree](const char *pszPrefix, GDALGenImgProjTransformPart &part)
3381
0
    {
3382
        /* ----------------------------------------------------------------- */
3383
        /*      Geotransform                                                 */
3384
        /* ----------------------------------------------------------------- */
3385
0
        if (const auto psGTNode =
3386
0
                CPLGetXMLNode(psTree, CPLSPrintf("%sGeoTransform", pszPrefix)))
3387
0
        {
3388
0
            GDALDeserializeGeoTransform(CPLGetXMLValue(psGTNode, "", ""),
3389
0
                                        part.adfGeoTransform);
3390
3391
0
            if (const auto psInvGTNode = CPLGetXMLNode(
3392
0
                    psTree, CPLSPrintf("%sInvGeoTransform", pszPrefix)))
3393
0
            {
3394
0
                GDALDeserializeGeoTransform(CPLGetXMLValue(psInvGTNode, "", ""),
3395
0
                                            part.adfInvGeoTransform);
3396
0
            }
3397
0
            else
3398
0
            {
3399
0
                if (!GDALInvGeoTransform(part.adfGeoTransform,
3400
0
                                         part.adfInvGeoTransform))
3401
0
                {
3402
0
                    CPLError(CE_Failure, CPLE_AppDefined,
3403
0
                             "Cannot invert geotransform");
3404
0
                }
3405
0
            }
3406
0
        }
3407
3408
        /* ---------------------------------------------------------------- */
3409
        /*      Transform                                                   */
3410
        /* ---------------------------------------------------------------- */
3411
0
        else
3412
0
        {
3413
0
            for (CPLXMLNode *psIter = psTree->psChild; psIter != nullptr;
3414
0
                 psIter = psIter->psNext)
3415
0
            {
3416
0
                if (psIter->eType == CXT_Element &&
3417
0
                    STARTS_WITH_CI(psIter->pszValue, pszPrefix))
3418
0
                {
3419
0
                    GDALDeserializeTransformer(psIter->psChild,
3420
0
                                               &part.pTransformer,
3421
0
                                               &part.pTransformArg);
3422
0
                    break;
3423
0
                }
3424
0
            }
3425
0
        }
3426
0
    };
3427
3428
0
    DeserializePart("Src", psInfo->sSrcParams);
3429
0
    DeserializePart("Dst", psInfo->sDstParams);
3430
3431
    /* -------------------------------------------------------------------- */
3432
    /*      Reproject transformer                                           */
3433
    /* -------------------------------------------------------------------- */
3434
0
    CPLXMLNode *psSubtree = CPLGetXMLNode(psTree, "ReprojectTransformer");
3435
0
    if (psSubtree != nullptr && psSubtree->psChild != nullptr)
3436
0
    {
3437
0
        GDALDeserializeTransformer(psSubtree->psChild, &psInfo->pReproject,
3438
0
                                   &psInfo->pReprojectArg);
3439
0
    }
3440
3441
0
    return psInfo;
3442
0
}
3443
3444
/************************************************************************/
3445
/*                 GDALCreateReprojectionTransformer()                  */
3446
/************************************************************************/
3447
3448
/**
3449
 * Create reprojection transformer.
3450
 *
3451
 * Creates a callback data structure suitable for use with
3452
 * GDALReprojectionTransformation() to represent a transformation from
3453
 * one geographic or projected coordinate system to another.  On input
3454
 * the coordinate systems are described in OpenGIS WKT format.
3455
 *
3456
 * Internally the OGRCoordinateTransformation object is used to implement
3457
 * the reprojection.
3458
 *
3459
 * @param pszSrcWKT the coordinate system for the source coordinate system.
3460
 * @param pszDstWKT the coordinate system for the destination coordinate
3461
 * system.
3462
 *
3463
 * @return Handle for use with GDALReprojectionTransform(), or NULL if the
3464
 * system fails to initialize the reprojection.
3465
 **/
3466
3467
void *GDALCreateReprojectionTransformer(const char *pszSrcWKT,
3468
                                        const char *pszDstWKT)
3469
3470
0
{
3471
    /* -------------------------------------------------------------------- */
3472
    /*      Ingest the SRS definitions.                                     */
3473
    /* -------------------------------------------------------------------- */
3474
0
    OGRSpatialReference oSrcSRS;
3475
0
    oSrcSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3476
0
    if (oSrcSRS.importFromWkt(pszSrcWKT) != OGRERR_NONE)
3477
0
    {
3478
0
        CPLError(CE_Failure, CPLE_AppDefined,
3479
0
                 "Failed to import coordinate system `%s'.", pszSrcWKT);
3480
0
        return nullptr;
3481
0
    }
3482
3483
0
    OGRSpatialReference oDstSRS;
3484
0
    oDstSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3485
0
    if (oDstSRS.importFromWkt(pszDstWKT) != OGRERR_NONE)
3486
0
    {
3487
0
        CPLError(CE_Failure, CPLE_AppDefined,
3488
0
                 "Failed to import coordinate system `%s'.", pszSrcWKT);
3489
0
        return nullptr;
3490
0
    }
3491
3492
0
    return GDALCreateReprojectionTransformerEx(
3493
0
        OGRSpatialReference::ToHandle(&oSrcSRS),
3494
0
        OGRSpatialReference::ToHandle(&oDstSRS), nullptr);
3495
0
}
3496
3497
/************************************************************************/
3498
/*                 GDALCreateReprojectionTransformerEx()                */
3499
/************************************************************************/
3500
3501
/**
3502
 * Create reprojection transformer.
3503
 *
3504
 * Creates a callback data structure suitable for use with
3505
 * GDALReprojectionTransformation() to represent a transformation from
3506
 * one geographic or projected coordinate system to another.
3507
 *
3508
 * Internally the OGRCoordinateTransformation object is used to implement
3509
 * the reprojection.
3510
 *
3511
 * @param hSrcSRS the coordinate system for the source coordinate system.
3512
 * @param hDstSRS the coordinate system for the destination coordinate
3513
 * system.
3514
 * @param papszOptions NULL-terminated list of options, or NULL. Currently
3515
 * supported options are:
3516
 * <ul>
3517
 * <li>AREA_OF_INTEREST=west_long,south_lat,east_long,north_lat: Values in
3518
 * degrees. longitudes in [-180,180], latitudes in [-90,90].</li>
3519
 * <li>COORDINATE_OPERATION=string: PROJ or WKT string representing a
3520
 * coordinate operation, overriding the default computed transformation.</li>
3521
 * <li>COORDINATE_EPOCH=decimal_year: Coordinate epoch, expressed as a
3522
 * decimal year. Useful for time-dependent coordinate operations.</li>
3523
 * <li> SRC_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch of source CRS,
3524
 * expressed as a decimal year. Useful for time-dependent coordinate
3525
 *operations.</li>
3526
 * <li> DST_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch
3527
 *of target CRS, expressed as a decimal year. Useful for time-dependent
3528
 *coordinate operations.</li>
3529
 * <li> ALLOW_BALLPARK=YES/NO: (GDAL &gt;= 3.11) Whether ballpark coordinate
3530
 * operations are allowed. Defaults to YES.</li>
3531
 * <li> ONLY_BEST=YES/NO/AUTO: (GDAL &gt;= 3.11) By default (at least in the
3532
 * PROJ 9.x series), PROJ may use coordinate
3533
 * operations that are not the "best" if resources (typically grids) needed
3534
 * to use them are missing. It will then fallback to other coordinate operations
3535
 * that have a lesser accuracy, for example using Helmert transformations,
3536
 * or in the absence of such operations, to ones with potential very rought
3537
 * accuracy, using "ballpark" transformations
3538
 * (see https://proj.org/glossary.html).
3539
 * When calling this method with YES, PROJ will only consider the
3540
 * "best" operation, and error out (at Transform() time) if they cannot be
3541
 * used.
3542
 * This method may be used together with ALLOW_BALLPARK=NO to
3543
 * only allow best operations that have a known accuracy.
3544
 * Note that this method has no effect on PROJ versions before 9.2.
3545
 * The default value for this option can be also set with the
3546
 * PROJ_ONLY_BEST_DEFAULT environment variable, or with the "only_best_default"
3547
 * setting of proj.ini. Calling SetOnlyBest() overrides such default value.</li>
3548
 * </ul>
3549
 *
3550
 * @return Handle for use with GDALReprojectionTransform(), or NULL if the
3551
 * system fails to initialize the reprojection.
3552
 *
3553
 * @since GDAL 3.0
3554
 **/
3555
3556
void *GDALCreateReprojectionTransformerEx(OGRSpatialReferenceH hSrcSRS,
3557
                                          OGRSpatialReferenceH hDstSRS,
3558
                                          const char *const *papszOptions)
3559
0
{
3560
0
    OGRSpatialReference *poSrcSRS = OGRSpatialReference::FromHandle(hSrcSRS);
3561
0
    OGRSpatialReference *poDstSRS = OGRSpatialReference::FromHandle(hDstSRS);
3562
3563
    /* -------------------------------------------------------------------- */
3564
    /*      Build the forward coordinate transformation.                    */
3565
    /* -------------------------------------------------------------------- */
3566
0
    double dfWestLongitudeDeg = 0.0;
3567
0
    double dfSouthLatitudeDeg = 0.0;
3568
0
    double dfEastLongitudeDeg = 0.0;
3569
0
    double dfNorthLatitudeDeg = 0.0;
3570
0
    const char *pszBBOX = CSLFetchNameValue(papszOptions, "AREA_OF_INTEREST");
3571
0
    if (pszBBOX)
3572
0
    {
3573
0
        char **papszTokens = CSLTokenizeString2(pszBBOX, ",", 0);
3574
0
        if (CSLCount(papszTokens) == 4)
3575
0
        {
3576
0
            dfWestLongitudeDeg = CPLAtof(papszTokens[0]);
3577
0
            dfSouthLatitudeDeg = CPLAtof(papszTokens[1]);
3578
0
            dfEastLongitudeDeg = CPLAtof(papszTokens[2]);
3579
0
            dfNorthLatitudeDeg = CPLAtof(papszTokens[3]);
3580
0
        }
3581
0
        CSLDestroy(papszTokens);
3582
0
    }
3583
0
    const char *pszCO = CSLFetchNameValue(papszOptions, "COORDINATE_OPERATION");
3584
3585
0
    OGRCoordinateTransformationOptions optionsFwd;
3586
0
    if (!(dfWestLongitudeDeg == 0.0 && dfSouthLatitudeDeg == 0.0 &&
3587
0
          dfEastLongitudeDeg == 0.0 && dfNorthLatitudeDeg == 0.0))
3588
0
    {
3589
0
        optionsFwd.SetAreaOfInterest(dfWestLongitudeDeg, dfSouthLatitudeDeg,
3590
0
                                     dfEastLongitudeDeg, dfNorthLatitudeDeg);
3591
0
    }
3592
0
    if (pszCO)
3593
0
    {
3594
0
        optionsFwd.SetCoordinateOperation(pszCO, false);
3595
0
    }
3596
3597
0
    const char *pszCENTER_LONG = CSLFetchNameValue(papszOptions, "CENTER_LONG");
3598
0
    if (pszCENTER_LONG)
3599
0
    {
3600
0
        optionsFwd.SetSourceCenterLong(CPLAtof(pszCENTER_LONG));
3601
0
    }
3602
3603
0
    optionsFwd.SetBallparkAllowed(CPLTestBool(
3604
0
        CSLFetchNameValueDef(papszOptions, "ALLOW_BALLPARK", "YES")));
3605
3606
0
    const char *pszOnlyBest =
3607
0
        CSLFetchNameValueDef(papszOptions, "ONLY_BEST", "AUTO");
3608
0
    if (!EQUAL(pszOnlyBest, "AUTO"))
3609
0
    {
3610
0
        optionsFwd.SetOnlyBest(CPLTestBool(pszOnlyBest));
3611
0
    }
3612
3613
0
    OGRCoordinateTransformation *poForwardTransform =
3614
0
        OGRCreateCoordinateTransformation(poSrcSRS, poDstSRS, optionsFwd);
3615
3616
0
    if (poForwardTransform == nullptr)
3617
        // OGRCreateCoordinateTransformation() will report errors on its own.
3618
0
        return nullptr;
3619
3620
0
    poForwardTransform->SetEmitErrors(false);
3621
3622
    /* -------------------------------------------------------------------- */
3623
    /*      Create a structure to hold the transform info, and also         */
3624
    /*      build reverse transform.  We assume that if the forward         */
3625
    /*      transform can be created, then so can the reverse one.          */
3626
    /* -------------------------------------------------------------------- */
3627
0
    GDALReprojectionTransformInfo *psInfo = new GDALReprojectionTransformInfo();
3628
3629
0
    psInfo->papszOptions = CSLDuplicate(papszOptions);
3630
0
    psInfo->poForwardTransform = poForwardTransform;
3631
0
    psInfo->dfTime = CPLAtof(CSLFetchNameValueDef(
3632
0
        papszOptions, "COORDINATE_EPOCH",
3633
0
        CSLFetchNameValueDef(
3634
0
            papszOptions, "DST_COORDINATE_EPOCH",
3635
0
            CSLFetchNameValueDef(papszOptions, "SRC_COORDINATE_EPOCH", "0"))));
3636
0
    psInfo->poReverseTransform = poForwardTransform->GetInverse();
3637
3638
0
    if (psInfo->poReverseTransform)
3639
0
        psInfo->poReverseTransform->SetEmitErrors(false);
3640
3641
0
    memcpy(psInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE,
3642
0
           strlen(GDAL_GTI2_SIGNATURE));
3643
0
    psInfo->sTI.pszClassName = GDAL_REPROJECTION_TRANSFORMER_CLASS_NAME;
3644
0
    psInfo->sTI.pfnTransform = GDALReprojectionTransform;
3645
0
    psInfo->sTI.pfnCleanup = GDALDestroyReprojectionTransformer;
3646
0
    psInfo->sTI.pfnSerialize = GDALSerializeReprojectionTransformer;
3647
3648
0
    return psInfo;
3649
0
}
3650
3651
/************************************************************************/
3652
/*                 GDALDestroyReprojectionTransformer()                 */
3653
/************************************************************************/
3654
3655
/**
3656
 * Destroy reprojection transformation.
3657
 *
3658
 * @param pTransformArg the transformation handle returned by
3659
 * GDALCreateReprojectionTransformer().
3660
 */
3661
3662
void GDALDestroyReprojectionTransformer(void *pTransformArg)
3663
3664
0
{
3665
0
    if (pTransformArg == nullptr)
3666
0
        return;
3667
3668
0
    GDALReprojectionTransformInfo *psInfo =
3669
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3670
3671
0
    if (psInfo->poForwardTransform)
3672
0
        OGRCoordinateTransformation::DestroyCT(psInfo->poForwardTransform);
3673
3674
0
    if (psInfo->poReverseTransform)
3675
0
        OGRCoordinateTransformation::DestroyCT(psInfo->poReverseTransform);
3676
3677
0
    CSLDestroy(psInfo->papszOptions);
3678
3679
0
    delete psInfo;
3680
0
}
3681
3682
/************************************************************************/
3683
/*                     GDALReprojectionTransform()                      */
3684
/************************************************************************/
3685
3686
/**
3687
 * Perform reprojection transformation.
3688
 *
3689
 * Actually performs the reprojection transformation described in
3690
 * GDALCreateReprojectionTransformer().  This function matches the
3691
 * GDALTransformerFunc() signature.  Details of the arguments are described
3692
 * there.
3693
 */
3694
3695
int GDALReprojectionTransform(void *pTransformArg, int bDstToSrc,
3696
                              int nPointCount, double *padfX, double *padfY,
3697
                              double *padfZ, int *panSuccess)
3698
3699
0
{
3700
0
    GDALReprojectionTransformInfo *psInfo =
3701
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3702
0
    int bSuccess;
3703
3704
0
    std::vector<double> adfTime;
3705
0
    double *padfT = nullptr;
3706
0
    if (psInfo->dfTime != 0.0 && nPointCount > 0)
3707
0
    {
3708
0
        adfTime.resize(nPointCount, psInfo->dfTime);
3709
0
        padfT = &adfTime[0];
3710
0
    }
3711
3712
0
    if (bDstToSrc)
3713
0
    {
3714
0
        if (psInfo->poReverseTransform == nullptr)
3715
0
        {
3716
0
            CPLError(
3717
0
                CE_Failure, CPLE_AppDefined,
3718
0
                "Inverse coordinate transformation cannot be instantiated");
3719
0
            if (panSuccess)
3720
0
            {
3721
0
                for (int i = 0; i < nPointCount; i++)
3722
0
                    panSuccess[i] = FALSE;
3723
0
            }
3724
0
            bSuccess = false;
3725
0
        }
3726
0
        else
3727
0
        {
3728
0
            bSuccess = psInfo->poReverseTransform->Transform(
3729
0
                nPointCount, padfX, padfY, padfZ, padfT, panSuccess);
3730
0
        }
3731
0
    }
3732
0
    else
3733
0
        bSuccess = psInfo->poForwardTransform->Transform(
3734
0
            nPointCount, padfX, padfY, padfZ, padfT, panSuccess);
3735
3736
0
    return bSuccess;
3737
0
}
3738
3739
/************************************************************************/
3740
/*                GDALSerializeReprojectionTransformer()                */
3741
/************************************************************************/
3742
3743
static CPLXMLNode *GDALSerializeReprojectionTransformer(void *pTransformArg)
3744
3745
0
{
3746
0
    CPLXMLNode *psTree;
3747
0
    GDALReprojectionTransformInfo *psInfo =
3748
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3749
3750
0
    psTree = CPLCreateXMLNode(nullptr, CXT_Element, "ReprojectionTransformer");
3751
3752
    /* -------------------------------------------------------------------- */
3753
    /*      Handle SourceCS.                                                */
3754
    /* -------------------------------------------------------------------- */
3755
0
    const auto ExportToWkt = [](const OGRSpatialReference *poSRS)
3756
0
    {
3757
        // Try first in WKT1 for backward compat
3758
0
        {
3759
0
            char *pszWKT = nullptr;
3760
0
            const char *const apszOptions[] = {"FORMAT=WKT1", nullptr};
3761
0
            CPLErrorHandlerPusher oHandler(CPLQuietErrorHandler);
3762
0
            CPLErrorStateBackuper oBackuper;
3763
0
            if (poSRS->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE)
3764
0
            {
3765
0
                std::string osRet(pszWKT);
3766
0
                CPLFree(pszWKT);
3767
0
                return osRet;
3768
0
            }
3769
0
            CPLFree(pszWKT);
3770
0
        }
3771
3772
0
        char *pszWKT = nullptr;
3773
0
        const char *const apszOptions[] = {"FORMAT=WKT2_2019", nullptr};
3774
0
        if (poSRS->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE)
3775
0
        {
3776
0
            std::string osRet(pszWKT);
3777
0
            CPLFree(pszWKT);
3778
0
            return osRet;
3779
0
        }
3780
0
        CPLFree(pszWKT);
3781
0
        return std::string();
3782
0
    };
3783
3784
0
    auto poSRS = psInfo->poForwardTransform->GetSourceCS();
3785
0
    if (poSRS)
3786
0
    {
3787
0
        const auto osWKT = ExportToWkt(poSRS);
3788
0
        CPLCreateXMLElementAndValue(psTree, "SourceSRS", osWKT.c_str());
3789
0
    }
3790
3791
    /* -------------------------------------------------------------------- */
3792
    /*      Handle DestinationCS.                                           */
3793
    /* -------------------------------------------------------------------- */
3794
0
    poSRS = psInfo->poForwardTransform->GetTargetCS();
3795
0
    if (poSRS)
3796
0
    {
3797
0
        const auto osWKT = ExportToWkt(poSRS);
3798
0
        CPLCreateXMLElementAndValue(psTree, "TargetSRS", osWKT.c_str());
3799
0
    }
3800
3801
    /* -------------------------------------------------------------------- */
3802
    /*      Serialize options.                                              */
3803
    /* -------------------------------------------------------------------- */
3804
0
    if (psInfo->papszOptions)
3805
0
    {
3806
0
        CPLXMLNode *psOptions =
3807
0
            CPLCreateXMLNode(psTree, CXT_Element, "Options");
3808
0
        for (auto iter = psInfo->papszOptions; *iter != nullptr; ++iter)
3809
0
        {
3810
0
            char *pszKey = nullptr;
3811
0
            const char *pszValue = CPLParseNameValue(*iter, &pszKey);
3812
0
            if (pszKey && pszValue)
3813
0
            {
3814
0
                auto elt =
3815
0
                    CPLCreateXMLElementAndValue(psOptions, "Option", pszValue);
3816
0
                CPLAddXMLAttributeAndValue(elt, "key", pszKey);
3817
0
            }
3818
0
            CPLFree(pszKey);
3819
0
        }
3820
0
    }
3821
3822
0
    return psTree;
3823
0
}
3824
3825
/************************************************************************/
3826
/*               GDALDeserializeReprojectionTransformer()               */
3827
/************************************************************************/
3828
3829
static void *GDALDeserializeReprojectionTransformer(CPLXMLNode *psTree)
3830
3831
0
{
3832
0
    const char *pszSourceSRS = CPLGetXMLValue(psTree, "SourceSRS", nullptr);
3833
0
    const char *pszTargetSRS = CPLGetXMLValue(psTree, "TargetSRS", nullptr);
3834
0
    char *pszSourceWKT = nullptr, *pszTargetWKT = nullptr;
3835
0
    void *pResult = nullptr;
3836
3837
0
    OGRSpatialReference oSrcSRS;
3838
0
    OGRSpatialReference oDstSRS;
3839
3840
0
    oSrcSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3841
0
    oDstSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3842
0
    if (pszSourceSRS != nullptr)
3843
0
    {
3844
0
        oSrcSRS.SetFromUserInput(pszSourceSRS);
3845
0
    }
3846
3847
0
    if (pszTargetSRS != nullptr)
3848
0
    {
3849
0
        oDstSRS.SetFromUserInput(pszTargetSRS);
3850
0
    }
3851
3852
0
    CPLStringList aosList;
3853
0
    const CPLXMLNode *psOptions = CPLGetXMLNode(psTree, "Options");
3854
0
    if (psOptions)
3855
0
    {
3856
0
        for (auto iter = psOptions->psChild; iter; iter = iter->psNext)
3857
0
        {
3858
0
            if (iter->eType == CXT_Element &&
3859
0
                strcmp(iter->pszValue, "Option") == 0)
3860
0
            {
3861
0
                const char *pszKey = CPLGetXMLValue(iter, "key", nullptr);
3862
0
                const char *pszValue = CPLGetXMLValue(iter, nullptr, nullptr);
3863
0
                if (pszKey && pszValue)
3864
0
                {
3865
0
                    aosList.SetNameValue(pszKey, pszValue);
3866
0
                }
3867
0
            }
3868
0
        }
3869
0
    }
3870
3871
0
    pResult = GDALCreateReprojectionTransformerEx(
3872
0
        !oSrcSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oSrcSRS) : nullptr,
3873
0
        !oDstSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oDstSRS) : nullptr,
3874
0
        aosList.List());
3875
3876
0
    CPLFree(pszSourceWKT);
3877
0
    CPLFree(pszTargetWKT);
3878
3879
0
    return pResult;
3880
0
}
3881
3882
/************************************************************************/
3883
/* ==================================================================== */
3884
/*      Approximate transformer.                                        */
3885
/* ==================================================================== */
3886
/************************************************************************/
3887
3888
/************************************************************************/
3889
/*                  GDALCreateSimilarApproxTransformer()                */
3890
/************************************************************************/
3891
3892
static void *GDALCreateSimilarApproxTransformer(void *hTransformArg,
3893
                                                double dfSrcRatioX,
3894
                                                double dfSrcRatioY)
3895
0
{
3896
0
    VALIDATE_POINTER1(hTransformArg, "GDALCreateSimilarApproxTransformer",
3897
0
                      nullptr);
3898
3899
0
    GDALApproxTransformInfo *psInfo =
3900
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
3901
3902
0
    void *pBaseCBData = GDALCreateSimilarTransformer(psInfo->pBaseCBData,
3903
0
                                                     dfSrcRatioX, dfSrcRatioY);
3904
0
    if (pBaseCBData == nullptr)
3905
0
    {
3906
0
        return nullptr;
3907
0
    }
3908
3909
0
    GDALApproxTransformInfo *psClonedInfo =
3910
0
        static_cast<GDALApproxTransformInfo *>(GDALCreateApproxTransformer2(
3911
0
            psInfo->pfnBaseTransformer, pBaseCBData, psInfo->dfMaxErrorForward,
3912
0
            psInfo->dfMaxErrorReverse));
3913
0
    psClonedInfo->bOwnSubtransformer = TRUE;
3914
3915
0
    return psClonedInfo;
3916
0
}
3917
3918
/************************************************************************/
3919
/*                   GDALSerializeApproxTransformer()                   */
3920
/************************************************************************/
3921
3922
static CPLXMLNode *GDALSerializeApproxTransformer(void *pTransformArg)
3923
3924
0
{
3925
0
    CPLXMLNode *psTree;
3926
0
    GDALApproxTransformInfo *psInfo =
3927
0
        static_cast<GDALApproxTransformInfo *>(pTransformArg);
3928
3929
0
    psTree = CPLCreateXMLNode(nullptr, CXT_Element, "ApproxTransformer");
3930
3931
    /* -------------------------------------------------------------------- */
3932
    /*      Attach max error.                                               */
3933
    /* -------------------------------------------------------------------- */
3934
0
    if (psInfo->dfMaxErrorForward == psInfo->dfMaxErrorReverse)
3935
0
    {
3936
0
        CPLCreateXMLElementAndValue(
3937
0
            psTree, "MaxError",
3938
0
            CPLString().Printf("%g", psInfo->dfMaxErrorForward));
3939
0
    }
3940
0
    else
3941
0
    {
3942
0
        CPLCreateXMLElementAndValue(
3943
0
            psTree, "MaxErrorForward",
3944
0
            CPLString().Printf("%g", psInfo->dfMaxErrorForward));
3945
0
        CPLCreateXMLElementAndValue(
3946
0
            psTree, "MaxErrorReverse",
3947
0
            CPLString().Printf("%g", psInfo->dfMaxErrorReverse));
3948
0
    }
3949
3950
    /* -------------------------------------------------------------------- */
3951
    /*      Capture underlying transformer.                                 */
3952
    /* -------------------------------------------------------------------- */
3953
0
    CPLXMLNode *psTransformerContainer =
3954
0
        CPLCreateXMLNode(psTree, CXT_Element, "BaseTransformer");
3955
3956
0
    CPLXMLNode *psTransformer = GDALSerializeTransformer(
3957
0
        psInfo->pfnBaseTransformer, psInfo->pBaseCBData);
3958
0
    if (psTransformer != nullptr)
3959
0
        CPLAddXMLChild(psTransformerContainer, psTransformer);
3960
3961
0
    return psTree;
3962
0
}
3963
3964
/************************************************************************/
3965
/*                    GDALCreateApproxTransformer()                     */
3966
/************************************************************************/
3967
3968
/**
3969
 * Create an approximating transformer.
3970
 *
3971
 * This function creates a context for an approximated transformer.  Basically
3972
 * a high precision transformer is supplied as input and internally linear
3973
 * approximations are computed to generate results to within a defined
3974
 * precision.
3975
 *
3976
 * The approximation is actually done at the point where GDALApproxTransform()
3977
 * calls are made, and depend on the assumption that they are roughly linear.
3978
 * The first and last point passed in must be the extreme values and the
3979
 * intermediate values should describe a curve between the end points.  The
3980
 * approximator transforms and centers using the approximate transformer, and
3981
 * then compares the true middle transformed value to a linear approximation
3982
 * based on the end points.  If the error is within the supplied threshold then
3983
 * the end points are used to linearly approximate all the values otherwise the
3984
 * input points are split into two smaller sets, and the function is recursively
3985
 * called until a sufficiently small set of points is found that the linear
3986
 * approximation is OK, or that all the points are exactly computed.
3987
 *
3988
 * This function is very suitable for approximating transformation results
3989
 * from output pixel/line space to input coordinates for warpers that operate
3990
 * on one input scanline at a time.  Care should be taken using it in other
3991
 * circumstances as little internal validation is done in order to keep things
3992
 * fast.
3993
 *
3994
 * @param pfnBaseTransformer the high precision transformer which should be
3995
 * approximated.
3996
 * @param pBaseTransformArg the callback argument for the high precision
3997
 * transformer.
3998
 * @param dfMaxError the maximum cartesian error in the "output" space that
3999
 * is to be accepted in the linear approximation, evaluated as a Manhattan
4000
 * distance.
4001
 *
4002
 * @return callback pointer suitable for use with GDALApproxTransform().  It
4003
 * should be deallocated with GDALDestroyApproxTransformer().
4004
 */
4005
4006
void *GDALCreateApproxTransformer(GDALTransformerFunc pfnBaseTransformer,
4007
                                  void *pBaseTransformArg, double dfMaxError)
4008
4009
0
{
4010
0
    return GDALCreateApproxTransformer2(pfnBaseTransformer, pBaseTransformArg,
4011
0
                                        dfMaxError, dfMaxError);
4012
0
}
4013
4014
static void *
4015
GDALCreateApproxTransformer2(GDALTransformerFunc pfnBaseTransformer,
4016
                             void *pBaseTransformArg, double dfMaxErrorForward,
4017
                             double dfMaxErrorReverse)
4018
4019
0
{
4020
0
    GDALApproxTransformInfo *psATInfo = new GDALApproxTransformInfo;
4021
0
    psATInfo->pfnBaseTransformer = pfnBaseTransformer;
4022
0
    psATInfo->pBaseCBData = pBaseTransformArg;
4023
0
    psATInfo->dfMaxErrorForward = dfMaxErrorForward;
4024
0
    psATInfo->dfMaxErrorReverse = dfMaxErrorReverse;
4025
0
    psATInfo->bOwnSubtransformer = FALSE;
4026
4027
0
    memcpy(psATInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE,
4028
0
           strlen(GDAL_GTI2_SIGNATURE));
4029
0
    psATInfo->sTI.pszClassName = GDAL_APPROX_TRANSFORMER_CLASS_NAME;
4030
0
    psATInfo->sTI.pfnTransform = GDALApproxTransform;
4031
0
    psATInfo->sTI.pfnCleanup = GDALDestroyApproxTransformer;
4032
0
    psATInfo->sTI.pfnSerialize = GDALSerializeApproxTransformer;
4033
0
    psATInfo->sTI.pfnCreateSimilar = GDALCreateSimilarApproxTransformer;
4034
4035
0
    return psATInfo;
4036
0
}
4037
4038
/************************************************************************/
4039
/*              GDALApproxTransformerOwnsSubtransformer()               */
4040
/************************************************************************/
4041
4042
/** Set bOwnSubtransformer flag */
4043
void GDALApproxTransformerOwnsSubtransformer(void *pCBData, int bOwnFlag)
4044
4045
0
{
4046
0
    GDALApproxTransformInfo *psATInfo =
4047
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4048
4049
0
    psATInfo->bOwnSubtransformer = bOwnFlag;
4050
0
}
4051
4052
/************************************************************************/
4053
/*                    GDALDestroyApproxTransformer()                    */
4054
/************************************************************************/
4055
4056
/**
4057
 * Cleanup approximate transformer.
4058
 *
4059
 * Deallocates the resources allocated by GDALCreateApproxTransformer().
4060
 *
4061
 * @param pCBData callback data originally returned by
4062
 * GDALCreateApproxTransformer().
4063
 */
4064
4065
void GDALDestroyApproxTransformer(void *pCBData)
4066
4067
0
{
4068
0
    if (pCBData == nullptr)
4069
0
        return;
4070
4071
0
    GDALApproxTransformInfo *psATInfo =
4072
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4073
4074
0
    if (psATInfo->bOwnSubtransformer)
4075
0
        GDALDestroyTransformer(psATInfo->pBaseCBData);
4076
4077
0
    delete psATInfo;
4078
0
}
4079
4080
/************************************************************************/
4081
/*                  GDALRefreshApproxTransformer()                      */
4082
/************************************************************************/
4083
4084
void GDALRefreshApproxTransformer(void *hTransformArg)
4085
0
{
4086
0
    GDALApproxTransformInfo *psInfo =
4087
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
4088
4089
0
    if (GDALIsTransformer(psInfo->pBaseCBData,
4090
0
                          GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4091
0
    {
4092
0
        GDALRefreshGenImgProjTransformer(psInfo->pBaseCBData);
4093
0
    }
4094
0
}
4095
4096
/************************************************************************/
4097
/*                      GDALApproxTransformInternal()                   */
4098
/************************************************************************/
4099
4100
static int GDALApproxTransformInternal(void *pCBData, int bDstToSrc,
4101
                                       int nPoints, double *x, double *y,
4102
                                       double *z, int *panSuccess,
4103
                                       // SME = Start, Middle, End.
4104
                                       const double xSMETransformed[3],
4105
                                       const double ySMETransformed[3],
4106
                                       const double zSMETransformed[3])
4107
0
{
4108
0
    GDALApproxTransformInfo *psATInfo =
4109
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4110
0
    const int nMiddle = (nPoints - 1) / 2;
4111
4112
#ifdef notdef_sanify_check
4113
    {
4114
        double x2[3] = {x[0], x[nMiddle], x[nPoints - 1]};
4115
        double y2[3] = {y[0], y[nMiddle], y[nPoints - 1]};
4116
        double z2[3] = {z[0], z[nMiddle], z[nPoints - 1]};
4117
        int anSuccess2[3] = {};
4118
4119
        const int bSuccess = psATInfo->pfnBaseTransformer(
4120
            psATInfo->pBaseCBData, bDstToSrc, 3, x2, y2, z2, anSuccess2);
4121
        CPLAssert(bSuccess);
4122
        CPLAssert(anSuccess2[0]);
4123
        CPLAssert(anSuccess2[1]);
4124
        CPLAssert(anSuccess2[2]);
4125
        CPLAssert(x2[0] == xSMETransformed[0]);
4126
        CPLAssert(y2[0] == ySMETransformed[0]);
4127
        CPLAssert(z2[0] == zSMETransformed[0]);
4128
        CPLAssert(x2[1] == xSMETransformed[1]);
4129
        CPLAssert(y2[1] == ySMETransformed[1]);
4130
        CPLAssert(z2[1] == zSMETransformed[1]);
4131
        CPLAssert(x2[2] == xSMETransformed[2]);
4132
        CPLAssert(y2[2] == ySMETransformed[2]);
4133
        CPLAssert(z2[2] == zSMETransformed[2]);
4134
    }
4135
#endif
4136
4137
#ifdef DEBUG_APPROX_TRANSFORMER
4138
    fprintf(stderr, "start (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4139
            x[0], y[0], xSMETransformed[0], ySMETransformed[0]);
4140
    fprintf(stderr, "middle (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4141
            x[nMiddle], y[nMiddle], xSMETransformed[1], ySMETransformed[1]);
4142
    fprintf(stderr, "end (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4143
            x[nPoints - 1], y[nPoints - 1], xSMETransformed[2],
4144
            ySMETransformed[2]);
4145
#endif
4146
4147
    /* -------------------------------------------------------------------- */
4148
    /*      Is the error at the middle acceptable relative to an            */
4149
    /*      interpolation of the middle position?                           */
4150
    /* -------------------------------------------------------------------- */
4151
0
    const double dfDeltaX =
4152
0
        (xSMETransformed[2] - xSMETransformed[0]) / (x[nPoints - 1] - x[0]);
4153
0
    const double dfDeltaY =
4154
0
        (ySMETransformed[2] - ySMETransformed[0]) / (x[nPoints - 1] - x[0]);
4155
0
    const double dfDeltaZ =
4156
0
        (zSMETransformed[2] - zSMETransformed[0]) / (x[nPoints - 1] - x[0]);
4157
4158
0
    const double dfError =
4159
0
        fabs((xSMETransformed[0] + dfDeltaX * (x[nMiddle] - x[0])) -
4160
0
             xSMETransformed[1]) +
4161
0
        fabs((ySMETransformed[0] + dfDeltaY * (x[nMiddle] - x[0])) -
4162
0
             ySMETransformed[1]);
4163
4164
0
    const double dfMaxError =
4165
0
        (bDstToSrc) ? psATInfo->dfMaxErrorReverse : psATInfo->dfMaxErrorForward;
4166
0
    if (dfError > dfMaxError)
4167
0
    {
4168
#if DEBUG_VERBOSE
4169
        CPLDebug("GDAL",
4170
                 "ApproxTransformer - "
4171
                 "error %g over threshold %g, subdivide %d points.",
4172
                 dfError, dfMaxError, nPoints);
4173
#endif
4174
4175
0
        double xMiddle[3] = {x[(nMiddle - 1) / 2], x[nMiddle - 1],
4176
0
                             x[nMiddle + (nPoints - nMiddle - 1) / 2]};
4177
0
        double yMiddle[3] = {y[(nMiddle - 1) / 2], y[nMiddle - 1],
4178
0
                             y[nMiddle + (nPoints - nMiddle - 1) / 2]};
4179
0
        double zMiddle[3] = {z[(nMiddle - 1) / 2], z[nMiddle - 1],
4180
0
                             z[nMiddle + (nPoints - nMiddle - 1) / 2]};
4181
4182
0
        const bool bUseBaseTransformForHalf1 =
4183
0
            nMiddle <= 5 || y[0] != y[nMiddle - 1] ||
4184
0
            y[0] != y[(nMiddle - 1) / 2] || x[0] == x[nMiddle - 1] ||
4185
0
            x[0] == x[(nMiddle - 1) / 2];
4186
0
        const bool bUseBaseTransformForHalf2 =
4187
0
            nPoints - nMiddle <= 5 || y[nMiddle] != y[nPoints - 1] ||
4188
0
            y[nMiddle] != y[nMiddle + (nPoints - nMiddle - 1) / 2] ||
4189
0
            x[nMiddle] == x[nPoints - 1] ||
4190
0
            x[nMiddle] == x[nMiddle + (nPoints - nMiddle - 1) / 2];
4191
4192
0
        int anSuccess2[3] = {};
4193
0
        int bSuccess = FALSE;
4194
0
        if (!bUseBaseTransformForHalf1 && !bUseBaseTransformForHalf2)
4195
0
            bSuccess = psATInfo->pfnBaseTransformer(
4196
0
                psATInfo->pBaseCBData, bDstToSrc, 3, xMiddle, yMiddle, zMiddle,
4197
0
                anSuccess2);
4198
0
        else if (!bUseBaseTransformForHalf1)
4199
0
        {
4200
0
            bSuccess = psATInfo->pfnBaseTransformer(
4201
0
                psATInfo->pBaseCBData, bDstToSrc, 2, xMiddle, yMiddle, zMiddle,
4202
0
                anSuccess2);
4203
0
            anSuccess2[2] = TRUE;
4204
0
        }
4205
0
        else if (!bUseBaseTransformForHalf2)
4206
0
        {
4207
0
            bSuccess = psATInfo->pfnBaseTransformer(
4208
0
                psATInfo->pBaseCBData, bDstToSrc, 1, xMiddle + 2, yMiddle + 2,
4209
0
                zMiddle + 2, anSuccess2 + 2);
4210
0
            anSuccess2[0] = TRUE;
4211
0
            anSuccess2[1] = TRUE;
4212
0
        }
4213
4214
0
        if (!bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2])
4215
0
        {
4216
0
            bSuccess = psATInfo->pfnBaseTransformer(
4217
0
                psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1,
4218
0
                z + 1, panSuccess + 1);
4219
0
            bSuccess &= psATInfo->pfnBaseTransformer(
4220
0
                psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2,
4221
0
                x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1,
4222
0
                panSuccess + nMiddle + 1);
4223
4224
0
            x[0] = xSMETransformed[0];
4225
0
            y[0] = ySMETransformed[0];
4226
0
            z[0] = zSMETransformed[0];
4227
0
            panSuccess[0] = TRUE;
4228
0
            x[nMiddle] = xSMETransformed[1];
4229
0
            y[nMiddle] = ySMETransformed[1];
4230
0
            z[nMiddle] = zSMETransformed[1];
4231
0
            panSuccess[nMiddle] = TRUE;
4232
0
            x[nPoints - 1] = xSMETransformed[2];
4233
0
            y[nPoints - 1] = ySMETransformed[2];
4234
0
            z[nPoints - 1] = zSMETransformed[2];
4235
0
            panSuccess[nPoints - 1] = TRUE;
4236
0
            return bSuccess;
4237
0
        }
4238
4239
0
        double x2[3] = {};
4240
0
        double y2[3] = {};
4241
0
        double z2[3] = {};
4242
0
        if (!bUseBaseTransformForHalf1)
4243
0
        {
4244
0
            x2[0] = xSMETransformed[0];
4245
0
            y2[0] = ySMETransformed[0];
4246
0
            z2[0] = zSMETransformed[0];
4247
0
            x2[1] = xMiddle[0];
4248
0
            y2[1] = yMiddle[0];
4249
0
            z2[1] = zMiddle[0];
4250
0
            x2[2] = xMiddle[1];
4251
0
            y2[2] = yMiddle[1];
4252
0
            z2[2] = zMiddle[1];
4253
4254
0
            bSuccess = GDALApproxTransformInternal(
4255
0
                psATInfo, bDstToSrc, nMiddle, x, y, z, panSuccess, x2, y2, z2);
4256
0
        }
4257
0
        else
4258
0
        {
4259
0
            bSuccess = psATInfo->pfnBaseTransformer(
4260
0
                psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1,
4261
0
                z + 1, panSuccess + 1);
4262
0
            x[0] = xSMETransformed[0];
4263
0
            y[0] = ySMETransformed[0];
4264
0
            z[0] = zSMETransformed[0];
4265
0
            panSuccess[0] = TRUE;
4266
0
        }
4267
4268
0
        if (!bSuccess)
4269
0
            return FALSE;
4270
4271
0
        if (!bUseBaseTransformForHalf2)
4272
0
        {
4273
0
            x2[0] = xSMETransformed[1];
4274
0
            y2[0] = ySMETransformed[1];
4275
0
            z2[0] = zSMETransformed[1];
4276
0
            x2[1] = xMiddle[2];
4277
0
            y2[1] = yMiddle[2];
4278
0
            z2[1] = zMiddle[2];
4279
0
            x2[2] = xSMETransformed[2];
4280
0
            y2[2] = ySMETransformed[2];
4281
0
            z2[2] = zSMETransformed[2];
4282
4283
0
            bSuccess = GDALApproxTransformInternal(
4284
0
                psATInfo, bDstToSrc, nPoints - nMiddle, x + nMiddle,
4285
0
                y + nMiddle, z + nMiddle, panSuccess + nMiddle, x2, y2, z2);
4286
0
        }
4287
0
        else
4288
0
        {
4289
0
            bSuccess = psATInfo->pfnBaseTransformer(
4290
0
                psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2,
4291
0
                x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1,
4292
0
                panSuccess + nMiddle + 1);
4293
4294
0
            x[nMiddle] = xSMETransformed[1];
4295
0
            y[nMiddle] = ySMETransformed[1];
4296
0
            z[nMiddle] = zSMETransformed[1];
4297
0
            panSuccess[nMiddle] = TRUE;
4298
0
            x[nPoints - 1] = xSMETransformed[2];
4299
0
            y[nPoints - 1] = ySMETransformed[2];
4300
0
            z[nPoints - 1] = zSMETransformed[2];
4301
0
            panSuccess[nPoints - 1] = TRUE;
4302
0
        }
4303
4304
0
        if (!bSuccess)
4305
0
            return FALSE;
4306
4307
0
        return TRUE;
4308
0
    }
4309
4310
    /* -------------------------------------------------------------------- */
4311
    /*      Error is OK since this is just used to compute output bounds    */
4312
    /*      of newly created file for gdalwarper.  So just use affine       */
4313
    /*      approximation of the reverse transform.  Eventually we          */
4314
    /*      should implement iterative searching to find a result within    */
4315
    /*      our error threshold.                                            */
4316
    /*      NOTE: the above comment is not true: gdalwarp uses approximator */
4317
    /*      also to compute the source pixel of each target pixel.          */
4318
    /* -------------------------------------------------------------------- */
4319
0
    for (int i = nPoints - 1; i >= 0; i--)
4320
0
    {
4321
#ifdef check_error
4322
        double xtemp = x[i];
4323
        double ytemp = y[i];
4324
        double ztemp = z[i];
4325
        double x_ori = xtemp;
4326
        double y_ori = ytemp;
4327
        int btemp = FALSE;
4328
        psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 1,
4329
                                     &xtemp, &ytemp, &ztemp, &btemp);
4330
#endif
4331
0
        const double dfDist = (x[i] - x[0]);
4332
0
        x[i] = xSMETransformed[0] + dfDeltaX * dfDist;
4333
0
        y[i] = ySMETransformed[0] + dfDeltaY * dfDist;
4334
0
        z[i] = zSMETransformed[0] + dfDeltaZ * dfDist;
4335
#ifdef check_error
4336
        const double dfError2 = fabs(x[i] - xtemp) + fabs(y[i] - ytemp);
4337
        if (dfError2 > 4 /*10 * dfMaxError*/)
4338
        {
4339
            /*ok*/ printf("Error = %f on (%f, %f)\n", dfError2, x_ori, y_ori);
4340
        }
4341
#endif
4342
0
        panSuccess[i] = TRUE;
4343
0
    }
4344
4345
0
    return TRUE;
4346
0
}
4347
4348
/************************************************************************/
4349
/*                        GDALApproxTransform()                         */
4350
/************************************************************************/
4351
4352
/**
4353
 * Perform approximate transformation.
4354
 *
4355
 * Actually performs the approximate transformation described in
4356
 * GDALCreateApproxTransformer().  This function matches the
4357
 * GDALTransformerFunc() signature.  Details of the arguments are described
4358
 * there.
4359
 */
4360
4361
int GDALApproxTransform(void *pCBData, int bDstToSrc, int nPoints, double *x,
4362
                        double *y, double *z, int *panSuccess)
4363
4364
0
{
4365
0
    GDALApproxTransformInfo *psATInfo =
4366
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4367
0
    double x2[3] = {};
4368
0
    double y2[3] = {};
4369
0
    double z2[3] = {};
4370
0
    int anSuccess2[3] = {};
4371
0
    int bSuccess;
4372
4373
0
    const int nMiddle = (nPoints - 1) / 2;
4374
4375
    /* -------------------------------------------------------------------- */
4376
    /*      Bail if our preconditions are not met, or if error is not       */
4377
    /*      acceptable.                                                     */
4378
    /* -------------------------------------------------------------------- */
4379
0
    int bRet = FALSE;
4380
0
    if (y[0] != y[nPoints - 1] || y[0] != y[nMiddle] ||
4381
0
        x[0] == x[nPoints - 1] || x[0] == x[nMiddle] ||
4382
0
        (psATInfo->dfMaxErrorForward == 0.0 &&
4383
0
         psATInfo->dfMaxErrorReverse == 0.0) ||
4384
0
        nPoints <= 5)
4385
0
    {
4386
0
        bRet = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc,
4387
0
                                            nPoints, x, y, z, panSuccess);
4388
0
        goto end;
4389
0
    }
4390
4391
    /* -------------------------------------------------------------------- */
4392
    /*      Transform first, last and middle point.                         */
4393
    /* -------------------------------------------------------------------- */
4394
0
    x2[0] = x[0];
4395
0
    y2[0] = y[0];
4396
0
    z2[0] = z[0];
4397
0
    x2[1] = x[nMiddle];
4398
0
    y2[1] = y[nMiddle];
4399
0
    z2[1] = z[nMiddle];
4400
0
    x2[2] = x[nPoints - 1];
4401
0
    y2[2] = y[nPoints - 1];
4402
0
    z2[2] = z[nPoints - 1];
4403
4404
0
    bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 3,
4405
0
                                            x2, y2, z2, anSuccess2);
4406
0
    if (!bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2])
4407
0
    {
4408
0
        bRet = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc,
4409
0
                                            nPoints, x, y, z, panSuccess);
4410
0
        goto end;
4411
0
    }
4412
4413
0
    bRet = GDALApproxTransformInternal(pCBData, bDstToSrc, nPoints, x, y, z,
4414
0
                                       panSuccess, x2, y2, z2);
4415
4416
0
end:
4417
#ifdef DEBUG_APPROX_TRANSFORMER
4418
    for (int i = 0; i < nPoints; i++)
4419
        fprintf(stderr, "[%d] (%.10f,%.10f) %d\n", /*ok*/
4420
                i, x[i], y[i], panSuccess[i]);
4421
#endif
4422
4423
0
    return bRet;
4424
0
}
4425
4426
/************************************************************************/
4427
/*                  GDALDeserializeApproxTransformer()                  */
4428
/************************************************************************/
4429
4430
static void *GDALDeserializeApproxTransformer(CPLXMLNode *psTree)
4431
4432
0
{
4433
0
    double dfMaxErrorForward = 0.25;
4434
0
    double dfMaxErrorReverse = 0.25;
4435
0
    const char *pszMaxError = CPLGetXMLValue(psTree, "MaxError", nullptr);
4436
0
    if (pszMaxError != nullptr)
4437
0
    {
4438
0
        dfMaxErrorForward = CPLAtof(pszMaxError);
4439
0
        dfMaxErrorReverse = dfMaxErrorForward;
4440
0
    }
4441
0
    const char *pszMaxErrorForward =
4442
0
        CPLGetXMLValue(psTree, "MaxErrorForward", nullptr);
4443
0
    if (pszMaxErrorForward != nullptr)
4444
0
    {
4445
0
        dfMaxErrorForward = CPLAtof(pszMaxErrorForward);
4446
0
    }
4447
0
    const char *pszMaxErrorReverse =
4448
0
        CPLGetXMLValue(psTree, "MaxErrorReverse", nullptr);
4449
0
    if (pszMaxErrorReverse != nullptr)
4450
0
    {
4451
0
        dfMaxErrorReverse = CPLAtof(pszMaxErrorReverse);
4452
0
    }
4453
4454
0
    GDALTransformerFunc pfnBaseTransform = nullptr;
4455
0
    void *pBaseCBData = nullptr;
4456
4457
0
    CPLXMLNode *psContainer = CPLGetXMLNode(psTree, "BaseTransformer");
4458
4459
0
    if (psContainer != nullptr && psContainer->psChild != nullptr)
4460
0
    {
4461
0
        GDALDeserializeTransformer(psContainer->psChild, &pfnBaseTransform,
4462
0
                                   &pBaseCBData);
4463
0
    }
4464
4465
0
    if (pfnBaseTransform == nullptr)
4466
0
    {
4467
0
        CPLError(CE_Failure, CPLE_AppDefined,
4468
0
                 "Cannot get base transform for approx transformer.");
4469
0
        return nullptr;
4470
0
    }
4471
4472
0
    void *pApproxCBData = GDALCreateApproxTransformer2(
4473
0
        pfnBaseTransform, pBaseCBData, dfMaxErrorForward, dfMaxErrorReverse);
4474
0
    GDALApproxTransformerOwnsSubtransformer(pApproxCBData, TRUE);
4475
4476
0
    return pApproxCBData;
4477
0
}
4478
4479
/************************************************************************/
4480
/*                 GDALTransformLonLatToDestApproxTransformer()         */
4481
/************************************************************************/
4482
4483
int GDALTransformLonLatToDestApproxTransformer(void *hTransformArg,
4484
                                               double *pdfX, double *pdfY)
4485
0
{
4486
0
    GDALApproxTransformInfo *psInfo =
4487
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
4488
4489
0
    if (GDALIsTransformer(psInfo->pBaseCBData,
4490
0
                          GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4491
0
    {
4492
0
        return GDALTransformLonLatToDestGenImgProjTransformer(
4493
0
            psInfo->pBaseCBData, pdfX, pdfY);
4494
0
    }
4495
0
    return false;
4496
0
}
4497
4498
/************************************************************************/
4499
/*                       GDALApplyGeoTransform()                        */
4500
/************************************************************************/
4501
4502
/**
4503
 * Apply GeoTransform to x/y coordinate.
4504
 *
4505
 * Applies the following computation, converting a (pixel, line) coordinate
4506
 * into a georeferenced (geo_x, geo_y) location.
4507
 * \code{.c}
4508
 *  *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1]
4509
 *                                 + dfLine  * padfGeoTransform[2];
4510
 *  *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4]
4511
 *                                 + dfLine  * padfGeoTransform[5];
4512
 * \endcode
4513
 *
4514
 * @param padfGeoTransform Six coefficient GeoTransform to apply.
4515
 * @param dfPixel Input pixel position.
4516
 * @param dfLine Input line position.
4517
 * @param pdfGeoX output location where geo_x (easting/longitude)
4518
 * location is placed.
4519
 * @param pdfGeoY output location where geo_y (northing/latitude)
4520
 * location is placed.
4521
 */
4522
4523
void CPL_STDCALL GDALApplyGeoTransform(const double *padfGeoTransform,
4524
                                       double dfPixel, double dfLine,
4525
                                       double *pdfGeoX, double *pdfGeoY)
4526
0
{
4527
0
    *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1] +
4528
0
               dfLine * padfGeoTransform[2];
4529
0
    *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4] +
4530
0
               dfLine * padfGeoTransform[5];
4531
0
}
4532
4533
/************************************************************************/
4534
/*                        GDALInvGeoTransform()                         */
4535
/************************************************************************/
4536
4537
/**
4538
 * Invert Geotransform.
4539
 *
4540
 * This function will invert a standard 3x2 set of GeoTransform coefficients.
4541
 * This converts the equation from being pixel to geo to being geo to pixel.
4542
 *
4543
 * @param gt_in Input geotransform (six doubles - unaltered).
4544
 * @param gt_out Output geotransform (six doubles - updated).
4545
 *
4546
 * @return TRUE on success or FALSE if the equation is uninvertable.
4547
 */
4548
4549
int CPL_STDCALL GDALInvGeoTransform(const double *gt_in, double *gt_out)
4550
4551
0
{
4552
    // Special case - no rotation - to avoid computing determinate
4553
    // and potential precision issues.
4554
0
    if (gt_in[2] == 0.0 && gt_in[4] == 0.0 && gt_in[1] != 0.0 &&
4555
0
        gt_in[5] != 0.0)
4556
0
    {
4557
        /*X = gt_in[0] + x * gt_in[1]
4558
          Y = gt_in[3] + y * gt_in[5]
4559
          -->
4560
          x = -gt_in[0] / gt_in[1] + (1 / gt_in[1]) * X
4561
          y = -gt_in[3] / gt_in[5] + (1 / gt_in[5]) * Y
4562
        */
4563
0
        gt_out[0] = -gt_in[0] / gt_in[1];
4564
0
        gt_out[1] = 1.0 / gt_in[1];
4565
0
        gt_out[2] = 0.0;
4566
0
        gt_out[3] = -gt_in[3] / gt_in[5];
4567
0
        gt_out[4] = 0.0;
4568
0
        gt_out[5] = 1.0 / gt_in[5];
4569
0
        return 1;
4570
0
    }
4571
4572
    // Assume a 3rd row that is [1 0 0].
4573
4574
    // Compute determinate.
4575
4576
0
    const double det = gt_in[1] * gt_in[5] - gt_in[2] * gt_in[4];
4577
0
    const double magnitude = std::max(std::max(fabs(gt_in[1]), fabs(gt_in[2])),
4578
0
                                      std::max(fabs(gt_in[4]), fabs(gt_in[5])));
4579
4580
0
    if (fabs(det) <= 1e-10 * magnitude * magnitude)
4581
0
        return 0;
4582
4583
0
    const double inv_det = 1.0 / det;
4584
4585
    // Compute adjoint, and divide by determinate.
4586
4587
0
    gt_out[1] = gt_in[5] * inv_det;
4588
0
    gt_out[4] = -gt_in[4] * inv_det;
4589
4590
0
    gt_out[2] = -gt_in[2] * inv_det;
4591
0
    gt_out[5] = gt_in[1] * inv_det;
4592
4593
0
    gt_out[0] = (gt_in[2] * gt_in[3] - gt_in[0] * gt_in[5]) * inv_det;
4594
0
    gt_out[3] = (-gt_in[1] * gt_in[3] + gt_in[0] * gt_in[4]) * inv_det;
4595
4596
0
    return 1;
4597
0
}
4598
4599
/************************************************************************/
4600
/*                      GDALSerializeTransformer()                      */
4601
/************************************************************************/
4602
4603
CPLXMLNode *GDALSerializeTransformer(GDALTransformerFunc /* pfnFunc */,
4604
                                     void *pTransformArg)
4605
0
{
4606
0
    VALIDATE_POINTER1(pTransformArg, "GDALSerializeTransformer", nullptr);
4607
4608
0
    GDALTransformerInfo *psInfo =
4609
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4610
4611
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4612
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4613
0
    {
4614
0
        CPLError(CE_Failure, CPLE_AppDefined,
4615
0
                 "Attempt to serialize non-GTI2 transformer.");
4616
0
        return nullptr;
4617
0
    }
4618
0
    else if (psInfo->pfnSerialize == nullptr)
4619
0
    {
4620
0
        CPLError(CE_Failure, CPLE_AppDefined,
4621
0
                 "No serialization function available for this transformer.");
4622
0
        return nullptr;
4623
0
    }
4624
4625
0
    return psInfo->pfnSerialize(pTransformArg);
4626
0
}
4627
4628
/************************************************************************/
4629
/*                  GDALRegisterTransformDeserializer()                 */
4630
/************************************************************************/
4631
4632
static CPLList *psListDeserializer = nullptr;
4633
static CPLMutex *hDeserializerMutex = nullptr;
4634
4635
typedef struct
4636
{
4637
    char *pszTransformName;
4638
    GDALTransformerFunc pfnTransformerFunc;
4639
    GDALTransformDeserializeFunc pfnDeserializeFunc;
4640
} TransformDeserializerInfo;
4641
4642
void *GDALRegisterTransformDeserializer(
4643
    const char *pszTransformName, GDALTransformerFunc pfnTransformerFunc,
4644
    GDALTransformDeserializeFunc pfnDeserializeFunc)
4645
0
{
4646
0
    TransformDeserializerInfo *psInfo =
4647
0
        static_cast<TransformDeserializerInfo *>(
4648
0
            CPLMalloc(sizeof(TransformDeserializerInfo)));
4649
0
    psInfo->pszTransformName = CPLStrdup(pszTransformName);
4650
0
    psInfo->pfnTransformerFunc = pfnTransformerFunc;
4651
0
    psInfo->pfnDeserializeFunc = pfnDeserializeFunc;
4652
4653
0
    CPLMutexHolderD(&hDeserializerMutex);
4654
0
    psListDeserializer = CPLListInsert(psListDeserializer, psInfo, 0);
4655
4656
0
    return psInfo;
4657
0
}
4658
4659
/************************************************************************/
4660
/*                GDALUnregisterTransformDeserializer()                 */
4661
/************************************************************************/
4662
4663
void GDALUnregisterTransformDeserializer(void *pData)
4664
0
{
4665
0
    CPLMutexHolderD(&hDeserializerMutex);
4666
0
    CPLList *psList = psListDeserializer;
4667
0
    CPLList *psLast = nullptr;
4668
0
    while (psList)
4669
0
    {
4670
0
        if (psList->pData == pData)
4671
0
        {
4672
0
            TransformDeserializerInfo *psInfo =
4673
0
                static_cast<TransformDeserializerInfo *>(pData);
4674
0
            CPLFree(psInfo->pszTransformName);
4675
0
            CPLFree(pData);
4676
0
            if (psLast)
4677
0
                psLast->psNext = psList->psNext;
4678
0
            else
4679
0
                psListDeserializer = nullptr;
4680
0
            CPLFree(psList);
4681
0
            break;
4682
0
        }
4683
0
        psLast = psList;
4684
0
        psList = psList->psNext;
4685
0
    }
4686
0
}
4687
4688
/************************************************************************/
4689
/*                GDALUnregisterTransformDeserializer()                 */
4690
/************************************************************************/
4691
4692
void GDALCleanupTransformDeserializerMutex()
4693
0
{
4694
0
    if (hDeserializerMutex != nullptr)
4695
0
    {
4696
0
        CPLDestroyMutex(hDeserializerMutex);
4697
0
        hDeserializerMutex = nullptr;
4698
0
    }
4699
0
}
4700
4701
/************************************************************************/
4702
/*                     GDALDeserializeTransformer()                     */
4703
/************************************************************************/
4704
4705
CPLErr GDALDeserializeTransformer(CPLXMLNode *psTree,
4706
                                  GDALTransformerFunc *ppfnFunc,
4707
                                  void **ppTransformArg)
4708
4709
0
{
4710
0
    *ppfnFunc = nullptr;
4711
0
    *ppTransformArg = nullptr;
4712
4713
0
    CPLErrorReset();
4714
4715
0
    if (psTree == nullptr || psTree->eType != CXT_Element)
4716
0
        CPLError(CE_Failure, CPLE_AppDefined,
4717
0
                 "Malformed element in GDALDeserializeTransformer");
4718
0
    else if (EQUAL(psTree->pszValue, "GenImgProjTransformer"))
4719
0
    {
4720
0
        *ppfnFunc = GDALGenImgProjTransform;
4721
0
        *ppTransformArg = GDALDeserializeGenImgProjTransformer(psTree);
4722
0
    }
4723
0
    else if (EQUAL(psTree->pszValue, "ReprojectionTransformer"))
4724
0
    {
4725
0
        *ppfnFunc = GDALReprojectionTransform;
4726
0
        *ppTransformArg = GDALDeserializeReprojectionTransformer(psTree);
4727
0
    }
4728
0
    else if (EQUAL(psTree->pszValue, "GCPTransformer"))
4729
0
    {
4730
0
        *ppfnFunc = GDALGCPTransform;
4731
0
        *ppTransformArg = GDALDeserializeGCPTransformer(psTree);
4732
0
    }
4733
0
    else if (EQUAL(psTree->pszValue, "TPSTransformer"))
4734
0
    {
4735
0
        *ppfnFunc = GDALTPSTransform;
4736
0
        *ppTransformArg = GDALDeserializeTPSTransformer(psTree);
4737
0
    }
4738
0
    else if (EQUAL(psTree->pszValue, "GeoLocTransformer"))
4739
0
    {
4740
0
        *ppfnFunc = GDALGeoLocTransform;
4741
0
        *ppTransformArg = GDALDeserializeGeoLocTransformer(psTree);
4742
0
    }
4743
0
    else if (EQUAL(psTree->pszValue, "RPCTransformer"))
4744
0
    {
4745
0
        *ppfnFunc = GDALRPCTransform;
4746
0
        *ppTransformArg = GDALDeserializeRPCTransformer(psTree);
4747
0
    }
4748
0
    else if (EQUAL(psTree->pszValue, "ApproxTransformer"))
4749
0
    {
4750
0
        *ppfnFunc = GDALApproxTransform;
4751
0
        *ppTransformArg = GDALDeserializeApproxTransformer(psTree);
4752
0
    }
4753
0
    else if (EQUAL(psTree->pszValue, "HomographyTransformer"))
4754
0
    {
4755
0
        *ppfnFunc = GDALHomographyTransform;
4756
0
        *ppTransformArg = GDALDeserializeHomographyTransformer(psTree);
4757
0
    }
4758
0
    else
4759
0
    {
4760
0
        GDALTransformDeserializeFunc pfnDeserializeFunc = nullptr;
4761
0
        {
4762
0
            CPLMutexHolderD(&hDeserializerMutex);
4763
0
            CPLList *psList = psListDeserializer;
4764
0
            while (psList)
4765
0
            {
4766
0
                TransformDeserializerInfo *psInfo =
4767
0
                    static_cast<TransformDeserializerInfo *>(psList->pData);
4768
0
                if (strcmp(psInfo->pszTransformName, psTree->pszValue) == 0)
4769
0
                {
4770
0
                    *ppfnFunc = psInfo->pfnTransformerFunc;
4771
0
                    pfnDeserializeFunc = psInfo->pfnDeserializeFunc;
4772
0
                    break;
4773
0
                }
4774
0
                psList = psList->psNext;
4775
0
            }
4776
0
        }
4777
4778
0
        if (pfnDeserializeFunc != nullptr)
4779
0
        {
4780
0
            *ppTransformArg = pfnDeserializeFunc(psTree);
4781
0
        }
4782
0
        else
4783
0
        {
4784
0
            CPLError(CE_Failure, CPLE_AppDefined,
4785
0
                     "Unrecognized element '%s' GDALDeserializeTransformer",
4786
0
                     psTree->pszValue);
4787
0
        }
4788
0
    }
4789
4790
0
    return CPLGetLastErrorType();
4791
0
}
4792
4793
/************************************************************************/
4794
/*                       GDALDestroyTransformer()                       */
4795
/************************************************************************/
4796
4797
void GDALDestroyTransformer(void *pTransformArg)
4798
4799
0
{
4800
0
    if (pTransformArg == nullptr)
4801
0
        return;
4802
4803
0
    GDALTransformerInfo *psInfo =
4804
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4805
4806
0
    if (memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4807
0
               strlen(GDAL_GTI2_SIGNATURE)) != 0)
4808
0
    {
4809
0
        CPLError(CE_Failure, CPLE_AppDefined,
4810
0
                 "Attempt to destroy non-GTI2 transformer.");
4811
0
        return;
4812
0
    }
4813
4814
0
    psInfo->pfnCleanup(pTransformArg);
4815
0
}
4816
4817
/************************************************************************/
4818
/*                         GDALUseTransformer()                         */
4819
/************************************************************************/
4820
4821
int GDALUseTransformer(void *pTransformArg, int bDstToSrc, int nPointCount,
4822
                       double *x, double *y, double *z, int *panSuccess)
4823
0
{
4824
0
    GDALTransformerInfo *psInfo =
4825
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4826
4827
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4828
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4829
0
    {
4830
0
        CPLError(CE_Failure, CPLE_AppDefined,
4831
0
                 "Attempt to use non-GTI2 transformer.");
4832
0
        return FALSE;
4833
0
    }
4834
4835
0
    return psInfo->pfnTransform(pTransformArg, bDstToSrc, nPointCount, x, y, z,
4836
0
                                panSuccess);
4837
0
}
4838
4839
/************************************************************************/
4840
/*                        GDALCloneTransformer()                        */
4841
/************************************************************************/
4842
4843
void *GDALCloneTransformer(void *pTransformArg)
4844
0
{
4845
0
    GDALTransformerInfo *psInfo =
4846
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4847
4848
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4849
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4850
0
    {
4851
0
        CPLError(CE_Failure, CPLE_AppDefined,
4852
0
                 "Attempt to clone non-GTI2 transformer.");
4853
0
        return nullptr;
4854
0
    }
4855
4856
0
    if (psInfo->pfnCreateSimilar != nullptr)
4857
0
    {
4858
0
        return psInfo->pfnCreateSimilar(psInfo, 1.0, 1.0);
4859
0
    }
4860
4861
0
    if (psInfo->pfnSerialize == nullptr)
4862
0
    {
4863
0
        CPLError(CE_Failure, CPLE_AppDefined,
4864
0
                 "No serialization function available for this transformer.");
4865
0
        return nullptr;
4866
0
    }
4867
4868
0
    CPLXMLNode *pSerialized = psInfo->pfnSerialize(pTransformArg);
4869
0
    if (pSerialized == nullptr)
4870
0
        return nullptr;
4871
0
    GDALTransformerFunc pfnTransformer = nullptr;
4872
0
    void *pClonedTransformArg = nullptr;
4873
0
    if (GDALDeserializeTransformer(pSerialized, &pfnTransformer,
4874
0
                                   &pClonedTransformArg) != CE_None)
4875
0
    {
4876
0
        CPLDestroyXMLNode(pSerialized);
4877
0
        CPLFree(pClonedTransformArg);
4878
0
        return nullptr;
4879
0
    }
4880
4881
0
    CPLDestroyXMLNode(pSerialized);
4882
0
    return pClonedTransformArg;
4883
0
}
4884
4885
/************************************************************************/
4886
/*                   GDALCreateSimilarTransformer()                     */
4887
/************************************************************************/
4888
4889
void *GDALCreateSimilarTransformer(void *pTransformArg, double dfRatioX,
4890
                                   double dfRatioY)
4891
0
{
4892
0
    GDALTransformerInfo *psInfo =
4893
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4894
4895
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4896
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4897
0
    {
4898
0
        CPLError(CE_Failure, CPLE_AppDefined,
4899
0
                 "Attempt to call CreateSimilar on a non-GTI2 transformer.");
4900
0
        return nullptr;
4901
0
    }
4902
4903
0
    if (psInfo->pfnCreateSimilar == nullptr)
4904
0
    {
4905
0
        CPLError(CE_Failure, CPLE_AppDefined,
4906
0
                 "No CreateSimilar function available for this transformer.");
4907
0
        return nullptr;
4908
0
    }
4909
4910
0
    return psInfo->pfnCreateSimilar(psInfo, dfRatioX, dfRatioY);
4911
0
}
4912
4913
/************************************************************************/
4914
/*                      GetGenImgProjTransformInfo()                    */
4915
/************************************************************************/
4916
4917
static GDALTransformerInfo *GetGenImgProjTransformInfo(const char *pszFunc,
4918
                                                       void *pTransformArg)
4919
0
{
4920
0
    GDALTransformerInfo *psInfo =
4921
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4922
4923
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4924
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4925
0
    {
4926
0
        CPLError(CE_Failure, CPLE_AppDefined,
4927
0
                 "Attempt to call %s on "
4928
0
                 "a non-GTI2 transformer.",
4929
0
                 pszFunc);
4930
0
        return nullptr;
4931
0
    }
4932
4933
0
    if (EQUAL(psInfo->pszClassName, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
4934
0
    {
4935
0
        GDALApproxTransformInfo *psATInfo =
4936
0
            static_cast<GDALApproxTransformInfo *>(pTransformArg);
4937
0
        psInfo = static_cast<GDALTransformerInfo *>(psATInfo->pBaseCBData);
4938
4939
0
        if (psInfo == nullptr ||
4940
0
            memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4941
0
                   strlen(GDAL_GTI2_SIGNATURE)) != 0)
4942
0
        {
4943
0
            CPLError(CE_Failure, CPLE_AppDefined,
4944
0
                     "Attempt to call %s on "
4945
0
                     "a non-GTI2 transformer.",
4946
0
                     pszFunc);
4947
0
            return nullptr;
4948
0
        }
4949
0
    }
4950
4951
0
    if (EQUAL(psInfo->pszClassName, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4952
0
    {
4953
0
        return psInfo;
4954
0
    }
4955
4956
0
    return nullptr;
4957
0
}
4958
4959
/************************************************************************/
4960
/*                 GDALSetTransformerDstGeoTransform()                  */
4961
/************************************************************************/
4962
4963
/**
4964
 * Set ApproxTransformer or GenImgProj output geotransform.
4965
 *
4966
 * This is a layer above GDALSetGenImgProjTransformerDstGeoTransform() that
4967
 * checks that the passed hTransformArg is compatible.
4968
 *
4969
 * Normally the "destination geotransform", or transformation between
4970
 * georeferenced output coordinates and pixel/line coordinates on the
4971
 * destination file is extracted from the destination file by
4972
 * GDALCreateGenImgProjTransformer() and stored in the GenImgProj private
4973
 * info.  However, sometimes it is inconvenient to have an output file
4974
 * handle with appropriate geotransform information when creating the
4975
 * transformation.  For these cases, this function can be used to apply
4976
 * the destination geotransform.
4977
 *
4978
 * @param pTransformArg the handle to update.
4979
 * @param padfGeoTransform the destination geotransform to apply (six doubles).
4980
 */
4981
4982
void GDALSetTransformerDstGeoTransform(void *pTransformArg,
4983
                                       const double *padfGeoTransform)
4984
0
{
4985
0
    VALIDATE_POINTER0(pTransformArg, "GDALSetTransformerDstGeoTransform");
4986
4987
0
    GDALTransformerInfo *psInfo = GetGenImgProjTransformInfo(
4988
0
        "GDALSetTransformerDstGeoTransform", pTransformArg);
4989
0
    if (psInfo)
4990
0
    {
4991
0
        GDALSetGenImgProjTransformerDstGeoTransform(psInfo, padfGeoTransform);
4992
0
    }
4993
0
}
4994
4995
/************************************************************************/
4996
/*                 GDALGetTransformerDstGeoTransform()                  */
4997
/************************************************************************/
4998
4999
/**
5000
 * Get ApproxTransformer or GenImgProj output geotransform.
5001
 *
5002
 * @param pTransformArg transformer handle.
5003
 * @param padfGeoTransform (output) the destination geotransform to return (six
5004
 * doubles).
5005
 */
5006
5007
void GDALGetTransformerDstGeoTransform(void *pTransformArg,
5008
                                       double *padfGeoTransform)
5009
0
{
5010
0
    VALIDATE_POINTER0(pTransformArg, "GDALGetTransformerDstGeoTransform");
5011
5012
0
    GDALTransformerInfo *psInfo = GetGenImgProjTransformInfo(
5013
0
        "GDALGetTransformerDstGeoTransform", pTransformArg);
5014
0
    if (psInfo)
5015
0
    {
5016
0
        GDALGenImgProjTransformInfo *psGenImgProjInfo =
5017
0
            reinterpret_cast<GDALGenImgProjTransformInfo *>(psInfo);
5018
5019
0
        memcpy(padfGeoTransform, psGenImgProjInfo->sDstParams.adfGeoTransform,
5020
0
               sizeof(double) * 6);
5021
0
    }
5022
0
}
5023
5024
/************************************************************************/
5025
/*            GDALTransformIsTranslationOnPixelBoundaries()             */
5026
/************************************************************************/
5027
5028
bool GDALTransformIsTranslationOnPixelBoundaries(GDALTransformerFunc,
5029
                                                 void *pTransformerArg)
5030
0
{
5031
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5032
0
    {
5033
0
        const auto *pApproxInfo =
5034
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5035
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5036
0
    }
5037
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5038
0
    {
5039
0
        const auto *pGenImgpProjInfo =
5040
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5041
0
        const auto IsCloseToInteger = [](double dfVal)
5042
0
        { return std::fabs(dfVal - std::round(dfVal)) <= 1e-6; };
5043
0
        return pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr &&
5044
0
               pGenImgpProjInfo->sDstParams.pTransformArg == nullptr &&
5045
0
               pGenImgpProjInfo->pReproject == nullptr &&
5046
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[1] ==
5047
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[1] &&
5048
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[5] ==
5049
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[5] &&
5050
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[2] ==
5051
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[2] &&
5052
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[4] ==
5053
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[4] &&
5054
               // Check that the georeferenced origin of the destination
5055
               // geotransform is close to be an integer value when transformed
5056
               // to source image coordinates
5057
0
               IsCloseToInteger(
5058
0
                   pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[0] +
5059
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[0] *
5060
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[1] +
5061
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[3] *
5062
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[2]) &&
5063
0
               IsCloseToInteger(
5064
0
                   pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[3] +
5065
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[0] *
5066
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[4] +
5067
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[3] *
5068
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[5]);
5069
0
    }
5070
0
    return false;
5071
0
}
5072
5073
/************************************************************************/
5074
/*                   GDALTransformIsAffineNoRotation()                  */
5075
/************************************************************************/
5076
5077
bool GDALTransformIsAffineNoRotation(GDALTransformerFunc, void *pTransformerArg)
5078
0
{
5079
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5080
0
    {
5081
0
        const auto *pApproxInfo =
5082
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5083
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5084
0
    }
5085
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5086
0
    {
5087
0
        const auto *pGenImgpProjInfo =
5088
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5089
0
        return pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr &&
5090
0
               pGenImgpProjInfo->sDstParams.pTransformArg == nullptr &&
5091
0
               pGenImgpProjInfo->pReproject == nullptr &&
5092
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[2] == 0 &&
5093
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[4] == 0 &&
5094
0
               pGenImgpProjInfo->sDstParams.adfGeoTransform[2] == 0 &&
5095
0
               pGenImgpProjInfo->sDstParams.adfGeoTransform[4] == 0;
5096
0
    }
5097
0
    return false;
5098
0
}
5099
5100
/************************************************************************/
5101
/*                        GDALTransformHasFastClone()                   */
5102
/************************************************************************/
5103
5104
/** Returns whether GDALCloneTransformer() on this transformer is
5105
 * "fast"
5106
 * Counter-examples are GCPs or TPSs transformers.
5107
 */
5108
bool GDALTransformHasFastClone(void *pTransformerArg)
5109
0
{
5110
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5111
0
    {
5112
0
        const auto *pApproxInfo =
5113
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5114
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5115
        // Fallback to next lines
5116
0
    }
5117
5118
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5119
0
    {
5120
0
        const auto *pGenImgpProjInfo =
5121
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5122
0
        return (pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr ||
5123
0
                GDALTransformHasFastClone(
5124
0
                    pGenImgpProjInfo->sSrcParams.pTransformArg)) &&
5125
0
               (pGenImgpProjInfo->sDstParams.pTransformArg == nullptr ||
5126
0
                GDALTransformHasFastClone(
5127
0
                    pGenImgpProjInfo->sDstParams.pTransformArg));
5128
0
    }
5129
0
    else if (GDALIsTransformer(pTransformerArg,
5130
0
                               GDAL_RPC_TRANSFORMER_CLASS_NAME))
5131
0
    {
5132
0
        return true;
5133
0
    }
5134
0
    else
5135
0
    {
5136
0
        return false;
5137
0
    }
5138
0
}