Coverage Report

Created: 2025-12-31 06:48

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 rough
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
    // Sanity check (see issue GH #13498)
3082
0
    if (nullptr == pTransformArgIn)
3083
0
        return FALSE;
3084
3085
0
    GDALGenImgProjTransformInfo *psInfo =
3086
0
        static_cast<GDALGenImgProjTransformInfo *>(pTransformArgIn);
3087
3088
#ifdef DEBUG_APPROX_TRANSFORMER
3089
    CPLAssert(nPointCount > 0);
3090
    countGDALGenImgProjTransform += nPointCount;
3091
#endif
3092
3093
0
    for (int i = 0; i < nPointCount; i++)
3094
0
    {
3095
0
        panSuccess[i] = (padfX[i] != HUGE_VAL && padfY[i] != HUGE_VAL);
3096
0
    }
3097
3098
0
    int ret = TRUE;
3099
3100
    /* -------------------------------------------------------------------- */
3101
    /*      Convert from src (dst) pixel/line to src (dst)                  */
3102
    /*      georeferenced coordinates.                                      */
3103
    /* -------------------------------------------------------------------- */
3104
0
    {
3105
0
        const auto params = bDstToSrc ? psInfo->sDstParams : psInfo->sSrcParams;
3106
0
        const double *padfGeoTransform = params.adfGeoTransform;
3107
0
        void *pTransformArg = params.pTransformArg;
3108
0
        GDALTransformerFunc pTransformer = params.pTransformer;
3109
3110
0
        if (pTransformArg != nullptr)
3111
0
        {
3112
0
            if (!pTransformer(pTransformArg, FALSE, nPointCount, padfX, padfY,
3113
0
                              padfZ, panSuccess))
3114
0
                ret = FALSE;
3115
0
        }
3116
0
        else
3117
0
        {
3118
0
            for (int i = 0; i < nPointCount; i++)
3119
0
            {
3120
0
                if (!panSuccess[i])
3121
0
                    continue;
3122
3123
0
                const double dfNewX = padfGeoTransform[0] +
3124
0
                                      padfX[i] * padfGeoTransform[1] +
3125
0
                                      padfY[i] * padfGeoTransform[2];
3126
0
                const double dfNewY = padfGeoTransform[3] +
3127
0
                                      padfX[i] * padfGeoTransform[4] +
3128
0
                                      padfY[i] * padfGeoTransform[5];
3129
3130
0
                padfX[i] = dfNewX;
3131
0
                padfY[i] = dfNewY;
3132
0
            }
3133
0
        }
3134
0
    }
3135
3136
    /* -------------------------------------------------------------------- */
3137
    /*      Reproject if needed.                                            */
3138
    /* -------------------------------------------------------------------- */
3139
0
    if (psInfo->pReprojectArg)
3140
0
    {
3141
0
        if (!psInfo->pReproject(psInfo->pReprojectArg, bDstToSrc, nPointCount,
3142
0
                                padfX, padfY, padfZ, panSuccess))
3143
0
            ret = FALSE;
3144
0
    }
3145
3146
    /* -------------------------------------------------------------------- */
3147
    /*      Convert dst (src) georef coordinates back to pixel/line.        */
3148
    /* -------------------------------------------------------------------- */
3149
0
    {
3150
0
        const auto params = bDstToSrc ? psInfo->sSrcParams : psInfo->sDstParams;
3151
0
        const double *padfInvGeoTransform = params.adfInvGeoTransform;
3152
0
        void *pTransformArg = params.pTransformArg;
3153
0
        GDALTransformerFunc pTransformer = params.pTransformer;
3154
3155
0
        if (pTransformArg != nullptr)
3156
0
        {
3157
0
            if (!pTransformer(pTransformArg, TRUE, nPointCount, padfX, padfY,
3158
0
                              padfZ, panSuccess))
3159
0
                ret = FALSE;
3160
0
        }
3161
0
        else
3162
0
        {
3163
0
            for (int i = 0; i < nPointCount; i++)
3164
0
            {
3165
0
                if (!panSuccess[i])
3166
0
                    continue;
3167
3168
0
                const double dfNewX = padfInvGeoTransform[0] +
3169
0
                                      padfX[i] * padfInvGeoTransform[1] +
3170
0
                                      padfY[i] * padfInvGeoTransform[2];
3171
0
                const double dfNewY = padfInvGeoTransform[3] +
3172
0
                                      padfX[i] * padfInvGeoTransform[4] +
3173
0
                                      padfY[i] * padfInvGeoTransform[5];
3174
3175
0
                padfX[i] = dfNewX;
3176
0
                padfY[i] = dfNewY;
3177
0
            }
3178
0
        }
3179
0
    }
3180
3181
0
    return ret;
3182
0
}
3183
3184
/************************************************************************/
3185
/*              GDALTransformLonLatToDestGenImgProjTransformer()        */
3186
/************************************************************************/
3187
3188
int GDALTransformLonLatToDestGenImgProjTransformer(void *hTransformArg,
3189
                                                   double *pdfX, double *pdfY)
3190
0
{
3191
0
    GDALGenImgProjTransformInfo *psInfo =
3192
0
        static_cast<GDALGenImgProjTransformInfo *>(hTransformArg);
3193
3194
0
    if (psInfo->pReprojectArg == nullptr ||
3195
0
        psInfo->pReproject != GDALReprojectionTransform)
3196
0
        return false;
3197
3198
0
    GDALReprojectionTransformInfo *psReprojInfo =
3199
0
        static_cast<GDALReprojectionTransformInfo *>(psInfo->pReprojectArg);
3200
0
    if (psReprojInfo->poForwardTransform == nullptr ||
3201
0
        psReprojInfo->poForwardTransform->GetSourceCS() == nullptr)
3202
0
        return false;
3203
3204
0
    double z = 0;
3205
0
    int success = true;
3206
0
    auto poSourceCRS = psReprojInfo->poForwardTransform->GetSourceCS();
3207
0
    if (poSourceCRS->IsGeographic() &&
3208
0
        std::fabs(poSourceCRS->GetAngularUnits() -
3209
0
                  CPLAtof(SRS_UA_DEGREE_CONV)) < 1e-9)
3210
0
    {
3211
        // Optimization to avoid creating a OGRCoordinateTransformation
3212
0
        OGRAxisOrientation eSourceFirstAxisOrient = OAO_Other;
3213
0
        poSourceCRS->GetAxis(nullptr, 0, &eSourceFirstAxisOrient);
3214
0
        const auto &mapping = poSourceCRS->GetDataAxisToSRSAxisMapping();
3215
0
        if ((mapping[0] == 2 && eSourceFirstAxisOrient == OAO_East) ||
3216
0
            (mapping[0] == 1 && eSourceFirstAxisOrient != OAO_East))
3217
0
        {
3218
0
            std::swap(*pdfX, *pdfY);
3219
0
        }
3220
0
    }
3221
0
    else
3222
0
    {
3223
0
        auto poLongLat =
3224
0
            std::unique_ptr<OGRSpatialReference>(poSourceCRS->CloneGeogCS());
3225
0
        if (poLongLat == nullptr)
3226
0
            return false;
3227
0
        poLongLat->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3228
3229
0
        const bool bCurrentCheckWithInvertProj =
3230
0
            GetCurrentCheckWithInvertPROJ();
3231
0
        if (!bCurrentCheckWithInvertProj)
3232
0
            CPLSetThreadLocalConfigOption("CHECK_WITH_INVERT_PROJ", "YES");
3233
0
        auto poCT = std::unique_ptr<OGRCoordinateTransformation>(
3234
0
            OGRCreateCoordinateTransformation(poLongLat.get(), poSourceCRS));
3235
0
        if (!bCurrentCheckWithInvertProj)
3236
0
            CPLSetThreadLocalConfigOption("CHECK_WITH_INVERT_PROJ", nullptr);
3237
0
        if (poCT == nullptr)
3238
0
            return false;
3239
3240
0
        poCT->SetEmitErrors(false);
3241
0
        if (!poCT->Transform(1, pdfX, pdfY))
3242
0
            return false;
3243
3244
0
        if (!psInfo->pReproject(psInfo->pReprojectArg, false, 1, pdfX, pdfY, &z,
3245
0
                                &success) ||
3246
0
            !success)
3247
0
        {
3248
0
            return false;
3249
0
        }
3250
0
    }
3251
3252
0
    double *padfGeoTransform = psInfo->sDstParams.adfInvGeoTransform;
3253
0
    void *pTransformArg = psInfo->sDstParams.pTransformArg;
3254
0
    GDALTransformerFunc pTransformer = psInfo->sDstParams.pTransformer;
3255
0
    if (pTransformArg != nullptr)
3256
0
    {
3257
0
        if (!pTransformer(pTransformArg, TRUE, 1, pdfX, pdfY, &z, &success) ||
3258
0
            !success)
3259
0
        {
3260
0
            return false;
3261
0
        }
3262
0
    }
3263
0
    else
3264
0
    {
3265
0
        const double dfNewX = padfGeoTransform[0] +
3266
0
                              pdfX[0] * padfGeoTransform[1] +
3267
0
                              pdfY[0] * padfGeoTransform[2];
3268
0
        const double dfNewY = padfGeoTransform[3] +
3269
0
                              pdfX[0] * padfGeoTransform[4] +
3270
0
                              pdfY[0] * padfGeoTransform[5];
3271
3272
0
        pdfX[0] = dfNewX;
3273
0
        pdfY[0] = dfNewY;
3274
0
    }
3275
3276
0
    return true;
3277
0
}
3278
3279
/************************************************************************/
3280
/*                 GDALSerializeGenImgProjTransformer()                 */
3281
/************************************************************************/
3282
3283
static CPLXMLNode *GDALSerializeGenImgProjTransformer(void *pTransformArg)
3284
3285
0
{
3286
0
    GDALGenImgProjTransformInfo *psInfo =
3287
0
        static_cast<GDALGenImgProjTransformInfo *>(pTransformArg);
3288
3289
0
    CPLXMLNode *psTree =
3290
0
        CPLCreateXMLNode(nullptr, CXT_Element, "GenImgProjTransformer");
3291
3292
0
    const auto SerializePart =
3293
0
        [psTree](const char *pszPrefix, const GDALGenImgProjTransformPart &part)
3294
0
    {
3295
0
        char szWork[200] = {};
3296
3297
        /* ------------------------------------------------------------- */
3298
        /*      Handle transformation.                                   */
3299
        /* ------------------------------------------------------------- */
3300
0
        if (part.pTransformArg != nullptr)
3301
0
        {
3302
0
            CPLXMLNode *psTransformer =
3303
0
                GDALSerializeTransformer(part.pTransformer, part.pTransformArg);
3304
0
            if (psTransformer != nullptr)
3305
0
            {
3306
0
                CPLXMLNode *psTransformerContainer = CPLCreateXMLNode(
3307
0
                    psTree, CXT_Element,
3308
0
                    CPLSPrintf("%s%s", pszPrefix, psTransformer->pszValue));
3309
3310
0
                CPLAddXMLChild(psTransformerContainer, psTransformer);
3311
0
            }
3312
0
        }
3313
3314
        /* ------------------------------------------------------------- */
3315
        /*      Handle geotransforms.                                    */
3316
        /* ------------------------------------------------------------- */
3317
0
        else
3318
0
        {
3319
0
            CPLsnprintf(szWork, sizeof(szWork),
3320
0
                        "%.17g,%.17g,%.17g,%.17g,%.17g,%.17g",
3321
0
                        part.adfGeoTransform[0], part.adfGeoTransform[1],
3322
0
                        part.adfGeoTransform[2], part.adfGeoTransform[3],
3323
0
                        part.adfGeoTransform[4], part.adfGeoTransform[5]);
3324
0
            CPLCreateXMLElementAndValue(
3325
0
                psTree, CPLSPrintf("%sGeoTransform", pszPrefix), szWork);
3326
3327
0
            CPLsnprintf(szWork, sizeof(szWork),
3328
0
                        "%.17g,%.17g,%.17g,%.17g,%.17g,%.17g",
3329
0
                        part.adfInvGeoTransform[0], part.adfInvGeoTransform[1],
3330
0
                        part.adfInvGeoTransform[2], part.adfInvGeoTransform[3],
3331
0
                        part.adfInvGeoTransform[4], part.adfInvGeoTransform[5]);
3332
0
            CPLCreateXMLElementAndValue(
3333
0
                psTree, CPLSPrintf("%sInvGeoTransform", pszPrefix), szWork);
3334
0
        }
3335
0
    };
3336
3337
0
    SerializePart("Src", psInfo->sSrcParams);
3338
0
    SerializePart("Dst", psInfo->sDstParams);
3339
3340
    /* -------------------------------------------------------------------- */
3341
    /*      Do we have a reprojection transformer?                          */
3342
    /* -------------------------------------------------------------------- */
3343
0
    if (psInfo->pReprojectArg != nullptr)
3344
0
    {
3345
3346
0
        CPLXMLNode *psTransformerContainer =
3347
0
            CPLCreateXMLNode(psTree, CXT_Element, "ReprojectTransformer");
3348
3349
0
        CPLXMLNode *psTransformer =
3350
0
            GDALSerializeTransformer(psInfo->pReproject, psInfo->pReprojectArg);
3351
0
        if (psTransformer != nullptr)
3352
0
            CPLAddXMLChild(psTransformerContainer, psTransformer);
3353
0
    }
3354
3355
0
    return psTree;
3356
0
}
3357
3358
/************************************************************************/
3359
/*                    GDALDeserializeGeoTransform()                     */
3360
/************************************************************************/
3361
3362
static void GDALDeserializeGeoTransform(const char *pszGT,
3363
                                        double adfGeoTransform[6])
