Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: High Performance Image Reprojector
4 : * Purpose: Implementation of high level convenience APIs for warper.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2003, Frank Warmerdam <warmerdam@pobox.com>
9 : * Copyright (c) 2008-2012, Even Rouault <even dot rouault at spatialys.com>
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #include "cpl_port.h"
15 : #include "gdalwarper.h"
16 :
17 : #include <stdlib.h>
18 : #include <string.h>
19 :
20 : #include <algorithm>
21 : #include <cmath>
22 : #include <limits>
23 :
24 : #include "cpl_conv.h"
25 : #include "cpl_error.h"
26 : #include "cpl_mask.h"
27 : #include "cpl_minixml.h"
28 : #include "cpl_progress.h"
29 : #include "cpl_string.h"
30 : #include "cpl_vsi.h"
31 : #include "gdal.h"
32 : #include "gdal_priv.h"
33 : #include "ogr_api.h"
34 : #include "ogr_core.h"
35 : #include "vrtdataset.h" // for VRTSerializeNoData
36 :
37 : #if (defined(__x86_64) || defined(_M_X64))
38 : #include <emmintrin.h>
39 : #endif
40 :
41 : /************************************************************************/
42 : /* GDALReprojectImage() */
43 : /************************************************************************/
44 :
45 : /**
46 : * Reproject image.
47 : *
48 : * This is a convenience function utilizing the GDALWarpOperation class to
49 : * reproject an image from a source to a destination. In particular, this
50 : * function takes care of establishing the transformation function to
51 : * implement the reprojection, and will default a variety of other
52 : * warp options.
53 : *
54 : * Nodata values set on destination dataset are taken into account.
55 : *
56 : * No metadata, projection info, or color tables are transferred
57 : * to the output file. Source overviews are not considered.
58 : *
59 : * For more advanced warping capabilities, consider using GDALWarp().
60 : *
61 : * @param hSrcDS the source image file.
62 : * @param pszSrcWKT the source projection. If NULL the source projection
63 : * is read from from hSrcDS.
64 : * @param hDstDS the destination image file.
65 : * @param pszDstWKT the destination projection. If NULL the destination
66 : * projection will be read from hDstDS.
67 : * @param eResampleAlg the type of resampling to use.
68 : * @param dfWarpMemoryLimit the amount of memory (in bytes) that the warp
69 : * API is allowed to use for caching. This is in addition to the memory
70 : * already allocated to the GDAL caching (as per GDALSetCacheMax()). May be
71 : * 0.0 to use default memory settings.
72 : * @param dfMaxError maximum error measured in input pixels that is allowed
73 : * in approximating the transformation (0.0 for exact calculations).
74 : * @param pfnProgress a GDALProgressFunc() compatible callback function for
75 : * reporting progress or NULL.
76 : * @param pProgressArg argument to be passed to pfnProgress. May be NULL.
77 : * @param psOptions warp options, normally NULL.
78 : *
79 : * @return CE_None on success or CE_Failure if something goes wrong.
80 : * @see GDALWarp()
81 : */
82 :
83 90 : CPLErr CPL_STDCALL GDALReprojectImage(
84 : GDALDatasetH hSrcDS, const char *pszSrcWKT, GDALDatasetH hDstDS,
85 : const char *pszDstWKT, GDALResampleAlg eResampleAlg,
86 : CPL_UNUSED double dfWarpMemoryLimit, double dfMaxError,
87 : GDALProgressFunc pfnProgress, void *pProgressArg,
88 : GDALWarpOptions *psOptions)
89 :
90 : {
91 : /* -------------------------------------------------------------------- */
92 : /* Setup a reprojection based transformer. */
93 : /* -------------------------------------------------------------------- */
94 90 : void *hTransformArg = GDALCreateGenImgProjTransformer(
95 : hSrcDS, pszSrcWKT, hDstDS, pszDstWKT, TRUE, 1000.0, 0);
96 :
97 90 : if (hTransformArg == nullptr)
98 0 : return CE_Failure;
99 :
100 : /* -------------------------------------------------------------------- */
101 : /* Create a copy of the user provided options, or a defaulted */
102 : /* options structure. */
103 : /* -------------------------------------------------------------------- */
104 : GDALWarpOptions *psWOptions = psOptions == nullptr
105 90 : ? GDALCreateWarpOptions()
106 1 : : GDALCloneWarpOptions(psOptions);
107 :
108 90 : psWOptions->eResampleAlg = eResampleAlg;
109 :
110 : /* -------------------------------------------------------------------- */
111 : /* Set transform. */
112 : /* -------------------------------------------------------------------- */
113 90 : if (dfMaxError > 0.0)
114 : {
115 3 : psWOptions->pTransformerArg = GDALCreateApproxTransformer(
116 : GDALGenImgProjTransform, hTransformArg, dfMaxError);
117 :
118 3 : psWOptions->pfnTransformer = GDALApproxTransform;
119 : }
120 : else
121 : {
122 87 : psWOptions->pfnTransformer = GDALGenImgProjTransform;
123 87 : psWOptions->pTransformerArg = hTransformArg;
124 : }
125 :
126 : /* -------------------------------------------------------------------- */
127 : /* Set file and band mapping. */
128 : /* -------------------------------------------------------------------- */
129 90 : psWOptions->hSrcDS = hSrcDS;
130 90 : psWOptions->hDstDS = hDstDS;
131 :
132 90 : int nSrcBands = GDALGetRasterCount(hSrcDS);
133 : {
134 90 : GDALRasterBandH hBand = GDALGetRasterBand(hSrcDS, nSrcBands);
135 90 : if (hBand && GDALGetRasterColorInterpretation(hBand) == GCI_AlphaBand)
136 : {
137 12 : psWOptions->nSrcAlphaBand = nSrcBands;
138 12 : nSrcBands--;
139 : }
140 : }
141 :
142 90 : int nDstBands = GDALGetRasterCount(hDstDS);
143 : {
144 90 : GDALRasterBandH hBand = GDALGetRasterBand(hDstDS, nDstBands);
145 90 : if (hBand && GDALGetRasterColorInterpretation(hBand) == GCI_AlphaBand)
146 : {
147 12 : psWOptions->nDstAlphaBand = nDstBands;
148 12 : nDstBands--;
149 : }
150 : }
151 :
152 90 : GDALWarpInitDefaultBandMapping(psWOptions, std::min(nSrcBands, nDstBands));
153 :
154 : /* -------------------------------------------------------------------- */
155 : /* Set source nodata values if the source dataset seems to have */
156 : /* any. Same for target nodata values */
157 : /* -------------------------------------------------------------------- */
158 204 : for (int iBand = 0; iBand < psWOptions->nBandCount; iBand++)
159 : {
160 114 : GDALRasterBandH hBand = GDALGetRasterBand(hSrcDS, iBand + 1);
161 :
162 114 : int bGotNoData = FALSE;
163 114 : double dfNoDataValue = GDALGetRasterNoDataValue(hBand, &bGotNoData);
164 114 : if (bGotNoData)
165 : {
166 7 : GDALWarpInitSrcNoDataReal(psWOptions, -1.1e20);
167 7 : psWOptions->padfSrcNoDataReal[iBand] = dfNoDataValue;
168 : }
169 :
170 : // Deal with target band.
171 114 : hBand = GDALGetRasterBand(hDstDS, iBand + 1);
172 :
173 114 : dfNoDataValue = GDALGetRasterNoDataValue(hBand, &bGotNoData);
174 114 : if (bGotNoData)
175 : {
176 2 : GDALWarpInitDstNoDataReal(psWOptions, -1.1e20);
177 2 : psWOptions->padfDstNoDataReal[iBand] = dfNoDataValue;
178 : }
179 : }
180 :
181 : /* -------------------------------------------------------------------- */
182 : /* Set the progress function. */
183 : /* -------------------------------------------------------------------- */
184 90 : if (pfnProgress != nullptr)
185 : {
186 3 : psWOptions->pfnProgress = pfnProgress;
187 3 : psWOptions->pProgressArg = pProgressArg;
188 : }
189 :
190 : /* -------------------------------------------------------------------- */
191 : /* Create a warp options based on the options. */
192 : /* -------------------------------------------------------------------- */
193 90 : GDALWarpOperation oWarper;
194 90 : CPLErr eErr = oWarper.Initialize(psWOptions);
195 :
196 90 : if (eErr == CE_None)
197 90 : eErr = oWarper.ChunkAndWarpImage(0, 0, GDALGetRasterXSize(hDstDS),
198 : GDALGetRasterYSize(hDstDS));
199 :
200 : /* -------------------------------------------------------------------- */
201 : /* Cleanup. */
202 : /* -------------------------------------------------------------------- */
203 90 : GDALDestroyGenImgProjTransformer(hTransformArg);
204 :
205 90 : if (dfMaxError > 0.0)
206 3 : GDALDestroyApproxTransformer(psWOptions->pTransformerArg);
207 :
208 90 : GDALDestroyWarpOptions(psWOptions);
209 :
210 90 : return eErr;
211 : }
212 :
213 : /************************************************************************/
214 : /* GDALCreateAndReprojectImage() */
215 : /* */
216 : /* This is a "quicky" reprojection API. */
217 : /************************************************************************/
218 :
219 : /** Reproject an image and create the target reprojected image */
220 0 : CPLErr CPL_STDCALL GDALCreateAndReprojectImage(
221 : GDALDatasetH hSrcDS, const char *pszSrcWKT, const char *pszDstFilename,
222 : const char *pszDstWKT, GDALDriverH hDstDriver, char **papszCreateOptions,
223 : GDALResampleAlg eResampleAlg, double dfWarpMemoryLimit, double dfMaxError,
224 : GDALProgressFunc pfnProgress, void *pProgressArg,
225 : GDALWarpOptions *psOptions)
226 :
227 : {
228 0 : VALIDATE_POINTER1(hSrcDS, "GDALCreateAndReprojectImage", CE_Failure);
229 :
230 : /* -------------------------------------------------------------------- */
231 : /* Default a few parameters. */
232 : /* -------------------------------------------------------------------- */
233 0 : if (hDstDriver == nullptr)
234 : {
235 0 : hDstDriver = GDALGetDriverByName("GTiff");
236 0 : if (hDstDriver == nullptr)
237 : {
238 0 : CPLError(CE_Failure, CPLE_AppDefined,
239 : "GDALCreateAndReprojectImage needs GTiff driver");
240 0 : return CE_Failure;
241 : }
242 : }
243 :
244 0 : if (pszSrcWKT == nullptr)
245 0 : pszSrcWKT = GDALGetProjectionRef(hSrcDS);
246 :
247 0 : if (pszDstWKT == nullptr)
248 0 : pszDstWKT = pszSrcWKT;
249 :
250 : /* -------------------------------------------------------------------- */
251 : /* Create a transformation object from the source to */
252 : /* destination coordinate system. */
253 : /* -------------------------------------------------------------------- */
254 0 : void *hTransformArg = GDALCreateGenImgProjTransformer(
255 : hSrcDS, pszSrcWKT, nullptr, pszDstWKT, TRUE, 1000.0, 0);
256 :
257 0 : if (hTransformArg == nullptr)
258 0 : return CE_Failure;
259 :
260 : /* -------------------------------------------------------------------- */
261 : /* Get approximate output definition. */
262 : /* -------------------------------------------------------------------- */
263 0 : double adfDstGeoTransform[6] = {};
264 0 : int nPixels = 0;
265 0 : int nLines = 0;
266 :
267 0 : if (GDALSuggestedWarpOutput(hSrcDS, GDALGenImgProjTransform, hTransformArg,
268 : adfDstGeoTransform, &nPixels,
269 0 : &nLines) != CE_None)
270 0 : return CE_Failure;
271 :
272 0 : GDALDestroyGenImgProjTransformer(hTransformArg);
273 :
274 : /* -------------------------------------------------------------------- */
275 : /* Create the output file. */
276 : /* -------------------------------------------------------------------- */
277 0 : GDALDatasetH hDstDS = GDALCreate(
278 : hDstDriver, pszDstFilename, nPixels, nLines, GDALGetRasterCount(hSrcDS),
279 : GDALGetRasterDataType(GDALGetRasterBand(hSrcDS, 1)),
280 : papszCreateOptions);
281 :
282 0 : if (hDstDS == nullptr)
283 0 : return CE_Failure;
284 :
285 : /* -------------------------------------------------------------------- */
286 : /* Write out the projection definition. */
287 : /* -------------------------------------------------------------------- */
288 0 : GDALSetProjection(hDstDS, pszDstWKT);
289 0 : GDALSetGeoTransform(hDstDS, adfDstGeoTransform);
290 :
291 : /* -------------------------------------------------------------------- */
292 : /* Perform the reprojection. */
293 : /* -------------------------------------------------------------------- */
294 0 : CPLErr eErr = GDALReprojectImage(
295 : hSrcDS, pszSrcWKT, hDstDS, pszDstWKT, eResampleAlg, dfWarpMemoryLimit,
296 : dfMaxError, pfnProgress, pProgressArg, psOptions);
297 :
298 0 : GDALClose(hDstDS);
299 :
300 0 : return eErr;
301 : }
302 :
303 : /************************************************************************/
304 : /* GDALWarpNoDataMaskerT() */
305 : /************************************************************************/
306 :
307 : template <class T>
308 724 : static CPLErr GDALWarpNoDataMaskerT(const double *padfNoData, size_t nPixels,
309 : const T *pData, GUInt32 *panValidityMask,
310 : int *pbOutAllValid)
311 : {
312 : // Nothing to do if value is out of range.
313 724 : if (padfNoData[0] < std::numeric_limits<T>::min() ||
314 1446 : padfNoData[0] > std::numeric_limits<T>::max() + 0.000001 ||
315 722 : padfNoData[1] != 0.0)
316 : {
317 2 : *pbOutAllValid = TRUE;
318 2 : return CE_None;
319 : }
320 :
321 722 : const int nNoData = static_cast<int>(floor(padfNoData[0] + 0.000001));
322 722 : int bAllValid = TRUE;
323 92333577 : for (size_t iOffset = 0; iOffset < nPixels; ++iOffset)
324 : {
325 92332825 : if (pData[iOffset] == nNoData)
326 : {
327 87204829 : bAllValid = FALSE;
328 87204829 : CPLMaskClear(panValidityMask, iOffset);
329 : }
330 : }
331 722 : *pbOutAllValid = bAllValid;
332 :
333 722 : return CE_None;
334 : }
335 :
336 : /************************************************************************/
337 : /* GDALWarpNoDataMasker() */
338 : /* */
339 : /* GDALMaskFunc for establishing a validity mask for a source */
340 : /* band based on a provided NODATA value. */
341 : /************************************************************************/
342 :
343 837 : CPLErr GDALWarpNoDataMasker(void *pMaskFuncArg, int nBandCount,
344 : GDALDataType eType, int /* nXOff */,
345 : int /* nYOff */, int nXSize, int nYSize,
346 : GByte **ppImageData, int bMaskIsFloat,
347 : void *pValidityMask, int *pbOutAllValid)
348 :
349 : {
350 837 : const double *padfNoData = static_cast<double *>(pMaskFuncArg);
351 837 : GUInt32 *panValidityMask = static_cast<GUInt32 *>(pValidityMask);
352 837 : const size_t nPixels = static_cast<size_t>(nXSize) * nYSize;
353 :
354 837 : *pbOutAllValid = FALSE;
355 :
356 837 : if (nBandCount != 1 || bMaskIsFloat)
357 : {
358 0 : CPLError(
359 : CE_Failure, CPLE_AppDefined,
360 : "Invalid nBandCount or bMaskIsFloat argument in SourceNoDataMask");
361 0 : return CE_Failure;
362 : }
363 :
364 837 : switch (eType)
365 : {
366 670 : case GDT_Byte:
367 670 : return GDALWarpNoDataMaskerT(padfNoData, nPixels,
368 : *ppImageData, // Already a GByte *.
369 670 : panValidityMask, pbOutAllValid);
370 :
371 52 : case GDT_Int16:
372 52 : return GDALWarpNoDataMaskerT(
373 : padfNoData, nPixels, reinterpret_cast<GInt16 *>(*ppImageData),
374 52 : panValidityMask, pbOutAllValid);
375 :
376 2 : case GDT_UInt16:
377 2 : return GDALWarpNoDataMaskerT(
378 : padfNoData, nPixels, reinterpret_cast<GUInt16 *>(*ppImageData),
379 2 : panValidityMask, pbOutAllValid);
380 :
381 76 : case GDT_Float32:
382 : {
383 76 : const float fNoData = static_cast<float>(padfNoData[0]);
384 76 : const float *pafData = reinterpret_cast<float *>(*ppImageData);
385 76 : const bool bIsNoDataNan = CPL_TO_BOOL(std::isnan(fNoData));
386 :
387 : // Nothing to do if value is out of range.
388 76 : if (padfNoData[1] != 0.0)
389 : {
390 0 : *pbOutAllValid = TRUE;
391 0 : return CE_None;
392 : }
393 :
394 76 : int bAllValid = TRUE;
395 45242 : for (size_t iOffset = 0; iOffset < nPixels; ++iOffset)
396 : {
397 45166 : float fVal = pafData[iOffset];
398 90102 : if ((bIsNoDataNan && std::isnan(fVal)) ||
399 44936 : (!bIsNoDataNan && ARE_REAL_EQUAL(fVal, fNoData)))
400 : {
401 37362 : bAllValid = FALSE;
402 37362 : CPLMaskClear(panValidityMask, iOffset);
403 : }
404 : }
405 76 : *pbOutAllValid = bAllValid;
406 : }
407 76 : break;
408 :
409 17 : case GDT_Float64:
410 : {
411 17 : const double dfNoData = padfNoData[0];
412 17 : const double *padfData = reinterpret_cast<double *>(*ppImageData);
413 17 : const bool bIsNoDataNan = CPL_TO_BOOL(std::isnan(dfNoData));
414 :
415 : // Nothing to do if value is out of range.
416 17 : if (padfNoData[1] != 0.0)
417 : {
418 0 : *pbOutAllValid = TRUE;
419 0 : return CE_None;
420 : }
421 :
422 17 : int bAllValid = TRUE;
423 2871 : for (size_t iOffset = 0; iOffset < nPixels; ++iOffset)
424 : {
425 2854 : double dfVal = padfData[iOffset];
426 3337 : if ((bIsNoDataNan && std::isnan(dfVal)) ||
427 483 : (!bIsNoDataNan && ARE_REAL_EQUAL(dfVal, dfNoData)))
428 : {
429 2470 : bAllValid = FALSE;
430 2470 : CPLMaskClear(panValidityMask, iOffset);
431 : }
432 : }
433 17 : *pbOutAllValid = bAllValid;
434 : }
435 17 : break;
436 :
437 20 : default:
438 : {
439 20 : const int nWordSize = GDALGetDataTypeSizeBytes(eType);
440 :
441 : const bool bIsNoDataRealNan =
442 20 : CPL_TO_BOOL(std::isnan(padfNoData[0]));
443 :
444 : double *padfWrk =
445 20 : static_cast<double *>(CPLMalloc(nXSize * sizeof(double) * 2));
446 20 : int bAllValid = TRUE;
447 274 : for (int iLine = 0; iLine < nYSize; iLine++)
448 : {
449 254 : GDALCopyWords((*ppImageData) + nWordSize * iLine * nXSize,
450 : eType, nWordSize, padfWrk, GDT_CFloat64, 16,
451 : nXSize);
452 :
453 6362 : for (int iPixel = 0; iPixel < nXSize; ++iPixel)
454 : {
455 0 : if (((bIsNoDataRealNan &&
456 12216 : std::isnan(padfWrk[iPixel * 2])) ||
457 12216 : (!bIsNoDataRealNan &&
458 6108 : ARE_REAL_EQUAL(padfWrk[iPixel * 2], padfNoData[0]))))
459 : {
460 805 : size_t iOffset =
461 805 : iPixel + static_cast<size_t>(iLine) * nXSize;
462 :
463 805 : bAllValid = FALSE;
464 805 : CPLMaskClear(panValidityMask, iOffset);
465 : }
466 : }
467 : }
468 20 : *pbOutAllValid = bAllValid;
469 :
470 20 : CPLFree(padfWrk);
471 : }
472 20 : break;
473 : }
474 :
475 113 : return CE_None;
476 : }
477 :
478 : /************************************************************************/
479 : /* GDALWarpSrcAlphaMasker() */
480 : /* */
481 : /* GDALMaskFunc for reading source simple 8bit alpha mask */
482 : /* information and building a floating point density mask from */
483 : /* it. */
484 : /************************************************************************/
485 :
486 79 : CPLErr GDALWarpSrcAlphaMasker(void *pMaskFuncArg, int /* nBandCount */,
487 : GDALDataType /* eType */, int nXOff, int nYOff,
488 : int nXSize, int nYSize, GByte ** /*ppImageData */,
489 : int bMaskIsFloat, void *pValidityMask,
490 : int *pbOutAllOpaque)
491 :
492 : {
493 79 : GDALWarpOptions *psWO = static_cast<GDALWarpOptions *>(pMaskFuncArg);
494 79 : float *pafMask = static_cast<float *>(pValidityMask);
495 79 : *pbOutAllOpaque = FALSE;
496 79 : const size_t nPixels = static_cast<size_t>(nXSize) * nYSize;
497 :
498 : /* -------------------------------------------------------------------- */
499 : /* Do some minimal checking. */
500 : /* -------------------------------------------------------------------- */
501 79 : if (!bMaskIsFloat)
502 : {
503 0 : CPLAssert(false);
504 : return CE_Failure;
505 : }
506 :
507 79 : if (psWO == nullptr || psWO->nSrcAlphaBand < 1)
508 : {
509 0 : CPLAssert(false);
510 : return CE_Failure;
511 : }
512 :
513 : /* -------------------------------------------------------------------- */
514 : /* Read the alpha band. */
515 : /* -------------------------------------------------------------------- */
516 : GDALRasterBandH hAlphaBand =
517 79 : GDALGetRasterBand(psWO->hSrcDS, psWO->nSrcAlphaBand);
518 79 : if (hAlphaBand == nullptr)
519 0 : return CE_Failure;
520 :
521 : // Rescale.
522 79 : const float inv_alpha_max = static_cast<float>(
523 79 : 1.0 / CPLAtof(CSLFetchNameValueDef(psWO->papszWarpOptions,
524 79 : "SRC_ALPHA_MAX", "255")));
525 79 : bool bOutAllOpaque = true;
526 :
527 79 : size_t iPixel = 0;
528 : CPLErr eErr;
529 :
530 : #if (defined(__x86_64) || defined(_M_X64))
531 79 : GDALDataType eDT = GDALGetRasterDataType(hAlphaBand);
532 : // Make sure that pafMask is at least 8-byte aligned, which should
533 : // normally be always the case if being a ptr returned by malloc().
534 79 : if ((eDT == GDT_Byte || eDT == GDT_UInt16) && CPL_IS_ALIGNED(pafMask, 8))
535 : {
536 : // Read data.
537 136 : eErr = GDALRasterIOEx(
538 : hAlphaBand, GF_Read, nXOff, nYOff, nXSize, nYSize, pafMask, nXSize,
539 : nYSize, eDT, static_cast<GSpacing>(sizeof(int)),
540 68 : static_cast<GSpacing>(sizeof(int)) * nXSize, nullptr);
541 :
542 68 : if (eErr != CE_None)
543 0 : return eErr;
544 :
545 : // Make sure we have the correct alignment before doing SSE
546 : // On Linux x86_64, the alignment should be always correct due
547 : // the alignment of malloc() being 16 byte.
548 68 : const GUInt32 mask = (eDT == GDT_Byte) ? 0xff : 0xffff;
549 68 : if (!CPL_IS_ALIGNED(pafMask, 16))
550 : {
551 0 : pafMask[iPixel] =
552 0 : (reinterpret_cast<GUInt32 *>(pafMask)[iPixel] & mask) *
553 0 : inv_alpha_max;
554 0 : if (pafMask[iPixel] >= 1.0f)
555 0 : pafMask[iPixel] = 1.0f;
556 : else
557 0 : bOutAllOpaque = false;
558 0 : iPixel++;
559 : }
560 68 : CPLAssert(CPL_IS_ALIGNED(pafMask + iPixel, 16));
561 68 : const __m128 xmm_inverse_alpha_max = _mm_load1_ps(&inv_alpha_max);
562 68 : const float one_single = 1.0f;
563 68 : const __m128 xmm_one = _mm_load1_ps(&one_single);
564 136 : const __m128i xmm_i_mask = _mm_set1_epi32(mask);
565 68 : __m128 xmmMaskNonOpaque0 = _mm_setzero_ps();
566 68 : __m128 xmmMaskNonOpaque1 = _mm_setzero_ps();
567 68 : __m128 xmmMaskNonOpaque2 = _mm_setzero_ps();
568 629236 : for (; iPixel + 6 * 4 - 1 < nPixels; iPixel += 6 * 4)
569 : {
570 1258340 : __m128 xmm_mask0 = _mm_cvtepi32_ps(_mm_and_si128(
571 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
572 629168 : pafMask + iPixel + 4 * 0))));
573 1258340 : __m128 xmm_mask1 = _mm_cvtepi32_ps(_mm_and_si128(
574 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
575 629168 : pafMask + iPixel + 4 * 1))));
576 1258340 : __m128 xmm_mask2 = _mm_cvtepi32_ps(_mm_and_si128(
577 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
578 629168 : pafMask + iPixel + 4 * 2))));
579 1258340 : __m128 xmm_mask3 = _mm_cvtepi32_ps(_mm_and_si128(
580 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
581 629168 : pafMask + iPixel + 4 * 3))));
582 1258340 : __m128 xmm_mask4 = _mm_cvtepi32_ps(_mm_and_si128(
583 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
584 629168 : pafMask + iPixel + 4 * 4))));
585 1887500 : __m128 xmm_mask5 = _mm_cvtepi32_ps(_mm_and_si128(
586 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
587 629168 : pafMask + iPixel + 4 * 5))));
588 629168 : xmm_mask0 = _mm_mul_ps(xmm_mask0, xmm_inverse_alpha_max);
589 629168 : xmm_mask1 = _mm_mul_ps(xmm_mask1, xmm_inverse_alpha_max);
590 629168 : xmm_mask2 = _mm_mul_ps(xmm_mask2, xmm_inverse_alpha_max);
591 629168 : xmm_mask3 = _mm_mul_ps(xmm_mask3, xmm_inverse_alpha_max);
592 629168 : xmm_mask4 = _mm_mul_ps(xmm_mask4, xmm_inverse_alpha_max);
593 629168 : xmm_mask5 = _mm_mul_ps(xmm_mask5, xmm_inverse_alpha_max);
594 : xmmMaskNonOpaque0 =
595 1258340 : _mm_or_ps(xmmMaskNonOpaque0, _mm_cmplt_ps(xmm_mask0, xmm_one));
596 : xmmMaskNonOpaque1 =
597 1258340 : _mm_or_ps(xmmMaskNonOpaque1, _mm_cmplt_ps(xmm_mask1, xmm_one));
598 : xmmMaskNonOpaque2 =
599 1258340 : _mm_or_ps(xmmMaskNonOpaque2, _mm_cmplt_ps(xmm_mask2, xmm_one));
600 : xmmMaskNonOpaque0 =
601 1258340 : _mm_or_ps(xmmMaskNonOpaque0, _mm_cmplt_ps(xmm_mask3, xmm_one));
602 : xmmMaskNonOpaque1 =
603 1258340 : _mm_or_ps(xmmMaskNonOpaque1, _mm_cmplt_ps(xmm_mask4, xmm_one));
604 : xmmMaskNonOpaque2 =
605 1258340 : _mm_or_ps(xmmMaskNonOpaque2, _mm_cmplt_ps(xmm_mask5, xmm_one));
606 629168 : xmm_mask0 = _mm_min_ps(xmm_mask0, xmm_one);
607 629168 : xmm_mask1 = _mm_min_ps(xmm_mask1, xmm_one);
608 629168 : xmm_mask2 = _mm_min_ps(xmm_mask2, xmm_one);
609 629168 : xmm_mask3 = _mm_min_ps(xmm_mask3, xmm_one);
610 629168 : xmm_mask4 = _mm_min_ps(xmm_mask4, xmm_one);
611 629168 : xmm_mask5 = _mm_min_ps(xmm_mask5, xmm_one);
612 629168 : _mm_store_ps(pafMask + iPixel + 4 * 0, xmm_mask0);
613 629168 : _mm_store_ps(pafMask + iPixel + 4 * 1, xmm_mask1);
614 629168 : _mm_store_ps(pafMask + iPixel + 4 * 2, xmm_mask2);
615 629168 : _mm_store_ps(pafMask + iPixel + 4 * 3, xmm_mask3);
616 629168 : _mm_store_ps(pafMask + iPixel + 4 * 4, xmm_mask4);
617 629168 : _mm_store_ps(pafMask + iPixel + 4 * 5, xmm_mask5);
618 : }
619 136 : if (_mm_movemask_ps(
620 : _mm_or_ps(_mm_or_ps(xmmMaskNonOpaque0, xmmMaskNonOpaque1),
621 68 : xmmMaskNonOpaque2)))
622 : {
623 43 : bOutAllOpaque = false;
624 : }
625 980 : for (; iPixel < nPixels; iPixel++)
626 : {
627 912 : pafMask[iPixel] =
628 912 : (reinterpret_cast<GUInt32 *>(pafMask)[iPixel] & mask) *
629 912 : inv_alpha_max;
630 912 : if (pafMask[iPixel] >= 1.0f)
631 442 : pafMask[iPixel] = 1.0f;
632 : else
633 470 : bOutAllOpaque = false;
634 68 : }
635 : }
636 : else
637 : #endif
638 : {
639 : // Read data.
640 11 : eErr = GDALRasterIO(hAlphaBand, GF_Read, nXOff, nYOff, nXSize, nYSize,
641 : pafMask, nXSize, nYSize, GDT_Float32, 0, 0);
642 :
643 11 : if (eErr != CE_None)
644 0 : return eErr;
645 :
646 : // TODO(rouault): Is loop unrolling by hand (r34564) actually helpful?
647 12969 : for (; iPixel + 3 < nPixels; iPixel += 4)
648 : {
649 12958 : pafMask[iPixel] = pafMask[iPixel] * inv_alpha_max;
650 12958 : if (pafMask[iPixel] >= 1.0f)
651 3028 : pafMask[iPixel] = 1.0f;
652 : else
653 9930 : bOutAllOpaque = false;
654 12958 : pafMask[iPixel + 1] = pafMask[iPixel + 1] * inv_alpha_max;
655 12958 : if (pafMask[iPixel + 1] >= 1.0f)
656 3018 : pafMask[iPixel + 1] = 1.0f;
657 : else
658 9940 : bOutAllOpaque = false;
659 12958 : pafMask[iPixel + 2] = pafMask[iPixel + 2] * inv_alpha_max;
660 12958 : if (pafMask[iPixel + 2] >= 1.0f)
661 3066 : pafMask[iPixel + 2] = 1.0f;
662 : else
663 9892 : bOutAllOpaque = false;
664 12958 : pafMask[iPixel + 3] = pafMask[iPixel + 3] * inv_alpha_max;
665 12958 : if (pafMask[iPixel + 3] >= 1.0f)
666 3016 : pafMask[iPixel + 3] = 1.0f;
667 : else
668 9942 : bOutAllOpaque = false;
669 : }
670 :
671 12 : for (; iPixel < nPixels; iPixel++)
672 : {
673 1 : pafMask[iPixel] = pafMask[iPixel] * inv_alpha_max;
674 1 : if (pafMask[iPixel] >= 1.0f)
675 0 : pafMask[iPixel] = 1.0f;
676 : else
677 1 : bOutAllOpaque = false;
678 : }
679 : }
680 :
681 79 : *pbOutAllOpaque = bOutAllOpaque;
682 :
683 79 : return CE_None;
684 : }
685 :
686 : /************************************************************************/
687 : /* GDALWarpSrcMaskMasker() */
688 : /* */
689 : /* GDALMaskFunc for reading source simple 8bit validity mask */
690 : /* information and building a one bit validity mask. */
691 : /************************************************************************/
692 :
693 2 : CPLErr GDALWarpSrcMaskMasker(void *pMaskFuncArg, int /* nBandCount */,
694 : GDALDataType /* eType */, int nXOff, int nYOff,
695 : int nXSize, int nYSize, GByte ** /*ppImageData */,
696 : int bMaskIsFloat, void *pValidityMask)
697 :
698 : {
699 2 : GDALWarpOptions *psWO = static_cast<GDALWarpOptions *>(pMaskFuncArg);
700 2 : GUInt32 *panMask = static_cast<GUInt32 *>(pValidityMask);
701 :
702 : /* -------------------------------------------------------------------- */
703 : /* Do some minimal checking. */
704 : /* -------------------------------------------------------------------- */
705 2 : if (bMaskIsFloat)
706 : {
707 0 : CPLAssert(false);
708 : return CE_Failure;
709 : }
710 :
711 2 : if (psWO == nullptr)
712 : {
713 0 : CPLAssert(false);
714 : return CE_Failure;
715 : }
716 :
717 : /* -------------------------------------------------------------------- */
718 : /* Allocate a temporary buffer to read mask byte data into. */
719 : /* -------------------------------------------------------------------- */
720 : GByte *pabySrcMask =
721 2 : static_cast<GByte *>(VSI_MALLOC2_VERBOSE(nXSize, nYSize));
722 2 : if (pabySrcMask == nullptr)
723 : {
724 0 : return CE_Failure;
725 : }
726 :
727 : /* -------------------------------------------------------------------- */
728 : /* Fetch our mask band. */
729 : /* -------------------------------------------------------------------- */
730 2 : GDALRasterBandH hMaskBand = nullptr;
731 : GDALRasterBandH hSrcBand =
732 2 : GDALGetRasterBand(psWO->hSrcDS, psWO->panSrcBands[0]);
733 2 : if (hSrcBand != nullptr)
734 2 : hMaskBand = GDALGetMaskBand(hSrcBand);
735 :
736 2 : if (hMaskBand == nullptr)
737 : {
738 0 : CPLAssert(false);
739 : return CE_Failure;
740 : }
741 :
742 : /* -------------------------------------------------------------------- */
743 : /* Read the mask band. */
744 : /* -------------------------------------------------------------------- */
745 2 : CPLErr eErr = GDALRasterIO(hMaskBand, GF_Read, nXOff, nYOff, nXSize, nYSize,
746 : pabySrcMask, nXSize, nYSize, GDT_Byte, 0, 0);
747 :
748 2 : if (eErr != CE_None)
749 : {
750 0 : CPLFree(pabySrcMask);
751 0 : return eErr;
752 : }
753 :
754 : /* -------------------------------------------------------------------- */
755 : /* Pack into 1 bit per pixel for validity. */
756 : /* -------------------------------------------------------------------- */
757 2 : const size_t nPixels = static_cast<size_t>(nXSize) * nYSize;
758 234579 : for (size_t iPixel = 0; iPixel < nPixels; iPixel++)
759 : {
760 234577 : if (pabySrcMask[iPixel] == 0)
761 31660 : CPLMaskClear(panMask, iPixel);
762 : }
763 :
764 2 : CPLFree(pabySrcMask);
765 :
766 2 : return CE_None;
767 : }
768 :
769 : /************************************************************************/
770 : /* GDALWarpDstAlphaMasker() */
771 : /* */
772 : /* GDALMaskFunc for reading or writing the destination simple */
773 : /* 8bit alpha mask information and building a floating point */
774 : /* density mask from it. Note, writing is distinguished */
775 : /* negative bandcount. */
776 : /************************************************************************/
777 :
778 490 : CPLErr GDALWarpDstAlphaMasker(void *pMaskFuncArg, int nBandCount,
779 : CPL_UNUSED GDALDataType /* eType */, int nXOff,
780 : int nYOff, int nXSize, int nYSize,
781 : GByte ** /*ppImageData */, int bMaskIsFloat,
782 : void *pValidityMask)
783 : {
784 : /* -------------------------------------------------------------------- */
785 : /* Do some minimal checking. */
786 : /* -------------------------------------------------------------------- */
787 490 : if (!bMaskIsFloat)
788 : {
789 0 : CPLAssert(false);
790 : return CE_Failure;
791 : }
792 :
793 490 : GDALWarpOptions *psWO = static_cast<GDALWarpOptions *>(pMaskFuncArg);
794 490 : if (psWO == nullptr || psWO->nDstAlphaBand < 1)
795 : {
796 0 : CPLAssert(false);
797 : return CE_Failure;
798 : }
799 :
800 490 : float *pafMask = static_cast<float *>(pValidityMask);
801 490 : const size_t nPixels = static_cast<size_t>(nXSize) * nYSize;
802 :
803 : GDALRasterBandH hAlphaBand =
804 490 : GDALGetRasterBand(psWO->hDstDS, psWO->nDstAlphaBand);
805 490 : if (hAlphaBand == nullptr)
806 0 : return CE_Failure;
807 :
808 490 : size_t iPixel = 0;
809 :
810 : /* -------------------------------------------------------------------- */
811 : /* Read alpha case. */
812 : /* -------------------------------------------------------------------- */
813 490 : if (nBandCount >= 0)
814 : {
815 : const char *pszInitDest =
816 245 : CSLFetchNameValue(psWO->papszWarpOptions, "INIT_DEST");
817 :
818 : // Special logic for destinations being initialized on-the-fly.
819 245 : if (pszInitDest != nullptr)
820 : {
821 155 : memset(pafMask, 0, nPixels * sizeof(float));
822 155 : return CE_None;
823 : }
824 :
825 : // Rescale.
826 90 : const float inv_alpha_max = static_cast<float>(
827 90 : 1.0 / CPLAtof(CSLFetchNameValueDef(psWO->papszWarpOptions,
828 90 : "DST_ALPHA_MAX", "255")));
829 :
830 : #if (defined(__x86_64) || defined(_M_X64))
831 90 : const GDALDataType eDT = GDALGetRasterDataType(hAlphaBand);
832 : // Make sure that pafMask is at least 8-byte aligned, which should
833 : // normally be always the case if being a ptr returned by malloc().
834 90 : if ((eDT == GDT_Byte || eDT == GDT_UInt16) &&
835 81 : CPL_IS_ALIGNED(pafMask, 8))
836 : {
837 : // Read data.
838 162 : const CPLErr eErr = GDALRasterIOEx(
839 : hAlphaBand, GF_Read, nXOff, nYOff, nXSize, nYSize, pafMask,
840 : nXSize, nYSize, eDT, static_cast<GSpacing>(sizeof(int)),
841 81 : static_cast<GSpacing>(sizeof(int)) * nXSize, nullptr);
842 :
843 81 : if (eErr != CE_None)
844 0 : return eErr;
845 :
846 : // Make sure we have the correct alignment before doing SSE
847 : // On Linux x86_64, the alignment should be always correct due
848 : // the alignment of malloc() being 16 byte.
849 81 : const GUInt32 mask = (eDT == GDT_Byte) ? 0xff : 0xffff;
850 81 : if (!CPL_IS_ALIGNED(pafMask, 16))
851 : {
852 0 : pafMask[iPixel] =
853 0 : (reinterpret_cast<GUInt32 *>(pafMask)[iPixel] & mask) *
854 0 : inv_alpha_max;
855 0 : pafMask[iPixel] = std::min(1.0f, pafMask[iPixel]);
856 0 : iPixel++;
857 : }
858 81 : CPLAssert(CPL_IS_ALIGNED(pafMask + iPixel, 16));
859 81 : const __m128 xmm_inverse_alpha_max = _mm_load1_ps(&inv_alpha_max);
860 81 : const float one_single = 1.0f;
861 81 : const __m128 xmm_one = _mm_load1_ps(&one_single);
862 162 : const __m128i xmm_i_mask = _mm_set1_epi32(mask);
863 126073 : for (; iPixel + 31 < nPixels; iPixel += 32)
864 : {
865 251984 : __m128 xmm_mask0 = _mm_cvtepi32_ps(_mm_and_si128(
866 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
867 125992 : pafMask + iPixel + 4 * 0))));
868 251984 : __m128 xmm_mask1 = _mm_cvtepi32_ps(_mm_and_si128(
869 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
870 125992 : pafMask + iPixel + 4 * 1))));
871 251984 : __m128 xmm_mask2 = _mm_cvtepi32_ps(_mm_and_si128(
872 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
873 125992 : pafMask + iPixel + 4 * 2))));
874 251984 : __m128 xmm_mask3 = _mm_cvtepi32_ps(_mm_and_si128(
875 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
876 125992 : pafMask + iPixel + 4 * 3))));
877 251984 : __m128 xmm_mask4 = _mm_cvtepi32_ps(_mm_and_si128(
878 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
879 125992 : pafMask + iPixel + 4 * 4))));
880 251984 : __m128 xmm_mask5 = _mm_cvtepi32_ps(_mm_and_si128(
881 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
882 125992 : pafMask + iPixel + 4 * 5))));
883 251984 : __m128 xmm_mask6 = _mm_cvtepi32_ps(_mm_and_si128(
884 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
885 125992 : pafMask + iPixel + 4 * 6))));
886 377976 : __m128 xmm_mask7 = _mm_cvtepi32_ps(_mm_and_si128(
887 : xmm_i_mask, _mm_load_si128(reinterpret_cast<__m128i *>(
888 125992 : pafMask + iPixel + 4 * 7))));
889 125992 : xmm_mask0 = _mm_mul_ps(xmm_mask0, xmm_inverse_alpha_max);
890 125992 : xmm_mask1 = _mm_mul_ps(xmm_mask1, xmm_inverse_alpha_max);
891 125992 : xmm_mask2 = _mm_mul_ps(xmm_mask2, xmm_inverse_alpha_max);
892 125992 : xmm_mask3 = _mm_mul_ps(xmm_mask3, xmm_inverse_alpha_max);
893 125992 : xmm_mask4 = _mm_mul_ps(xmm_mask4, xmm_inverse_alpha_max);
894 125992 : xmm_mask5 = _mm_mul_ps(xmm_mask5, xmm_inverse_alpha_max);
895 125992 : xmm_mask6 = _mm_mul_ps(xmm_mask6, xmm_inverse_alpha_max);
896 125992 : xmm_mask7 = _mm_mul_ps(xmm_mask7, xmm_inverse_alpha_max);
897 125992 : xmm_mask0 = _mm_min_ps(xmm_mask0, xmm_one);
898 125992 : xmm_mask1 = _mm_min_ps(xmm_mask1, xmm_one);
899 125992 : xmm_mask2 = _mm_min_ps(xmm_mask2, xmm_one);
900 125992 : xmm_mask3 = _mm_min_ps(xmm_mask3, xmm_one);
901 125992 : xmm_mask4 = _mm_min_ps(xmm_mask4, xmm_one);
902 125992 : xmm_mask5 = _mm_min_ps(xmm_mask5, xmm_one);
903 125992 : xmm_mask6 = _mm_min_ps(xmm_mask6, xmm_one);
904 125992 : xmm_mask7 = _mm_min_ps(xmm_mask7, xmm_one);
905 125992 : _mm_store_ps(pafMask + iPixel + 4 * 0, xmm_mask0);
906 125992 : _mm_store_ps(pafMask + iPixel + 4 * 1, xmm_mask1);
907 125992 : _mm_store_ps(pafMask + iPixel + 4 * 2, xmm_mask2);
908 125992 : _mm_store_ps(pafMask + iPixel + 4 * 3, xmm_mask3);
909 125992 : _mm_store_ps(pafMask + iPixel + 4 * 4, xmm_mask4);
910 125992 : _mm_store_ps(pafMask + iPixel + 4 * 5, xmm_mask5);
911 125992 : _mm_store_ps(pafMask + iPixel + 4 * 6, xmm_mask6);
912 125992 : _mm_store_ps(pafMask + iPixel + 4 * 7, xmm_mask7);
913 : }
914 825 : for (; iPixel < nPixels; iPixel++)
915 : {
916 744 : pafMask[iPixel] =
917 744 : (reinterpret_cast<GUInt32 *>(pafMask)[iPixel] & mask) *
918 744 : inv_alpha_max;
919 744 : pafMask[iPixel] = std::min(1.0f, pafMask[iPixel]);
920 81 : }
921 : }
922 : else
923 : #endif
924 : {
925 : // Read data.
926 : const CPLErr eErr =
927 9 : GDALRasterIO(hAlphaBand, GF_Read, nXOff, nYOff, nXSize, nYSize,
928 : pafMask, nXSize, nYSize, GDT_Float32, 0, 0);
929 :
930 9 : if (eErr != CE_None)
931 0 : return eErr;
932 :
933 842 : for (; iPixel < nPixels; iPixel++)
934 : {
935 833 : pafMask[iPixel] = pafMask[iPixel] * inv_alpha_max;
936 833 : pafMask[iPixel] = std::min(1.0f, pafMask[iPixel]);
937 : }
938 : }
939 :
940 90 : return CE_None;
941 : }
942 :
943 : /* -------------------------------------------------------------------- */
944 : /* Write alpha case. */
945 : /* -------------------------------------------------------------------- */
946 : else
947 : {
948 245 : GDALDataType eDT = GDALGetRasterDataType(hAlphaBand);
949 : const float cst_alpha_max =
950 245 : static_cast<float>(CPLAtof(CSLFetchNameValueDef(
951 245 : psWO->papszWarpOptions, "DST_ALPHA_MAX", "255"))) +
952 29 : ((eDT == GDT_Byte || eDT == GDT_Int16 || eDT == GDT_UInt16 ||
953 1 : eDT == GDT_Int32 || eDT == GDT_UInt32)
954 274 : ? 0.1f
955 245 : : 0.0f);
956 :
957 245 : CPLErr eErr = CE_None;
958 :
959 : #if (defined(__x86_64) || defined(_M_X64))
960 : // Make sure that pafMask is at least 8-byte aligned, which should
961 : // normally be always the case if being a ptr returned by malloc()
962 245 : if ((eDT == GDT_Byte || eDT == GDT_Int16 || eDT == GDT_UInt16) &&
963 244 : CPL_IS_ALIGNED(pafMask, 8))
964 : {
965 : // Make sure we have the correct alignment before doing SSE
966 : // On Linux x86_64, the alignment should be always correct due
967 : // the alignment of malloc() being 16 byte
968 244 : if (!CPL_IS_ALIGNED(pafMask, 16))
969 : {
970 0 : reinterpret_cast<int *>(pafMask)[iPixel] =
971 0 : static_cast<int>(pafMask[iPixel] * cst_alpha_max);
972 0 : iPixel++;
973 : }
974 244 : CPLAssert(CPL_IS_ALIGNED(pafMask + iPixel, 16));
975 244 : const __m128 xmm_alpha_max = _mm_load1_ps(&cst_alpha_max);
976 279059 : for (; iPixel + 31 < nPixels; iPixel += 32)
977 : {
978 278815 : __m128 xmm_mask0 = _mm_load_ps(pafMask + iPixel + 4 * 0);
979 278815 : __m128 xmm_mask1 = _mm_load_ps(pafMask + iPixel + 4 * 1);
980 278815 : __m128 xmm_mask2 = _mm_load_ps(pafMask + iPixel + 4 * 2);
981 278815 : __m128 xmm_mask3 = _mm_load_ps(pafMask + iPixel + 4 * 3);
982 278815 : __m128 xmm_mask4 = _mm_load_ps(pafMask + iPixel + 4 * 4);
983 278815 : __m128 xmm_mask5 = _mm_load_ps(pafMask + iPixel + 4 * 5);
984 278815 : __m128 xmm_mask6 = _mm_load_ps(pafMask + iPixel + 4 * 6);
985 557630 : __m128 xmm_mask7 = _mm_load_ps(pafMask + iPixel + 4 * 7);
986 278815 : xmm_mask0 = _mm_mul_ps(xmm_mask0, xmm_alpha_max);
987 278815 : xmm_mask1 = _mm_mul_ps(xmm_mask1, xmm_alpha_max);
988 278815 : xmm_mask2 = _mm_mul_ps(xmm_mask2, xmm_alpha_max);
989 278815 : xmm_mask3 = _mm_mul_ps(xmm_mask3, xmm_alpha_max);
990 278815 : xmm_mask4 = _mm_mul_ps(xmm_mask4, xmm_alpha_max);
991 278815 : xmm_mask5 = _mm_mul_ps(xmm_mask5, xmm_alpha_max);
992 278815 : xmm_mask6 = _mm_mul_ps(xmm_mask6, xmm_alpha_max);
993 278815 : xmm_mask7 = _mm_mul_ps(xmm_mask7, xmm_alpha_max);
994 : // Truncate to int.
995 278815 : _mm_store_si128(
996 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 0),
997 : _mm_cvttps_epi32(xmm_mask0));
998 278815 : _mm_store_si128(
999 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 1),
1000 : _mm_cvttps_epi32(xmm_mask1));
1001 278815 : _mm_store_si128(
1002 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 2),
1003 : _mm_cvttps_epi32(xmm_mask2));
1004 278815 : _mm_store_si128(
1005 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 3),
1006 : _mm_cvttps_epi32(xmm_mask3));
1007 278815 : _mm_store_si128(
1008 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 4),
1009 : _mm_cvttps_epi32(xmm_mask4));
1010 278815 : _mm_store_si128(
1011 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 5),
1012 : _mm_cvttps_epi32(xmm_mask5));
1013 278815 : _mm_store_si128(
1014 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 6),
1015 : _mm_cvttps_epi32(xmm_mask6));
1016 278815 : _mm_store_si128(
1017 278815 : reinterpret_cast<__m128i *>(pafMask + iPixel + 4 * 7),
1018 : _mm_cvttps_epi32(xmm_mask7));
1019 : }
1020 1804 : for (; iPixel < nPixels; iPixel++)
1021 1560 : reinterpret_cast<int *>(pafMask)[iPixel] =
1022 1560 : static_cast<int>(pafMask[iPixel] * cst_alpha_max);
1023 :
1024 : // Write data.
1025 : // Assumes little endianness here.
1026 488 : eErr = GDALRasterIOEx(
1027 : hAlphaBand, GF_Write, nXOff, nYOff, nXSize, nYSize, pafMask,
1028 : nXSize, nYSize, eDT, static_cast<GSpacing>(sizeof(int)),
1029 244 : static_cast<GSpacing>(sizeof(int)) * nXSize, nullptr);
1030 : }
1031 : else
1032 : #endif
1033 : {
1034 9 : for (; iPixel + 3 < nPixels; iPixel += 4)
1035 : {
1036 8 : pafMask[iPixel + 0] = static_cast<float>(
1037 8 : static_cast<int>(pafMask[iPixel + 0] * cst_alpha_max));
1038 8 : pafMask[iPixel + 1] = static_cast<float>(
1039 8 : static_cast<int>(pafMask[iPixel + 1] * cst_alpha_max));
1040 8 : pafMask[iPixel + 2] = static_cast<float>(
1041 8 : static_cast<int>(pafMask[iPixel + 2] * cst_alpha_max));
1042 8 : pafMask[iPixel + 3] = static_cast<float>(
1043 8 : static_cast<int>(pafMask[iPixel + 3] * cst_alpha_max));
1044 : }
1045 2 : for (; iPixel < nPixels; iPixel++)
1046 1 : pafMask[iPixel] = static_cast<float>(
1047 1 : static_cast<int>(pafMask[iPixel] * cst_alpha_max));
1048 :
1049 : // Write data.
1050 :
1051 : eErr =
1052 1 : GDALRasterIO(hAlphaBand, GF_Write, nXOff, nYOff, nXSize, nYSize,
1053 : pafMask, nXSize, nYSize, GDT_Float32, 0, 0);
1054 : }
1055 245 : return eErr;
1056 : }
1057 : }
1058 :
1059 : /************************************************************************/
1060 : /* ==================================================================== */
1061 : /* GDALWarpOptions */
1062 : /* ==================================================================== */
1063 : /************************************************************************/
1064 :
1065 : /**
1066 : * \var char **GDALWarpOptions::papszWarpOptions;
1067 : *
1068 : * A string list of additional options controlling the warp operation in
1069 : * name=value format. A suitable string list can be prepared with
1070 : * CSLSetNameValue().
1071 : *
1072 : * The following values are currently supported:
1073 : * <ul>
1074 : * <li>INIT_DEST=[value] or INIT_DEST=NO_DATA: This option forces the
1075 : * destination image to be initialized to the indicated value (for all bands)
1076 : * or indicates that it should be initialized to the NO_DATA value in
1077 : * padfDstNoDataReal/padfDstNoDataImag. If this value isn't set the
1078 : * destination image will be read and overlaid.</li>
1079 : *
1080 : * <li>WRITE_FLUSH=YES/NO: This option forces a flush to disk of data after
1081 : * each chunk is processed. In some cases this helps ensure a serial
1082 : * writing of the output data otherwise a block of data may be written to disk
1083 : * each time a block of data is read for the input buffer resulting in a lot
1084 : * of extra seeking around the disk, and reduced IO throughput. The default
1085 : * at this time is NO.</li>
1086 : *
1087 : * <li>SKIP_NOSOURCE=YES/NO: Skip all processing for chunks for which there
1088 : * is no corresponding input data. This will disable initializing the
1089 : * destination (INIT_DEST) and all other processing, and so should be used
1090 : * carefully. Mostly useful to short circuit a lot of extra work in mosaicing
1091 : * situations. Starting with GDAL 2.4, gdalwarp will automatically enable this
1092 : * option when it is assumed to be safe to do so.</li>
1093 : *
1094 : * <li>UNIFIED_SRC_NODATA=YES/NO/PARTIAL: This setting determines
1095 : * how to take into account nodata values when there are several input bands.
1096 : * <ul>
1097 : * <li>When YES, all bands are considered as nodata if and only if, all bands
1098 : * match the corresponding nodata values.
1099 : * Note: UNIFIED_SRC_NODATA=YES is set by default, when called from gdalwarp
1100 : * / GDALWarp() with an explicit -srcnodata setting.
1101 : *
1102 : * Example with nodata values at (1, 2, 3) and target alpha band requested.
1103 : * <ul>
1104 : * <li>input pixel = (1, 2, 3) ==> output pixel = (0, 0, 0, 0)</li>
1105 : * <li>input pixel = (1, 2, 127) ==> output pixel = (1, 2, 127, 255)</li>
1106 : * </ul>
1107 : * </li>
1108 : * <li>When NO, nodata masking values is considered independently for each band.
1109 : * A potential target alpha band will always be valid if there are multiple
1110 : * bands.
1111 : *
1112 : * Example with nodata values at (1, 2, 3) and target alpha band requested.
1113 : * <ul>
1114 : * <li>input pixel = (1, 2, 3) ==> output pixel = (0, 0, 0, 255)</li>
1115 : * <li>input pixel = (1, 2, 127) ==> output pixel = (0, 0, 127, 255)</li>
1116 : * </ul>
1117 : *
1118 : * Note: NO was the default behavior before GDAL 3.3.2
1119 : * </li>
1120 : * <li>When PARTIAL, or not specified at all (default behavior),
1121 : * nodata masking values is considered independently for each band.
1122 : * But, and this is the difference with NO, if for a given pixel, it
1123 : * evaluates to the nodata value of each band, the target pixel is
1124 : * considered as globally invalid, which impacts the value of a potential
1125 : * target alpha band.
1126 : *
1127 : * Note: PARTIAL is new to GDAL 3.3.2 and should not be used with
1128 : * earlier versions. The default behavior of GDAL < 3.3.2 was NO.
1129 : *
1130 : * Example with nodata values at (1, 2, 3) and target alpha band requested.
1131 : * <ul>
1132 : * <li>input pixel = (1, 2, 3) ==> output pixel = (0, 0, 0, 0)</li>
1133 : * <li>input pixel = (1, 2, 127) ==> output pixel = (0, 0, 127, 255)</li>
1134 : * </ul>
1135 : * </li>
1136 : * </ul>
1137 : * </li>
1138 : *
1139 : * <li>CUTLINE: This may contain the WKT geometry for a cutline. It will
1140 : * be converted into a geometry by GDALWarpOperation::Initialize() and assigned
1141 : * to the GDALWarpOptions hCutline field. The coordinates must be expressed
1142 : * in source pixel/line coordinates. Note: this is different from the
1143 : * assumptions made for the -cutline option of the gdalwarp utility !</li>
1144 : *
1145 : * <li>CUTLINE_BLEND_DIST: This may be set with a distance in pixels which
1146 : * will be assigned to the dfCutlineBlendDist field in the GDALWarpOptions.</li>
1147 : *
1148 : * <li>CUTLINE_ALL_TOUCHED: This defaults to FALSE, but may be set to TRUE
1149 : * to enable ALL_TOUCHEd mode when rasterizing cutline polygons. This is
1150 : * useful to ensure that that all pixels overlapping the cutline polygon
1151 : * will be selected, not just those whose center point falls within the
1152 : * polygon.</li>
1153 : *
1154 : * <li>XSCALE: Ratio expressing the resampling factor (number of destination
1155 : * pixels per source pixel) along the target horizontal axis.
1156 : * The scale is used to determine the number of source pixels along the x-axis
1157 : * that are considered by the resampling algorithm.
1158 : * Equals to one for no resampling, below one for downsampling
1159 : * and above one for upsampling. This is automatically computed, for each
1160 : * processing chunk, and may thus vary among them, depending on the
1161 : * shape of output regions vs input regions. Such variations can be undesired
1162 : * in some situations. If the resampling factor can be considered as constant
1163 : * over the warped area, setting a constant value can lead to more reproducible
1164 : * pixel output.</li>
1165 : *
1166 : * <li>YSCALE: Same as XSCALE, but along the horizontal axis.</li>
1167 : *
1168 : * <li>OPTIMIZE_SIZE: This defaults to FALSE, but may be set to TRUE
1169 : * typically when writing to a compressed dataset (GeoTIFF with
1170 : * COMPRESS creation option set for example) for achieving a smaller
1171 : * file size. This is achieved by writing at once data aligned on full
1172 : * blocks of the target dataset, which avoids partial writes of
1173 : * compressed blocks and lost space when they are rewritten at the end
1174 : * of the file. However sticking to target block size may cause major
1175 : * processing slowdown for some particular reprojections. Starting
1176 : * with GDAL 3.8, OPTIMIZE_SIZE mode is automatically enabled when it is safe
1177 : * to do so.
1178 : * As this parameter influences the shape of warping chunk, and by default the
1179 : * XSCALE and YSCALE parameters are computed per warping chunk, this parameter may
1180 : * influence the pixel output.
1181 : * </li>
1182 : *
1183 : * <li>NUM_THREADS: (GDAL >= 1.10) Can be set to a numeric value or ALL_CPUS to
1184 : * set the number of threads to use to parallelize the computation part of the
1185 : * warping. If not set, computation will be done in a single thread.</li>
1186 : *
1187 : * <li>STREAMABLE_OUTPUT: (GDAL >= 2.0) This defaults to FALSE, but may
1188 : * be set to TRUE typically when writing to a streamed file. The
1189 : * gdalwarp utility automatically sets this option when writing to
1190 : * /vsistdout/ or a named pipe (on Unix). This option has performance
1191 : * impacts for some reprojections. Note: band interleaved output is
1192 : * not currently supported by the warping algorithm in a streamable
1193 : * compatible way.</li>
1194 : *
1195 : * <li>SRC_COORD_PRECISION: (GDAL >= 2.0). Advanced setting. This
1196 : * defaults to 0, to indicate that no rounding of computing source
1197 : * image coordinates corresponding to the target image must be
1198 : * done. If greater than 0 (and typically below 1), this value,
1199 : * expressed in pixel, will be used to round computed source image
1200 : * coordinates. The purpose of this option is to make the results of
1201 : * warping with the approximated transformer more reproducible and not
1202 : * sensitive to changes in warping memory size. To achieve that,
1203 : * SRC_COORD_PRECISION must be at least 10 times greater than the
1204 : * error threshold. The higher the SRC_COORD_PRECISION/error_threshold
1205 : * ratio, the higher the performance will be, since exact
1206 : * reprojections must statistically be done with a frequency of
1207 : * 4*error_threshold/SRC_COORD_PRECISION.</li>
1208 : *
1209 : * <li>SRC_ALPHA_MAX: (GDAL >= 2.2). Maximum value for the alpha band of the
1210 : * source dataset. If the value is not set and the alpha band has a NBITS
1211 : * metadata item, it is used to set SRC_ALPHA_MAX = 2^NBITS-1. Otherwise, if the
1212 : * value is not set and the alpha band is of type UInt16 (resp Int16), 65535
1213 : * (resp 32767) is used. Otherwise, 255 is used.</li>
1214 : *
1215 : * <li>DST_ALPHA_MAX: (GDAL >= 2.2). Maximum value for the alpha band of the
1216 : * destination dataset. If the value is not set and the alpha band has a NBITS
1217 : * metadata item, it is used to set DST_ALPHA_MAX = 2^NBITS-1. Otherwise, if the
1218 : * value is not set and the alpha band is of type UInt16 (resp Int16), 65535
1219 : * (resp 32767) is used. Otherwise, 255 is used.</li>
1220 : * </ul>
1221 : *
1222 : * Normally when computing the source raster data to
1223 : * load to generate a particular output area, the warper samples transforms
1224 : * 21 points along each edge of the destination region back onto the source
1225 : * file, and uses this to compute a bounding window on the source image that
1226 : * is sufficient. Depending on the transformation in effect, the source
1227 : * window may be a bit too small, or even missing large areas. Problem
1228 : * situations are those where the transformation is very non-linear or
1229 : * "inside out". Examples are transforming from WGS84 to Polar Stereographic
1230 : * for areas around the pole, or transformations where some of the image is
1231 : * untransformable. The following options provide some additional control
1232 : * to deal with errors in computing the source window:
1233 : * <ul>
1234 : *
1235 : * <li>SAMPLE_GRID=YES/NO: Setting this option to YES will force the sampling to
1236 : * include internal points as well as edge points which can be important if
1237 : * the transformation is esoteric inside out, or if large sections of the
1238 : * destination image are not transformable into the source coordinate
1239 : * system.</li>
1240 : *
1241 : * <li>SAMPLE_STEPS: Modifies the density of the sampling grid. The default
1242 : * number of steps is 21. Increasing this can increase the computational
1243 : * cost, but improves the accuracy with which the source region is
1244 : * computed.
1245 : * Starting with GDAL 3.7, this can be set to ALL to mean to sample
1246 : * along all edge points of the destination region (if SAMPLE_GRID=NO or not
1247 : * specified), or all points of the destination region if SAMPLE_GRID=YES.</li>
1248 : *
1249 : * <li>SOURCE_EXTRA: This is a number of extra pixels added around the source
1250 : * window for a given request, and by default it is 1 to take care of rounding
1251 : * error. Setting this larger will increase the amount of data that needs to
1252 : * be read, but can avoid missing source data.</li>
1253 : * <li>APPLY_VERTICAL_SHIFT=YES/NO: Force the use of vertical shift.
1254 : * This option is generally not necessary, except when using an explicit
1255 : * coordinate transformation (COORDINATE_OPERATION), and not specifying
1256 : * an explicit source and target SRS.</li>
1257 : * <li>MULT_FACTOR_VERTICAL_SHIFT: Multiplication factor for the vertical
1258 : * shift. Default 1.0</li>
1259 : *
1260 : * <li>EXCLUDED_VALUES: (GDAL >= 3.9) Comma-separated tuple of values
1261 : * (thus typically "R,G,B"), that are ignored as contributing source
1262 : * pixels during resampling. The number of values in the tuple must be the same
1263 : * as the number of bands, excluding the alpha band.
1264 : * Several tuples of excluded values may be specified using the
1265 : * "(R1,G1,B2),(R2,G2,B2)" syntax.
1266 : * Only taken into account by Average currently.
1267 : * This concept is a bit similar to nodata/alpha, but the main difference is
1268 : * that pixels matching one of the excluded value tuples are still considered
1269 : * as valid, when determining the target pixel validity/density.
1270 : * </li>
1271 : *
1272 : * <li>EXCLUDED_VALUES_PCT_THRESHOLD=[0-100]: (GDAL >= 3.9) Minimum percentage
1273 : * of source pixels that must be set at one of the EXCLUDED_VALUES to cause
1274 : * the excluded value, that is in majority among source pixels, to be used as the
1275 : * target pixel value. Default value is 50 (%).
1276 : * Only taken into account by Average currently.</li>
1277 : *
1278 : * <li>NODATA_VALUES_PCT_THRESHOLD=[0-100]: (GDAL >= 3.9) Minimum percentage
1279 : * of source pixels that must be at nodata (or alpha=0 or any other way to express
1280 : * transparent pixel) to cause the target pixel value to not be set. Default
1281 : * value is 100 (%), which means that a target pixel is not set only if all
1282 : * contributing source pixels are not set.
1283 : * Note that NODATA_VALUES_PCT_THRESHOLD is taken into account before
1284 : * EXCLUDED_VALUES_PCT_THRESHOLD.
1285 : * Only taken into account by Average currently.</li>
1286 : *
1287 : * <li>MODE_TIES=FIRST/MIN/MAX: (GDAL >= 3.11) Strategy to use when breaking
1288 : * ties with MODE resampling. By default, the first value encountered will be used.
1289 : * Alternatively, the minimum or maximum value can be selected.</li>
1290 : *
1291 : * </ul>
1292 : */
1293 :
1294 : /************************************************************************/
1295 : /* GDALCreateWarpOptions() */
1296 : /************************************************************************/
1297 :
1298 : /** Create a warp options structure.
1299 : *
1300 : * Must be deallocated with GDALDestroyWarpOptions()
1301 : */
1302 2971 : GDALWarpOptions *CPL_STDCALL GDALCreateWarpOptions()
1303 :
1304 : {
1305 : GDALWarpOptions *psOptions =
1306 2971 : static_cast<GDALWarpOptions *>(CPLCalloc(sizeof(GDALWarpOptions), 1));
1307 :
1308 2971 : psOptions->nBandCount = 0;
1309 2971 : psOptions->eResampleAlg = GRA_NearestNeighbour;
1310 2971 : psOptions->pfnProgress = GDALDummyProgress;
1311 2971 : psOptions->eWorkingDataType = GDT_Unknown;
1312 2971 : psOptions->eTieStrategy = GWKTS_First;
1313 :
1314 2971 : return psOptions;
1315 : }
1316 :
1317 : /************************************************************************/
1318 : /* GDALDestroyWarpOptions() */
1319 : /************************************************************************/
1320 :
1321 : /** Destroy a warp options structure. */
1322 2971 : void CPL_STDCALL GDALDestroyWarpOptions(GDALWarpOptions *psOptions)
1323 :
1324 : {
1325 2971 : if (psOptions == nullptr)
1326 0 : return;
1327 :
1328 2971 : CSLDestroy(psOptions->papszWarpOptions);
1329 2971 : CPLFree(psOptions->panSrcBands);
1330 2971 : CPLFree(psOptions->panDstBands);
1331 2971 : CPLFree(psOptions->padfSrcNoDataReal);
1332 2971 : CPLFree(psOptions->padfSrcNoDataImag);
1333 2971 : CPLFree(psOptions->padfDstNoDataReal);
1334 2971 : CPLFree(psOptions->padfDstNoDataImag);
1335 2971 : CPLFree(psOptions->papfnSrcPerBandValidityMaskFunc);
1336 2971 : CPLFree(psOptions->papSrcPerBandValidityMaskFuncArg);
1337 :
1338 2971 : if (psOptions->hCutline != nullptr)
1339 48 : delete static_cast<OGRGeometry *>(psOptions->hCutline);
1340 :
1341 2971 : CPLFree(psOptions);
1342 : }
1343 :
1344 : #define COPY_MEM(target, type, count) \
1345 : do \
1346 : { \
1347 : if ((psSrcOptions->target) != nullptr && (count) != 0) \
1348 : { \
1349 : (psDstOptions->target) = \
1350 : static_cast<type *>(CPLMalloc(sizeof(type) * (count))); \
1351 : memcpy((psDstOptions->target), (psSrcOptions->target), \
1352 : sizeof(type) * (count)); \
1353 : } \
1354 : else \
1355 : (psDstOptions->target) = nullptr; \
1356 : } while (false)
1357 :
1358 : /************************************************************************/
1359 : /* GDALCloneWarpOptions() */
1360 : /************************************************************************/
1361 :
1362 : /** Clone a warp options structure.
1363 : *
1364 : * Must be deallocated with GDALDestroyWarpOptions()
1365 : */
1366 : GDALWarpOptions *CPL_STDCALL
1367 1579 : GDALCloneWarpOptions(const GDALWarpOptions *psSrcOptions)
1368 :
1369 : {
1370 1579 : GDALWarpOptions *psDstOptions = GDALCreateWarpOptions();
1371 :
1372 1579 : memcpy(psDstOptions, psSrcOptions, sizeof(GDALWarpOptions));
1373 :
1374 1579 : if (psSrcOptions->papszWarpOptions != nullptr)
1375 1318 : psDstOptions->papszWarpOptions =
1376 1318 : CSLDuplicate(psSrcOptions->papszWarpOptions);
1377 :
1378 1579 : COPY_MEM(panSrcBands, int, psSrcOptions->nBandCount);
1379 1579 : COPY_MEM(panDstBands, int, psSrcOptions->nBandCount);
1380 1579 : COPY_MEM(padfSrcNoDataReal, double, psSrcOptions->nBandCount);
1381 1579 : COPY_MEM(padfSrcNoDataImag, double, psSrcOptions->nBandCount);
1382 1579 : COPY_MEM(padfDstNoDataReal, double, psSrcOptions->nBandCount);
1383 1579 : COPY_MEM(padfDstNoDataImag, double, psSrcOptions->nBandCount);
1384 : // cppcheck-suppress pointerSize
1385 1579 : COPY_MEM(papfnSrcPerBandValidityMaskFunc, GDALMaskFunc,
1386 : psSrcOptions->nBandCount);
1387 1579 : psDstOptions->papSrcPerBandValidityMaskFuncArg = nullptr;
1388 :
1389 1579 : if (psSrcOptions->hCutline != nullptr)
1390 10 : psDstOptions->hCutline =
1391 10 : OGR_G_Clone(static_cast<OGRGeometryH>(psSrcOptions->hCutline));
1392 1579 : psDstOptions->dfCutlineBlendDist = psSrcOptions->dfCutlineBlendDist;
1393 :
1394 1579 : return psDstOptions;
1395 : }
1396 :
1397 : namespace
1398 : {
1399 143 : void InitNoData(int nBandCount, double **ppdNoDataReal, double dDataReal)
1400 : {
1401 143 : if (nBandCount <= 0)
1402 : {
1403 0 : return;
1404 : }
1405 143 : if (*ppdNoDataReal != nullptr)
1406 : {
1407 32 : return;
1408 : }
1409 :
1410 111 : *ppdNoDataReal =
1411 111 : static_cast<double *>(CPLMalloc(sizeof(double) * nBandCount));
1412 :
1413 254 : for (int i = 0; i < nBandCount; ++i)
1414 : {
1415 143 : (*ppdNoDataReal)[i] = dDataReal;
1416 : }
1417 : }
1418 : } // namespace
1419 :
1420 : /************************************************************************/
1421 : /* GDALWarpInitDstNoDataReal() */
1422 : /************************************************************************/
1423 :
1424 : /**
1425 : * \brief Initialize padfDstNoDataReal with specified value.
1426 : *
1427 : * @param psOptionsIn options to initialize.
1428 : * @param dNoDataReal value to initialize to.
1429 : *
1430 : */
1431 38 : void CPL_STDCALL GDALWarpInitDstNoDataReal(GDALWarpOptions *psOptionsIn,
1432 : double dNoDataReal)
1433 : {
1434 38 : VALIDATE_POINTER0(psOptionsIn, "GDALWarpInitDstNoDataReal");
1435 38 : InitNoData(psOptionsIn->nBandCount, &psOptionsIn->padfDstNoDataReal,
1436 : dNoDataReal);
1437 : }
1438 :
1439 : /************************************************************************/
1440 : /* GDALWarpInitSrcNoDataReal() */
1441 : /************************************************************************/
1442 :
1443 : /**
1444 : * \brief Initialize padfSrcNoDataReal with specified value.
1445 : *
1446 : * @param psOptionsIn options to initialize.
1447 : * @param dNoDataReal value to initialize to.
1448 : *
1449 : */
1450 39 : void CPL_STDCALL GDALWarpInitSrcNoDataReal(GDALWarpOptions *psOptionsIn,
1451 : double dNoDataReal)
1452 : {
1453 39 : VALIDATE_POINTER0(psOptionsIn, "GDALWarpInitSrcNoDataReal");
1454 39 : InitNoData(psOptionsIn->nBandCount, &psOptionsIn->padfSrcNoDataReal,
1455 : dNoDataReal);
1456 : }
1457 :
1458 : /************************************************************************/
1459 : /* GDALWarpInitNoDataReal() */
1460 : /************************************************************************/
1461 :
1462 : /**
1463 : * \brief Initialize padfSrcNoDataReal and padfDstNoDataReal with specified
1464 : * value.
1465 : *
1466 : * @param psOptionsIn options to initialize.
1467 : * @param dNoDataReal value to initialize to.
1468 : *
1469 : */
1470 1 : void CPL_STDCALL GDALWarpInitNoDataReal(GDALWarpOptions *psOptionsIn,
1471 : double dNoDataReal)
1472 : {
1473 1 : GDALWarpInitDstNoDataReal(psOptionsIn, dNoDataReal);
1474 1 : GDALWarpInitSrcNoDataReal(psOptionsIn, dNoDataReal);
1475 1 : }
1476 :
1477 : /************************************************************************/
1478 : /* GDALWarpInitDstNoDataImag() */
1479 : /************************************************************************/
1480 :
1481 : /**
1482 : * \brief Initialize padfDstNoDataImag with specified value.
1483 : *
1484 : * @param psOptionsIn options to initialize.
1485 : * @param dNoDataImag value to initialize to.
1486 : *
1487 : */
1488 35 : void CPL_STDCALL GDALWarpInitDstNoDataImag(GDALWarpOptions *psOptionsIn,
1489 : double dNoDataImag)
1490 : {
1491 35 : VALIDATE_POINTER0(psOptionsIn, "GDALWarpInitDstNoDataImag");
1492 35 : InitNoData(psOptionsIn->nBandCount, &psOptionsIn->padfDstNoDataImag,
1493 : dNoDataImag);
1494 : }
1495 :
1496 : /************************************************************************/
1497 : /* GDALWarpInitSrcNoDataImag() */
1498 : /************************************************************************/
1499 :
1500 : /**
1501 : * \brief Initialize padfSrcNoDataImag with specified value.
1502 : *
1503 : * @param psOptionsIn options to initialize.
1504 : * @param dNoDataImag value to initialize to.
1505 : *
1506 : */
1507 31 : void CPL_STDCALL GDALWarpInitSrcNoDataImag(GDALWarpOptions *psOptionsIn,
1508 : double dNoDataImag)
1509 : {
1510 31 : VALIDATE_POINTER0(psOptionsIn, "GDALWarpInitSrcNoDataImag");
1511 31 : InitNoData(psOptionsIn->nBandCount, &psOptionsIn->padfSrcNoDataImag,
1512 : dNoDataImag);
1513 : }
1514 :
1515 : /************************************************************************/
1516 : /* GDALWarpResolveWorkingDataType() */
1517 : /************************************************************************/
1518 :
1519 : /**
1520 : * \brief If the working data type is unknown, this method will determine
1521 : * a valid working data type to support the data in the src and dest
1522 : * data sets and any noData values.
1523 : *
1524 : * @param psOptions options to initialize.
1525 : *
1526 : */
1527 1475 : void CPL_STDCALL GDALWarpResolveWorkingDataType(GDALWarpOptions *psOptions)
1528 : {
1529 1475 : if (psOptions == nullptr)
1530 : {
1531 0 : return;
1532 : }
1533 : /* -------------------------------------------------------------------- */
1534 : /* If no working data type was provided, set one now. */
1535 : /* */
1536 : /* Ensure that the working data type can encapsulate any value */
1537 : /* in the target, source, and the no data for either. */
1538 : /* -------------------------------------------------------------------- */
1539 1475 : if (psOptions->eWorkingDataType != GDT_Unknown)
1540 : {
1541 326 : return;
1542 : }
1543 :
1544 1149 : psOptions->eWorkingDataType = GDT_Byte;
1545 :
1546 : // If none of the provided input nodata values can be represented in the
1547 : // data type of the corresponding source band, ignore them.
1548 1149 : if (psOptions->hSrcDS && psOptions->padfSrcNoDataReal)
1549 : {
1550 90 : int nCountInvalidSrcNoDataReal = 0;
1551 232 : for (int iBand = 0; iBand < psOptions->nBandCount; iBand++)
1552 : {
1553 284 : GDALRasterBandH hSrcBand = GDALGetRasterBand(
1554 142 : psOptions->hSrcDS, psOptions->panSrcBands[iBand]);
1555 :
1556 284 : if (hSrcBand &&
1557 142 : !GDALIsValueExactAs(psOptions->padfSrcNoDataReal[iBand],
1558 : GDALGetRasterDataType(hSrcBand)))
1559 : {
1560 2 : nCountInvalidSrcNoDataReal++;
1561 : }
1562 : }
1563 90 : if (nCountInvalidSrcNoDataReal == psOptions->nBandCount)
1564 : {
1565 2 : CPLFree(psOptions->padfSrcNoDataReal);
1566 2 : psOptions->padfSrcNoDataReal = nullptr;
1567 2 : CPLFree(psOptions->padfSrcNoDataImag);
1568 2 : psOptions->padfSrcNoDataImag = nullptr;
1569 : }
1570 : }
1571 :
1572 2578 : for (int iBand = 0; iBand < psOptions->nBandCount; iBand++)
1573 : {
1574 1429 : if (psOptions->hDstDS != nullptr)
1575 : {
1576 2758 : GDALRasterBandH hDstBand = GDALGetRasterBand(
1577 1379 : psOptions->hDstDS, psOptions->panDstBands[iBand]);
1578 :
1579 1379 : if (hDstBand != nullptr)
1580 : {
1581 1379 : psOptions->eWorkingDataType =
1582 1379 : GDALDataTypeUnion(psOptions->eWorkingDataType,
1583 : GDALGetRasterDataType(hDstBand));
1584 : }
1585 : }
1586 :
1587 1429 : if (psOptions->hSrcDS != nullptr)
1588 : {
1589 2818 : GDALRasterBandH hSrcBand = GDALGetRasterBand(
1590 1409 : psOptions->hSrcDS, psOptions->panSrcBands[iBand]);
1591 :
1592 1409 : if (hSrcBand != nullptr)
1593 : {
1594 1409 : psOptions->eWorkingDataType =
1595 1409 : GDALDataTypeUnion(psOptions->eWorkingDataType,
1596 : GDALGetRasterDataType(hSrcBand));
1597 : }
1598 : }
1599 :
1600 1429 : if (psOptions->padfSrcNoDataReal != nullptr)
1601 : {
1602 150 : psOptions->eWorkingDataType = GDALDataTypeUnionWithValue(
1603 : psOptions->eWorkingDataType,
1604 150 : psOptions->padfSrcNoDataReal[iBand], false);
1605 : }
1606 :
1607 1429 : if (psOptions->padfSrcNoDataImag != nullptr &&
1608 4 : psOptions->padfSrcNoDataImag[iBand] != 0.0)
1609 : {
1610 3 : psOptions->eWorkingDataType = GDALDataTypeUnionWithValue(
1611 : psOptions->eWorkingDataType,
1612 3 : psOptions->padfSrcNoDataImag[iBand], true);
1613 : }
1614 :
1615 1429 : if (psOptions->padfDstNoDataReal != nullptr)
1616 : {
1617 121 : psOptions->eWorkingDataType = GDALDataTypeUnionWithValue(
1618 : psOptions->eWorkingDataType,
1619 121 : psOptions->padfDstNoDataReal[iBand], false);
1620 : }
1621 :
1622 1429 : if (psOptions->padfDstNoDataImag != nullptr &&
1623 32 : psOptions->padfDstNoDataImag[iBand] != 0.0)
1624 : {
1625 3 : psOptions->eWorkingDataType = GDALDataTypeUnionWithValue(
1626 : psOptions->eWorkingDataType,
1627 3 : psOptions->padfDstNoDataImag[iBand], true);
1628 : }
1629 : }
1630 :
1631 2298 : const bool bApplyVerticalShift = CPLFetchBool(
1632 1149 : psOptions->papszWarpOptions, "APPLY_VERTICAL_SHIFT", false);
1633 1168 : if (bApplyVerticalShift &&
1634 19 : GDALDataTypeIsInteger(psOptions->eWorkingDataType))
1635 : {
1636 16 : const double dfMultFactorVerticalShift = CPLAtof(CSLFetchNameValueDef(
1637 16 : psOptions->papszWarpOptions, "MULT_FACTOR_VERTICAL_SHIFT", "1.0"));
1638 16 : if (dfMultFactorVerticalShift != 1)
1639 : {
1640 0 : psOptions->eWorkingDataType =
1641 0 : GDALDataTypeUnion(psOptions->eWorkingDataType, GDT_Float32);
1642 : }
1643 : }
1644 : }
1645 :
1646 : /************************************************************************/
1647 : /* GDALWarpInitDefaultBandMapping() */
1648 : /************************************************************************/
1649 :
1650 : /**
1651 : * \brief Init src and dst band mappings such that Bands[i] = i+1
1652 : * for nBandCount
1653 : * Does nothing if psOptionsIn->nBandCount is non-zero.
1654 : *
1655 : * @param psOptionsIn options to initialize.
1656 : * @param nBandCount bands to initialize for.
1657 : *
1658 : */
1659 312 : void CPL_STDCALL GDALWarpInitDefaultBandMapping(GDALWarpOptions *psOptionsIn,
1660 : int nBandCount)
1661 : {
1662 312 : if (psOptionsIn->nBandCount != 0)
1663 : {
1664 0 : return;
1665 : }
1666 :
1667 312 : psOptionsIn->nBandCount = nBandCount;
1668 :
1669 312 : psOptionsIn->panSrcBands =
1670 312 : static_cast<int *>(CPLMalloc(sizeof(int) * psOptionsIn->nBandCount));
1671 312 : psOptionsIn->panDstBands =
1672 312 : static_cast<int *>(CPLMalloc(sizeof(int) * psOptionsIn->nBandCount));
1673 :
1674 820 : for (int i = 0; i < psOptionsIn->nBandCount; i++)
1675 : {
1676 508 : psOptionsIn->panSrcBands[i] = i + 1;
1677 508 : psOptionsIn->panDstBands[i] = i + 1;
1678 : }
1679 : }
1680 :
1681 : /************************************************************************/
1682 : /* GDALSerializeWarpOptions() */
1683 : /************************************************************************/
1684 :
1685 77 : CPLXMLNode *CPL_STDCALL GDALSerializeWarpOptions(const GDALWarpOptions *psWO)
1686 :
1687 : {
1688 : /* -------------------------------------------------------------------- */
1689 : /* Create root. */
1690 : /* -------------------------------------------------------------------- */
1691 : CPLXMLNode *psTree =
1692 77 : CPLCreateXMLNode(nullptr, CXT_Element, "GDALWarpOptions");
1693 :
1694 : /* -------------------------------------------------------------------- */
1695 : /* WarpMemoryLimit */
1696 : /* -------------------------------------------------------------------- */
1697 77 : CPLCreateXMLElementAndValue(
1698 : psTree, "WarpMemoryLimit",
1699 154 : CPLString().Printf("%g", psWO->dfWarpMemoryLimit));
1700 :
1701 : /* -------------------------------------------------------------------- */
1702 : /* ResampleAlg */
1703 : /* -------------------------------------------------------------------- */
1704 77 : const char *pszAlgName = nullptr;
1705 :
1706 77 : if (psWO->eResampleAlg == GRA_NearestNeighbour)
1707 77 : pszAlgName = "NearestNeighbour";
1708 0 : else if (psWO->eResampleAlg == GRA_Bilinear)
1709 0 : pszAlgName = "Bilinear";
1710 0 : else if (psWO->eResampleAlg == GRA_Cubic)
1711 0 : pszAlgName = "Cubic";
1712 0 : else if (psWO->eResampleAlg == GRA_CubicSpline)
1713 0 : pszAlgName = "CubicSpline";
1714 0 : else if (psWO->eResampleAlg == GRA_Lanczos)
1715 0 : pszAlgName = "Lanczos";
1716 0 : else if (psWO->eResampleAlg == GRA_Average)
1717 0 : pszAlgName = "Average";
1718 0 : else if (psWO->eResampleAlg == GRA_RMS)
1719 0 : pszAlgName = "RootMeanSquare";
1720 0 : else if (psWO->eResampleAlg == GRA_Mode)
1721 0 : pszAlgName = "Mode";
1722 0 : else if (psWO->eResampleAlg == GRA_Max)
1723 0 : pszAlgName = "Maximum";
1724 0 : else if (psWO->eResampleAlg == GRA_Min)
1725 0 : pszAlgName = "Minimum";
1726 0 : else if (psWO->eResampleAlg == GRA_Med)
1727 0 : pszAlgName = "Median";
1728 0 : else if (psWO->eResampleAlg == GRA_Q1)
1729 0 : pszAlgName = "Quartile1";
1730 0 : else if (psWO->eResampleAlg == GRA_Q3)
1731 0 : pszAlgName = "Quartile3";
1732 0 : else if (psWO->eResampleAlg == GRA_Sum)
1733 0 : pszAlgName = "Sum";
1734 : else
1735 0 : pszAlgName = "Unknown";
1736 :
1737 77 : CPLCreateXMLElementAndValue(psTree, "ResampleAlg", pszAlgName);
1738 :
1739 : /* -------------------------------------------------------------------- */
1740 : /* Working Data Type */
1741 : /* -------------------------------------------------------------------- */
1742 77 : CPLCreateXMLElementAndValue(psTree, "WorkingDataType",
1743 77 : GDALGetDataTypeName(psWO->eWorkingDataType));
1744 :
1745 : /* -------------------------------------------------------------------- */
1746 : /* Name/value warp options. */
1747 : /* -------------------------------------------------------------------- */
1748 314 : for (int iWO = 0; psWO->papszWarpOptions != nullptr &&
1749 314 : psWO->papszWarpOptions[iWO] != nullptr;
1750 : iWO++)
1751 : {
1752 237 : char *pszName = nullptr;
1753 : const char *pszValue =
1754 237 : CPLParseNameValue(psWO->papszWarpOptions[iWO], &pszName);
1755 :
1756 : // EXTRA_ELTS is an internal detail that we will recover
1757 : // no need to serialize it.
1758 : // And CUTLINE is also serialized in a special way
1759 237 : if (pszName != nullptr && !EQUAL(pszName, "EXTRA_ELTS") &&
1760 160 : !EQUAL(pszName, "CUTLINE"))
1761 : {
1762 : CPLXMLNode *psOption =
1763 160 : CPLCreateXMLElementAndValue(psTree, "Option", pszValue);
1764 :
1765 160 : CPLCreateXMLNode(CPLCreateXMLNode(psOption, CXT_Attribute, "name"),
1766 : CXT_Text, pszName);
1767 : }
1768 :
1769 237 : CPLFree(pszName);
1770 : }
1771 :
1772 : /* -------------------------------------------------------------------- */
1773 : /* Source and Destination Data Source */
1774 : /* -------------------------------------------------------------------- */
1775 77 : if (psWO->hSrcDS != nullptr)
1776 : {
1777 77 : CPLCreateXMLElementAndValue(psTree, "SourceDataset",
1778 77 : GDALGetDescription(psWO->hSrcDS));
1779 :
1780 : CSLConstList papszOpenOptions =
1781 77 : GDALDataset::FromHandle(psWO->hSrcDS)->GetOpenOptions();
1782 77 : GDALSerializeOpenOptionsToXML(psTree, papszOpenOptions);
1783 : }
1784 :
1785 154 : if (psWO->hDstDS != nullptr &&
1786 77 : strlen(GDALGetDescription(psWO->hDstDS)) != 0)
1787 : {
1788 0 : CPLCreateXMLElementAndValue(psTree, "DestinationDataset",
1789 0 : GDALGetDescription(psWO->hDstDS));
1790 : }
1791 :
1792 : /* -------------------------------------------------------------------- */
1793 : /* Serialize transformer. */
1794 : /* -------------------------------------------------------------------- */
1795 77 : if (psWO->pfnTransformer != nullptr)
1796 : {
1797 : CPLXMLNode *psTransformerContainer =
1798 77 : CPLCreateXMLNode(psTree, CXT_Element, "Transformer");
1799 :
1800 154 : CPLXMLNode *psTransformerTree = GDALSerializeTransformer(
1801 77 : psWO->pfnTransformer, psWO->pTransformerArg);
1802 :
1803 77 : if (psTransformerTree != nullptr)
1804 77 : CPLAddXMLChild(psTransformerContainer, psTransformerTree);
1805 : }
1806 :
1807 : /* -------------------------------------------------------------------- */
1808 : /* Band count and lists. */
1809 : /* -------------------------------------------------------------------- */
1810 77 : CPLXMLNode *psBandList = nullptr;
1811 :
1812 77 : if (psWO->nBandCount != 0)
1813 77 : psBandList = CPLCreateXMLNode(psTree, CXT_Element, "BandList");
1814 :
1815 240 : for (int i = 0; i < psWO->nBandCount; i++)
1816 : {
1817 : CPLXMLNode *psBand;
1818 :
1819 163 : psBand = CPLCreateXMLNode(psBandList, CXT_Element, "BandMapping");
1820 163 : if (psWO->panSrcBands != nullptr)
1821 163 : CPLCreateXMLNode(CPLCreateXMLNode(psBand, CXT_Attribute, "src"),
1822 : CXT_Text,
1823 326 : CPLString().Printf("%d", psWO->panSrcBands[i]));
1824 163 : if (psWO->panDstBands != nullptr)
1825 163 : CPLCreateXMLNode(CPLCreateXMLNode(psBand, CXT_Attribute, "dst"),
1826 : CXT_Text,
1827 326 : CPLString().Printf("%d", psWO->panDstBands[i]));
1828 :
1829 163 : if (psWO->padfSrcNoDataReal != nullptr)
1830 : {
1831 13 : CPLCreateXMLElementAndValue(
1832 : psBand, "SrcNoDataReal",
1833 13 : VRTSerializeNoData(psWO->padfSrcNoDataReal[i],
1834 13 : psWO->eWorkingDataType, 16)
1835 : .c_str());
1836 : }
1837 :
1838 163 : if (psWO->padfSrcNoDataImag != nullptr)
1839 : {
1840 2 : if (std::isnan(psWO->padfSrcNoDataImag[i]))
1841 0 : CPLCreateXMLElementAndValue(psBand, "SrcNoDataImag", "nan");
1842 : else
1843 2 : CPLCreateXMLElementAndValue(
1844 : psBand, "SrcNoDataImag",
1845 4 : CPLString().Printf("%.16g", psWO->padfSrcNoDataImag[i]));
1846 : }
1847 : // Compatibility with GDAL <= 2.2: if we serialize a SrcNoDataReal,
1848 : // it needs a SrcNoDataImag as well
1849 161 : else if (psWO->padfSrcNoDataReal != nullptr)
1850 : {
1851 11 : CPLCreateXMLElementAndValue(psBand, "SrcNoDataImag", "0");
1852 : }
1853 :
1854 163 : if (psWO->padfDstNoDataReal != nullptr)
1855 : {
1856 13 : CPLCreateXMLElementAndValue(
1857 : psBand, "DstNoDataReal",
1858 13 : VRTSerializeNoData(psWO->padfDstNoDataReal[i],
1859 13 : psWO->eWorkingDataType, 16)
1860 : .c_str());
1861 : }
1862 :
1863 163 : if (psWO->padfDstNoDataImag != nullptr)
1864 : {
1865 2 : if (std::isnan(psWO->padfDstNoDataImag[i]))
1866 0 : CPLCreateXMLElementAndValue(psBand, "DstNoDataImag", "nan");
1867 : else
1868 2 : CPLCreateXMLElementAndValue(
1869 : psBand, "DstNoDataImag",
1870 4 : CPLString().Printf("%.16g", psWO->padfDstNoDataImag[i]));
1871 : }
1872 : // Compatibility with GDAL <= 2.2: if we serialize a DstNoDataReal,
1873 : // it needs a SrcNoDataImag as well
1874 161 : else if (psWO->padfDstNoDataReal != nullptr)
1875 : {
1876 11 : CPLCreateXMLElementAndValue(psBand, "DstNoDataImag", "0");
1877 : }
1878 : }
1879 :
1880 : /* -------------------------------------------------------------------- */
1881 : /* Alpha bands. */
1882 : /* -------------------------------------------------------------------- */
1883 77 : if (psWO->nSrcAlphaBand > 0)
1884 0 : CPLCreateXMLElementAndValue(
1885 : psTree, "SrcAlphaBand",
1886 0 : CPLString().Printf("%d", psWO->nSrcAlphaBand));
1887 :
1888 77 : if (psWO->nDstAlphaBand > 0)
1889 23 : CPLCreateXMLElementAndValue(
1890 : psTree, "DstAlphaBand",
1891 46 : CPLString().Printf("%d", psWO->nDstAlphaBand));
1892 :
1893 : /* -------------------------------------------------------------------- */
1894 : /* Cutline. */
1895 : /* -------------------------------------------------------------------- */
1896 77 : if (psWO->hCutline != nullptr)
1897 : {
1898 0 : char *pszWKT = nullptr;
1899 0 : if (OGR_G_ExportToWkt(static_cast<OGRGeometryH>(psWO->hCutline),
1900 0 : &pszWKT) == OGRERR_NONE)
1901 : {
1902 0 : CPLCreateXMLElementAndValue(psTree, "Cutline", pszWKT);
1903 : }
1904 0 : CPLFree(pszWKT);
1905 : }
1906 :
1907 77 : if (psWO->dfCutlineBlendDist != 0.0)
1908 0 : CPLCreateXMLElementAndValue(
1909 : psTree, "CutlineBlendDist",
1910 0 : CPLString().Printf("%.5g", psWO->dfCutlineBlendDist));
1911 :
1912 77 : return psTree;
1913 : }
1914 :
1915 : /************************************************************************/
1916 : /* GDALDeserializeWarpOptions() */
1917 : /************************************************************************/
1918 :
1919 198 : GDALWarpOptions *CPL_STDCALL GDALDeserializeWarpOptions(CPLXMLNode *psTree)
1920 :
1921 : {
1922 198 : CPLErrorReset();
1923 :
1924 : /* -------------------------------------------------------------------- */
1925 : /* Verify this is the right kind of object. */
1926 : /* -------------------------------------------------------------------- */
1927 198 : if (psTree == nullptr || psTree->eType != CXT_Element ||
1928 198 : !EQUAL(psTree->pszValue, "GDALWarpOptions"))
1929 : {
1930 0 : CPLError(CE_Failure, CPLE_AppDefined,
1931 : "Wrong node, unable to deserialize GDALWarpOptions.");
1932 0 : return nullptr;
1933 : }
1934 :
1935 : /* -------------------------------------------------------------------- */
1936 : /* Create pre-initialized warp options. */
1937 : /* -------------------------------------------------------------------- */
1938 198 : GDALWarpOptions *psWO = GDALCreateWarpOptions();
1939 :
1940 : /* -------------------------------------------------------------------- */
1941 : /* Warp memory limit. */
1942 : /* -------------------------------------------------------------------- */
1943 198 : psWO->dfWarpMemoryLimit =
1944 198 : CPLAtof(CPLGetXMLValue(psTree, "WarpMemoryLimit", "0.0"));
1945 :
1946 : /* -------------------------------------------------------------------- */
1947 : /* resample algorithm */
1948 : /* -------------------------------------------------------------------- */
1949 198 : const char *pszValue = CPLGetXMLValue(psTree, "ResampleAlg", "Default");
1950 :
1951 198 : if (EQUAL(pszValue, "NearestNeighbour"))
1952 140 : psWO->eResampleAlg = GRA_NearestNeighbour;
1953 58 : else if (EQUAL(pszValue, "Bilinear"))
1954 8 : psWO->eResampleAlg = GRA_Bilinear;
1955 50 : else if (EQUAL(pszValue, "Cubic"))
1956 9 : psWO->eResampleAlg = GRA_Cubic;
1957 41 : else if (EQUAL(pszValue, "CubicSpline"))
1958 9 : psWO->eResampleAlg = GRA_CubicSpline;
1959 32 : else if (EQUAL(pszValue, "Lanczos"))
1960 4 : psWO->eResampleAlg = GRA_Lanczos;
1961 28 : else if (EQUAL(pszValue, "Average"))
1962 6 : psWO->eResampleAlg = GRA_Average;
1963 22 : else if (EQUAL(pszValue, "RootMeanSquare"))
1964 5 : psWO->eResampleAlg = GRA_RMS;
1965 17 : else if (EQUAL(pszValue, "Mode"))
1966 4 : psWO->eResampleAlg = GRA_Mode;
1967 13 : else if (EQUAL(pszValue, "Maximum"))
1968 3 : psWO->eResampleAlg = GRA_Max;
1969 10 : else if (EQUAL(pszValue, "Minimum"))
1970 2 : psWO->eResampleAlg = GRA_Min;
1971 8 : else if (EQUAL(pszValue, "Median"))
1972 3 : psWO->eResampleAlg = GRA_Med;
1973 5 : else if (EQUAL(pszValue, "Quartile1"))
1974 2 : psWO->eResampleAlg = GRA_Q1;
1975 3 : else if (EQUAL(pszValue, "Quartile3"))
1976 2 : psWO->eResampleAlg = GRA_Q3;
1977 1 : else if (EQUAL(pszValue, "Sum"))
1978 1 : psWO->eResampleAlg = GRA_Sum;
1979 0 : else if (EQUAL(pszValue, "Default"))
1980 : /* leave as is */;
1981 : else
1982 : {
1983 0 : CPLError(CE_Failure, CPLE_AppDefined,
1984 : "Unrecognised ResampleAlg value '%s'.", pszValue);
1985 : }
1986 :
1987 : /* -------------------------------------------------------------------- */
1988 : /* Working data type. */
1989 : /* -------------------------------------------------------------------- */
1990 198 : psWO->eWorkingDataType = GDALGetDataTypeByName(
1991 : CPLGetXMLValue(psTree, "WorkingDataType", "Unknown"));
1992 :
1993 : /* -------------------------------------------------------------------- */
1994 : /* Name/value warp options. */
1995 : /* -------------------------------------------------------------------- */
1996 1849 : for (CPLXMLNode *psItem = psTree->psChild; psItem != nullptr;
1997 1651 : psItem = psItem->psNext)
1998 : {
1999 1651 : if (psItem->eType == CXT_Element && EQUAL(psItem->pszValue, "Option"))
2000 : {
2001 346 : const char *pszName = CPLGetXMLValue(psItem, "Name", nullptr);
2002 346 : pszValue = CPLGetXMLValue(psItem, "", nullptr);
2003 :
2004 346 : if (pszName != nullptr && pszValue != nullptr)
2005 : {
2006 346 : psWO->papszWarpOptions =
2007 346 : CSLSetNameValue(psWO->papszWarpOptions, pszName, pszValue);
2008 : }
2009 : }
2010 : }
2011 :
2012 : /* -------------------------------------------------------------------- */
2013 : /* Source Dataset. */
2014 : /* -------------------------------------------------------------------- */
2015 198 : pszValue = CPLGetXMLValue(psTree, "SourceDataset", nullptr);
2016 :
2017 198 : if (pszValue != nullptr)
2018 : {
2019 : CPLXMLNode *psGeoLocNode =
2020 198 : CPLSearchXMLNode(psTree, "GeoLocTransformer");
2021 198 : if (psGeoLocNode)
2022 : {
2023 1 : CPLCreateXMLElementAndValue(psGeoLocNode, "SourceDataset",
2024 : pszValue);
2025 : }
2026 :
2027 396 : CPLConfigOptionSetter oSetter("CPL_ALLOW_VSISTDIN", "NO", true);
2028 :
2029 198 : char **papszOpenOptions = GDALDeserializeOpenOptionsFromXML(psTree);
2030 198 : psWO->hSrcDS =
2031 198 : GDALOpenEx(pszValue, GDAL_OF_RASTER | GDAL_OF_VERBOSE_ERROR,
2032 : nullptr, papszOpenOptions, nullptr);
2033 198 : CSLDestroy(papszOpenOptions);
2034 : }
2035 :
2036 : /* -------------------------------------------------------------------- */
2037 : /* Destination Dataset. */
2038 : /* -------------------------------------------------------------------- */
2039 198 : pszValue = CPLGetXMLValue(psTree, "DestinationDataset", nullptr);
2040 :
2041 198 : if (pszValue != nullptr)
2042 : {
2043 0 : psWO->hDstDS = GDALOpenShared(pszValue, GA_Update);
2044 : }
2045 :
2046 : /* -------------------------------------------------------------------- */
2047 : /* First, count band mappings so we can establish the bandcount. */
2048 : /* -------------------------------------------------------------------- */
2049 198 : CPLXMLNode *psBandTree = CPLGetXMLNode(psTree, "BandList");
2050 :
2051 198 : int nBandCount = 0;
2052 198 : CPLXMLNode *psBand = psBandTree ? psBandTree->psChild : nullptr;
2053 563 : for (; psBand != nullptr; psBand = psBand->psNext)
2054 : {
2055 365 : if (psBand->eType != CXT_Element ||
2056 365 : !EQUAL(psBand->pszValue, "BandMapping"))
2057 0 : continue;
2058 :
2059 365 : nBandCount++;
2060 : }
2061 :
2062 198 : GDALWarpInitDefaultBandMapping(psWO, nBandCount);
2063 :
2064 : /* ==================================================================== */
2065 : /* Now actually process each bandmapping. */
2066 : /* ==================================================================== */
2067 198 : int iBand = 0;
2068 :
2069 198 : psBand = psBandTree ? psBandTree->psChild : nullptr;
2070 :
2071 563 : for (; psBand != nullptr; psBand = psBand->psNext)
2072 : {
2073 365 : if (psBand->eType != CXT_Element ||
2074 365 : !EQUAL(psBand->pszValue, "BandMapping"))
2075 0 : continue;
2076 :
2077 : /* --------------------------------------------------------------------
2078 : */
2079 : /* Source band */
2080 : /* --------------------------------------------------------------------
2081 : */
2082 365 : pszValue = CPLGetXMLValue(psBand, "src", nullptr);
2083 365 : if (pszValue != nullptr)
2084 365 : psWO->panSrcBands[iBand] = atoi(pszValue);
2085 :
2086 : /* --------------------------------------------------------------------
2087 : */
2088 : /* Destination band. */
2089 : /* --------------------------------------------------------------------
2090 : */
2091 365 : pszValue = CPLGetXMLValue(psBand, "dst", nullptr);
2092 365 : if (pszValue != nullptr)
2093 365 : psWO->panDstBands[iBand] = atoi(pszValue);
2094 :
2095 66 : const auto NormalizeValue = [](const char *pszValueIn,
2096 : GDALDataType eDataType) -> double
2097 : {
2098 78 : if (eDataType == GDT_Float32 &&
2099 78 : CPLString().Printf(
2100 12 : "%.16g", -std::numeric_limits<float>::max()) == pszValueIn)
2101 : {
2102 0 : return -std::numeric_limits<float>::max();
2103 : }
2104 78 : else if (eDataType == GDT_Float32 &&
2105 78 : CPLString().Printf("%.16g",
2106 12 : std::numeric_limits<float>::max()) ==
2107 : pszValueIn)
2108 : {
2109 0 : return std::numeric_limits<float>::max();
2110 : }
2111 : else
2112 : {
2113 66 : return CPLAtof(pszValueIn);
2114 : }
2115 : };
2116 :
2117 : /* --------------------------------------------------------------------
2118 : */
2119 : /* Source nodata. */
2120 : /* --------------------------------------------------------------------
2121 : */
2122 365 : pszValue = CPLGetXMLValue(psBand, "SrcNoDataReal", nullptr);
2123 365 : if (pszValue != nullptr)
2124 : {
2125 31 : GDALWarpInitSrcNoDataReal(psWO, -1.1e20);
2126 62 : psWO->padfSrcNoDataReal[iBand] =
2127 31 : NormalizeValue(pszValue, psWO->eWorkingDataType);
2128 : }
2129 :
2130 365 : pszValue = CPLGetXMLValue(psBand, "SrcNoDataImag", nullptr);
2131 365 : if (pszValue != nullptr)
2132 : {
2133 31 : GDALWarpInitSrcNoDataImag(psWO, 0);
2134 31 : psWO->padfSrcNoDataImag[iBand] = CPLAtof(pszValue);
2135 : }
2136 :
2137 : /* --------------------------------------------------------------------
2138 : */
2139 : /* Destination nodata. */
2140 : /* --------------------------------------------------------------------
2141 : */
2142 365 : pszValue = CPLGetXMLValue(psBand, "DstNoDataReal", nullptr);
2143 365 : if (pszValue != nullptr)
2144 : {
2145 35 : GDALWarpInitDstNoDataReal(psWO, -1.1e20);
2146 70 : psWO->padfDstNoDataReal[iBand] =
2147 35 : NormalizeValue(pszValue, psWO->eWorkingDataType);
2148 : }
2149 :
2150 365 : pszValue = CPLGetXMLValue(psBand, "DstNoDataImag", nullptr);
2151 365 : if (pszValue != nullptr)
2152 : {
2153 35 : GDALWarpInitDstNoDataImag(psWO, 0);
2154 35 : psWO->padfDstNoDataImag[iBand] = CPLAtof(pszValue);
2155 : }
2156 :
2157 365 : iBand++;
2158 : }
2159 :
2160 : /* -------------------------------------------------------------------- */
2161 : /* Alpha bands. */
2162 : /* -------------------------------------------------------------------- */
2163 198 : psWO->nSrcAlphaBand = atoi(CPLGetXMLValue(psTree, "SrcAlphaBand", "0"));
2164 198 : psWO->nDstAlphaBand = atoi(CPLGetXMLValue(psTree, "DstAlphaBand", "0"));
2165 :
2166 : /* -------------------------------------------------------------------- */
2167 : /* Cutline. */
2168 : /* -------------------------------------------------------------------- */
2169 198 : const char *pszWKT = CPLGetXMLValue(psTree, "Cutline", nullptr);
2170 198 : if (pszWKT)
2171 : {
2172 7 : char *pszWKTTemp = const_cast<char *>(pszWKT);
2173 7 : OGRGeometryH hCutline = nullptr;
2174 7 : OGR_G_CreateFromWkt(&pszWKTTemp, nullptr, &hCutline);
2175 7 : psWO->hCutline = hCutline;
2176 : }
2177 :
2178 198 : psWO->dfCutlineBlendDist =
2179 198 : CPLAtof(CPLGetXMLValue(psTree, "CutlineBlendDist", "0"));
2180 :
2181 : /* -------------------------------------------------------------------- */
2182 : /* Transformation. */
2183 : /* -------------------------------------------------------------------- */
2184 198 : CPLXMLNode *psTransformer = CPLGetXMLNode(psTree, "Transformer");
2185 :
2186 198 : if (psTransformer != nullptr && psTransformer->psChild != nullptr)
2187 : {
2188 198 : GDALDeserializeTransformer(psTransformer->psChild,
2189 : &(psWO->pfnTransformer),
2190 : &(psWO->pTransformerArg));
2191 : }
2192 :
2193 : /* -------------------------------------------------------------------- */
2194 : /* If any error has occurred, cleanup else return success. */
2195 : /* -------------------------------------------------------------------- */
2196 198 : if (CPLGetLastErrorType() != CE_None)
2197 : {
2198 0 : if (psWO->pTransformerArg)
2199 : {
2200 0 : GDALDestroyTransformer(psWO->pTransformerArg);
2201 0 : psWO->pTransformerArg = nullptr;
2202 : }
2203 0 : if (psWO->hSrcDS != nullptr)
2204 : {
2205 0 : GDALClose(psWO->hSrcDS);
2206 0 : psWO->hSrcDS = nullptr;
2207 : }
2208 0 : if (psWO->hDstDS != nullptr)
2209 : {
2210 0 : GDALClose(psWO->hDstDS);
2211 0 : psWO->hDstDS = nullptr;
2212 : }
2213 0 : GDALDestroyWarpOptions(psWO);
2214 0 : return nullptr;
2215 : }
2216 :
2217 198 : return psWO;
2218 : }
|