3364
0
{
3365
0
    CPLsscanf(pszGT, "%lf,%lf,%lf,%lf,%lf,%lf", adfGeoTransform + 0,
3366
0
              adfGeoTransform + 1, adfGeoTransform + 2, adfGeoTransform + 3,
3367
0
              adfGeoTransform + 4, adfGeoTransform + 5);
3368
0
}
3369
3370
/************************************************************************/
3371
/*                GDALDeserializeGenImgProjTransformer()                */
3372
/************************************************************************/
3373
3374
void *GDALDeserializeGenImgProjTransformer(CPLXMLNode *psTree)
3375
3376
0
{
3377
    /* -------------------------------------------------------------------- */
3378
    /*      Initialize the transform info.                                  */
3379
    /* -------------------------------------------------------------------- */
3380
0
    GDALGenImgProjTransformInfo *psInfo =
3381
0
        GDALCreateGenImgProjTransformerInternal();
3382
3383
0
    const auto DeserializePart =
3384
0
        [psTree](const char *pszPrefix, GDALGenImgProjTransformPart &part)
3385
0
    {
3386
        /* ----------------------------------------------------------------- */
3387
        /*      Geotransform                                                 */
3388
        /* ----------------------------------------------------------------- */
3389
0
        if (const auto psGTNode =
3390
0
                CPLGetXMLNode(psTree, CPLSPrintf("%sGeoTransform", pszPrefix)))
3391
0
        {
3392
0
            GDALDeserializeGeoTransform(CPLGetXMLValue(psGTNode, "", ""),
3393
0
                                        part.adfGeoTransform);
3394
3395
0
            if (const auto psInvGTNode = CPLGetXMLNode(
3396
0
                    psTree, CPLSPrintf("%sInvGeoTransform", pszPrefix)))
3397
0
            {
3398
0
                GDALDeserializeGeoTransform(CPLGetXMLValue(psInvGTNode, "", ""),
3399
0
                                            part.adfInvGeoTransform);
3400
0
            }
3401
0
            else
3402
0
            {
3403
0
                if (!GDALInvGeoTransform(part.adfGeoTransform,
3404
0
                                         part.adfInvGeoTransform))
3405
0
                {
3406
0
                    CPLError(CE_Failure, CPLE_AppDefined,
3407
0
                             "Cannot invert geotransform");
3408
0
                }
3409
0
            }
3410
0
        }
3411
3412
        /* ---------------------------------------------------------------- */
3413
        /*      Transform                                                   */
3414
        /* ---------------------------------------------------------------- */
3415
0
        else
3416
0
        {
3417
0
            for (CPLXMLNode *psIter = psTree->psChild; psIter != nullptr;
3418
0
                 psIter = psIter->psNext)
3419
0
            {
3420
0
                if (psIter->eType == CXT_Element &&
3421
0
                    STARTS_WITH_CI(psIter->pszValue, pszPrefix))
3422
0
                {
3423
0
                    GDALDeserializeTransformer(psIter->psChild,
3424
0
                                               &part.pTransformer,
3425
0
                                               &part.pTransformArg);
3426
0
                    break;
3427
0
                }
3428
0
            }
3429
0
        }
3430
0
    };
3431
3432
0
    DeserializePart("Src", psInfo->sSrcParams);
3433
0
    DeserializePart("Dst", psInfo->sDstParams);
3434
3435
    /* -------------------------------------------------------------------- */
3436
    /*      Reproject transformer                                           */
3437
    /* -------------------------------------------------------------------- */
3438
0
    CPLXMLNode *psSubtree = CPLGetXMLNode(psTree, "ReprojectTransformer");
3439
0
    if (psSubtree != nullptr && psSubtree->psChild != nullptr)
3440
0
    {
3441
0
        GDALDeserializeTransformer(psSubtree->psChild, &psInfo->pReproject,
3442
0
                                   &psInfo->pReprojectArg);
3443
0
    }
3444
3445
0
    return psInfo;
3446
0
}
3447
3448
/************************************************************************/
3449
/*                 GDALCreateReprojectionTransformer()                  */
3450
/************************************************************************/
3451
3452
/**
3453
 * Create reprojection transformer.
3454
 *
3455
 * Creates a callback data structure suitable for use with
3456
 * GDALReprojectionTransformation() to represent a transformation from
3457
 * one geographic or projected coordinate system to another.  On input
3458
 * the coordinate systems are described in OpenGIS WKT format.
3459
 *
3460
 * Internally the OGRCoordinateTransformation object is used to implement
3461
 * the reprojection.
3462
 *
3463
 * @param pszSrcWKT the coordinate system for the source coordinate system.
3464
 * @param pszDstWKT the coordinate system for the destination coordinate
3465
 * system.
3466
 *
3467
 * @return Handle for use with GDALReprojectionTransform(), or NULL if the
3468
 * system fails to initialize the reprojection.
3469
 **/
3470
3471
void *GDALCreateReprojectionTransformer(const char *pszSrcWKT,
3472
                                        const char *pszDstWKT)
3473
3474
0
{
3475
    /* -------------------------------------------------------------------- */
3476
    /*      Ingest the SRS definitions.                                     */
3477
    /* -------------------------------------------------------------------- */
3478
0
    OGRSpatialReference oSrcSRS;
3479
0
    oSrcSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3480
0
    if (oSrcSRS.importFromWkt(pszSrcWKT) != OGRERR_NONE)
3481
0
    {
3482
0
        CPLError(CE_Failure, CPLE_AppDefined,
3483
0
                 "Failed to import coordinate system `%s'.", pszSrcWKT);
3484
0
        return nullptr;
3485
0
    }
3486
3487
0
    OGRSpatialReference oDstSRS;
3488
0
    oDstSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3489
0
    if (oDstSRS.importFromWkt(pszDstWKT) != OGRERR_NONE)
3490
0
    {
3491
0
        CPLError(CE_Failure, CPLE_AppDefined,
3492
0
                 "Failed to import coordinate system `%s'.", pszSrcWKT);
3493
0
        return nullptr;
3494
0
    }
3495
3496
0
    return GDALCreateReprojectionTransformerEx(
3497
0
        OGRSpatialReference::ToHandle(&oSrcSRS),
3498
0
        OGRSpatialReference::ToHandle(&oDstSRS), nullptr);
3499
0
}
3500
3501
/************************************************************************/
3502
/*                 GDALCreateReprojectionTransformerEx()                */
3503
/************************************************************************/
3504
3505
/**
3506
 * Create reprojection transformer.
3507
 *
3508
 * Creates a callback data structure suitable for use with
3509
 * GDALReprojectionTransformation() to represent a transformation from
3510
 * one geographic or projected coordinate system to another.
3511
 *
3512
 * Internally the OGRCoordinateTransformation object is used to implement
3513
 * the reprojection.
3514
 *
3515
 * @param hSrcSRS the coordinate system for the source coordinate system.
3516
 * @param hDstSRS the coordinate system for the destination coordinate
3517
 * system.
3518
 * @param papszOptions NULL-terminated list of options, or NULL. Currently
3519
 * supported options are:
3520
 * <ul>
3521
 * <li>AREA_OF_INTEREST=west_long,south_lat,east_long,north_lat: Values in
3522
 * degrees. longitudes in [-180,180], latitudes in [-90,90].</li>
3523
 * <li>COORDINATE_OPERATION=string: PROJ or WKT string representing a
3524
 * coordinate operation, overriding the default computed transformation.</li>
3525
 * <li>COORDINATE_EPOCH=decimal_year: Coordinate epoch, expressed as a
3526
 * decimal year. Useful for time-dependent coordinate operations.</li>
3527
 * <li> SRC_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch of source CRS,
3528
 * expressed as a decimal year. Useful for time-dependent coordinate
3529
 *operations.</li>
3530
 * <li> DST_COORDINATE_EPOCH: (GDAL &gt;= 3.4) Coordinate epoch
3531
 *of target CRS, expressed as a decimal year. Useful for time-dependent
3532
 *coordinate operations.</li>
3533
 * <li> ALLOW_BALLPARK=YES/NO: (GDAL &gt;= 3.11) Whether ballpark coordinate
3534
 * operations are allowed. Defaults to YES.</li>
3535
 * <li> ONLY_BEST=YES/NO/AUTO: (GDAL &gt;= 3.11) By default (at least in the
3536
 * PROJ 9.x series), PROJ may use coordinate
3537
 * operations that are not the "best" if resources (typically grids) needed
3538
 * to use them are missing. It will then fallback to other coordinate operations
3539
 * that have a lesser accuracy, for example using Helmert transformations,
3540
 * or in the absence of such operations, to ones with potential very rough
3541
 * accuracy, using "ballpark" transformations
3542
 * (see https://proj.org/glossary.html).
3543
 * When calling this method with YES, PROJ will only consider the
3544
 * "best" operation, and error out (at Transform() time) if they cannot be
3545
 * used.
3546
 * This method may be used together with ALLOW_BALLPARK=NO to
3547
 * only allow best operations that have a known accuracy.
3548
 * Note that this method has no effect on PROJ versions before 9.2.
3549
 * The default value for this option can be also set with the
3550
 * PROJ_ONLY_BEST_DEFAULT environment variable, or with the "only_best_default"
3551
 * setting of proj.ini. Calling SetOnlyBest() overrides such default value.</li>
3552
 * </ul>
3553
 *
3554
 * @return Handle for use with GDALReprojectionTransform(), or NULL if the
3555
 * system fails to initialize the reprojection.
3556
 *
3557
 * @since GDAL 3.0
3558
 **/
3559
3560
void *GDALCreateReprojectionTransformerEx(OGRSpatialReferenceH hSrcSRS,
3561
                                          OGRSpatialReferenceH hDstSRS,
3562
                                          const char *const *papszOptions)
3563
0
{
3564
0
    OGRSpatialReference *poSrcSRS = OGRSpatialReference::FromHandle(hSrcSRS);
3565
0
    OGRSpatialReference *poDstSRS = OGRSpatialReference::FromHandle(hDstSRS);
3566
3567
    /* -------------------------------------------------------------------- */
3568
    /*      Build the forward coordinate transformation.                    */
3569
    /* -------------------------------------------------------------------- */
3570
0
    double dfWestLongitudeDeg = 0.0;
3571
0
    double dfSouthLatitudeDeg = 0.0;
3572
0
    double dfEastLongitudeDeg = 0.0;
3573
0
    double dfNorthLatitudeDeg = 0.0;
3574
0
    const char *pszBBOX = CSLFetchNameValue(papszOptions, "AREA_OF_INTEREST");
3575
0
    if (pszBBOX)
3576
0
    {
3577
0
        char **papszTokens = CSLTokenizeString2(pszBBOX, ",", 0);
3578
0
        if (CSLCount(papszTokens) == 4)
3579
0
        {
3580
0
            dfWestLongitudeDeg = CPLAtof(papszTokens[0]);
3581
0
            dfSouthLatitudeDeg = CPLAtof(papszTokens[1]);
3582
0
            dfEastLongitudeDeg = CPLAtof(papszTokens[2]);
3583
0
            dfNorthLatitudeDeg = CPLAtof(papszTokens[3]);
3584
0
        }
3585
0
        CSLDestroy(papszTokens);
3586
0
    }
3587
0
    const char *pszCO = CSLFetchNameValue(papszOptions, "COORDINATE_OPERATION");
3588
3589
0
    OGRCoordinateTransformationOptions optionsFwd;
3590
0
    if (!(dfWestLongitudeDeg == 0.0 && dfSouthLatitudeDeg == 0.0 &&
3591
0
          dfEastLongitudeDeg == 0.0 && dfNorthLatitudeDeg == 0.0))
3592
0
    {
3593
0
        optionsFwd.SetAreaOfInterest(dfWestLongitudeDeg, dfSouthLatitudeDeg,
3594
0
                                     dfEastLongitudeDeg, dfNorthLatitudeDeg);
3595
0
    }
3596
0
    if (pszCO)
3597
0
    {
3598
0
        optionsFwd.SetCoordinateOperation(pszCO, false);
3599
0
    }
3600
3601
0
    const char *pszCENTER_LONG = CSLFetchNameValue(papszOptions, "CENTER_LONG");
3602
0
    if (pszCENTER_LONG)
3603
0
    {
3604
0
        optionsFwd.SetSourceCenterLong(CPLAtof(pszCENTER_LONG));
3605
0
    }
3606
3607
0
    optionsFwd.SetBallparkAllowed(CPLTestBool(
3608
0
        CSLFetchNameValueDef(papszOptions, "ALLOW_BALLPARK", "YES")));
3609
3610
0
    const char *pszOnlyBest =
3611
0
        CSLFetchNameValueDef(papszOptions, "ONLY_BEST", "AUTO");
3612
0
    if (!EQUAL(pszOnlyBest, "AUTO"))
3613
0
    {
3614
0
        optionsFwd.SetOnlyBest(CPLTestBool(pszOnlyBest));
3615
0
    }
3616
3617
0
    OGRCoordinateTransformation *poForwardTransform =
3618
0
        OGRCreateCoordinateTransformation(poSrcSRS, poDstSRS, optionsFwd);
3619
3620
0
    if (poForwardTransform == nullptr)
3621
        // OGRCreateCoordinateTransformation() will report errors on its own.
3622
0
        return nullptr;
3623
3624
0
    poForwardTransform->SetEmitErrors(false);
3625
3626
    /* -------------------------------------------------------------------- */
3627
    /*      Create a structure to hold the transform info, and also         */
3628
    /*      build reverse transform.  We assume that if the forward         */
3629
    /*      transform can be created, then so can the reverse one.          */
3630
    /* -------------------------------------------------------------------- */
3631
0
    GDALReprojectionTransformInfo *psInfo = new GDALReprojectionTransformInfo();
3632
3633
0
    psInfo->papszOptions = CSLDuplicate(papszOptions);
3634
0
    psInfo->poForwardTransform = poForwardTransform;
3635
0
    psInfo->dfTime = CPLAtof(CSLFetchNameValueDef(
3636
0
        papszOptions, "COORDINATE_EPOCH",
3637
0
        CSLFetchNameValueDef(
3638
0
            papszOptions, "DST_COORDINATE_EPOCH",
3639
0
            CSLFetchNameValueDef(papszOptions, "SRC_COORDINATE_EPOCH", "0"))));
3640
0
    psInfo->poReverseTransform = poForwardTransform->GetInverse();
3641
3642
0
    if (psInfo->poReverseTransform)
3643
0
        psInfo->poReverseTransform->SetEmitErrors(false);
3644
3645
0
    memcpy(psInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE,
3646
0
           strlen(GDAL_GTI2_SIGNATURE));
3647
0
    psInfo->sTI.pszClassName = GDAL_REPROJECTION_TRANSFORMER_CLASS_NAME;
3648
0
    psInfo->sTI.pfnTransform = GDALReprojectionTransform;
3649
0
    psInfo->sTI.pfnCleanup = GDALDestroyReprojectionTransformer;
3650
0
    psInfo->sTI.pfnSerialize = GDALSerializeReprojectionTransformer;
3651
3652
0
    return psInfo;
3653
0
}
3654
3655
/************************************************************************/
3656
/*                 GDALDestroyReprojectionTransformer()                 */
3657
/************************************************************************/
3658
3659
/**
3660
 * Destroy reprojection transformation.
3661
 *
3662
 * @param pTransformArg the transformation handle returned by
3663
 * GDALCreateReprojectionTransformer().
3664
 */
3665
3666
void GDALDestroyReprojectionTransformer(void *pTransformArg)
3667
3668
0
{
3669
0
    if (pTransformArg == nullptr)
3670
0
        return;
3671
3672
0
    GDALReprojectionTransformInfo *psInfo =
3673
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3674
3675
0
    if (psInfo->poForwardTransform)
3676
0
        OGRCoordinateTransformation::DestroyCT(psInfo->poForwardTransform);
3677
3678
0
    if (psInfo->poReverseTransform)
3679
0
        OGRCoordinateTransformation::DestroyCT(psInfo->poReverseTransform);
3680
3681
0
    CSLDestroy(psInfo->papszOptions);
3682
3683
0
    delete psInfo;
3684
0
}
3685
3686
/************************************************************************/
3687
/*                     GDALReprojectionTransform()                      */
3688
/************************************************************************/
3689
3690
/**
3691
 * Perform reprojection transformation.
3692
 *
3693
 * Actually performs the reprojection transformation described in
3694
 * GDALCreateReprojectionTransformer().  This function matches the
3695
 * GDALTransformerFunc() signature.  Details of the arguments are described
3696
 * there.
3697
 */
3698
3699
int GDALReprojectionTransform(void *pTransformArg, int bDstToSrc,
3700
                              int nPointCount, double *padfX, double *padfY,
3701
                              double *padfZ, int *panSuccess)
3702
3703
0
{
3704
0
    GDALReprojectionTransformInfo *psInfo =
3705
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3706
0
    int bSuccess;
3707
3708
0
    std::vector<double> adfTime;
3709
0
    double *padfT = nullptr;
3710
0
    if (psInfo->dfTime != 0.0 && nPointCount > 0)
3711
0
    {
3712
0
        adfTime.resize(nPointCount, psInfo->dfTime);
3713
0
        padfT = &adfTime[0];
3714
0
    }
3715
3716
0
    if (bDstToSrc)
3717
0
    {
3718
0
        if (psInfo->poReverseTransform == nullptr)
3719
0
        {
3720
0
            CPLError(
3721
0
                CE_Failure, CPLE_AppDefined,
3722
0
                "Inverse coordinate transformation cannot be instantiated");
3723
0
            if (panSuccess)
3724
0
            {
3725
0
                for (int i = 0; i < nPointCount; i++)
3726
0
                    panSuccess[i] = FALSE;
3727
0
            }
3728
0
            bSuccess = false;
3729
0
        }
3730
0
        else
3731
0
        {
3732
0
            bSuccess = psInfo->poReverseTransform->Transform(
3733
0
                nPointCount, padfX, padfY, padfZ, padfT, panSuccess);
3734
0
        }
3735
0
    }
3736
0
    else
3737
0
        bSuccess = psInfo->poForwardTransform->Transform(
3738
0
            nPointCount, padfX, padfY, padfZ, padfT, panSuccess);
3739
3740
0
    return bSuccess;
3741
0
}
3742
3743
/************************************************************************/
3744
/*                GDALSerializeReprojectionTransformer()                */
3745
/************************************************************************/
3746
3747
static CPLXMLNode *GDALSerializeReprojectionTransformer(void *pTransformArg)
3748
3749
0
{
3750
0
    CPLXMLNode *psTree;
3751
0
    GDALReprojectionTransformInfo *psInfo =
3752
0
        static_cast<GDALReprojectionTransformInfo *>(pTransformArg);
3753
3754
0
    psTree = CPLCreateXMLNode(nullptr, CXT_Element, "ReprojectionTransformer");
3755
3756
    /* -------------------------------------------------------------------- */
3757
    /*      Handle SourceCS.                                                */
3758
    /* -------------------------------------------------------------------- */
3759
0
    const auto ExportToWkt = [](const OGRSpatialReference *poSRS)
3760
0
    {
3761
        // Try first in WKT1 for backward compat
3762
0
        {
3763
0
            char *pszWKT = nullptr;
3764
0
            const char *const apszOptions[] = {"FORMAT=WKT1", nullptr};
3765
0
            CPLErrorHandlerPusher oHandler(CPLQuietErrorHandler);
3766
0
            CPLErrorStateBackuper oBackuper;
3767
0
            if (poSRS->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE)
3768
0
            {
3769
0
                std::string osRet(pszWKT);
3770
0
                CPLFree(pszWKT);
3771
0
                return osRet;
3772
0
            }
3773
0
            CPLFree(pszWKT);
3774
0
        }
3775
3776
0
        char *pszWKT = nullptr;
3777
0
        const char *const apszOptions[] = {"FORMAT=WKT2_2019", nullptr};
3778
0
        if (poSRS->exportToWkt(&pszWKT, apszOptions) == OGRERR_NONE)
3779
0
        {
3780
0
            std::string osRet(pszWKT);
3781
0
            CPLFree(pszWKT);
3782
0
            return osRet;
3783
0
        }
3784
0
        CPLFree(pszWKT);
3785
0
        return std::string();
3786
0
    };
3787
3788
0
    auto poSRS = psInfo->poForwardTransform->GetSourceCS();
3789
0
    if (poSRS)
3790
0
    {
3791
0
        const auto osWKT = ExportToWkt(poSRS);
3792
0
        CPLCreateXMLElementAndValue(psTree, "SourceSRS", osWKT.c_str());
3793
0
    }
3794
3795
    /* -------------------------------------------------------------------- */
3796
    /*      Handle DestinationCS.                                           */
3797
    /* -------------------------------------------------------------------- */
3798
0
    poSRS = psInfo->poForwardTransform->GetTargetCS();
3799
0
    if (poSRS)
3800
0
    {
3801
0
        const auto osWKT = ExportToWkt(poSRS);
3802
0
        CPLCreateXMLElementAndValue(psTree, "TargetSRS", osWKT.c_str());
3803
0
    }
3804
3805
    /* -------------------------------------------------------------------- */
3806
    /*      Serialize options.                                              */
3807
    /* -------------------------------------------------------------------- */
3808
0
    if (psInfo->papszOptions)
3809
0
    {
3810
0
        CPLXMLNode *psOptions =
3811
0
            CPLCreateXMLNode(psTree, CXT_Element, "Options");
3812
0
        for (auto iter = psInfo->papszOptions; *iter != nullptr; ++iter)
3813
0
        {
3814
0
            char *pszKey = nullptr;
3815
0
            const char *pszValue = CPLParseNameValue(*iter, &pszKey);
3816
0
            if (pszKey && pszValue)
3817
0
            {
3818
0
                auto elt =
3819
0
                    CPLCreateXMLElementAndValue(psOptions, "Option", pszValue);
3820
0
                CPLAddXMLAttributeAndValue(elt, "key", pszKey);
3821
0
            }
3822
0
            CPLFree(pszKey);
3823
0
        }
3824
0
    }
3825
3826
0
    return psTree;
3827
0
}
3828
3829
/************************************************************************/
3830
/*               GDALDeserializeReprojectionTransformer()               */
3831
/************************************************************************/
3832
3833
static void *GDALDeserializeReprojectionTransformer(CPLXMLNode *psTree)
3834
3835
0
{
3836
0
    const char *pszSourceSRS = CPLGetXMLValue(psTree, "SourceSRS", nullptr);
3837
0
    const char *pszTargetSRS = CPLGetXMLValue(psTree, "TargetSRS", nullptr);
3838
0
    char *pszSourceWKT = nullptr, *pszTargetWKT = nullptr;
3839
0
    void *pResult = nullptr;
3840
3841
0
    OGRSpatialReference oSrcSRS;
3842
0
    OGRSpatialReference oDstSRS;
3843
3844
0
    oSrcSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3845
0
    oDstSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
3846
0
    if (pszSourceSRS != nullptr)
3847
0
    {
3848
0
        oSrcSRS.SetFromUserInput(pszSourceSRS);
3849
0
    }
3850
3851
0
    if (pszTargetSRS != nullptr)
3852
0
    {
3853
0
        oDstSRS.SetFromUserInput(pszTargetSRS);
3854
0
    }
3855
3856
0
    CPLStringList aosList;
3857
0
    const CPLXMLNode *psOptions = CPLGetXMLNode(psTree, "Options");
3858
0
    if (psOptions)
3859
0
    {
3860
0
        for (auto iter = psOptions->psChild; iter; iter = iter->psNext)
3861
0
        {
3862
0
            if (iter->eType == CXT_Element &&
3863
0
                strcmp(iter->pszValue, "Option") == 0)
3864
0
            {
3865
0
                const char *pszKey = CPLGetXMLValue(iter, "key", nullptr);
3866
0
                const char *pszValue = CPLGetXMLValue(iter, nullptr, nullptr);
3867
0
                if (pszKey && pszValue)
3868
0
                {
3869
0
                    aosList.SetNameValue(pszKey, pszValue);
3870
0
                }
3871
0
            }
3872
0
        }
3873
0
    }
3874
3875
0
    pResult = GDALCreateReprojectionTransformerEx(
3876
0
        !oSrcSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oSrcSRS) : nullptr,
3877
0
        !oDstSRS.IsEmpty() ? OGRSpatialReference::ToHandle(&oDstSRS) : nullptr,
3878
0
        aosList.List());
3879
3880
0
    CPLFree(pszSourceWKT);
3881
0
    CPLFree(pszTargetWKT);
3882
3883
0
    return pResult;
3884
0
}
3885
3886
/************************************************************************/
3887
/* ==================================================================== */
3888
/*      Approximate transformer.                                        */
3889
/* ==================================================================== */
3890
/************************************************************************/
3891
3892
/************************************************************************/
3893
/*                  GDALCreateSimilarApproxTransformer()                */
3894
/************************************************************************/
3895
3896
static void *GDALCreateSimilarApproxTransformer(void *hTransformArg,
3897
                                                double dfSrcRatioX,
3898
                                                double dfSrcRatioY)
3899
0
{
3900
0
    VALIDATE_POINTER1(hTransformArg, "GDALCreateSimilarApproxTransformer",
3901
0
                      nullptr);
3902
3903
0
    GDALApproxTransformInfo *psInfo =
3904
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
3905
3906
0
    void *pBaseCBData = GDALCreateSimilarTransformer(psInfo->pBaseCBData,
3907
0
                                                     dfSrcRatioX, dfSrcRatioY);
3908
0
    if (pBaseCBData == nullptr)
3909
0
    {
3910
0
        return nullptr;
3911
0
    }
3912
3913
0
    GDALApproxTransformInfo *psClonedInfo =
3914
0
        static_cast<GDALApproxTransformInfo *>(GDALCreateApproxTransformer2(
3915
0
            psInfo->pfnBaseTransformer, pBaseCBData, psInfo->dfMaxErrorForward,
3916
0
            psInfo->dfMaxErrorReverse));
3917
0
    psClonedInfo->bOwnSubtransformer = TRUE;
3918
3919
0
    return psClonedInfo;
3920
0
}
3921
3922
/************************************************************************/
3923
/*                   GDALSerializeApproxTransformer()                   */
3924
/************************************************************************/
3925
3926
static CPLXMLNode *GDALSerializeApproxTransformer(void *pTransformArg)
3927
3928
0
{
3929
0
    CPLXMLNode *psTree;
3930
0
    GDALApproxTransformInfo *psInfo =
3931
0
        static_cast<GDALApproxTransformInfo *>(pTransformArg);
3932
3933
0
    psTree = CPLCreateXMLNode(nullptr, CXT_Element, "ApproxTransformer");
3934
3935
    /* -------------------------------------------------------------------- */
3936
    /*      Attach max error.                                               */
3937
    /* -------------------------------------------------------------------- */
3938
0
    if (psInfo->dfMaxErrorForward == psInfo->dfMaxErrorReverse)
3939
0
    {
3940
0
        CPLCreateXMLElementAndValue(
3941
0
            psTree, "MaxError",
3942
0
            CPLString().Printf("%g", psInfo->dfMaxErrorForward));
3943
0
    }
3944
0
    else
3945
0
    {
3946
0
        CPLCreateXMLElementAndValue(
3947
0
            psTree, "MaxErrorForward",
3948
0
            CPLString().Printf("%g", psInfo->dfMaxErrorForward));
3949
0
        CPLCreateXMLElementAndValue(
3950
0
            psTree, "MaxErrorReverse",
3951
0
            CPLString().Printf("%g", psInfo->dfMaxErrorReverse));
3952
0
    }
3953
3954
    /* -------------------------------------------------------------------- */
3955
    /*      Capture underlying transformer.                                 */
3956
    /* -------------------------------------------------------------------- */
3957
0
    CPLXMLNode *psTransformerContainer =
3958
0
        CPLCreateXMLNode(psTree, CXT_Element, "BaseTransformer");
3959
3960
0
    CPLXMLNode *psTransformer = GDALSerializeTransformer(
3961
0
        psInfo->pfnBaseTransformer, psInfo->pBaseCBData);
3962
0
    if (psTransformer != nullptr)
3963
0
        CPLAddXMLChild(psTransformerContainer, psTransformer);
3964
3965
0
    return psTree;
3966
0
}
3967
3968
/************************************************************************/
3969
/*                    GDALCreateApproxTransformer()                     */
3970
/************************************************************************/
3971
3972
/**
3973
 * Create an approximating transformer.
3974
 *
3975
 * This function creates a context for an approximated transformer.  Basically
3976
 * a high precision transformer is supplied as input and internally linear
3977
 * approximations are computed to generate results to within a defined
3978
 * precision.
3979
 *
3980
 * The approximation is actually done at the point where GDALApproxTransform()
3981
 * calls are made, and depend on the assumption that they are roughly linear.
3982
 * The first and last point passed in must be the extreme values and the
3983
 * intermediate values should describe a curve between the end points.  The
3984
 * approximator transforms and centers using the approximate transformer, and
3985
 * then compares the true middle transformed value to a linear approximation
3986
 * based on the end points.  If the error is within the supplied threshold then
3987
 * the end points are used to linearly approximate all the values otherwise the
3988
 * input points are split into two smaller sets, and the function is recursively
3989
 * called until a sufficiently small set of points is found that the linear
3990
 * approximation is OK, or that all the points are exactly computed.
3991
 *
3992
 * This function is very suitable for approximating transformation results
3993
 * from output pixel/line space to input coordinates for warpers that operate
3994
 * on one input scanline at a time.  Care should be taken using it in other
3995
 * circumstances as little internal validation is done in order to keep things
3996
 * fast.
3997
 *
3998
 * @param pfnBaseTransformer the high precision transformer which should be
3999
 * approximated.
4000
 * @param pBaseTransformArg the callback argument for the high precision
4001
 * transformer.
4002
 * @param dfMaxError the maximum cartesian error in the "output" space that
4003
 * is to be accepted in the linear approximation, evaluated as a Manhattan
4004
 * distance.
4005
 *
4006
 * @return callback pointer suitable for use with GDALApproxTransform().  It
4007
 * should be deallocated with GDALDestroyApproxTransformer().
4008
 */
4009
4010
void *GDALCreateApproxTransformer(GDALTransformerFunc pfnBaseTransformer,
4011
                                  void *pBaseTransformArg, double dfMaxError)
4012
4013
0
{
4014
0
    return GDALCreateApproxTransformer2(pfnBaseTransformer, pBaseTransformArg,
4015
0
                                        dfMaxError, dfMaxError);
4016
0
}
4017
4018
static void *
4019
GDALCreateApproxTransformer2(GDALTransformerFunc pfnBaseTransformer,
4020
                             void *pBaseTransformArg, double dfMaxErrorForward,
4021
                             double dfMaxErrorReverse)
4022
4023
0
{
4024
0
    GDALApproxTransformInfo *psATInfo = new GDALApproxTransformInfo;
4025
0
    psATInfo->pfnBaseTransformer = pfnBaseTransformer;
4026
0
    psATInfo->pBaseCBData = pBaseTransformArg;
4027
0
    psATInfo->dfMaxErrorForward = dfMaxErrorForward;
4028
0
    psATInfo->dfMaxErrorReverse = dfMaxErrorReverse;
4029
0
    psATInfo->bOwnSubtransformer = FALSE;
4030
4031
0
    memcpy(psATInfo->sTI.abySignature, GDAL_GTI2_SIGNATURE,
4032
0
           strlen(GDAL_GTI2_SIGNATURE));
4033
0
    psATInfo->sTI.pszClassName = GDAL_APPROX_TRANSFORMER_CLASS_NAME;
4034
0
    psATInfo->sTI.pfnTransform = GDALApproxTransform;
4035
0
    psATInfo->sTI.pfnCleanup = GDALDestroyApproxTransformer;
4036
0
    psATInfo->sTI.pfnSerialize = GDALSerializeApproxTransformer;
4037
0
    psATInfo->sTI.pfnCreateSimilar = GDALCreateSimilarApproxTransformer;
4038
4039
0
    return psATInfo;
4040
0
}
4041
4042
/************************************************************************/
4043
/*              GDALApproxTransformerOwnsSubtransformer()               */
4044
/************************************************************************/
4045
4046
/** Set bOwnSubtransformer flag */
4047
void GDALApproxTransformerOwnsSubtransformer(void *pCBData, int bOwnFlag)
4048
4049
0
{
4050
0
    GDALApproxTransformInfo *psATInfo =
4051
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4052
4053
0
    psATInfo->bOwnSubtransformer = bOwnFlag;
4054
0
}
4055
4056
/************************************************************************/
4057
/*                    GDALDestroyApproxTransformer()                    */
4058
/************************************************************************/
4059
4060
/**
4061
 * Cleanup approximate transformer.
4062
 *
4063
 * Deallocates the resources allocated by GDALCreateApproxTransformer().
4064
 *
4065
 * @param pCBData callback data originally returned by
4066
 * GDALCreateApproxTransformer().
4067
 */
4068
4069
void GDALDestroyApproxTransformer(void *pCBData)
4070
4071
0
{
4072
0
    if (pCBData == nullptr)
4073
0
        return;
4074
4075
0
    GDALApproxTransformInfo *psATInfo =
4076
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4077
4078
0
    if (psATInfo->bOwnSubtransformer)
4079
0
        GDALDestroyTransformer(psATInfo->pBaseCBData);
4080
4081
0
    delete psATInfo;
4082
0
}
4083
4084
/************************************************************************/
4085
/*                  GDALRefreshApproxTransformer()                      */
4086
/************************************************************************/
4087
4088
void GDALRefreshApproxTransformer(void *hTransformArg)
4089
0
{
4090
0
    GDALApproxTransformInfo *psInfo =
4091
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
4092
4093
0
    if (GDALIsTransformer(psInfo->pBaseCBData,
4094
0
                          GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4095
0
    {
4096
0
        GDALRefreshGenImgProjTransformer(psInfo->pBaseCBData);
4097
0
    }
4098
0
}
4099
4100
/************************************************************************/
4101
/*                      GDALApproxTransformInternal()                   */
4102
/************************************************************************/
4103
4104
static int GDALApproxTransformInternal(void *pCBData, int bDstToSrc,
4105
                                       int nPoints, double *x, double *y,
4106
                                       double *z, int *panSuccess,
4107
                                       // SME = Start, Middle, End.
4108
                                       const double xSMETransformed[3],
4109
                                       const double ySMETransformed[3],
4110
                                       const double zSMETransformed[3])
4111
0
{
4112
0
    GDALApproxTransformInfo *psATInfo =
4113
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4114
0
    const int nMiddle = (nPoints - 1) / 2;
4115
4116
#ifdef notdef_sanify_check
4117
    {
4118
        double x2[3] = {x[0], x[nMiddle], x[nPoints - 1]};
4119
        double y2[3] = {y[0], y[nMiddle], y[nPoints - 1]};
4120
        double z2[3] = {z[0], z[nMiddle], z[nPoints - 1]};
4121
        int anSuccess2[3] = {};
4122
4123
        const int bSuccess = psATInfo->pfnBaseTransformer(
4124
            psATInfo->pBaseCBData, bDstToSrc, 3, x2, y2, z2, anSuccess2);
4125
        CPLAssert(bSuccess);
4126
        CPLAssert(anSuccess2[0]);
4127
        CPLAssert(anSuccess2[1]);
4128
        CPLAssert(anSuccess2[2]);
4129
        CPLAssert(x2[0] == xSMETransformed[0]);
4130
        CPLAssert(y2[0] == ySMETransformed[0]);
4131
        CPLAssert(z2[0] == zSMETransformed[0]);
4132
        CPLAssert(x2[1] == xSMETransformed[1]);
4133
        CPLAssert(y2[1] == ySMETransformed[1]);
4134
        CPLAssert(z2[1] == zSMETransformed[1]);
4135
        CPLAssert(x2[2] == xSMETransformed[2]);
4136
        CPLAssert(y2[2] == ySMETransformed[2]);
4137
        CPLAssert(z2[2] == zSMETransformed[2]);
4138
    }
4139
#endif
4140
4141
#ifdef DEBUG_APPROX_TRANSFORMER
4142
    fprintf(stderr, "start (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4143
            x[0], y[0], xSMETransformed[0], ySMETransformed[0]);
4144
    fprintf(stderr, "middle (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4145
            x[nMiddle], y[nMiddle], xSMETransformed[1], ySMETransformed[1]);
4146
    fprintf(stderr, "end (%.3f,%.3f) -> (%.3f,%.3f)\n", /*ok*/
4147
            x[nPoints - 1], y[nPoints - 1], xSMETransformed[2],
4148
            ySMETransformed[2]);
4149
#endif
4150
4151
    /* -------------------------------------------------------------------- */
4152
    /*      Is the error at the middle acceptable relative to an            */
4153
    /*      interpolation of the middle position?                           */
4154
    /* -------------------------------------------------------------------- */
4155
0
    const double dfDeltaX =
4156
0
        (xSMETransformed[2] - xSMETransformed[0]) / (x[nPoints - 1] - x[0]);
4157
0
    const double dfDeltaY =
4158
0
        (ySMETransformed[2] - ySMETransformed[0]) / (x[nPoints - 1] - x[0]);
4159
0
    const double dfDeltaZ =
4160
0
        (zSMETransformed[2] - zSMETransformed[0]) / (x[nPoints - 1] - x[0]);
4161
4162
0
    const double dfError =
4163
0
        fabs((xSMETransformed[0] + dfDeltaX * (x[nMiddle] - x[0])) -
4164
0
             xSMETransformed[1]) +
4165
0
        fabs((ySMETransformed[0] + dfDeltaY * (x[nMiddle] - x[0])) -
4166
0
             ySMETransformed[1]);
4167
4168
0
    const double dfMaxError =
4169
0
        (bDstToSrc) ? psATInfo->dfMaxErrorReverse : psATInfo->dfMaxErrorForward;
4170
0
    if (dfError > dfMaxError)
4171
0
    {
4172
#if DEBUG_VERBOSE
4173
        CPLDebug("GDAL",
4174
                 "ApproxTransformer - "
4175
                 "error %g over threshold %g, subdivide %d points.",
4176
                 dfError, dfMaxError, nPoints);
4177
#endif
4178
4179
0
        double xMiddle[3] = {x[(nMiddle - 1) / 2], x[nMiddle - 1],
4180
0
                             x[nMiddle + (nPoints - nMiddle - 1) / 2]};
4181
0
        double yMiddle[3] = {y[(nMiddle - 1) / 2], y[nMiddle - 1],
4182
0
                             y[nMiddle + (nPoints - nMiddle - 1) / 2]};
4183
0
        double zMiddle[3] = {z[(nMiddle - 1) / 2], z[nMiddle - 1],
4184
0
                             z[nMiddle + (nPoints - nMiddle - 1) / 2]};
4185
4186
0
        const bool bUseBaseTransformForHalf1 =
4187
0
            nMiddle <= 5 || y[0] != y[nMiddle - 1] ||
4188
0
            y[0] != y[(nMiddle - 1) / 2] || x[0] == x[nMiddle - 1] ||
4189
0
            x[0] == x[(nMiddle - 1) / 2];
4190
0
        const bool bUseBaseTransformForHalf2 =
4191
0
            nPoints - nMiddle <= 5 || y[nMiddle] != y[nPoints - 1] ||
4192
0
            y[nMiddle] != y[nMiddle + (nPoints - nMiddle - 1) / 2] ||
4193
0
            x[nMiddle] == x[nPoints - 1] ||
4194
0
            x[nMiddle] == x[nMiddle + (nPoints - nMiddle - 1) / 2];
4195
4196
0
        int anSuccess2[3] = {};
4197
0
        int bSuccess = FALSE;
4198
0
        if (!bUseBaseTransformForHalf1 && !bUseBaseTransformForHalf2)
4199
0
            bSuccess = psATInfo->pfnBaseTransformer(
4200
0
                psATInfo->pBaseCBData, bDstToSrc, 3, xMiddle, yMiddle, zMiddle,
4201
0
                anSuccess2);
4202
0
        else if (!bUseBaseTransformForHalf1)
4203
0
        {
4204
0
            bSuccess = psATInfo->pfnBaseTransformer(
4205
0
                psATInfo->pBaseCBData, bDstToSrc, 2, xMiddle, yMiddle, zMiddle,
4206
0
                anSuccess2);
4207
0
            anSuccess2[2] = TRUE;
4208
0
        }
4209
0
        else if (!bUseBaseTransformForHalf2)
4210
0
        {
4211
0
            bSuccess = psATInfo->pfnBaseTransformer(
4212
0
                psATInfo->pBaseCBData, bDstToSrc, 1, xMiddle + 2, yMiddle + 2,
4213
0
                zMiddle + 2, anSuccess2 + 2);
4214
0
            anSuccess2[0] = TRUE;
4215
0
            anSuccess2[1] = TRUE;
4216
0
        }
4217
4218
0
        if (!bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2])
4219
0
        {
4220
0
            bSuccess = psATInfo->pfnBaseTransformer(
4221
0
                psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1,
4222
0
                z + 1, panSuccess + 1);
4223
0
            bSuccess &= psATInfo->pfnBaseTransformer(
4224
0
                psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2,
4225
0
                x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1,
4226
0
                panSuccess + nMiddle + 1);
4227
4228
0
            x[0] = xSMETransformed[0];
4229
0
            y[0] = ySMETransformed[0];
4230
0
            z[0] = zSMETransformed[0];
4231
0
            panSuccess[0] = TRUE;
4232
0
            x[nMiddle] = xSMETransformed[1];
4233
0
            y[nMiddle] = ySMETransformed[1];
4234
0
            z[nMiddle] = zSMETransformed[1];
4235
0
            panSuccess[nMiddle] = TRUE;
4236
0
            x[nPoints - 1] = xSMETransformed[2];
4237
0
            y[nPoints - 1] = ySMETransformed[2];
4238
0
            z[nPoints - 1] = zSMETransformed[2];
4239
0
            panSuccess[nPoints - 1] = TRUE;
4240
0
            return bSuccess;
4241
0
        }
4242
4243
0
        double x2[3] = {};
4244
0
        double y2[3] = {};
4245
0
        double z2[3] = {};
4246
0
        if (!bUseBaseTransformForHalf1)
4247
0
        {
4248
0
            x2[0] = xSMETransformed[0];
4249
0
            y2[0] = ySMETransformed[0];
4250
0
            z2[0] = zSMETransformed[0];
4251
0
            x2[1] = xMiddle[0];
4252
0
            y2[1] = yMiddle[0];
4253
0
            z2[1] = zMiddle[0];
4254
0
            x2[2] = xMiddle[1];
4255
0
            y2[2] = yMiddle[1];
4256
0
            z2[2] = zMiddle[1];
4257
4258
0
            bSuccess = GDALApproxTransformInternal(
4259
0
                psATInfo, bDstToSrc, nMiddle, x, y, z, panSuccess, x2, y2, z2);
4260
0
        }
4261
0
        else
4262
0
        {
4263
0
            bSuccess = psATInfo->pfnBaseTransformer(
4264
0
                psATInfo->pBaseCBData, bDstToSrc, nMiddle - 1, x + 1, y + 1,
4265
0
                z + 1, panSuccess + 1);
4266
0
            x[0] = xSMETransformed[0];
4267
0
            y[0] = ySMETransformed[0];
4268
0
            z[0] = zSMETransformed[0];
4269
0
            panSuccess[0] = TRUE;
4270
0
        }
4271
4272
0
        if (!bSuccess)
4273
0
            return FALSE;
4274
4275
0
        if (!bUseBaseTransformForHalf2)
4276
0
        {
4277
0
            x2[0] = xSMETransformed[1];
4278
0
            y2[0] = ySMETransformed[1];
4279
0
            z2[0] = zSMETransformed[1];
4280
0
            x2[1] = xMiddle[2];
4281
0
            y2[1] = yMiddle[2];
4282
0
            z2[1] = zMiddle[2];
4283
0
            x2[2] = xSMETransformed[2];
4284
0
            y2[2] = ySMETransformed[2];
4285
0
            z2[2] = zSMETransformed[2];
4286
4287
0
            bSuccess = GDALApproxTransformInternal(
4288
0
                psATInfo, bDstToSrc, nPoints - nMiddle, x + nMiddle,
4289
0
                y + nMiddle, z + nMiddle, panSuccess + nMiddle, x2, y2, z2);
4290
0
        }
4291
0
        else
4292
0
        {
4293
0
            bSuccess = psATInfo->pfnBaseTransformer(
4294
0
                psATInfo->pBaseCBData, bDstToSrc, nPoints - nMiddle - 2,
4295
0
                x + nMiddle + 1, y + nMiddle + 1, z + nMiddle + 1,
4296
0
                panSuccess + nMiddle + 1);
4297
4298
0
            x[nMiddle] = xSMETransformed[1];
4299
0
            y[nMiddle] = ySMETransformed[1];
4300
0
            z[nMiddle] = zSMETransformed[1];
4301
0
            panSuccess[nMiddle] = TRUE;
4302
0
            x[nPoints - 1] = xSMETransformed[2];
4303
0
            y[nPoints - 1] = ySMETransformed[2];
4304
0
            z[nPoints - 1] = zSMETransformed[2];
4305
0
            panSuccess[nPoints - 1] = TRUE;
4306
0
        }
4307
4308
0
        if (!bSuccess)
4309
0
            return FALSE;
4310
4311
0
        return TRUE;
4312
0
    }
4313
4314
    /* -------------------------------------------------------------------- */
4315
    /*      Error is OK since this is just used to compute output bounds    */
4316
    /*      of newly created file for gdalwarper.  So just use affine       */
4317
    /*      approximation of the reverse transform.  Eventually we          */
4318
    /*      should implement iterative searching to find a result within    */
4319
    /*      our error threshold.                                            */
4320
    /*      NOTE: the above comment is not true: gdalwarp uses approximator */
4321
    /*      also to compute the source pixel of each target pixel.          */
4322
    /* -------------------------------------------------------------------- */
4323
0
    for (int i = nPoints - 1; i >= 0; i--)
4324
0
    {
4325
#ifdef check_error
4326
        double xtemp = x[i];
4327
        double ytemp = y[i];
4328
        double ztemp = z[i];
4329
        double x_ori = xtemp;
4330
        double y_ori = ytemp;
4331
        int btemp = FALSE;
4332
        psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 1,
4333
                                     &xtemp, &ytemp, &ztemp, &btemp);
4334
#endif
4335
0
        const double dfDist = (x[i] - x[0]);
4336
0
        x[i] = xSMETransformed[0] + dfDeltaX * dfDist;
4337
0
        y[i] = ySMETransformed[0] + dfDeltaY * dfDist;
4338
0
        z[i] = zSMETransformed[0] + dfDeltaZ * dfDist;
4339
#ifdef check_error
4340
        const double dfError2 = fabs(x[i] - xtemp) + fabs(y[i] - ytemp);
4341
        if (dfError2 > 4 /*10 * dfMaxError*/)
4342
        {
4343
            /*ok*/ printf("Error = %f on (%f, %f)\n", dfError2, x_ori, y_ori);
4344
        }
4345
#endif
4346
0
        panSuccess[i] = TRUE;
4347
0
    }
4348
4349
0
    return TRUE;
4350
0
}
4351
4352
/************************************************************************/
4353
/*                        GDALApproxTransform()                         */
4354
/************************************************************************/
4355
4356
/**
4357
 * Perform approximate transformation.
4358
 *
4359
 * Actually performs the approximate transformation described in
4360
 * GDALCreateApproxTransformer().  This function matches the
4361
 * GDALTransformerFunc() signature.  Details of the arguments are described
4362
 * there.
4363
 */
4364
4365
int GDALApproxTransform(void *pCBData, int bDstToSrc, int nPoints, double *x,
4366
                        double *y, double *z, int *panSuccess)
4367
4368
0
{
4369
0
    GDALApproxTransformInfo *psATInfo =
4370
0
        static_cast<GDALApproxTransformInfo *>(pCBData);
4371
0
    double x2[3] = {};
4372
0
    double y2[3] = {};
4373
0
    double z2[3] = {};
4374
0
    int anSuccess2[3] = {};
4375
0
    int bSuccess;
4376
4377
0
    const int nMiddle = (nPoints - 1) / 2;
4378
4379
    /* -------------------------------------------------------------------- */
4380
    /*      Bail if our preconditions are not met, or if error is not       */
4381
    /*      acceptable.                                                     */
4382
    /* -------------------------------------------------------------------- */
4383
0
    int bRet = FALSE;
4384
0
    if (y[0] != y[nPoints - 1] || y[0] != y[nMiddle] ||
4385
0
        x[0] == x[nPoints - 1] || x[0] == x[nMiddle] ||
4386
0
        (psATInfo->dfMaxErrorForward == 0.0 &&
4387
0
         psATInfo->dfMaxErrorReverse == 0.0) ||
4388
0
        nPoints <= 5)
4389
0
    {
4390
0
        bRet = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc,
4391
0
                                            nPoints, x, y, z, panSuccess);
4392
0
        goto end;
4393
0
    }
4394
4395
    /* -------------------------------------------------------------------- */
4396
    /*      Transform first, last and middle point.                         */
4397
    /* -------------------------------------------------------------------- */
4398
0
    x2[0] = x[0];
4399
0
    y2[0] = y[0];
4400
0
    z2[0] = z[0];
4401
0
    x2[1] = x[nMiddle];
4402
0
    y2[1] = y[nMiddle];
4403
0
    z2[1] = z[nMiddle];
4404
0
    x2[2] = x[nPoints - 1];
4405
0
    y2[2] = y[nPoints - 1];
4406
0
    z2[2] = z[nPoints - 1];
4407
4408
0
    bSuccess = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc, 3,
4409
0
                                            x2, y2, z2, anSuccess2);
4410
0
    if (!bSuccess || !anSuccess2[0] || !anSuccess2[1] || !anSuccess2[2])
4411
0
    {
4412
0
        bRet = psATInfo->pfnBaseTransformer(psATInfo->pBaseCBData, bDstToSrc,
4413
0
                                            nPoints, x, y, z, panSuccess);
4414
0
        goto end;
4415
0
    }
4416
4417
0
    bRet = GDALApproxTransformInternal(pCBData, bDstToSrc, nPoints, x, y, z,
4418
0
                                       panSuccess, x2, y2, z2);
4419
4420
0
end:
4421
#ifdef DEBUG_APPROX_TRANSFORMER
4422
    for (int i = 0; i < nPoints; i++)
4423
        fprintf(stderr, "[%d] (%.10f,%.10f) %d\n", /*ok*/
4424
                i, x[i], y[i], panSuccess[i]);
4425
#endif
4426
4427
0
    return bRet;
4428
0
}
4429
4430
/************************************************************************/
4431
/*                  GDALDeserializeApproxTransformer()                  */
4432
/************************************************************************/
4433
4434
static void *GDALDeserializeApproxTransformer(CPLXMLNode *psTree)
4435
4436
0
{
4437
0
    double dfMaxErrorForward = 0.25;
4438
0
    double dfMaxErrorReverse = 0.25;
4439
0
    const char *pszMaxError = CPLGetXMLValue(psTree, "MaxError", nullptr);
4440
0
    if (pszMaxError != nullptr)
4441
0
    {
4442
0
        dfMaxErrorForward = CPLAtof(pszMaxError);
4443
0
        dfMaxErrorReverse = dfMaxErrorForward;
4444
0
    }
4445
0
    const char *pszMaxErrorForward =
4446
0
        CPLGetXMLValue(psTree, "MaxErrorForward", nullptr);
4447
0
    if (pszMaxErrorForward != nullptr)
4448
0
    {
4449
0
        dfMaxErrorForward = CPLAtof(pszMaxErrorForward);
4450
0
    }
4451
0
    const char *pszMaxErrorReverse =
4452
0
        CPLGetXMLValue(psTree, "MaxErrorReverse", nullptr);
4453
0
    if (pszMaxErrorReverse != nullptr)
4454
0
    {
4455
0
        dfMaxErrorReverse = CPLAtof(pszMaxErrorReverse);
4456
0
    }
4457
4458
0
    GDALTransformerFunc pfnBaseTransform = nullptr;
4459
0
    void *pBaseCBData = nullptr;
4460
4461
0
    CPLXMLNode *psContainer = CPLGetXMLNode(psTree, "BaseTransformer");
4462
4463
0
    if (psContainer != nullptr && psContainer->psChild != nullptr)
4464
0
    {
4465
0
        GDALDeserializeTransformer(psContainer->psChild, &pfnBaseTransform,
4466
0
                                   &pBaseCBData);
4467
0
    }
4468
4469
0
    if (pfnBaseTransform == nullptr)
4470
0
    {
4471
0
        CPLError(CE_Failure, CPLE_AppDefined,
4472
0
                 "Cannot get base transform for approx transformer.");
4473
0
        return nullptr;
4474
0
    }
4475
4476
0
    void *pApproxCBData = GDALCreateApproxTransformer2(
4477
0
        pfnBaseTransform, pBaseCBData, dfMaxErrorForward, dfMaxErrorReverse);
4478
0
    GDALApproxTransformerOwnsSubtransformer(pApproxCBData, TRUE);
4479
4480
0
    return pApproxCBData;
4481
0
}
4482
4483
/************************************************************************/
4484
/*                 GDALTransformLonLatToDestApproxTransformer()         */
4485
/************************************************************************/
4486
4487
int GDALTransformLonLatToDestApproxTransformer(void *hTransformArg,
4488
                                               double *pdfX, double *pdfY)
4489
0
{
4490
0
    GDALApproxTransformInfo *psInfo =
4491
0
        static_cast<GDALApproxTransformInfo *>(hTransformArg);
4492
4493
0
    if (GDALIsTransformer(psInfo->pBaseCBData,
4494
0
                          GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4495
0
    {
4496
0
        return GDALTransformLonLatToDestGenImgProjTransformer(
4497
0
            psInfo->pBaseCBData, pdfX, pdfY);
4498
0
    }
4499
0
    return false;
4500
0
}
4501
4502
/************************************************************************/
4503
/*                       GDALApplyGeoTransform()                        */
4504
/************************************************************************/
4505
4506
/**
4507
 * Apply GeoTransform to x/y coordinate.
4508
 *
4509
 * Applies the following computation, converting a (pixel, line) coordinate
4510
 * into a georeferenced (geo_x, geo_y) location.
4511
 * \code{.c}
4512
 *  *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1]
4513
 *                                 + dfLine  * padfGeoTransform[2];
4514
 *  *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4]
4515
 *                                 + dfLine  * padfGeoTransform[5];
4516
 * \endcode
4517
 *
4518
 * @param padfGeoTransform Six coefficient GeoTransform to apply.
4519
 * @param dfPixel Input pixel position.
4520
 * @param dfLine Input line position.
4521
 * @param pdfGeoX output location where geo_x (easting/longitude)
4522
 * location is placed.
4523
 * @param pdfGeoY output location where geo_y (northing/latitude)
4524
 * location is placed.
4525
 */
4526
4527
void CPL_STDCALL GDALApplyGeoTransform(const double *padfGeoTransform,
4528
                                       double dfPixel, double dfLine,
4529
                                       double *pdfGeoX, double *pdfGeoY)
4530
0
{
4531
0
    *pdfGeoX = padfGeoTransform[0] + dfPixel * padfGeoTransform[1] +
4532
0
               dfLine * padfGeoTransform[2];
4533
0
    *pdfGeoY = padfGeoTransform[3] + dfPixel * padfGeoTransform[4] +
4534
0
               dfLine * padfGeoTransform[5];
4535
0
}
4536
4537
/************************************************************************/
4538
/*                        GDALInvGeoTransform()                         */
4539
/************************************************************************/
4540
4541
/**
4542
 * Invert Geotransform.
4543
 *
4544
 * This function will invert a standard 3x2 set of GeoTransform coefficients.
4545
 * This converts the equation from being pixel to geo to being geo to pixel.
4546
 *
4547
 * @param gt_in Input geotransform (six doubles - unaltered).
4548
 * @param gt_out Output geotransform (six doubles - updated).
4549
 *
4550
 * @return TRUE on success or FALSE if the equation is uninvertable.
4551
 */
4552
4553
int CPL_STDCALL GDALInvGeoTransform(const double *gt_in, double *gt_out)
4554
4555
0
{
4556
    // Special case - no rotation - to avoid computing determinate
4557
    // and potential precision issues.
4558
0
    if (gt_in[2] == 0.0 && gt_in[4] == 0.0 && gt_in[1] != 0.0 &&
4559
0
        gt_in[5] != 0.0)
4560
0
    {
4561
        /*X = gt_in[0] + x * gt_in[1]
4562
          Y = gt_in[3] + y * gt_in[5]
4563
          -->
4564
          x = -gt_in[0] / gt_in[1] + (1 / gt_in[1]) * X
4565
          y = -gt_in[3] / gt_in[5] + (1 / gt_in[5]) * Y
4566
        */
4567
0
        gt_out[0] = -gt_in[0] / gt_in[1];
4568
0
        gt_out[1] = 1.0 / gt_in[1];
4569
0
        gt_out[2] = 0.0;
4570
0
        gt_out[3] = -gt_in[3] / gt_in[5];
4571
0
        gt_out[4] = 0.0;
4572
0
        gt_out[5] = 1.0 / gt_in[5];
4573
0
        return 1;
4574
0
    }
4575
4576
    // Assume a 3rd row that is [1 0 0].
4577
4578
    // Compute determinate.
4579
4580
0
    const double det = gt_in[1] * gt_in[5] - gt_in[2] * gt_in[4];
4581
0
    const double magnitude = std::max(std::max(fabs(gt_in[1]), fabs(gt_in[2])),
4582
0
                                      std::max(fabs(gt_in[4]), fabs(gt_in[5])));
4583
4584
0
    if (fabs(det) <= 1e-10 * magnitude * magnitude)
4585
0
        return 0;
4586
4587
0
    const double inv_det = 1.0 / det;
4588
4589
    // Compute adjoint, and divide by determinate.
4590
4591
0
    gt_out[1] = gt_in[5] * inv_det;
4592
0
    gt_out[4] = -gt_in[4] * inv_det;
4593
4594
0
    gt_out[2] = -gt_in[2] * inv_det;
4595
0
    gt_out[5] = gt_in[1] * inv_det;
4596
4597
0
    gt_out[0] = (gt_in[2] * gt_in[3] - gt_in[0] * gt_in[5]) * inv_det;
4598
0
    gt_out[3] = (-gt_in[1] * gt_in[3] + gt_in[0] * gt_in[4]) * inv_det;
4599
4600
0
    return 1;
4601
0
}
4602
4603
/************************************************************************/
4604
/*                      GDALSerializeTransformer()                      */
4605
/************************************************************************/
4606
4607
CPLXMLNode *GDALSerializeTransformer(GDALTransformerFunc /* pfnFunc */,
4608
                                     void *pTransformArg)
4609
0
{
4610
0
    VALIDATE_POINTER1(pTransformArg, "GDALSerializeTransformer", nullptr);
4611
4612
0
    GDALTransformerInfo *psInfo =
4613
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4614
4615
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4616
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4617
0
    {
4618
0
        CPLError(CE_Failure, CPLE_AppDefined,
4619
0
                 "Attempt to serialize non-GTI2 transformer.");
4620
0
        return nullptr;
4621
0
    }
4622
0
    else if (psInfo->pfnSerialize == nullptr)
4623
0
    {
4624
0
        CPLError(CE_Failure, CPLE_AppDefined,
4625
0
                 "No serialization function available for this transformer.");
4626
0
        return nullptr;
4627
0
    }
4628
4629
0
    return psInfo->pfnSerialize(pTransformArg);
4630
0
}
4631
4632
/************************************************************************/
4633
/*                  GDALRegisterTransformDeserializer()                 */
4634
/************************************************************************/
4635
4636
static CPLList *psListDeserializer = nullptr;
4637
static CPLMutex *hDeserializerMutex = nullptr;
4638
4639
typedef struct
4640
{
4641
    char *pszTransformName;
4642
    GDALTransformerFunc pfnTransformerFunc;
4643
    GDALTransformDeserializeFunc pfnDeserializeFunc;
4644
} TransformDeserializerInfo;
4645
4646
void *GDALRegisterTransformDeserializer(
4647
    const char *pszTransformName, GDALTransformerFunc pfnTransformerFunc,
4648
    GDALTransformDeserializeFunc pfnDeserializeFunc)
4649
0
{
4650
0
    TransformDeserializerInfo *psInfo =
4651
0
        static_cast<TransformDeserializerInfo *>(
4652
0
            CPLMalloc(sizeof(TransformDeserializerInfo)));
4653
0
    psInfo->pszTransformName = CPLStrdup(pszTransformName);
4654
0
    psInfo->pfnTransformerFunc = pfnTransformerFunc;
4655
0
    psInfo->pfnDeserializeFunc = pfnDeserializeFunc;
4656
4657
0
    CPLMutexHolderD(&hDeserializerMutex);
4658
0
    psListDeserializer = CPLListInsert(psListDeserializer, psInfo, 0);
4659
4660
0
    return psInfo;
4661
0
}
4662
4663
/************************************************************************/
4664
/*                GDALUnregisterTransformDeserializer()                 */
4665
/************************************************************************/
4666
4667
void GDALUnregisterTransformDeserializer(void *pData)
4668
0
{
4669
0
    CPLMutexHolderD(&hDeserializerMutex);
4670
0
    CPLList *psList = psListDeserializer;
4671
0
    CPLList *psLast = nullptr;
4672
0
    while (psList)
4673
0
    {
4674
0
        if (psList->pData == pData)
4675
0
        {
4676
0
            TransformDeserializerInfo *psInfo =
4677
0
                static_cast<TransformDeserializerInfo *>(pData);
4678
0
            CPLFree(psInfo->pszTransformName);
4679
0
            CPLFree(pData);
4680
0
            if (psLast)
4681
0
                psLast->psNext = psList->psNext;
4682
0
            else
4683
0
                psListDeserializer = nullptr;
4684
0
            CPLFree(psList);
4685
0
            break;
4686
0
        }
4687
0
        psLast = psList;
4688
0
        psList = psList->psNext;
4689
0
    }
4690
0
}
4691
4692
/************************************************************************/
4693
/*                GDALUnregisterTransformDeserializer()                 */
4694
/************************************************************************/
4695
4696
void GDALCleanupTransformDeserializerMutex()
4697
0
{
4698
0
    if (hDeserializerMutex != nullptr)
4699
0
    {
4700
0
        CPLDestroyMutex(hDeserializerMutex);
4701
0
        hDeserializerMutex = nullptr;
4702
0
    }
4703
0
}
4704
4705
/************************************************************************/
4706
/*                     GDALDeserializeTransformer()                     */
4707
/************************************************************************/
4708
4709
CPLErr GDALDeserializeTransformer(CPLXMLNode *psTree,
4710
                                  GDALTransformerFunc *ppfnFunc,
4711
                                  void **ppTransformArg)
4712
4713
0
{
4714
0
    *ppfnFunc = nullptr;
4715
0
    *ppTransformArg = nullptr;
4716
4717
0
    CPLErrorReset();
4718
4719
0
    if (psTree == nullptr || psTree->eType != CXT_Element)
4720
0
        CPLError(CE_Failure, CPLE_AppDefined,
4721
0
                 "Malformed element in GDALDeserializeTransformer");
4722
0
    else if (EQUAL(psTree->pszValue, "GenImgProjTransformer"))
4723
0
    {
4724
0
        *ppfnFunc = GDALGenImgProjTransform;
4725
0
        *ppTransformArg = GDALDeserializeGenImgProjTransformer(psTree);
4726
0
    }
4727
0
    else if (EQUAL(psTree->pszValue, "ReprojectionTransformer"))
4728
0
    {
4729
0
        *ppfnFunc = GDALReprojectionTransform;
4730
0
        *ppTransformArg = GDALDeserializeReprojectionTransformer(psTree);
4731
0
    }
4732
0
    else if (EQUAL(psTree->pszValue, "GCPTransformer"))
4733
0
    {
4734
0
        *ppfnFunc = GDALGCPTransform;
4735
0
        *ppTransformArg = GDALDeserializeGCPTransformer(psTree);
4736
0
    }
4737
0
    else if (EQUAL(psTree->pszValue, "TPSTransformer"))
4738
0
    {
4739
0
        *ppfnFunc = GDALTPSTransform;
4740
0
        *ppTransformArg = GDALDeserializeTPSTransformer(psTree);
4741
0
    }
4742
0
    else if (EQUAL(psTree->pszValue, "GeoLocTransformer"))
4743
0
    {
4744
0
        *ppfnFunc = GDALGeoLocTransform;
4745
0
        *ppTransformArg = GDALDeserializeGeoLocTransformer(psTree);
4746
0
    }
4747
0
    else if (EQUAL(psTree->pszValue, "RPCTransformer"))
4748
0
    {
4749
0
        *ppfnFunc = GDALRPCTransform;
4750
0
        *ppTransformArg = GDALDeserializeRPCTransformer(psTree);
4751
0
    }
4752
0
    else if (EQUAL(psTree->pszValue, "ApproxTransformer"))
4753
0
    {
4754
0
        *ppfnFunc = GDALApproxTransform;
4755
0
        *ppTransformArg = GDALDeserializeApproxTransformer(psTree);
4756
0
    }
4757
0
    else if (EQUAL(psTree->pszValue, "HomographyTransformer"))
4758
0
    {
4759
0
        *ppfnFunc = GDALHomographyTransform;
4760
0
        *ppTransformArg = GDALDeserializeHomographyTransformer(psTree);
4761
0
    }
4762
0
    else
4763
0
    {
4764
0
        GDALTransformDeserializeFunc pfnDeserializeFunc = nullptr;
4765
0
        {
4766
0
            CPLMutexHolderD(&hDeserializerMutex);
4767
0
            CPLList *psList = psListDeserializer;
4768
0
            while (psList)
4769
0
            {
4770
0
                TransformDeserializerInfo *psInfo =
4771
0
                    static_cast<TransformDeserializerInfo *>(psList->pData);
4772
0
                if (strcmp(psInfo->pszTransformName, psTree->pszValue) == 0)
4773
0
                {
4774
0
                    *ppfnFunc = psInfo->pfnTransformerFunc;
4775
0
                    pfnDeserializeFunc = psInfo->pfnDeserializeFunc;
4776
0
                    break;
4777
0
                }
4778
0
                psList = psList->psNext;
4779
0
            }
4780
0
        }
4781
4782
0
        if (pfnDeserializeFunc != nullptr)
4783
0
        {
4784
0
            *ppTransformArg = pfnDeserializeFunc(psTree);
4785
0
        }
4786
0
        else
4787
0
        {
4788
0
            CPLError(CE_Failure, CPLE_AppDefined,
4789
0
                     "Unrecognized element '%s' GDALDeserializeTransformer",
4790
0
                     psTree->pszValue);
4791
0
        }
4792
0
    }
4793
4794
0
    return CPLGetLastErrorType();
4795
0
}
4796
4797
/************************************************************************/
4798
/*                       GDALDestroyTransformer()                       */
4799
/************************************************************************/
4800
4801
void GDALDestroyTransformer(void *pTransformArg)
4802
4803
0
{
4804
0
    if (pTransformArg == nullptr)
4805
0
        return;
4806
4807
0
    GDALTransformerInfo *psInfo =
4808
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4809
4810
0
    if (memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4811
0
               strlen(GDAL_GTI2_SIGNATURE)) != 0)
4812
0
    {
4813
0
        CPLError(CE_Failure, CPLE_AppDefined,
4814
0
                 "Attempt to destroy non-GTI2 transformer.");
4815
0
        return;
4816
0
    }
4817
4818
0
    psInfo->pfnCleanup(pTransformArg);
4819
0
}
4820
4821
/************************************************************************/
4822
/*                         GDALUseTransformer()                         */
4823
/************************************************************************/
4824
4825
int GDALUseTransformer(void *pTransformArg, int bDstToSrc, int nPointCount,
4826
                       double *x, double *y, double *z, int *panSuccess)
4827
0
{
4828
0
    GDALTransformerInfo *psInfo =
4829
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4830
4831
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4832
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4833
0
    {
4834
0
        CPLError(CE_Failure, CPLE_AppDefined,
4835
0
                 "Attempt to use non-GTI2 transformer.");
4836
0
        return FALSE;
4837
0
    }
4838
4839
0
    return psInfo->pfnTransform(pTransformArg, bDstToSrc, nPointCount, x, y, z,
4840
0
                                panSuccess);
4841
0
}
4842
4843
/************************************************************************/
4844
/*                        GDALCloneTransformer()                        */
4845
/************************************************************************/
4846
4847
void *GDALCloneTransformer(void *pTransformArg)
4848
0
{
4849
0
    GDALTransformerInfo *psInfo =
4850
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4851
4852
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4853
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4854
0
    {
4855
0
        CPLError(CE_Failure, CPLE_AppDefined,
4856
0
                 "Attempt to clone non-GTI2 transformer.");
4857
0
        return nullptr;
4858
0
    }
4859
4860
0
    if (psInfo->pfnCreateSimilar != nullptr)
4861
0
    {
4862
0
        return psInfo->pfnCreateSimilar(psInfo, 1.0, 1.0);
4863
0
    }
4864
4865
0
    if (psInfo->pfnSerialize == nullptr)
4866
0
    {
4867
0
        CPLError(CE_Failure, CPLE_AppDefined,
4868
0
                 "No serialization function available for this transformer.");
4869
0
        return nullptr;
4870
0
    }
4871
4872
0
    CPLXMLNode *pSerialized = psInfo->pfnSerialize(pTransformArg);
4873
0
    if (pSerialized == nullptr)
4874
0
        return nullptr;
4875
0
    GDALTransformerFunc pfnTransformer = nullptr;
4876
0
    void *pClonedTransformArg = nullptr;
4877
0
    if (GDALDeserializeTransformer(pSerialized, &pfnTransformer,
4878
0
                                   &pClonedTransformArg) != CE_None)
4879
0
    {
4880
0
        CPLDestroyXMLNode(pSerialized);
4881
0
        CPLFree(pClonedTransformArg);
4882
0
        return nullptr;
4883
0
    }
4884
4885
0
    CPLDestroyXMLNode(pSerialized);
4886
0
    return pClonedTransformArg;
4887
0
}
4888
4889
/************************************************************************/
4890
/*                   GDALCreateSimilarTransformer()                     */
4891
/************************************************************************/
4892
4893
void *GDALCreateSimilarTransformer(void *pTransformArg, double dfRatioX,
4894
                                   double dfRatioY)
4895
0
{
4896
0
    GDALTransformerInfo *psInfo =
4897
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4898
4899
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4900
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4901
0
    {
4902
0
        CPLError(CE_Failure, CPLE_AppDefined,
4903
0
                 "Attempt to call CreateSimilar on a non-GTI2 transformer.");
4904
0
        return nullptr;
4905
0
    }
4906
4907
0
    if (psInfo->pfnCreateSimilar == nullptr)
4908
0
    {
4909
0
        CPLError(CE_Failure, CPLE_AppDefined,
4910
0
                 "No CreateSimilar function available for this transformer.");
4911
0
        return nullptr;
4912
0
    }
4913
4914
0
    return psInfo->pfnCreateSimilar(psInfo, dfRatioX, dfRatioY);
4915
0
}
4916
4917
/************************************************************************/
4918
/*                      GetGenImgProjTransformInfo()                    */
4919
/************************************************************************/
4920
4921
static GDALTransformerInfo *GetGenImgProjTransformInfo(const char *pszFunc,
4922
                                                       void *pTransformArg)
4923
0
{
4924
0
    GDALTransformerInfo *psInfo =
4925
0
        static_cast<GDALTransformerInfo *>(pTransformArg);
4926
4927
0
    if (psInfo == nullptr || memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4928
0
                                    strlen(GDAL_GTI2_SIGNATURE)) != 0)
4929
0
    {
4930
0
        CPLError(CE_Failure, CPLE_AppDefined,
4931
0
                 "Attempt to call %s on "
4932
0
                 "a non-GTI2 transformer.",
4933
0
                 pszFunc);
4934
0
        return nullptr;
4935
0
    }
4936
4937
0
    if (EQUAL(psInfo->pszClassName, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
4938
0
    {
4939
0
        GDALApproxTransformInfo *psATInfo =
4940
0
            static_cast<GDALApproxTransformInfo *>(pTransformArg);
4941
0
        psInfo = static_cast<GDALTransformerInfo *>(psATInfo->pBaseCBData);
4942
4943
0
        if (psInfo == nullptr ||
4944
0
            memcmp(psInfo->abySignature, GDAL_GTI2_SIGNATURE,
4945
0
                   strlen(GDAL_GTI2_SIGNATURE)) != 0)
4946
0
        {
4947
0
            CPLError(CE_Failure, CPLE_AppDefined,
4948
0
                     "Attempt to call %s on "
4949
0
                     "a non-GTI2 transformer.",
4950
0
                     pszFunc);
4951
0
            return nullptr;
4952
0
        }
4953
0
    }
4954
4955
0
    if (EQUAL(psInfo->pszClassName, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
4956
0
    {
4957
0
        return psInfo;
4958
0
    }
4959
4960
0
    return nullptr;
4961
0
}
4962
4963
/************************************************************************/
4964
/*                 GDALSetTransformerDstGeoTransform()                  */
4965
/************************************************************************/
4966
4967
/**
4968
 * Set ApproxTransformer or GenImgProj output geotransform.
4969
 *
4970
 * This is a layer above GDALSetGenImgProjTransformerDstGeoTransform() that
4971
 * checks that the passed hTransformArg is compatible.
4972
 *
4973
 * Normally the "destination geotransform", or transformation between
4974
 * georeferenced output coordinates and pixel/line coordinates on the
4975
 * destination file is extracted from the destination file by
4976
 * GDALCreateGenImgProjTransformer() and stored in the GenImgProj private
4977
 * info.  However, sometimes it is inconvenient to have an output file
4978
 * handle with appropriate geotransform information when creating the
4979
 * transformation.  For these cases, this function can be used to apply
4980
 * the destination geotransform.
4981
 *
4982
 * @param pTransformArg the handle to update.
4983
 * @param padfGeoTransform the destination geotransform to apply (six doubles).
4984
 */
4985
4986
void GDALSetTransformerDstGeoTransform(void *pTransformArg,
4987
                                       const double *padfGeoTransform)
4988
0
{
4989
0
    VALIDATE_POINTER0(pTransformArg, "GDALSetTransformerDstGeoTransform");
4990
4991
0
    GDALTransformerInfo *psInfo = GetGenImgProjTransformInfo(
4992
0
        "GDALSetTransformerDstGeoTransform", pTransformArg);
4993
0
    if (psInfo)
4994
0
    {
4995
0
        GDALSetGenImgProjTransformerDstGeoTransform(psInfo, padfGeoTransform);
4996
0
    }
4997
0
}
4998
4999
/************************************************************************/
5000
/*                 GDALGetTransformerDstGeoTransform()                  */
5001
/************************************************************************/
5002
5003
/**
5004
 * Get ApproxTransformer or GenImgProj output geotransform.
5005
 *
5006
 * @param pTransformArg transformer handle.
5007
 * @param padfGeoTransform (output) the destination geotransform to return (six
5008
 * doubles).
5009
 */
5010
5011
void GDALGetTransformerDstGeoTransform(void *pTransformArg,
5012
                                       double *padfGeoTransform)
5013
0
{
5014
0
    VALIDATE_POINTER0(pTransformArg, "GDALGetTransformerDstGeoTransform");
5015
5016
0
    GDALTransformerInfo *psInfo = GetGenImgProjTransformInfo(
5017
0
        "GDALGetTransformerDstGeoTransform", pTransformArg);
5018
0
    if (psInfo)
5019
0
    {
5020
0
        GDALGenImgProjTransformInfo *psGenImgProjInfo =
5021
0
            reinterpret_cast<GDALGenImgProjTransformInfo *>(psInfo);
5022
5023
0
        memcpy(padfGeoTransform, psGenImgProjInfo->sDstParams.adfGeoTransform,
5024
0
               sizeof(double) * 6);
5025
0
    }
5026
0
}
5027
5028
/************************************************************************/
5029
/*            GDALTransformIsTranslationOnPixelBoundaries()             */
5030
/************************************************************************/
5031
5032
bool GDALTransformIsTranslationOnPixelBoundaries(GDALTransformerFunc,
5033
                                                 void *pTransformerArg)
5034
0
{
5035
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5036
0
    {
5037
0
        const auto *pApproxInfo =
5038
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5039
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5040
0
    }
5041
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5042
0
    {
5043
0
        const auto *pGenImgpProjInfo =
5044
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5045
0
        const auto IsCloseToInteger = [](double dfVal)
5046
0
        { return std::fabs(dfVal - std::round(dfVal)) <= 1e-6; };
5047
0
        return pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr &&
5048
0
               pGenImgpProjInfo->sDstParams.pTransformArg == nullptr &&
5049
0
               pGenImgpProjInfo->pReproject == nullptr &&
5050
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[1] ==
5051
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[1] &&
5052
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[5] ==
5053
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[5] &&
5054
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[2] ==
5055
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[2] &&
5056
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[4] ==
5057
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[4] &&
5058
               // Check that the georeferenced origin of the destination
5059
               // geotransform is close to be an integer value when transformed
5060
               // to source image coordinates
5061
0
               IsCloseToInteger(
5062
0
                   pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[0] +
5063
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[0] *
5064
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[1] +
5065
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[3] *
5066
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[2]) &&
5067
0
               IsCloseToInteger(
5068
0
                   pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[3] +
5069
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[0] *
5070
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[4] +
5071
0
                   pGenImgpProjInfo->sDstParams.adfGeoTransform[3] *
5072
0
                       pGenImgpProjInfo->sSrcParams.adfInvGeoTransform[5]);
5073
0
    }
5074
0
    return false;
5075
0
}
5076
5077
/************************************************************************/
5078
/*                   GDALTransformIsAffineNoRotation()                  */
5079
/************************************************************************/
5080
5081
bool GDALTransformIsAffineNoRotation(GDALTransformerFunc, void *pTransformerArg)
5082
0
{
5083
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5084
0
    {
5085
0
        const auto *pApproxInfo =
5086
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5087
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5088
0
    }
5089
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5090
0
    {
5091
0
        const auto *pGenImgpProjInfo =
5092
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5093
0
        return pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr &&
5094
0
               pGenImgpProjInfo->sDstParams.pTransformArg == nullptr &&
5095
0
               pGenImgpProjInfo->pReproject == nullptr &&
5096
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[2] == 0 &&
5097
0
               pGenImgpProjInfo->sSrcParams.adfGeoTransform[4] == 0 &&
5098
0
               pGenImgpProjInfo->sDstParams.adfGeoTransform[2] == 0 &&
5099
0
               pGenImgpProjInfo->sDstParams.adfGeoTransform[4] == 0;
5100
0
    }
5101
0
    return false;
5102
0
}
5103
5104
/************************************************************************/
5105
/*                        GDALTransformHasFastClone()                   */
5106
/************************************************************************/
5107
5108
/** Returns whether GDALCloneTransformer() on this transformer is
5109
 * "fast"
5110
 * Counter-examples are GCPs or TPSs transformers.
5111
 */
5112
bool GDALTransformHasFastClone(void *pTransformerArg)
5113
0
{
5114
0
    if (GDALIsTransformer(pTransformerArg, GDAL_APPROX_TRANSFORMER_CLASS_NAME))
5115
0
    {
5116
0
        const auto *pApproxInfo =
5117
0
            static_cast<const GDALApproxTransformInfo *>(pTransformerArg);
5118
0
        pTransformerArg = pApproxInfo->pBaseCBData;
5119
        // Fallback to next lines
5120
0
    }
5121
5122
0
    if (GDALIsTransformer(pTransformerArg, GDAL_GEN_IMG_TRANSFORMER_CLASS_NAME))
5123
0
    {
5124
0
        const auto *pGenImgpProjInfo =
5125
0
            static_cast<GDALGenImgProjTransformInfo *>(pTransformerArg);
5126
0
        return (pGenImgpProjInfo->sSrcParams.pTransformArg == nullptr ||
5127
0
                GDALTransformHasFastClone(
5128
0
                    pGenImgpProjInfo->sSrcParams.pTransformArg)) &&
5129
0
               (pGenImgpProjInfo->sDstParams.pTransformArg == nullptr ||
5130
0
                GDALTransformHasFastClone(
5131
0
                    pGenImgpProjInfo->sDstParams.pTransformArg));
5132
0
    }
5133
0
    else if (GDALIsTransformer(pTransformerArg,
5134
0
                               GDAL_RPC_TRANSFORMER_CLASS_NAME))
5135
0
    {
5136
0
        return true;
5137
0
    }
5138
0
    else
5139
0
    {
5140
0
        return false;
5141
0
    }
5142
0
}